| /* |
| * Copyright (C) 2014 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License |
| */ |
| |
| package com.android.systemui.statusbar; |
| |
| import static java.lang.annotation.RetentionPolicy.SOURCE; |
| |
| import android.animation.ObjectAnimator; |
| import android.animation.ValueAnimator; |
| import android.annotation.IntDef; |
| import android.util.FloatProperty; |
| import android.view.animation.Interpolator; |
| |
| import com.android.internal.annotations.GuardedBy; |
| import com.android.systemui.Interpolators; |
| import com.android.systemui.statusbar.notification.stack.StackStateAnimator; |
| import com.android.systemui.statusbar.phone.StatusBar; |
| |
| import java.lang.annotation.Retention; |
| import java.util.ArrayList; |
| import java.util.Comparator; |
| |
| /** |
| * Tracks and reports on {@link StatusBarState}. |
| */ |
| public class StatusBarStateController { |
| private static final String TAG = "SbStateController"; |
| |
| private static final int MAX_STATE = StatusBarState.FULLSCREEN_USER_SWITCHER; |
| private static final int MIN_STATE = StatusBarState.SHADE; |
| |
| private static final Comparator <RankedListener> mComparator |
| = (o1, o2) -> Integer.compare(o1.rank, o2.rank); |
| private static final FloatProperty<StatusBarStateController> SET_DARK_AMOUNT_PROPERTY = |
| new FloatProperty<StatusBarStateController>("mDozeAmount") { |
| |
| @Override |
| public void setValue(StatusBarStateController object, float value) { |
| object.setDozeAmountInternal(value); |
| } |
| |
| @Override |
| public Float get(StatusBarStateController object) { |
| return object.mDozeAmount; |
| } |
| }; |
| |
| private final ArrayList<RankedListener> mListeners = new ArrayList<>(); |
| private int mState; |
| private int mLastState; |
| private boolean mLeaveOpenOnKeyguardHide; |
| private boolean mKeyguardRequested; |
| |
| /** |
| * If the device is currently dozing or not. |
| */ |
| private boolean mIsDozing; |
| |
| /** |
| * Current {@link #mDozeAmount} animator. |
| */ |
| private ValueAnimator mDarkAnimator; |
| |
| /** |
| * Current doze amount in this frame. |
| */ |
| private float mDozeAmount; |
| |
| /** |
| * Where the animator will stop. |
| */ |
| private float mDozeAmountTarget; |
| |
| /** |
| * The type of interpolator that should be used to the doze animation. |
| */ |
| private Interpolator mDozeInterpolator = Interpolators.FAST_OUT_SLOW_IN; |
| |
| // TODO: b/115739177 (remove this explicit ordering if we can) |
| @Retention(SOURCE) |
| @IntDef({RANK_STATUS_BAR, RANK_STATUS_BAR_WINDOW_CONTROLLER, RANK_STACK_SCROLLER, RANK_SHELF}) |
| public @interface SbStateListenerRank {} |
| // This is the set of known dependencies when updating StatusBarState |
| public static final int RANK_STATUS_BAR = 0; |
| public static final int RANK_STATUS_BAR_WINDOW_CONTROLLER = 1; |
| public static final int RANK_STACK_SCROLLER = 2; |
| public static final int RANK_SHELF = 3; |
| |
| public int getState() { |
| return mState; |
| } |
| |
| /** |
| * Update the status bar state |
| * @param state see {@link StatusBarState} for valid options |
| * @return {@code true} if the state changed, else {@code false} |
| */ |
| public boolean setState(int state) { |
| if (state > MAX_STATE || state < MIN_STATE) { |
| throw new IllegalArgumentException("Invalid state " + state); |
| } |
| if (state == mState) { |
| return false; |
| } |
| synchronized (mListeners) { |
| for (RankedListener rl : new ArrayList<>(mListeners)) { |
| rl.listener.onStatePreChange(mState, state); |
| } |
| mLastState = mState; |
| mState = state; |
| for (RankedListener rl : new ArrayList<>(mListeners)) { |
| rl.listener.onStateChanged(mState); |
| } |
| |
| for (RankedListener rl : new ArrayList<>(mListeners)) { |
| rl.listener.onStatePostChange(); |
| } |
| } |
| |
| return true; |
| } |
| |
| public boolean isDozing() { |
| return mIsDozing; |
| } |
| |
| public float getDozeAmount() { |
| return mDozeAmount; |
| } |
| |
| public float getInterpolatedDozeAmount() { |
| return mDozeInterpolator.getInterpolation(mDozeAmount); |
| } |
| |
| /** |
| * Update the dozing state from {@link StatusBar}'s perspective |
| * @param isDozing well, are we dozing? |
| * @return {@code true} if the state changed, else {@code false} |
| */ |
| @SuppressWarnings("UnusedReturnValue") |
| public boolean setIsDozing(boolean isDozing) { |
| if (mIsDozing == isDozing) { |
| return false; |
| } |
| |
| mIsDozing = isDozing; |
| |
| synchronized (mListeners) { |
| for (RankedListener rl : new ArrayList<>(mListeners)) { |
| rl.listener.onDozingChanged(isDozing); |
| } |
| } |
| |
| return true; |
| } |
| |
| /** |
| * Changes the current doze amount. |
| * |
| * @param dozeAmount New doze/dark amount. |
| * @param animated If change should be animated or not. This will cancel current animations. |
| */ |
| public void setDozeAmount(float dozeAmount, boolean animated) { |
| if (mDarkAnimator != null && mDarkAnimator.isRunning()) { |
| if (animated && mDozeAmountTarget == dozeAmount) { |
| return; |
| } else { |
| mDarkAnimator.cancel(); |
| } |
| } |
| |
| mDozeAmountTarget = dozeAmount; |
| if (animated) { |
| startDozeAnimation(); |
| } else { |
| setDozeAmountInternal(dozeAmount); |
| } |
| } |
| |
| private void startDozeAnimation() { |
| if (mDozeAmount == 0f || mDozeAmount == 1f) { |
| mDozeInterpolator = mIsDozing |
| ? Interpolators.FAST_OUT_SLOW_IN |
| : Interpolators.TOUCH_RESPONSE_REVERSE; |
| } |
| mDarkAnimator = ObjectAnimator.ofFloat(this, SET_DARK_AMOUNT_PROPERTY, mDozeAmountTarget); |
| mDarkAnimator.setInterpolator(Interpolators.LINEAR); |
| mDarkAnimator.setDuration(StackStateAnimator.ANIMATION_DURATION_WAKEUP); |
| mDarkAnimator.start(); |
| } |
| |
| private void setDozeAmountInternal(float dozeAmount) { |
| mDozeAmount = dozeAmount; |
| float interpolatedAmount = mDozeInterpolator.getInterpolation(dozeAmount); |
| synchronized (mListeners) { |
| for (RankedListener rl : new ArrayList<>(mListeners)) { |
| rl.listener.onDozeAmountChanged(mDozeAmount, interpolatedAmount); |
| } |
| } |
| } |
| |
| public boolean goingToFullShade() { |
| return mState == StatusBarState.SHADE && mLeaveOpenOnKeyguardHide; |
| } |
| |
| public void setLeaveOpenOnKeyguardHide(boolean leaveOpen) { |
| mLeaveOpenOnKeyguardHide = leaveOpen; |
| } |
| |
| public boolean leaveOpenOnKeyguardHide() { |
| return mLeaveOpenOnKeyguardHide; |
| } |
| |
| public boolean fromShadeLocked() { |
| return mLastState == StatusBarState.SHADE_LOCKED; |
| } |
| |
| public void addListener(StateListener listener) { |
| synchronized (mListeners) { |
| addListenerInternalLocked(listener, Integer.MAX_VALUE); |
| } |
| } |
| |
| /** |
| * Add a listener and a rank based on the priority of this message |
| * @param listener the listener |
| * @param rank the order in which you'd like to be called. Ranked listeners will be |
| * notified before unranked, and we will sort ranked listeners from low to high |
| * |
| * @deprecated This method exists only to solve latent inter-dependencies from refactoring |
| * StatusBarState out of StatusBar.java. Any new listeners should be built not to need ranking |
| * (i.e., they are non-dependent on the order of operations of StatusBarState listeners). |
| */ |
| public void addListener(StateListener listener, @SbStateListenerRank int rank) { |
| synchronized (mListeners) { |
| addListenerInternalLocked(listener, rank); |
| } |
| } |
| |
| @GuardedBy("mListeners") |
| private void addListenerInternalLocked(StateListener listener, int rank) { |
| // Protect against double-subscribe |
| for (RankedListener rl : mListeners) { |
| if (rl.listener.equals(listener)) { |
| return; |
| } |
| } |
| |
| RankedListener rl = new RankedListener(listener, rank); |
| mListeners.add(rl); |
| mListeners.sort(mComparator); |
| } |
| |
| public void removeListener(StateListener listener) { |
| synchronized (mListeners) { |
| mListeners.removeIf((it) -> it.listener.equals(listener)); |
| } |
| } |
| |
| public void setKeyguardRequested(boolean keyguardRequested) { |
| mKeyguardRequested = keyguardRequested; |
| } |
| |
| public boolean isKeyguardRequested() { |
| return mKeyguardRequested; |
| } |
| |
| public static String describe(int state) { |
| return StatusBarState.toShortString(state); |
| } |
| |
| private class RankedListener { |
| private final StateListener listener; |
| private final int rank; |
| |
| private RankedListener(StateListener l, int r) { |
| listener = l; |
| rank = r; |
| } |
| } |
| |
| /** |
| * Listener for StatusBarState updates |
| */ |
| public interface StateListener { |
| |
| /** |
| * Callback before the new state is applied, for those who need to preempt the change. |
| * |
| * @param oldState state before the change |
| * @param newState new state to be applied in {@link #onStateChanged} |
| */ |
| public default void onStatePreChange(int oldState, int newState) { |
| } |
| |
| /** |
| * Callback after all listeners have had a chance to update based on the state change |
| */ |
| public default void onStatePostChange() { |
| } |
| |
| /** |
| * Required callback. Get the new state and do what you will with it. Keep in mind that |
| * other listeners are typically unordered and don't rely on your work being done before |
| * other peers. |
| * |
| * Only called if the state is actually different. |
| * |
| * @param newState the new {@link StatusBarState} |
| */ |
| public void onStateChanged(int newState); |
| |
| /** |
| * Callback to be notified when Dozing changes. Dozing is stored separately from state. |
| * |
| * @param isDozing {@code true} if dozing according to {@link StatusBar} |
| */ |
| public default void onDozingChanged(boolean isDozing) {} |
| |
| /** |
| * Callback to be notified when the doze amount changes. Useful for animations. |
| * Note: this will be called for each animation frame. Please be careful to avoid |
| * performance regressions. |
| * |
| * @param linear A number from 0 to 1, where 1 means that the device is dozing. |
| * @param eased Same as {@code linear} but transformed by an interpolator. |
| */ |
| default void onDozeAmountChanged(float linear, float eased) {} |
| } |
| } |