Android无障碍简单开发->模拟滑动点击输入等

	似乎除了input type=password的edit text 其他所有的UI都可以进行操作。直接开始。

首先配置环境。
创建一个service 清单中代码如下,就算用AS直接创建,也需要配置一些权限等东西。

 <service
            android:name=".MyAccessibilityService"
            android:enabled="true"
            android:exported="true"
            android:label="@string/accessibility_service_label"
            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" />
        </service>

label 无障碍的名字,显示在系统辅助功能->无障碍界面的标题。权限不能掉,而且必须放在这。不能和其他权限一样放在上面。resource是无障碍服务的相关配置。xml文件如下:

//我直接写的注释 复制粘贴记得删除 不然会报错
<accessibility-service
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:accessibilityEventTypes="typeAllMask"
    android:accessibilityFeedbackType="feedbackGeneric"
    android:accessibilityFlags="flagReportViewIds"
    android:canRetrieveWindowContent="true"
    android:notificationTimeout="100"
    android:description="@string/description"//无障碍的文字描述,展示在系统设置界面
    android:canPerformGestures="true"//允许模拟手势
    android:packageNames="com.tencent.mobileqq"/>//要检测的窗口或应用的包名

各个属性的内容和含义看这个无障碍参数详情
然后新建一个MyAccessibilityService继承AccessibilityService
重写onAccessibilityEvent()和onInterrupt()
在onAccessibilityEvent()中操作,在检测到目标包名在前台运行的时候会回调这个方法,然后我们在这获取根视图。

   try {
            //拿到根节点
            AccessibilityNodeInfo rootInfo = getRootInActiveWindow();
            if (rootInfo == null) {
                return;
            }
            //开始找目标节点,这里拎出来细讲,直接往下看正文
            if (rootInfo.getChildCount() != 0) {
                if (rootInfo == null || TextUtils.isEmpty(rootInfo.getClassName())) {
                    return;
                }
                //开始去找
                findByID(rootInfo, "com.tencent.mobileqq:id/chat_item_content_layout");
            }
        } catch (Exception e) {
            
        }

findByID方法如下:

 private void findByID(AccessibilityNodeInfo rootInfo, String text) {
        if (rootInfo.getChildCount() > 0) {
            for (int i = 0; i < rootInfo.getChildCount(); i++) {
                AccessibilityNodeInfo child = rootInfo.getChild(i);
                try {
                    if (child.findAccessibilityNodeInfosByViewId(text).size() > 0) {
                        for (AccessibilityNodeInfo info : child.findAccessibilityNodeInfosByViewId(text)) {
                            performClick(getClickable(info));
                            //模仿全局手势
//                        performGlobalAction(GLOBAL_ACTION_NOTIFICATIONS);
                        }
                    }
                } catch (NullPointerException e) {
                }
                findByID(child, text);//递归一直找一层层的全部遍历
            }
        }
    }

id通过DDMS看 在这里插入图片描述
在这里插入图片描述
有些控件处在listview或recyclerview下,就导致id重复,所以再用text匹配。
performClick(getClickable(info));这里的一个方法。

   //有些节点不可点击 点击交给父级甚至父级的父级...来做的。
    private AccessibilityNodeInfo getClickable(AccessibilityNodeInfo info) {
        Log.i(TAG, info.getClassName() + ": " + info.isClickable());
        if (info.isClickable()) {
            return info;//如果可以点击就返回
        } else {//不可点击就检查父级 一直递归
            return getClickable(info.getParent());
        }
    }

此外通过文本内容来找到视图

  private AccessibilityNodeInfo findByText(AccessibilityNodeInfo rootInfo, String text) {
        if (rootInfo.getChildCount() > 0) {
            for (int i = 0; i < rootInfo.getChildCount(); i++) {
                AccessibilityNodeInfo child = rootInfo.getChild(i);
                try {
                    if (child.findAccessibilityNodeInfosByText(text).size() > 0) {
                        for (AccessibilityNodeInfo info : child.findAccessibilityNodeInfosByText(text)) {
                            
                                performClick(getClickable(info));
                                return null;
                            
                        }
                    }
                } catch (NullPointerException e) {
                }
                findByText(child, text);
            }
        }
        return null;

    }

找到了视图之后,就可以做各种动作,如输入框输入:

 private void changeInput(AccessibilityNodeInfo info,String text) {  //改变editText的内容
        Bundle arguments = new Bundle();
        arguments.putCharSequence(AccessibilityNodeInfo.ACTION_ARGUMENT_SET_TEXT_CHARSEQUENCE,
                text);
        info.performAction(AccessibilityNodeInfo.ACTION_SET_TEXT, arguments);
    }

再如模拟手势:

    private void MyGesture(){//仿滑动
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
            Path path = new Path();
            path.moveTo(1000, 1000);//滑动起点
            path.lineTo(2000, 1000);//滑动终点
            GestureDescription.Builder builder = new GestureDescription.Builder();
            GestureDescription description = builder.addStroke(new GestureDescription.StrokeDescription(path, 100L, 100L)).build();
            //100L 第一个是开始的时间,第二个是持续时间
            dispatchGesture(description, new MyCallBack(), null);
        }
    }

模拟手势监听的回调:

  //模拟手势的监听
    @RequiresApi(api = Build.VERSION_CODES.N)
    private class MyCallBack extends GestureResultCallback {
        public MyCallBack() {
            super();
        }

        @Override
        public void onCompleted(GestureDescription gestureDescription) {
            super.onCompleted(gestureDescription);
           
        }

        @Override
        public void onCancelled(GestureDescription gestureDescription) {
            super.onCancelled(gestureDescription);
      
        }
    }

只是用无障碍可以实现点击 向上滑动 向下滑动,输入等等,左右滑动需要通过模拟手势来达到,但是模拟手势需要7.0+。
因为系统原因,无障碍服务每次运行完都得重新开启
快捷跳转 到无障碍设置界面

 Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
   startActivity(intent);

开启无障碍服务之后,再启动无障碍服务的Service

    startService(new Intent(this, MyAccessibilityService.class));

无障碍服务基本可以捕捉所有界面并点击,包括6.0+弹窗授权。。。很是无语。
就到这吧 有啥不对的欢迎指出!
源码文件下载

### 回答1: Android Studio是一款主要用于Android应用开发的集成开发环境。无障碍开发是指开发者在设计和开发Android应用时,考虑到用户的多样性和特殊需求,通过提供辅助功能和无障碍服务,使得应用能够更好地满足各类用户的需求,包括具备视觉、听觉或运动上的障碍的用户。 在Android Studio中,开发者可以针对无障碍开发进行一些实例操作。例如,可以在布局文件中为控件添加一些属性来支持无障碍功能。比如,可以为按钮添加属性`android:contentDescription`,用于描述按钮的功能或作用,当用户因视觉障碍而无法看到按钮时,屏读读屏软件会读出该描述信息。 此外,开发者可以通过使用无障碍服务来实现更复杂的无障碍功能。无障碍服务是Android系统提供的一种机制,可以使应用在后台监听和接收来自系统的无障碍事件,并且可以进行一些相应的操作。例如,通过无障碍服务可以捕获用户的点击事件并执行一系列自定义的操作,比如读取特定的界面元素、发送通知等等。 开发者还可以通过使用AccessibilityNodeInfo类来实现一些无障碍功能。通过此类,开发者可以获取和修改界面上的控件元素的属性、文本内容等信息。 总结来说,Android Studio提供了一整套的工具和API来支持无障碍开发开发者可以根据具体的需求灵活地进行相应的实例操作,以提供更好的用户体验和无障碍支持。 ### 回答2: Android Studio是一款强大的集成开发环境,可以用于开发Android应用程序。无障碍开发是一种为了帮助有特殊需求的用户(如视力障碍、听力障碍、身体障碍等)更好地使用应用程序而进行的开发方法。下面是一个Android Studio无障碍开发的实例。 假设我们要开发一个阅读应用程序,帮助视力障碍用户更好地阅读电子书。首先,我们需要确保应用程序的可访问性。在Android Studio中,我们可以通过向布局添加一些无障碍属性来实现这一点。比如,我们可以为文本视图添加“contentDescription”属性,描述该视图的内容。这样,屏幕阅读器可以读出文本内容给视力障碍用户听。 接下来,我们可以使用无障碍服务来提供更多的辅助功能。例如,我们可以创建一个自定义的无障碍服务,在用户滑动屏幕时,通过音频反馈告诉用户当前页面的标题。为了实现这个功能,我们需要在应用程序的主活动中注册这个无障碍服务,并在服务中实现相关的逻辑。 除了辅助功能,我们还可以使用无障碍事件来增强应用程序的可访问性。例如,我们可以在应用程序中增加一个按钮,当用户点击该按钮时,屏幕阅读器将会通知用户当前页面的标题和内容。为了实现这个功能,我们需要在按钮的点击事件监听器中发送无障碍事件,并在应用程序的主活动中注册并处理这个事件。 总之,Android Studio提供了丰富的工具和功能来支持无障碍开发。通过使用无障碍属性、无障碍服务和无障碍事件,我们可以轻松地为特殊需求的用户提供一个可访问的应用程序,帮助他们更好地使用我们的应用程序。 ### 回答3: Android Studio是一款流行的Android应用开发环境,可以提供无障碍开发的支持。无障碍开发旨在提供给残障人士使用的应用程序,使他们可以轻松地通过触摸屏幕、语音命令或其他辅助技术来访问应用程序的功能。 在Android Studio中,开发者可以通过一系列的工具和API来实现无障碍开发。其中包括: 1.辅助功能服务:Android Studio提供了一套辅助功能服务,使开发者可以为应用程序添加可访问的功能。这些功能包括屏幕阅读器、文本到语音转换、语音识别等。 2.可访问性属性:开发者可以使用可访问性属性来标记应用程序中的视图元素,以便辅助技术可以正确地解释和操作这些元素。例如,开发者可以为按钮添加可访问性属性,以便屏幕阅读器可以读出按钮的标签和状态。 3.键盘快捷键:Android Studio提供一系列的键盘快捷键,使开发者可以在无需使用鼠标的情况下进行开发。这对于视觉障碍的开发者尤为重要。 4.无障碍检查工具:Android Studio还提供了一个无障碍检查工具,可以帮助开发者检查应用程序中是否遵循无障碍开发的最佳实践。开发者可以使用这个工具来发现并修复潜在的无障碍问题。 无障碍开发实例可以包括为视觉障碍的用户优化应用程序中的界面,使用屏幕阅读器测试应用程序的可访问性,为语音输入提供支持,等等。通过使用Android Studio的无障碍开发功能,开发者可以更好地满足残障人士的需求,为他们提供更好的应用体验。
评论 16
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值