前言
全局懸浮窗是項(xiàng)目中的一個(gè)常見(jiàn)需求,目前比較常見(jiàn)的實(shí)現(xiàn)是將要懸浮的View
添加到WindowManager
中
這種方案的主要痛點(diǎn)在于需要用戶申請(qǐng)TYPE_SYSTEM_ALERT
權(quán)限,并且需要用戶去設(shè)置中手動(dòng)打開(kāi),使用起來(lái)很不方便督弓,同時(shí)需要申請(qǐng)權(quán)限可能會(huì)勸退用戶.
針對(duì)這種情況下面介紹一種不需要權(quán)限的懸浮窗方案
效果圖
首先看下最終的效果圖
特性
- 1.不需要申請(qǐng)權(quán)限澳盐,可以直接打開(kāi)懸浮窗蜗顽,使用便捷
- 2.支持自定義布局柔滔,自定義顯示樣式,自定義初始顯示位置
- 3.支持拖拽,可自動(dòng)吸附到屏幕邊緣
- 4.可過(guò)濾不需要顯示懸浮窗的黑名單界面
- 5.支持自定義點(diǎn)擊事件,可支持展開(kāi)折疊等功能
- 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)用,使用起來(lái)非常方便
1.初始化
EasyFloat
.layout(R.layout.layout_float_view)
.blackList(mutableListOf(ThirdActivity::class.java))
.layoutParams(initLayoutParams())
.listener {
initListener(it)
}
.show(this)
復(fù)制代碼
如上所示:
1.通過(guò)layout
指定自定義布局
2.通過(guò)blackList
指定不展示懸浮窗界面
3.通過(guò)layoutParams
指定初始展示位置
4.通過(guò)listener
處理自定義點(diǎn)擊事件
2.銷毀懸浮窗
EasyFloat.dismiss(this)
復(fù)制代碼
直接調(diào)用dismiss
銷毀即可
主要原理
我們都知道届榄,當(dāng)我們需要設(shè)置布局的時(shí)候浅乔,是通過(guò)setContentView
設(shè)置的
而setContentView
實(shí)際上是將我們的布局添加到了DecoreView
上,布局層級(jí)如下所示:
1.Activity
類似于一個(gè)框架,負(fù)責(zé)容器生命周期及活動(dòng)铝条,窗口通過(guò) Window
來(lái)管理靖苇;
2.Window
負(fù)責(zé)窗口管理(實(shí)際是子類 PhoneWindow
),窗口的繪制和渲染交給 DecorView
完成班缰;
3.DecorView
是 View
樹(shù)的根贤壁,開(kāi)發(fā)人員為 Activity
定義的 layout
將成為 DecorView
的子視圖 ContentParent
的子視圖;
4.layout.xml
是開(kāi)發(fā)人員定義的布局文件埠忘,最終 inflate
為 DecorView
的子組件脾拆;
由上我們可以想到一個(gè)方案:
我們?cè)?code>Activity onStart時(shí),將要懸浮的View
添加到ContentParent
上就可以實(shí)現(xiàn)不需要權(quán)限的懸浮窗了
當(dāng)然我們還需要注意以下幾點(diǎn)
1.因?yàn)槲覀冃枰诙鄠€(gè)頁(yè)面展示懸浮窗莹妒,可以通過(guò)ActivityLifecycleCallbacks
監(jiān)聽(tīng)所有Activity
的生命周期,onStart
時(shí)添加名船,onStop
時(shí)移除
2.因?yàn)橐诙鄠€(gè)頁(yè)面共享狀態(tài),所以應(yīng)該有一個(gè)單例類管理View
旨怠,做到只創(chuàng)建一個(gè)View
包帚,頁(yè)面切換時(shí)只做添加與移除
3.因?yàn)橐砑拥?code>ContentParent中,持有了Activity
的引用运吓,所以要注意處理內(nèi)存泄漏的問(wèn)題渴邦,在項(xiàng)目中我們使用了弱引用來(lái)防止內(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è)開(kāi)源框架的過(guò)程中,主要借鑒了EnFloatingView的一些思路
并在其基礎(chǔ)上進(jìn)行了一定的封裝拘哨,優(yōu)化了API
調(diào)用并解決了滑動(dòng)沖突等一些問(wèn)題
項(xiàng)目地址
EasyFloat
開(kāi)源不易谋梭,如果項(xiàng)目對(duì)你有所幫助,歡迎點(diǎn)贊,Star
,收藏~