业余时间了解了Android无障碍服务的一些有趣功能,比如微信自动抢红包、应用宝的一键安装功能等。大致原理是监听手机窗体内容变化,拿到对应的View,进行点击、长按等Touch操作,下面我们就借助 AccessibilityService 这个服务类实现模拟点击功能。

class MyAccessibilityService : AccessibilityService() {
override fun onAccessibilityEvent(event: AccessibilityEvent) {
//获取指定包名应用
val packageName = event.packageName
//只使用界面变化的监听,避免点击事件监听进入死循环
if(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED != event.eventType){
return
}
if ("com.yufs.accessibility" == packageName) {
LogUtils.e("Thread:${Thread.currentThread().name},event:${event}")
//找到对应node,开始点击
val nodeInfo = AsUtils.findNodeInfo(
this,
"com.yufs.accessibility:id/btn_click_node",
"节点模拟点击",
""
)
nodeInfo?.let {
thread {
LogUtils.e("找到节点,三秒后执行点击事件")
Thread.sleep(3000)
AsUtils.performClickNodeInfo(it)
}
}
}
}
override fun onInterrupt() {
LogUtils.e("onInterrupt==>")
}
/**
* 服务连接成功
*/
override fun onServiceConnected() {
super.onServiceConnected()
thread {
//便于设置完成后返回来看到显示效果
Thread.sleep(5000)
LogUtils.e("坐标点击:500,515")
AsUtils.click(this, 507f, 512f)
}
}
}
onAccessibilityEvent 是最重要的方法,所有界面的操作事件都会回调此方法,不限于当前应用,可以监听手机上其他应用的界面。例如界面变化时,我们打印AccessibilityEvent的信息,可以看到基本的包信息、类名等

<service
android:name=".service.MyAccessibilityService"
android:description="@string/description_in_xml"
android:exported="true"
android:enabled="true"
android:label="@string/app_name"
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_config" />
</service>
<?xml version="1.0" encoding="utf-8"?>
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:accessibilityEventTypes="typeAllMask"
android:accessibilityFeedbackType="feedbackGeneric"
android:canPerformGestures="true"
android:canRetrieveWindowContent="true"
android:canRequestTouchExplorationMode="true"
android:description="@string/description_in_xml"
android:settingsActivity="com.yufs.accessibility.MainActivity"
android:notificationTimeout="100" />
<!--
accessibilityEventTypes:可以接收的事件类型,例如界面变化、点击等。typeAllMask 接收所有,根据实际情况选择合适的类型,减少电量的消耗
accessibilityFeedbackType:事件的反馈类型,暂时不知道用作啥
canPerformGestures:是否支持手势
canRetrieveWindowContent:是否允许读取窗口中的内容
canRequestTouchExplorationMode:在这种模式下,被触摸的项目会被大声说出,用户界面可以被激活通过手势探索
description:服务的描述文字
settingsActivity:开启服务界面显示一个设置按钮,可以返回应用指定界面
notificationTimeout:事件的发送间隔事件,单位毫秒
-->

/**
* 坐标模拟点击:最低api24,即要求Android7.0以上
*/
fun click(accessibilityService: AccessibilityService, x: Float, y: Float) {
val builder = GestureDescription.Builder()
val path = Path()
path.moveTo(x, y)
path.lineTo(x, y)
builder.addStroke(GestureDescription.StrokeDescription(path, 0, 1))
val gesture = builder.build()
accessibilityService.dispatchGesture(
gesture,
object : AccessibilityService.GestureResultCallback() {
override fun onCancelled(gestureDescription: GestureDescription) {
super.onCancelled(gestureDescription)
}
override fun onCompleted(gestureDescription: GestureDescription) {
super.onCompleted(gestureDescription)
}
},
null
)
}
获取应用中view的id可以通过Android SDK中monitor.bat查看,双击运行

拿到resouce id 就可以查找view
/**
* 查找节点信息
*
* @param id
* @param text
* @param contentDescription
* @return null表示未找到
*/
fun findNodeInfo(
service: AccessibilityService,
id: String,
text: String,
contentDescription: String
): AccessibilityNodeInfo? {
if (TextUtils.isEmpty(text) && TextUtils.isEmpty(contentDescription)) {
return null
}
val nodeInfo = service.rootInActiveWindow
if (nodeInfo != null) {
val list = nodeInfo.findAccessibilityNodeInfosByViewId(id)
for (n in list) {
val nodeInfoText =
if (TextUtils.isEmpty(n.text)) "" else n.text
.toString()
val nodeContentDescription =
if (TextUtils.isEmpty(n.contentDescription)) "" else n.contentDescription
.toString()
if (TextUtils.isEmpty(text)) {
if (contentDescription == nodeContentDescription) {
return n
}
} else {
if (text == nodeInfoText) {
return n
}
}
}
}
return null
}
/**
* 点击节点
* @return true表示点击成功
*/
fun performClickNodeInfo(nodeInfo: AccessibilityNodeInfo?): Boolean {
if (nodeInfo != null) {
if (nodeInfo.isClickable) {
nodeInfo.performAction(AccessibilityNodeInfo.ACTION_CLICK)
return true
} else {
val parent = nodeInfo.parent
if (parent != null) {
val isParentClickSuccess = performClickNodeInfo(parent)
parent.recycle()
return isParentClickSuccess
}
}
}
return false
}
通过学习无障碍服务可以实现很多重复性操作,方便用户解决更多的问题,
很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo
我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("
我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b
您如何在Rails中的实时服务器上进行有效调试,无论是在测试版/生产服务器上?我试过直接在服务器上修改文件,然后重启应用,但是修改好像没有生效,或者需要很长时间(缓存?)我也试过在本地做“脚本/服务器生产”,但是那很慢另一种选择是编码和部署,但效率很低。有人对他们如何有效地做到这一点有任何见解吗? 最佳答案 我会回答你的问题,即使我不同意这种热修补服务器代码的方式:)首先,你真的确定你已经重启了服务器吗?您可以通过跟踪日志文件来检查它。您更改的代码显示的View可能会被缓存。缓存页面位于tmp/cache文件夹下。您可以尝试手动删除
我们目前正在为ROR3.2开发自定义cms引擎。在这个过程中,我们希望成为我们的rails应用程序中的一等公民的几个类类型起源,这意味着它们应该驻留在应用程序的app文件夹下,它是插件。目前我们有以下类型:数据源数据类型查看我在app文件夹下创建了多个目录来保存这些:应用/数据源应用/数据类型应用/View更多类型将随之而来,我有点担心应用程序文件夹被这么多目录污染。因此,我想将它们移动到一个子目录/模块中,该子目录/模块包含cms定义的所有类型。所有类都应位于MyCms命名空间内,目录布局应如下所示:应用程序/my_cms/data_source应用程序/my_cms/data_ty
最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路