悬浮小窗口在直播项目中还是比较常见的需求,在直播间打开其它界面或者退到桌面继续播放直播内容,下面是我的实现方案。
实现思路
其实很容易想到,可以通过 WindowManager
的 addView()
实现,不过需要用户同意悬浮窗的权限,如果你的项目是应用内的悬浮窗,也可以去获取 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)) {
// 用户同意了就可以进行其它操作了
}
}
}
处理滑动和惯性滑动
滑动的处理是通过 WindowManager
的updateViewLayout()
方法,根据滑动的距离改变Window
的位置,不过处理滑动前要计算一下滑动的距离是否超过了可滑动的边界,不能让我们的悬浮窗跑到屏幕外面去
惯性滑动就是模板代码了,交给GestureDetector
的onFling()
方法,然后通过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()