使用AccessibilityService主要有以下几步:
1、自定义Service继承自AccessibilityService;
public class MyAccessibilityService extends AccessibilityService {
private static final String TAG = "MyAccessibilityService";
private static final String LISTEN_PACKAGENAME = "com.xxx.xxx";
@Override
public void onCreate() {
super.onCreate();
}
//服务连接成功时的回调
@Override
protected void onServiceConnected() {
super.onServiceConnected();
Log.e(TAG, "MyAccessibilityService onServiceConnected");
}
//接收到系统发送AccessibilityEvent时的回调
@Override
public void onAccessibilityEvent(AccessibilityEvent accessibilityEvent) {
Log.e(TAG, "MyAccessibilityService onAccessibilityEvent: " + accessibilityEvent);
String packageName = accessibilityEvent.getPackageName().toString();
if (LISTEN_PACKAGENAME.equals(packageName)){
Log.e(TAG, "MyAccessibilityService packageName true");
int eventType = accessibilityEvent.getEventType();
switch (eventType){
case AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED:
Log.e(TAG, "MyAccessibilityService TYPE_VIEW_ACCESSIBILITY_FOCUSED");
break;
case AccessibilityEvent.TYPE_VIEW_CLICKED:
try {
Log.e(TAG, "MyAccessibilityService TYPE_VIEW_CLICKED");
String text = accessibilityEvent.getText().toString();
Log.e(TAG, "MyAccessibilityService TYPE_VIEW_CLICKED text: " + text);
AccessibilityNodeInfo ani = accessibilityEvent.getSource();
//找到当前点击按钮的父级
AccessibilityNodeInfo parentAni = ani.getParent();
//找到父级的父级
AccessibilityNodeInfo grandParentAni = parentAni.getParent();
//根据控件的文字标签查找对应的控件,找到"xxx"文字控件对应的node
List<AccessibilityNodeInfo> anis = grandParentAni.findAccessibilityNodeInfosByText("xxx");
//根据控件的id查找对应的控件,packageName:id/idName
List<AccessibilityNodeInfo> anis = grandParentAni.findAccessibilityNodeInfosByViewId("com.xxx.xxx:id/idName");
} catch (Exception e) {
e.printStackTrace();
Log.e(TAG, e.getMessage());
}
break;
case AccessibilityEvent.TYPE_VIEW_FOCUSED:
Log.e(TAG, "MyAccessibilityService TYPE_VIEW_FOCUSED");
break;
case AccessibilityEvent.TYPE_VIEW_TEXT_CHANGED:
Log.e(TAG, "MyAccessibilityService TYPE_VIEW_TEXT_CHANGED");
break;
case AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED:
try {
Log.e(TAG, "MyAccessibilityService TYPE_WINDOW_STATE_CHANGED");
} catch (Exception e) {
e.printStackTrace();
Log.e(TAG, e.getMessage());
}
break;
}
}else{
Log.e(TAG, "MyAccessibilityService packageName error: " + packageName);
}
}
//服务中断时的回调
@Override
public void onInterrupt() {
Log.e(TAG, "MyAccessibilityService onInterrupt");
}
}
2、在AndroidManifest.xml进行AccessibilityService的注册配置
<!-- android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"是为了确保只有系统可以绑定该服务 -->
<service
android:name=".MyAccessibilityService"
android:label="xxx"
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/accessible_service_config_test" />
</service>
3、在res/xml文件夹下新建accessible_service_config_test.xml,内容如下:
<?xml version="1.0" encoding="utf-8"?>
<!-- typeAllMask:接收所有事件。
==========窗口事件相关(常用)==========
typeWindowStateChanged:监听窗口状态变化,比如打开一个popupWindow,dialog,Activity切换等等。
typeWindowContentChanged:监听窗口内容改变,比如根布局子view的变化。
typeWindowsChanged:监听屏幕上显示的系统窗口中的事件更改。 此事件类型只应由系统分派。
typeNotificationStateChanged:监听通知变化,比如notifacation和toast。
============View事件相关==========
typeViewClicked:监听view点击事件。
typeViewLongClicked:监听view长按事件。
typeViewFocused:监听view焦点事件。
typeViewSelected:监听AdapterView中的上下文选择事件。
typeViewTextChanged:监听EditText的文本改变事件。
typeViewHoverEnter、typeViewHoverExit:监听view的视图悬停进入和退出事件。
typeViewScrolled:监听view滚动,此类事件通常不直接发送。
typeViewTextSelectionChanged:监听EditText选择改变事件。
typeViewAccessibilityFocused:监听view获得可访问性焦点事件。
typeViewAccessibilityFocusCleared:监听view清除可访问性焦点事件。
============手势事件相关==========
typeGestureDetectionStart、typeGestureDetectionEnd:监听手势开始和结束事件。
typeTouchInteractionStart、typeTouchInteractionEnd:监听用户触摸屏幕事件的开始和结束。
typeTouchExplorationGestureStart、typeTouchExplorationGestureEnd:监听触摸探索手势的开始和结束。
android:description :辅助功能描述,描述该辅助功能用来干嘛的
android:packageNames :指定辅助功能监听的包名,不指定表示监听所有应用
android:accessibilityEventTypes:辅助功能处理事件类型,一般配置为typeAllMask表示接收所有事件
android:accessibilityFlags:辅助功能查找截点方式,一般配置为flagDefault默认方式。
android:accessibilityFeedbackType:操作相应按钮以后辅助功能给用户的反馈类型,包括声音,震动等。
android:notificationTimeout:相应时间设置
android:canRetrieveWindowContent:是否允许辅助功能获得窗口的节点信息,为了能正常实用辅助功能,请务必保持该项配置为true
-->
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:accessibilityEventTypes="typeAllMask"
android:accessibilityFeedbackType="feedbackGeneric"
android:accessibilityFlags="flagReportViewIds"
android:canRetrieveWindowContent="true"
android:description="xxx" //这个决定在无障碍系统设置中显示的名字
android:notificationTimeout="10"
android:canPerformGestures="true"
android:packageNames="com.xxxx.xxx" /> //这个是要监控的包名,可以监控多个包名,用","隔开
4、判断无障碍权限是否开启
public boolean isAccessibilitySettingsOn(Context mContext, Class<? extends AccessibilityService> clazz) {
int accessibilityEnabled = 0;
final String service = mContext.getPackageName() + "/" + clazz.getCanonicalName();
try {
accessibilityEnabled = Settings.Secure.getInt(mContext.getApplicationContext().getContentResolver(),
Settings.Secure.ACCESSIBILITY_ENABLED);
} catch (Settings.SettingNotFoundException e) {
e.printStackTrace();
}
TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':');
if (accessibilityEnabled == 1) {
String settingValue = Settings.Secure.getString(mContext.getApplicationContext().getContentResolver(),
Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);
if (settingValue != null) {
mStringColonSplitter.setString(settingValue);
while (mStringColonSplitter.hasNext()) {
String accessibilityService = mStringColonSplitter.next();
if (accessibilityService.equalsIgnoreCase(service)) {
return true;
}
}
}
}
return false;
}
如果没有开启,通过startActivity(new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS));可以跳转到设置页进行开启
注意:
1、如果无障碍里面的权限未开启,监听功能失效,所以如果监听无反应,请检查是否正常开启权限;
2、查找节点除了findAccessibilityNodeInfosByText("xxx")和findAccessibilityNodeInfosByViewId("com.xxx.xxx:id/idName")之外,还可以获取父节点,然后根据索引查找,但是一旦界面发生改变,可能会出现索引错乱的问题