前沿
android 弹窗好几种,全局弹窗是什么?和普通Dialog(必须依附activity上下文的弹窗)有什么区别?
逛技术blog发现【全局dialog】这个名词,之前用FragmentDialog,自定义dialog。以及dialog的模态窗口,dialog非模态窗口。全局弹窗dialog比较新鲜。(对于我来说)。分享如下
下定义: 全局弹窗不依赖activity上下文的弹窗。及时关闭app,同样可以浮在系统桌面上。
实现全局弹窗
方式有两种:在服务中弹窗,使用windowManager进行弹窗设置。
方式1:开启服务,在service中进行,service中来弹出dialog。用new Dialog().show()的话,会抛异常。然后通过看源码。发现,dialog可以是可是设置成系统的alert的。
mDialog.getWindow().setType((WindowManager.LayoutParams.TYPE_SYSTEM_ALERT));
- 1
清单文件进行声明必要的权限 api<23
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
- 1
UML 类图关系
简单介绍下,后面会附上源码下载地址
1.使用系统api,observer,Observable 进行showdialog的通知。具体使用方案多种(接口回调方式,强引用server)能实现需求并且不出bug都是好方案。这里使用observer,Observable进行类间解耦。
2.服务的开启,startService(this,GlobalService.class),清单文件声明必要的权限,service声明都是必不可少的。
3. 系统api,observer,Observable的使用需要进行观察者的注册,dialogObservable.addObserver(new GlobalService())
package com.tseng.alldilaog;
import android.app.Application;
import android.content.Context;
import com.tseng.alldilaog.dialog.DialogObservable;
import com.tseng.alldilaog.dialog.GlobalService;
/**
* Author: yangweichao
* Date: 2018/8/14 下午3:41
* Description:
*/
public class MyApp extends Application {
private static Context mContext;
private static DialogObservable dialogObservable;
@Override
public void onCreate() {
super.onCreate();
mContext = getApplicationContext();
dialogObservable = new DialogObservable();
dialogObservable.addObserver(new GlobalService());
}
public static Context getmContext() {
return mContext;
}
public static void showDialog(String msg) {
dialogObservable.showDialog(msg);
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
4.使用方法,全局直接调用弹窗方法
public void fff(View view) {
MyApp.showDialog("你好啊");
}
- 1
- 2
- 3
方式2
采用 windowManager方式,这里不进行说明,请看引用部分 ⬇️ 引用
特别注意:
android 版本迭代过程中需要处理不同版本兼容,否则,崩溃是免不了的
在api >=23中加入运行时权限检查,并把 SYSTEM_ALERT_WINDOW废弃,替换为SYSTEM_OVERLAY_WINDOW,所以在使用全局弹窗入口进行权限检查。
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.SYSTEM_OVERLAY_WINDOW"/>
- 1
- 2
//兼容api23版本
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (Settings.canDrawOverlays(MainActivity.this)) {
Intent intent = new Intent(MainActivity.this, GlobalService.class);
startService(intent);
} else {
//若没有权限,提示获取.
Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
Toast.makeText(MainActivity.this, "需要取得权限以使用悬浮窗", Toast.LENGTH_SHORT).show();
startActivity(intent);
}
} else {
//SDK在23以下,不用管.
Intent intent = new Intent(MainActivity.this, GlobalService.class);
startService(intent);
finish();
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
不检查权限会报如下异常:
rocess: com.tseng.alldilaog, PID: 824
android.view.WindowManager$BadTokenException: Unable to <span class="hljs-keyword">add</span> window android<span class="hljs-preprocessor">.view</span><span class="hljs-preprocessor">.ViewRootImpl</span>$W@7795399 -- permission denied for window type 2003
at android.view.ViewRootImpl.setView(ViewRootImpl.java:928)
at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:342)
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:97)
at android.app.Dialog.show(Dialog.java:419)
at com.tseng.alldilaog.dialog.MyService$1.dispatchMessage(MyService.java:28)
at android.os.Looper.loop(Looper.java:154)
at android.app.ActivityThread.main(ActivityThread.java:6682)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1534)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1424)
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
在dialog,show()的地方同样添加兼容代码,防止异常发生。
该方法 update(),来自Observable,需要进行复写,处理dialog ui展示
@Override
public void update(Observable observable, Object data) {
String msg = (String) data;
if (msg != null) {
if (mDialog == null) {
mDialog = new Dialog(MyApp.getmContext());
mDialog.setContentView(R.layout.show_dialog);
}
if (mDialog != null && !mDialog.isShowing()) {
mDialog.setTitle(msg);
// 加入系统服务
// if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {//6.0
// mDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY);
// } else {
// mDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
// }
//8.0系统加强后台管理,禁止在其他应用和窗口弹提醒弹窗,如果要弹,必须使用TYPE_APPLICATION_OVERLAY
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
mDialog.getWindow().setType((WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY));
}else {
mDialog.getWindow().setType((WindowManager.LayoutParams.TYPE_SYSTEM_ALERT));
}
mDialog.show();
}
} else {
if (mDialog != null) {
mDialog.cancel();
mDialog = null;
}
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
android 8.0 对悬浮窗的的优化
demo下载: https://github.com/yatou252303/tseng/tree/master
最后,希望此篇博客对大家有所帮助,欢迎提出问题及建议共同探讨,如有兴趣可以关注我的博客,谢谢!
参考:
https://www.jianshu.com/p/634cd056b90c Android悬浮窗TYPE_TOAST小结: 源码分析
https://www.jianshu.com/u/38373cf49077 Android 8.0完美适配全局dialog 悬浮窗弹出
https://www.cnblogs.com/lizhanqi/p/8214319.html permission denied for window type 2003
https://blog.csdn.net/pangzaifei/article/details/43155997 全手机弹出的dialog和观察者设计模式