Android Launcher 界面添加未接來電 / 未讀短信條數顯示

原文首發地址 Android Launcher 界面添加未接來電 / 未讀短信條數顯示

本文代碼基于 Android 6.0(高通),原生不支持未接來電以及未讀短信的圖標右上角數目顯示。

涉及到幾個文件:

  1. packages/apps/Launcher3/src/com/android/launcher3/Launcher.java
  2. packages/apps/Launcher3/src/com/android/launcher3/Utilities.java
  3. packages/apps/Launcher3/AndroidManifest.xml

具體的思路:Launcher 中注冊 ContentObserver 來監聽短信和電話數據庫,當數據發生變化時,讀取讀取數據庫未讀數目之后通過 ICON_NAME ,重新繪制短信,電話的圖標(在原圖右上角畫圓和數字)。


具體代碼:

1. Launcher.java

  • 首先是定義兩個 ContentObserver
private final static int UPDATE_MMS_ICON = 826;  
private final static int UPDATE_CALL_ICON = 1206;

public class SMSContentObserver extends ContentObserver {
    private Handler mHandler;

    public SMSContentObserver(Context context, Handler handler) {
        super(handler);
        mHandler = handler;
    }

    @Override
    public void onChange(boolean selfChange) {
        Log.i("Launcher-","SMSContentObserver onChange");
        mHandler.removeMessages(UPDATE_MMS_ICON);
        Message msg = mHandler.obtainMessage(UPDATE_MMS_ICON);
        msg.obj = getMissMmsCount();
        mHandler.sendMessage(msg);
    }
}

public class CallContentObserver extends ContentObserver {
    private Handler mHandler;
    public CallContentObserver(Context context, Handler handler) {
        super(handler);
        mHandler = handler;
    }

    @Override
    public void onChange(boolean selfChange) {
        Log.i("Launcher-","CallContentObserver onChange");
        mHandler.removeMessages(UPDATE_CALL_ICON);
        Message msg = mHandler.obtainMessage(UPDATE_CALL_ICON);
        msg.obj = getMissCallCount();
        mHandler.sendMessage(msg);
    }
}
  • 在 onCreate() 中注冊 ContentObserver
private SMSContentObserver smsContentObserver = null;  
private CallContentObserver callContentObserver = null;  

smsContentObserver = new SMSContentObserver(this,mHandler);
callContentObserver =new CallContentObserver(this,mHandler);

getContentResolver().registerContentObserver(Calls.CONTENT_URI,true,callContentObserver);
getContentResolver().registerContentObserver(Uri.parse("content://mms-sms/"),true,smsContentObserver);
  • 定義兩個 ICON_NAME
// 這兩個 ICON_NAME 根據自己實際系統短信和電話頁面對應包名填寫
private final static String PHONE_ICON_NAME = "com.android.dialer.DialtactsActivity";
private final static String MMS_ICON_NAME = "com.android.messaging.ui.conversationlist.ConversationListActivity";
  • mHandler 中處理
@Thunk
final Handler mHandler = new Handler(new Handler.Callback() {
    @Override
    public boolean handleMessage(Message msg) {
        Log.i("Launcher-","mHandler msg.what = " + msg.what);
        if (msg.what == ADVANCE_MSG) {
            int i = 0;
            for (View key : mWidgetsToAdvance.keySet()) {
                final View v = key.findViewById(mWidgetsToAdvance.get(key).autoAdvanceViewId);
                final int delay = mAdvanceStagger * i;
                if (v instanceof Advanceable) {
                    mHandler.postDelayed(new Runnable() {
                        public void run() {
                            ((Advanceable) v).advance();
                        }
                    }, delay);
                }
                i++;
            }
            sendAdvanceMessage(mAdvanceInterval);
        }
        //ADD BY Bruce Yang FOR SHOW UNREAD MMS
        else if (msg.what == UPDATE_MMS_ICON) {
            setMmsOrPhoneNum(MMS_ICON_NAME, getMissMmsCount());
        } else if (msg.what == UPDATE_CALL_ICON) {
            setMmsOrPhoneNum(PHONE_ICON_NAME, getMissCallCount());
        }
        return true;
    }
});
  • 獲取數據庫中未讀數目
 private int getMissMmsCount() {
    Log.i("Launcher-","getMissMmsCount");
    int missSmsCount = 0;
    Cursor cursorSMS = null;
    Cursor cursorMMS = null;
    try {
        cursorSMS = getContentResolver().query(
                Uri.parse("content://sms"), null, "(read=0 and type=1)",
                null, null);
        cursorMMS = getContentResolver().query(
                Uri.parse("content://mms"), null, "(read=0)", null,
                null);
    } catch (SQLiteException e) {
        return missSmsCount;
    }
    if (cursorSMS != null) {
        missSmsCount = cursorSMS.getCount();
        cursorSMS.close();
    }
    if (cursorMMS != null) {
            missSmsCount = missSmsCount + cursorMMS.getCount();
        cursorMMS.close();
    }

    Log.i("Launcher-","getMissMmsCount  missSmsCount = " + missSmsCount);
    return missSmsCount;
}

private int getMissCallCount() {
    Log.i("Launcher-","getMissCallCount");
    int missCallCount = 0;
    Uri missingCallUri = Calls.CONTENT_URI;
    String where = Calls.TYPE + "='" + Calls.MISSED_TYPE + "'"
            + " AND new=1";
    Cursor cursorCall = null;
    try {
        cursorCall = getContentResolver().query(missingCallUri,
                null, where, null, null);
    } catch (SQLiteException e) {
        return missCallCount;
    }

    if (cursorCall != null) {
        missCallCount = cursorCall.getCount();
        cursorCall.close();
    }
    Log.i("Launcher-","getMissCallCount  missCallCount = " + missCallCount);
    return missCallCount;
}
  • 設置 ICON 未讀數
/**
    *
    * @param flag 更新電話或短信 ICON
    * @param missCount 未讀數
*/
private void setMmsOrPhoneNum(final String flag, final int missCount) {
    Log.i("Launcher-","flag = "+flag +" missCount = "+missCount);
    if(mWorkspace == null) return;
    ArrayList<CellLayout> cellLayouts = mWorkspace.getWorkspaceAndHotseatCellLayouts();
    for (final CellLayout layoutParent: cellLayouts) {
        final ViewGroup shortcutAndWidgetContainer = layoutParent.getShortcutsAndWidgets();

        mWorkspace.post(new Runnable() {
            public void run() {

                int childCount = shortcutAndWidgetContainer.getChildCount();
                for (int j = 0; j <childCount; j++) {
                    View view = shortcutAndWidgetContainer.getChildAt(j);

                    Object tag = view.getTag();
                    if (tag instanceof ShortcutInfo) {
                        final ShortcutInfo info = (ShortcutInfo) tag;
                        final Intent intent = info.intent;

                        if (intent != null) {
                            final ComponentName name = intent.getComponent();
                            if (name != null && name.getClassName().equals(flag)) {
                                BubbleTextView bv = (BubbleTextView) view;
                                Bitmap defaultIconBitmap = Bitmap.createBitmap(info.getIcon(mIconCache));
                                Bitmap bitmap = Utilities.createIconBitmap(defaultIconBitmap, missCount);
                                bv.setCompoundDrawablesWithIntrinsicBounds(null,
                                        new FastBitmapDrawable(bitmap),
                                        null, null);
                            }
                        }
                    }
                }
            }
        });
    }
}
  • 第一次啟動 Launcher 就能獲取未讀數目,在 finishBindingItems() 添加邏輯
//ADD BY Bruce Yang
int missCall = getMissCallCount();
int missMms = getMissMmsCount();
if(missCall != 0) {
    setMmsOrPhoneNum(PHONE_ICON_NAME, missCall);
}
if(missMms != 0) {
    setMmsOrPhoneNum(MMS_ICON_NAME, missMms);
}
  • 在 onDestroy() 中反注冊 ContentObserver
getContentResolver().unregisterContentObserver(smsContentObserver);
getContentResolver().unregisterContentObserver(callContentObserver);

2. Utilities.java

新增構造方法,用于重新繪制帶數字的應用圖標。

//add by Bruce Yang for ...
static Bitmap createIconBitmap(Bitmap b, int count) {
    Bitmap bitmap = b.copy(Bitmap.Config.ARGB_8888,true);
    Log.i("Launcher-","b.isMutable() = "+b.isMutable()); // 如果為 false 就會拋出 java.lang.IllegalStateException 異常, http://bbs.csdn.net/topics/370021698
    if (count == 0) return b;
    int textureWidth = bitmap.getWidth();
    final Canvas canvas = sCanvas;
    Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.DEV_KERN_TEXT_FLAG);
    canvas.setBitmap(bitmap);

    paint.setColor(Color.RED);
    canvas.drawCircle(textureWidth - 17-6, 16+6, 16+6, paint);
    paint.setColor(Color.WHITE);
    paint.setStyle(Paint.Style.STROKE);
    paint.setStrokeWidth(2);
    canvas.drawCircle(textureWidth - 17-6, 16+6, 16+6, paint);

    Paint countPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.DEV_KERN_TEXT_FLAG);
    countPaint.setColor(Color.WHITE);
    countPaint.setTextSize(26f);
    countPaint.setTypeface(Typeface.DEFAULT_BOLD);

    float x = textureWidth - 24-4;
    if (count > 9) x -= 4+6;

    if (count > 99) {
        countPaint.setTextSize(22f);
        String text = String.valueOf(99) + "+";
        canvas.drawText(text, x-2, 25+5, countPaint);
    } else {
        String text = String.valueOf(count);
        canvas.drawText(text,x, 25+5, countPaint);
    }
    return bitmap;
}

3. AndroidManifest.xml

添加如下兩個權限,由于 Android 6.0 以上需要動態權限申請,這里為了直接獲取權限,可以將 targetSdkVersion 改成 21(原來是 23)。

<uses-permission android:name="android.permission.READ_SMS" />
<uses-permission android:name="android.permission.READ_CALL_LOG" />

4. 修改對比

修改前:

image

修改后:

image

參考文章:

http://blog.csdn.net/chenxiong668/article/details/12851357
http://blog.csdn.net/kerancsdn/article/details/26705767

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,333評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,491評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,263評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,946評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,708評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,409評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,939評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,774評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,209評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,641評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,872評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,650評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內容