blob: ca1b2a91f75573cd6c75e630c4dac37b4eea5f0c [file] [log] [blame]
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Daniel Sandler325dc232013-06-05 22:57:57 -040017package com.android.launcher3;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080018
kholoud mohamedd5a4d602022-04-28 15:55:40 +010019import static android.app.admin.DevicePolicyManager.ACTION_DEVICE_POLICY_RESOURCE_UPDATED;
20
Sunny Goyala7a5bf32020-01-05 15:35:29 +053021import static com.android.launcher3.LauncherAppState.ACTION_FORCE_ROLOAD;
Oli Thompson3ea1acb2024-05-13 11:06:50 +000022import static com.android.launcher3.LauncherPrefs.WORK_EDU_STEP;
Zak Cohen3eeb41d2020-02-14 14:15:13 -080023import static com.android.launcher3.config.FeatureFlags.IS_STUDIO_BUILD;
Rohit Goyal86a1b1a2024-03-11 12:27:19 +000024import static com.android.launcher3.icons.cache.BaseIconCache.EMPTY_CLASS_NAME;
Sunny Goyal6449a212024-01-12 09:40:43 -080025import static com.android.launcher3.model.PackageUpdatedTask.OP_UPDATE;
Himanshu Gupta9aab4d42023-10-12 16:53:00 +010026import static com.android.launcher3.pm.UserCache.ACTION_PROFILE_AVAILABLE;
27import static com.android.launcher3.pm.UserCache.ACTION_PROFILE_UNAVAILABLE;
Vinit Nayak134e4292023-06-01 17:09:43 -070028import static com.android.launcher3.testing.shared.TestProtocol.sDebugTracing;
Sunny Goyala7a5bf32020-01-05 15:35:29 +053029import static com.android.launcher3.util.Executors.MAIN_EXECUTOR;
30import static com.android.launcher3.util.Executors.MODEL_EXECUTOR;
Sunny Goyala7a5bf32020-01-05 15:35:29 +053031
Rohit Goyal86a1b1a2024-03-11 12:27:19 +000032import android.content.ComponentName;
Sunny Goyal14168432019-10-24 15:59:49 -070033import android.content.Context;
Sunny Goyalf599ccf2014-07-08 13:01:29 -070034import android.content.Intent;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070035import android.content.pm.PackageInstaller;
Sunny Goyal6bbf6002019-04-17 18:38:52 -070036import android.content.pm.ShortcutInfo;
Sunny Goyal7c74e4a2016-12-15 15:53:17 -080037import android.os.UserHandle;
Winson Chunga90303b2013-11-15 13:05:06 -080038import android.text.TextUtils;
Winson Chungaafa03c2010-06-11 17:34:16 -070039import android.util.Log;
Sunny Goyala474a9b2017-05-04 16:47:11 -070040import android.util.Pair;
Michael Jurka34c2e6c2013-12-13 16:07:45 +010041
Pinyao Ting777c13e2022-09-16 09:44:26 -070042import androidx.annotation.NonNull;
Sunny Goyal6fe3eec2019-08-15 14:53:41 -070043import androidx.annotation.Nullable;
Sunny Goyal14168432019-10-24 15:59:49 -070044import androidx.annotation.WorkerThread;
Sunny Goyal6fe3eec2019-08-15 14:53:41 -070045
Sunny Goyal669b71f2023-01-27 14:37:07 -080046import com.android.launcher3.celllayout.CellPosMapper;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070047import com.android.launcher3.config.FeatureFlags;
Sunny Goyalf840f102018-09-21 14:41:05 -070048import com.android.launcher3.icons.IconCache;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070049import com.android.launcher3.model.AddWorkspaceItemsTask;
Sunny Goyal87dcde62019-07-17 20:35:56 -070050import com.android.launcher3.model.AllAppsList;
Sunny Goyal77954ba2024-03-25 11:53:17 -070051import com.android.launcher3.model.BaseLauncherBinder;
Sunny Goyale9956a72016-09-01 17:24:47 -070052import com.android.launcher3.model.BgDataModel;
Sunny Goyal87dcde62019-07-17 20:35:56 -070053import com.android.launcher3.model.BgDataModel.Callbacks;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070054import com.android.launcher3.model.CacheDataUpdatedTask;
Sunny Goyal60e68c92020-08-12 13:59:27 -070055import com.android.launcher3.model.ItemInstallQueue;
Sunny Goyalb434fde2017-06-06 10:46:59 -070056import com.android.launcher3.model.LoaderTask;
Sunny Goyalce953a32023-04-14 14:08:37 -070057import com.android.launcher3.model.ModelDbController;
Sunny Goyal8b74cc72020-07-27 17:50:33 -070058import com.android.launcher3.model.ModelDelegate;
Sunny Goyal6449a212024-01-12 09:40:43 -080059import com.android.launcher3.model.ModelLauncherCallbacks;
Sunny Goyal99389382024-05-01 14:47:43 -070060import com.android.launcher3.model.ModelTaskController;
Sunny Goyal43bf11d2017-02-02 13:52:53 -080061import com.android.launcher3.model.ModelWriter;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070062import com.android.launcher3.model.PackageInstallStateChangedTask;
63import com.android.launcher3.model.PackageUpdatedTask;
kholoud mohamedd5a4d602022-04-28 15:55:40 +010064import com.android.launcher3.model.ReloadStringCacheTask;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070065import com.android.launcher3.model.ShortcutsChangedTask;
66import com.android.launcher3.model.UserLockStateChangedTask;
Sunny Goyale396abf2020-04-06 15:11:17 -070067import com.android.launcher3.model.data.AppInfo;
68import com.android.launcher3.model.data.ItemInfo;
69import com.android.launcher3.model.data.WorkspaceItemInfo;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070070import com.android.launcher3.pm.InstallSessionTracker;
71import com.android.launcher3.pm.PackageInstallInfo;
Sunny Goyal337c81f2019-12-10 12:19:13 -080072import com.android.launcher3.pm.UserCache;
Sunny Goyalfa395362019-12-11 10:00:47 -080073import com.android.launcher3.shortcuts.ShortcutRequest;
Sunny Goyalea600c72020-07-09 19:31:40 -070074import com.android.launcher3.util.IntSet;
Jon Miranda088b9c22019-09-17 11:51:49 -070075import com.android.launcher3.util.ItemInfoMatcher;
Rohit Goyal86a1b1a2024-03-11 12:27:19 +000076import com.android.launcher3.util.PackageManagerHelper;
Tony Wickham86222d22017-03-29 15:30:43 -070077import com.android.launcher3.util.PackageUserKey;
Sunny Goyalaaf7d1d2016-05-17 13:38:54 -070078import com.android.launcher3.util.Preconditions;
Romain Guyedcce092010-03-04 13:03:17 -080079
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -080080import java.io.FileDescriptor;
81import java.io.PrintWriter;
Michael Jurkac2f801e2011-07-12 14:19:46 -070082import java.util.ArrayList;
Winson Chungb8b2a5a2012-07-12 17:55:31 -070083import java.util.HashSet;
Michael Jurkac2f801e2011-07-12 14:19:46 -070084import java.util.List;
Sunny Goyaldd96a5e2017-02-17 11:22:34 -080085import java.util.concurrent.CancellationException;
Sunny Goyal57e08662021-08-06 11:52:10 -070086import java.util.function.Consumer;
Sunny Goyal8c48d8b2019-01-25 15:10:18 -080087import java.util.function.Supplier;
Michael Jurkac2f801e2011-07-12 14:19:46 -070088
The Android Open Source Project31dd5032009-03-03 19:32:27 -080089/**
90 * Maintains in-memory state of the Launcher. It is expected that there should be only one
91 * LauncherModel object held in a static. Also provide APIs for updating the database state
The Android Open Source Projectbc219c32009-03-09 11:52:14 -070092 * for the Launcher.
The Android Open Source Project31dd5032009-03-03 19:32:27 -080093 */
Sunny Goyal6449a212024-01-12 09:40:43 -080094public class LauncherModel implements InstallSessionTracker.Callback {
Chris Wrenee523362014-09-09 10:09:02 -040095 private static final boolean DEBUG_RECEIVER = false;
Chris Wrenb358f812014-04-16 13:37:00 -040096
Joe Onorato9c1289c2009-08-17 11:03:03 -040097 static final String TAG = "Launcher.Model";
The Android Open Source Projectf96811c2009-03-18 17:39:48 -070098
Pinyao Ting777c13e2022-09-16 09:44:26 -070099 @NonNull
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530100 private final LauncherAppState mApp;
Pinyao Ting777c13e2022-09-16 09:44:26 -0700101 @NonNull
Winson Chung94e8ad02024-05-08 21:14:57 +0000102 private final PackageManagerHelper mPmHelper;
103 @NonNull
Sunny Goyalce953a32023-04-14 14:08:37 -0700104 private final ModelDbController mModelDbController;
105 @NonNull
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530106 private final Object mLock = new Object();
Pinyao Ting777c13e2022-09-16 09:44:26 -0700107 @Nullable
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530108 private LoaderTask mLoaderTask;
109 private boolean mIsLoaderTaskRunning;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800110
Charlie Anderson673a81b2023-07-21 15:05:33 -0400111 // only allow this once per reboot to reload work apps
112 private boolean mShouldReloadWorkProfile = true;
113
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800114 // Indicates whether the current model data is valid or not.
115 // We start off with everything not loaded. After that, we assume that
Joe Onoratocc67f472010-06-08 10:54:30 -0700116 // our monitoring of the package manager provides all updates and we never
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800117 // need to do a requery. This is only ever touched from the loader thread.
118 private boolean mModelLoaded;
Sunny Goyal777d4902021-08-27 21:22:17 +0000119 private boolean mModelDestroyed = false;
Hyunyoung Song6aa37292017-02-06 10:46:24 -0800120 public boolean isModelLoaded() {
121 synchronized (mLock) {
Sunny Goyal777d4902021-08-27 21:22:17 +0000122 return mModelLoaded && mLoaderTask == null && !mModelDestroyed;
Hyunyoung Song6aa37292017-02-06 10:46:24 -0800123 }
124 }
Joe Onoratocc67f472010-06-08 10:54:30 -0700125
Pinyao Ting777c13e2022-09-16 09:44:26 -0700126 @NonNull
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530127 private final ArrayList<Callbacks> mCallbacksList = new ArrayList<>(1);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800128
Michael Jurkaa8c760d2011-04-28 14:59:33 -0700129 // < only access in worker thread >
Pinyao Ting777c13e2022-09-16 09:44:26 -0700130 @NonNull
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800131 private final AllAppsList mBgAllAppsList;
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700132
Sunny Goyale9956a72016-09-01 17:24:47 -0700133 /**
134 * All the static data should be accessed on the background thread, A lock should be acquired
135 * on this object when accessing any data from this model.
136 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700137 @NonNull
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530138 private final BgDataModel mBgDataModel = new BgDataModel();
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700139
Pinyao Ting777c13e2022-09-16 09:44:26 -0700140 @NonNull
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700141 private final ModelDelegate mModelDelegate;
142
Sunny Goyal7b9e28f2023-05-17 12:44:03 -0700143 private int mLastLoadId = -1;
144
Sunny Goyalc6e97692017-06-02 13:46:55 -0700145 // Runnable to check if the shortcuts permission has changed.
Pinyao Ting777c13e2022-09-16 09:44:26 -0700146 @NonNull
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700147 private final Runnable mDataValidationCheck = new Runnable() {
Sunny Goyalc6e97692017-06-02 13:46:55 -0700148 @Override
149 public void run() {
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700150 if (mModelLoaded) {
151 mModelDelegate.validateData();
Sunny Goyalc6e97692017-06-02 13:46:55 -0700152 }
153 }
154 };
Kenny Guyed131872014-04-30 03:02:21 +0100155
Pinyao Ting777c13e2022-09-16 09:44:26 -0700156 LauncherModel(@NonNull final Context context, @NonNull final LauncherAppState app,
157 @NonNull final IconCache iconCache, @NonNull final AppFilter appFilter,
Winson Chung94e8ad02024-05-08 21:14:57 +0000158 @NonNull final PackageManagerHelper pmHelper, final boolean isPrimaryInstance) {
Daniel Sandlere4f98912013-06-25 15:13:26 -0400159 mApp = app;
Winson Chung94e8ad02024-05-08 21:14:57 +0000160 mPmHelper = pmHelper;
Sunny Goyalce953a32023-04-14 14:08:37 -0700161 mModelDbController = new ModelDbController(context);
Bjorn Bringert1307f632013-10-03 22:31:03 +0100162 mBgAllAppsList = new AllAppsList(iconCache, appFilter);
Winson Chung94e8ad02024-05-08 21:14:57 +0000163 mModelDelegate = ModelDelegate.newInstance(context, app, mPmHelper, mBgAllAppsList,
164 mBgDataModel, isPrimaryInstance);
Joe Onorato0589f0f2010-02-08 13:44:00 -0800165 }
166
Pinyao Ting777c13e2022-09-16 09:44:26 -0700167 @NonNull
Schneider Victor-tulias3ec8f552021-04-13 14:17:41 -0700168 public ModelDelegate getModelDelegate() {
169 return mModelDelegate;
170 }
171
Sunny Goyalce953a32023-04-14 14:08:37 -0700172 public ModelDbController getModelDbController() {
173 return mModelDbController;
174 }
175
Sunny Goyal6449a212024-01-12 09:40:43 -0800176 public ModelLauncherCallbacks newModelCallbacks() {
177 return new ModelLauncherCallbacks(this::enqueueModelUpdateTask);
178 }
179
Sunny Goyal71b5c0b2015-01-08 16:59:04 -0800180 /**
181 * Adds the provided items to the workspace.
Sunny Goyal71b5c0b2015-01-08 16:59:04 -0800182 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700183 public void addAndBindAddedWorkspaceItems(
184 @NonNull final List<Pair<ItemInfo, Object>> itemList) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530185 for (Callbacks cb : getCallbacks()) {
186 cb.preAddApps();
Tony Wickham6a71a5b2018-08-21 11:40:23 -0700187 }
Sunny Goyal91498ab2017-10-05 15:57:40 -0700188 enqueueModelUpdateTask(new AddWorkspaceItemsTask(itemList));
Winson Chung64359a52013-07-08 17:17:08 -0700189 }
190
Pinyao Ting777c13e2022-09-16 09:44:26 -0700191 @NonNull
Sebastian Franco2d023e22023-11-15 16:48:34 -0600192 public ModelWriter getWriter(final boolean verifyChanges, CellPosMapper cellPosMapper,
193 @Nullable final Callbacks owner) {
194 return new ModelWriter(mApp.getContext(), this, mBgDataModel, verifyChanges, cellPosMapper,
195 owner);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800196 }
197
Sunny Goyal14168432019-10-24 15:59:49 -0700198 /**
199 * Called when the icon for an app changes, outside of package event
200 */
201 @WorkerThread
Pinyao Ting777c13e2022-09-16 09:44:26 -0700202 public void onAppIconChanged(@NonNull final String packageName,
203 @NonNull final UserHandle user) {
Sunny Goyal14168432019-10-24 15:59:49 -0700204 // Update the icon for the calendar package
205 Context context = mApp.getContext();
Sunny Goyal6449a212024-01-12 09:40:43 -0800206 enqueueModelUpdateTask(new PackageUpdatedTask(OP_UPDATE, user, packageName));
Sunny Goyal14168432019-10-24 15:59:49 -0700207
Sunny Goyalfa395362019-12-11 10:00:47 -0800208 List<ShortcutInfo> pinnedShortcuts = new ShortcutRequest(context, user)
209 .forPackage(packageName).query(ShortcutRequest.PINNED);
Sunny Goyal14168432019-10-24 15:59:49 -0700210 if (!pinnedShortcuts.isEmpty()) {
211 enqueueModelUpdateTask(new ShortcutsChangedTask(packageName, pinnedShortcuts, user,
212 false));
213 }
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700214 }
215
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700216 /**
Sunny Goyal0fc3d122020-08-11 18:49:28 -0700217 * Called when the workspace items have drastically changed
218 */
219 public void onWorkspaceUiChanged() {
220 MODEL_EXECUTOR.execute(mModelDelegate::workspaceLoadComplete);
221 }
222
223 /**
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700224 * Called when the model is destroyed
225 */
226 public void destroy() {
Sunny Goyal777d4902021-08-27 21:22:17 +0000227 mModelDestroyed = true;
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700228 MODEL_EXECUTOR.execute(mModelDelegate::destroy);
229 }
230
Pinyao Ting777c13e2022-09-16 09:44:26 -0700231 public void onBroadcastIntent(@NonNull final Intent intent) {
Vinit Nayak134e4292023-06-01 17:09:43 -0700232 if (DEBUG_RECEIVER || sDebugTracing) Log.d(TAG, "onReceive intent=" + intent);
Joe Onorato36115782010-06-17 13:28:48 -0400233 final String action = intent.getAction();
Kenny Guyed131872014-04-30 03:02:21 +0100234 if (Intent.ACTION_LOCALE_CHANGED.equals(action)) {
Reena Lee93f824a2011-09-23 17:20:28 -0700235 // If we have changed locale we need to clear out the labels in all apps/workspace.
236 forceReload();
kholoud mohamedd5a4d602022-04-28 15:55:40 +0100237 } else if (ACTION_DEVICE_POLICY_RESOURCE_UPDATED.equals(action)) {
238 enqueueModelUpdateTask(new ReloadStringCacheTask(mModelDelegate));
Zak Cohen3eeb41d2020-02-14 14:15:13 -0800239 } else if (IS_STUDIO_BUILD && ACTION_FORCE_ROLOAD.equals(action)) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530240 for (Callbacks cb : getCallbacks()) {
241 if (cb instanceof Launcher) {
242 ((Launcher) cb).recreate();
243 }
244 }
Joe Onoratoe9ad59e2010-10-29 17:35:36 -0700245 }
246 }
247
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800248 /**
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700249 * Called then there use a user event
250 * @see UserCache#addUserEventListener
251 */
252 public void onUserEvent(UserHandle user, String action) {
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700253 if (Intent.ACTION_MANAGED_PROFILE_AVAILABLE.equals(action)
Charlie Anderson673a81b2023-07-21 15:05:33 -0400254 && mShouldReloadWorkProfile) {
255 mShouldReloadWorkProfile = false;
256 forceReload();
257 } else if (Intent.ACTION_MANAGED_PROFILE_AVAILABLE.equals(action)
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700258 || Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE.equals(action)) {
Charlie Anderson673a81b2023-07-21 15:05:33 -0400259 mShouldReloadWorkProfile = false;
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700260 enqueueModelUpdateTask(new PackageUpdatedTask(
261 PackageUpdatedTask.OP_USER_AVAILABILITY_CHANGE, user));
Charlie Anderson673a81b2023-07-21 15:05:33 -0400262 } else if (UserCache.ACTION_PROFILE_LOCKED.equals(action)
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700263 || UserCache.ACTION_PROFILE_UNLOCKED.equals(action)) {
264 enqueueModelUpdateTask(new UserLockStateChangedTask(
265 user, UserCache.ACTION_PROFILE_UNLOCKED.equals(action)));
Charlie Anderson673a81b2023-07-21 15:05:33 -0400266 } else if (UserCache.ACTION_PROFILE_ADDED.equals(action)
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700267 || UserCache.ACTION_PROFILE_REMOVED.equals(action)) {
268 forceReload();
Himanshu Gupta9aab4d42023-10-12 16:53:00 +0100269 } else if (ACTION_PROFILE_AVAILABLE.equals(action)
270 || ACTION_PROFILE_UNAVAILABLE.equals(action)) {
271 /*
272 * This broadcast is only available when android.os.Flags.allowPrivateProfile() is set.
273 * For Work-profile this broadcast will be sent in addition to
274 * ACTION_MANAGED_PROFILE_AVAILABLE/UNAVAILABLE.
275 * So effectively, this if block only handles the non-work profile case.
276 */
277 enqueueModelUpdateTask(new PackageUpdatedTask(
278 PackageUpdatedTask.OP_USER_AVAILABILITY_CHANGE, user));
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700279 }
Oli Thompson3ea1acb2024-05-13 11:06:50 +0000280 if (Intent.ACTION_MANAGED_PROFILE_REMOVED.equals(action)) {
281 LauncherPrefs.get(mApp.getContext()).put(WORK_EDU_STEP, 0);
282 }
Sunny Goyalc80e60d2023-06-07 12:52:30 -0700283 }
284
285 /**
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800286 * Reloads the workspace items from the DB and re-binds the workspace. This should generally
287 * not be called as DB updates are automatically followed by UI update
288 */
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530289 public void forceReload() {
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800290 synchronized (mLock) {
291 // Stop any existing loaders first, so they don't set mModelLoaded to true later
Sunny Goyale86f11f2017-06-06 14:33:18 -0700292 stopLoader();
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800293 mModelLoaded = false;
294 }
Winson Chungf0c6ae02012-03-21 16:10:31 -0700295
Sunny Goyal29947f02017-12-18 13:49:44 -0800296 // Start the loader if launcher is already running, otherwise the loader will run,
297 // the next time launcher starts
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530298 if (hasCallbacks()) {
299 startLoader();
Joe Onoratoe9ad59e2010-10-29 17:35:36 -0700300 }
Joe Onorato36115782010-06-17 13:28:48 -0400301 }
Joe Onoratof99f8c12009-10-31 17:27:36 -0400302
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530303 /**
304 * Rebinds all existing callbacks with already loaded model
305 */
306 public void rebindCallbacks() {
307 if (hasCallbacks()) {
308 startLoader();
309 }
310 }
311
312 /**
313 * Removes an existing callback
314 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700315 public void removeCallbacks(@NonNull final Callbacks callbacks) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530316 synchronized (mCallbacksList) {
317 Preconditions.assertUIThread();
318 if (mCallbacksList.remove(callbacks)) {
319 if (stopLoader()) {
320 // Rebind existing callbacks
321 startLoader();
322 }
323 }
324 }
325 }
326
327 /**
328 * Adds a callbacks to receive model updates
329 * @return true if workspace load was performed synchronously
330 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700331 public boolean addCallbacksAndLoad(@NonNull final Callbacks callbacks) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530332 synchronized (mLock) {
333 addCallbacks(callbacks);
Sunny Goyal711c5962021-06-23 12:36:18 -0700334 return startLoader(new Callbacks[] { callbacks });
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530335
336 }
337 }
338
339 /**
340 * Adds a callbacks to receive model updates
341 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700342 public void addCallbacks(@NonNull final Callbacks callbacks) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530343 Preconditions.assertUIThread();
344 synchronized (mCallbacksList) {
345 mCallbacksList.add(callbacks);
346 }
Adam Cohen1a85c582014-09-30 09:48:49 -0700347 }
348
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700349 /**
350 * Starts the loader. Tries to bind {@params synchronousBindPage} synchronously if possible.
351 * @return true if the page could be bound synchronously.
352 */
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530353 public boolean startLoader() {
Sunny Goyal711c5962021-06-23 12:36:18 -0700354 return startLoader(new Callbacks[0]);
355 }
356
Pinyao Ting777c13e2022-09-16 09:44:26 -0700357 private boolean startLoader(@NonNull final Callbacks[] newCallbacks) {
Sunny Goyal756adbc2015-04-16 15:20:51 -0700358 // Enable queue before starting loader. It will get disabled in Launcher#finishBindingItems
Sunny Goyal60e68c92020-08-12 13:59:27 -0700359 ItemInstallQueue.INSTANCE.get(mApp.getContext())
360 .pauseModelPush(ItemInstallQueue.FLAG_LOADER_RUNNING);
Joe Onorato36115782010-06-17 13:28:48 -0400361 synchronized (mLock) {
Sunny Goyal711c5962021-06-23 12:36:18 -0700362 // If there is already one running, tell it to stop.
363 boolean wasRunning = stopLoader();
364 boolean bindDirectly = mModelLoaded && !mIsLoaderTaskRunning;
365 boolean bindAllCallbacks = wasRunning || !bindDirectly || newCallbacks.length == 0;
366 final Callbacks[] callbacksList = bindAllCallbacks ? getCallbacks() : newCallbacks;
367
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530368 if (callbacksList.length > 0) {
Sunny Goyal527c7d32015-08-28 15:19:36 -0700369 // Clear any pending bind-runnables from the synchronized load process.
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530370 for (Callbacks cb : callbacksList) {
Sunny Goyal5fb83a42020-08-10 10:50:36 -0700371 MAIN_EXECUTOR.execute(cb::clearPendingBinds);
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530372 }
Sunny Goyal527c7d32015-08-28 15:19:36 -0700373
Sunny Goyal77954ba2024-03-25 11:53:17 -0700374 BaseLauncherBinder launcherBinder = new BaseLauncherBinder(
Sunny Goyal5fb83a42020-08-10 10:50:36 -0700375 mApp, mBgDataModel, mBgAllAppsList, callbacksList);
Sunny Goyal711c5962021-06-23 12:36:18 -0700376 if (bindDirectly) {
Sunny Goyalf8d6ed22017-06-01 14:26:38 -0700377 // Divide the set of loaded items into those that we are binding synchronously,
378 // and everything else that is to be bound normally (asynchronously).
Fengjiang Li5a36c172023-04-13 18:18:27 +0000379 launcherBinder.bindWorkspace(bindAllCallbacks, /* isBindSync= */ true);
Sunny Goyalf8d6ed22017-06-01 14:26:38 -0700380 // For now, continue posting the binding of AllApps as there are other
381 // issues that arise from that.
Stefan Andoniane82476a2023-01-10 22:46:31 +0000382 launcherBinder.bindAllApps();
383 launcherBinder.bindDeepShortcuts();
384 launcherBinder.bindWidgets();
Stefan Andoniane4609a22023-04-07 19:28:05 +0000385 if (FeatureFlags.CHANGE_MODEL_DELEGATE_LOADING_ORDER.get()) {
386 mModelDelegate.bindAllModelExtras(callbacksList);
387 }
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700388 return true;
Winson Chungb8b2a5a2012-07-12 17:55:31 -0700389 } else {
Sunny Goyal57e08662021-08-06 11:52:10 -0700390 stopLoader();
391 mLoaderTask = new LoaderTask(
Stefan Andoniane82476a2023-01-10 22:46:31 +0000392 mApp, mBgAllAppsList, mBgDataModel, mModelDelegate, launcherBinder);
Sunny Goyal57e08662021-08-06 11:52:10 -0700393
394 // Always post the loader task, instead of running directly
395 // (even on same thread) so that we exit any nested synchronized blocks
396 MODEL_EXECUTOR.post(mLoaderTask);
Winson Chungb8b2a5a2012-07-12 17:55:31 -0700397 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400398 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400399 }
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700400 return false;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400401 }
402
Sunny Goyale86f11f2017-06-06 14:33:18 -0700403 /**
404 * If there is already a loader task running, tell it to stop.
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530405 * @return true if an existing loader was stopped.
Sunny Goyale86f11f2017-06-06 14:33:18 -0700406 */
Sunny Goyal711c5962021-06-23 12:36:18 -0700407 private boolean stopLoader() {
Joe Onorato36115782010-06-17 13:28:48 -0400408 synchronized (mLock) {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700409 LoaderTask oldTask = mLoaderTask;
410 mLoaderTask = null;
411 if (oldTask != null) {
412 oldTask.stopLocked();
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530413 return true;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400414 }
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530415 return false;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400416 }
Joe Onorato36115782010-06-17 13:28:48 -0400417 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400418
Sunny Goyal57e08662021-08-06 11:52:10 -0700419 /**
420 * Loads the model if not loaded
421 * @param callback called with the data model upon successful load or null on model thread.
422 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700423 public void loadAsync(@NonNull final Consumer<BgDataModel> callback) {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700424 synchronized (mLock) {
Sunny Goyal57e08662021-08-06 11:52:10 -0700425 if (!mModelLoaded && !mIsLoaderTaskRunning) {
426 startLoader();
Winson Chungd6519662018-02-16 03:23:51 +0000427 }
428 }
Sunny Goyal57e08662021-08-06 11:52:10 -0700429 MODEL_EXECUTOR.post(() -> callback.accept(isModelLoaded() ? mBgDataModel : null));
Winson Chungd6519662018-02-16 03:23:51 +0000430 }
431
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700432 @Override
Pinyao Ting777c13e2022-09-16 09:44:26 -0700433 public void onInstallSessionCreated(@NonNull final PackageInstallInfo sessionInfo) {
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700434 if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {
Sunny Goyal99389382024-05-01 14:47:43 -0700435 enqueueModelUpdateTask((taskController, dataModel, apps) -> {
436 apps.addPromiseApp(mApp.getContext(), sessionInfo);
437 taskController.bindApplicationsIfNeeded();
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700438 });
439 }
440 }
441
442 @Override
Pinyao Ting777c13e2022-09-16 09:44:26 -0700443 public void onSessionFailure(@NonNull final String packageName,
444 @NonNull final UserHandle user) {
Sunny Goyal99389382024-05-01 14:47:43 -0700445 enqueueModelUpdateTask((taskController, dataModel, apps) -> {
446 IconCache iconCache = mApp.getIconCache();
447 final IntSet removedIds = new IntSet();
448 HashSet<WorkspaceItemInfo> archivedWorkspaceItemsToCacheRefresh = new HashSet<>();
449 boolean isAppArchived = PackageManagerHelper.INSTANCE.get(mApp.getContext())
450 .isAppArchivedForUser(packageName, user);
451 synchronized (dataModel) {
452 if (isAppArchived) {
453 // Remove package icon cache entry for archived app in case of a session
454 // failure.
455 mApp.getIconCache().remove(
456 new ComponentName(packageName, packageName + EMPTY_CLASS_NAME),
457 user);
458 }
Rohit Goyal86a1b1a2024-03-11 12:27:19 +0000459
Sunny Goyal99389382024-05-01 14:47:43 -0700460 for (ItemInfo info : dataModel.itemsIdMap) {
461 if (info instanceof WorkspaceItemInfo
462 && ((WorkspaceItemInfo) info).hasPromiseIconUi()
463 && user.equals(info.user)
464 && info.getIntent() != null) {
465 if (TextUtils.equals(packageName, info.getIntent().getPackage())) {
466 removedIds.add(info.id);
467 }
468 if (((WorkspaceItemInfo) info).isArchived()) {
469 WorkspaceItemInfo workspaceItem = (WorkspaceItemInfo) info;
470 // Refresh icons on the workspace for archived apps.
471 iconCache.getTitleAndIcon(workspaceItem,
472 workspaceItem.usingLowResIcon());
473 archivedWorkspaceItemsToCacheRefresh.add(workspaceItem);
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700474 }
475 }
476 }
477
Rohit Goyal86a1b1a2024-03-11 12:27:19 +0000478 if (isAppArchived) {
Sunny Goyal99389382024-05-01 14:47:43 -0700479 apps.updateIconsAndLabels(new HashSet<>(List.of(packageName)), user);
Jakob Schneider4f5b6012024-02-07 17:29:25 +0000480 }
Mario Bertschler817afa32017-03-15 11:56:47 -0700481 }
Sunny Goyal99389382024-05-01 14:47:43 -0700482
Sebastian Franco6d200f62024-06-12 19:39:49 -0600483 if (!removedIds.isEmpty() && !isAppArchived) {
Sunny Goyal99389382024-05-01 14:47:43 -0700484 taskController.deleteAndBindComponentsRemoved(
485 ItemInfoMatcher.ofItemIds(removedIds),
486 "removed because install session failed");
487 }
488 if (!archivedWorkspaceItemsToCacheRefresh.isEmpty()) {
489 taskController.bindUpdatedWorkspaceItems(
490 archivedWorkspaceItemsToCacheRefresh.stream().toList());
491 }
492 if (isAppArchived) {
493 taskController.bindApplicationsIfNeeded();
494 }
Mario Bertschler817afa32017-03-15 11:56:47 -0700495 });
496 }
497
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700498 @Override
Pinyao Ting777c13e2022-09-16 09:44:26 -0700499 public void onPackageStateChanged(@NonNull final PackageInstallInfo installInfo) {
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700500 enqueueModelUpdateTask(new PackageInstallStateChangedTask(installInfo));
501 }
502
503 /**
504 * Updates the icons and label of all pending icons for the provided package name.
505 */
506 @Override
Pinyao Ting777c13e2022-09-16 09:44:26 -0700507 public void onUpdateSessionDisplay(@NonNull final PackageUserKey key,
508 @NonNull final PackageInstaller.SessionInfo info) {
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700509 mApp.getIconCache().updateSessionCache(key, info);
510
511 HashSet<String> packages = new HashSet<>();
512 packages.add(key.mPackageName);
513 enqueueModelUpdateTask(new CacheDataUpdatedTask(
514 CacheDataUpdatedTask.OP_SESSION_UPDATE, key.mUser, packages));
515 }
516
Sunny Goyalc6e97692017-06-02 13:46:55 -0700517 public class LoaderTransaction implements AutoCloseable {
518
Pinyao Ting777c13e2022-09-16 09:44:26 -0700519 @NonNull
Sunny Goyalc6e97692017-06-02 13:46:55 -0700520 private final LoaderTask mTask;
521
Pinyao Ting777c13e2022-09-16 09:44:26 -0700522 private LoaderTransaction(@NonNull final LoaderTask task) throws CancellationException {
Sunny Goyalc6e97692017-06-02 13:46:55 -0700523 synchronized (mLock) {
524 if (mLoaderTask != task) {
525 throw new CancellationException("Loader already stopped");
526 }
Sunny Goyal7b9e28f2023-05-17 12:44:03 -0700527 mLastLoadId++;
Sunny Goyalc6e97692017-06-02 13:46:55 -0700528 mTask = task;
529 mIsLoaderTaskRunning = true;
530 mModelLoaded = false;
531 }
532 }
533
534 public void commit() {
535 synchronized (mLock) {
536 // Everything loaded bind the data.
537 mModelLoaded = true;
538 }
539 }
540
541 @Override
542 public void close() {
543 synchronized (mLock) {
544 // If we are still the last one to be scheduled, remove ourselves.
545 if (mLoaderTask == mTask) {
546 mLoaderTask = null;
547 }
548 mIsLoaderTaskRunning = false;
549 }
550 }
551 }
552
Pinyao Ting777c13e2022-09-16 09:44:26 -0700553 public LoaderTransaction beginLoader(@NonNull final LoaderTask task)
554 throws CancellationException {
Sunny Goyalc6e97692017-06-02 13:46:55 -0700555 return new LoaderTransaction(task);
556 }
557
Joe Onorato36115782010-06-17 13:28:48 -0400558 /**
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700559 * Refreshes the cached shortcuts if the shortcut permission has changed.
560 * Current implementation simply reloads the workspace, but it can be optimized to
Sunny Goyal337c81f2019-12-10 12:19:13 -0800561 * use partial updates similar to {@link UserCache}
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700562 */
Sunny Goyal8b74cc72020-07-27 17:50:33 -0700563 public void validateModelDataOnResume() {
564 MODEL_EXECUTOR.getHandler().removeCallbacks(mDataValidationCheck);
565 MODEL_EXECUTOR.post(mDataValidationCheck);
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700566 }
567
568 /**
Sunny Goyal75b0f552015-05-20 21:57:06 -0700569 * Called when the icons for packages have been updated in the icon cache.
570 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700571 public void onPackageIconsUpdated(@NonNull final HashSet<String> updatedPackages,
572 @NonNull final UserHandle user) {
Sunny Goyal75b0f552015-05-20 21:57:06 -0700573 // If any package icon has changed (app was updated while launcher was dead),
574 // update the corresponding shortcuts.
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700575 enqueueModelUpdateTask(new CacheDataUpdatedTask(
576 CacheDataUpdatedTask.OP_CACHE_UPDATE, user, updatedPackages));
Sunny Goyal75b0f552015-05-20 21:57:06 -0700577 }
578
Sunny Goyalac8154a2018-09-26 12:00:30 -0700579 /**
580 * Called when the labels for the widgets has updated in the icon cache.
581 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700582 public void onWidgetLabelsUpdated(@NonNull final HashSet<String> updatedPackages,
583 @NonNull final UserHandle user) {
Sunny Goyal99389382024-05-01 14:47:43 -0700584 enqueueModelUpdateTask((taskController, dataModel, apps) -> {
585 dataModel.widgetsModel.onPackageIconsUpdated(updatedPackages, user, mApp);
586 taskController.bindUpdatedWidgets(dataModel);
Sunny Goyalac8154a2018-09-26 12:00:30 -0700587 });
588 }
589
Pinyao Ting777c13e2022-09-16 09:44:26 -0700590 public void enqueueModelUpdateTask(@NonNull final ModelUpdateTask task) {
Sunny Goyal777d4902021-08-27 21:22:17 +0000591 if (mModelDestroyed) {
592 return;
593 }
Sunny Goyal99389382024-05-01 14:47:43 -0700594 MODEL_EXECUTOR.execute(() -> {
595 if (!isModelLoaded()) {
596 // Loader has not yet run.
597 return;
598 }
599 ModelTaskController controller = new ModelTaskController(
600 mApp, mBgDataModel, mBgAllAppsList, this, MAIN_EXECUTOR);
601 task.execute(controller, mBgDataModel, mBgAllAppsList);
602 });
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700603 }
604
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700605 /**
606 * A task to be executed on the current callbacks on the UI thread.
607 * If there is no current callbacks, the task is ignored.
608 */
609 public interface CallbackTask {
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700610
Pinyao Ting777c13e2022-09-16 09:44:26 -0700611 void execute(@NonNull Callbacks callbacks);
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700612 }
613
Sunny Goyal99389382024-05-01 14:47:43 -0700614 public interface ModelUpdateTask {
Joe Onorato36115782010-06-17 13:28:48 -0400615
Sunny Goyal99389382024-05-01 14:47:43 -0700616 void execute(@NonNull ModelTaskController taskController,
617 @NonNull BgDataModel dataModel, @NonNull AllAppsList apps);
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800618 }
619
Pinyao Ting777c13e2022-09-16 09:44:26 -0700620 public void updateAndBindWorkspaceItem(@NonNull final WorkspaceItemInfo si,
621 @NonNull final ShortcutInfo info) {
Sunny Goyal95899162019-03-27 16:03:06 -0700622 updateAndBindWorkspaceItem(() -> {
Sunny Goyal83fd25e2018-07-09 16:47:01 -0700623 si.updateFromDeepShortcutInfo(info, mApp.getContext());
Sunny Goyal18204e42020-02-06 11:28:01 -0800624 mApp.getIconCache().getShortcutIcon(si, info);
Sunny Goyal83fd25e2018-07-09 16:47:01 -0700625 return si;
Sunny Goyal1b072632017-01-18 11:30:23 -0800626 });
627 }
628
Sunny Goyal10923b32016-07-20 15:42:44 -0700629 /**
Sunny Goyal1cc1c9a2017-01-06 16:32:57 -0800630 * Utility method to update a shortcut on the background thread.
Sunny Goyal10923b32016-07-20 15:42:44 -0700631 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700632 public void updateAndBindWorkspaceItem(
633 @NonNull final Supplier<WorkspaceItemInfo> itemProvider) {
Sunny Goyal99389382024-05-01 14:47:43 -0700634 enqueueModelUpdateTask((taskController, dataModel, apps) -> {
635 WorkspaceItemInfo info = itemProvider.get();
636 taskController.getModelWriter().updateItemInDatabase(info);
637 ArrayList<WorkspaceItemInfo> update = new ArrayList<>();
638 update.add(info);
639 taskController.bindUpdatedWorkspaceItems(update);
Sunny Goyal10923b32016-07-20 15:42:44 -0700640 });
641 }
642
Sunny Goyalc6e97692017-06-02 13:46:55 -0700643 public void refreshAndBindWidgetsAndShortcuts(@Nullable final PackageUserKey packageUser) {
Sunny Goyal99389382024-05-01 14:47:43 -0700644 enqueueModelUpdateTask((taskController, dataModel, apps) -> {
645 dataModel.widgetsModel.update(taskController.getApp(), packageUser);
646 taskController.bindUpdatedWidgets(dataModel);
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800647 });
Michael Jurkac402cd92013-05-20 15:49:32 +0200648 }
649
Pinyao Ting777c13e2022-09-16 09:44:26 -0700650 public void dumpState(@Nullable final String prefix, @Nullable final FileDescriptor fd,
651 @NonNull final PrintWriter writer, @NonNull final String[] args) {
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -0800652 if (args.length > 0 && TextUtils.equals(args[0], "--all")) {
653 writer.println(prefix + "All apps list: size=" + mBgAllAppsList.data.size());
654 for (AppInfo info : mBgAllAppsList.data) {
Sunny Goyal3808a692019-10-25 13:41:28 -0700655 writer.println(prefix + " title=\"" + info.title
656 + "\" bitmapIcon=" + info.bitmap.icon
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -0800657 + " componentName=" + info.componentName.getPackageName());
658 }
Andy Wickhambb993c42021-02-26 12:56:25 -0800659 writer.println();
Joe Onorato36115782010-06-17 13:28:48 -0400660 }
Andy Wickhambb993c42021-02-26 12:56:25 -0800661 mModelDelegate.dump(prefix, fd, writer, args);
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530662 mBgDataModel.dump(prefix, fd, writer, args);
Joe Onoratobe386092009-11-17 17:32:16 -0800663 }
Sunny Goyale0f58d72014-11-10 18:05:31 -0800664
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530665 /**
666 * Returns true if there are any callbacks attached to the model
667 */
668 public boolean hasCallbacks() {
669 synchronized (mCallbacksList) {
670 return !mCallbacksList.isEmpty();
671 }
672 }
673
674 /**
675 * Returns an array of currently attached callbacks
676 */
Pinyao Ting777c13e2022-09-16 09:44:26 -0700677 @NonNull
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530678 public Callbacks[] getCallbacks() {
679 synchronized (mCallbacksList) {
680 return mCallbacksList.toArray(new Callbacks[mCallbacksList.size()]);
681 }
Sunny Goyale0f58d72014-11-10 18:05:31 -0800682 }
Sunny Goyal7b9e28f2023-05-17 12:44:03 -0700683
684 /**
685 * Returns the ID for the last model load. If the load ID doesn't match for a transaction, the
686 * transaction should be ignored.
687 */
688 public int getLastLoadId() {
689 return mLastLoadId;
690 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800691}