首頁 > 軟體

Android onbackpressed實現返回鍵的攔截和彈窗流程分析

2023-01-06 14:00:38

前言

此文章主要解決攔截使用者點選手機底部導航欄中的返回鍵時該事件的攔截;

此方法依然可以適用於fragment

onBackPressed()

這是 activity 中自帶的攔截返回的方法;

每當我們按下底部導航的返回鍵時,就會呼叫該事件,我們可以通過重寫它來阻止 activity 的銷燬

譬如下方程式碼,我們直接把 super 註釋掉,那麼點選返回鍵時就不會有任何反應,因為我們並沒有針對性的進行響應的事件處理

override fun onBackPressed() {
    // super.onBackPressed()
}

onBackPressedDispatcher

fragment 沒有提供 onBackPressed 方法,故大多數情況下使用 android 推薦的 onBackPressedDispatcher 來進行返回鍵的攔截操作;

接下來是一個攔截返回並彈窗,詢問是否退出的一個小案例;

為便於大家理解,故直接使用 activity 而非 fragment;

首先使用屬性委託 lazy,即懶載入的方式定義一個 onBackPressedDispatcher 變數

class NoteActivity : AppCompatActivity() {
    private val dispatcher by lazy {
        NoteActivity().onBackPressedDispatcher
    }
    override fun onCreate(savedInstanceState: Bundle?) {
        ...
    }
}

接著在 oncreate 方法中註冊 onBackPressedDispatcher

註冊的方式即新增一個回撥函數用處處理攔截時的動作,使用 addCallback

addCallback 有兩個引數,第一個為上下文,第二個為 OnBackPressedCallback 函數方法(下面的程式碼使用了 lambda 的形式提高效率)

OnBackPressedCallback 函數接收一個引數,我們一般都填 true;

它還需要實現一個方法 handleOnBackPressed ,這裡面就是處理對應實踐的地方了

在 handleOnBackPressed 方法內:

  • 如果想阻止返回事件,那就什麼都不做
  • 如果想允許返回事件,那就使用 finish,結束掉本 activity
override fun onCreate(savedInstanceState: Bundle?) {
    ...
    dispatcher.addCallback(
        this,
        object : OnBackPressedCallback(true) {
            override fun handleOnBackPressed() {
                // 構造一個AlertDialog,具體怎麼用不細說
                AlertDialog.Builder(this@NoteActivity).apply {
                    setTitle("Oops!")
                    setMessage("請寫上一個標題奧,不然咱沒法儲存")
                    setIcon(R.drawable.ic_info)
                    // 點選了確認按鈕,直接使用finish銷燬本activity
                    setPositiveButton("直接退出") { dialog, which ->
                        finish()
                    }
                    // 點了否定,就啥都不做,即不處理返回事件,那自然就不會銷燬了
                    setNegativeButton("我再想想", null)
                }.show()
            }
        }
    )
}

最後,在 onBackPressed 中觸發 onBackPressedDispatcher (因為返回鍵點選事件都由 onBackPressed 所攔截,onBackPressedDispatcher 僅僅是充當了一個助手的作用)

注意刪掉或者註釋原來的的 super,否則你點返回鍵就直接退出了,那就前功盡棄了!!!

override fun onBackPressed() {
    dispatcher.onBackPressed()
}

全部程式碼展示:

package com.zhiyiyi.sqlitedemo
import android.content.DialogInterface
import android.database.sqlite.SQLiteDatabase
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.view.KeyEvent
import android.view.MotionEvent
import androidx.activity.OnBackPressedCallback
import androidx.appcompat.app.AlertDialog
import com.zhiyiyi.sqlitedemo.databinding.ActivityNoteBinding
import com.zhiyiyi.sqlitedemo.utils.DBHelper
class NoteActivity : AppCompatActivity() {
    private lateinit var binding: ActivityNoteBinding
    // 懶載入
    private val dispatcher by lazy {
        NoteActivity().onBackPressedDispatcher
    }
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = ActivityNoteBinding.inflate(layoutInflater)
        setContentView(binding.root)
        // 註冊回撥事件
        dispatcher.addCallback(
            this,
            object : OnBackPressedCallback(true) {
                override fun handleOnBackPressed() {
                    AlertDialog.Builder(this@NoteActivity).apply {
                        setTitle("Oops!")
                        setMessage("請寫上一個標題奧,不然咱沒法儲存")
                        setIcon(R.drawable.ic_info)
                        setPositiveButton("直接退出") { dialog, which ->
                            finish()
                        }
                        setNegativeButton("我再想想", null)
                    }.show()
                }
            }
        )
    }
    // 檢測返回鍵點選事件
    override fun onBackPressed() {
        dispatcher.onBackPressed()
    }
}

到此這篇關於Android onbackpressed實現返回鍵的攔截和彈窗流程分析的文章就介紹到這了,更多相關Android onbackpressed內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com