/** * 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.widget; import static android.app.Activity.RESULT_CANCELED; import static com.android.launcher3.util.Executors.MAIN_EXECUTOR; import android.appwidget.AppWidgetHost; import android.appwidget.AppWidgetHostView; import android.appwidget.AppWidgetManager; import android.appwidget.AppWidgetProviderInfo; import android.content.ActivityNotFoundException; import android.content.Context; import android.content.Intent; import android.os.Bundle; import android.util.SparseArray; import android.widget.RemoteViews; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import com.android.launcher3.BaseActivity; import com.android.launcher3.BaseDraggingActivity; import com.android.launcher3.LauncherAppState; import com.android.launcher3.R; import com.android.launcher3.Utilities; import com.android.launcher3.config.FeatureFlags; import com.android.launcher3.model.WidgetsModel; import com.android.launcher3.model.data.ItemInfo; import com.android.launcher3.testing.TestLogging; import com.android.launcher3.testing.shared.TestProtocol; import com.android.launcher3.util.ResourceBasedOverride; import com.android.launcher3.widget.custom.CustomWidgetManager; import java.util.function.IntConsumer; /** * A wrapper for LauncherAppWidgetHost. This class is created so the AppWidgetHost could run in * background. */ public class LauncherWidgetHolder { public static final int APPWIDGET_HOST_ID = 1024; protected static final int FLAG_LISTENING = 1; protected static final int FLAG_STATE_IS_NORMAL = 1 << 1; protected static final int FLAG_ACTIVITY_STARTED = 1 << 2; protected static final int FLAG_ACTIVITY_RESUMED = 1 << 3; private static final int FLAGS_SHOULD_LISTEN = FLAG_STATE_IS_NORMAL | FLAG_ACTIVITY_STARTED | FLAG_ACTIVITY_RESUMED; @NonNull private final Context mContext; @NonNull private final AppWidgetHost mWidgetHost; @NonNull private final SparseArray mViews = new SparseArray<>(); @NonNull private final SparseArray mPendingViews = new SparseArray<>(); @NonNull private final SparseArray mDeferredViews = new SparseArray<>(); protected int mFlags = FLAG_STATE_IS_NORMAL; // TODO(b/191735836): Replace with ActivityOptions.KEY_SPLASH_SCREEN_STYLE when un-hidden private static final String KEY_SPLASH_SCREEN_STYLE = "android.activity.splashScreenStyle"; // TODO(b/191735836): Replace with SplashScreen.SPLASH_SCREEN_STYLE_EMPTY when un-hidden private static final int SPLASH_SCREEN_STYLE_EMPTY = 0; protected LauncherWidgetHolder(@NonNull Context context, @Nullable IntConsumer appWidgetRemovedCallback) { mContext = context; mWidgetHost = createHost(context, appWidgetRemovedCallback); } protected AppWidgetHost createHost( Context context, @Nullable IntConsumer appWidgetRemovedCallback) { return new LauncherAppWidgetHost(context, appWidgetRemovedCallback, this); } /** * Starts listening to the widget updates from the server side */ public void startListening() { if (WidgetsModel.GO_DISABLE_WIDGETS) { return; } setListeningFlag(true); try { mWidgetHost.startListening(); } catch (Exception e) { if (!Utilities.isBinderSizeError(e)) { throw new RuntimeException(e); } // We're willing to let this slide. The exception is being caused by the list of // RemoteViews which is being passed back. The startListening relationship will // have been established by this point, and we will end up populating the // widgets upon bind anyway. See issue 14255011 for more context. } updateDeferredView(); } /** * Update any views which have been deferred because the host was not listening. */ protected void updateDeferredView() { // We go in reverse order and inflate any deferred or cached widget for (int i = mViews.size() - 1; i >= 0; i--) { LauncherAppWidgetHostView view = mViews.valueAt(i); if (view instanceof DeferredAppWidgetHostView) { view.reInflate(); } if (FeatureFlags.ENABLE_CACHED_WIDGET.get()) { final int appWidgetId = mViews.keyAt(i); if (view == mDeferredViews.get(appWidgetId)) { // If the widget view was deferred, we'll need to call super.createView here // to make the binder call to system process to fetch cumulative updates to this // widget, as well as setting up this view for future updates. mWidgetHost.createView(view.mLauncher, appWidgetId, view.getAppWidgetInfo()); // At this point #onCreateView should have been called, which in turn returned // the deferred view. There's no reason to keep the reference anymore, so we // removed it here. mDeferredViews.remove(appWidgetId); } } } } /** * Registers an "activity started/stopped" event. */ public void setActivityStarted(boolean isStarted) { setShouldListenFlag(FLAG_ACTIVITY_STARTED, isStarted); } /** * Registers an "activity paused/resumed" event. */ public void setActivityResumed(boolean isResumed) { setShouldListenFlag(FLAG_ACTIVITY_RESUMED, isResumed); } /** * Set the NORMAL state of the widget host * @param isNormal True if setting the host to be in normal state, false otherwise */ public void setStateIsNormal(boolean isNormal) { setShouldListenFlag(FLAG_STATE_IS_NORMAL, isNormal); } /** * Delete the specified app widget from the host * @param appWidgetId The ID of the app widget to be deleted */ public void deleteAppWidgetId(int appWidgetId) { mWidgetHost.deleteAppWidgetId(appWidgetId); mViews.remove(appWidgetId); if (FeatureFlags.ENABLE_CACHED_WIDGET.get()) { final LauncherAppState state = LauncherAppState.getInstance(mContext); synchronized (state.mCachedRemoteViews) { state.mCachedRemoteViews.delete(appWidgetId); } } } /** * Add the pending view to the host for complete configuration in further steps * @param appWidgetId The ID of the specified app widget * @param view The {@link PendingAppWidgetHostView} of the app widget */ public void addPendingView(int appWidgetId, @NonNull PendingAppWidgetHostView view) { mPendingViews.put(appWidgetId, view); } /** * @param appWidgetId The app widget id of the specified widget * @return The {@link PendingAppWidgetHostView} of the widget if it exists, null otherwise */ @Nullable protected PendingAppWidgetHostView getPendingView(int appWidgetId) { return mPendingViews.get(appWidgetId); } protected void removePendingView(int appWidgetId) { mPendingViews.remove(appWidgetId); } /** * Called when the launcher is destroyed */ public void destroy() { // No-op } /** * @return The allocated app widget id if allocation is successful, returns -1 otherwise */ public int allocateAppWidgetId() { if (WidgetsModel.GO_DISABLE_WIDGETS) { return AppWidgetManager.INVALID_APPWIDGET_ID; } return mWidgetHost.allocateAppWidgetId(); } /** * Add a listener that is triggered when the providers of the widgets are changed * @param listener The listener that notifies when the providers changed */ public void addProviderChangeListener(@NonNull ProviderChangedListener listener) { LauncherAppWidgetHost tempHost = (LauncherAppWidgetHost) mWidgetHost; tempHost.addProviderChangeListener(listener); } /** * Remove the specified listener from the host * @param listener The listener that is to be removed from the host */ public void removeProviderChangeListener(ProviderChangedListener listener) { LauncherAppWidgetHost tempHost = (LauncherAppWidgetHost) mWidgetHost; tempHost.removeProviderChangeListener(listener); } /** * Starts the configuration activity for the widget * @param activity The activity in which to start the configuration page * @param widgetId The ID of the widget * @param requestCode The request code */ public void startConfigActivity(@NonNull BaseDraggingActivity activity, int widgetId, int requestCode) { if (WidgetsModel.GO_DISABLE_WIDGETS) { sendActionCancelled(activity, requestCode); return; } try { TestLogging.recordEvent(TestProtocol.SEQUENCE_MAIN, "start: startConfigActivity"); mWidgetHost.startAppWidgetConfigureActivityForResult(activity, widgetId, 0, requestCode, getConfigurationActivityOptions(activity, widgetId)); } catch (ActivityNotFoundException | SecurityException e) { Toast.makeText(activity, R.string.activity_not_found, Toast.LENGTH_SHORT).show(); sendActionCancelled(activity, requestCode); } } private void sendActionCancelled(final BaseActivity activity, final int requestCode) { MAIN_EXECUTOR.execute( () -> activity.onActivityResult(requestCode, RESULT_CANCELED, null)); } /** * Returns an {@link android.app.ActivityOptions} bundle from the {code activity} for launching * the configuration of the {@code widgetId} app widget, or null of options cannot be produced. */ @Nullable protected Bundle getConfigurationActivityOptions(@NonNull BaseDraggingActivity activity, int widgetId) { LauncherAppWidgetHostView view = mViews.get(widgetId); if (view == null) { return activity.makeDefaultActivityOptions( -1 /* SPLASH_SCREEN_STYLE_UNDEFINED */).toBundle(); } Object tag = view.getTag(); if (!(tag instanceof ItemInfo)) { return activity.makeDefaultActivityOptions( -1 /* SPLASH_SCREEN_STYLE_UNDEFINED */).toBundle(); } Bundle bundle = activity.getActivityLaunchOptions(view, (ItemInfo) tag).toBundle(); bundle.putInt(KEY_SPLASH_SCREEN_STYLE, SPLASH_SCREEN_STYLE_EMPTY); return bundle; } /** * Starts the binding flow for the widget * @param activity The activity for which to bind the widget * @param appWidgetId The ID of the widget * @param info The {@link AppWidgetProviderInfo} of the widget * @param requestCode The request code */ public void startBindFlow(@NonNull BaseActivity activity, int appWidgetId, @NonNull AppWidgetProviderInfo info, int requestCode) { if (WidgetsModel.GO_DISABLE_WIDGETS) { sendActionCancelled(activity, requestCode); return; } Intent intent = new Intent(AppWidgetManager.ACTION_APPWIDGET_BIND) .putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetId) .putExtra(AppWidgetManager.EXTRA_APPWIDGET_PROVIDER, info.provider) .putExtra(AppWidgetManager.EXTRA_APPWIDGET_PROVIDER_PROFILE, info.getProfile()); // TODO: we need to make sure that this accounts for the options bundle. // intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_OPTIONS, options); activity.startActivityForResult(intent, requestCode); } /** * Stop the host from listening to the widget updates */ public void stopListening() { if (WidgetsModel.GO_DISABLE_WIDGETS) { return; } if (FeatureFlags.ENABLE_CACHED_WIDGET.get()) { // Cache the content from the widgets when Launcher stops listening to widget updates final LauncherAppState state = LauncherAppState.getInstance(mContext); synchronized (state.mCachedRemoteViews) { for (int i = 0; i < mViews.size(); i++) { final int appWidgetId = mViews.keyAt(i); final LauncherAppWidgetHostView view = mViews.get(appWidgetId); state.mCachedRemoteViews.put(appWidgetId, view.mLastRemoteViews); } } } mWidgetHost.stopListening(); setListeningFlag(false); } protected void setListeningFlag(final boolean isListening) { if (isListening) { mFlags |= FLAG_LISTENING; return; } mFlags &= ~FLAG_LISTENING; } /** * @return The app widget ids */ @NonNull public int[] getAppWidgetIds() { return mWidgetHost.getAppWidgetIds(); } /** * Create a view for the specified app widget * @param context The activity context for which the view is created * @param appWidgetId The ID of the widget * @param appWidget The {@link LauncherAppWidgetProviderInfo} of the widget * @return A view for the widget */ @NonNull public AppWidgetHostView createView(@NonNull Context context, int appWidgetId, @NonNull LauncherAppWidgetProviderInfo appWidget) { if (appWidget.isCustomWidget()) { LauncherAppWidgetHostView lahv = new LauncherAppWidgetHostView(context); lahv.setAppWidget(0, appWidget); CustomWidgetManager.INSTANCE.get(context).onViewCreated(lahv); return lahv; } else if ((mFlags & FLAG_LISTENING) == 0) { // Since the launcher hasn't started listening to widget updates, we can't simply call // super.createView here because the later will make a binder call to retrieve // RemoteViews from system process. // TODO: have launcher always listens to widget updates in background so that this // check can be removed altogether. if (FeatureFlags.ENABLE_CACHED_WIDGET.get()) { final RemoteViews cachedRemoteViews = getCachedRemoteViews(appWidgetId); if (cachedRemoteViews != null) { // We've found RemoteViews from cache for this widget, so we will instantiate a // widget host view and populate it with the cached RemoteViews. final LauncherAppWidgetHostView view = new LauncherAppWidgetHostView(context); view.setAppWidget(appWidgetId, appWidget); view.updateAppWidget(cachedRemoteViews); mDeferredViews.put(appWidgetId, view); mViews.put(appWidgetId, view); return view; } } // If cache misses or not enabled, a placeholder for the widget will be returned. DeferredAppWidgetHostView view = new DeferredAppWidgetHostView(context); view.setAppWidget(appWidgetId, appWidget); mViews.put(appWidgetId, view); return view; } else { try { return mWidgetHost.createView(context, appWidgetId, appWidget); } catch (Exception e) { if (!Utilities.isBinderSizeError(e)) { throw new RuntimeException(e); } // If the exception was thrown while fetching the remote views, let the view stay. // This will ensure that if the widget posts a valid update later, the view // will update. LauncherAppWidgetHostView view = mViews.get(appWidgetId); if (view == null) { view = onCreateView(mContext, appWidgetId, appWidget); } view.setAppWidget(appWidgetId, appWidget); view.switchToErrorView(); return view; } } } /** * Listener for getting notifications on provider changes. */ public interface ProviderChangedListener { /** * Notify the listener that the providers have changed */ void notifyWidgetProvidersChanged(); } /** * Called to return a proper view when creating a view * @param context The context for which the widget view is created * @param appWidgetId The ID of the added widget * @param appWidget The provider info of the added widget * @return A view for the specified app widget */ @NonNull public LauncherAppWidgetHostView onCreateView(Context context, int appWidgetId, AppWidgetProviderInfo appWidget) { final LauncherAppWidgetHostView view; if (getPendingView(appWidgetId) != null) { view = getPendingView(appWidgetId); removePendingView(appWidgetId); } else if (mDeferredViews.get(appWidgetId) != null) { // In case the widget view is deferred, we will simply return the deferred view as // opposed to instantiate a new instance of LauncherAppWidgetHostView since launcher // already added the former to the workspace. view = mDeferredViews.get(appWidgetId); } else { view = new LauncherAppWidgetHostView(context); } mViews.put(appWidgetId, view); return view; } /** * Clears all the views from the host */ public void clearViews() { LauncherAppWidgetHost tempHost = (LauncherAppWidgetHost) mWidgetHost; tempHost.clearViews(); if (FeatureFlags.ENABLE_CACHED_WIDGET.get()) { // Clear previously cached content from existing widgets mDeferredViews.clear(); } mViews.clear(); } /** * @return True if the host is listening to the updates, false otherwise */ public boolean isListening() { return (mFlags & FLAG_LISTENING) != 0; } /** * Sets or unsets a flag the can change whether the widget host should be in the listening * state. */ private void setShouldListenFlag(int flag, boolean on) { if (on) { mFlags |= flag; } else { mFlags &= ~flag; } final boolean listening = isListening(); if (!listening && shouldListen(mFlags)) { // Postpone starting listening until all flags are on. startListening(); } else if (listening && (mFlags & FLAG_ACTIVITY_STARTED) == 0) { // Postpone stopping listening until the activity is stopped. stopListening(); } } /** * Returns true if the holder should be listening for widget updates based * on the provided state flags. */ protected boolean shouldListen(int flags) { return (flags & FLAGS_SHOULD_LISTEN) == FLAGS_SHOULD_LISTEN; } @Nullable private RemoteViews getCachedRemoteViews(int appWidgetId) { final LauncherAppState state = LauncherAppState.getInstance(mContext); synchronized (state.mCachedRemoteViews) { return state.mCachedRemoteViews.get(appWidgetId); } } /** * Returns the new LauncherWidgetHolder instance */ public static LauncherWidgetHolder newInstance(Context context) { return HolderFactory.newFactory(context).newInstance(context, null); } /** * A factory class that generates new instances of {@code LauncherWidgetHolder} */ public static class HolderFactory implements ResourceBasedOverride { /** * @param context The context of the caller * @param appWidgetRemovedCallback The callback that is called when widgets are removed * @return A new instance of {@code LauncherWidgetHolder} */ public LauncherWidgetHolder newInstance(@NonNull Context context, @Nullable IntConsumer appWidgetRemovedCallback) { return new LauncherWidgetHolder(context, appWidgetRemovedCallback); } /** * @param context The context of the caller * @return A new instance of factory class for widget holders. If not specified, returning * {@code HolderFactory} by default. */ public static HolderFactory newFactory(Context context) { return Overrides.getObject( HolderFactory.class, context, R.string.widget_holder_factory_class); } } }