溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android輔助權(quán)限的示例分析

發(fā)布時(shí)間:2021-08-20 13:47:05 來源:億速云 閱讀:129 作者:小新 欄目:移動(dòng)開發(fā)

這篇文章給大家分享的是有關(guān)Android輔助權(quán)限的示例分析的內(nèi)容。小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過來看看吧。

一、介紹

輔助功能服務(wù)在后臺(tái)運(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)評(pí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找到對(duì)應(yīng)的組件(注:方法返回的是集合,找到的組件不一點(diǎn)唯一,同時(shí)這里的text不單單是我們理解的 TextView 的 Text,還包括一些組件的 ContentDescription)

accessibilityNodeInfo.findAccessibilityNodeInfosByText(text)

2、根據(jù)組件 id 找到對(duì)應(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

Android輔助權(quán)限的示例分析

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);
    }
   }
  }
 }

感謝各位的閱讀!關(guān)于“Android輔助權(quán)限的示例分析”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI