前言
全局懸浮窗是項(xiàng)目中的一個(gè)常見需求,目前比較常見的實(shí)現(xiàn)是將要懸浮的View添加到WindowManager中
這種方案的主要痛點(diǎn)在于需要用戶申請(qǐng)TYPE_SYSTEM_ALERT權(quán)限,并且需要用戶去設(shè)置中手動(dòng)打開,使用起來很不方便,同時(shí)需要申請(qǐng)權(quán)限可能會(huì)勸退用戶.
針對(duì)這種情況下面介紹一種不需要權(quán)限的懸浮窗方案
效果圖
首先看下最終的效果圖
特性
- 1.不需要申請(qǐng)權(quán)限,可以直接打開懸浮窗,使用便捷
- 2.支持自定義布局,自定義顯示樣式,自定義初始顯示位置
- 3.支持拖拽,可自動(dòng)吸附到屏幕邊緣
- 4.可過濾不需要顯示懸浮窗的黑名單界面
- 5.支持自定義點(diǎn)擊事件,可支持展開折疊等功能
- 6.
API鏈?zhǔn)秸{(diào)用,使用簡(jiǎn)潔優(yōu)雅
集成
第 1 步:在工程的 build.gradle 中添加:
allprojects {
repositories {
...
mavenCentral()
}
}
復(fù)制代碼
第2步:在應(yīng)用的 build.gradle 中添加:
dependencies {
implementation 'io.github.shenzhen2017:easyfloat:1.0.0'
}
復(fù)制代碼
使用
API鏈?zhǔn)秸{(diào)用,使用起來非常方便
1.初始化
EasyFloat
.layout(R.layout.layout_float_view)
.blackList(mutableListOf(ThirdActivity::class.java))
.layoutParams(initLayoutParams())
.listener {
initListener(it)
}
.show(this)
復(fù)制代碼
如上所示:
1.通過layout指定自定義布局
2.通過blackList指定不展示懸浮窗界面
3.通過layoutParams指定初始展示位置
4.通過listener處理自定義點(diǎn)擊事件
2.銷毀懸浮窗
EasyFloat.dismiss(this)
復(fù)制代碼
直接調(diào)用dismiss銷毀即可
主要原理
我們都知道,當(dāng)我們需要設(shè)置布局的時(shí)候,是通過setContentView設(shè)置的
而setContentView實(shí)際上是將我們的布局添加到了DecoreView上,布局層級(jí)如下所示:
1.Activity 類似于一個(gè)框架,負(fù)責(zé)容器生命周期及活動(dòng),窗口通過 Window 來管理;
2.Window 負(fù)責(zé)窗口管理(實(shí)際是子類 PhoneWindow),窗口的繪制和渲染交給 DecorView完成;
3.DecorView 是 View 樹的根,開發(fā)人員為 Activity 定義的 layout 將成為 DecorView 的子視圖 ContentParent 的子視圖;
4.layout.xml 是開發(fā)人員定義的布局文件,最終 inflate 為 DecorView 的子組件;
由上我們可以想到一個(gè)方案:
我們?cè)?code>Activity onStart時(shí),將要懸浮的View添加到ContentParent上就可以實(shí)現(xiàn)不需要權(quán)限的懸浮窗了
當(dāng)然我們還需要注意以下幾點(diǎn)
1.因?yàn)槲覀冃枰诙鄠€(gè)頁面展示懸浮窗,可以通過ActivityLifecycleCallbacks監(jiān)聽所有Activity的生命周期,onStart時(shí)添加,onStop時(shí)移除
2.因?yàn)橐诙鄠€(gè)頁面共享狀態(tài),所以應(yīng)該有一個(gè)單例類管理View,做到只創(chuàng)建一個(gè)View,頁面切換時(shí)只做添加與移除
3.因?yàn)橐砑拥?code>ContentParent中,持有了Activity的引用,所以要注意處理內(nèi)存泄漏的問題,在項(xiàng)目中我們使用了弱引用來防止內(nèi)存泄漏
部分代碼如下:
object EasyFloat : Application.ActivityLifecycleCallbacks {
override fun onActivityStarted(activity: Activity) {
FloatingView.get().attach(activity)
}
override fun onActivityStopped(activity: Activity) {
FloatingView.get().detach(activity)
}
fun show(activity: Activity) {
initShow(activity)
activity.application.registerActivityLifecycleCallbacks(this)
}
fun dismiss(activity: Activity) {
FloatingView.get().remove()
FloatingView.get().detach(activity)
activity.application.unregisterActivityLifecycleCallbacks(this)
}
}
復(fù)制代碼
總結(jié)
特別鳴謝
在實(shí)現(xiàn)這個(gè)開源框架的過程中,主要借鑒了EnFloatingView的一些思路
并在其基礎(chǔ)上進(jìn)行了一定的封裝,優(yōu)化了API調(diào)用并解決了滑動(dòng)沖突等一些問題
項(xiàng)目地址
EasyFloat
開源不易,如果項(xiàng)目對(duì)你有所幫助,歡迎點(diǎn)贊,Star,收藏~