/* * Copyright (C) 2019 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.quickstep; import static android.content.Intent.ACTION_PACKAGE_ADDED; import static android.content.Intent.ACTION_PACKAGE_CHANGED; import static android.content.Intent.ACTION_PACKAGE_REMOVED; import static com.android.launcher3.Utilities.createHomeIntent; import static com.android.launcher3.config.FeatureFlags.SEPARATE_RECENTS_ACTIVITY; import static com.android.launcher3.util.PackageManagerHelper.getPackageFilter; import static com.android.systemui.shared.system.PackageManagerWrapper.ACTION_PREFERRED_ACTIVITY_CHANGED; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.pm.ActivityInfo; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.util.SparseIntArray; import com.android.launcher3.tracing.OverviewComponentObserverProto; import com.android.launcher3.tracing.TouchInteractionServiceProto; import com.android.launcher3.util.SimpleBroadcastReceiver; import com.android.systemui.shared.system.PackageManagerWrapper; import java.io.PrintWriter; import java.util.ArrayList; import java.util.Objects; import java.util.function.Consumer; /** * Class to keep track of the current overview component based off user preferences and app updates * and provide callers the relevant classes. */ public final class OverviewComponentObserver { private final BroadcastReceiver mUserPreferenceChangeReceiver = new SimpleBroadcastReceiver(this::updateOverviewTargets); private final BroadcastReceiver mOtherHomeAppUpdateReceiver = new SimpleBroadcastReceiver(this::updateOverviewTargets); private final Context mContext; private final RecentsAnimationDeviceState mDeviceState; private final Intent mCurrentHomeIntent; private final Intent mMyHomeIntent; private final Intent mFallbackIntent; private final SparseIntArray mConfigChangesMap = new SparseIntArray(); private Consumer mOverviewChangeListener = b -> { }; private String mUpdateRegisteredPackage; private BaseActivityInterface mActivityInterface; private Intent mOverviewIntent; private boolean mIsHomeAndOverviewSame; private boolean mIsDefaultHome; private boolean mIsHomeDisabled; public OverviewComponentObserver(Context context, RecentsAnimationDeviceState deviceState) { mContext = context; mDeviceState = deviceState; mCurrentHomeIntent = createHomeIntent(); mMyHomeIntent = new Intent(mCurrentHomeIntent).setPackage(mContext.getPackageName()); ResolveInfo info = context.getPackageManager().resolveActivity(mMyHomeIntent, 0); ComponentName myHomeComponent = new ComponentName(context.getPackageName(), info.activityInfo.name); mMyHomeIntent.setComponent(myHomeComponent); mConfigChangesMap.append(myHomeComponent.hashCode(), info.activityInfo.configChanges); ComponentName fallbackComponent = new ComponentName(mContext, RecentsActivity.class); mFallbackIntent = new Intent(Intent.ACTION_MAIN) .addCategory(Intent.CATEGORY_DEFAULT) .setComponent(fallbackComponent) .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); try { ActivityInfo fallbackInfo = context.getPackageManager().getActivityInfo( mFallbackIntent.getComponent(), 0 /* flags */); mConfigChangesMap.append(fallbackComponent.hashCode(), fallbackInfo.configChanges); } catch (PackageManager.NameNotFoundException ignored) { /* Impossible */ } mContext.registerReceiver(mUserPreferenceChangeReceiver, new IntentFilter(ACTION_PREFERRED_ACTIVITY_CHANGED)); updateOverviewTargets(); } /** * Sets a listener for changes in {@link #isHomeAndOverviewSame()} */ public void setOverviewChangeListener(Consumer overviewChangeListener) { mOverviewChangeListener = overviewChangeListener; } public void onSystemUiStateChanged() { if (mDeviceState.isHomeDisabled() != mIsHomeDisabled) { updateOverviewTargets(); } // Notify ALL_APPS touch controller when one handed mode state activated or deactivated if (mDeviceState.isOneHandedModeEnabled()) { mActivityInterface.onOneHandedModeStateChanged(mDeviceState.isOneHandedModeActive()); } } private void updateOverviewTargets(Intent unused) { updateOverviewTargets(); } /** * Update overview intent and {@link BaseActivityInterface} based off the current launcher home * component. */ private void updateOverviewTargets() { ComponentName defaultHome = PackageManagerWrapper.getInstance() .getHomeActivities(new ArrayList<>()); mIsHomeDisabled = mDeviceState.isHomeDisabled(); mIsDefaultHome = Objects.equals(mMyHomeIntent.getComponent(), defaultHome); // Set assistant visibility to 0 from launcher's perspective, ensures any elements that // launcher made invisible become visible again before the new activity control helper // becomes active. if (mActivityInterface != null) { mActivityInterface.onAssistantVisibilityChanged(0.f); } if (SEPARATE_RECENTS_ACTIVITY.get()) { mIsDefaultHome = false; if (defaultHome == null) { defaultHome = mMyHomeIntent.getComponent(); } } if (!mDeviceState.isHomeDisabled() && (defaultHome == null || mIsDefaultHome)) { // User default home is same as out home app. Use Overview integrated in Launcher. mActivityInterface = LauncherActivityInterface.INSTANCE; mIsHomeAndOverviewSame = true; mOverviewIntent = mMyHomeIntent; mCurrentHomeIntent.setComponent(mMyHomeIntent.getComponent()); // Remove any update listener as we don't care about other packages. unregisterOtherHomeAppUpdateReceiver(); } else { // The default home app is a different launcher. Use the fallback Overview instead. mActivityInterface = FallbackActivityInterface.INSTANCE; mIsHomeAndOverviewSame = false; mOverviewIntent = mFallbackIntent; mCurrentHomeIntent.setComponent(defaultHome); // User's default home app can change as a result of package updates of this app (such // as uninstalling the app or removing the "Launcher" feature in an update). // Listen for package updates of this app (and remove any previously attached // package listener). if (defaultHome == null) { unregisterOtherHomeAppUpdateReceiver(); } else if (!defaultHome.getPackageName().equals(mUpdateRegisteredPackage)) { unregisterOtherHomeAppUpdateReceiver(); mUpdateRegisteredPackage = defaultHome.getPackageName(); mContext.registerReceiver(mOtherHomeAppUpdateReceiver, getPackageFilter( mUpdateRegisteredPackage, ACTION_PACKAGE_ADDED, ACTION_PACKAGE_CHANGED, ACTION_PACKAGE_REMOVED)); } } mOverviewChangeListener.accept(mIsHomeAndOverviewSame); } /** * Clean up any registered receivers. */ public void onDestroy() { mContext.unregisterReceiver(mUserPreferenceChangeReceiver); unregisterOtherHomeAppUpdateReceiver(); } private void unregisterOtherHomeAppUpdateReceiver() { if (mUpdateRegisteredPackage != null) { mContext.unregisterReceiver(mOtherHomeAppUpdateReceiver); mUpdateRegisteredPackage = null; } } /** * @return {@code true} if the overview component is able to handle the configuration changes. */ boolean canHandleConfigChanges(ComponentName component, int changes) { final int orientationChange = ActivityInfo.CONFIG_ORIENTATION | ActivityInfo.CONFIG_SCREEN_SIZE; if ((changes & orientationChange) == orientationChange) { // This is just an approximate guess for simple orientation change because the changes // may contain non-public bits (e.g. window configuration). return true; } int configMask = mConfigChangesMap.get(component.hashCode()); return configMask != 0 && (~configMask & changes) == 0; } /** * Get the intent for overview activity. It is used when lockscreen is shown and home was died * in background, we still want to restart the one that will be used after unlock. * * @return the overview intent */ Intent getOverviewIntentIgnoreSysUiState() { return mIsDefaultHome ? mMyHomeIntent : mOverviewIntent; } /** * Get the current intent for going to the overview activity. * * @return the overview intent */ public Intent getOverviewIntent() { return mOverviewIntent; } /** * Get the current intent for going to the home activity. */ public Intent getHomeIntent() { return mCurrentHomeIntent; } /** * Returns true if home and overview are same activity. */ public boolean isHomeAndOverviewSame() { return mIsHomeAndOverviewSame; } /** * Get the current activity control helper for managing interactions to the overview activity. * * @return the current activity control helper */ public BaseActivityInterface getActivityInterface() { return mActivityInterface; } public void dump(PrintWriter pw) { pw.println("OverviewComponentObserver:"); pw.println(" isDefaultHome=" + mIsDefaultHome); pw.println(" isHomeDisabled=" + mIsHomeDisabled); pw.println(" homeAndOverviewSame=" + mIsHomeAndOverviewSame); pw.println(" overviewIntent=" + mOverviewIntent); pw.println(" homeIntent=" + mCurrentHomeIntent); } /** * Used for winscope tracing, see launcher_trace.proto * @see com.android.systemui.shared.tracing.ProtoTraceable#writeToProto * @param serviceProto The parent of this proto message. */ public void writeToProto(TouchInteractionServiceProto.Builder serviceProto) { OverviewComponentObserverProto.Builder overviewComponentObserver = OverviewComponentObserverProto.newBuilder(); overviewComponentObserver.setOverviewActivityStarted(mActivityInterface.isStarted()); overviewComponentObserver.setOverviewActivityResumed(mActivityInterface.isResumed()); serviceProto.setOverviewComponentObvserver(overviewComponentObserver); } }