Android輔助權(quán)限的介紹和配置完整記錄
前言
本文旨在介紹AccessibilityService如果更優(yōu)雅的使用,以及使用過程遇到的問題,該怎么解決。
一、介紹
輔助功能服務(wù)在后臺運(yùn)行,并在觸發(fā)AccessibilityEvent時(shí)由系統(tǒng)接收回調(diào)。這樣的事件表示用戶界面中的一些狀態(tài)轉(zhuǎn)換,例如,焦點(diǎn)已經(jīng)改變,按鈕被點(diǎn)擊等等?,F(xiàn)在常用于自動(dòng)化業(yè)務(wù)中,例如:微信自動(dòng)搶紅包插件,微商自動(dòng)加附近好友,自動(dòng)評論朋友,點(diǎn)贊朋友圈,甚至運(yùn)用在群控系統(tǒng),進(jìn)行刷單。
二、配置
1、新建Service并繼承AccessibilityService
/** * 核心服務(wù):執(zhí)行自動(dòng)化任務(wù) * Created by czc on 2017/6/13. */ public class TaskService_ extends AccessibilityService{ @Override public void onAccessibilityEvent(AccessibilityEvent event) { //注意這個(gè)方法回調(diào),是在主線程,不要在這里執(zhí)行耗時(shí)操作 } @Override public void onInterrupt() { } }
2、并配置AndroidManifest.xml
<service android:name=".service.TaskService" android:enabled="true" android:exported="true" android:label="@string/app_name_setting" android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"> <intent-filter> <action android:name="android.accessibilityservice.AccessibilityService"/> </intent-filter> <meta-data android:name="android.accessibilityservice" android:resource="@xml/accessibility"/> </service>
3、在res目錄下新建xml文件夾,并新建配置文件accessibility.xml
<?xml version="1.0" encoding="utf-8"?> <accessibility-service xmlns:android="http://schemas.android.com/apk/res/android" <!--監(jiān)視的動(dòng)作--> android:accessibilityEventTypes="typeAllMask" <!--提供反饋類型,語音震動(dòng)等等。--> android:accessibilityFeedbackType="feedbackGeneric" <!--監(jiān)視的view的狀態(tài),注意這里設(shè)置flagDefault會(huì)到時(shí)候部分界面狀態(tài)改變,不觸發(fā)onAccessibilityEvent(AccessibilityEvent event)的回調(diào)--> android:accessibilityFlags="flagDefault|flagRetrieveInteractiveWindows|flagIncludeNotImportantViews|flagReportViewIds|flagRequestTouchExplorationMode" <!--是否要能夠檢索活動(dòng)窗口的內(nèi)容,此設(shè)置不能在運(yùn)行時(shí)改變--> android:canRetrieveWindowContent="true" <!--功能描述--> android:description="@string/description" <!--同一事件間隔時(shí)間名--> android:notificationTimeout="100" <!--監(jiān)控的軟件包名--> android:packageNames="com.tencent.mm,com.eg.android.AlipayGphone" />
三、核心方法
1、根據(jù)界面text找到對應(yīng)的組件(注:方法返回的是集合,找到的組件不一點(diǎn)唯一,同時(shí)這里的text不單單是我們理解的 TextView 的 Text,還包括一些組件的 ContentDescription)
accessibilityNodeInfo.findAccessibilityNodeInfosByText(text)
2、根據(jù)組件 id 找到對應(yīng)的組件(注:方法返回的是集合,找到的組件不一點(diǎn)唯一,組件的 id 獲取可以通過 Android Studio 內(nèi)置的工具 monitor 獲取,該工具路徑:C:\Users\Dell\AppData\Local\Android\Sdk\tools)
accessibilityNodeInfo.findAccessibilityNodeInfosByViewId(id)
使用 Monitor 工具獲取節(jié)點(diǎn) id
Monitor選擇id
四、輔助權(quán)限判斷是否開啟
public static boolean hasServicePermission(Context ct, Class serviceClass) { int ok = 0; try { ok = Settings.Secure.getInt(ct.getContentResolver(), Settings.Secure.ACCESSIBILITY_ENABLED); } catch (Settings.SettingNotFoundException e) { } TextUtils.SimpleStringSplitter ms = new TextUtils.SimpleStringSplitter(':'); if (ok == 1) { String settingValue = Settings.Secure.getString(ct.getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); if (settingValue != null) { ms.setString(settingValue); while (ms.hasNext()) { String accessibilityService = ms.next(); if (accessibilityService.contains(serviceClass.getSimpleName())) { return true; } } } } return false; }
五、輔助的開啟方法
1.root 授權(quán)環(huán)境下,無需引導(dǎo)用戶到系統(tǒng)設(shè)置頁面開啟
public static void openServicePermissonRoot(Context ct, Class service) { String cmd1 = "settings put secure enabled_accessibility_services " + ct.getPackageName() + "/" + service.getName(); String cmd2 = "settings put secure accessibility_enabled 1"; String[] cmds = new String[]{cmd1, cmd2}; ShellUtils.execCmd(cmds, true); }
2.targetSdk 版本小于23的情況下,部分手機(jī)也可通過以下代碼開啟權(quán)限,為了兼容,最好 try...catch 以下異常
public static void openServicePermission(Context ct, Class serviceClass) { Set<ComponentName> enabledServices = getEnabledServicesFromSettings(ct, serviceClass); if (null == enabledServices) { return; } ComponentName toggledService = ComponentName.unflattenFromString(ct.getPackageName() + "/" + serviceClass.getName()); final boolean accessibilityEnabled = true; enabledServices.add(toggledService); // Update the enabled services setting. StringBuilder enabledServicesBuilder = new StringBuilder(); for (ComponentName enabledService : enabledServices) { enabledServicesBuilder.append(enabledService.flattenToString()); enabledServicesBuilder.append(":"); } final int enabledServicesBuilderLength = enabledServicesBuilder.length(); if (enabledServicesBuilderLength > 0) { enabledServicesBuilder.deleteCharAt(enabledServicesBuilderLength - 1); } Settings.Secure.putString(ct.getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES, enabledServicesBuilder.toString()); // Update accessibility enabled. Settings.Secure.putInt(ct.getContentResolver(), Settings.Secure.ACCESSIBILITY_ENABLED, accessibilityEnabled ? 1 : 0); } public static Set<ComponentName> getEnabledServicesFromSettings(Context context, Class serviceClass) { String enabledServicesSetting = Settings.Secure.getString(context.getContentResolver(), Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES); if (enabledServicesSetting == null) { enabledServicesSetting = ""; } Set<ComponentName> enabledServices = new HashSet<ComponentName>(); TextUtils.SimpleStringSplitter colonSplitter = new TextUtils.SimpleStringSplitter(':'); colonSplitter.setString(enabledServicesSetting); while (colonSplitter.hasNext()) { String componentNameString = colonSplitter.next(); ComponentName enabledService = ComponentName.unflattenFromString(componentNameString); if (enabledService != null) { if (enabledService.flattenToString().contains(serviceClass.getSimpleName())) { return null; } enabledServices.add(enabledService); } } return enabledServices; }
3.引導(dǎo)用戶到系統(tǒng)設(shè)置界面開啟權(quán)限
public static void jumpSystemSetting(Context ct) { // jump to setting permission Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); ct.startActivity(intent); }
4.結(jié)合一起,我們可以這樣開啟輔助權(quán)限
public static void openServicePermissonCompat(final Context ct, final Class service) { //輔助權(quán)限:如果root,先申請root權(quán)限 if (isAppRoot()) { if (!hasServicePermission(ct, service)) { new Thread(new Runnable() { @Override public void run() { openServicePermissonRoot(ct, service); } }).start(); } } else { try { openServicePermission(ct, service); } catch (Exception e) { e.printStackTrace(); if (!hasServicePermission(ct, service)) { jumpSystemSetting(ct); } } } }
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
- Android 操作系統(tǒng)獲取Root權(quán)限 原理詳細(xì)解析
- Android獲取ROOT權(quán)限的實(shí)例代碼
- Android權(quán)限控制之自定義權(quán)限
- Android中Root權(quán)限獲取的簡單代碼
- Android需要提升權(quán)限的操作方法
- android編程判斷應(yīng)用是否具有某個(gè)權(quán)限的方法
- Android無需申請權(quán)限撥打電話的兩種方式
- Android用代碼獲取手機(jī)root之后的最高權(quán)限
- Android 手動(dòng)獲取判斷處理權(quán)限
- 談?wù)凙ndroid6.0運(yùn)行時(shí)的權(quán)限處理
相關(guān)文章
Android開發(fā)之圖形圖像與動(dòng)畫(一)Paint和Canvas類學(xué)習(xí)
Paint類代表畫筆,用來描述圖形的顏色和風(fēng)格,如線寬,顏色,透明度和填充效果等信息;Canvas類代表畫布,通過該類提供的構(gòu)造方法,可以繪制各種圖形;感興趣的朋友可以了解下啊,希望本文對你有所幫助2013-01-01Android頁面之間進(jìn)行數(shù)據(jù)回傳的方法分析
這篇文章主要介紹了Android頁面之間進(jìn)行數(shù)據(jù)回傳的方法,結(jié)合實(shí)例形式分析了Android頁面之間進(jìn)行數(shù)據(jù)的傳遞與處理技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06Android 判斷SIM卡屬于哪個(gè)移動(dòng)運(yùn)營商詳解及實(shí)例
這篇文章主要介紹了Android 判斷SIM卡屬于哪個(gè)移動(dòng)運(yùn)營商詳解相關(guān)資料,并附實(shí)例代碼,具有一定參考價(jià)值,需要的朋友可以參考下2016-11-11Android利用ViewPager實(shí)現(xiàn)可滑動(dòng)放大縮小畫廊效果
這篇文章主要介紹了Android利用ViewPager實(shí)現(xiàn)可滑動(dòng)放大縮小畫廊效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08Android貝塞爾曲線實(shí)現(xiàn)直播點(diǎn)贊效果
這篇文章主要為大家詳細(xì)介紹了Android貝塞爾曲線實(shí)現(xiàn)直播點(diǎn)贊效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-03-03一文帶你搞清楚Android游戲發(fā)行切包資源ID那點(diǎn)事
這篇文章主要介紹了Android 解決游戲發(fā)行切包資源ID的一些問題,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2023-05-05