mirror of
https://github.com/LawnchairLauncher/lawnchair.git
synced 2026-02-28 07:46:55 +00:00
Makes the scrim opaque for phones so that it looks like the launcher phone all apps layout. Flag: com.android.wm.shell.enable_tiny_taskbar Test: Manual Bug: 341784466 Change-Id: I0352bf7c13c4ea0a5e1bdecf8b4573fd2008d44b
308 lines
11 KiB
Java
308 lines
11 KiB
Java
/*
|
|
* Copyright (C) 2022 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.launcher3.taskbar.allapps;
|
|
|
|
import static com.android.app.animation.Interpolators.EMPHASIZED;
|
|
import static com.android.launcher3.Flags.enablePredictiveBackGesture;
|
|
import static com.android.launcher3.touch.AllAppsSwipeController.ALL_APPS_FADE_MANUAL;
|
|
import static com.android.launcher3.touch.AllAppsSwipeController.SCRIM_FADE_MANUAL;
|
|
|
|
import android.animation.Animator;
|
|
import android.content.Context;
|
|
import android.graphics.Canvas;
|
|
import android.graphics.Rect;
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
import android.util.AttributeSet;
|
|
import android.view.MotionEvent;
|
|
import android.view.View;
|
|
import android.view.animation.Interpolator;
|
|
import android.window.OnBackInvokedDispatcher;
|
|
|
|
import androidx.annotation.Nullable;
|
|
|
|
import com.android.app.animation.Interpolators;
|
|
import com.android.launcher3.DeviceProfile;
|
|
import com.android.launcher3.Insettable;
|
|
import com.android.launcher3.R;
|
|
import com.android.launcher3.anim.AnimatorListeners;
|
|
import com.android.launcher3.anim.PendingAnimation;
|
|
import com.android.launcher3.config.FeatureFlags;
|
|
import com.android.launcher3.taskbar.allapps.TaskbarAllAppsViewController.TaskbarAllAppsCallbacks;
|
|
import com.android.launcher3.taskbar.overlay.TaskbarOverlayContext;
|
|
import com.android.launcher3.util.Themes;
|
|
import com.android.launcher3.views.AbstractSlideInView;
|
|
|
|
/** Wrapper for taskbar all apps with slide-in behavior. */
|
|
public class TaskbarAllAppsSlideInView extends AbstractSlideInView<TaskbarOverlayContext>
|
|
implements Insettable, DeviceProfile.OnDeviceProfileChangeListener {
|
|
private final Handler mHandler;
|
|
|
|
private TaskbarAllAppsContainerView mAppsView;
|
|
private float mShiftRange;
|
|
private @Nullable Runnable mShowOnFullyAttachedToWindowRunnable;
|
|
|
|
// Initialized in init.
|
|
private TaskbarAllAppsCallbacks mAllAppsCallbacks;
|
|
|
|
public TaskbarAllAppsSlideInView(Context context, AttributeSet attrs) {
|
|
this(context, attrs, 0);
|
|
}
|
|
|
|
public TaskbarAllAppsSlideInView(Context context, AttributeSet attrs,
|
|
int defStyleAttr) {
|
|
super(context, attrs, defStyleAttr);
|
|
mHandler = new Handler(Looper.myLooper());
|
|
}
|
|
|
|
void init(TaskbarAllAppsCallbacks callbacks) {
|
|
mAllAppsCallbacks = callbacks;
|
|
}
|
|
|
|
/** Opens the all apps view. */
|
|
void show(boolean animate) {
|
|
if (mIsOpen || mOpenCloseAnimation.getAnimationPlayer().isRunning()) {
|
|
return;
|
|
}
|
|
mIsOpen = true;
|
|
|
|
addOnAttachStateChangeListener(new OnAttachStateChangeListener() {
|
|
@Override
|
|
public void onViewAttachedToWindow(View v) {
|
|
removeOnAttachStateChangeListener(this);
|
|
// Wait for view and its descendants to be fully attached before starting open.
|
|
mShowOnFullyAttachedToWindowRunnable = () -> showOnFullyAttachedToWindow(animate);
|
|
mHandler.post(mShowOnFullyAttachedToWindowRunnable);
|
|
}
|
|
|
|
@Override
|
|
public void onViewDetachedFromWindow(View v) {
|
|
removeOnAttachStateChangeListener(this);
|
|
}
|
|
});
|
|
attachToContainer();
|
|
}
|
|
|
|
private void showOnFullyAttachedToWindow(boolean animate) {
|
|
mAllAppsCallbacks.onAllAppsTransitionStart(true);
|
|
if (!animate) {
|
|
mAllAppsCallbacks.onAllAppsTransitionEnd(true);
|
|
setTranslationShift(TRANSLATION_SHIFT_OPENED);
|
|
return;
|
|
}
|
|
|
|
setUpOpenAnimation(mAllAppsCallbacks.getOpenDuration());
|
|
Animator animator = mOpenCloseAnimation.getAnimationPlayer();
|
|
animator.setInterpolator(EMPHASIZED);
|
|
animator.addListener(AnimatorListeners.forEndCallback(() -> {
|
|
if (mIsOpen) {
|
|
mAllAppsCallbacks.onAllAppsTransitionEnd(true);
|
|
}
|
|
}));
|
|
animator.start();
|
|
}
|
|
|
|
@Override
|
|
protected void onOpenCloseAnimationPending(PendingAnimation animation) {
|
|
final boolean isOpening = mToTranslationShift == TRANSLATION_SHIFT_OPENED;
|
|
|
|
if (mActivityContext.getDeviceProfile().isPhone) {
|
|
final Interpolator allAppsFadeInterpolator =
|
|
isOpening ? ALL_APPS_FADE_MANUAL : Interpolators.reverse(ALL_APPS_FADE_MANUAL);
|
|
animation.setViewAlpha(mAppsView, 1 - mToTranslationShift, allAppsFadeInterpolator);
|
|
}
|
|
|
|
mAllAppsCallbacks.onAllAppsAnimationPending(animation, isOpening);
|
|
}
|
|
|
|
@Override
|
|
protected Interpolator getScrimInterpolator() {
|
|
if (mActivityContext.getDeviceProfile().isTablet) {
|
|
return super.getScrimInterpolator();
|
|
}
|
|
return mToTranslationShift == TRANSLATION_SHIFT_OPENED
|
|
? SCRIM_FADE_MANUAL
|
|
: Interpolators.reverse(SCRIM_FADE_MANUAL);
|
|
}
|
|
|
|
/** The apps container inside this view. */
|
|
TaskbarAllAppsContainerView getAppsView() {
|
|
return mAppsView;
|
|
}
|
|
|
|
@Override
|
|
protected void handleClose(boolean animate) {
|
|
if (mShowOnFullyAttachedToWindowRunnable != null) {
|
|
mHandler.removeCallbacks(mShowOnFullyAttachedToWindowRunnable);
|
|
mShowOnFullyAttachedToWindowRunnable = null;
|
|
}
|
|
if (mIsOpen) {
|
|
mAllAppsCallbacks.onAllAppsTransitionStart(false);
|
|
}
|
|
handleClose(animate, mAllAppsCallbacks.getCloseDuration());
|
|
}
|
|
|
|
@Override
|
|
protected void onCloseComplete() {
|
|
mAllAppsCallbacks.onAllAppsTransitionEnd(false);
|
|
super.onCloseComplete();
|
|
}
|
|
|
|
@Override
|
|
protected Interpolator getIdleInterpolator() {
|
|
return EMPHASIZED;
|
|
}
|
|
|
|
@Override
|
|
protected boolean isOfType(int type) {
|
|
return (type & TYPE_TASKBAR_ALL_APPS) != 0;
|
|
}
|
|
|
|
@Override
|
|
protected void onFinishInflate() {
|
|
super.onFinishInflate();
|
|
mAppsView = findViewById(R.id.apps_view);
|
|
if (mActivityContext.getDeviceProfile().isPhone) {
|
|
mAppsView.setAlpha(0);
|
|
}
|
|
mContent = mAppsView;
|
|
|
|
// Setup header protection for search bar, if enabled.
|
|
if (FeatureFlags.ENABLE_ALL_APPS_SEARCH_IN_TASKBAR.get()) {
|
|
mAppsView.setOnInvalidateHeaderListener(this::invalidate);
|
|
}
|
|
|
|
DeviceProfile dp = mActivityContext.getDeviceProfile();
|
|
setShiftRange(dp.allAppsShiftRange);
|
|
}
|
|
|
|
@Override
|
|
protected void onAttachedToWindow() {
|
|
super.onAttachedToWindow();
|
|
mActivityContext.addOnDeviceProfileChangeListener(this);
|
|
if (enablePredictiveBackGesture()) {
|
|
mAppsView.getAppsRecyclerViewContainer().setOutlineProvider(mViewOutlineProvider);
|
|
mAppsView.getAppsRecyclerViewContainer().setClipToOutline(true);
|
|
OnBackInvokedDispatcher dispatcher = findOnBackInvokedDispatcher();
|
|
if (dispatcher != null) {
|
|
dispatcher.registerOnBackInvokedCallback(
|
|
OnBackInvokedDispatcher.PRIORITY_DEFAULT, this);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onDetachedFromWindow() {
|
|
super.onDetachedFromWindow();
|
|
mActivityContext.removeOnDeviceProfileChangeListener(this);
|
|
if (enablePredictiveBackGesture()) {
|
|
mAppsView.getAppsRecyclerViewContainer().setOutlineProvider(null);
|
|
mAppsView.getAppsRecyclerViewContainer().setClipToOutline(false);
|
|
OnBackInvokedDispatcher dispatcher = findOnBackInvokedDispatcher();
|
|
if (dispatcher != null) {
|
|
dispatcher.unregisterOnBackInvokedCallback(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void dispatchDraw(Canvas canvas) {
|
|
// We should call drawOnScrimWithBottomOffset() rather than drawOnScrimWithScale(). Because
|
|
// for taskbar all apps, the scrim view is a child view of AbstractSlideInView. Thus scaling
|
|
// down in AbstractSlideInView#onScaleProgressChanged() with SCALE_PROPERTY has already
|
|
// done the job - there is no need to re-apply scale effect here. But it also means we need
|
|
// to pass extra bottom offset to background scrim to fill the bottom gap during predictive
|
|
// back swipe.
|
|
mAppsView.drawOnScrimWithBottomOffset(canvas, getBottomOffsetPx());
|
|
super.dispatchDraw(canvas);
|
|
}
|
|
|
|
@Override
|
|
protected void onLayout(boolean changed, int l, int t, int r, int b) {
|
|
super.onLayout(changed, l, t, r, b);
|
|
setTranslationShift(mTranslationShift);
|
|
}
|
|
|
|
@Override
|
|
protected int getScrimColor(Context context) {
|
|
return mActivityContext.getDeviceProfile().isPhone
|
|
? Themes.getAttrColor(context, R.attr.allAppsScrimColor)
|
|
: context.getColor(R.color.widgets_picker_scrim);
|
|
}
|
|
|
|
@Override
|
|
public boolean onControllerInterceptTouchEvent(MotionEvent ev) {
|
|
if (ev.getAction() == MotionEvent.ACTION_DOWN) {
|
|
mNoIntercept = !mAppsView.shouldContainerScroll(ev)
|
|
|| getTopOpenViewWithType(
|
|
mActivityContext, TYPE_TOUCH_CONTROLLER_NO_INTERCEPT) != null;
|
|
}
|
|
return super.onControllerInterceptTouchEvent(ev);
|
|
}
|
|
|
|
@Override
|
|
public void setInsets(Rect insets) {
|
|
mAppsView.setInsets(insets);
|
|
}
|
|
|
|
@Override
|
|
public void onDeviceProfileChanged(DeviceProfile dp) {
|
|
setShiftRange(dp.allAppsShiftRange);
|
|
setTranslationShift(TRANSLATION_SHIFT_OPENED);
|
|
}
|
|
|
|
private void setShiftRange(float shiftRange) {
|
|
mShiftRange = shiftRange;
|
|
}
|
|
|
|
@Override
|
|
protected float getShiftRange() {
|
|
return mShiftRange;
|
|
}
|
|
|
|
@Override
|
|
protected boolean isEventOverContent(MotionEvent ev) {
|
|
return getPopupContainer().isEventOverView(mAppsView.getVisibleContainerView(), ev);
|
|
}
|
|
|
|
/**
|
|
* In taskbar all apps search mode, we should scale down content inside all apps, rather
|
|
* than the whole all apps bottom sheet, to indicate we will navigate back within the all apps.
|
|
*/
|
|
@Override
|
|
public boolean shouldAnimateContentViewInBackSwipe() {
|
|
return mAllAppsCallbacks.canHandleSearchBackInvoked();
|
|
}
|
|
|
|
@Override
|
|
protected void onUserSwipeToDismissProgressChanged() {
|
|
super.onUserSwipeToDismissProgressChanged();
|
|
mAppsView.setClipChildren(!mIsDismissInProgress);
|
|
mAppsView.getAppsRecyclerViewContainer().setClipChildren(!mIsDismissInProgress);
|
|
}
|
|
|
|
@Override
|
|
public void onBackInvoked() {
|
|
if (mAllAppsCallbacks.handleSearchBackInvoked()) {
|
|
// We need to scale back taskbar all apps if we navigate back within search inside all
|
|
// apps
|
|
post(this::animateSwipeToDismissProgressToStart);
|
|
} else {
|
|
super.onBackInvoked();
|
|
}
|
|
}
|
|
}
|