Android使用AccessibilityService监听第三方app

使用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")之外,还可以获取父节点,然后根据索引查找,但是一旦界面发生改变,可能会出现索引错乱的问题

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
使用 AccessibilityService监听第三方应用播放的音乐,你需要先创建一个继承自 AccessibilityService 的服务类,并在 AndroidManifest.xml 文件中声明该服务类。在服务类中,你需要重写 onAccessibilityEvent() 方法,并在该方法中获取当前活动的窗口信息,判断当前是否有音乐播放器应用正在运行,并获取该应用的包名。 接下来,你可以通过使用 MediaSessionManager 来获取当前正在播放的媒体信息,例如音乐名称、歌手、专辑等。在获取到这些信息后,你可以将它们发送到你的服务器或进行其他处理。 以下是一个示例代码,用于监听第三方应用播放的音乐: ```java public class MusicAccessibilityService extends AccessibilityService { private MediaSessionManager mMediaSessionManager; private MediaController mMediaController; @Override protected void onServiceConnected() { super.onServiceConnected(); mMediaSessionManager = (MediaSessionManager) getSystemService(Context.MEDIA_SESSION_SERVICE); } @Override public void onAccessibilityEvent(AccessibilityEvent event) { if (event.getEventType() == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) { String packageName = event.getPackageName().toString(); if (isMusicPlayerApp(packageName)) { List<MediaController> controllers = mMediaSessionManager.getActiveSessions(null); for (MediaController controller : controllers) { if (controller != null && controller.getMetadata() != null) { String title = controller.getMetadata().getString(MediaMetadata.METADATA_KEY_TITLE); String artist = controller.getMetadata().getString(MediaMetadata.METADATA_KEY_ARTIST); String album = controller.getMetadata().getString(MediaMetadata.METADATA_KEY_ALBUM); // 将获取到的媒体信息发送到服务器或进行其他处理 } } } } } private boolean isMusicPlayerApp(String packageName) { // 判断是否为音乐播放器应用 return packageName.equals("com.android.music") || packageName.equals("com.google.android.music"); } @Override public void onInterrupt() { } } ``` 需要注意的是,使用 AccessibilityService监听第三方应用播放的音乐需要用户授权才能正常工作。你需要在启动服务时请求用户授权,并在用户同意授权后才能正常工作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

晒干的老咸鱼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值