Application显示Dialog

2,952 阅读1分钟

项目里有个需求,在某个页面可能调接口或收到消息,需要在弹一个dialog,最后决定在application里实现这个弹出操作

首次尝试

使用applcationContext,并设置dialog为系统级 and 设置权限 android.permission.SYSTEM_ALERT_WINDOW

   AlertDialog.Builder builder = new AlertDialog.Builder(getApplicationContext());
   builder.setMessage("俺がガンダムだ")
           .setCancelable(false);
   AlertDialog alert = builder.create();
   alert.setCancelable(false);
   alert.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
   // 话说这个窗口类型已经弃用了,最新的是悬浮窗类型 不过效果一样
   alert.show();

效果

Android 6.0以下正常运行,6.0以上就不弹出了出了,场面一度十分尴尬。

也考虑了弹出前先请求个权限,但是脑补了一下场景还是放弃了

第二次尝试

其实这个需求的难点是在applcation里获取要显示页面的context,所以就可以用上神奇的ActivityLifecycleCallbacks了

   private Context mContext;
   private void initLifecycle() {
       registerActivityLifecycleCallbacks(new ActivityLifecycleCallbacks() {
           @Override
           public void onActivityCreated(Activity activity, Bundle savedInstanceState) {
               if(activity.getParent()!=null){//如果这个视图是嵌入的子视图
                   mContext = activity.getParent();
               }else {
                   mContext = activity;
               }
           }
           @Override
           public void onActivityStarted(Activity activity) {
               if(activity.getParent()!=null){
                   mContext = activity.getParent();
               }else {
                   mContext = activity;
               }
           }

           @Override
           public void onActivityResumed(Activity activity) {
               if(activity.getParent()!=null){
                   mContext = activity.getParent();
               }else {
                   mContext = activity;
               }
           }
          //.....
       });

   }

   private void showDialog(){
       AlertDialog.Builder builder = new AlertDialog.Builder(mContext);
       builder.setMessage("俺がガンダムだ")
           .setCancelable(false);
       AlertDialog alert = builder.create();
       alert.setCancelable(false);
       alert.show();
   }

这次的效果

succeed

扩展

根据资料,还可以用service的方式来实现,不过感觉会有点麻烦,有空的同学可以试试