本文记录两种情况下Android的toast无法显示的问题及提供对应的解决方案
关闭通知权限
在Android系统中将通知栏权限,我们可以发现toast无法展示出来(绝大部分机子,oppo和vivo一定系统及以上仍然可以正常显示,应该是由于厂商做了什么处理)。通过查看Toast的源码我们可以发现,在Androird中toast的实现是通过WindowManager来创建,其显示又与NotificationManager有关,所以在通知栏权限的情况下,toast是显现不出来的。
因此,我们要做的就是绕过权限,通过查阅资料和Toast源码,有以下几种解决方法:
1.使用WindowManager,自己写一个toast,绕过NotificationManager,并改变 WindowManager.LayoutParams的type
2.使用Dialog、DialogFragment、PopupWindow等弹窗控件
3.使用Snackbar代替
分析
1、第一种解决方案,经过尝试,改变type类型仍然可能在部分机型上无法显示toast,甚至于需要比通知栏权限更加敏感的权限
2、第二种解决方案,使用弹窗控件,其严重依赖于activity,对于toast这种通知来说显得过重
3、使用Snackbar代替,其本质是通过在布局上添加view的形式,因此不存在机型适配问题。单存在两个问题,一是由于在布局上添加view来实现,显示时可能被Dialog、PopupWindow等遮挡;二是因为依赖于布局,在acitivity被销毁时候,布局被销毁,不像toast是个独立的window,因此无法跨页面传输,解决方法是在当前界面(A)的onDestroy里将需要显示的内容和剩余时间传递到上一个界面(B),当前界面销毁后,界面(B)继续显示。
所以最终决定采取方案三
Snackbar代替Toast
通过查看Snackbar的源码,我们可以发现Snackbar根据我们在public static Snackbar make(@NonNull View view, @NonNull CharSequence text,@Duration int duration)中传入的view来一层层往上找布局,最终将Snackbar添加进对应的布局。正常来说我们只要传入当前Activity的根布局就能正常显示。对于被Dialog、PopupWindow等遮挡的问题,需要特殊处理,以为其是另起个window,要将Dialog、PopupWindow的根布局传入Snackbar。
待补充(Snackbar中自定义视图使其UI更加接近toast、对应代码)
多次点击后,显示不出toast
在Android系统7.0及以上机型,快速点击显示toast,发现toast显示几个后,就不再显示。
其处理方法是在每次toast的时候,把之前显示的toast调用cancle方法
在开发中还发现在部分机型上(貌似是魅族),用代码连续show两次toast,发现toast无法显示
待补充(对应代码)