翻譯Android Weekly--No, You Can Not Override the Home Button... But You Don't Have To!
有關Home鍵
每個Android開發(fā)者都曾經(jīng)問個這問題虾攻,“我能不能覆蓋Home鍵楼雹?”。答曰:
當然不行琳要!處理返回按鈕已經(jīng)足夠艱難了闲礼,如果Home鍵能覆蓋那就更是一團糟牍汹。
真相如下:
- 如果Home鍵沒有返回桌面琅翻,用戶會很失望
- 用戶會困在app中
- 覆蓋不了所有情況
1.由于來電而退出app的情況;
2.掛電話的情況柑贞;
3.點擊切換app按鈕的情況。
所以聂抢,你很可能是想知道用戶什么時候離開你的應用的钧嘶,而不是把用戶困在應用中。如果Application
中有onStop()
方法就相當簡單啦琳疏,對不對有决?
Application
中遺失的onStart()
和onStop()
想一想,為什么在onStop()
方法中停止空盼?讓我們深入了解前臺/后臺
(forground/background
)的生命周期书幕,當狀態(tài)發(fā)生改變時,能了然于胸揽趾。
為什么需要知道當前的狀態(tài)呢台汇?假設我們的應用會收到一條通知。如果當時應用處于前臺運行篱瞎,肯定會展示出應用內相應的通知界面苟呐。相反,如果應用處于后臺運行俐筋,那在通知欄進行展示就比較合適牵素。
例如其他情況,你想知道應用從前臺運行到切換至后臺的會話長短澄者,又或者當用戶需要處理其他事務離開時笆呆,你需要把你應用的緩存清空。
謝天謝地粱挡,你可以使用可靠的方式獲取這些信息赠幕,而不是使用令人抓狂的ActivityManager.getRunningTask,也不是令人蛋疼的Activity
的生命周期(onStop()
總是不緊跟onStart()
調用)。
應用切換至后臺
從API 14
(Android 4.0 ICS
)抱怔,我們可以調用Application.onTrimMemory(int level)劣坊,這個方法包含了一個等級叫TRIM_MEMORY_UI_HIDDEN,用于記錄應用即將進入后臺運行屈留。
下面是一個自定義Applicaiton
的使用
public class MyApplication extends Application
{
@Override
public void onTrimMemory(int level) {
super.onTrimMemory(level);
if (level == TRIM_MEMORY_UI_HIDDEN) {
isBackground = true;
notifyBackground();
}
}
}
啊哈局冰!這樣就能知道應用切換至后臺運行啦。
手機熄屏
Application.onTrimMemory(int level)
在手機熄屏時不回調怎么辦灌危?用Intent.ACTION_SCREEN_OFF
注冊BroadcastReceiver
public class MyApplication extends Application {
// ...
@Override
public void onCreate() {
super.onCreate();
// ...
IntentFilter screenOffFilter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
registerReceiver(new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
}, screenOffFilter);
}
}
注意:無需監(jiān)聽屏幕點亮的動作康二,下面會全部搞掂。
應用切換至前臺
沒有任何flag
或者trim level
來判斷應用切換至前臺勇蝙,覆寫Activity.onResume()
是最好的方法沫勿。在基類Activity
中復寫它是一個選擇,但無須如此。
一個更簡潔的做法是产雹,利用Application.registerActivityLifeStyleCallbacks()诫惭,如名字描述一樣字币,可以覆寫每一個生命周期函數(shù)抛蚁。在這個例子中糠排,在不侵入式改動每個Activity
的代碼的前提下肥缔,在Activity.onResume()
中執(zhí)行了代碼冗荸。
下面是一個自定義Application
:
public class MyApplication extends Application {
// ...
@Override
public void onCreate() {
super.onCreate();
registerActivityLifecycleCallbacks(new Application.ActivityLifecycleCallbacks() {
// ...
@Override
public void onActivityResumed(Activity activity) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
// ...
});
}
// ...
}
組織起來
下面是一個應用前后臺切換的完整例子恢总。
public class MyApplication extends Application {
// Starts as true in order to be notified on first launch
private boolean isBackground = true;
@Override
public void onCreate() {
super.onCreate();
listenForForeground();
listenForScreenTurningOff();
}
private void listenForForeground() {
registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {
//...
@Override
public void onActivityResumed(Activity activity) {
if (isBackground) {
isBackground = false;
notifyForeground();
}
}
//...
});
}
private void listenForScreenTurningOff() {
IntentFilter screenStateFilter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
registerReceiver(new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
isBackground = true;
notifyBackground();
}
}, screenStateFilter);
}
@Override
public void onTrimMemory(int level) {
super.onTrimMemory(level);
if (level == TRIM_MEMORY_UI_HIDDEN) {
isBackground = true;
notifyBackground();
}
}
private void notifyForeground() {
// This is where you can notify listeners, handle session tracking, etc
}
private void notifyBackground() {
// This is where you can notify listeners, handle session tracking, etc
}
public boolean isBackground() {
return isBackground;
}
}
總結
- API 14及以上
- 用
Application.onTrimLevel(int level)
和TRIM_MEMORY_UI_HIDDEN
判斷應用是否切換至后臺運行探膊。 - 通過
INTENT.ACTION_SCREEN_OFF
注冊廣播接受器監(jiān)聽屏幕熄滅 - 注冊
Activity.registerLifeStyleCallback
監(jiān)聽應用切換至前臺運行 - 別奢望覆寫
Home鍵
- 做個好人