詳解Android 全域性彈出對話方塊SYSTEM_ALERT_WINDOW許可權

NO IMAGE

專案中為了實現賬號多裝置登入的監聽 一個賬號在別的裝置登入時在該裝置上需要彈出對話方塊提示 故而用到全域性對話方塊

方案一、

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