# Launcher90 新安裝應(yīng)用title 提示---小圓點(diǎn)
###? ? 前言:
? 縱觀市面上的華為绅项、小米产上、vivo手機(jī) 在桌面或者文件夾中有新安裝應(yīng)用時(shí)會(huì)在title左邊顯示一個(gè)小圓點(diǎn)(顏色主流為綠色)從而提示用戶此應(yīng)用為新安裝的app,可能有以下原因吧: 有一定的導(dǎo)流作用 促使用戶去打開此新安裝的應(yīng)用凶朗,從UI的角度 頁(yè)面較為美觀吧远荠。
? 先看下效果圖:
### 方案和問題:
在sugar版本上可以看到實(shí)現(xiàn)方案為在icon的右上角繪制一個(gè)小圓點(diǎn) 表示新安裝應(yīng)用砸抛,在Icon上面繪制圓點(diǎn)好處為厅翔,iconsize 相同乖坠,畫的小圓點(diǎn)類似角標(biāo)可以準(zhǔn)確確定位置 基本適用于所有應(yīng)用。不會(huì)有位置偏移變化刀闷。
現(xiàn)在改為在新安裝的應(yīng)用title前面添加小圓點(diǎn) 提示熊泵,由于每個(gè)應(yīng)用title長(zhǎng)短不一,需要?jiǎng)討B(tài)計(jì)算title長(zhǎng)度 使小圓點(diǎn)等距離的美觀顯示在桌面上甸昏⊥绶郑看了小米的顯示效果為 小圓點(diǎn)距離title的第一個(gè)字距離都相同,這樣才美觀施蜜。那么在draw 這個(gè)小圓點(diǎn)時(shí) 就需要?jiǎng)討B(tài)計(jì)算title首字母的坐標(biāo)了卒蘸,這樣才能根據(jù)不同的title動(dòng)態(tài)等距離的顯示小圓點(diǎn),美觀的展示給用戶翻默。
先看下所有APP 調(diào)試效果圖缸沃。
會(huì)根據(jù)每個(gè)應(yīng)用的title前面顯示一個(gè)小圓點(diǎn),距離需要等寬修械。
### 具體實(shí)現(xiàn)方案:
在新安裝APP的title顯示小圓點(diǎn)在BubbleTextView.java 類中實(shí)現(xiàn)趾牧,需要在桌面添加創(chuàng)建此icon shortcut時(shí) 增加判斷是否是新安裝應(yīng)用。
首先看下shortcut 實(shí)現(xiàn)肯污,
代碼位置:Z:\work\Launcher90\src\com\android\launcher3\BubbleTextView.java
由于桌面文件夾和deep shortcut hotset都是繼承BubbleTextView.java 所以在創(chuàng)建是需增加判斷過濾翘单。
protected void drawRemindIcon(Canvas canvas) {
? ? ? ? if (getTag() != null && getTag() instanceof ShortcutInfo) {
? ? ? ? ? ? ShortcutInfo info = (ShortcutInfo) getTag();
? ? ? ? ? ? DeviceProfile grid = mActivity.getDeviceProfile();
? ? ? ? ? ? int paddingTop = mIconSize;
? ? ? ? ? ? int paddingleft = mOffsetRemindLeft;
? ? ? ? ? ? int position = 0;
? ? ? ? ? ? Layout layout = getLayout();
? ? ? ? ? ? Rect bound = new Rect();
? ? ? ? ? ? int line = layout.getLineForOffset(position);
? ? ? ? ? ? layout.getLineBounds(line, bound);
? ? ? ? ? ? int yAxisTop = bound.top;//字符頂部y坐標(biāo)
? ? ? ? ? ? int yAxisBottom = bound.bottom;//字符底部y坐標(biāo)
? ? ? ? ? ? float xAxisLeft = layout.getPrimaryHorizontal(position);//字符左邊x坐標(biāo)
? ? ? ? ? ? float xAxisRight = layout.getSecondaryHorizontal(position);//字符右邊x坐標(biāo)
? ? ? ? ? ? if (mDisplay == DISPLAY_WORKSPACE) {
? ? ? ? ? ? ? ? paddingleft = (int) xAxisLeft;
? ? ? ? ? ? ? ? paddingTop = mIconSize + (yAxisBottom - yAxisTop) / 2 + 2 * grid.iconDrawablePaddingPx;
? ? ? ? ? ? } else if (mDisplay == DISPLAY_FOLDER) {
? ? ? ? ? ? ? ? paddingleft = (int) xAxisLeft - mOffsetRemindLeft;
? ? ? ? ? ? ? ? paddingTop = mIconSize + (yAxisBottom - yAxisTop) / 4 + grid.iconDrawablePaddingPx/2;
? ? ? ? ? ? } else if (mDisplay == DISPLAY_SHORTCUT) {
? ? ? ? ? ? ? ? isShowRemind = false;
? ? ? ? ? ? } else if (mDisplay == DISPLAY_WIDGET) {
? ? ? ? ? ? ? ? isShowRemind = false;
? ? ? ? ? ? }
? ? ? ? ? ? LogUtil.d(TAG, "yAxisTop:" + yAxisTop + "yAxisBottom" + yAxisBottom + "xAxisLeft:" + xAxisLeft + "xAxisRight" + xAxisRight + "grid.iconDrawablePaddingPx" + grid.iconDrawablePaddingPx);
? ? ? ? ? ? if (info.remind && isShowRemind && shouldTextBeVisible()) {
? ? ? ? ? ? ? ? canvas.save();
? ? ? ? ? ? ? ? canvas.drawBitmap(LauncherAppState.getInstance(mContext).getBitmapAndDrawableCache().getRemindIcon(), paddingleft, paddingTop, null);
? ? ? ? ? ? ? ? canvas.restore();
? ? ? ? ? ? }
? ? ? ? }
? ? }
? ? 此段會(huì)增加判斷只在桌面快捷方式 文件夾內(nèi)部顯示小圓點(diǎn) ,文件夾前面的需要后面特殊處理蹦渣。
? ? 主要用到 Layout layout = getLayout();
? ? ? ? ? ? Rect bound = new Rect();
? ? ? ? ? ? int line = layout.getLineForOffset(position);
? ? ? ? ? ? layout.getLineBounds(line, bound);
? ? ? ? ? ? int yAxisTop = bound.top;//字符頂部y坐標(biāo)
? ? ? ? ? ? int yAxisBottom = bound.bottom;//字符底部y坐標(biāo)
? ? ? ? ? ? float xAxisLeft = layout.getPrimaryHorizontal(position);//字符左邊x坐標(biāo)
? ? ? ? ? ? float xAxisRight = layout.getSecondaryHorizontal(position);//字符右邊x坐標(biāo)
可以動(dòng)態(tài)計(jì)算title最前面字符的左邊哄芜,有了它就可以canvas.drawBitmap 了。
快捷方式實(shí)現(xiàn)了柬唯。
接下來看下文件夾的實(shí)現(xiàn)方式认臊,文件夾需要特殊處理,要根據(jù)folder 內(nèi)部是否有新安裝應(yīng)用锄奢,如果folder內(nèi)部有新安裝應(yīng)用失晴,在打開文件夾是顯示在應(yīng)用前面冤议,當(dāng)關(guān)閉文件夾時(shí)會(huì)顯示子文件夾title的前面(由于文件夾name 是可以自己定義的需要?jiǎng)討B(tài)計(jì)算顯示在前面);
? ? 先看下調(diào)試效果师坎。
? ? 源碼位置:
? ? Z:\work\Launcher90\src\com\android\launcher3\folder\FolderIcon.java
? ? 在dispatchDraw(Canvas canvas)方法中循環(huán)遍歷下folder內(nèi)部是否有新安裝應(yīng)用。
? ? ? int count = mInfo.contents.size();
? ? ? ? boolean needremind = false;
? ? ? ? for (int i=0; i<count; i++) {
? ? ? ? ? ? ShortcutInfo info = mInfo.contents.get(i);
? ? ? ? ? ? if (CustomConfigManager.getInstance().supportFolderIconUnread) {
? ? ? ? ? ? ? ? if (info.unreadNum > 0) {
? ? ? ? ? ? ? ? ? ? needremind = false;
? ? ? ? ? ? ? ? ? ? break;
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }
? ? ? ? ? ? if (info.remind) {
? ? ? ? ? ? ? ? needremind = true;
? ? ? ? ? ? ? ? if (CustomConfigManager.getInstance().supportFolderIconUnread) {
? ? ? ? ? ? ? ? ? ? continue;
? ? ? ? ? ? ? ? }else {
? ? ? ? ? ? ? ? ? ? break;
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? if (needremind) {
? ? ? ? ? ? drawFolderRemind(canvas);
? ? ? ? }
? ? ? ? private void drawFolderRemind(Canvas canvas) {
? ? ? ? canvas.save();
? ? ? ? int textureWidth = this.getWidth();
? ? ? ? int paddingTop = this.getPaddingTop();
? ? ? ? int paddingleft = mOffsetRemindLeft;
? ? ? ? DeviceProfile grid = mLauncher.getDeviceProfile();
? ? ? ? DisplayMetrics dm = mLauncher.getResources().getDisplayMetrics();
? ? ? ? int position = 0;
? ? ? ? if (mFolderName == null) {
? ? ? ? ? ? return;
? ? ? ? }
? ? ? ? Layout layout = mFolderName.getLayout();
? ? ? ? Rect bound = new Rect();
? ? ? ? int line = layout.getLineForOffset(position);
? ? ? ? layout.getLineBounds(line, bound);
? ? ? ? int yAxisTop = bound.top;//字符頂部y坐標(biāo)
? ? ? ? int xAxisCenter = bound.centerX();//字符頂部y坐標(biāo)
? ? ? ? int yAxisBottom = bound.bottom;//字符底部y坐標(biāo)
? ? ? ? float xAxisLeft = layout.getPrimaryHorizontal(position);//字符左邊x坐標(biāo)
? ? ? ? float xAxisRight = layout.getSecondaryHorizontal(position);//字符右邊x坐標(biāo)
? ? ? ? paddingTop = grid.iconSizePx + (yAxisBottom - yAxisTop) / 2 + 2 * grid.iconDrawablePaddingPx;
? ? ? ? paddingleft = textureWidth/2 - (xAxisCenter - (int) xAxisLeft)-mOffsetRemindLeft;
? ? ? ? canvas.drawBitmap(LauncherAppState.getInstance(mLauncher).getBitmapAndDrawableCache().getRemindIcon(), paddingleft, paddingTop, null);
? ? ? ? canvas.restore();
? ? }
? ? 和shortcut 創(chuàng)建流程基本類似 需要計(jì)算文件夾name左邊堪滨,在調(diào)試過程中 發(fā)現(xiàn) 文件夾的name首字母坐標(biāo)不是left值 需要this.getWidth()/2 減去 name中心坐標(biāo)和首字母坐標(biāo)胯陋,會(huì)隨首字母坐標(biāo)變left變小.
? ? 到此顯示布局邏輯基本實(shí)現(xiàn)了接下來需要控制判斷哪些是新安裝的應(yīng)用才顯示。
### 如何判斷新安裝應(yīng)用:
源碼位置:
Z:\work\Launcher90\src\com\android\launcher3\Workspace.java
Z:\work\Launcher90\src\com\ape\launcher3\MyosLauncher.java
本次實(shí)現(xiàn)方案沒有保存在數(shù)據(jù)庫(kù)中袱箱,在監(jiān)聽新安裝應(yīng)用時(shí) 保存此應(yīng)用的包名到sp,
然后再luncher onresume 或者
public void finishBindingItems() 分別刷新下遏乔。
? public void refreshRemindIcon() {
? ? ? ? LogUtil.d(TAG, "refreshRemindIcon()");
? ? ? ? Set<String> set = DownloadRemind.getAllReminValue(MyosLauncher.this);
? ? ? ? final HashMap<String, Integer> remindMap = new HashMap<>();
? ? ? ? LogUtil.d(TAG, "remindMap");
? ? ? ? for (String s : set) {
? ? ? ? ? ? int state = DownloadRemind.isRemindApp(MyosLauncher.this, s) ? 0 : 1;
? ? ? ? ? ? LogUtil.d(TAG, "refreshRemindIcon : packageName = " + s + ", state = " + state);
? ? ? ? ? ? remindMap.put(s, state);
? ? ? ? }
? ? ? ? mHandler.post(new Runnable() {
? ? ? ? ? ? public void run() {
? ? ? ? ? ? ? ? if (mWorkspace != null) {
? ? ? ? ? ? ? ? ? ? mWorkspace.updateShortcutsAndFoldersRemind(remindMap);
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }
? ? ? ? });
? ? }
刷新遍歷 ShortcutInfo 、FolderInfo发笔、Hotseat是否包含在sp中的包名remind info 信息盟萨。
public void updateShortcutsAndFoldersRemind(HashMap<String, Integer> remindMap) {
? ? ? ? final ArrayList<ShortcutAndWidgetContainer> childrenLayouts = getAllShortcutAndWidgetContainers();
? ? ? ? int childCount = 0;
? ? ? ? View view = null;
? ? ? ? Object tag = null;
? ? ? ? for (ShortcutAndWidgetContainer layout : childrenLayouts) {
? ? ? ? ? ? childCount = layout.getChildCount();
? ? ? ? ? ? for (int j = 0; j < childCount; j++) {
? ? ? ? ? ? ? ? view = layout.getChildAt(j);
? ? ? ? ? ? ? ? tag = view.getTag();
? ? ? ? ? ? ? ? if (tag instanceof ShortcutInfo) {
? ? ? ? ? ? ? ? ? ? final ShortcutInfo info = (ShortcutInfo) tag;
? ? ? ? ? ? ? ? ? ? final Intent intent = info.intent;
? ? ? ? ? ? ? ? ? ? final ComponentName componentName = intent.getComponent();
? ? ? ? ? ? ? ? ? ? if (componentName != null && componentName.getPackageName() != null
? ? ? ? ? ? ? ? ? ? ? ? ? ? && remindMap.containsKey(componentName.getPackageName())) {
? ? ? ? ? ? ? ? ? ? ? ? ////this is remind app
? ? ? ? ? ? ? ? ? ? ? ? if (remindMap.get(componentName.getPackageName()) == 0) {
? ? ? ? ? ? ? ? ? ? ? ? ? ? info.remind = true;
? ? ? ? ? ? ? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? ? ? ? ? ? ? info.remind = false;
? ? ? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? ? ? ((BubbleTextView)view).invalidate();
? ? ? ? ? ? ? ? ? ? //((BubbleTextView) view).mFavorite.invalidate();
? ? ? ? ? ? ? ? } else if (tag instanceof FolderInfo) {
? ? ? ? ? ? ? ? ? ? ((FolderIcon) view).updateFolderRemind(remindMap);
? ? ? ? ? ? ? ? ? ? ((FolderIcon) view).invalidate();
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? mLauncher.getHotseat().updateRemindApps(remindMap);
? ? }
### title圓點(diǎn)如何消失:
? 目前處理 判別應(yīng)用是否進(jìn)入新應(yīng)用,為從桌面點(diǎn)擊 startactivity 即認(rèn)為不是新應(yīng)用了需要移除標(biāo)志位了讨。
? public boolean startActivitySafely(View v, Intent intent, ItemInfo item) {
? ? ? ? boolean success = super.startActivitySafely(v, intent, item);
? ? ? ? if (success && v instanceof BubbleTextView) {
? ? ? ? ? ? // This is set to the view that launched the activity that navigated the user away
? ? ? ? ? ? // from launcher. Since there is no callback for when the activity has finished
? ? ? ? ? ? // launching, enable the press state and keep this reference to reset the press
? ? ? ? ? ? // state when we return to launcher.
? ? ? ? ? ? BubbleTextView btv = (BubbleTextView) v;
? ? ? ? ? ? btv.setStayPressed(true);
? ? ? ? ? ? setOnResumeCallback(btv);
? ? ? ? ? ? if (DownloadRemind.isRemindApp(this, intent.getComponent().getPackageName())) {
? ? ? ? ? ? ? ? DownloadRemind.setDownloadAppsRemind(this, intent.getComponent().getPackageName(), false);
? ? ? ? ? ? }
? ? ? ? ? ? LogUtil.d(TAG, "startActivitySafely success:" + success + "PackageName:" + intent.getComponent().getPackageName());
? ? ? ? }
? ? ? ? return success;
? ? }
? 設(shè)置對(duì)應(yīng)的包名 value 為false;
? 以上為新安裝應(yīng)用的title 小圓點(diǎn)draw位置捻激、顯示、和消失基本邏輯 方案前计。方法很多如有好的更優(yōu)的方法 歡迎指正胞谭。