blob: e61b7a85130e82b0d461d156d6de21f93a4ea6a9 [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
Sunny Goyala7a5bf32020-01-05 15:35:29 +053019import static com.android.launcher3.LauncherAppState.ACTION_FORCE_ROLOAD;
Zak Cohen3eeb41d2020-02-14 14:15:13 -080020import static com.android.launcher3.config.FeatureFlags.IS_STUDIO_BUILD;
Sunny Goyala7a5bf32020-01-05 15:35:29 +053021import static com.android.launcher3.util.Executors.MAIN_EXECUTOR;
22import static com.android.launcher3.util.Executors.MODEL_EXECUTOR;
23import static com.android.launcher3.util.PackageManagerHelper.hasShortcutsPermission;
24
Sunny Goyal14168432019-10-24 15:59:49 -070025import android.content.Context;
Sunny Goyalf599ccf2014-07-08 13:01:29 -070026import android.content.Intent;
Sunny Goyale7b00122019-10-02 16:13:34 -070027import android.content.pm.LauncherApps;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070028import android.content.pm.PackageInstaller;
Sunny Goyal6bbf6002019-04-17 18:38:52 -070029import android.content.pm.ShortcutInfo;
Sunny Goyal7c74e4a2016-12-15 15:53:17 -080030import android.os.UserHandle;
Winson Chunga90303b2013-11-15 13:05:06 -080031import android.text.TextUtils;
Winson Chungaafa03c2010-06-11 17:34:16 -070032import android.util.Log;
Sunny Goyala474a9b2017-05-04 16:47:11 -070033import android.util.Pair;
Michael Jurka34c2e6c2013-12-13 16:07:45 +010034
Sunny Goyal6fe3eec2019-08-15 14:53:41 -070035import androidx.annotation.Nullable;
Sunny Goyal14168432019-10-24 15:59:49 -070036import androidx.annotation.WorkerThread;
Sunny Goyal6fe3eec2019-08-15 14:53:41 -070037
Sunny Goyal045b4fa2019-09-20 12:51:37 -070038import com.android.launcher3.config.FeatureFlags;
Sunny Goyalf840f102018-09-21 14:41:05 -070039import com.android.launcher3.icons.IconCache;
Samuel Fufacc1e1072019-09-06 16:19:11 -070040import com.android.launcher3.logging.FileLog;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070041import com.android.launcher3.model.AddWorkspaceItemsTask;
Sunny Goyal87dcde62019-07-17 20:35:56 -070042import com.android.launcher3.model.AllAppsList;
Sunny Goyale75d9f12018-01-31 11:59:00 -080043import com.android.launcher3.model.BaseModelUpdateTask;
Sunny Goyale9956a72016-09-01 17:24:47 -070044import com.android.launcher3.model.BgDataModel;
Sunny Goyal87dcde62019-07-17 20:35:56 -070045import com.android.launcher3.model.BgDataModel.Callbacks;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070046import com.android.launcher3.model.CacheDataUpdatedTask;
Sunny Goyalf8d6ed22017-06-01 14:26:38 -070047import com.android.launcher3.model.LoaderResults;
Sunny Goyalb434fde2017-06-06 10:46:59 -070048import com.android.launcher3.model.LoaderTask;
Sunny Goyal43bf11d2017-02-02 13:52:53 -080049import com.android.launcher3.model.ModelWriter;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -070050import com.android.launcher3.model.PackageInstallStateChangedTask;
51import com.android.launcher3.model.PackageUpdatedTask;
52import com.android.launcher3.model.ShortcutsChangedTask;
53import com.android.launcher3.model.UserLockStateChangedTask;
Sunny Goyal045b4fa2019-09-20 12:51:37 -070054import com.android.launcher3.pm.InstallSessionTracker;
55import com.android.launcher3.pm.PackageInstallInfo;
Sunny Goyal337c81f2019-12-10 12:19:13 -080056import com.android.launcher3.pm.UserCache;
Sunny Goyalfa395362019-12-11 10:00:47 -080057import com.android.launcher3.shortcuts.ShortcutRequest;
Jon Miranda088b9c22019-09-17 11:51:49 -070058import com.android.launcher3.util.IntSparseArrayMap;
59import com.android.launcher3.util.ItemInfoMatcher;
Sunny Goyala7a5bf32020-01-05 15:35:29 +053060import com.android.launcher3.util.LooperExecutor;
Tony Wickham86222d22017-03-29 15:30:43 -070061import com.android.launcher3.util.PackageUserKey;
Sunny Goyalaaf7d1d2016-05-17 13:38:54 -070062import com.android.launcher3.util.Preconditions;
Romain Guyedcce092010-03-04 13:03:17 -080063
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -080064import java.io.FileDescriptor;
65import java.io.PrintWriter;
Michael Jurkac2f801e2011-07-12 14:19:46 -070066import java.util.ArrayList;
Winson Chungb8b2a5a2012-07-12 17:55:31 -070067import java.util.HashSet;
Michael Jurkac2f801e2011-07-12 14:19:46 -070068import java.util.List;
Sunny Goyaldd96a5e2017-02-17 11:22:34 -080069import java.util.concurrent.CancellationException;
Sunny Goyal527c7d32015-08-28 15:19:36 -070070import java.util.concurrent.Executor;
Sunny Goyal8c48d8b2019-01-25 15:10:18 -080071import java.util.function.Supplier;
Michael Jurkac2f801e2011-07-12 14:19:46 -070072
The Android Open Source Project31dd5032009-03-03 19:32:27 -080073/**
74 * Maintains in-memory state of the Launcher. It is expected that there should be only one
75 * LauncherModel object held in a static. Also provide APIs for updating the database state
The Android Open Source Projectbc219c32009-03-09 11:52:14 -070076 * for the Launcher.
The Android Open Source Project31dd5032009-03-03 19:32:27 -080077 */
Sunny Goyale7b00122019-10-02 16:13:34 -070078public class LauncherModel extends LauncherApps.Callback implements InstallSessionTracker.Callback {
Chris Wrenee523362014-09-09 10:09:02 -040079 private static final boolean DEBUG_RECEIVER = false;
Chris Wrenb358f812014-04-16 13:37:00 -040080
Joe Onorato9c1289c2009-08-17 11:03:03 -040081 static final String TAG = "Launcher.Model";
The Android Open Source Projectf96811c2009-03-18 17:39:48 -070082
Sunny Goyala7a5bf32020-01-05 15:35:29 +053083 private final LauncherAppState mApp;
84 private final Object mLock = new Object();
85 private final LooperExecutor mMainExecutor = MAIN_EXECUTOR;
86
87 private LoaderTask mLoaderTask;
88 private boolean mIsLoaderTaskRunning;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080089
Sunny Goyaldd96a5e2017-02-17 11:22:34 -080090 // Indicates whether the current model data is valid or not.
91 // We start off with everything not loaded. After that, we assume that
Joe Onoratocc67f472010-06-08 10:54:30 -070092 // our monitoring of the package manager provides all updates and we never
Sunny Goyaldd96a5e2017-02-17 11:22:34 -080093 // need to do a requery. This is only ever touched from the loader thread.
94 private boolean mModelLoaded;
Hyunyoung Song6aa37292017-02-06 10:46:24 -080095 public boolean isModelLoaded() {
96 synchronized (mLock) {
97 return mModelLoaded && mLoaderTask == null;
98 }
99 }
Joe Onoratocc67f472010-06-08 10:54:30 -0700100
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530101 private final ArrayList<Callbacks> mCallbacksList = new ArrayList<>(1);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800102
Michael Jurkaa8c760d2011-04-28 14:59:33 -0700103 // < only access in worker thread >
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800104 private final AllAppsList mBgAllAppsList;
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700105
Sunny Goyale9956a72016-09-01 17:24:47 -0700106 /**
107 * All the static data should be accessed on the background thread, A lock should be acquired
108 * on this object when accessing any data from this model.
109 */
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530110 private final BgDataModel mBgDataModel = new BgDataModel();
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700111
Sunny Goyalc6e97692017-06-02 13:46:55 -0700112 // Runnable to check if the shortcuts permission has changed.
113 private final Runnable mShortcutPermissionCheckRunnable = new Runnable() {
114 @Override
115 public void run() {
Sunny Goyalfa395362019-12-11 10:00:47 -0800116 if (mModelLoaded && hasShortcutsPermission(mApp.getContext())
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530117 != mBgDataModel.hasShortcutHostPermission) {
Sunny Goyalfa395362019-12-11 10:00:47 -0800118 forceReload();
Sunny Goyalc6e97692017-06-02 13:46:55 -0700119 }
120 }
121 };
Kenny Guyed131872014-04-30 03:02:21 +0100122
Sunny Goyaldde4fd92016-11-21 16:02:39 +0530123 LauncherModel(LauncherAppState app, IconCache iconCache, AppFilter appFilter) {
Daniel Sandlere4f98912013-06-25 15:13:26 -0400124 mApp = app;
Bjorn Bringert1307f632013-10-03 22:31:03 +0100125 mBgAllAppsList = new AllAppsList(iconCache, appFilter);
Joe Onorato0589f0f2010-02-08 13:44:00 -0800126 }
127
Sunny Goyal71b5c0b2015-01-08 16:59:04 -0800128 /**
129 * Adds the provided items to the workspace.
Sunny Goyal71b5c0b2015-01-08 16:59:04 -0800130 */
Sunny Goyal91498ab2017-10-05 15:57:40 -0700131 public void addAndBindAddedWorkspaceItems(List<Pair<ItemInfo, Object>> itemList) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530132 for (Callbacks cb : getCallbacks()) {
133 cb.preAddApps();
Tony Wickham6a71a5b2018-08-21 11:40:23 -0700134 }
Sunny Goyal91498ab2017-10-05 15:57:40 -0700135 enqueueModelUpdateTask(new AddWorkspaceItemsTask(itemList));
Winson Chung64359a52013-07-08 17:17:08 -0700136 }
137
Sunny Goyal605bcf32018-03-02 15:16:12 -0800138 public ModelWriter getWriter(boolean hasVerticalHotseat, boolean verifyChanges) {
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530139 return new ModelWriter(mApp.getContext(), this, mBgDataModel,
Sunny Goyal605bcf32018-03-02 15:16:12 -0800140 hasVerticalHotseat, verifyChanges);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800141 }
142
Kenny Guyed131872014-04-30 03:02:21 +0100143 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800144 public void onPackageChanged(String packageName, UserHandle user) {
Kenny Guyed131872014-04-30 03:02:21 +0100145 int op = PackageUpdatedTask.OP_UPDATE;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700146 enqueueModelUpdateTask(new PackageUpdatedTask(op, user, packageName));
Kenny Guyed131872014-04-30 03:02:21 +0100147 }
148
149 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800150 public void onPackageRemoved(String packageName, UserHandle user) {
Sunny Goyalc2936bc2016-09-01 15:50:36 -0700151 onPackagesRemoved(user, packageName);
152 }
153
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800154 public void onPackagesRemoved(UserHandle user, String... packages) {
Kenny Guyed131872014-04-30 03:02:21 +0100155 int op = PackageUpdatedTask.OP_REMOVE;
Sunny Goyalbb18f4c2020-02-25 15:27:06 -0800156 FileLog.d(TAG, "package removed received " + TextUtils.join(",", packages));
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700157 enqueueModelUpdateTask(new PackageUpdatedTask(op, user, packages));
Kenny Guyed131872014-04-30 03:02:21 +0100158 }
159
160 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800161 public void onPackageAdded(String packageName, UserHandle user) {
Kenny Guyed131872014-04-30 03:02:21 +0100162 int op = PackageUpdatedTask.OP_ADD;
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700163 enqueueModelUpdateTask(new PackageUpdatedTask(op, user, packageName));
Kenny Guyed131872014-04-30 03:02:21 +0100164 }
165
166 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800167 public void onPackagesAvailable(String[] packageNames, UserHandle user,
Kenny Guyed131872014-04-30 03:02:21 +0100168 boolean replacing) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700169 enqueueModelUpdateTask(
170 new PackageUpdatedTask(PackageUpdatedTask.OP_UPDATE, user, packageNames));
Kenny Guyed131872014-04-30 03:02:21 +0100171 }
172
173 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800174 public void onPackagesUnavailable(String[] packageNames, UserHandle user,
Kenny Guyed131872014-04-30 03:02:21 +0100175 boolean replacing) {
176 if (!replacing) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700177 enqueueModelUpdateTask(new PackageUpdatedTask(
178 PackageUpdatedTask.OP_UNAVAILABLE, user, packageNames));
Kenny Guyed131872014-04-30 03:02:21 +0100179 }
Kenny Guyed131872014-04-30 03:02:21 +0100180 }
181
Kenny Guy44cba692016-01-21 19:50:02 +0000182 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800183 public void onPackagesSuspended(String[] packageNames, UserHandle user) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700184 enqueueModelUpdateTask(new PackageUpdatedTask(
185 PackageUpdatedTask.OP_SUSPEND, user, packageNames));
Kenny Guy44cba692016-01-21 19:50:02 +0000186 }
187
188 @Override
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800189 public void onPackagesUnsuspended(String[] packageNames, UserHandle user) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700190 enqueueModelUpdateTask(new PackageUpdatedTask(
191 PackageUpdatedTask.OP_UNSUSPEND, user, packageNames));
Kenny Guy44cba692016-01-21 19:50:02 +0000192 }
193
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700194 @Override
Sunny Goyal6bbf6002019-04-17 18:38:52 -0700195 public void onShortcutsChanged(String packageName, List<ShortcutInfo> shortcuts,
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800196 UserHandle user) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700197 enqueueModelUpdateTask(new ShortcutsChangedTask(packageName, shortcuts, user, true));
Sunny Goyal50941fb2016-08-04 12:03:52 -0700198 }
199
Sunny Goyal14168432019-10-24 15:59:49 -0700200 /**
201 * Called when the icon for an app changes, outside of package event
202 */
203 @WorkerThread
204 public void onAppIconChanged(String packageName, UserHandle user) {
205 // Update the icon for the calendar package
206 Context context = mApp.getContext();
207 onPackageChanged(packageName, user);
208
Sunny Goyalfa395362019-12-11 10:00:47 -0800209 List<ShortcutInfo> pinnedShortcuts = new ShortcutRequest(context, user)
210 .forPackage(packageName).query(ShortcutRequest.PINNED);
Sunny Goyal14168432019-10-24 15:59:49 -0700211 if (!pinnedShortcuts.isEmpty()) {
212 enqueueModelUpdateTask(new ShortcutsChangedTask(packageName, pinnedShortcuts, user,
213 false));
214 }
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700215 }
216
Sunny Goyale7b00122019-10-02 16:13:34 -0700217 public void onBroadcastIntent(Intent intent) {
Chris Wrenb358f812014-04-16 13:37:00 -0400218 if (DEBUG_RECEIVER) Log.d(TAG, "onReceive intent=" + intent);
Joe Onorato36115782010-06-17 13:28:48 -0400219 final String action = intent.getAction();
Kenny Guyed131872014-04-30 03:02:21 +0100220 if (Intent.ACTION_LOCALE_CHANGED.equals(action)) {
Reena Lee93f824a2011-09-23 17:20:28 -0700221 // If we have changed locale we need to clear out the labels in all apps/workspace.
222 forceReload();
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700223 } else if (Intent.ACTION_MANAGED_PROFILE_AVAILABLE.equals(action) ||
224 Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE.equals(action) ||
225 Intent.ACTION_MANAGED_PROFILE_UNLOCKED.equals(action)) {
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800226 UserHandle user = intent.getParcelableExtra(Intent.EXTRA_USER);
Sunny Goyalda891c12016-03-18 18:29:24 -0700227 if (user != null) {
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700228 if (Intent.ACTION_MANAGED_PROFILE_AVAILABLE.equals(action) ||
229 Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE.equals(action)) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700230 enqueueModelUpdateTask(new PackageUpdatedTask(
231 PackageUpdatedTask.OP_USER_AVAILABILITY_CHANGE, user));
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700232 }
233
234 // ACTION_MANAGED_PROFILE_UNAVAILABLE sends the profile back to locked mode, so
235 // we need to run the state change task again.
236 if (Intent.ACTION_MANAGED_PROFILE_UNAVAILABLE.equals(action) ||
237 Intent.ACTION_MANAGED_PROFILE_UNLOCKED.equals(action)) {
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700238 enqueueModelUpdateTask(new UserLockStateChangedTask(user));
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700239 }
Sunny Goyalda891c12016-03-18 18:29:24 -0700240 }
Zak Cohen3eeb41d2020-02-14 14:15:13 -0800241 } else if (IS_STUDIO_BUILD && ACTION_FORCE_ROLOAD.equals(action)) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530242 for (Callbacks cb : getCallbacks()) {
243 if (cb instanceof Launcher) {
244 ((Launcher) cb).recreate();
245 }
246 }
Joe Onoratoe9ad59e2010-10-29 17:35:36 -0700247 }
248 }
249
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800250 /**
251 * Reloads the workspace items from the DB and re-binds the workspace. This should generally
252 * not be called as DB updates are automatically followed by UI update
253 */
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530254 public void forceReload() {
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800255 synchronized (mLock) {
256 // Stop any existing loaders first, so they don't set mModelLoaded to true later
Sunny Goyale86f11f2017-06-06 14:33:18 -0700257 stopLoader();
Sunny Goyaldd96a5e2017-02-17 11:22:34 -0800258 mModelLoaded = false;
259 }
Winson Chungf0c6ae02012-03-21 16:10:31 -0700260
Sunny Goyal29947f02017-12-18 13:49:44 -0800261 // Start the loader if launcher is already running, otherwise the loader will run,
262 // the next time launcher starts
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530263 if (hasCallbacks()) {
264 startLoader();
Joe Onoratoe9ad59e2010-10-29 17:35:36 -0700265 }
Joe Onorato36115782010-06-17 13:28:48 -0400266 }
Joe Onoratof99f8c12009-10-31 17:27:36 -0400267
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530268 /**
269 * Rebinds all existing callbacks with already loaded model
270 */
271 public void rebindCallbacks() {
272 if (hasCallbacks()) {
273 startLoader();
274 }
275 }
276
277 /**
278 * Removes an existing callback
279 */
280 public void removeCallbacks(Callbacks callbacks) {
281 synchronized (mCallbacksList) {
282 Preconditions.assertUIThread();
283 if (mCallbacksList.remove(callbacks)) {
284 if (stopLoader()) {
285 // Rebind existing callbacks
286 startLoader();
287 }
288 }
289 }
290 }
291
292 /**
293 * Adds a callbacks to receive model updates
294 * @return true if workspace load was performed synchronously
295 */
296 public boolean addCallbacksAndLoad(Callbacks callbacks) {
297 synchronized (mLock) {
298 addCallbacks(callbacks);
299 return startLoader();
300
301 }
302 }
303
304 /**
305 * Adds a callbacks to receive model updates
306 */
307 public void addCallbacks(Callbacks callbacks) {
308 Preconditions.assertUIThread();
309 synchronized (mCallbacksList) {
310 mCallbacksList.add(callbacks);
311 }
Adam Cohen1a85c582014-09-30 09:48:49 -0700312 }
313
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700314 /**
315 * Starts the loader. Tries to bind {@params synchronousBindPage} synchronously if possible.
316 * @return true if the page could be bound synchronously.
317 */
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530318 public boolean startLoader() {
Sunny Goyal756adbc2015-04-16 15:20:51 -0700319 // Enable queue before starting loader. It will get disabled in Launcher#finishBindingItems
Sunny Goyala474a9b2017-05-04 16:47:11 -0700320 InstallShortcutReceiver.enableInstallQueue(InstallShortcutReceiver.FLAG_LOADER_RUNNING);
Joe Onorato36115782010-06-17 13:28:48 -0400321 synchronized (mLock) {
Joe Onorato36115782010-06-17 13:28:48 -0400322 // Don't bother to start the thread if we know it's not going to do anything
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530323 final Callbacks[] callbacksList = getCallbacks();
324 if (callbacksList.length > 0) {
Sunny Goyal527c7d32015-08-28 15:19:36 -0700325 // Clear any pending bind-runnables from the synchronized load process.
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530326 for (Callbacks cb : callbacksList) {
327 mMainExecutor.execute(cb::clearPendingBinds);
328 }
Sunny Goyal527c7d32015-08-28 15:19:36 -0700329
Joe Onorato36115782010-06-17 13:28:48 -0400330 // If there is already one running, tell it to stop.
Sunny Goyale86f11f2017-06-06 14:33:18 -0700331 stopLoader();
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530332 LoaderResults loaderResults = new LoaderResults(
333 mApp, mBgDataModel, mBgAllAppsList, callbacksList, mMainExecutor);
Sunny Goyale86f11f2017-06-06 14:33:18 -0700334 if (mModelLoaded && !mIsLoaderTaskRunning) {
Sunny Goyalf8d6ed22017-06-01 14:26:38 -0700335 // Divide the set of loaded items into those that we are binding synchronously,
336 // and everything else that is to be bound normally (asynchronously).
337 loaderResults.bindWorkspace();
338 // For now, continue posting the binding of AllApps as there are other
339 // issues that arise from that.
340 loaderResults.bindAllApps();
341 loaderResults.bindDeepShortcuts();
Sunny Goyalc6e97692017-06-02 13:46:55 -0700342 loaderResults.bindWidgets();
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700343 return true;
Winson Chungb8b2a5a2012-07-12 17:55:31 -0700344 } else {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700345 startLoaderForResults(loaderResults);
Winson Chungb8b2a5a2012-07-12 17:55:31 -0700346 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400347 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400348 }
Sunny Goyalb5b9ad62016-04-02 11:23:39 -0700349 return false;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400350 }
351
Sunny Goyale86f11f2017-06-06 14:33:18 -0700352 /**
353 * If there is already a loader task running, tell it to stop.
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530354 * @return true if an existing loader was stopped.
Sunny Goyale86f11f2017-06-06 14:33:18 -0700355 */
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530356 public boolean stopLoader() {
Joe Onorato36115782010-06-17 13:28:48 -0400357 synchronized (mLock) {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700358 LoaderTask oldTask = mLoaderTask;
359 mLoaderTask = null;
360 if (oldTask != null) {
361 oldTask.stopLocked();
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530362 return true;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400363 }
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530364 return false;
Joe Onorato9c1289c2009-08-17 11:03:03 -0400365 }
Joe Onorato36115782010-06-17 13:28:48 -0400366 }
Joe Onorato9c1289c2009-08-17 11:03:03 -0400367
Sunny Goyale86f11f2017-06-06 14:33:18 -0700368 public void startLoaderForResults(LoaderResults results) {
369 synchronized (mLock) {
370 stopLoader();
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530371 mLoaderTask = new LoaderTask(mApp, mBgAllAppsList, mBgDataModel, results);
Sunny Goyal51f220d2019-05-29 11:05:30 -0700372
373 // Always post the loader task, instead of running directly (even on same thread) so
374 // that we exit any nested synchronized blocks
Sunny Goyal6fe3eec2019-08-15 14:53:41 -0700375 MODEL_EXECUTOR.post(mLoaderTask);
Sunny Goyale86f11f2017-06-06 14:33:18 -0700376 }
377 }
378
Winson Chungd6519662018-02-16 03:23:51 +0000379 public void startLoaderForResultsIfNotLoaded(LoaderResults results) {
380 synchronized (mLock) {
381 if (!isModelLoaded()) {
382 Log.d(TAG, "Workspace not loaded, loading now");
383 startLoaderForResults(results);
384 }
385 }
386 }
387
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700388 @Override
Mario Bertschler817afa32017-03-15 11:56:47 -0700389 public void onInstallSessionCreated(final PackageInstallInfo sessionInfo) {
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700390 if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {
391 enqueueModelUpdateTask(new BaseModelUpdateTask() {
392 @Override
393 public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList apps) {
394 apps.addPromiseApp(app.getContext(), sessionInfo);
395 bindApplicationsIfNeeded();
396 }
397 });
398 }
399 }
400
401 @Override
402 public void onSessionFailure(String packageName, UserHandle user) {
403 if (!FeatureFlags.PROMISE_APPS_NEW_INSTALLS.get()) {
404 return;
405 }
Sunny Goyale86f11f2017-06-06 14:33:18 -0700406 enqueueModelUpdateTask(new BaseModelUpdateTask() {
Mario Bertschler817afa32017-03-15 11:56:47 -0700407 @Override
408 public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList apps) {
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700409 final IntSparseArrayMap<Boolean> removedIds = new IntSparseArrayMap<>();
410 synchronized (dataModel) {
411 for (ItemInfo info : dataModel.itemsIdMap) {
412 if (info instanceof WorkspaceItemInfo
413 && ((WorkspaceItemInfo) info).hasPromiseIconUi()
414 && user.equals(info.user)
415 && info.getIntent() != null
416 && TextUtils.equals(packageName, info.getIntent().getPackage())) {
417 removedIds.put(info.id, true /* remove */);
418 }
419 }
420 }
421
422 if (!removedIds.isEmpty()) {
423 deleteAndBindComponentsRemoved(ItemInfoMatcher.ofItemIds(removedIds, false));
424 }
Mario Bertschler817afa32017-03-15 11:56:47 -0700425 }
426 });
427 }
428
Sunny Goyal045b4fa2019-09-20 12:51:37 -0700429 @Override
430 public void onPackageStateChanged(PackageInstallInfo installInfo) {
431 enqueueModelUpdateTask(new PackageInstallStateChangedTask(installInfo));
432 }
433
434 /**
435 * Updates the icons and label of all pending icons for the provided package name.
436 */
437 @Override
438 public void onUpdateSessionDisplay(PackageUserKey key, PackageInstaller.SessionInfo info) {
439 mApp.getIconCache().updateSessionCache(key, info);
440
441 HashSet<String> packages = new HashSet<>();
442 packages.add(key.mPackageName);
443 enqueueModelUpdateTask(new CacheDataUpdatedTask(
444 CacheDataUpdatedTask.OP_SESSION_UPDATE, key.mUser, packages));
445 }
446
Sunny Goyalc6e97692017-06-02 13:46:55 -0700447 public class LoaderTransaction implements AutoCloseable {
448
449 private final LoaderTask mTask;
450
451 private LoaderTransaction(LoaderTask task) throws CancellationException {
452 synchronized (mLock) {
453 if (mLoaderTask != task) {
454 throw new CancellationException("Loader already stopped");
455 }
456 mTask = task;
457 mIsLoaderTaskRunning = true;
458 mModelLoaded = false;
459 }
460 }
461
462 public void commit() {
463 synchronized (mLock) {
464 // Everything loaded bind the data.
465 mModelLoaded = true;
466 }
467 }
468
469 @Override
470 public void close() {
471 synchronized (mLock) {
472 // If we are still the last one to be scheduled, remove ourselves.
473 if (mLoaderTask == mTask) {
474 mLoaderTask = null;
475 }
476 mIsLoaderTaskRunning = false;
477 }
478 }
479 }
480
481 public LoaderTransaction beginLoader(LoaderTask task) throws CancellationException {
482 return new LoaderTransaction(task);
483 }
484
Joe Onorato36115782010-06-17 13:28:48 -0400485 /**
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700486 * Refreshes the cached shortcuts if the shortcut permission has changed.
487 * Current implementation simply reloads the workspace, but it can be optimized to
Sunny Goyal337c81f2019-12-10 12:19:13 -0800488 * use partial updates similar to {@link UserCache}
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700489 */
490 public void refreshShortcutsIfRequired() {
Sunny Goyal6fe3eec2019-08-15 14:53:41 -0700491 MODEL_EXECUTOR.getHandler().removeCallbacks(mShortcutPermissionCheckRunnable);
492 MODEL_EXECUTOR.post(mShortcutPermissionCheckRunnable);
Sunny Goyal95f3d6b2016-08-10 16:09:29 -0700493 }
494
495 /**
Sunny Goyal75b0f552015-05-20 21:57:06 -0700496 * Called when the icons for packages have been updated in the icon cache.
497 */
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800498 public void onPackageIconsUpdated(HashSet<String> updatedPackages, UserHandle user) {
Sunny Goyal75b0f552015-05-20 21:57:06 -0700499 // If any package icon has changed (app was updated while launcher was dead),
500 // update the corresponding shortcuts.
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700501 enqueueModelUpdateTask(new CacheDataUpdatedTask(
502 CacheDataUpdatedTask.OP_CACHE_UPDATE, user, updatedPackages));
Sunny Goyal75b0f552015-05-20 21:57:06 -0700503 }
504
Sunny Goyalac8154a2018-09-26 12:00:30 -0700505 /**
506 * Called when the labels for the widgets has updated in the icon cache.
507 */
508 public void onWidgetLabelsUpdated(HashSet<String> updatedPackages, UserHandle user) {
509 enqueueModelUpdateTask(new BaseModelUpdateTask() {
510 @Override
511 public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList apps) {
512 dataModel.widgetsModel.onPackageIconsUpdated(updatedPackages, user, app);
513 bindUpdatedWidgets(dataModel);
514 }
515 });
516 }
517
Sunny Goyale86f11f2017-06-06 14:33:18 -0700518 public void enqueueModelUpdateTask(ModelUpdateTask task) {
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530519 task.init(mApp, this, mBgDataModel, mBgAllAppsList, mMainExecutor);
Sunny Goyal6fe3eec2019-08-15 14:53:41 -0700520 MODEL_EXECUTOR.execute(task);
Sunny Goyald3b87ef2016-07-28 12:11:54 -0700521 }
522
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700523 /**
524 * A task to be executed on the current callbacks on the UI thread.
525 * If there is no current callbacks, the task is ignored.
526 */
527 public interface CallbackTask {
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700528
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700529 void execute(Callbacks callbacks);
Tony Wickhambfbf7f92016-05-19 11:19:39 -0700530 }
531
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700532 /**
533 * A runnable which changes/updates the data model of the launcher based on certain events.
534 */
Sunny Goyale86f11f2017-06-06 14:33:18 -0700535 public interface ModelUpdateTask extends Runnable {
Joe Onorato36115782010-06-17 13:28:48 -0400536
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700537 /**
Sunny Goyale86f11f2017-06-06 14:33:18 -0700538 * Called before the task is posted to initialize the internal state.
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700539 */
Sunny Goyale86f11f2017-06-06 14:33:18 -0700540 void init(LauncherAppState app, LauncherModel model,
541 BgDataModel dataModel, AllAppsList allAppsList, Executor uiExecutor);
Sunny Goyale0f58d72014-11-10 18:05:31 -0800542
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800543 }
544
Sunny Goyal6bbf6002019-04-17 18:38:52 -0700545 public void updateAndBindWorkspaceItem(WorkspaceItemInfo si, ShortcutInfo info) {
Sunny Goyal95899162019-03-27 16:03:06 -0700546 updateAndBindWorkspaceItem(() -> {
Sunny Goyal83fd25e2018-07-09 16:47:01 -0700547 si.updateFromDeepShortcutInfo(info, mApp.getContext());
Sunny Goyal18204e42020-02-06 11:28:01 -0800548 mApp.getIconCache().getShortcutIcon(si, info);
Sunny Goyal83fd25e2018-07-09 16:47:01 -0700549 return si;
Sunny Goyal1b072632017-01-18 11:30:23 -0800550 });
551 }
552
Sunny Goyal10923b32016-07-20 15:42:44 -0700553 /**
Sunny Goyal1cc1c9a2017-01-06 16:32:57 -0800554 * Utility method to update a shortcut on the background thread.
Sunny Goyal10923b32016-07-20 15:42:44 -0700555 */
Sunny Goyal95899162019-03-27 16:03:06 -0700556 public void updateAndBindWorkspaceItem(final Supplier<WorkspaceItemInfo> itemProvider) {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700557 enqueueModelUpdateTask(new BaseModelUpdateTask() {
Sunny Goyal10923b32016-07-20 15:42:44 -0700558 @Override
Sunny Goyalf0ba8b72016-09-09 15:47:55 -0700559 public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList apps) {
Sunny Goyal95899162019-03-27 16:03:06 -0700560 WorkspaceItemInfo info = itemProvider.get();
Sunny Goyalf266deb2018-06-21 09:59:08 -0700561 getModelWriter().updateItemInDatabase(info);
Sunny Goyal95899162019-03-27 16:03:06 -0700562 ArrayList<WorkspaceItemInfo> update = new ArrayList<>();
Sunny Goyal10923b32016-07-20 15:42:44 -0700563 update.add(info);
Sunny Goyal95899162019-03-27 16:03:06 -0700564 bindUpdatedWorkspaceItems(update);
Sunny Goyal10923b32016-07-20 15:42:44 -0700565 }
566 });
567 }
568
Sunny Goyalc6e97692017-06-02 13:46:55 -0700569 public void refreshAndBindWidgetsAndShortcuts(@Nullable final PackageUserKey packageUser) {
Sunny Goyale86f11f2017-06-06 14:33:18 -0700570 enqueueModelUpdateTask(new BaseModelUpdateTask() {
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800571 @Override
Sunny Goyalc6e97692017-06-02 13:46:55 -0700572 public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList apps) {
573 dataModel.widgetsModel.update(app, packageUser);
574 bindUpdatedWidgets(dataModel);
Sunny Goyal31860582015-09-18 08:38:57 -0700575 }
Sunny Goyal2e1efb42016-03-03 16:58:55 -0800576 });
Michael Jurkac402cd92013-05-20 15:49:32 +0200577 }
578
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -0800579 public void dumpState(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) {
580 if (args.length > 0 && TextUtils.equals(args[0], "--all")) {
581 writer.println(prefix + "All apps list: size=" + mBgAllAppsList.data.size());
582 for (AppInfo info : mBgAllAppsList.data) {
Sunny Goyal3808a692019-10-25 13:41:28 -0700583 writer.println(prefix + " title=\"" + info.title
584 + "\" bitmapIcon=" + info.bitmap.icon
Hyunyoung Song3c7d9cb2017-01-30 15:11:27 -0800585 + " componentName=" + info.componentName.getPackageName());
586 }
Joe Onorato36115782010-06-17 13:28:48 -0400587 }
Sunny Goyal9ae9b602019-12-18 19:29:00 +0530588 mBgDataModel.dump(prefix, fd, writer, args);
Joe Onoratobe386092009-11-17 17:32:16 -0800589 }
Sunny Goyale0f58d72014-11-10 18:05:31 -0800590
Sunny Goyala7a5bf32020-01-05 15:35:29 +0530591 /**
592 * Returns true if there are any callbacks attached to the model
593 */
594 public boolean hasCallbacks() {
595 synchronized (mCallbacksList) {
596 return !mCallbacksList.isEmpty();
597 }
598 }
599
600 /**
601 * Returns an array of currently attached callbacks
602 */
603 public Callbacks[] getCallbacks() {
604 synchronized (mCallbacksList) {
605 return mCallbacksList.toArray(new Callbacks[mCallbacksList.size()]);
606 }
Sunny Goyale0f58d72014-11-10 18:05:31 -0800607 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800608}