不需要權(quán)限的懸浮窗方案(轉(zhuǎn))

前言

全局懸浮窗是項(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)限的懸浮窗方案

效果圖

首先看下最終的效果圖

p1.gif

特性

  • 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í)如下所示:

p1.gif

1.Activity 類似于一個(gè)框架,負(fù)責(zé)容器生命周期及活動(dòng),窗口通過 Window 來管理;
2.Window 負(fù)責(zé)窗口管理(實(shí)際是子類 PhoneWindow),窗口的繪制和渲染交給 DecorView完成;
3.DecorViewView 樹的根,開發(fā)人員為 Activity 定義的 layout 將成為 DecorView 的子視圖 ContentParent 的子視圖;
4.layout.xml 是開發(fā)人員定義的布局文件,最終 inflateDecorView 的子組件;

由上我們可以想到一個(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,收藏~

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容