Android 全局Dialog

前沿

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));

清单文件进行声明必要的权限 api<23

  <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

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);

    }
}

4.使用方法,全局直接调用弹窗方法

public void fff(View view) {
        MyApp.showDialog("你好啊");
    }
方式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"/>
    //兼容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();
        }

不检查权限会报如下异常:

rocess: com.tseng.alldilaog, PID: 824
    android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$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)

在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;
            }
        }


    }
android 8.0 对悬浮窗的的优化

Android8.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和观察者设计模式

Android全局Dialog是一种可以在应用程序的任何界面中显示的对话框。它可以覆盖在当前界面上方,并且不会阻止用户与其他部分进行交互。可以使用全局Dialog来显示一些重要的信息、警告、确认对话框等。 要创建一个全局Dialog,可以使用AndroidDialog类。首先,需要在应用程序的主题中设置一个全局样式,该样式将应用于所有的Dialog。可以在styles.xml文件中定义这个样式,例如: ```xml <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> <!-- 全局Dialog样式 --> <item name="android:dialogTheme">@style/CustomDialogTheme</item> </style> <style name="CustomDialogTheme" parent="Theme.AppCompat.Dialog"> <!-- 自定义全局Dialog样式 --> <item name="android:windowIsFloating">true</item> <!-- 其他自定义属性 --> </style> ``` 然后,在代码中创建Dialog实例并显示出来。可以在任何需要显示全局Dialog的地方调用以下代码: ```java Dialog dialog = new Dialog(context, R.style.CustomDialogTheme); dialog.setContentView(R.layout.dialog_layout); // 设置其他Dialog属性和内容 dialog.show(); ``` 这里的`R.layout.dialog_layout`是自定义的对话框布局文件,可以根据需要进行修改。 需要注意的是,全局Dialog并不是一种推荐的UI设计方式,因为它可能会打断用户的操作流程,并且可能会给用户带来困扰。在使用全局Dialog时,应该谨慎考虑,并确保它的使用是合理的,不会对用户体验造成负面影响。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

灯塔@kuaidao

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值