Android AccessibilityService无障碍服务(二)

简介: 当服务未开启时,快速的跳转到开启服务的界面。if (!OpenAccessibilitySettingHelper.isAccessibilitySettingsOn(this, AccessibilitySampleService.
  1. 当服务未开启时,快速的跳转到开启服务的界面。
if (!OpenAccessibilitySettingHelper.isAccessibilitySettingsOn(this,
            AccessibilitySampleService.class.getName())){// 判断服务是否开启
          OpenAccessibilitySettingHelper.jumpToSettingPage(this);// 跳转到开启页面
        }else {
          Toast.makeText(this, "服务已开启", Toast.LENGTH_SHORT).show();
        }

用到的方法具体实现:

/**
 * 开启无障碍服务帮助类
 * Created by mazaiting on 2017/8/18.
 */
public class OpenAccessibilitySettingHelper {

  /**
   * 跳转到无障碍服务设置页面
   * @param context 设备上下文
   */
  public static void jumpToSettingPage(Context context){
    Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    context.startActivity(intent);
  }

  /**
   * 判断是否有辅助功能权限
   * @return true 已开启
   *          false 未开启
   */
  public static boolean isAccessibilitySettingsOn(Context context,String className){
    if (context == null){
      return false;
    }
    ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
    List<ActivityManager.RunningServiceInfo> runningServices =
        activityManager.getRunningServices(100);// 获取正在运行的服务列表
    if (runningServices.size()<0){
      return false;
    }
    for (int i=0;i<runningServices.size();i++){
      ComponentName service = runningServices.get(i).service;
      if (service.getClassName().equals(className)){
        return true;
      }
    }
    return false;
  }
}

2.模拟点击,创建模拟点击的Activity为AccessibilityNormalSampleActivity,并在AndroidManifest.xml将AccessibilityNormalSampleActivity与AccessibilitySampleService配置在同一个进程,若不在同一进程,则获取到的AccessibilityService与AccessibilityEvent为空。

android:process=":BackgroundService"

配置文件为:

<!-- 注册辅助功能服务 -->
    <service
        android:name=".service.AccessibilitySampleService"
        android:enabled="true"
        android:exported="true"
        android:label="@string/accessibility_tip"
        android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE"
        android:process=":BackgroundService">

      <!-- android:label="@string/accessibility_tip" 在设置中显示的文字 -->
      <intent-filter>
        <action android:name="android.accessibilityservice.AccessibilityService" />
      </intent-filter>
      <!-- 通过xml文件完成辅助功能相关配置,也可以在onServiceConnected中动态配置 -->
      <meta-data
          android:name="android.accessibilityservice"
          android:resource="@xml/accessibility_config" />
    </service>

    <activity android:name=".ui.AccessibilityNormalSampleActivity"
        android:process=":BackgroundService"></activity>

AccessibilityNormalSampleActivity界面布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/activity_accessibility_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:orientation="vertical">

  <CheckBox
      android:id="@+id/normal_sample_checkbox"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="复选框开关"/>

  <RadioButton
      android:id="@+id/normal_sample_radiobutton"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_marginTop="10dp"
      android:text="单选按钮"/>

  <ToggleButton
      android:id="@+id/normal_sample_togglebutton"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_marginTop="10dp"/>

  <Button
      android:id="@+id/normal_sample_back"
      android:layout_marginTop="20dp"
      android:text="退出本页面"
      android:layout_width="match_parent"
      android:layout_height="wrap_content" />
</LinearLayout>

3.创建一个单例类来控制模拟点击AccessibilityOperator。

/**
 * 控制无障碍服务
 * Created by mazaiting on 2017/8/18.
 */
public class AccessibilityOperator {
  private static final String TAG = "AccessibilityOperator";
  private static AccessibilityOperator mInstance;
  private AccessibilityOperator(){}
  public static AccessibilityOperator getInstance() {
    if (mInstance == null){
      synchronized (AccessibilityOperator.class){
        if (mInstance == null){
          mInstance = new AccessibilityOperator();
        }
      }
    }
    return mInstance;
  }
}
  1. 创建一个用来模拟点击的界面
    在要点击的Activity中创建一个Handler来执行延时消息,创建AccessibilityOperator对象,在onCreate方法中获取单例对象。
  private Handler mHandler = new Handler(Looper.getMainLooper());
  private AccessibilityOperator accessibilityOperator;

  @Override protected void onCreate(Bundle savedInstanceState) {
      .....// 省略布局填充
      accessibilityOperator = AccessibilityOperator.getInstance();
    }

并在onResume方法中进行模拟点击。此处只是调用AccessibilityOperator中的方法,因此直接贴代码:

@Override protected void onResume() {
    super.onResume();
    // 执行延时任务
    clickText();
  }

  /**
   * 按文本点击
   */
  private void clickText() {
    clickTextItem("复选框",1);
    clickTextItem("单选按钮",2);
    clickTextItem("关闭",3);
    clickTextItem("退出本页面",4);
  }

/**
   * 文本单个延时点击
   * @param text 文本内容
   * @param num 延时倍数
   */
  private void clickTextItem(final String text,int num) {
    mHandler.postDelayed(new Runnable() {
      @Override public void run() {
        final boolean isSuccess = accessibilityOperator.clickText(text);
        runOnUiThread(new Runnable() {
          @Override public void run() {
            popToast(isSuccess, text);
          }
        });
      }
    },2000*num);
  }

  /**
   * 弹出吐司
   * @param isSuccess
   * @param msg
   */
  private void popToast(boolean isSuccess, String msg) {
    if (isSuccess) {
      Toast.makeText(this, msg + "点击成功", Toast.LENGTH_SHORT).show();
    } else {
      Toast.makeText(this, msg + "点击失败", Toast.LENGTH_SHORT).show();
    }
  }
  1. 在accessibilityOperator.clickText(text)文本时,系统会先调用AccessibilitySampleService中的onAccessibilityEvent方法,因此在AccessibilityOperator创建一个updateEvent方法,来为AccessibilityService服务与AccessibilityEvent事件赋值。
  private AccessibilityService mAccessibilityService;
  private AccessibilityEvent mAccessibilityEvent;
  /**
   * 更新事件
   * @param service
   * @param event
   */
  public void updateEvent(AccessibilityService service, AccessibilityEvent event) {
    if (mAccessibilityService == null && service != null){
      mAccessibilityService = service;
    }
    if (event != null){
      mAccessibilityEvent = event;
    }
  }
  1. 对AccessibilityService与AccessibilityEvent赋值之后就可以正常使用了。clickText方法的完整内容代码:
  /**
   * 根据Text搜索所有符合条件的节点,模糊搜索方式
   * @param text
   * @return
   */
  public boolean clickText(String text) {
    AccessibilityNodeInfo nodeInfo = getRootNodeInfo();
    if (nodeInfo!=null){
      List<AccessibilityNodeInfo> nodeInfos =
          nodeInfo.findAccessibilityNodeInfosByText(text);
      return performClick(nodeInfos);
    }
    return false;
  }

/**
   * 获取根节点
   * @return
   */
  private AccessibilityNodeInfo getRootNodeInfo() {
    Log.e(TAG, "getRootNodeInfo: ");
    AccessibilityEvent curEvent = mAccessibilityEvent;
    AccessibilityNodeInfo nodeInfo = null;
    if (Build.VERSION.SDK_INT >= 16){
      if (mAccessibilityService!=null){
        // 获得窗体根节点
        nodeInfo = mAccessibilityService.getRootInActiveWindow();
      }
    }else {
      nodeInfo = curEvent.getSource();
    }
    return nodeInfo;
  }

  /**
   * 模拟点击
   * @param nodeInfos
   * @return true 成功; false 失败。
   */
  private boolean performClick(List<AccessibilityNodeInfo> nodeInfos) {
    if (nodeInfos!=null && !nodeInfos.isEmpty()){// 判断是否非空
      AccessibilityNodeInfo nodeInfo;
      for (int i=0;i<nodeInfos.size();i++){
        nodeInfo = nodeInfos.get(i);// 获得要点击的View
        // 进行模拟点击
        if (nodeInfo.isEnabled()){// 如果可以点击
          return nodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK);
        }
      }
    }
    return false;
  }

getRootNodeInfo()返回的AccessibilityNodeInfo可以对它进行遍历,查询它的子节点

    AccessibilityNodeInfo nodeInfo = getRootNodeInfo();
    if (nodeInfo!=null){
      for (int i=0;i<nodeInfo.getChildCount();i++){
        AccessibilityNodeInfo child = nodeInfo.getChild(i);
        Log.e(TAG, "clickText: "+child.toString());
      }
    }
目录
相关文章
|
2月前
|
安全 Java 网络安全
Android远程连接和登录FTPS服务代码(commons.net库)
Android远程连接和登录FTPS服务代码(commons.net库)
34 1
|
3月前
|
XML Android开发 数据格式
🌐Android国际化与本地化全攻略!让你的App走遍全球无障碍!🌍
在全球化背景下,实现Android应用的国际化与本地化至关重要。本文以一款旅游指南App为例,详细介绍如何通过资源文件拆分与命名、适配布局与方向、处理日期时间及货币格式、考虑文化习俗等步骤,完成多语言支持和本地化调整。通过邀请用户测试并收集反馈,确保应用能无缝融入不同市场,提升用户体验与满意度。
125 3
|
3月前
|
JavaScript 前端开发 Android开发
让Vite+Vue3项目在Android端离线打开(不需要起服务)
让Vite+Vue3项目在Android端离线打开(不需要起服务)
126 10
|
3月前
|
调度 Android开发 UED
Android经典实战之Android 14前台服务适配
本文介绍了在Android 14中适配前台服务的关键步骤与最佳实践,包括指定服务类型、请求权限、优化用户体验及使用WorkManager等。通过遵循这些指南,确保应用在新系统上顺畅运行并提升用户体验。
276 6
|
3月前
|
安全 API 开发工具
Android平台RTMP推送|轻量级RTSP服务如何实现麦克风|扬声器声音采集切换
Android平台扬声器播放声音的采集,在无纸化同屏等场景下,意义很大,早期低版本的Android设备,是没法直接采集扬声器audio的(从Android 10开始支持),所以,如果需要采集扬声器audio,需要先做系统版本判断,添加相应的权限。
|
3月前
|
编解码 开发工具 Android开发
Android平台实现屏幕录制(屏幕投影)|音频播放采集|麦克风采集并推送RTMP或轻量级RTSP服务
Android平台屏幕采集、音频播放声音采集、麦克风采集编码打包推送到RTMP和轻量级RTSP服务的相关技术实现,做成高稳定低延迟的同屏系统,还需要有配套好的RTMP、RTSP直播播放器
|
4月前
|
数据处理 开发工具 数据安全/隐私保护
Android平台RTMP推送|轻量级RTSP服务|GB28181接入之文字、png图片水印的精进之路
本文探讨了Android平台上推流模块中添加文字与PNG水印的技术演进。自2015年起,为了满足应急指挥及安防领域的需求,逐步发展出三代水印技术:第一代为静态文字与图像水印;第二代实现了动态更新水印内容的能力,例如实时位置与时间信息;至第三代,则优化了数据传输效率,直接使用Bitmap对象传递水印数据至JNI层,减少了内存拷贝次数。这些迭代不仅提升了用户体验和技术效率,也体现了开发者追求极致与不断创新的精神。
|
4月前
|
数据采集 编解码 开发工具
Android平台实现无纸化同屏并推送RTMP或轻量级RTSP服务(毫秒级延迟)
一个好的无纸化同屏系统,需要考虑的有整体组网、分辨率、码率、实时延迟、音视频同步和连续性等各个指标,做容易,做好难
|
4月前
|
监控 开发工具 Android开发
Android平台实现RTSP拉流转发至轻量级RTSP服务
为满足Android平台上从外部RTSP摄像头拉流并提供轻量级RTSP服务的需求,利用大牛直播SDK实现了相关功能。SDK支持开始与停止拉流、音频视频数据回调处理及RTSP服务的启动与发布等操作。拉流仅需将未解码数据回调,对性能影响小。音频和视频数据经由特定接口传递给发布端进行处理。此外,SDK还提供了获取RTSP会话数量的功能。此方案适用于监控和巡检等低延迟应用场景,并支持二次水印添加等功能。
|
4月前
|
编解码 API 开发工具
Android平台轻量级RTSP服务模块二次封装版调用说明
本文介绍了Android平台上轻量级RTSP服务模块的二次封装实践,旨在简化开发流程,让开发者能更专注于业务逻辑。通过`LibPublisherWrapper`类提供的API,可在应用中轻松初始化RTSP服务、配置视频参数(如分辨率、编码类型)、启动与停止RTSP服务及流发布,并获取RTSP会话数量。此外,还展示了如何处理音频和视频数据的采集与推送。最后,文章提供了从启动服务到销毁资源的完整示例,帮助开发者快速集成实时流媒体功能。