mirror of
https://github.com/LawnchairLauncher/lawnchair.git
synced 2026-02-19 18:58:19 +00:00
This reverts commit d82503fc42.
Reason for revert: Introduced crashes
Change-Id: I0956f746e25ccfe655169580fd802d33f99c0652
418 lines
14 KiB
Java
418 lines
14 KiB
Java
/*
|
|
* Copyright (C) 2017 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;
|
|
|
|
import static com.android.launcher3.util.FlagDebugUtils.appendFlag;
|
|
import static com.android.launcher3.util.FlagDebugUtils.formatFlagChange;
|
|
import static com.android.launcher3.util.SystemUiController.UI_STATE_FULLSCREEN_TASK;
|
|
|
|
import static java.lang.annotation.RetentionPolicy.SOURCE;
|
|
|
|
import android.app.Activity;
|
|
import android.content.Context;
|
|
import android.content.ContextWrapper;
|
|
import android.content.Intent;
|
|
import android.content.res.Configuration;
|
|
import android.os.Bundle;
|
|
import android.util.Log;
|
|
import android.window.OnBackInvokedDispatcher;
|
|
|
|
import androidx.annotation.IntDef;
|
|
|
|
import com.android.launcher3.DeviceProfile.OnDeviceProfileChangeListener;
|
|
import com.android.launcher3.logging.StatsLogManager;
|
|
import com.android.launcher3.testing.TestLogging;
|
|
import com.android.launcher3.testing.shared.TestProtocol;
|
|
import com.android.launcher3.util.RunnableList;
|
|
import com.android.launcher3.util.SystemUiController;
|
|
import com.android.launcher3.util.ViewCache;
|
|
import com.android.launcher3.views.ActivityContext;
|
|
import com.android.launcher3.views.ScrimView;
|
|
|
|
import java.io.PrintWriter;
|
|
import java.lang.annotation.Retention;
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.StringJoiner;
|
|
|
|
/**
|
|
* Launcher BaseActivity
|
|
*/
|
|
public abstract class BaseActivity extends Activity implements ActivityContext {
|
|
|
|
private static final String TAG = "BaseActivity";
|
|
static final boolean DEBUG = false;
|
|
|
|
public static final int INVISIBLE_BY_STATE_HANDLER = 1 << 0;
|
|
public static final int INVISIBLE_BY_APP_TRANSITIONS = 1 << 1;
|
|
public static final int INVISIBLE_BY_PENDING_FLAGS = 1 << 2;
|
|
|
|
// This is not treated as invisibility flag, but adds as a hint for an incomplete transition.
|
|
// When the wallpaper animation runs, it replaces this flag with a proper invisibility
|
|
// flag, INVISIBLE_BY_PENDING_FLAGS only for the duration of that animation.
|
|
public static final int PENDING_INVISIBLE_BY_WALLPAPER_ANIMATION = 1 << 3;
|
|
|
|
private static final int INVISIBLE_FLAGS =
|
|
INVISIBLE_BY_STATE_HANDLER | INVISIBLE_BY_APP_TRANSITIONS | INVISIBLE_BY_PENDING_FLAGS;
|
|
public static final int STATE_HANDLER_INVISIBILITY_FLAGS =
|
|
INVISIBLE_BY_STATE_HANDLER | PENDING_INVISIBLE_BY_WALLPAPER_ANIMATION;
|
|
public static final int INVISIBLE_ALL =
|
|
INVISIBLE_FLAGS | PENDING_INVISIBLE_BY_WALLPAPER_ANIMATION;
|
|
|
|
@Retention(SOURCE)
|
|
@IntDef(
|
|
flag = true,
|
|
value = {INVISIBLE_BY_STATE_HANDLER, INVISIBLE_BY_APP_TRANSITIONS,
|
|
INVISIBLE_BY_PENDING_FLAGS, PENDING_INVISIBLE_BY_WALLPAPER_ANIMATION})
|
|
public @interface InvisibilityFlags {
|
|
}
|
|
|
|
private final ArrayList<OnDeviceProfileChangeListener> mDPChangeListeners = new ArrayList<>();
|
|
private final ArrayList<MultiWindowModeChangedListener> mMultiWindowModeChangedListeners =
|
|
new ArrayList<>();
|
|
|
|
protected DeviceProfile mDeviceProfile;
|
|
protected SystemUiController mSystemUiController;
|
|
private StatsLogManager mStatsLogManager;
|
|
|
|
|
|
public static final int ACTIVITY_STATE_STARTED = 1 << 0;
|
|
public static final int ACTIVITY_STATE_RESUMED = 1 << 1;
|
|
|
|
/**
|
|
* State flags indicating that the activity has received one frame after resume, and was
|
|
* not immediately paused.
|
|
*/
|
|
public static final int ACTIVITY_STATE_DEFERRED_RESUMED = 1 << 2;
|
|
|
|
public static final int ACTIVITY_STATE_WINDOW_FOCUSED = 1 << 3;
|
|
|
|
/**
|
|
* State flag indicating if the user is active or the activity when to background as a result
|
|
* of user action.
|
|
*
|
|
* @see #isUserActive()
|
|
*/
|
|
public static final int ACTIVITY_STATE_USER_ACTIVE = 1 << 4;
|
|
|
|
/**
|
|
* State flag indicating that a state transition is in progress
|
|
*/
|
|
public static final int ACTIVITY_STATE_TRANSITION_ACTIVE = 1 << 6;
|
|
|
|
@Retention(SOURCE)
|
|
@IntDef(
|
|
flag = true,
|
|
value = {ACTIVITY_STATE_STARTED,
|
|
ACTIVITY_STATE_RESUMED,
|
|
ACTIVITY_STATE_DEFERRED_RESUMED,
|
|
ACTIVITY_STATE_WINDOW_FOCUSED,
|
|
ACTIVITY_STATE_USER_ACTIVE,
|
|
ACTIVITY_STATE_TRANSITION_ACTIVE})
|
|
public @interface ActivityFlags {
|
|
}
|
|
|
|
/** Returns a human-readable string for the specified {@link ActivityFlags}. */
|
|
public static String getActivityStateString(@ActivityFlags int flags) {
|
|
StringJoiner result = new StringJoiner("|");
|
|
appendFlag(result, flags, ACTIVITY_STATE_STARTED, "state_started");
|
|
appendFlag(result, flags, ACTIVITY_STATE_RESUMED, "state_resumed");
|
|
appendFlag(result, flags, ACTIVITY_STATE_DEFERRED_RESUMED, "state_deferred_resumed");
|
|
appendFlag(result, flags, ACTIVITY_STATE_WINDOW_FOCUSED, "state_window_focused");
|
|
appendFlag(result, flags, ACTIVITY_STATE_USER_ACTIVE, "state_user_active");
|
|
appendFlag(result, flags, ACTIVITY_STATE_TRANSITION_ACTIVE, "state_transition_active");
|
|
return result.toString();
|
|
}
|
|
|
|
@ActivityFlags
|
|
private int mActivityFlags;
|
|
|
|
// When the recents animation is running, the visibility of the Launcher is managed by the
|
|
// animation
|
|
@InvisibilityFlags
|
|
private int mForceInvisible;
|
|
|
|
private final ViewCache mViewCache = new ViewCache();
|
|
|
|
@Retention(SOURCE)
|
|
@IntDef({EVENT_STARTED, EVENT_RESUMED, EVENT_STOPPED, EVENT_DESTROYED})
|
|
public @interface ActivityEvent { }
|
|
public static final int EVENT_STARTED = 0;
|
|
public static final int EVENT_RESUMED = 1;
|
|
public static final int EVENT_STOPPED = 2;
|
|
public static final int EVENT_DESTROYED = 3;
|
|
|
|
// Callback array that corresponds to events defined in @ActivityEvent
|
|
private final RunnableList[] mEventCallbacks =
|
|
{new RunnableList(), new RunnableList(), new RunnableList(), new RunnableList()};
|
|
|
|
@Override
|
|
public ViewCache getViewCache() {
|
|
return mViewCache;
|
|
}
|
|
|
|
@Override
|
|
public DeviceProfile getDeviceProfile() {
|
|
return mDeviceProfile;
|
|
}
|
|
|
|
@Override
|
|
public List<OnDeviceProfileChangeListener> getOnDeviceProfileChangeListeners() {
|
|
return mDPChangeListeners;
|
|
}
|
|
|
|
/**
|
|
* Returns {@link StatsLogManager} for user event logging.
|
|
*/
|
|
@Override
|
|
public StatsLogManager getStatsLogManager() {
|
|
if (mStatsLogManager == null) {
|
|
mStatsLogManager = StatsLogManager.newInstance(this);
|
|
}
|
|
return mStatsLogManager;
|
|
}
|
|
|
|
public SystemUiController getSystemUiController() {
|
|
if (mSystemUiController == null) {
|
|
mSystemUiController = new SystemUiController(getWindow().getDecorView());
|
|
}
|
|
return mSystemUiController;
|
|
}
|
|
|
|
public ScrimView getScrimView() {
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
}
|
|
|
|
@Override
|
|
protected void onCreate(Bundle savedInstanceState) {
|
|
super.onCreate(savedInstanceState);
|
|
registerBackDispatcher();
|
|
}
|
|
|
|
@Override
|
|
protected void onStart() {
|
|
addActivityFlags(ACTIVITY_STATE_STARTED);
|
|
super.onStart();
|
|
mEventCallbacks[EVENT_STARTED].executeAllAndClear();
|
|
}
|
|
|
|
@Override
|
|
protected void onResume() {
|
|
setResumed();
|
|
super.onResume();
|
|
mEventCallbacks[EVENT_RESUMED].executeAllAndClear();
|
|
}
|
|
|
|
@Override
|
|
protected void onUserLeaveHint() {
|
|
removeActivityFlags(ACTIVITY_STATE_USER_ACTIVE);
|
|
super.onUserLeaveHint();
|
|
}
|
|
|
|
@Override
|
|
public void onMultiWindowModeChanged(boolean isInMultiWindowMode, Configuration newConfig) {
|
|
super.onMultiWindowModeChanged(isInMultiWindowMode, newConfig);
|
|
for (int i = mMultiWindowModeChangedListeners.size() - 1; i >= 0; i--) {
|
|
mMultiWindowModeChangedListeners.get(i).onMultiWindowModeChanged(isInMultiWindowMode);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onStop() {
|
|
removeActivityFlags(ACTIVITY_STATE_STARTED | ACTIVITY_STATE_USER_ACTIVE);
|
|
mForceInvisible = 0;
|
|
super.onStop();
|
|
mEventCallbacks[EVENT_STOPPED].executeAllAndClear();
|
|
|
|
|
|
// Reset the overridden sysui flags used for the task-swipe launch animation, this is a
|
|
// catch all for if we do not get resumed (and therefore not paused below)
|
|
getSystemUiController().updateUiState(UI_STATE_FULLSCREEN_TASK, 0);
|
|
}
|
|
|
|
@Override
|
|
protected void onDestroy() {
|
|
super.onDestroy();
|
|
mEventCallbacks[EVENT_DESTROYED].executeAllAndClear();
|
|
}
|
|
|
|
@Override
|
|
protected void onPause() {
|
|
setPaused();
|
|
super.onPause();
|
|
|
|
// Reset the overridden sysui flags used for the task-swipe launch animation, we do this
|
|
// here instead of at the end of the animation because the start of the new activity does
|
|
// not happen immediately, which would cause us to reset to launcher's sysui flags and then
|
|
// back to the new app (causing a flash)
|
|
getSystemUiController().updateUiState(UI_STATE_FULLSCREEN_TASK, 0);
|
|
}
|
|
|
|
@Override
|
|
public void onWindowFocusChanged(boolean hasFocus) {
|
|
super.onWindowFocusChanged(hasFocus);
|
|
if (hasFocus) {
|
|
addActivityFlags(ACTIVITY_STATE_WINDOW_FOCUSED);
|
|
} else {
|
|
removeActivityFlags(ACTIVITY_STATE_WINDOW_FOCUSED);
|
|
}
|
|
}
|
|
|
|
protected void registerBackDispatcher() {
|
|
if (Utilities.ATLEAST_T) {
|
|
getOnBackInvokedDispatcher().registerOnBackInvokedCallback(
|
|
OnBackInvokedDispatcher.PRIORITY_DEFAULT,
|
|
() -> {
|
|
onBackPressed();
|
|
TestLogging.recordEvent(TestProtocol.SEQUENCE_MAIN, "onBackInvoked");
|
|
});
|
|
}
|
|
}
|
|
|
|
public boolean isStarted() {
|
|
return (mActivityFlags & ACTIVITY_STATE_STARTED) != 0;
|
|
}
|
|
|
|
/**
|
|
* isResumed in already defined as a hidden final method in Activity.java
|
|
*/
|
|
public boolean hasBeenResumed() {
|
|
return (mActivityFlags & ACTIVITY_STATE_RESUMED) != 0;
|
|
}
|
|
|
|
/**
|
|
* Sets the activity to appear as paused.
|
|
*/
|
|
public void setPaused() {
|
|
removeActivityFlags(ACTIVITY_STATE_RESUMED | ACTIVITY_STATE_DEFERRED_RESUMED);
|
|
}
|
|
|
|
/**
|
|
* Sets the activity to appear as resumed.
|
|
*/
|
|
public void setResumed() {
|
|
addActivityFlags(ACTIVITY_STATE_RESUMED | ACTIVITY_STATE_USER_ACTIVE);
|
|
}
|
|
|
|
public boolean isUserActive() {
|
|
return (mActivityFlags & ACTIVITY_STATE_USER_ACTIVE) != 0;
|
|
}
|
|
|
|
public int getActivityFlags() {
|
|
return mActivityFlags;
|
|
}
|
|
|
|
protected void addActivityFlags(int toAdd) {
|
|
final int oldFlags = mActivityFlags;
|
|
mActivityFlags |= toAdd;
|
|
if (DEBUG) {
|
|
Log.d(TAG, "Launcher flags updated: " + formatFlagChange(mActivityFlags, oldFlags,
|
|
BaseActivity::getActivityStateString));
|
|
}
|
|
onActivityFlagsChanged(toAdd);
|
|
}
|
|
|
|
protected void removeActivityFlags(int toRemove) {
|
|
final int oldFlags = mActivityFlags;
|
|
mActivityFlags &= ~toRemove;
|
|
if (DEBUG) {
|
|
Log.d(TAG, "Launcher flags updated: " + formatFlagChange(mActivityFlags, oldFlags,
|
|
BaseActivity::getActivityStateString));
|
|
}
|
|
|
|
onActivityFlagsChanged(toRemove);
|
|
}
|
|
|
|
protected void onActivityFlagsChanged(int changeBits) {
|
|
}
|
|
|
|
public void addMultiWindowModeChangedListener(MultiWindowModeChangedListener listener) {
|
|
mMultiWindowModeChangedListeners.add(listener);
|
|
}
|
|
|
|
public void removeMultiWindowModeChangedListener(MultiWindowModeChangedListener listener) {
|
|
mMultiWindowModeChangedListeners.remove(listener);
|
|
}
|
|
|
|
/**
|
|
* Used to set the override visibility state, used only to handle the transition home with the
|
|
* recents animation.
|
|
*
|
|
* @see QuickstepTransitionManager#createWallpaperOpenRunner
|
|
*/
|
|
public void addForceInvisibleFlag(@InvisibilityFlags int flag) {
|
|
mForceInvisible |= flag;
|
|
}
|
|
|
|
public void clearForceInvisibleFlag(@InvisibilityFlags int flag) {
|
|
mForceInvisible &= ~flag;
|
|
}
|
|
|
|
/**
|
|
* @return Wether this activity should be considered invisible regardless of actual visibility.
|
|
*/
|
|
public boolean isForceInvisible() {
|
|
return hasSomeInvisibleFlag(INVISIBLE_FLAGS);
|
|
}
|
|
|
|
public boolean hasSomeInvisibleFlag(int mask) {
|
|
return (mForceInvisible & mask) != 0;
|
|
}
|
|
|
|
/**
|
|
* Adds a callback for the provided activity event
|
|
*/
|
|
public void addEventCallback(@ActivityEvent int event, Runnable callback) {
|
|
mEventCallbacks[event].add(callback);
|
|
}
|
|
|
|
/** Removes a previously added callback */
|
|
public void removeEventCallback(@ActivityEvent int event, Runnable callback) {
|
|
mEventCallbacks[event].remove(callback);
|
|
}
|
|
|
|
public interface MultiWindowModeChangedListener {
|
|
void onMultiWindowModeChanged(boolean isInMultiWindowMode);
|
|
}
|
|
|
|
protected void dumpMisc(String prefix, PrintWriter writer) {
|
|
writer.println(prefix + "deviceProfile isTransposed="
|
|
+ getDeviceProfile().isVerticalBarLayout());
|
|
writer.println(prefix + "orientation=" + getResources().getConfiguration().orientation);
|
|
writer.println(prefix + "mSystemUiController: " + mSystemUiController);
|
|
writer.println(prefix + "mActivityFlags: " + getActivityStateString(mActivityFlags));
|
|
writer.println(prefix + "mForceInvisible: " + mForceInvisible);
|
|
}
|
|
|
|
public static <T extends BaseActivity> T fromContext(Context context) {
|
|
if (context instanceof BaseActivity) {
|
|
return (T) context;
|
|
} else if (context instanceof ActivityContextDelegate) {
|
|
return (T) ((ActivityContextDelegate) context).mDelegate;
|
|
} else if (context instanceof ContextWrapper) {
|
|
return fromContext(((ContextWrapper) context).getBaseContext());
|
|
} else {
|
|
throw new IllegalArgumentException("Cannot find BaseActivity in parent tree");
|
|
}
|
|
}
|
|
}
|