前言
在 Android 11 Toast 的行为发生了变更
- 禁止后台自定义 Toast
- text toast 不允许自定义
- setView() 被弃用
- 新增 Toast.Callback 回调
Android 11 API 变更
禁止后台自定义 Toast
自定义 Toast 「不能」 在 app 处于后台时显示,取而代之会显示 「"Background custom toast blocked for package [packageName] See g.co/dev/toast."」 的文本 toast
禁止后台自定义 Toast
普通的 text toast
不受影响
普通的 text toast 不受影响
text toast 不允许自定义
默认的 toast 是 text toast
,如果想使用自定义的 toast ,需要调用 setView() 方法
在 targetSdkVersion 为 R 或更高时,调用 setGravity 和 setMargin 方法将不进行任何操作
❝
官方文档中所述的 Android R 仅影响 「text toast」 ,而自定义的 toast 不受影响
❞
调用无效,仅影响 text toast
调用无效,仅影响 test toast
如图,在 test toast 中调用 setGravity 和 setMargin 方法,但 toast 位置并未居中
在 test toast 中调用 setGravity 和 setMargin 方法
并未居中,方法不生效
setView() 被弃用
setView() 方法被标记弃用
❝
「Deprecated」 表示该功能目前仍可以使用,但可能会在将来的 Android 版本中删除。建议开发人员避免长期使用此功能
❞
setView 被弃用
可以看到,官方在一步步禁止自定义 Toast
目前是 targetSdkVersion 为 R 或更高的 app 禁止后台弹出自定义 Toast
同时将 setView() 方法标记弃用,当该方法从源码中移除后,自定 Toast 的方式将被彻底消灭
❝
当然,官方提供了相应的替代品,使用 Snackbar
❞
新增 Toast.Callback 回调
添加了新的回调(Toast.Callback
),以通知 Toast 显示和隐藏。可以通过以下方法轻松将其添加到 Toast 中:
val toast = Toast.makeText(this, R.string.simple2_toast, Toast.LENGTH_SHORT)toast.addCallback(object : Toast.Callback() { override fun onToastShown() { super.onToastShown() Log.d(TAG, "onToastShown") } override fun onToastHidden() { super.onToastHidden() Log.d(TAG, "onToastHidden") }})toast.show()
一些小 tips 及 demo
demo 在这 ,切换 Flavor 即可指定不同的 targetSdkVersion
切换 Flavor
在写 demo 时遇到一些小问题
tip1
Handler()
无参构造方法和 Handler(Handler.Callback)
构造方法 被弃用了
无参构造器被弃用
简单来讲就是在初始化 Handler 时要显示的配置 Looper
Handler 使用不当会有这样一种 bug,例如在子线程通过无参构造函数创建 Handler,您可能会看到这样的异常
错误日志
抛出异常源码
详细内容这里就不讲了,这是 Android 开发者的必备知识
官方通过强制使用传入 Looper 的 Handler 构造器来避免使用中的问题
tip2
过去使用 Toast 构造器创建 Toast 对象 并调用 setText 方法会崩溃,targetSdkVersion 为 R 时不会崩溃
相同的代码 targetSdkVersion 低版本会崩溃
崩溃,但设置位置生效
异常log
API 29 源码
API 29 中调用 setText() 方法时要保证 mNextView 不为空,而 mNextView 是调用 setView 赋值的
API 29 setView 源码
因此过去使用 Toast 构造器创建 toast 对象无法创建普通的 text toast,必须调用 setView 方法
至于 API 30 肯定在这里做了修改,由于现在看不到源码,我也猜测不出官方的用意