悬浮窗权限突破及兼容性处理

转载请注明出处:
http://blog.csdn.net/brucehurrican/article/details/64129000

通常业务需要在桌面显示悬浮窗来展示某些功能,如360的悬浮球,MIUI 系统显示悬浮窗。

常用的悬浮窗代码写法如下

private static void showSmall1(Context context) {
        WindowManager windowManager = getWindowManager(context);
        int screenWidth = windowManager.getDefaultDisplay().getWidth();
        int screenHeight = windowManager.getDefaultDisplay().getHeight();
        if (null == smallWindow) {
            smallWindow = new FWSmallView(context);
            if (smallWindowParams == null) {
                smallWindowParams = new WindowManager.LayoutParams();
                smallWindowParams.type = WindowManager.LayoutParams.TYPE_PHONE;
                smallWindowParams.format = PixelFormat.RGBA_8888;
                smallWindowParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
                smallWindowParams.gravity = Gravity.LEFT | Gravity.TOP;
                smallWindowParams.width = FWSmallView.viewWidth;
                smallWindowParams.height = FWSmallView.viewHeight;
                smallWindowParams.x = screenWidth;
                smallWindowParams.y = screenHeight / 2;
            }
            smallWindow.setParams(smallWindowParams);
            windowManager.addView(smallWindow, smallWindowParams);
        }
    }

type 类型可以选择的有 TYPE_PHONE,TYPE_SYSTEM_ALERT和 TYPE_TOAST,对于 TYPE_PHONE,TYPE_SYSTEM_ALERT 需要在 AndroidManifest.xml 中声明权限

uses-permission android:name=”android.permission.SYSTEM_ALERT_WINDOW”

否则会报错:

permission denied for this window type

原因是在 com.android.server.policy.PhoneWindowManager 类中
方法 checkAddPermission中

public int checkAddPermission(WindowManager.LayoutParams attrs, int[] outAppOp) {
        int type = attrs.type;
        outAppOp[0] = AppOpsManager.OP_NONE;
        if (!((type >= FIRST_APPLICATION_WINDOW && type <= LAST_APPLICATION_WINDOW)
                || (type >= FIRST_SUB_WINDOW && type <= LAST_SUB_WINDOW)
                || (type >= FIRST_SYSTEM_WINDOW && type <= LAST_SYSTEM_WINDOW))) {
            return WindowManagerGlobal.ADD_INVALID_TYPE;
        }
        if (type < FIRST_SYSTEM_WINDOW || type > LAST_SYSTEM_WINDOW) {
            // Window manager will make sure these are okay.
            return WindowManagerGlobal.ADD_OKAY;
        }
        String permission = null;
        switch (type) {
            case TYPE_TOAST:
                // XXX right now the app process has complete control over
                // this...  should introduce a token to let the system
                // monitor/control what they are doing.
                outAppOp[0] = AppOpsManager.OP_TOAST_WINDOW;
                break;
            case TYPE_DREAM:
            case TYPE_INPUT_METHOD:
            case TYPE_WALLPAPER:
            case TYPE_PRIVATE_PRESENTATION:
            case TYPE_VOICE_INTERACTION:
            case TYPE_ACCESSIBILITY_OVERLAY:
                // The window manager will check these.
                break;
            case TYPE_PHONE:
            case TYPE_PRIORITY_PHONE:
            case TYPE_SYSTEM_ALERT:
            case TYPE_SYSTEM_ERROR:
            case TYPE_SYSTEM_OVERLAY:
                permission = android.Manifest.permission.SYSTEM_ALERT_WINDOW;
                outAppOp[0] = AppOpsManager.OP_SYSTEM_ALERT_WINDOW;
                break;
            default:
                permission = android.Manifest.permission.INTERNAL_SYSTEM_WINDOW;
        }
        if (permission != null) {
            if (permission == android.Manifest.permission.SYSTEM_ALERT_WINDOW) {
                final int callingUid = Binder.getCallingUid();
                // system processes will be automatically allowed privilege to draw
                if (callingUid == Process.SYSTEM_UID) {
                    return WindowManagerGlobal.ADD_OKAY;
                }
                // check if user has enabled this operation. SecurityException will be thrown if
                // this app has not been allowed by the user
                final int mode = mAppOpsManager.checkOp(outAppOp[0], callingUid,
                        attrs.packageName);
                switch (mode) {
                    case AppOpsManager.MODE_ALLOWED:
                    case AppOpsManager.MODE_IGNORED:
                        // although we return ADD_OKAY for MODE_IGNORED, the added window will
                        // actually be hidden in WindowManagerService
                        return WindowManagerGlobal.ADD_OKAY;
                    case AppOpsManager.MODE_ERRORED:
                        return WindowManagerGlobal.ADD_PERMISSION_DENIED;
                    default:
                        // in the default mode, we will make a decision here based on
                        // checkCallingPermission()
                        if (mContext.checkCallingPermission(permission) !=
                                PackageManager.PERMISSION_GRANTED) {
                            return WindowManagerGlobal.ADD_PERMISSION_DENIED;
                        } else {
                            return WindowManagerGlobal.ADD_OKAY;
                        }
                }
            }
            if (mContext.checkCallingOrSelfPermission(permission)
                    != PackageManager.PERMISSION_GRANTED) {
                return WindowManagerGlobal.ADD_PERMISSION_DENIED;
            }
        }
        return WindowManagerGlobal.ADD_OKAY;
    }

从源码中可以看到当声明为 TYPE_PHONE, TYPE_SYSTEM_ALERT等类型时,需要进行权限检查。

这种通过自定义 view, FWSmallView 的方式定制悬浮窗样式,通过 WindowManager addView() 显示自定义的悬浮窗。这种方法在原生类的手机中是可以显示的,但是介于国内第三方厂商修改的 ROM,有的机型在用户未授权的情况下只能在应用内显示或者不显示,如果要在桌面显示的话,必须要用户授权,如华为荣耀系列,小米/红米手机,魅族手机,都需要用户的手动授权才能显示悬浮窗。需要注意的是,以小米/红米手机使用的 MIUI 系统默认在安装 app 时,悬浮窗功能默认是关闭的,而且开启该功能的操作路径比较深,MIUI 开启步骤有两种方法,

  • 安全中心-授权管理-应用权限管理-XX应用-显示悬浮窗,打开

  • 设置-更多应用-XX应用-权限管理-显示悬浮窗,打开

两种方法对于用户来说,操作都有点麻烦。
通过在代码中判断当前 view 是否是显示状态

view.getVisibility()

获取到的值是0,即View.VISIBLE,就是说,当 AndroidManifest.xml 中声明了权限的场景下,view.getVisibility()获取到的是View.VISIBLE,但是在MIUI系统中却看不到悬浮窗,此时权限管理中的悬浮窗是关闭的。推测是 MIUI针对悬浮窗功能做了自己的限制,这个限制不是来源于 android 系统的限制,完全是 MIUI应用层的限制。

那么有没有什么方法不可以绕过 MIUI的用户授权呢?答案是肯定的,上面介绍的WindowManager.LayoutParams类型中还有一个没有说,那就是 TYPE_TOAST,通过设置 TYPE_TOAST可以让悬浮窗功能正常显示不需要 MIUI用户授权。

对于使用 TYPE_TOAST类型的悬浮窗,有个版本限制需要注意,如果用户系统版本> 19时,可以接收点击/触摸事件,如果<19时则不能接收点击/触摸事件。

还有一种方式通过调用Toast 的 setView() 方法传入自定义布局文件,以自定义 toast 形式来显示悬浮窗,在MIUI、华为荣耀、魅族手机上是不需要用户授权的。但是,这种自定义 toast 的方式,有两个缺点
1. 不能控制悬浮窗显示的时间,只有两个参数 LENGTH_SHORT,LENGTH_LONG,对应的时间分别为,3.5秒,2秒。
对应源码如下:
com.android.server.notification.NotificationManagerService

static final int LONG_DELAY = 3500; // 3.5 seconds
    static final int SHORT_DELAY = 2000; // 2 seconds
private void scheduleTimeoutLocked(ToastRecord r)
    {
        mHandler.removeCallbacksAndMessages(r);
        Message m = Message.obtain(mHandler, MESSAGE_TIMEOUT, r);
        long delay = r.duration == Toast.LENGTH_LONG ? LONG_DELAY : SHORT_DELAY;
        mHandler.sendMessageDelayed(m, delay);
    }

2.自定义的 toast 也不能处理点击/触摸事件。

既然自定义的 toast 不行的话,只能试试修改系统的 toast 来达到显示悬浮窗的目的了。

通过反射的方式来替换系统 toast 的显示方式,达到自由控制悬浮窗显示/关闭,并处理点击/触摸事件。
代码如下:

private void initTN() {
        int screenWidth = windowManager.getDefaultDisplay().getWidth();
        int screenHeight = windowManager.getDefaultDisplay().getHeight();
        try {
            Field tnField = toast.getClass().getDeclaredField("mTN");
            tnField.setAccessible(true);
            mTN = tnField.get(toast);
            show = mTN.getClass().getMethod("show");
            hide = mTN.getClass().getMethod("hide");

            Field tnParamsField = mTN.getClass().getDeclaredField("mParams");
            tnParamsField.setAccessible(true);
            if (null == mParams) {
                mParams = (WindowManager.LayoutParams) tnParamsField.get(mTN);
                mParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                        | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
                /**设置动画*/
                mParams.windowAnimations = android.R.anim.fade_in;
                if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT){
                    mParams.type = WindowManager.LayoutParams.TYPE_PHONE;
                }
                mParams.x = screenWidth;
                mParams.y = screenHeight / 2;
            }

            /**调用tn.show()之前一定要先设置mNextView*/
            Field tnNextViewField = mTN.getClass().getDeclaredField("mNextView");
            tnNextViewField.setAccessible(true);
            tnNextViewField.set(mTN, toast.getView());
        } catch (Exception e) {
            e.printStackTrace();
        }
        toast.setGravity(Gravity.LEFT | Gravity.TOP,mParams.x ,mParams.y);
    }

通过上述方法反射获取toast 源码中的 show,hide,mTN。

show.invoke(mTN);
显示悬浮窗
hide.invoke(mTN);
关闭悬浮窗

并且,这个关闭的时间是可以自己控制的。

悬浮窗免授权显示的介绍就到这了。下面我来介绍下,怎样处理用户的点击/触摸事件了。

如果想要完成像360悬浮球那样,用户可以拖动小球的功能,需要重写 onTouchEvent 方法,源码如下:

public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                // 手指按下时记录必要数据,纵坐标的值都需要减去状态栏高度
                xInView = event.getX();
                yInView = event.getY();
                xDownInScreen = event.getRawX();
                yDownInScreen = event.getRawY() - getStatusBarHeight();
                xInScreen = event.getRawX();
                yInScreen = event.getRawY() - getStatusBarHeight();
                break;
            case MotionEvent.ACTION_MOVE:
                xInScreen = event.getRawX();
                yInScreen = event.getRawY() - getStatusBarHeight();
                // 手指移动的时候更新小悬浮窗的位置
                updateViewPosition();
                break;
            case MotionEvent.ACTION_UP:
                // 如果手指离开屏幕时,xDownInScreen和xInScreen相等,且yDownInScreen和yInScreen相等,则视为触发了单击事件。
                if ((xDownInScreen == xInScreen) && (yDownInScreen == yInScreen)) {
                    openBigWindow();
                } else if (Math.abs(xDownInScreen - xInScreen) <= 10 && Math.abs(yDownInScreen - yInScreen) <= 10) {
                    openBigWindow();
                }
                LogUtils.i("isShow->"+FWmanager.isWindowShowing());
                break;
            default:
                break;
        }
        return true;
    }

这里我遇到了几个坑,
1.在 SONY的一款机型上发现的,测试时,不论怎样点击布局上的按钮,悬浮窗都无法实现点击功能。通过调试发现,每次触摸时,在MotionEvent.ACTION_UP时,SONY系统自动会将当前触摸的坐标进行偏移,导致按照(xDownInScreen == xInScreen) && (yDownInScreen == yInScreen)来判断用户点击事件时始终为 false。这样就达不到“点击”的效果了。通过加上Math.abs(xDownInScreen - xInScreen) <= 10 && Math.abs(yDownInScreen - yInScreen) <= 10来判断,如果为true,则认为用户是“点击”而非“拖动”,10只是个经验值。
2.在部分三星手机上出现,无法响点击/应触摸事件,解决办法是在 initTN() 前加入如下代码:

view.setOnTouchListener(new OnTouchListener() {
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                switch (event.getAction()) {
                    case MotionEvent.ACTION_DOWN:
                        // 手指按下时记录必要数据,纵坐标的值都需要减去状态栏高度
                        xInView = event.getX();
                        yInView = event.getY();
                        xDownInScreen = event.getRawX();
                        yDownInScreen = event.getRawY() - getStatusBarHeight();
                        xInScreen = event.getRawX();
                        yInScreen = event.getRawY() - getStatusBarHeight();
                        break;
                    case MotionEvent.ACTION_MOVE:
                        xInScreen = event.getRawX();
                        yInScreen = event.getRawY() - getStatusBarHeight();
                        // 手指移动的时候更新小悬浮窗的位置
                        updateViewPosition();
                        break;
                    case MotionEvent.ACTION_UP:
                        // 如果手指离开屏幕时,xDownInScreen和xInScreen相等,且yDownInScreen和yInScreen相等,则视为触发了单击事件。
                        if ((xDownInScreen == xInScreen) && (yDownInScreen == yInScreen)) {
                            openBigWindow();
                        } else if (Math.abs(xDownInScreen - xInScreen) <= 10 && Math.abs(yDownInScreen - yInScreen) <= 10) {
                            openBigWindow();
                        }
                        LogUtils.i("isShow->" + FWmanager.isWindowShowing());
                        break;
                    default:
                        break;
                }
                return true;
            }
        });

3.在部分 MIUI 8开发版中,出现了拖动悬浮窗时系统崩溃的情况。定位代码时发现在调用

windowManager.updateViewLayout(this, mParams);

出现 crash,即使我将该调用

try{}
catch(Exception e){}

都捕获不到异常,推测,可能是 MIUI 8开发版中,在处理界面刷新时不允许开发者 hook 系统 toast,并刷新当前窗口。如果你知道原因的话,希望告诉我下,thx。

demo传送门

参考资料:
小米手机显示悬浮窗
仿360悬浮窗
悬浮窗小结
toast 源码分析

你可能感兴趣的:(android学习笔记,悬浮窗,权限,MIUI)