今天逛Android官方文檔的時候發(fā)現(xiàn)了一個比較簡單也有點意思的東西。自定義返回導(dǎo)航。這篇文章就這個簡單說一下:
自定義返回導(dǎo)航是啥射赛;
怎么用咪辱;
簡單場景犬辰;
看一下源碼楚殿;
什么是自定義返回導(dǎo)航
簡單點來說就是可以對返回進行攔截了。有人就會說了:之前也可以攔截呀C取!
對喝滞,以前對后退做攔截的的方式基本上是對onKeyDown
方法做攔截(也可以對onBackPressed
做攔截)阁将,但onKeyDown
職能比較多,包含了很多物理按鍵囤躁,現(xiàn)在android官方將返回拿出來單獨處理肯定會好一點的(原理就是對onBackPressed
做攔截)冀痕。
是這樣的
通過activity的getOnBackPressedDispatcher()
方法你可以得到一個OnBackPressedDispatcher
,看名字就知道狸演,這是用來調(diào)度返回用的言蛇,而我們可以通過它的addCallback
方法將我們想要的返回邏輯加給他,就像這樣:
val callback = onBackPressedDispatcher.addCallback(this,true){
//back code
}
這是kotlin對addCallback
優(yōu)化的一個擴展函數(shù)宵距,如果是java的話腊尚,是這樣的:
OnBackPressedCallback callback = new OnBackPressedCallback(true) {//true 開啟自定義返回邏輯
@Override
public void handleOnBackPressed() {
//back code
}
};
getOnBackPressedDispatcher().addCallback(callback);
然后我們可以通過callback的setEnabled
方法對這個自定義返回邏輯開啟和關(guān)閉。callback.setEnabled(false);
關(guān)閉返回攔截满哪,最主要的功能就這么多婿斥。
一個實現(xiàn)場景
我使用自定義返回導(dǎo)航改了一下之前寫的后退提示劝篷,通常在退出最后一個activity的時候提醒一下用戶:再點擊就退出這個app啦
這樣的需求。改后對比了一下簡潔了好多(也因為是kotlin):
fun AppCompatActivity.exit(message: String = "再按一次返回鍵退出", duration: Int = Toast.LENGTH_SHORT) {
onBackPressedDispatcher.addCallback(this, true) {
val toast = Toast.makeText(this@exit, message, duration)
toast.show()
isEnabled = false
toast.view.postDelayed(
Runnable {
isEnabled = true
},
if (toast.duration == Toast.LENGTH_SHORT) 4000 else 7000)
}
}
最后看一下源碼
我們除了要知道怎么用之外民宿,最好要知道怎么實現(xiàn)的娇妓,這樣知識才是完整的。而且有關(guān)自定義返回導(dǎo)航的代碼也不多活鹰,完完全全加起來還不夠一千行應(yīng)該哈恰。
從androidX
開始,在activity中(準(zhǔn)確來說是在ComponentActivity這一層)后退部分的邏輯變了志群,變成這樣了
//ComponentActivity
public void onBackPressed() {
mOnBackPressedDispatcher.onBackPressed();
}
后退的邏輯全都交給了后退調(diào)度器了着绷。
點進去一看:
//OnBackPressedDispatcher
public void onBackPressed() {
Iterator<OnBackPressedCallback> iterator =
mOnBackPressedCallbacks.descendingIterator();//得到所有注冊到調(diào)度器中的各種后退邏輯
while (iterator.hasNext()) {
OnBackPressedCallback callback = iterator.next();
if (callback.isEnabled()) {//如果callback是可用的話
callback.handleOnBackPressed();//執(zhí)行自定義后退邏輯
return;
}
}
if (mFallbackOnBackPressed != null) {//如果沒有執(zhí)行任何自定義后退邏輯,那么就執(zhí)行一下這個锌云,如果mFallbackOnBackPressed存在的話
mFallbackOnBackPressed.run();
}
}
但是我們始終沒有看到原本后退的邏輯荠医,也就是返回到上一個activity的后退邏輯,但是有這么個東西mFallbackOnBackPressed
桑涎,無論從邏輯還是名字彬向,原本的后退邏輯
這個邏輯應(yīng)該就藏在這個run方法里了。
往上找找這個實例:
//OnBackPressedDispatcher
public OnBackPressedDispatcher(@Nullable Runnable fallbackOnBackPressed) {
mFallbackOnBackPressed = fallbackOnBackPressed;
}
原來是初始化的時候設(shè)置的石洗,那我們就可以回到ComponentActivity再看看OnBackPressedDispatcher是在哪里被實例化的了:
//ComponentActivity
private final OnBackPressedDispatcher mOnBackPressedDispatcher =
new OnBackPressedDispatcher(new Runnable() {
@Override
public void run() {
ComponentActivity.super.onBackPressed();//原本的后退邏輯
}
});
好了幢泼,整個邏輯下來就通了。
小建議
在使用自定義返回導(dǎo)航的時候讲衫,注意把LifecycleOwner
加上缕棵。