在執(zhí)行自動(dòng)化服務(wù)的流程中,我們其實(shí)并不希望被用戶的操作中斷流程,所以有什么方法在用戶點(diǎn)擊自動(dòng)化操作的過(guò)程中,避免用戶再次操作呢轧飞?那就是開(kāi)啟一個(gè)全局透明的懸浮窗衅鹿,進(jìn)行屏蔽觸摸事件。
一过咬、懸浮窗
其實(shí)一開(kāi)始塘安,我是想當(dāng)然的跟以前一樣,開(kāi)啟一個(gè)全屏的透明的懸浮窗援奢,進(jìn)行遮罩的作用,但是發(fā)現(xiàn)忍捡,設(shè)置 Type 為 TYPE_TOAST 或者 TYPE_SYSTEM_ALERT 這樣的懸浮窗某些類型的不同集漾,會(huì)導(dǎo)致不單單把用戶的操作屏蔽了,甚至窗口的一些狀態(tài)改變也屏蔽的砸脊,導(dǎo)致輔助權(quán)限的 onAccessibilityEvent() 方法不回調(diào)具篇,于是去找官方文檔,查找相關(guān)懸浮窗的 Type 類型設(shè)置凌埂。然后被我找到這個(gè)屬性值的 Type :
LayoutParams.TYPE_ACCESSIBILITY_OVERLAY
我們?cè)賮?lái)看官方解釋:
Windows that are overlaid only by a connected AccessibilityService for interception of user interactions without changing the windows an accessibility service can introspect. In particular, an accessibility service can introspect only windows that a sighted user can interact with which is they can touch these windows or can type into these windows. For example, if there is a full screen accessibility overlay that is touchable, the windows below it will be introspectable by an accessibility service even though they are covered by a touchable window.
雖然官方寫的一大堆驱显,但是我們大概能 get 到里面的意思,其實(shí)就是設(shè)置為這個(gè)類型的懸浮窗瞳抓,能夠使輔助功能繼續(xù)響應(yīng)相關(guān)窗口與內(nèi)容的變化埃疫。經(jīng)測(cè)試,果然設(shè)置這個(gè)類型的懸浮窗孩哑,可以一方面屏蔽用戶的觸摸事件栓霜,另一方繼續(xù)響應(yīng)自動(dòng)點(diǎn)擊的相關(guān)操作。
public void createFullScreenView(Context context) {
WindowManager windowManager = getWindowManager(context);
if (fullScreenView == null) {
fullScreenView = new FloatWindowFullScreenView(context);
LayoutParams fullScreenParams = new LayoutParams();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP_MR1) {
fullScreenParams.type = LayoutParams.TYPE_ACCESSIBILITY_OVERLAY;
} else {
fullScreenParams.type = LayoutParams.TYPE_TOAST;
}
fullScreenParams.format = PixelFormat.TRANSLUCENT;
fullScreenParams.flags |= WindowManager.LayoutParams.FLAG_FULLSCREEN
| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| LayoutParams.FLAG_KEEP_SCREEN_ON
| WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL;
fullScreenParams.gravity = Gravity.CENTER;
windowManager.addView(fullScreenView, fullScreenParams);
}
}
值得注意的是横蜒,這個(gè)屬性是在 android 5.1 之后加入進(jìn)來(lái)胳蛮,對(duì)于之前的版本,經(jīng)測(cè)試丛晌,使用 Toast 類型仅炊,也能執(zhí)行相關(guān)操作,至于為什么 5.1 之后不繼續(xù)使用Toast類型呢澎蛛,這里面涉及到懸浮窗的開(kāi)啟問(wèn)題了抚垄,可自行百度懸浮窗的開(kāi)啟相關(guān)文章。
二瓶竭、懸浮窗的 Context
我們一般開(kāi)啟懸浮窗的過(guò)程中督勺,Context 的傳遞我們使用的 Service 或者 Activity,不過(guò)如果設(shè)置為 TYPE_ACCESSIBILITY_OVERLAY 的懸浮窗斤贰,是只能傳入你繼承自 AccessibilityService 的服務(wù)(Context智哀,否則會(huì)報(bào) Is Activity Running 這個(gè)異常,那如何在這個(gè)服務(wù)里面開(kāi)啟懸浮窗呢荧恍?我是使用廣播的形式去開(kāi)啟的:
// 注冊(cè)廣播接聽(tīng)者
IntentFilter filter = new IntentFilter();
filter.addAction(Const.ACTION_SHOW_COVER_VIEW);
filter.addAction(Const.ACTION_SHOW_SMALL_VIEW);
filter.addAction(Const.ACTION_SET_COVER_VIEW_TIPS);
registerReceiver(mReceiver, filter);
....省略其他代碼
private BroadcastReceiver mReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (action.equals(Const.ACTION_SHOW_COVER_VIEW)) {
if (!FloatWindowManager.getInstance().isFullWindowShowing()) {
FloatWindowManager.getInstance().createFullScreenView(TaskService.this);
}
String toast = intent.getStringExtra(Const.EXTRA_WINDOW_TOAST);
if (!StringUtils.isEmpty(toast)) {
FloatWindowManager.getInstance().showToast(toast);
}
} else if (action.equals(Const.ACTION_SHOW_SMALL_VIEW)) {
if (!FloatWindowManager.getInstance().isSmallWindowShowing()) {
FloatWindowManager.getInstance().createSmallWindow(TaskService.this);
}
}else if (action.equals(Const.ACTION_SET_COVER_VIEW_TIPS)) {
if (FloatWindowManager.getInstance().isFullWindowShowing()) {
FloatWindowManager.getInstance().showTipst(intent.getStringExtra("tips"));
}
}
}
};
三瓷叫、懸浮窗的實(shí)現(xiàn)
在懸浮窗的UI設(shè)計(jì)上屯吊,我們需要將其設(shè)置為透明背景,這樣對(duì)用戶是無(wú)感的摹菠,整個(gè)自動(dòng)化流程中盒卸,其實(shí)是相當(dāng)于屏幕有個(gè)用戶看不到的“保護(hù)罩”在確保著你的自動(dòng)化業(yè)務(wù)不被“打擾”。在布局上次氨,我們需要實(shí)現(xiàn)最外層的根布局的點(diǎn)擊事件蔽介,這樣在用戶點(diǎn)擊屏幕的時(shí)候,彈窗 Toast 友好提示用戶:自動(dòng)化業(yè)務(wù)正在執(zhí)行煮寡,請(qǐng)停止業(yè)務(wù)才能操作虹蓄。
同時(shí)懸浮窗提供“停止”按鈕,可以終止業(yè)務(wù)并關(guān)閉全屏透明懸浮窗幸撕。
四薇组、使用場(chǎng)景
部分軟件需要開(kāi)啟許多權(quán)限才能保證軟件的正常使用,例如市面上的某鎖屏軟件坐儿,他們需要涉及相當(dāng)多的權(quán)限律胀,如果一個(gè)個(gè)讓用戶去開(kāi)啟,可能找不到對(duì)應(yīng)的權(quán)限怎么開(kāi)啟貌矿,于是他們把這個(gè)流程簡(jiǎn)化成腳本炭菌,只要用戶開(kāi)啟輔助權(quán)限,則跳轉(zhuǎn)到權(quán)限開(kāi)啟流程站叼,自動(dòng)到權(quán)限頁(yè)面娃兽,把例如:開(kāi)機(jī)自啟動(dòng)權(quán)限,讀取通知尽楔,獲取位置等權(quán)限開(kāi)啟投储。當(dāng)然這個(gè)過(guò)程是被一個(gè)界面遮蓋了的,用戶是看不到執(zhí)行了什么操作的(這也暴露android的安全性問(wèn)題)阔馋。