Android6.0系統(tǒng)懸浮窗權(quán)限的問(wèn)題解決方法

Android的窗口體系中,WindowManager占有非常重要的地位,平時(shí)我們使用懸浮窗會(huì)遇到一些權(quán)限的問(wèn)題。
當(dāng) Android工程在
targetSdkVersion 23
編譯,Android6.0及其以上版本手機(jī)使用懸浮窗功能時(shí)候,會(huì)發(fā)生

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

的異常,導(dǎo)致程序崩潰。
當(dāng)Android工程在
targetSdkVersion 22
編譯,Android6.0及其以上版本手機(jī)使用懸浮窗功能會(huì)正常使用
其實(shí)原因很簡(jiǎn)單,大部分的Android6.0手機(jī)(尤其是三星,谷歌原生手機(jī))
在大于等于23版本下編譯,懸浮窗權(quán)限默認(rèn)是關(guān)閉沒(méi)有權(quán)限,然在小于23版本下編譯懸浮窗權(quán)限是開(kāi)啟有權(quán)限的。
所以在大于23版本下編譯時(shí)需要去檢測(cè)懸浮窗權(quán)限,并且獲取懸浮窗權(quán)限,下面我就羅列下怎么去檢測(cè)懸浮窗權(quán)限并且獲取懸浮窗權(quán)限

if(Build.VERSION.*SDK_INT*>=23)
{
       if(Settings.*canDrawOverlays*(this))
       {
           //有懸浮窗權(quán)限開(kāi)啟服務(wù)綁定 綁定權(quán)限
           Intent intent = new Intent(MainActivity.this, FBService.class);
           startService(intent);

       }else{
           //沒(méi)有懸浮窗權(quán)限m,去開(kāi)啟懸浮窗權(quán)限
           try{
                   Intent  intent=new Intent(Settings.*ACTION_MANAGE_OVERLAY_PERMISSION*);
             startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);
           }catch (Exception e)
           {
               e.printStackTrace();
           }

       }
} else{
    //默認(rèn)有懸浮窗權(quán)限  但是 華為, 小米,oppo等手機(jī)會(huì)有自己的一套Android6.0以下  會(huì)有自己的一套懸浮窗權(quán)限管理 也需要做適配
    Intent intent = new Intent(MainActivity.this, FBService.class);
    startService(intent);
}    

每次使用懸浮窗的時(shí)候都要去檢測(cè)權(quán)限,因?yàn)閼腋〈皺?quán)限是可以手動(dòng)關(guān)閉的。
位置位于 (三星S6為例Android6.0.1版本)設(shè)置-- 應(yīng)用程序--應(yīng)用程序管理器 -- 更多 --可出現(xiàn)在頂部的應(yīng)用程序 --- 選擇你的APP -- 運(yùn)行在其他應(yīng)用的上層顯示

下圖所示:

當(dāng)你點(diǎn)擊懸浮窗權(quán)限app開(kāi)關(guān)時(shí)候 退出 會(huì)在activity 有回調(diào)方法。


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, "權(quán)限授予失敗,無(wú)法開(kāi)啟懸浮窗", Toast.LENGTH_SHORT).show();
            } else {
                       Toast.makeText(this, "權(quán)限授予成功!", Toast.LENGTH_SHORT).show();
               //有懸浮窗權(quán)限開(kāi)啟服務(wù)綁定 綁定權(quán)限
                Intent intent = new Intent(MainActivity.this, FBService.class);
                startService(intent);
            }
        }
    }
}

下面就羅列下Service中啟動(dòng)懸浮的實(shí)現(xiàn)
首先 你需要在配置文件懸浮窗權(quán)限

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

第二需要在服務(wù)中開(kāi)啟懸浮窗


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*;// 系統(tǒng)提示window
        mParams.format = PixelFormat.*TRANSLUCENT*;// 支持透明
        // mParams.format = PixelFormat.RGBA_8888;
        mParams.flags |= WindowManager.LayoutParams.*FLAG_NOT_TOUCH_MODAL *| WindowManager.LayoutParams.*FLAG_NOT_FOCUSABLE*;// 焦點(diǎn)
        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);// 添加窗口
      
    }
       

第三,關(guān)閉服務(wù)時(shí)候,關(guān)閉懸浮窗

@Override
public void onDestroy() {
    if (wManager!=null&&windowView != null) {
            wManager.removeView(windowView);
    }
    super.onDestroy();
}

這就是在Android下面開(kāi)啟懸浮窗權(quán)限的全部流程。

Android訂閱是探討Android開(kāi)發(fā)的公眾號(hào),分享最有價(jià)值的Android干貨文章

歡迎關(guān)注我們,一起討論技術(shù),掃描和長(zhǎng)按下方的二維碼可快速關(guān)注我們

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容