Android 直播中的悬浮小窗以及封装

4,339 阅读2分钟

悬浮小窗口在直播项目中还是比较常见的需求,在直播间打开其它界面或者退到桌面继续播放直播内容,下面是我的实现方案。

效果图

实现思路

其实很容易想到,可以通过 WindowManageraddView()实现,不过需要用户同意悬浮窗的权限,如果你的项目是应用内的悬浮窗,也可以去获取 DecorView给它添加一个View,这样就不用去获取权限了

悬浮窗权限获取

manifest添加权限

<!--    悬浮窗权限-->
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

Window设置一下type类型,这里 8.0 版本之后类型有改变这里做个判断

if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
    layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
} else {
    layoutParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT
}

权限判断

悬浮窗需要用户手动开启,这里需要给用户做一个引导

if (!Settings.canDrawOverlays(this)) { // 判断是否有权限
    val intent = Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION)
    // packageName 是应用的包名
    intent.data = Uri.parse("package:$packageName")
    // 直接打开系统的同意界面给用户操作
    startActivityForResult(intent, REQ_CODE_1000)
}

@RequiresApi(Build.VERSION_CODES.M)
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)
    if (requestCode == REQ_CODE_1000) {
        if (Settings.canDrawOverlays(this)) {
            // 用户同意了就可以进行其它操作了
        }
    }
}

处理滑动和惯性滑动

滑动的处理是通过 WindowManagerupdateViewLayout()方法,根据滑动的距离改变Window的位置,不过处理滑动前要计算一下滑动的距离是否超过了可滑动的边界,不能让我们的悬浮窗跑到屏幕外面去

惯性滑动就是模板代码了,交给GestureDetectoronFling()方法,然后通过OverScroller计算当前滑动的值

还要一点需要注意,我们的悬浮窗不仅要处理滑动时间还要处理点击事件,当用户点击窗口的时候还要回到直播页面的,当前面吧onTouchEvent交给GestureDetector去处理时点击事件就不起作用了,也就是setOnClickListener设置没有作用了,这边要手动调一下这个回调

override fun onSingleTapConfirmed(e: MotionEvent?): Boolean {
  // 触发 onClick() 回调
    performClick()
    return true
}

封装思路

悬浮窗有一些可变部分和不变部分,可变的就是悬浮窗的样式以及具体播放的操作,不变的就是滑动的操作,所以把这两部分分开,使用适配器模式连接这两部分,在适配器里面用户可以自定义布局和实现自己的业务逻辑。由于悬浮窗要作用于全局,同一时间只能有一个,所以用一个单例类FloatWindow去管理 View 的添加删除操作。

// 创建 FloatView 设置 Adapter
val floatView = FloatView(this).apply {
    setAdapter(SimpleAdapter())
}
// 把 View 设置给 Window
FloatWindow.getInstance(this).bindView(floatView)
// 移除 Window
FloatWindow.getInstance(this).removeView()

源码地址

github地址

效果图