Android WindowManger实现桌面悬浮窗功能

 更新时间:2023年04月28日 14:15:23   作者:_小马快跑  
这篇文章主要介绍了Android WindowManger实现桌面悬浮窗功能,他们基本都是在Activity之上显示的,如果想实现在桌面显示的悬浮窗效果,需要用到WindowManager来实现了,需要的朋友可以参考下

如果想实现一个在桌面显示的悬浮窗,用DialogPopupWindowToast等已经不能实现了,他们基本都是在Activity之上显示的,如果想实现在桌面显示的悬浮窗效果,需要用到WindowManager来实现了。

效果图

使用WindowManager实现

  • 添加一个悬浮窗:
        sys_view = new SmallWindowView(mContext);
        sys_view.setText("50%");
        sys_view.setOnTouchListener(this);
        windowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
        int screenWidth = 0, screenHeight = 0;
        if (windowManager != null) {
            //获取屏幕的宽和高
            Point point = new Point();
            windowManager.getDefaultDisplay().getSize(point);
            screenWidth = point.x;
            screenHeight = point.y;
            layoutParams = new WindowManager.LayoutParams();
//            layoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
//            layoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
            layoutParams.width = 200;
            layoutParams.height = 200;
            //设置type
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                //26及以上必须使用TYPE_APPLICATION_OVERLAY   @deprecated TYPE_PHONE
                layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
            } else {
                layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
            }
            //设置flags
            layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                    | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED;
            layoutParams.gravity = Gravity.START | Gravity.TOP;
            //背景设置成透明
            layoutParams.format = PixelFormat.TRANSPARENT;
            layoutParams.x = screenWidth;
            layoutParams.y = screenHeight / 2;
            //将View添加到屏幕上
            windowManager.addView(sys_view, layoutParams);
        }
  • 更新悬浮窗位置:
windowManager.updateViewLayout(sys_view, layoutParams);
  • 关闭悬浮窗:
windowManager.removeView(sys_view);

通过上面的代码就可以实现一个桌面悬浮窗功能了。

注意:在6.0以上,需要在Manifest.xml中声明 <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />权限并且在开启悬浮窗时动态判断权限,如果没有此权限需要跳到设置页面去设置,看下官方文档的说明:

SYSTEM_ALERT_WINDOW

分析

1、添加悬浮窗: 通过Context.getSystemService(Context.WINDOW_SERVICE)获得一个WindowManager(以下简称VM), VM是外界访问Window的入口,ActivityDialogToast等其视图都是依附在Window之上的,WindowView的直接管理者,VM继承自ViewManager,其添加、刷新、删除方法也是来自ViewManager:

public interface ViewManager
{   public void addView(View view, ViewGroup.LayoutParams params);
    public void updateViewLayout(View view, ViewGroup.LayoutParams params);
    public void removeView(View view);
}

VM有一个静态内部类WindowManager.LayoutParamsWindow的各个属性在这个内部类中设置:

  • LayoutParams.TYPE 如果TargetSdkVersion<26 ,那么可以直接使用LayoutParams.TYPE_PHONE或者LayoutParams.TYPE_SYSTEM_ALERT,在TargetSdkVersion>=26时,TYPE_PHONETYPE_SYSTEM_ALERT都已经废弃了,需要使用TYPE_APPLICATION_OVERLAY来标识TYPE
  • LayoutParams.FLAGS FLAGS表示Window的属性,通过FLAGS可以控制Window的显示特性,常用的几个特性: LayoutParams.FLAG_NOT_TOUCH_MODAL : 使用了此标识,可以将点击事件传递到悬浮窗以外的区域,反之其他区域的Window将接收不到事件。 LayoutParams.FLAG_NOT_FOCUSABLE : 表示悬浮窗Window不需要获取焦点,也不需要获取各种输入事件,事件会直接传递给下层的具有焦点的Window LayoutParams.FLAG_SHOW_WHEN_LOCKED : 此模式可以让Window显示在锁屏的界面上
  • LayoutParams.FORMAT 悬浮窗Window的背景格式,一般设置成PixelFormat.TRANSPARENT透明即可
  • LayoutParams.X & LayoutParams.Y 悬浮窗Window在屏幕上的坐标值,可以根据X&Y的值来刷新Window在屏幕上的位置
  • LayoutParams.Width & LayoutParams.Height 悬浮窗Window的宽度和高度

2、更新悬浮窗位置: 在ViewOnTouchEvent中或OnTouch中更新layoutParams.xlayoutParams.y的值并通过windowManager.updateViewLayout()重新设置悬浮窗Window在屏幕中的位置,如下:

    @Override
    public boolean onTouch(View v, MotionEvent event) {
        int mInScreenX = (int) event.getRawX();
        int mInScreenY = (int) event.getRawY();
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mLastX = (int) event.getRawX();
                mLastY = (int) event.getRawY();
                break;
            case MotionEvent.ACTION_MOVE:
                layoutParams.x += mInScreenX - mLastX;
                layoutParams.y += mInScreenY - mLastY;
                mLastX = mInScreenX;
                mLastY = mInScreenY;
                windowManager.updateViewLayout(sys_view, layoutParams);
                break;
            case MotionEvent.ACTION_UP:
                break;
        }
        return true;
    }

3、删除悬浮窗: 删除比较简单,直接调用windowManager.removeView(view)viewWindow中删除即可。

问题

6.0以上使用时,需要动态申请该悬浮窗权限,如下:

//判断有没有悬浮窗权限,没有去申请
if(!Settings.canDrawOverlays(context)){
     Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                Uri.parse("package:" + context.getPackageName()));
     context.startActivityForResult(intent, REQUEST_CODE);
}
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        switch (requestCode) {
            case REQUEST_CODE:
                if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) return;
                if (!WindowUtil.canOverDraw(this)) {
                    toast("悬浮窗权限未开启,请在设置中手动打开");
                    return;
                }
                WindowController.getInstance().showThumbWindow();
                break;
        }
    }

通过Settings.canDrawOverlays(context)判断是否有悬浮窗权限,如果没有,跳转到设置页面去设置,并在onActivityResult ()中得到申请结果,看似很完美,但在实际测试中,发现在8.0以上的手机上有问题,即使在设置中同意了权限,8.0的手机Settings.canDrawOverlays(context)总是返回false,不过在关闭页面重新调用此方法时,又返回的true,感觉是有一定的延迟,google了一下,发现别人同样遇到了这个问题,貌似已经给google提交了bug单,可以看此博客: http://paskov.vmsoft-bg.com/settings-candrawoverlays-allays-returns-false-on-android-o/ ,不过博客中的解决方法用我的8.0手机(HUAWEI MATE10)依然不起作用,暂时还没深入研究,有解决此问题的还希望不吝赐教。

以上例子的源码地址:https://github.com/crazyqiang

参考

【1】developer.android.com/reference/a…

到此这篇关于Android WindowManger实现桌面悬浮窗功能的文章就介绍到这了,更多相关Android桌面悬浮窗内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 微信公众平台开发入门教程(SAE方倍工作室)

    微信公众平台开发入门教程(SAE方倍工作室)

    在这篇微信公众平台开发教程中,我们假定你已经有了PHP语言程序、MySQL数据库、计算机网络通讯、及HTTP/XML/CSS/JS等基础
    2014-05-05
  • Android 基于agora 开发视频会议的代码

    Android 基于agora 开发视频会议的代码

    这篇文章主要介绍了Android 基于agora 开发视频会议,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • 详解Android 视频播放时停止后台运行的方法

    详解Android 视频播放时停止后台运行的方法

    这篇文章主要介绍了详解Android 视频播放时停止后台运行的方法的相关资料,需要的朋友可以参考下
    2017-06-06
  • Gradle配置教程之自定义APK名称与输出路径

    Gradle配置教程之自定义APK名称与输出路径

    Gradle是一个基于JVM的富有突破性构建工具,下面这篇文章主要给大家介绍了关于Gradle配置教程之自定义APK名称与输出路径的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2018-03-03
  • Android中监听短信的两种方法

    Android中监听短信的两种方法

    这篇文章主要介绍了Android中监听短信的两种方法,本文讲解了监听广播、采用观察方法,监听短信数据库两种方法,需要的朋友可以参考下
    2015-04-04
  • Android 自定义Switch开关按钮的样式实例详解

    Android 自定义Switch开关按钮的样式实例详解

    本文主要讲的是在Android原生Switch控件的基础上进行样式自定义,内容很简单,但是在实现的过程中还是遇到了一些问题,在此记录下来,需要的朋友参考下吧
    2017-12-12
  • 如何将Android Studio卸载干净

    如何将Android Studio卸载干净

    这篇文章主要介绍了如何将Android Studio卸载干净,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-09-09
  • Flutter路由fluro引入配置和使用的具体方法

    Flutter路由fluro引入配置和使用的具体方法

    Flutter本身提供了路由机制,本文主要介绍了Flutter fluro配置使用,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-09-09
  • 实例讲解Android中的View类以及自定义View控件的方法

    实例讲解Android中的View类以及自定义View控件的方法

    这篇文章主要介绍了Android中的View类以及自定义View控件的方法,讲解了如何继承View类并且展示了一个对View进行重绘的例子,需要的朋友可以参考下
    2016-04-04
  • Android实现音乐视频播放

    Android实现音乐视频播放

    这篇文章主要为大家详细介绍了Android实现音乐视频播放,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-05-05

最新评论