Android M及以上版本系统 悬浮窗权限 的解决方案

 

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, MyService.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, MyService.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中启动悬浮的实现
首先 你需要在配置文件悬浮窗权限

第二需要在服务中开启悬浮窗

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);// 添加窗口          }       

第三,关闭服务时候,关闭悬浮窗

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

这就是在Android下面开启悬浮窗权限的全部流程。

更多相关文章

  1. Android系统中的ROOT和System权限的区别
  2. Android中的权限问题
  3. android 安全机制
  4. 如何编程实现开启或者关闭GPS
  5. 专访丰生强:Android软件安全与逆向分析
  6. Android(安卓)内容提供器---创建内容提供器(元素)
  7. Android(安卓)6.0 (marshmallow)中新的新技术
  8. android boot.img 拆包,修改adb 具有root权限,以i9250为例
  9. 如何使Android应用程序获得root权限

随机推荐

  1. Android应用.国际化-屏幕适配-样式与主题
  2. android中PhoneWindow、DecorView
  3. android设置保持窗口常亮
  4. 关闭程序
  5. Android Native Development Kit (NDK)
  6. Version Code 版本设置
  7. Android 滑动开关控件
  8. Android MediaPlayer
  9. Android 学习之- 单选按钮、复选框、状态
  10. Android中下载文件的使用