详解Android 全局弹出对话框SYSTEM_ALERT_WINDOW权限
项目中为了实现账号多设备登录的监听 一个账号在别的设备登录时在该设备上需要弹出对话框提示 故而用到全局对话框
方案一、
1、在开发中有时会用到全局弹出对话框但必须在manifest中申请权限:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
2、创建Dialog
AlertDialog.Builder builder=new AlertDialog.Builder(this); builder.setIcon(R.drawable.logo_mini); builder.setTitle("下线通知").setMessage("该账号在另一台Android设备上登录。") .setPositiveButton("重新登陆", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { //do somthing } }).setNegativeButton("退出",new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { Intent i=new Intent(CoreService.this,LoginActivity.class); i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); startActivity(i); } }); alertDialog = builder.create(); alertDialog.setCancelable(false); alertDialog.setCanceledOnTouchOutside(false); alertDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT); alertDialog.show();
注意要设置Dialog的Window类型为WindowManager.LayoutParams.TYPE_SYSTEM_ALERT。
方案二、
采用方案一带来的问题:
用户在安装应用时会询问用户是否授权问题
同时在小米手机上默认是禁止系统弹出框的,应用中的系统弹出框将不能够弹出
那能不能不申请系统权限,又能弹出提示框提示用户呢?
这里妨转换思路,既然系统弹出框弹不出,我们就不采用系统弹出框,给他弹个Activity替代。但这个时候注意,要再Service是中或者ApplicationContext中startActivity时要添加额外标志ntent.FLAG_ACTIVITY_NEW_TASK:
Intent i=new Intent(this,ActDialog.class); i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); startActivity(i); //代码中ActDialog其实是一个Activity把主题给它应用@android:style/Theme.Dialog //把activity做的像一个Dialog的风格
这样解决了权限申请和小米手机全局Dialog默认无法展示的问题。
最后补充一下在方案一中也可以不申请权限就能弹出对话框,将弹出Window type 改为LayoutParams.TYPE_TOAST,但是这种类型的弹出框是不能接受事件处理的。
使用WindowManager实现全局对话框
/** * 显示弹出框 * * @param context */ public static void showPopupWindow(final Context context, final DialogCallback callback) { // 获取WindowManager final WindowManager mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE); final WindowManager.LayoutParams params = new WindowManager.LayoutParams(); // 类型 params.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT; // 设置flag params.flags = WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; // 如果设置了WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,弹出的View收不到Back键的事件 // 不设置这个弹出框的透明遮罩显示为黑色 params.format = PixelFormat.TRANSLUCENT; // FLAG_NOT_TOUCH_MODAL不阻塞事件传递到后面的窗口 // 设置 FLAG_NOT_FOCUSABLE 悬浮窗口较小时,后面的应用图标由不可长按变为可长按 // 不设置这个flag的话,home页的划屏会有问题 params.width = WindowManager.LayoutParams.WRAP_CONTENT; params.height = WindowManager.LayoutParams.WRAP_CONTENT; params.gravity = Gravity.CENTER; TextView textView = new TextView(context); textView.setText("sfgsfdsfbsadfbasdfg"); textView.setTextSize(100); final View mView = LayoutInflater.from(context).inflate(R.layout.item_dialog_exit, null); TextView tv_itemdialog_title = (TextView) mView.findViewById(R.id.tv_itemdialog_title); TextView tv_itemdialog_ok = (TextView) mView.findViewById(R.id.tv_itemdialog_ok); TextView tv_itemdialog_close = (TextView) mView.findViewById(R.id.tv_itemdialog_close); tv_itemdialog_ok.setText("重新登录"); tv_itemdialog_close.setText("退出登录"); tv_itemdialog_title.setText("该账户在其他设备登录,若不是您在操作,请及时修改密码以防泄露信息"); tv_itemdialog_ok.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 隐藏弹窗 mWindowManager.removeView(mView); callback.onPositive(); } }); tv_itemdialog_close.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { mWindowManager.removeView(mView); callback.onNegative(); } }); mWindowManager.addView(textView, params); }
栏 目:Android代码
下一篇:在Android中如何使用DataBinding详解(Kotlin)
本文标题:详解Android 全局弹出对话框SYSTEM_ALERT_WINDOW权限
本文地址:http://www.codeinn.net/misctech/194260.html