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 = newIntent(MainActivity.this,FBService.class);startService(intent);}else{
//没有悬浮窗权限m,去开启悬浮窗权限try{
Intent intent=newIntent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);}catch(Exception e)
{
e.printStackTrace();}
}
} else{
//默认有悬浮窗权限 但是 华为, 小米,oppo等手机会有自己的一套Android6.0以下 会有自己的一套悬浮窗权限管理 也需要做适配Intent intent = newIntent(MainActivity.this,FBService.class);startService(intent);}
每次使用悬浮窗的时候都要去检测权限,因为悬浮窗权限是可以手动关闭的。
位置位于 (三星S6为例Android6.0.1版本)设置-- 应用程序--应用程序管理器 -- 更多 --可出现在顶部的应用程序 --- 选择你的APP -- 运行在其他应用的上层显示
下图所示:
当你点击悬浮窗权限app开关时候 退出 会在activity 有回调方法。
protected voidonActivityResult(intrequestCode, intresultCode,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 = newIntent(MainActivity.this,FBService.class);startService(intent);}
}
}
}
下面就罗列下Service中启动悬浮的实现
首先 你需要在配置文件悬浮窗权限
android:name='android.permission.SYSTEM_ALERT_WINDOW'/>
第二需要在服务中开启悬浮窗
public classFBService extendsService {
privateWindowManager wManager;// 窗口管理者privateWindowManager.LayoutParams mParams;// 窗口的属性privateFloatButtonLayout windowView;privateSurfaceHolder holder;public static finalString ACTION_ALPHA= 'com.fb.alpha';privateServiceReceiver receiver;@Overridepublic voidonCreate() {
super.onCreate();wManager= (WindowManager) getSystemService(Context.WINDOW_SERVICE);mParams= newWindowManager.LayoutParams(WindowManager.LayoutParams.TYPE_SYSTEM_ERROR,WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,PixelFormat.TRANSPARENT);mParams.type= WindowManager.LayoutParams.TYPE_SYSTEM_ERROR;// 系统提示windowmParams.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 voidonDestroy() {
if(wManager!=null&&windowView!= null) {
wManager.removeView(windowView);}
super.onDestroy();}
这就是在Android下面开启悬浮窗权限的全部流程。
Android订阅是探讨Android开发的公众号,分享最有价值的Android干货文章