Merge "Refactored rotation code from nav bar fragment to its button"
This commit is contained in:
committed by
Android (Google) Code Review
commit
0cb4e4777a
@@ -288,4 +288,10 @@ public class ButtonDispatcher {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes when button is detached from window.
|
||||
*/
|
||||
protected void onDestroy() {
|
||||
}
|
||||
}
|
||||
|
||||
@@ -18,8 +18,10 @@ package com.android.systemui.statusbar.phone;
|
||||
|
||||
import android.annotation.DrawableRes;
|
||||
import android.annotation.IdRes;
|
||||
import android.annotation.NonNull;
|
||||
import android.content.Context;
|
||||
import android.view.View;
|
||||
|
||||
import com.android.systemui.statusbar.policy.KeyButtonDrawable;
|
||||
import com.android.systemui.statusbar.policy.KeyButtonView;
|
||||
|
||||
@@ -29,6 +31,9 @@ import com.android.systemui.statusbar.policy.KeyButtonView;
|
||||
*/
|
||||
public class ContextualButton extends ButtonDispatcher {
|
||||
|
||||
private ContextButtonListener mListener;
|
||||
private ContextualButtonGroup mGroup;
|
||||
|
||||
protected final @DrawableRes int mIconResId;
|
||||
|
||||
/**
|
||||
@@ -64,6 +69,48 @@ public class ContextualButton extends ButtonDispatcher {
|
||||
currentDrawable.clearAnimationCallbacks();
|
||||
currentDrawable.resetAnimation();
|
||||
}
|
||||
|
||||
if (mListener != null) {
|
||||
mListener.onVisibilityChanged(this, visibility == View.VISIBLE);
|
||||
}
|
||||
}
|
||||
|
||||
public void setListener(ContextButtonListener listener) {
|
||||
mListener = listener;
|
||||
}
|
||||
|
||||
/**
|
||||
* Show this button based on its priority compared to other buttons in the group. If not
|
||||
* attached to a group it will set its own visibility to be visible.
|
||||
* @return if visible
|
||||
*/
|
||||
public boolean show() {
|
||||
if (mGroup == null) {
|
||||
setVisibility(View.VISIBLE);
|
||||
return true;
|
||||
}
|
||||
return mGroup.setButtonVisiblity(getId(), true /* visible */) == View.VISIBLE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Hide this button.
|
||||
* @return if visible
|
||||
*/
|
||||
public boolean hide() {
|
||||
if (mGroup == null) {
|
||||
setVisibility(View.INVISIBLE);
|
||||
return false;
|
||||
}
|
||||
return mGroup.setButtonVisiblity(getId(), false /* visible */) != View.VISIBLE;
|
||||
}
|
||||
|
||||
/**
|
||||
* Called when this button was added to the group. Keep a reference to the group to show based
|
||||
* on priority compared to other buttons.
|
||||
* @param group the holder of all the buttons
|
||||
*/
|
||||
void attachToGroup(@NonNull ContextualButtonGroup group) {
|
||||
mGroup = group;
|
||||
}
|
||||
|
||||
protected KeyButtonDrawable getNewDrawable() {
|
||||
@@ -79,4 +126,8 @@ public class ContextualButton extends ButtonDispatcher {
|
||||
protected Context getContext() {
|
||||
return getCurrentView().getContext();
|
||||
}
|
||||
|
||||
public interface ContextButtonListener {
|
||||
void onVisibilityChanged(ContextualButton button, boolean visible);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -40,6 +40,7 @@ public class ContextualButtonGroup extends ButtonDispatcher {
|
||||
* @param button the button added to the group
|
||||
*/
|
||||
public void addButton(@NonNull ContextualButton button) {
|
||||
button.attachToGroup(this);
|
||||
mButtonData.add(new ButtonData(button));
|
||||
}
|
||||
|
||||
|
||||
@@ -19,26 +19,20 @@ import static android.app.StatusBarManager.NAVIGATION_HINT_IME_SHOWN;
|
||||
import static android.app.StatusBarManager.WINDOW_STATE_SHOWING;
|
||||
import static android.app.StatusBarManager.windowStateToString;
|
||||
|
||||
import static com.android.internal.view.RotationPolicy.NATURAL_ROTATION;
|
||||
|
||||
import static com.android.systemui.OverviewProxyService.OverviewProxyListener;
|
||||
import static com.android.systemui.shared.system.NavigationBarCompat.InteractionType;
|
||||
import static com.android.systemui.statusbar.phone.BarTransitions.MODE_LIGHTS_OUT_TRANSPARENT;
|
||||
import static com.android.systemui.statusbar.phone.BarTransitions.MODE_SEMI_TRANSPARENT;
|
||||
import static com.android.systemui.statusbar.phone.BarTransitions.MODE_TRANSPARENT;
|
||||
import static com.android.systemui.statusbar.phone.StatusBar.DEBUG_WINDOW_STATE;
|
||||
import static com.android.systemui.statusbar.phone.StatusBar.dumpBarTransitions;
|
||||
import static com.android.systemui.OverviewProxyService.OverviewProxyListener;
|
||||
|
||||
import android.accessibilityservice.AccessibilityServiceInfo;
|
||||
import android.animation.Animator;
|
||||
import android.animation.AnimatorListenerAdapter;
|
||||
import android.animation.ObjectAnimator;
|
||||
import android.annotation.IdRes;
|
||||
import android.annotation.Nullable;
|
||||
import android.app.ActivityManager;
|
||||
import android.app.ActivityTaskManager;
|
||||
import android.app.Fragment;
|
||||
import android.app.IActivityManager;
|
||||
import android.app.IActivityTaskManager;
|
||||
import android.app.StatusBarManager;
|
||||
import android.content.BroadcastReceiver;
|
||||
@@ -55,11 +49,9 @@ import android.os.Binder;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.IBinder;
|
||||
import android.os.Message;
|
||||
import android.os.RemoteException;
|
||||
import android.os.UserHandle;
|
||||
import android.provider.Settings;
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
import android.telecom.TelecomManager;
|
||||
import android.text.TextUtils;
|
||||
import android.util.Log;
|
||||
@@ -72,16 +64,16 @@ import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.view.WindowManager;
|
||||
import android.view.WindowManager.LayoutParams;
|
||||
import android.view.WindowManagerGlobal;
|
||||
import android.view.accessibility.AccessibilityEvent;
|
||||
import android.view.accessibility.AccessibilityManager;
|
||||
import android.view.accessibility.AccessibilityManager.AccessibilityServicesStateChangeListener;
|
||||
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
|
||||
import com.android.internal.logging.MetricsLogger;
|
||||
import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
|
||||
import com.android.internal.util.LatencyTracker;
|
||||
import com.android.systemui.Dependency;
|
||||
import com.android.systemui.Interpolators;
|
||||
import com.android.systemui.OverviewProxyService;
|
||||
import com.android.systemui.R;
|
||||
import com.android.systemui.SysUiServiceProvider;
|
||||
@@ -90,22 +82,20 @@ import com.android.systemui.fragments.FragmentHostManager;
|
||||
import com.android.systemui.fragments.FragmentHostManager.FragmentListener;
|
||||
import com.android.systemui.recents.Recents;
|
||||
import com.android.systemui.shared.system.ActivityManagerWrapper;
|
||||
import com.android.systemui.shared.system.TaskStackChangeListener;
|
||||
import com.android.systemui.stackdivider.Divider;
|
||||
import com.android.systemui.statusbar.CommandQueue;
|
||||
import com.android.systemui.statusbar.CommandQueue.Callbacks;
|
||||
import com.android.systemui.statusbar.notification.stack.StackStateAnimator;
|
||||
import com.android.systemui.statusbar.phone.ContextualButton.ContextButtonListener;
|
||||
import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper;
|
||||
import com.android.systemui.statusbar.policy.DeviceProvisionedController;
|
||||
import com.android.systemui.statusbar.policy.KeyButtonDrawable;
|
||||
import com.android.systemui.statusbar.policy.KeyButtonView;
|
||||
import com.android.systemui.statusbar.policy.RotationLockController;
|
||||
import com.android.systemui.statusbar.notification.stack.StackStateAnimator;
|
||||
|
||||
import java.io.FileDescriptor;
|
||||
import java.io.PrintWriter;
|
||||
import java.util.List;
|
||||
import java.util.Locale;
|
||||
import java.util.Optional;
|
||||
import java.util.function.Consumer;
|
||||
|
||||
/**
|
||||
* Fragment containing the NavigationBarFragment. Contains logic for what happens
|
||||
@@ -115,15 +105,9 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
|
||||
public static final String TAG = "NavigationBar";
|
||||
private static final boolean DEBUG = false;
|
||||
private static final boolean DEBUG_ROTATION = true;
|
||||
private static final String EXTRA_DISABLE_STATE = "disabled_state";
|
||||
private static final String EXTRA_DISABLE2_STATE = "disabled2_state";
|
||||
|
||||
private final static int BUTTON_FADE_IN_OUT_DURATION_MS = 100;
|
||||
private final static int NAVBAR_HIDDEN_PENDING_ICON_TIMEOUT_MS = 20000;
|
||||
|
||||
private static final int NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION = 3;
|
||||
|
||||
/** Allow some time inbetween the long press for back and recents. */
|
||||
private static final int LOCK_TO_APP_GESTURE_TOLERENCE = 200;
|
||||
|
||||
@@ -137,7 +121,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
|
||||
private int mNavigationIconHints = 0;
|
||||
private int mNavigationBarMode;
|
||||
private boolean mAccessibilityFeedbackEnabled;
|
||||
private AccessibilityManager mAccessibilityManager;
|
||||
private MagnificationContentObserver mMagnificationObserver;
|
||||
private ContentResolver mContentResolver;
|
||||
@@ -162,18 +145,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
|
||||
public boolean mHomeBlockedThisTouch;
|
||||
|
||||
private int mLastRotationSuggestion;
|
||||
private boolean mPendingRotationSuggestion;
|
||||
private boolean mHoveringRotationSuggestion;
|
||||
private RotationLockController mRotationLockController;
|
||||
private TaskStackListenerImpl mTaskStackListener;
|
||||
|
||||
private final Runnable mRemoveRotationProposal = () -> setRotateSuggestionButtonState(false);
|
||||
private final Runnable mCancelPendingRotationProposal =
|
||||
() -> mPendingRotationSuggestion = false;
|
||||
private Animator mRotateHideAnimator;
|
||||
private ViewRippler mViewRippler = new ViewRippler();
|
||||
|
||||
private final OverviewProxyListener mOverviewProxyListener = new OverviewProxyListener() {
|
||||
@Override
|
||||
public void onConnectionChanged(boolean isConnected) {
|
||||
@@ -184,7 +155,7 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
@Override
|
||||
public void onQuickStepStarted() {
|
||||
// Use navbar dragging as a signal to hide the rotate button
|
||||
setRotateSuggestionButtonState(false);
|
||||
mNavigationBarView.getRotateSuggestionButton().setRotateSuggestionButtonState(false);
|
||||
|
||||
// Hide the notifications panel when quick step starts
|
||||
mStatusBar.collapsePanel(true /* animate */);
|
||||
@@ -210,6 +181,17 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
}
|
||||
};
|
||||
|
||||
private final ContextButtonListener mRotationButtonListener = new ContextButtonListener() {
|
||||
@Override
|
||||
public void onVisibilityChanged(ContextualButton button, boolean visible) {
|
||||
if (visible) {
|
||||
// If the button will actually become visible and the navbar is about to hide,
|
||||
// tell the statusbar to keep it around for longer
|
||||
mStatusBar.touchAutoHide();
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
// ----- Fragment Lifecycle Callbacks -----
|
||||
|
||||
@Override
|
||||
@@ -237,26 +219,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
}
|
||||
mAssistManager = Dependency.get(AssistManager.class);
|
||||
mOverviewProxyService = Dependency.get(OverviewProxyService.class);
|
||||
|
||||
try {
|
||||
WindowManagerGlobal.getWindowManagerService()
|
||||
.watchRotation(mRotationWatcher, getContext().getDisplayId());
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
}
|
||||
|
||||
mRotationLockController = Dependency.get(RotationLockController.class);
|
||||
|
||||
// Reset user rotation pref to match that of the WindowManager if starting in locked mode
|
||||
// This will automatically happen when switching from auto-rotate to locked mode
|
||||
if (mRotationLockController.isRotationLocked()) {
|
||||
final int winRotation = mWindowManager.getDefaultDisplay().getRotation();
|
||||
mRotationLockController.setRotationLockedAtAngle(true, winRotation);
|
||||
}
|
||||
|
||||
// Register the task stack listener
|
||||
mTaskStackListener = new TaskStackListenerImpl();
|
||||
ActivityManagerWrapper.getInstance().registerTaskStackListener(mTaskStackListener);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -266,15 +228,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
Dependency.get(AccessibilityManagerWrapper.class).removeCallback(
|
||||
mAccessibilityListener);
|
||||
mContentResolver.unregisterContentObserver(mMagnificationObserver);
|
||||
try {
|
||||
WindowManagerGlobal.getWindowManagerService()
|
||||
.removeRotationWatcher(mRotationWatcher);
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
}
|
||||
|
||||
// Unregister the task stack listener
|
||||
ActivityManagerWrapper.getInstance().unregisterTaskStackListener(mTaskStackListener);
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -307,6 +260,17 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
getContext().registerReceiverAsUser(mBroadcastReceiver, UserHandle.ALL, filter, null, null);
|
||||
notifyNavigationBarScreenOn();
|
||||
mOverviewProxyService.addCallback(mOverviewProxyListener);
|
||||
|
||||
RotationContextButton rotationButton = mNavigationBarView.getRotateSuggestionButton();
|
||||
rotationButton.setListener(mRotationButtonListener);
|
||||
rotationButton.addRotationCallback(mRotationWatcher);
|
||||
|
||||
// Reset user rotation pref to match that of the WindowManager if starting in locked mode
|
||||
// This will automatically happen when switching from auto-rotate to locked mode
|
||||
if (rotationButton.isRotationLocked()) {
|
||||
final int winRotation = mWindowManager.getDefaultDisplay().getRotation();
|
||||
rotationButton.setRotationLockedAtAngle(winRotation);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -417,229 +381,31 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
mNavigationBarWindowState = state;
|
||||
if (DEBUG_WINDOW_STATE) Log.d(TAG, "Navigation bar " + windowStateToString(state));
|
||||
|
||||
// If the navbar is visible, show the rotate button if there's a pending suggestion
|
||||
if (state == WINDOW_STATE_SHOWING && mPendingRotationSuggestion) {
|
||||
showAndLogRotationSuggestion();
|
||||
}
|
||||
mNavigationBarView.getRotateSuggestionButton()
|
||||
.onNavigationBarWindowVisibilityChange(state == WINDOW_STATE_SHOWING);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onRotationProposal(final int rotation, boolean isValid) {
|
||||
final int winRotation = mWindowManager.getDefaultDisplay().getRotation();
|
||||
final boolean rotateSuggestionsDisabled = hasDisable2RotateSuggestionFlag(mDisabledFlags2);
|
||||
if (DEBUG_ROTATION) {
|
||||
final boolean rotateSuggestionsDisabled = RotationContextButton
|
||||
.hasDisable2RotateSuggestionFlag(mDisabledFlags2);
|
||||
if (RotationContextButton.DEBUG_ROTATION) {
|
||||
Log.v(TAG, "onRotationProposal proposedRotation=" + Surface.rotationToString(rotation)
|
||||
+ ", winRotation=" + Surface.rotationToString(winRotation)
|
||||
+ ", isValid=" + isValid + ", mNavBarWindowState="
|
||||
+ StatusBarManager.windowStateToString(mNavigationBarWindowState)
|
||||
+ ", rotateSuggestionsDisabled=" + rotateSuggestionsDisabled
|
||||
+ ", isRotateButtonVisible=" + (mNavigationBarView == null ? "null" :
|
||||
mNavigationBarView.isRotateButtonVisible()));
|
||||
mNavigationBarView.getRotateSuggestionButton().isVisible()));
|
||||
}
|
||||
|
||||
// Respect the disabled flag, no need for action as flag change callback will handle hiding
|
||||
if (rotateSuggestionsDisabled) return;
|
||||
|
||||
// This method will be called on rotation suggestion changes even if the proposed rotation
|
||||
// is not valid for the top app. Use invalid rotation choices as a signal to remove the
|
||||
// rotate button if shown.
|
||||
if (!isValid) {
|
||||
setRotateSuggestionButtonState(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// If window rotation matches suggested rotation, remove any current suggestions
|
||||
if (rotation == winRotation) {
|
||||
getView().removeCallbacks(mRemoveRotationProposal);
|
||||
setRotateSuggestionButtonState(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// Prepare to show the navbar icon by updating the icon style to change anim params
|
||||
mLastRotationSuggestion = rotation; // Remember rotation for click
|
||||
if (mNavigationBarView != null) {
|
||||
final boolean rotationCCW = isRotationAnimationCCW(winRotation, rotation);
|
||||
int style;
|
||||
if (winRotation == Surface.ROTATION_0 || winRotation == Surface.ROTATION_180) {
|
||||
style = rotationCCW ? R.style.RotateButtonCCWStart90 :
|
||||
R.style.RotateButtonCWStart90;
|
||||
} else { // 90 or 270
|
||||
style = rotationCCW ? R.style.RotateButtonCCWStart0 :
|
||||
R.style.RotateButtonCWStart0;
|
||||
}
|
||||
mNavigationBarView.updateRotateSuggestionButtonStyle(style);
|
||||
}
|
||||
|
||||
if (mNavigationBarWindowState != WINDOW_STATE_SHOWING) {
|
||||
// If the navbar isn't shown, flag the rotate icon to be shown should the navbar become
|
||||
// visible given some time limit.
|
||||
mPendingRotationSuggestion = true;
|
||||
getView().removeCallbacks(mCancelPendingRotationProposal);
|
||||
getView().postDelayed(mCancelPendingRotationProposal,
|
||||
NAVBAR_HIDDEN_PENDING_ICON_TIMEOUT_MS);
|
||||
|
||||
} else { // The navbar is visible so show the icon right away
|
||||
showAndLogRotationSuggestion();
|
||||
}
|
||||
}
|
||||
|
||||
private void onRotationSuggestionsDisabled() {
|
||||
// Immediately hide the rotate button and clear any planned removal
|
||||
setRotateSuggestionButtonState(false, true);
|
||||
|
||||
// This method can be called before view setup is done, ensure getView isn't null
|
||||
final View v = getView();
|
||||
if (v != null) v.removeCallbacks(mRemoveRotationProposal);
|
||||
}
|
||||
|
||||
private void showAndLogRotationSuggestion() {
|
||||
setRotateSuggestionButtonState(true);
|
||||
rescheduleRotationTimeout(false);
|
||||
mMetricsLogger.visible(MetricsEvent.ROTATION_SUGGESTION_SHOWN);
|
||||
}
|
||||
|
||||
private boolean isRotationAnimationCCW(int from, int to) {
|
||||
// All 180deg WM rotation animations are CCW, match that
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_90) return false;
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_180) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_270) return true;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_0) return true;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_180) return false;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_270) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_0) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_90) return true;
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_270) return false;
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_0) return false;
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_90) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_180) return true;
|
||||
return false; // Default
|
||||
}
|
||||
|
||||
public void setRotateSuggestionButtonState(final boolean visible) {
|
||||
setRotateSuggestionButtonState(visible, false);
|
||||
}
|
||||
|
||||
public void setRotateSuggestionButtonState(final boolean visible, final boolean force) {
|
||||
if (mNavigationBarView == null) return;
|
||||
|
||||
// At any point the the button can become invisible because an a11y service became active.
|
||||
// Similarly, a call to make the button visible may be rejected because an a11y service is
|
||||
// active. Must account for this.
|
||||
|
||||
ButtonDispatcher rotBtn = mNavigationBarView.getRotateSuggestionButton();
|
||||
final boolean currentlyVisible = mNavigationBarView.isRotateButtonVisible();
|
||||
|
||||
// Rerun a show animation to indicate change but don't rerun a hide animation
|
||||
if (!visible && !currentlyVisible) return;
|
||||
|
||||
View view = rotBtn.getCurrentView();
|
||||
if (view == null) return;
|
||||
|
||||
KeyButtonDrawable kbd = rotBtn.getImageDrawable();
|
||||
if (kbd == null) return;
|
||||
|
||||
// Clear any pending suggestion flag as it has either been nullified or is being shown
|
||||
mPendingRotationSuggestion = false;
|
||||
if (getView() != null) getView().removeCallbacks(mCancelPendingRotationProposal);
|
||||
|
||||
// Handle the visibility change and animation
|
||||
if (visible) { // Appear and change (cannot force)
|
||||
// Stop and clear any currently running hide animations
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) {
|
||||
mRotateHideAnimator.cancel();
|
||||
}
|
||||
mRotateHideAnimator = null;
|
||||
|
||||
// Reset the alpha if any has changed due to hide animation
|
||||
view.setAlpha(1f);
|
||||
|
||||
// Run the rotate icon's animation if it has one
|
||||
if (kbd.canAnimate()) {
|
||||
kbd.resetAnimation();
|
||||
kbd.startAnimation();
|
||||
}
|
||||
|
||||
if (!isRotateSuggestionIntroduced()) mViewRippler.start(view);
|
||||
|
||||
// Set visibility, may fail if a11y service is active.
|
||||
// If invisible, call will stop animation.
|
||||
int appliedVisibility = mNavigationBarView.setRotateButtonVisibility(true);
|
||||
if (appliedVisibility == View.VISIBLE) {
|
||||
// If the button will actually become visible and the navbar is about to hide,
|
||||
// tell the statusbar to keep it around for longer
|
||||
mStatusBar.touchAutoHide();
|
||||
}
|
||||
|
||||
} else { // Hide
|
||||
|
||||
mViewRippler.stop(); // Prevent any pending ripples, force hide or not
|
||||
|
||||
if (force) {
|
||||
// If a hide animator is running stop it and make invisible
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) {
|
||||
mRotateHideAnimator.pause();
|
||||
}
|
||||
mNavigationBarView.setRotateButtonVisibility(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// Don't start any new hide animations if one is running
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) return;
|
||||
|
||||
ObjectAnimator fadeOut = ObjectAnimator.ofFloat(view, "alpha",
|
||||
0f);
|
||||
fadeOut.setDuration(BUTTON_FADE_IN_OUT_DURATION_MS);
|
||||
fadeOut.setInterpolator(Interpolators.LINEAR);
|
||||
fadeOut.addListener(new AnimatorListenerAdapter() {
|
||||
@Override
|
||||
public void onAnimationEnd(Animator animation) {
|
||||
mNavigationBarView.setRotateButtonVisibility(false);
|
||||
}
|
||||
});
|
||||
|
||||
mRotateHideAnimator = fadeOut;
|
||||
fadeOut.start();
|
||||
}
|
||||
}
|
||||
|
||||
private void rescheduleRotationTimeout(final boolean reasonHover) {
|
||||
// May be called due to a new rotation proposal or a change in hover state
|
||||
if (reasonHover) {
|
||||
// Don't reschedule if a hide animator is running
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) return;
|
||||
// Don't reschedule if not visible
|
||||
if (!mNavigationBarView.isRotateButtonVisible()) return;
|
||||
}
|
||||
|
||||
getView().removeCallbacks(mRemoveRotationProposal); // Stop any pending removal
|
||||
getView().postDelayed(mRemoveRotationProposal,
|
||||
computeRotationProposalTimeout()); // Schedule timeout
|
||||
}
|
||||
|
||||
private int computeRotationProposalTimeout() {
|
||||
if (mAccessibilityFeedbackEnabled) return 20000;
|
||||
if (mHoveringRotationSuggestion) return 16000;
|
||||
return 10000;
|
||||
}
|
||||
|
||||
private boolean isRotateSuggestionIntroduced() {
|
||||
ContentResolver cr = getContext().getContentResolver();
|
||||
return Settings.Secure.getInt(cr, Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED, 0)
|
||||
>= NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION;
|
||||
}
|
||||
|
||||
private void incrementNumAcceptedRotationSuggestionsIfNeeded() {
|
||||
// Get the number of accepted suggestions
|
||||
ContentResolver cr = getContext().getContentResolver();
|
||||
final int numSuggestions = Settings.Secure.getInt(cr,
|
||||
Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED, 0);
|
||||
|
||||
// Increment the number of accepted suggestions only if it would change intro mode
|
||||
if (numSuggestions < NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION) {
|
||||
Settings.Secure.putInt(cr, Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED,
|
||||
numSuggestions + 1);
|
||||
}
|
||||
mNavigationBarView.getRotateSuggestionButton()
|
||||
.onRotationProposal(rotation, winRotation, isValid);
|
||||
}
|
||||
|
||||
// Injected from StatusBar at creation.
|
||||
@@ -712,12 +478,7 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
|
||||
private void setDisabled2Flags(int state2) {
|
||||
// Method only called on change of disable2 flags
|
||||
final boolean rotateSuggestionsDisabled = hasDisable2RotateSuggestionFlag(state2);
|
||||
if (rotateSuggestionsDisabled) onRotationSuggestionsDisabled();
|
||||
}
|
||||
|
||||
private boolean hasDisable2RotateSuggestionFlag(int disable2Flags) {
|
||||
return (disable2Flags & StatusBarManager.DISABLE2_ROTATE_SUGGESTIONS) != 0;
|
||||
mNavigationBarView.getRotateSuggestionButton().onDisable2FlagChanged(state2);
|
||||
}
|
||||
|
||||
// ----- Internal stuffz -----
|
||||
@@ -782,9 +543,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
accessibilityButton.setOnLongClickListener(this::onAccessibilityLongClick);
|
||||
updateAccessibilityServicesState(mAccessibilityManager);
|
||||
|
||||
ButtonDispatcher rotateSuggestionButton = mNavigationBarView.getRotateSuggestionButton();
|
||||
rotateSuggestionButton.setOnClickListener(this::onRotateSuggestionClick);
|
||||
rotateSuggestionButton.setOnHoverListener(this::onRotateSuggestionHover);
|
||||
updateScreenPinningGestures();
|
||||
}
|
||||
|
||||
@@ -1004,27 +762,14 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
}
|
||||
}
|
||||
|
||||
mAccessibilityFeedbackEnabled = feedbackEnabled;
|
||||
mNavigationBarView.getRotateSuggestionButton()
|
||||
.setAccessibilityFeedbackEnabled(feedbackEnabled);
|
||||
|
||||
final boolean showAccessibilityButton = requestingServices >= 1;
|
||||
final boolean targetSelection = requestingServices >= 2;
|
||||
mNavigationBarView.setAccessibilityButtonState(showAccessibilityButton, targetSelection);
|
||||
}
|
||||
|
||||
private void onRotateSuggestionClick(View v) {
|
||||
mMetricsLogger.action(MetricsEvent.ACTION_ROTATION_SUGGESTION_ACCEPTED);
|
||||
incrementNumAcceptedRotationSuggestionsIfNeeded();
|
||||
mRotationLockController.setRotationLockedAtAngle(true, mLastRotationSuggestion);
|
||||
}
|
||||
|
||||
private boolean onRotateSuggestionHover(View v, MotionEvent event) {
|
||||
final int action = event.getActionMasked();
|
||||
mHoveringRotationSuggestion = (action == MotionEvent.ACTION_HOVER_ENTER)
|
||||
|| (action == MotionEvent.ACTION_HOVER_MOVE);
|
||||
rescheduleRotationTimeout(true);
|
||||
return false; // Must return false so a11y hover events are dispatched correctly.
|
||||
}
|
||||
|
||||
// ----- Methods that StatusBar talks to (should be minimized) -----
|
||||
|
||||
public void setLightBarController(LightBarController lightBarController) {
|
||||
@@ -1070,36 +815,10 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
}
|
||||
}
|
||||
|
||||
private final Stub mRotationWatcher = new Stub() {
|
||||
@Override
|
||||
public void onRotationChanged(final int rotation) throws RemoteException {
|
||||
// We need this to be scheduled as early as possible to beat the redrawing of
|
||||
// window in response to the orientation change.
|
||||
Handler h = getView().getHandler();
|
||||
Message msg = Message.obtain(h, () -> {
|
||||
|
||||
// If the screen rotation changes while locked, potentially update lock to flow with
|
||||
// new screen rotation and hide any showing suggestions.
|
||||
if (mRotationLockController.isRotationLocked()) {
|
||||
if (shouldOverrideUserLockPrefs(rotation)) {
|
||||
mRotationLockController.setRotationLockedAtAngle(true, rotation);
|
||||
}
|
||||
setRotateSuggestionButtonState(false, true);
|
||||
}
|
||||
|
||||
if (mNavigationBarView != null
|
||||
&& mNavigationBarView.needsReorient(rotation)) {
|
||||
repositionNavigationBar();
|
||||
}
|
||||
});
|
||||
msg.setAsynchronous(true);
|
||||
h.sendMessageAtFrontOfQueue(msg);
|
||||
}
|
||||
|
||||
private boolean shouldOverrideUserLockPrefs(final int rotation) {
|
||||
// Only override user prefs when returning to the natural rotation (normally portrait).
|
||||
// Don't let apps that force landscape or 180 alter user lock.
|
||||
return rotation == NATURAL_ROTATION;
|
||||
private final Consumer<Integer> mRotationWatcher = rotation -> {
|
||||
if (mNavigationBarView != null
|
||||
&& mNavigationBarView.needsReorient(rotation)) {
|
||||
repositionNavigationBar();
|
||||
}
|
||||
};
|
||||
|
||||
@@ -1118,69 +837,6 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
|
||||
}
|
||||
};
|
||||
|
||||
class TaskStackListenerImpl extends TaskStackChangeListener {
|
||||
// Invalidate any rotation suggestion on task change or activity orientation change
|
||||
// Note: all callbacks happen on main thread
|
||||
|
||||
@Override
|
||||
public void onTaskStackChanged() {
|
||||
setRotateSuggestionButtonState(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onTaskRemoved(int taskId) {
|
||||
setRotateSuggestionButtonState(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onTaskMovedToFront(int taskId) {
|
||||
setRotateSuggestionButtonState(false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActivityRequestedOrientationChanged(int taskId, int requestedOrientation) {
|
||||
// Only hide the icon if the top task changes its requestedOrientation
|
||||
// Launcher can alter its requestedOrientation while it's not on top, don't hide on this
|
||||
Optional.ofNullable(ActivityManagerWrapper.getInstance())
|
||||
.map(ActivityManagerWrapper::getRunningTask)
|
||||
.ifPresent(a -> {
|
||||
if (a.id == taskId) setRotateSuggestionButtonState(false);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
private class ViewRippler {
|
||||
private static final int RIPPLE_OFFSET_MS = 50;
|
||||
private static final int RIPPLE_INTERVAL_MS = 2000;
|
||||
private View mRoot;
|
||||
|
||||
public void start(View root) {
|
||||
stop(); // Stop any pending ripple animations
|
||||
|
||||
mRoot = root;
|
||||
|
||||
// Schedule pending ripples, offset the 1st to avoid problems with visibility change
|
||||
mRoot.postOnAnimationDelayed(mRipple, RIPPLE_OFFSET_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 2*RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 3*RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 4*RIPPLE_INTERVAL_MS);
|
||||
}
|
||||
|
||||
public void stop() {
|
||||
if (mRoot != null) mRoot.removeCallbacks(mRipple);
|
||||
}
|
||||
|
||||
private final Runnable mRipple = new Runnable() {
|
||||
@Override
|
||||
public void run() { // Cause the ripple to fire via false presses
|
||||
if (!mRoot.isAttachedToWindow()) return;
|
||||
mRoot.setPressed(true);
|
||||
mRoot.setPressed(false);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
public static View create(Context context, FragmentListener listener) {
|
||||
WindowManager.LayoutParams lp = new WindowManager.LayoutParams(
|
||||
LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT,
|
||||
|
||||
@@ -29,7 +29,6 @@ import android.animation.PropertyValuesHolder;
|
||||
import android.animation.TimeInterpolator;
|
||||
import android.animation.ValueAnimator;
|
||||
import android.annotation.DrawableRes;
|
||||
import android.annotation.StyleRes;
|
||||
import android.app.StatusBarManager;
|
||||
import android.content.Context;
|
||||
import android.content.res.Configuration;
|
||||
@@ -271,7 +270,7 @@ public class NavigationBarView extends FrameLayout implements PluginListener<Nav
|
||||
final ContextualButton imeSwitcherButton = new ContextualButton(R.id.ime_switcher,
|
||||
R.drawable.ic_ime_switcher_default);
|
||||
final RotationContextButton rotateSuggestionButton = new RotationContextButton(
|
||||
R.id.rotate_suggestion, R.drawable.ic_sysbar_rotate_button,
|
||||
R.id.rotate_suggestion, R.drawable.ic_sysbar_rotate_button, getContext(),
|
||||
R.style.RotateButtonCCWStart90);
|
||||
final ContextualButton accessibilityButton =
|
||||
new ContextualButton(R.id.accessibility_button,
|
||||
@@ -418,8 +417,9 @@ public class NavigationBarView extends FrameLayout implements PluginListener<Nav
|
||||
return mButtonDispatchers.get(R.id.accessibility_button);
|
||||
}
|
||||
|
||||
public ButtonDispatcher getRotateSuggestionButton() {
|
||||
return mButtonDispatchers.get(R.id.rotate_suggestion);
|
||||
public RotationContextButton getRotateSuggestionButton() {
|
||||
return (RotationContextButton) mContextualButtonGroup
|
||||
.getContextButton(R.id.rotate_suggestion);
|
||||
}
|
||||
|
||||
public SparseArray<ButtonDispatcher> getButtonDispatchers() {
|
||||
@@ -746,27 +746,12 @@ public class NavigationBarView extends FrameLayout implements PluginListener<Nav
|
||||
mContextualButtonGroup.setButtonVisiblity(R.id.menu, show);
|
||||
}
|
||||
|
||||
public void updateRotateSuggestionButtonStyle(@StyleRes int style) {
|
||||
RotationContextButton button = (RotationContextButton) mContextualButtonGroup
|
||||
.getContextButton(R.id.rotate_suggestion);
|
||||
button.setStyle(style);
|
||||
button.updateIcon();
|
||||
}
|
||||
|
||||
public void setAccessibilityButtonState(final boolean visible, final boolean longClickable) {
|
||||
mLongClickableAccessibilityButton = longClickable;
|
||||
getAccessibilityButton().setLongClickable(longClickable);
|
||||
mContextualButtonGroup.setButtonVisiblity(R.id.accessibility_button, visible);
|
||||
}
|
||||
|
||||
public int setRotateButtonVisibility(boolean visible) {
|
||||
return mContextualButtonGroup.setButtonVisiblity(R.id.rotate_suggestion, visible);
|
||||
}
|
||||
|
||||
public boolean isRotateButtonVisible() {
|
||||
return getRotateSuggestionButton().isVisible();
|
||||
}
|
||||
|
||||
void hideRecentsOnboarding() {
|
||||
mRecentsOnboarding.hide(true);
|
||||
}
|
||||
@@ -1058,6 +1043,9 @@ public class NavigationBarView extends FrameLayout implements PluginListener<Nav
|
||||
mGestureHelper.destroy();
|
||||
}
|
||||
setUpSwipeUpOnboarding(false);
|
||||
for (int i = 0; i < mButtonDispatchers.size(); ++i) {
|
||||
mButtonDispatchers.valueAt(i).onDestroy();
|
||||
}
|
||||
}
|
||||
|
||||
private void setUpSwipeUpOnboarding(boolean connectedToOverviewProxy) {
|
||||
|
||||
@@ -16,29 +16,269 @@
|
||||
|
||||
package com.android.systemui.statusbar.phone;
|
||||
|
||||
import static com.android.internal.view.RotationPolicy.NATURAL_ROTATION;
|
||||
|
||||
import android.animation.Animator;
|
||||
import android.animation.AnimatorListenerAdapter;
|
||||
import android.animation.ObjectAnimator;
|
||||
import android.annotation.DrawableRes;
|
||||
import android.annotation.IdRes;
|
||||
import android.annotation.NonNull;
|
||||
import android.annotation.StyleRes;
|
||||
import android.app.StatusBarManager;
|
||||
import android.content.ContentResolver;
|
||||
import android.content.Context;
|
||||
import android.content.ContextWrapper;
|
||||
import android.os.Handler;
|
||||
import android.os.Message;
|
||||
import android.os.RemoteException;
|
||||
import android.provider.Settings;
|
||||
import android.view.ContextThemeWrapper;
|
||||
import android.view.IRotationWatcher.Stub;
|
||||
import android.view.MotionEvent;
|
||||
import android.view.Surface;
|
||||
import android.view.View;
|
||||
import android.view.WindowManagerGlobal;
|
||||
|
||||
import com.android.systemui.Dependency;
|
||||
import com.android.systemui.Interpolators;
|
||||
import com.android.internal.logging.MetricsLogger;
|
||||
import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
|
||||
import com.android.systemui.R;
|
||||
import com.android.systemui.recents.misc.SysUiTaskStackChangeListener;
|
||||
import com.android.systemui.shared.system.ActivityManagerWrapper;
|
||||
import com.android.systemui.statusbar.policy.KeyButtonDrawable;
|
||||
import com.android.systemui.util.Utils;
|
||||
import com.android.systemui.statusbar.policy.RotationLockController;
|
||||
|
||||
import java.util.Optional;
|
||||
import java.util.function.Consumer;
|
||||
|
||||
public class RotationContextButton extends ContextualButton {
|
||||
public static final boolean DEBUG_ROTATION = false;
|
||||
|
||||
private static final int BUTTON_FADE_IN_OUT_DURATION_MS = 100;
|
||||
private static final int NAVBAR_HIDDEN_PENDING_ICON_TIMEOUT_MS = 20000;
|
||||
|
||||
private static final int NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION = 3;
|
||||
|
||||
private @StyleRes int mStyleRes;
|
||||
|
||||
public RotationContextButton(@IdRes int buttonResId, @DrawableRes int iconResId,
|
||||
@StyleRes int style) {
|
||||
super(buttonResId, iconResId);
|
||||
mStyleRes = style;
|
||||
private int mLastRotationSuggestion;
|
||||
private boolean mPendingRotationSuggestion;
|
||||
private boolean mHoveringRotationSuggestion;
|
||||
private RotationLockController mRotationLockController;
|
||||
private TaskStackListenerImpl mTaskStackListener;
|
||||
private Consumer<Integer> mRotWatcherListener;
|
||||
private boolean mIsNavigationBarShowing;
|
||||
|
||||
private final Runnable mRemoveRotationProposal =
|
||||
() -> setRotateSuggestionButtonState(false /* visible */);
|
||||
private final Runnable mCancelPendingRotationProposal =
|
||||
() -> mPendingRotationSuggestion = false;
|
||||
private Animator mRotateHideAnimator;
|
||||
private boolean mAccessibilityFeedbackEnabled;
|
||||
|
||||
private final MetricsLogger mMetricsLogger = Dependency.get(MetricsLogger.class);
|
||||
private final ViewRippler mViewRippler = new ViewRippler();
|
||||
|
||||
private final Stub mRotationWatcher = new Stub() {
|
||||
@Override
|
||||
public void onRotationChanged(final int rotation) throws RemoteException {
|
||||
// We need this to be scheduled as early as possible to beat the redrawing of
|
||||
// window in response to the orientation change.
|
||||
Handler h = getCurrentView().getHandler();
|
||||
Message msg = Message.obtain(h, () -> {
|
||||
// If the screen rotation changes while locked, potentially update lock to flow with
|
||||
// new screen rotation and hide any showing suggestions.
|
||||
if (mRotationLockController.isRotationLocked()) {
|
||||
if (shouldOverrideUserLockPrefs(rotation)) {
|
||||
setRotationLockedAtAngle(rotation);
|
||||
}
|
||||
setRotateSuggestionButtonState(false /* visible */, true /* forced */);
|
||||
}
|
||||
|
||||
if (mRotWatcherListener != null) {
|
||||
mRotWatcherListener.accept(rotation);
|
||||
}
|
||||
});
|
||||
msg.setAsynchronous(true);
|
||||
h.sendMessageAtFrontOfQueue(msg);
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Determines if rotation suggestions disabled2 flag exists in flag
|
||||
* @param disable2Flags see if rotation suggestion flag exists in this flag
|
||||
* @return whether flag exists
|
||||
*/
|
||||
static boolean hasDisable2RotateSuggestionFlag(int disable2Flags) {
|
||||
return (disable2Flags & StatusBarManager.DISABLE2_ROTATE_SUGGESTIONS) != 0;
|
||||
}
|
||||
|
||||
public void setStyle(@StyleRes int styleRes) {
|
||||
mStyleRes = styleRes;
|
||||
public RotationContextButton(@IdRes int buttonResId, @DrawableRes int iconResId,
|
||||
@NonNull Context context, @StyleRes int style) {
|
||||
super(buttonResId, iconResId);
|
||||
|
||||
mStyleRes = style;
|
||||
mIsNavigationBarShowing = true;
|
||||
mRotationLockController = Dependency.get(RotationLockController.class);
|
||||
|
||||
// Register the task stack listener
|
||||
mTaskStackListener = new TaskStackListenerImpl();
|
||||
ActivityManagerWrapper.getInstance().registerTaskStackListener(mTaskStackListener);
|
||||
setOnClickListener(this::onRotateSuggestionClick);
|
||||
setOnHoverListener(this::onRotateSuggestionHover);
|
||||
|
||||
try {
|
||||
WindowManagerGlobal.getWindowManagerService()
|
||||
.watchRotation(mRotationWatcher, context.getDisplay().getDisplayId());
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
}
|
||||
}
|
||||
|
||||
public void addRotationCallback(Consumer<Integer> watcher) {
|
||||
mRotWatcherListener = watcher;
|
||||
}
|
||||
|
||||
public void setRotateSuggestionButtonState(boolean visible) {
|
||||
setRotateSuggestionButtonState(visible, false /* force */);
|
||||
}
|
||||
|
||||
public void setRotateSuggestionButtonState(final boolean visible, final boolean force) {
|
||||
// At any point the the button can become invisible because an a11y service became active.
|
||||
// Similarly, a call to make the button visible may be rejected because an a11y service is
|
||||
// active. Must account for this.
|
||||
// Rerun a show animation to indicate change but don't rerun a hide animation
|
||||
if (!visible && !isVisible()) return;
|
||||
|
||||
final View view = getCurrentView();
|
||||
if (view == null) return;
|
||||
|
||||
final KeyButtonDrawable currentDrawable = getImageDrawable();
|
||||
if (currentDrawable == null) return;
|
||||
|
||||
// Clear any pending suggestion flag as it has either been nullified or is being shown
|
||||
mPendingRotationSuggestion = false;
|
||||
view.removeCallbacks(mCancelPendingRotationProposal);
|
||||
|
||||
// Handle the visibility change and animation
|
||||
if (visible) { // Appear and change (cannot force)
|
||||
// Stop and clear any currently running hide animations
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) {
|
||||
mRotateHideAnimator.cancel();
|
||||
}
|
||||
mRotateHideAnimator = null;
|
||||
|
||||
// Reset the alpha if any has changed due to hide animation
|
||||
view.setAlpha(1f);
|
||||
|
||||
// Run the rotate icon's animation if it has one
|
||||
if (currentDrawable.canAnimate()) {
|
||||
currentDrawable.resetAnimation();
|
||||
currentDrawable.startAnimation();
|
||||
}
|
||||
|
||||
if (!isRotateSuggestionIntroduced()) mViewRippler.start(view);
|
||||
|
||||
// Set visibility unless a11y service is active.
|
||||
show();
|
||||
} else { // Hide
|
||||
mViewRippler.stop(); // Prevent any pending ripples, force hide or not
|
||||
|
||||
if (force) {
|
||||
// If a hide animator is running stop it and make invisible
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) {
|
||||
mRotateHideAnimator.pause();
|
||||
}
|
||||
hide();
|
||||
return;
|
||||
}
|
||||
|
||||
// Don't start any new hide animations if one is running
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) return;
|
||||
|
||||
ObjectAnimator fadeOut = ObjectAnimator.ofFloat(view, "alpha", 0f);
|
||||
fadeOut.setDuration(BUTTON_FADE_IN_OUT_DURATION_MS);
|
||||
fadeOut.setInterpolator(Interpolators.LINEAR);
|
||||
fadeOut.addListener(new AnimatorListenerAdapter() {
|
||||
@Override
|
||||
public void onAnimationEnd(Animator animation) {
|
||||
hide();
|
||||
}
|
||||
});
|
||||
|
||||
mRotateHideAnimator = fadeOut;
|
||||
fadeOut.start();
|
||||
}
|
||||
}
|
||||
|
||||
public void setAccessibilityFeedbackEnabled(boolean flag) {
|
||||
mAccessibilityFeedbackEnabled = flag;
|
||||
}
|
||||
|
||||
public void setRotationLockedAtAngle(int rotationSuggestion) {
|
||||
mRotationLockController.setRotationLockedAtAngle(true /* locked */, rotationSuggestion);
|
||||
}
|
||||
|
||||
public boolean isRotationLocked() {
|
||||
return mRotationLockController.isRotationLocked();
|
||||
}
|
||||
|
||||
public void onRotationProposal(int rotation, int windowRotation, boolean isValid) {
|
||||
// This method will be called on rotation suggestion changes even if the proposed rotation
|
||||
// is not valid for the top app. Use invalid rotation choices as a signal to remove the
|
||||
// rotate button if shown.
|
||||
if (!isValid) {
|
||||
setRotateSuggestionButtonState(false /* visible */);
|
||||
return;
|
||||
}
|
||||
|
||||
// If window rotation matches suggested rotation, remove any current suggestions
|
||||
if (rotation == windowRotation) {
|
||||
getCurrentView().removeCallbacks(mRemoveRotationProposal);
|
||||
setRotateSuggestionButtonState(false /* visible */);
|
||||
return;
|
||||
}
|
||||
|
||||
// Prepare to show the navbar icon by updating the icon style to change anim params
|
||||
mLastRotationSuggestion = rotation; // Remember rotation for click
|
||||
final boolean rotationCCW = isRotationAnimationCCW(windowRotation, rotation);
|
||||
int style;
|
||||
if (windowRotation == Surface.ROTATION_0 || windowRotation == Surface.ROTATION_180) {
|
||||
style = rotationCCW ? R.style.RotateButtonCCWStart90 : R.style.RotateButtonCWStart90;
|
||||
} else { // 90 or 270
|
||||
style = rotationCCW ? R.style.RotateButtonCCWStart0 : R.style.RotateButtonCWStart0;
|
||||
}
|
||||
mStyleRes = style;
|
||||
updateIcon();
|
||||
|
||||
if (mIsNavigationBarShowing) {
|
||||
// The navbar is visible so show the icon right away
|
||||
showAndLogRotationSuggestion();
|
||||
} else {
|
||||
// If the navbar isn't shown, flag the rotate icon to be shown should the navbar become
|
||||
// visible given some time limit.
|
||||
mPendingRotationSuggestion = true;
|
||||
getCurrentView().removeCallbacks(mCancelPendingRotationProposal);
|
||||
getCurrentView().postDelayed(mCancelPendingRotationProposal,
|
||||
NAVBAR_HIDDEN_PENDING_ICON_TIMEOUT_MS);
|
||||
}
|
||||
}
|
||||
|
||||
public void onDisable2FlagChanged(int state2) {
|
||||
final boolean rotateSuggestionsDisabled = hasDisable2RotateSuggestionFlag(state2);
|
||||
if (rotateSuggestionsDisabled) onRotationSuggestionsDisabled();
|
||||
}
|
||||
|
||||
public void onNavigationBarWindowVisibilityChange(boolean showing) {
|
||||
if (mIsNavigationBarShowing != showing) {
|
||||
mIsNavigationBarShowing = showing;
|
||||
|
||||
// If the navbar is visible, show the rotate button if there's a pending suggestion
|
||||
if (showing && mPendingRotationSuggestion) {
|
||||
showAndLogRotationSuggestion();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -58,4 +298,168 @@ public class RotationContextButton extends ContextualButton {
|
||||
Context context = new ContextThemeWrapper(getContext().getApplicationContext(), mStyleRes);
|
||||
return KeyButtonDrawable.create(context, mIconResId, false /* shadow */);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDestroy() {
|
||||
// Unregister the task stack listener
|
||||
ActivityManagerWrapper.getInstance().unregisterTaskStackListener(mTaskStackListener);
|
||||
|
||||
try {
|
||||
WindowManagerGlobal.getWindowManagerService().removeRotationWatcher(mRotationWatcher);
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
}
|
||||
}
|
||||
|
||||
private void onRotateSuggestionClick(View v) {
|
||||
mMetricsLogger.action(MetricsEvent.ACTION_ROTATION_SUGGESTION_ACCEPTED);
|
||||
incrementNumAcceptedRotationSuggestionsIfNeeded();
|
||||
setRotationLockedAtAngle(mLastRotationSuggestion);
|
||||
}
|
||||
|
||||
private boolean onRotateSuggestionHover(View v, MotionEvent event) {
|
||||
final int action = event.getActionMasked();
|
||||
mHoveringRotationSuggestion = (action == MotionEvent.ACTION_HOVER_ENTER)
|
||||
|| (action == MotionEvent.ACTION_HOVER_MOVE);
|
||||
rescheduleRotationTimeout(true /* reasonHover */);
|
||||
return false; // Must return false so a11y hover events are dispatched correctly.
|
||||
}
|
||||
|
||||
private void onRotationSuggestionsDisabled() {
|
||||
// Immediately hide the rotate button and clear any planned removal
|
||||
setRotateSuggestionButtonState(false /* visible */, true /* force */);
|
||||
getCurrentView().removeCallbacks(mRemoveRotationProposal);
|
||||
}
|
||||
|
||||
private void showAndLogRotationSuggestion() {
|
||||
setRotateSuggestionButtonState(true /* visible */);
|
||||
rescheduleRotationTimeout(false /* reasonHover */);
|
||||
mMetricsLogger.visible(MetricsEvent.ROTATION_SUGGESTION_SHOWN);
|
||||
}
|
||||
|
||||
private boolean shouldOverrideUserLockPrefs(final int rotation) {
|
||||
// Only override user prefs when returning to the natural rotation (normally portrait).
|
||||
// Don't let apps that force landscape or 180 alter user lock.
|
||||
return rotation == NATURAL_ROTATION;
|
||||
}
|
||||
|
||||
private boolean isRotationAnimationCCW(int from, int to) {
|
||||
// All 180deg WM rotation animations are CCW, match that
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_90) return false;
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_180) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_0 && to == Surface.ROTATION_270) return true;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_0) return true;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_180) return false;
|
||||
if (from == Surface.ROTATION_90 && to == Surface.ROTATION_270) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_0) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_90) return true;
|
||||
if (from == Surface.ROTATION_180 && to == Surface.ROTATION_270) return false;
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_0) return false;
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_90) return true; //180d so CCW
|
||||
if (from == Surface.ROTATION_270 && to == Surface.ROTATION_180) return true;
|
||||
return false; // Default
|
||||
}
|
||||
|
||||
private void rescheduleRotationTimeout(final boolean reasonHover) {
|
||||
// May be called due to a new rotation proposal or a change in hover state
|
||||
if (reasonHover) {
|
||||
// Don't reschedule if a hide animator is running
|
||||
if (mRotateHideAnimator != null && mRotateHideAnimator.isRunning()) return;
|
||||
// Don't reschedule if not visible
|
||||
if (!isVisible()) return;
|
||||
}
|
||||
|
||||
// Stop any pending removal
|
||||
getCurrentView().removeCallbacks(mRemoveRotationProposal);
|
||||
// Schedule timeout
|
||||
getCurrentView().postDelayed(mRemoveRotationProposal, computeRotationProposalTimeout());
|
||||
}
|
||||
|
||||
private int computeRotationProposalTimeout() {
|
||||
if (mAccessibilityFeedbackEnabled) return 20000;
|
||||
if (mHoveringRotationSuggestion) return 16000;
|
||||
return 10000;
|
||||
}
|
||||
|
||||
private boolean isRotateSuggestionIntroduced() {
|
||||
ContentResolver cr = getContext().getContentResolver();
|
||||
return Settings.Secure.getInt(cr, Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED, 0)
|
||||
>= NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION;
|
||||
}
|
||||
|
||||
private void incrementNumAcceptedRotationSuggestionsIfNeeded() {
|
||||
// Get the number of accepted suggestions
|
||||
ContentResolver cr = getContext().getContentResolver();
|
||||
final int numSuggestions = Settings.Secure.getInt(cr,
|
||||
Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED, 0);
|
||||
|
||||
// Increment the number of accepted suggestions only if it would change intro mode
|
||||
if (numSuggestions < NUM_ACCEPTED_ROTATION_SUGGESTIONS_FOR_INTRODUCTION) {
|
||||
Settings.Secure.putInt(cr, Settings.Secure.NUM_ROTATION_SUGGESTIONS_ACCEPTED,
|
||||
numSuggestions + 1);
|
||||
}
|
||||
}
|
||||
|
||||
private class TaskStackListenerImpl extends SysUiTaskStackChangeListener {
|
||||
// Invalidate any rotation suggestion on task change or activity orientation change
|
||||
// Note: all callbacks happen on main thread
|
||||
|
||||
@Override
|
||||
public void onTaskStackChanged() {
|
||||
setRotateSuggestionButtonState(false /* visible */);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onTaskRemoved(int taskId) {
|
||||
setRotateSuggestionButtonState(false /* visible */);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onTaskMovedToFront(int taskId) {
|
||||
setRotateSuggestionButtonState(false /* visible */);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActivityRequestedOrientationChanged(int taskId, int requestedOrientation) {
|
||||
// Only hide the icon if the top task changes its requestedOrientation
|
||||
// Launcher can alter its requestedOrientation while it's not on top, don't hide on this
|
||||
Optional.ofNullable(ActivityManagerWrapper.getInstance())
|
||||
.map(ActivityManagerWrapper::getRunningTask)
|
||||
.ifPresent(a -> {
|
||||
if (a.id == taskId) setRotateSuggestionButtonState(false /* visible */);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
private class ViewRippler {
|
||||
private static final int RIPPLE_OFFSET_MS = 50;
|
||||
private static final int RIPPLE_INTERVAL_MS = 2000;
|
||||
private View mRoot;
|
||||
|
||||
public void start(View root) {
|
||||
stop(); // Stop any pending ripple animations
|
||||
|
||||
mRoot = root;
|
||||
|
||||
// Schedule pending ripples, offset the 1st to avoid problems with visibility change
|
||||
mRoot.postOnAnimationDelayed(mRipple, RIPPLE_OFFSET_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 2 * RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 3 * RIPPLE_INTERVAL_MS);
|
||||
mRoot.postOnAnimationDelayed(mRipple, 4 * RIPPLE_INTERVAL_MS);
|
||||
}
|
||||
|
||||
public void stop() {
|
||||
if (mRoot != null) mRoot.removeCallbacks(mRipple);
|
||||
}
|
||||
|
||||
private final Runnable mRipple = new Runnable() {
|
||||
@Override
|
||||
public void run() { // Cause the ripple to fire via false presses
|
||||
if (!mRoot.isAttachedToWindow()) return;
|
||||
mRoot.setPressed(true /* pressed */);
|
||||
mRoot.setPressed(false /* pressed */);
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,107 @@
|
||||
/*
|
||||
* Copyright (C) 2018 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.phone;
|
||||
|
||||
import static org.mockito.Mockito.doReturn;
|
||||
import static org.mockito.Mockito.mock;
|
||||
import static org.mockito.Mockito.spy;
|
||||
import static org.mockito.Mockito.times;
|
||||
import static org.mockito.Mockito.verify;
|
||||
|
||||
import android.support.test.InstrumentationRegistry;
|
||||
import android.support.test.filters.SmallTest;
|
||||
import android.support.test.runner.AndroidJUnit4;
|
||||
import android.view.View;
|
||||
|
||||
import com.android.systemui.SysuiTestableContext;
|
||||
import com.android.systemui.SysuiTestCase;
|
||||
import com.android.systemui.TestableDependency;
|
||||
import com.android.systemui.statusbar.policy.KeyButtonDrawable;
|
||||
import com.android.systemui.statusbar.policy.RotationLockController;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.mockito.MockitoAnnotations;
|
||||
|
||||
/** atest NavigationBarRotationContextTest */
|
||||
@SmallTest
|
||||
@RunWith(AndroidJUnit4.class)
|
||||
public class NavigationBarRotationContextTest extends SysuiTestCase {
|
||||
static final int RES_UNDEF = 0;
|
||||
static final int DEFAULT_ROTATE = 0;
|
||||
|
||||
@Rule
|
||||
public final SysuiTestableContext mContext = new SysuiTestableContext(
|
||||
InstrumentationRegistry.getContext(), getLeakCheck());
|
||||
private final TestableDependency mDependency = new TestableDependency(mContext);
|
||||
private RotationContextButton mButton;
|
||||
|
||||
@Before
|
||||
public void setup() {
|
||||
MockitoAnnotations.initMocks(this);
|
||||
mDependency.injectMockDependency(RotationLockController.class);
|
||||
|
||||
final View view = new View(mContext);
|
||||
mButton = spy(new RotationContextButton(RES_UNDEF, RES_UNDEF, mContext, RES_UNDEF));
|
||||
final KeyButtonDrawable kbd = mock(KeyButtonDrawable.class);
|
||||
doReturn(view).when(mButton).getCurrentView();
|
||||
doReturn(kbd).when(mButton).getNewDrawable();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOnInvalidRotationProposal() {
|
||||
mButton.onRotationProposal(DEFAULT_ROTATE, DEFAULT_ROTATE + 1, false /* isValid */);
|
||||
verify(mButton, times(1)).setRotateSuggestionButtonState(false /* visible */);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOnSameRotationProposal() {
|
||||
mButton.onRotationProposal(DEFAULT_ROTATE, DEFAULT_ROTATE, true /* isValid */);
|
||||
verify(mButton, times(1)).setRotateSuggestionButtonState(false /* visible */);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOnRotationProposalShowButtonShowNav() {
|
||||
// No navigation bar should not call to set visibility state
|
||||
mButton.onNavigationBarWindowVisibilityChange(false /* showing */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(false /* visible */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(true /* visible */);
|
||||
|
||||
// No navigation bar with rotation change should not call to set visibility state
|
||||
mButton.onRotationProposal(DEFAULT_ROTATE, DEFAULT_ROTATE + 1, true /* isValid */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(false /* visible */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(true /* visible */);
|
||||
|
||||
// Since rotation has changed rotation should be pending, show mButton when showing nav bar
|
||||
mButton.onNavigationBarWindowVisibilityChange(true /* showing */);
|
||||
verify(mButton, times(1)).setRotateSuggestionButtonState(true /* visible */);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testOnRotationProposalShowButton() {
|
||||
// Navigation bar being visible should not call to set visibility state
|
||||
mButton.onNavigationBarWindowVisibilityChange(true /* showing */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(false /* visible */);
|
||||
verify(mButton, times(0)).setRotateSuggestionButtonState(true /* visible */);
|
||||
|
||||
// Navigation bar is visible and rotation requested
|
||||
mButton.onRotationProposal(DEFAULT_ROTATE, DEFAULT_ROTATE + 1, true /* isValid */);
|
||||
verify(mButton, times(1)).setRotateSuggestionButtonState(true /* visible */);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user