AndroidAOSP定制隐藏某个应用的图标
AndroidAOSP定制隐藏某个应用的图标
1.前言:
之前在做AOSP定制的时候需要隐藏某些App的图标,或者默认不显示某个定制的App图标,这样可以让用户感觉不到已经安装了某个App,或者在做系统定制的时候需要修改桌面icon,有些系统的App图标默认不需要显示,因为原生的系统App很多,有些用户又不需要全部在桌面显示,这时候就需要隐藏某些App的图标了,直接上代码。
2.源码路径如下:
/packages/apps/Launcher3/src/com/android/launcher3/model/LoaderTask
private List<LauncherActivityInfo> loadAllApps() {final List<UserHandle> profiles = mUserCache.getUserProfiles();List<LauncherActivityInfo> allActivityList = new ArrayList<>();// Clear the list of appsmBgAllAppsList.clear();for (UserHandle user : profiles) {// Query for the set of appsfinal List<LauncherActivityInfo> apps = mLauncherApps.getActivityList(null, user);// Fail if we don't have any apps// TODO: Fix this. Only fail for the current user.if (apps == null || apps.isEmpty()) {return allActivityList;}boolean quietMode = mUserManagerState.isUserQuiet(user);// Create the ApplicationInfosfor (int i = 0; i < apps.size(); i++) {LauncherActivityInfo app = apps.get(i);// This builds the icon bitmaps.mBgAllAppsList.add(new AppInfo(app, user, quietMode), app);}allActivityList.addAll(apps);}if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {// get all active sessions and add them to the all apps listfor (PackageInstaller.SessionInfo info :mSessionHelper.getAllVerifiedSessions()) {mBgAllAppsList.addPromiseApp(mApp.getContext(),PackageInstallInfo.fromInstallingState(info));}}for (AppInfo item : mBgDataModel.cachedPredictedItems) {List<LauncherActivityInfo> l = mLauncherApps.getActivityList(item.componentName.getPackageName(), item.user);for (LauncherActivityInfo info : l) {boolean quietMode = mUserManagerState.isUserQuiet(item.user);mBgAllAppsList.add(new AppInfo(info, item.user, quietMode), info);}}mBgAllAppsList.setFlags(FLAG_QUIET_MODE_ENABLED,mUserManagerState.isAnyProfileQuietModeEnabled());mBgAllAppsList.setFlags(FLAG_HAS_SHORTCUT_PERMISSION,hasShortcutsPermission(mApp.getContext()));mBgAllAppsList.setFlags(FLAG_QUIET_MODE_CHANGE_PERMISSION,mApp.getContext().checkSelfPermission("android.permission.MODIFY_QUIET_MODE")== PackageManager.PERMISSION_GRANTED);mBgAllAppsList.getAndResetChangeFlag();return allActivityList;
}
3.修改的核心方法如下:

private List<LauncherActivityInfo> loadAllApps() {final List<UserHandle> profiles = mUserCache.getUserProfiles();List<LauncherActivityInfo> allActivityList = new ArrayList<>();// Clear the list of appsmBgAllAppsList.clear();for (UserHandle user : profiles) {// Query for the set of appsfinal List<LauncherActivityInfo> apps = mLauncherApps.getActivityList(null, user);// Fail if we don't have any apps// TODO: Fix this. Only fail for the current user.if (apps == null || apps.isEmpty()) {return allActivityList;}boolean quietMode = mUserManagerState.isUserQuiet(user);// Create the ApplicationInfosfor (int i = 0; i < apps.size(); i++) {LauncherActivityInfo app = apps.get(i);// This builds the icon bitmaps.String pkg = app.getComponentName().getPackageName();if (pkg.equals("com.android.calendar")|| pkg.equals("com.android.deskclock")|| pkg.equals("com.android.dialer")|| pkg.equals("com.android.quicksearchbox")|| pkg.equals("com.android.contacts")|| pkg.equals("com.android.camera2")|| pkg.equals("com.android.email")|| pkg.equals("com.android.calculator2")|| pkg.equals("org.chromium.webview_shell")|| pkg.equals("com.android.gallery3d")|| pkg.equals("com.android.music")|| pkg.equals("com.android.browser")|| pkg.equals("com.google.android.browser")) {continue;}mBgAllAppsList.add(new AppInfo(app, user, quietMode), app);}allActivityList.addAll(apps);}
4.隐藏系统通讯录图标:
private List<LauncherActivityInfo> loadAllApps() {final List<UserHandle> profiles = mUserCache.getUserProfiles();List<LauncherActivityInfo> allActivityList = new ArrayList<>();// Clear the list of appsmBgAllAppsList.clear();for (UserHandle user : profiles) {// Query for the set of appsfinal List<LauncherActivityInfo> apps = mLauncherApps.getActivityList(null, user);// Fail if we don't have any apps// TODO: Fix this. Only fail for the current user.if (apps == null || apps.isEmpty()) {return allActivityList;}boolean quietMode = mUserManagerState.isUserQuiet(user);// Create the ApplicationInfosfor (int i = 0; i < apps.size(); i++) {LauncherActivityInfo app = apps.get(i);// This builds the icon bitmaps.String pkg = app.getComponentName().getPackageName();if (pkg.equals("com.android.contacts")) {continue;}mBgAllAppsList.add(new AppInfo(app, user, quietMode), app);}allActivityList.addAll(apps);}if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {// get all active sessions and add them to the all apps listfor (PackageInstaller.SessionInfo info :mSessionHelper.getAllVerifiedSessions()) {mBgAllAppsList.addPromiseApp(mApp.getContext(),PackageInstallInfo.fromInstallingState(info));}}for (AppInfo item : mBgDataModel.cachedPredictedItems) {List<LauncherActivityInfo> l = mLauncherApps.getActivityList(item.componentName.getPackageName(), item.user);for (LauncherActivityInfo info : l) {boolean quietMode = mUserManagerState.isUserQuiet(item.user);mBgAllAppsList.add(new AppInfo(info, item.user, quietMode), info);}}mBgAllAppsList.setFlags(FLAG_QUIET_MODE_ENABLED,mUserManagerState.isAnyProfileQuietModeEnabled());mBgAllAppsList.setFlags(FLAG_HAS_SHORTCUT_PERMISSION,hasShortcutsPermission(mApp.getContext()));mBgAllAppsList.setFlags(FLAG_QUIET_MODE_CHANGE_PERMISSION,mApp.getContext().checkSelfPermission("android.permission.MODIFY_QUIET_MODE")== PackageManager.PERMISSION_GRANTED);mBgAllAppsList.getAndResetChangeFlag();return allActivityList;
}
5.隐藏系统相册图标:
private List<LauncherActivityInfo> loadAllApps() {final List<UserHandle> profiles = mUserCache.getUserProfiles();List<LauncherActivityInfo> allActivityList = new ArrayList<>();// Clear the list of appsmBgAllAppsList.clear();for (UserHandle user : profiles) {// Query for the set of appsfinal List<LauncherActivityInfo> apps = mLauncherApps.getActivityList(null, user);// Fail if we don't have any apps// TODO: Fix this. Only fail for the current user.if (apps == null || apps.isEmpty()) {return allActivityList;}boolean quietMode = mUserManagerState.isUserQuiet(user);// Create the ApplicationInfosfor (int i = 0; i < apps.size(); i++) {LauncherActivityInfo app = apps.get(i);// This builds the icon bitmaps.String pkg = app.getComponentName().getPackageName();if (pkg.equals("com.android.gallery3d")|| pkg.equals("com.android.camera2")) {continue;}mBgAllAppsList.add(new AppInfo(app, user, quietMode), app);}allActivityList.addAll(apps);}if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {// get all active sessions and add them to the all apps listfor (PackageInstaller.SessionInfo info :mSessionHelper.getAllVerifiedSessions()) {mBgAllAppsList.addPromiseApp(mApp.getContext(),PackageInstallInfo.fromInstallingState(info));}}for (AppInfo item : mBgDataModel.cachedPredictedItems) {List<LauncherActivityInfo> l = mLauncherApps.getActivityList(item.componentName.getPackageName(), item.user);for (LauncherActivityInfo info : l) {boolean quietMode = mUserManagerState.isUserQuiet(item.user);mBgAllAppsList.add(new AppInfo(info, item.user, quietMode), info);}}mBgAllAppsList.setFlags(FLAG_QUIET_MODE_ENABLED,mUserManagerState.isAnyProfileQuietModeEnabled());mBgAllAppsList.setFlags(FLAG_HAS_SHORTCUT_PERMISSION,hasShortcutsPermission(mApp.getContext()));mBgAllAppsList.setFlags(FLAG_QUIET_MODE_CHANGE_PERMISSION,mApp.getContext().checkSelfPermission("android.permission.MODIFY_QUIET_MODE")== PackageManager.PERMISSION_GRANTED);mBgAllAppsList.getAndResetChangeFlag();return allActivityList;
}
6.隐藏系统webview图标:
这里有个坑需要注意!!!
这里有个坑需要注意!!!
这里有个坑需要注意!!!
重要的事说三遍,webview在系统应用中可能会有多个包名,如果只是单纯添加某一个会不生效,本人是做过多次测试和修改,最后是经过查找资料和询问大佬实践多次才解决此问题,当时一度以为代码出现问题快emo了,最后才发现是系统webview可能安装了多个应用,导致存在多个包名,使用下面的方式进行过滤,感兴趣的朋友自己尝试.
private List<LauncherActivityInfo> loadAllApps() {final List<UserHandle> profiles = mUserCache.getUserProfiles();List<LauncherActivityInfo> allActivityList = new ArrayList<>();// Clear the list of appsmBgAllAppsList.clear();for (UserHandle user : profiles) {// Query for the set of appsfinal List<LauncherActivityInfo> apps = mLauncherApps.getActivityList(null, user);// Fail if we don't have any apps// TODO: Fix this. Only fail for the current user.if (apps == null || apps.isEmpty()) {return allActivityList;}boolean quietMode = mUserManagerState.isUserQuiet(user);// Create the ApplicationInfosfor (int i = 0; i < apps.size(); i++) {LauncherActivityInfo app = apps.get(i);// This builds the icon bitmaps.String pkg = app.getComponentName().getPackageName();if (pkg.equals("com.android.browser")|| pkg.equals("com.android.webview")|| pkg.equals("org.chromium.webview_shell")|| pkg.equals("com.google.android.browser")) {continue;}mBgAllAppsList.add(new AppInfo(app, user, quietMode), app);}allActivityList.addAll(apps);}if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {// get all active sessions and add them to the all apps listfor (PackageInstaller.SessionInfo info :mSessionHelper.getAllVerifiedSessions()) {mBgAllAppsList.addPromiseApp(mApp.getContext(),PackageInstallInfo.fromInstallingState(info));}}for (AppInfo item : mBgDataModel.cachedPredictedItems) {List<LauncherActivityInfo> l = mLauncherApps.getActivityList(item.componentName.getPackageName(), item.user);for (LauncherActivityInfo info : l) {boolean quietMode = mUserManagerState.isUserQuiet(item.user);mBgAllAppsList.add(new AppInfo(info, item.user, quietMode), info);}}mBgAllAppsList.setFlags(FLAG_QUIET_MODE_ENABLED,mUserManagerState.isAnyProfileQuietModeEnabled());mBgAllAppsList.setFlags(FLAG_HAS_SHORTCUT_PERMISSION,hasShortcutsPermission(mApp.getContext()));mBgAllAppsList.setFlags(FLAG_QUIET_MODE_CHANGE_PERMISSION,mApp.getContext().checkSelfPermission("android.permission.MODIFY_QUIET_MODE")== PackageManager.PERMISSION_GRANTED);mBgAllAppsList.getAndResetChangeFlag();return allActivityList;
}
7.封装工具类:
在源码路径添加:/packages/apps/Launcher3/src/com/android/launcher3/util/HideAppIconUtils
HideAppIconUtils工具类
package com.android.launcher3.util;import java.util.LinkedList;
import java.util.List;
/*** @author: smile* @date: 2025/2/24 23:45* @desc: 隐藏指定App的图标*/
public class HideAppIconUtils {private static final List<String> mPackageMap = new LinkedList<>();private HideAppIconUtils() {addHideAppList();}//需要隐藏的系统App包名,添加到这里private void addHideAppList() {mPackageMap.add("com.android.calendar");mPackageMap.add("com.android.deskclock");mPackageMap.add("com.android.quicksearchbox");mPackageMap.add("com.android.contacts");mPackageMap.add("com.android.camera2");mPackageMap.add("com.android.email");mPackageMap.add("com.android.calculator2");mPackageMap.add("com.android.gallery3d");mPackageMap.add("com.android.music");mPackageMap.add("org.chromium.webview_shell");mPackageMap.add("com.android.browser");mPackageMap.add("com.google.android.browser");}private static class HideAppIconUtilsHolder {private static final HideAppIconUtils INSTALL = new HideAppIconUtils();}public static HideAppIconUtils getInstall() {return HideAppIconUtilsHolder.INSTALL;}public boolean isHidedApkPackageName(String apkPkg) {return mPackageMap.contains(apkPkg);}}
}
8.更新时修改Launche3:
源码路径:/packages/apps/Launcher3/src/com/android/launcher3/model/PackageUpdatedTask
PackageUpdatedTask类代码如下:此类看源码注释就是说明在App包名发生改变或者桌面App发生变化时有一个线程Handler会更新或者删除相应的内容.
/** Copyright (C) 2016 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.model;import static com.android.launcher3.model.BgDataModel.Callbacks.FLAG_QUIET_MODE_ENABLED;
import static com.android.launcher3.model.data.WorkspaceItemInfo.FLAG_AUTOINSTALL_ICON;
import static com.android.launcher3.model.data.WorkspaceItemInfo.FLAG_RESTORED_ICON;import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.pm.LauncherApps;
import android.content.pm.ShortcutInfo;
import android.os.Process;
import android.os.UserHandle;
import android.os.UserManager;
import android.util.Log;
import android.util.Pair;import com.android.launcher3.InstallShortcutReceiver;
import com.android.launcher3.LauncherAppState;
import com.android.launcher3.LauncherSettings.Favorites;
import com.android.launcher3.SessionCommitReceiver;
import com.android.launcher3.Utilities;
import com.android.launcher3.config.FeatureFlags;
import com.android.launcher3.icons.BitmapInfo;
import com.android.launcher3.icons.IconCache;
import com.android.launcher3.icons.LauncherIcons;
import com.android.launcher3.logging.FileLog;
import com.android.launcher3.model.data.ItemInfo;
import com.android.launcher3.model.data.LauncherAppWidgetInfo;
import com.android.launcher3.model.data.WorkspaceItemInfo;
import com.android.launcher3.pm.UserCache;
import com.android.launcher3.shortcuts.ShortcutRequest;
import com.android.launcher3.util.FlagOp;
import com.android.launcher3.util.IntSparseArrayMap;
import com.android.launcher3.util.ItemInfoMatcher;
import com.android.launcher3.util.PackageManagerHelper;
import com.android.launcher3.util.PackageUserKey;
import com.android.launcher3.util.SafeCloseable;
import com.android.launcher3.model.data.AppInfo;import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;/*** Handles updates due to changes in package manager (app installed/updated/removed)* or when a user availability changes.*/
public class PackageUpdatedTask extends BaseModelUpdateTask {private static final boolean DEBUG = false;private static final String TAG = "PackageUpdatedTask";public static final int OP_NONE = 0;public static final int OP_ADD = 1;public static final int OP_UPDATE = 2;public static final int OP_REMOVE = 3; // uninstalledpublic static final int OP_UNAVAILABLE = 4; // external media unmountedpublic static final int OP_SUSPEND = 5; // package suspendedpublic static final int OP_UNSUSPEND = 6; // package unsuspendedpublic static final int OP_USER_AVAILABILITY_CHANGE = 7; // user available/unavailableprivate final int mOp;private final UserHandle mUser;private final String[] mPackages;public PackageUpdatedTask(int op, UserHandle user, String... packages) {mOp = op;mUser = user;mPackages = packages;}@Overridepublic void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList appsList) {final Context context = app.getContext();final IconCache iconCache = app.getIconCache();final String[] packages = mPackages;final int N = packages.length;FlagOp flagOp = FlagOp.NO_OP;final HashSet<String> packageSet = new HashSet<>(Arrays.asList(packages));ItemInfoMatcher matcher = ItemInfoMatcher.ofPackages(packageSet, mUser);final HashSet<ComponentName> removedComponents = new HashSet<>();switch (mOp) {case OP_ADD: {for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.addPackage " + packages[i]);String pkg = packages[i];if ("com.android.calendar".equals(pkg)|| "com.android.deskclock".equals(pkg)|| "com.android.dialer".equals(pkg)|| "com.android.browser".equals(pkg)|| "com.android.contacts".equals(pkg)|| "com.android.camera2".equals(pkg)|| "com.android.email".equals(pkg)|| "com.android.calculator2".equals(pkg)|| "com.android.webview".equals(pkg)|| "com.android.gallery3d".equals(pkg)|| "com.android.music".equals(pkg)) {continue;}iconCache.updateIconsForPkg(packages[i], mUser);if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {appsList.removePackage(packages[i], mUser);}appsList.addPackage(context, packages[i], mUser);// Automatically add homescreen icon for work profile apps for below O device.if (!Utilities.ATLEAST_OREO && !Process.myUserHandle().equals(mUser)) {SessionCommitReceiver.queueAppIconAddition(context, packages[i], mUser);}}flagOp = FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;}case OP_UPDATE:try (SafeCloseable t =appsList.trackRemoves(a -> removedComponents.add(a.componentName))) {for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.updatePackage " + packages[i]);String pkg = packages[i];if ("com.android.calendar".equals(pkg)|| "com.android.deskclock".equals(pkg)|| "com.android.dialer".equals(pkg)|| "com.android.browser".equals(pkg)|| "com.android.contacts".equals(pkg)|| "com.android.camera2".equals(pkg)|| "com.android.email".equals(pkg)|| "com.android.calculator2".equals(pkg)|| "com.android.webview".equals(pkg)|| "com.android.gallery3d".equals(pkg)|| "com.android.music".equals(pkg)) {continue;}iconCache.updateIconsForPkg(packages[i], mUser);appsList.updatePackage(context, packages[i], mUser);app.getWidgetCache().removePackage(packages[i], mUser);}}// Since package was just updated, the target must be available now.flagOp = FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;case OP_REMOVE: {for (int i = 0; i < N; i++) {FileLog.d(TAG, "Removing app icon" + packages[i]);iconCache.removeIconsForPkg(packages[i], mUser);}// Fall through}case OP_UNAVAILABLE:for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.removePackage " + packages[i]);appsList.removePackage(packages[i], mUser);app.getWidgetCache().removePackage(packages[i], mUser);}flagOp = FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;case OP_SUSPEND:case OP_UNSUSPEND:flagOp = mOp == OP_SUSPEND ?FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_SUSPENDED) :FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_SUSPENDED);if (DEBUG) Log.d(TAG, "mAllAppsList.(un)suspend " + N);appsList.updateDisabledFlags(matcher, flagOp);break;case OP_USER_AVAILABILITY_CHANGE: {UserManagerState ums = new UserManagerState();ums.init(UserCache.INSTANCE.get(context),context.getSystemService(UserManager.class));flagOp = ums.isUserQuiet(mUser)? FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_QUIET_USER): FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_QUIET_USER);// We want to update all packages for this user.matcher = ItemInfoMatcher.ofUser(mUser);appsList.updateDisabledFlags(matcher, flagOp);// We are not synchronizing here, as int operations are atomicappsList.setFlags(FLAG_QUIET_MODE_ENABLED, ums.isAnyProfileQuietModeEnabled());break;}}bindApplicationsIfNeeded();final IntSparseArrayMap<Boolean> removedShortcuts = new IntSparseArrayMap<>();// Update shortcut infosif (mOp == OP_ADD || flagOp != FlagOp.NO_OP) {final ArrayList<WorkspaceItemInfo> updatedWorkspaceItems = new ArrayList<>();final ArrayList<LauncherAppWidgetInfo> widgets = new ArrayList<>();// For system apps, package manager send OP_UPDATE when an app is enabled.final boolean isNewApkAvailable = mOp == OP_ADD || mOp == OP_UPDATE;synchronized (dataModel) {for (ItemInfo info : dataModel.itemsIdMap) {if (info instanceof WorkspaceItemInfo && mUser.equals(info.user)) {WorkspaceItemInfo si = (WorkspaceItemInfo) info;boolean infoUpdated = false;boolean shortcutUpdated = false;// Update shortcuts which use iconResource.if ((si.iconResource != null)&& packageSet.contains(si.iconResource.packageName)) {LauncherIcons li = LauncherIcons.obtain(context);BitmapInfo iconInfo = li.createIconBitmap(si.iconResource);li.recycle();if (iconInfo != null) {si.bitmap = iconInfo;infoUpdated = true;}}ComponentName cn = si.getTargetComponent();if (cn != null && matcher.matches(si, cn)) {String packageName = cn.getPackageName();if (si.hasStatusFlag(WorkspaceItemInfo.FLAG_SUPPORTS_WEB_UI)) {removedShortcuts.put(si.id, false);if (mOp == OP_REMOVE) {continue;}}if (si.isPromise() && isNewApkAvailable) {boolean isTargetValid = true;if (si.itemType == Favorites.ITEM_TYPE_DEEP_SHORTCUT) {List<ShortcutInfo> shortcut =new ShortcutRequest(context, mUser).forPackage(cn.getPackageName(),si.getDeepShortcutId()).query(ShortcutRequest.PINNED);if (shortcut.isEmpty()) {isTargetValid = false;} else {si.updateFromDeepShortcutInfo(shortcut.get(0), context);infoUpdated = true;}} else if (!cn.getClassName().equals(IconCache.EMPTY_CLASS_NAME)) {isTargetValid = context.getSystemService(LauncherApps.class).isActivityEnabled(cn, mUser);}if (si.hasStatusFlag(FLAG_RESTORED_ICON | FLAG_AUTOINSTALL_ICON)) {if (updateWorkspaceItemIntent(context, si, packageName)) {infoUpdated = true;} else if (si.hasPromiseIconUi()) {removedShortcuts.put(si.id, true);continue;}} else if (!isTargetValid) {removedShortcuts.put(si.id, true);FileLog.e(TAG, "Restored shortcut no longer valid "+ si.getIntent());continue;} else {si.status = WorkspaceItemInfo.DEFAULT;infoUpdated = true;}} else if (isNewApkAvailable && removedComponents.contains(cn)) {if (updateWorkspaceItemIntent(context, si, packageName)) {infoUpdated = true;}}if (isNewApkAvailable &&si.itemType == Favorites.ITEM_TYPE_APPLICATION) {iconCache.getTitleAndIcon(si, si.usingLowResIcon());infoUpdated = true;}int oldRuntimeFlags = si.runtimeStatusFlags;si.runtimeStatusFlags = flagOp.apply(si.runtimeStatusFlags);if (si.runtimeStatusFlags != oldRuntimeFlags) {shortcutUpdated = true;}}if (infoUpdated || shortcutUpdated) {updatedWorkspaceItems.add(si);}if (infoUpdated) {getModelWriter().updateItemInDatabase(si);}} else if (info instanceof LauncherAppWidgetInfo && isNewApkAvailable) {LauncherAppWidgetInfo widgetInfo = (LauncherAppWidgetInfo) info;if (mUser.equals(widgetInfo.user)&& widgetInfo.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY)&& packageSet.contains(widgetInfo.providerName.getPackageName())) {widgetInfo.restoreStatus &=~LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY &~LauncherAppWidgetInfo.FLAG_RESTORE_STARTED;// adding this flag ensures that launcher shows 'click to setup'// if the widget has a config activity. In case there is no config// activity, it will be marked as 'restored' during bind.widgetInfo.restoreStatus |= LauncherAppWidgetInfo.FLAG_UI_NOT_READY;widgets.add(widgetInfo);getModelWriter().updateItemInDatabase(widgetInfo);}}}}bindUpdatedWorkspaceItems(updatedWorkspaceItems);if (!removedShortcuts.isEmpty()) {deleteAndBindComponentsRemoved(ItemInfoMatcher.ofItemIds(removedShortcuts, false));}if (!widgets.isEmpty()) {scheduleCallbackTask(c -> c.bindWidgetsRestored(widgets));}}final HashSet<String> removedPackages = new HashSet<>();if (mOp == OP_REMOVE) {// Mark all packages in the broadcast to be removedCollections.addAll(removedPackages, packages);// No need to update the removedComponents as// removedPackages is a super-set of removedComponents} else if (mOp == OP_UPDATE) {// Mark disabled packages in the broadcast to be removedfinal LauncherApps launcherApps = context.getSystemService(LauncherApps.class);for (int i=0; i<N; i++) {if (!launcherApps.isPackageEnabled(packages[i], mUser)) {removedPackages.add(packages[i]);}}}if (!removedPackages.isEmpty() || !removedComponents.isEmpty()) {ItemInfoMatcher removeMatch = ItemInfoMatcher.ofPackages(removedPackages, mUser).or(ItemInfoMatcher.ofComponents(removedComponents, mUser)).and(ItemInfoMatcher.ofItemIds(removedShortcuts, true));deleteAndBindComponentsRemoved(removeMatch);// Remove any queued items from the install queueInstallShortcutReceiver.removeFromInstallQueue(context, removedPackages, mUser);}if (Utilities.ATLEAST_OREO && mOp == OP_ADD) {// Load widgets for the new package. Changes due to app updates are handled through// AppWidgetHost events, this is just to initialize the long-press options.for (int i = 0; i < N; i++) {dataModel.widgetsModel.update(app, new PackageUserKey(packages[i], mUser));}bindUpdatedWidgets(dataModel);}// add startif (FeatureFlags.REMOVE_DRAWER) {bindAllAppsToWorkspace(app, appsList);}// add end}/*** Updates {@param si}'s intent to point to a new ComponentName.* @return Whether the shortcut intent was changed.*/private boolean updateWorkspaceItemIntent(Context context,WorkspaceItemInfo si, String packageName) {// Try to find the best match activity.Intent intent = new PackageManagerHelper(context).getAppLaunchIntent(packageName, mUser);if (intent != null) {si.intent = intent;si.status = WorkspaceItemInfo.DEFAULT;return true;}return false;}// add startprivate void bindAllAppsToWorkspace(LauncherAppState app, AllAppsList mBgAllAppsList){if (mBgAllAppsList.data.size() > 0) {// AppInfoComparator mAppNameComparator = new AppInfoComparator(mApp.getContext());ArrayList<AppInfo> appInfos = new ArrayList<AppInfo>(mBgAllAppsList.data);// Collections.sort(appInfos, mAppNameComparator);ArrayList<Pair<ItemInfo, Object>> installQueue = new ArrayList<>();for (AppInfo info : appInfos) {installQueue.add(Pair.create((ItemInfo) info, null));}app.getModel().addAndBindAddedWorkspaceItems(installQueue);}}// add end
}
9.更新时关键修改如下:
@Override
public void execute(LauncherAppState app, BgDataModel dataModel, AllAppsList appsList) {final Context context = app.getContext();final IconCache iconCache = app.getIconCache();final String[] packages = mPackages;final int N = packages.length;FlagOp flagOp = FlagOp.NO_OP;final HashSet<String> packageSet = new HashSet<>(Arrays.asList(packages));ItemInfoMatcher matcher = ItemInfoMatcher.ofPackages(packageSet, mUser);final HashSet<ComponentName> removedComponents = new HashSet<>();switch (mOp) {case OP_ADD: {for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.addPackage " + packages[i]);String pkg = packages[i];if ("com.android.calendar".equals(pkg)|| "com.android.deskclock".equals(pkg)|| "com.android.dialer".equals(pkg)|| "com.android.browser".equals(pkg)|| "com.android.contacts".equals(pkg)|| "com.android.camera2".equals(pkg)|| "com.android.email".equals(pkg)|| "com.android.calculator2".equals(pkg)|| "com.android.webview".equals(pkg)|| "com.android.gallery3d".equals(pkg)|| "com.android.music".equals(pkg)) {continue;}iconCache.updateIconsForPkg(packages[i], mUser);if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) {appsList.removePackage(packages[i], mUser);}appsList.addPackage(context, packages[i], mUser);// Automatically add homescreen icon for work profile apps for below O device.if (!Utilities.ATLEAST_OREO && !Process.myUserHandle().equals(mUser)) {SessionCommitReceiver.queueAppIconAddition(context, packages[i], mUser);}}flagOp = FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;}case OP_UPDATE:try (SafeCloseable t =appsList.trackRemoves(a -> removedComponents.add(a.componentName))) {for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.updatePackage " + packages[i]);String pkg = packages[i];if ("com.android.calendar".equals(pkg)|| "com.android.deskclock".equals(pkg)|| "com.android.dialer".equals(pkg)|| "com.android.browser".equals(pkg)|| "com.android.contacts".equals(pkg)|| "com.android.camera2".equals(pkg)|| "com.android.email".equals(pkg)|| "com.android.calculator2".equals(pkg)|| "com.android.webview".equals(pkg)|| "com.android.gallery3d".equals(pkg)|| "com.android.music".equals(pkg)) {continue;}iconCache.updateIconsForPkg(packages[i], mUser);appsList.updatePackage(context, packages[i], mUser);app.getWidgetCache().removePackage(packages[i], mUser);}}// Since package was just updated, the target must be available now.flagOp = FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;case OP_REMOVE: {for (int i = 0; i < N; i++) {FileLog.d(TAG, "Removing app icon" + packages[i]);iconCache.removeIconsForPkg(packages[i], mUser);}// Fall through}case OP_UNAVAILABLE:for (int i = 0; i < N; i++) {if (DEBUG) Log.d(TAG, "mAllAppsList.removePackage " + packages[i]);appsList.removePackage(packages[i], mUser);app.getWidgetCache().removePackage(packages[i], mUser);}flagOp = FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE);break;case OP_SUSPEND:case OP_UNSUSPEND:flagOp = mOp == OP_SUSPEND ?FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_SUSPENDED) :FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_SUSPENDED);if (DEBUG) Log.d(TAG, "mAllAppsList.(un)suspend " + N);appsList.updateDisabledFlags(matcher, flagOp);break;case OP_USER_AVAILABILITY_CHANGE: {UserManagerState ums = new UserManagerState();ums.init(UserCache.INSTANCE.get(context),context.getSystemService(UserManager.class));flagOp = ums.isUserQuiet(mUser)? FlagOp.addFlag(WorkspaceItemInfo.FLAG_DISABLED_QUIET_USER): FlagOp.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_QUIET_USER);// We want to update all packages for this user.matcher = ItemInfoMatcher.ofUser(mUser);appsList.updateDisabledFlags(matcher, flagOp);// We are not synchronizing here, as int operations are atomicappsList.setFlags(FLAG_QUIET_MODE_ENABLED, ums.isAnyProfileQuietModeEnabled());break;}}
10.没修改的效果如下:

11.修改后的效果如下:
可以看到修改之后桌面很干净,没有哪些乱七八糟的应用图标,之前是一堆系统应用图标,给人焕然一新的感觉,而且还修改了排列的图标位置和列数等等,这些后面再讲解.

12.总结:
这里大家可以根据自己的需求进行隐藏,我这里只是举例子,因为我有些系统应用图标实际不需要隐藏,所以最后可以看到只有相册和电子邮件,Google商店和Google Chrome浏览器是我自己定制的apk,所以默认就显示在首屏,当然这些也可以去掉隐藏,像第一屏有些是在默认的Launcher3布局里面设置。Android9和11基本上在隐藏图标配置和修改是一样的,大家可以自己尝试,当然后面讲解Launcher3修改默认的抽屉和首屏布局列表,这时候Android9和11有很多不一样的 地方,大家可以自行尝试,今天的内容就到这里打卡睡觉.
相关文章:
AndroidAOSP定制隐藏某个应用的图标
AndroidAOSP定制隐藏某个应用的图标 1.前言: 之前在做AOSP定制的时候需要隐藏某些App的图标,或者默认不显示某个定制的App图标,这样可以让用户感觉不到已经安装了某个App,或者在做系统定制的时候需要修改桌面icon,有些系统的App图标默认不需要显示&…...
最小化重投影误差求解PnP
问题描述 已知n个空间点 P i [ x i , y i , z i ] T P_i[x_i,y_i,z_i]^T Pi[xi,yi,zi]T,其投影的像素坐标 p i [ u i , v i ] T p_i[u_i,v_i]^T pi[ui,vi]T求相机的位姿R,T。 问题分析 根据相机模型,像素点和空间点的位置…...
玩转Docker | 使用Docker部署IT-tools工具箱
玩转Docker | 使用Docker部署IT-tools工具箱 前言一、 IT-tools介绍简介主要特点二、系统要求环境要求环境检查Docker版本检查检查操作系统版本三、部署IT-tools服务下载镜像创建容器检查容器状态检查服务端口安全设置四、访问IT-tools应用五、测试与使用六、总结前言 在信息技…...
unity学习52:UI的最基础组件 rect transform,锚点anchor,支点/轴心点 pivot
目录 1 image 图像:最简单的UI 1.1 图像的基本属性 1.2 rect transform 1.3 image的component: 精灵 → 图片 1.4 修改颜色color 1.5 修改材质 1.6 raycast target 1.7 maskable 可遮罩 1.8 imageType 1.9 native size 原生大小 2 rect transform 2.1 …...
【Python系列】PYTHONUNBUFFERED=1的作用
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...
DeepSeek-R1技术全解析:如何以十分之一成本实现OpenAI级性能?
一、现象级爆火背后的技术逻辑 2025年1月20日,中国AI公司深度求索(DeepSeek)发布新一代大模型R1,其性能直接对标OpenAI的o1版本,但训练成本仅为后者的1/20(600万美元 vs. 1.2亿美元)࿰…...
Linux中的cgdb的基本使用
1.cgdb的简介 Linux中的cgdb是一个基于GDB(GNU Debugger)的图形化调试前端,它结合了GDB的命令行界面功能和代码查看窗口,为开发者提供了一个更为直观的调试体验。 cgdb的作用和功能: 直观调试体验:cgdb提供…...
Qt layout
文章目录 Qt layout**关键机制****验证示例****常见误区****最佳实践****总结**关键点总结:示例代码说明:结论: Qt layout 在 Qt 中,当调用 widget->setLayout(layout) 时,layout 的父对象会被自动设置为该 widget…...
解决idea2019创建springboot项目爆红的问题
通过spring Initializr创建springboot项目时,由于idea版本太低,创建完成后需要手动修改pom.xml,对小白不太友好 一个简便的方法,配置好pom.xml文件的各个版本: 在 https://start.aliyun.com/ 上选择好后复制pom.xml代…...
DeepSeek 提示词:基础结构
🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,精通Java编…...
自动驾驶两个传感器之间的坐标系转换
有两种方式可以实现两个坐标系的转换。 车身坐标系下一个点p_car,需要转换到相机坐标系下,旋转矩阵R_car2Cam,平移矩阵T_car2Cam。点p_car在相机坐标系下记p_cam. 方法1:先旋转再平移 p_cam T_car2Cam * p_car T_car2Cam 需要注…...
[实现Rpc] 客户端 | Requestor | RpcCaller的设计实现
目录 Requestor类的实现 框架 完善 onResponse处理回复 完整代码 RpcCaller类的实现 1. 同步调用 call 2. 异步调用 call 3. 回调调用 call Requestor类的实现 (1)主要功能: 客户端发送请求的功能,进行请求描述对服务器…...
flutter: table calendar笔记
pub dev:table_calendar 3.2.0 我来详细解释 TableCalendar 是如何根据不同的 CalendarFormat 来显示界面的。主要逻辑在 CalendarCore 中实现。 核心逻辑分为以下几个部分: 页面数量计算 - _getPageCount 方法根据不同格式计算总页数: in…...
smolagents学习笔记系列(五)Tools-in-depth-guide
这篇文章锁定官网教程中的 Tools-in-depth-guide 章节,主要介绍了如何详细构造自己的Tools,在之前的博文 smolagents学习笔记系列(二)Agents - Guided tour 中我初步介绍了下如何将一个函数或一个类声明成 smolagents 的工具&…...
axios几种请求类型的格式
Axios 是一个基于 Promise 的 HTTP 客户端,广泛用于浏览器和 Node.js 中发送 HTTP 请求。它支持多种请求格式,包括 GET、POST、PUT、DELETE 等。也叫RESTful 目录 一、axios几种请求类型的格式 1、get请求 2、post请求 3、put请求 4、delete请求 二…...
架构设计系列(六):缓存
一、概述 在应用对外提供服务的时候其稳定性,性能会受到诸多因素的影响。缓存的作用是将频繁访问的数据缓存起来,避免资源重复消耗,提升系统服务的吞吐量。 二、缓存的应用场景 2.1 客户端 HTTP响应可以被浏览器缓存。我们第一次通过HTTP请…...
个人电脑小参数GPT预训练、SFT、RLHF、蒸馏、CoT、Lora过程实践——MiniMind图文版教程
最近看到Github上开源了一个小模型的repo,是真正拉低LLM的学习门槛,让每个人都能从理解每一行代码, 从零开始亲手训练一个极小的语言模型。开源地址: GitHub - jingyaogong/minimind: 🚀🚀 「大模型」2小时…...
MySQL 中的事务隔离级别有哪些?MySQL 默认的事务隔离级别是什么?为什么选择这个级别?数据库的脏读、不可重复读和幻读分别是什么?
MySQL 中的事务隔离级别有哪些? 1. 读未提交(Read Uncommitted) 特点:一个事务可以读取另一个事务未提交的数据。如果一个事务对数据进行了修改但尚未提交,其他事务仍能读取到这些未提交的修改。优缺点: …...
格式工厂 FormatFactory v5.18.便携版 ——多功能媒体文件转换工具
格式工厂 FormatFactory v5.18.便携版 ——多功能媒体文件转换工具 功能:视频 音频 图片 文档PDF格式 各种转换,同格式调整压缩比例,调整大小 特色:果风图标 好看; 支持多任务队列,完成自动关机 下载地址࿱…...
python爬虫学习第十一篇爬取指定类型数据
最近在学习Python爬虫的过程中,尝试用爬虫获取指定类型的数据。今天,我想和大家分享一下我的实践过程和遇到的问题。 一、实现目标 目标是从一个网站的API接口获取不同类型的食品数据。 比如,第一步我想获取汉堡、小食、甜品等不同类型的数…...
Android 实现 RTMP 推流:快速集成指南
简介 在 Android 设备上实现 RTMP 推流,可以用于直播、远程监控等应用场景。本文将基于 rtmp-rtsp-stream-client-java 库,介绍如何在 Android 端快速集成 RTMP 推流,包括权限管理、相机预览、推流控制等关键步骤。 步骤 1. 配置 Maven 仓库 在 settings.gradle.kts 中添…...
KafkaTool
Offset Explorer 第一次打开需要配置kafka相关配置连接 随便先启动一个Kafka(先启动zookeeper) 设置key value 记得刷新...
基于C++“简单且有效”的“数据库连接池”
前言 数据库连接池在开发中应该是很常用的一个组件,他可以很好的节省连接数据库的时间开销;本文基使用C实现了一个简单的数据库连接池,代码量只有400行只有,但是压力测试效果很好;欢迎收藏 关注,本人将会…...
简单易懂,解析Go语言中的struct结构体
目录 4. struct 结构体4.1 初始化4.2 内嵌字段4.3 可见性4.4 方法与函数4.4.1 区别4.4.2 闭包 4.5 Tag 字段标签4.5.1定义4.5.2 Tag规范4.5.3 Tag意义 4. struct 结构体 go的结构体类似于其他语言中的class,主要区别就是go的结构体没有继承这一概念,但可…...
爬虫第九篇-结束爬虫循环
最近在学习Python爬虫的过程中,遇到了一个很有趣的问题:如何优雅地结束爬虫循环?今天,我想和大家分享一下我的发现和心得。 一、爬虫循环结束的常见问题 在写爬虫时,我们经常会遇到这样的情况:当爬取到的…...
国产编辑器EverEdit - 洞察秋毫!文件比较功能!
1 文件比较 1.1 应用场景 项目开发过程中,可能不同的部分会由不同的人在负责,存在一个文件多人编辑的情况,用户需要寻找差异,并将文档进行合并,比较专业的文本比较工具为BeyondCompare,WinMerge等。 如…...
QARepVGG--含demo实现
文章目录 前言引入Demo实现总结 前言 在上一篇博文RepVGG中,介绍了RepVGG网络。RepVGG 作为一种高效的重参数化网络,通过训练时的多分支结构(3x3卷积、1x1卷积、恒等映射)和推理时的单分支合并,在精度与速度间取得了优…...
五、 Spring Framework基础:Spring Data JPA基本用法与 Repository 接口
深入解析 Spring Data JPA:基本用法与 Repository 接口 Spring Data JPA 是 Spring 框架中用于简化数据访问层开发的核心模块。它基于 JPA 规范,底层使用 Hibernate 实现,通过接口继承和方法命名规则,自动实现增删改查等常见操作…...
如何实现在Redis集群情况下,同一类数据固定保存在同一个Redis实例中
1. 使用哈希标签(Hash Tags) 概述 Redis Cluster使用一致性哈希算法来分配数据到不同的节点上。为了确保相同类型的数据被分配到同一个Redis实例上,可以利用哈希标签(Hash Tags)。哈希标签是指在键名中用花括号 {} 包…...
kotlin 知识点 七 泛型的高级特性
对泛型进行实化 泛型实化这个功能对于绝大多数Java 程序员来讲是非常陌生的,因为Java 中完全没有这个概 念。而如果我们想要深刻地理解泛型实化,就要先解释一下Java 的泛型擦除机制才行。 在JDK 1.5之前,Java 是没有泛型功能的,…...
