Android的窗口體系中,WindowManager占有非常重要的地位,平時我們使用懸浮窗會遇到一些權限的問題。
當 Android工程在
targetSdkVersion 23
編譯,Android6.0及其以上版本手機使用懸浮窗功能時候,會發生
java.lang.RuntimeException: Unable to create service com.fb.tangyc.fbtools.service.FBService: android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@123e0ab -- permission denied for this window type
的異常,導致程序崩潰。
當Android工程在
targetSdkVersion 22
編譯,Android6.0及其以上版本手機使用懸浮窗功能會正常使用
其實原因很簡單,大部分的Android6.0手機(尤其是三星,谷歌原生手機)
在大于等于23版本下編譯,懸浮窗權限默認是關閉沒有權限,然在小于23版本下編譯懸浮窗權限是開啟有權限的。
所以在大于23版本下編譯時需要去檢測懸浮窗權限,并且獲取懸浮窗權限,下面我就羅列下怎么去檢測懸浮窗權限并且獲取懸浮窗權限
if(Build.VERSION.*SDK_INT*>=23)
{
if(Settings.*canDrawOverlays*(this))
{
//有懸浮窗權限開啟服務綁定 綁定權限
Intent intent = new Intent(MainActivity.this, FBService.class);
startService(intent);
}else{
//沒有懸浮窗權限m,去開啟懸浮窗權限
try{
Intent intent=new Intent(Settings.*ACTION_MANAGE_OVERLAY_PERMISSION*);
startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);
}catch (Exception e)
{
e.printStackTrace();
}
}
} else{
//默認有懸浮窗權限 但是 華為, 小米,oppo等手機會有自己的一套Android6.0以下 會有自己的一套懸浮窗權限管理 也需要做適配
Intent intent = new Intent(MainActivity.this, FBService.class);
startService(intent);
}
每次使用懸浮窗的時候都要去檢測權限,因為懸浮窗權限是可以手動關閉的。
位置位于 (三星S6為例Android6.0.1版本)設置-- 應用程序--應用程序管理器 -- 更多 --可出現在頂部的應用程序 --- 選擇你的APP -- 運行在其他應用的上層顯示
當你點擊懸浮窗權限app開關時候 退出 會在activity 有回調方法。
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == OVERLAY_PERMISSION_REQ_CODE) {
if(Build.VERSION.SDK_INT>=23) {
if (!Settings.canDrawOverlays(this)) {
Toast.makeText(this, "權限授予失敗,無法開啟懸浮窗", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(this, "權限授予成功!", Toast.LENGTH_SHORT).show();
//有懸浮窗權限開啟服務綁定 綁定權限
Intent intent = new Intent(MainActivity.this, FBService.class);
startService(intent);
}
}
}
}
下面就羅列下Service中啟動懸浮的實現
首先 你需要在配置文件懸浮窗權限
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
第二需要在服務中開啟懸浮窗
public class FBService extends Service {
private WindowManager wManager;// 窗口管理者
private WindowManager.LayoutParams mParams;// 窗口的屬性
private FloatButtonLayout windowView;
private SurfaceHolder holder;
public static final String *ACTION_ALPHA *= "com.fb.alpha";
private ServiceReceiver receiver;
@Override
public void onCreate() {
super.onCreate();
wManager = (WindowManager) getSystemService(Context.*WINDOW_SERVICE*);
mParams = new WindowManager.LayoutParams(WindowManager.LayoutParams.*TYPE_SYSTEM_ERROR*, WindowManager.LayoutParams.*FLAG_NOT_FOCUSABLE*, PixelFormat.*TRANSPARENT*);
mParams.type = WindowManager.LayoutParams.*TYPE_SYSTEM_ERROR*;// 系統提示window
mParams.format = PixelFormat.*TRANSLUCENT*;// 支持透明
// mParams.format = PixelFormat.RGBA_8888;
mParams.flags |= WindowManager.LayoutParams.*FLAG_NOT_TOUCH_MODAL *| WindowManager.LayoutParams.*FLAG_NOT_FOCUSABLE*;// 焦點
mParams.width = WindowManager.LayoutParams.*WRAP_CONTENT*;// 窗口的寬和高
mParams.height = WindowManager.LayoutParams.*WRAP_CONTENT*;
mParams.gravity = Gravity.*LEFT *| Gravity.*TOP*;
mParams.y = SharedPreferencesUtils.*getSharedPreferencesUtils*().getParamsY(getApplicationContext());
mParams.x = SharedPreferencesUtils.*getSharedPreferencesUtils*().getParamsX(getApplicationContext());
mParams.windowAnimations = android.R.style.*Animation_Toast*;
// mParams.alpha = 0.8f;//窗口的透明度
LayoutInflater layoutInflater = LayoutInflater.*from*(getApplicationContext());
windowView = (FloatButtonLayout) layoutInflater.inflate(R.layout.*float_button_layout*, null);
wManager.addView(windowView, mParams);// 添加窗口
}
第三,關閉服務時候,關閉懸浮窗
@Override
public void onDestroy() {
if (wManager!=null&&windowView != null) {
wManager.removeView(windowView);
}
super.onDestroy();
}
這就是在Android下面開啟懸浮窗權限的全部流程。
Android訂閱是探討Android開發的公眾號,分享最有價值的Android干貨文章
歡迎關注我們,一起討論技術,掃描和長按下方的二維碼可快速關注我們