美文网首页
Android跨进程模拟触屏事件

Android跨进程模拟触屏事件

作者: babybus_hentai | 来源:发表于2016-03-23 18:59 被阅读3102次

    跨进程模拟触屏事件的作用##

    有很多在一个Activity中实现虚拟触摸的方法,但是无法做到跨进程虚拟触摸。无论是Google提供的Monkey还是MonkeyRunner都不能很好的脱离PC进行虚拟触碰,更别说写一个后台进程实现一些tricky的虚拟触碰。

    模拟触屏事件也可以做许多事情,例如侦听支付宝解锁的手势加以模拟之类的,而且属于系统层面的模拟是应用无法分辨的,在此只介绍原理和技术。

    触屏事件##

    在Android中,所有的传感器在发生时间时都会想Linux层对应的传感器文件发送一个event。整个过程可以看成是传感器在收到操作后会产生一个硬件中断,硬件中断转换为相应的软中断消息,软中断会使系统调用sendevent,之后在通过sendevent做处理跟操作。本文只讨论如何利用软件的方式模拟硬件产生中断

    关于getevent###

    在Android工具中,包含一个getevent,侦听系统接收到的event,可以直接通过adb命令行getevent直接查看

    华为荣耀手机运行getevent会出现如下信息:
    <pre>
    add device 1: /dev/input/event0
    name: "mtk-kpd"
    could not get driver version for /dev/input/mouse0, Not a typewriter
    add device 2: /dev/input/event3
    name: "cyttsp4_mt"
    add device 3: /dev/input/event2
    name: "hwmdata"
    add device 4: /dev/input/event1
    name: "ACCDET"
    </pre>
    很显然,上面的每个adb device就相当于一个传感器,对应一个/dev/input/even[i]文件

    然后尝试点击屏幕,窗口中会显示10个event:
    <pre>
    /dev/input/event3: 0001 014a 00000001
    /dev/input/event3: 0003 0039 0000005c
    /dev/input/event3: 0003 0035 000001a5
    /dev/input/event3: 0003 0036 00000270
    /dev/input/event3: 0003 003a 00000056
    /dev/input/event3: 0000 0000 00000000
    /dev/input/event3: 0003 0039 ffffffff
    /dev/input/event3: 0000 0000 00000000
    /dev/input/event3: 0001 014a 00000000
    /dev/input/event3: 0000 0000 00000000
    </pre>
    尝试多次可能出现消息条数不同的结果,这与点击的力度与时间有关

    十六进制内容不利于阅读和理解可以调用getevent -l做同样的操作,返回的信息会以常量的名字出现,下面是触屏事件的消息内容:
    <pre>
    /dev/input/event3: EV_KEY BTN_TOUCH DOWN
    /dev/input/event3: EV_ABS ABS_MT_TRACKING_ID 00000060
    /dev/input/event3: EV_ABS ABS_MT_POSITION_X 000001fa
    /dev/input/event3: EV_ABS ABS_MT_POSITION_Y 000002db
    /dev/input/event3: EV_ABS ABS_MT_PRESSURE 0000003d
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    /dev/input/event3: EV_ABS ABS_MT_POSITION_X 000001f9
    /dev/input/event3: EV_ABS ABS_MT_POSITION_Y 000002d9
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    /dev/input/event3: EV_ABS ABS_MT_POSITION_X 000001f8
    /dev/input/event3: EV_ABS ABS_MT_POSITION_Y 000002d8
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    /dev/input/event3: EV_ABS ABS_MT_POSITION_X 000001f7
    /dev/input/event3: EV_ABS ABS_MT_POSITION_Y 000002d7
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    /dev/input/event3: EV_ABS ABS_MT_TRACKING_ID ffffffff
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    /dev/input/event3: EV_KEY BTN_TOUCH UP
    /dev/input/event3: EV_SYN SYN_REPORT 00000000
    </pre>
    粗略观察可以发现,消息主要包含点击的坐标、力度、时间、范围、类型等信息。EV_SYN是中断的类型,可以理解为硬件中断对应的消息类型

    关于sendevent###

    首先先尝试一下最简单的电源按钮,试图唤醒手机。
    根据getevent的数据,将其数值翻译成10进制,对应的KEY_POWER是0x0074,也就是10进制下的116。在adb shell中输入如下命令(并不需要很快连续输入,可以慢慢来):
    <pre>
    shell@hammerhead:/ $ sendevent /dev/input/event0 1 116 1
    shell@hammerhead:/ $ sendevent /dev/input/event0 0 0 0
    shell@hammerhead:/ $ sendevent /dev/input/event0 1 116 0
    shell@hammerhead:/ $ sendevent /dev/input/event0 0 0 0
    </pre>
    如果尝试了,会发现只输入前2行手机屏幕就会亮。但是如果不继续输入后面2行,系统会认为用户一直按住了电源键,因此这时在物理按下一次电源键,手机屏幕不会关闭,需要再按两次才行。

    分析高级操作##

    google的官方文档中Touch Device Driver Requirements 这一节介绍了Multi-touch设备的event相关的一些参数,显然现在所有的Android设备都支持多点触控,以下为简要翻译:
    <pre>
    ABS_MT_POSITION_X: (必须) 报告触碰的X坐标。
    ABS_MT_POSITION_Y: (必须) 报告触碰的Y坐标。
    ABS_MT_PRESSURE: (可选) 报告触碰的压力大小或者信号强度。
    ABS_MT_TOUCH_MAJOR: (可选) 报告触碰的代表性区域, 或者触碰的最长的尺寸。
    ABS_MT_TOUCH_MINOR: (可选) 报告触碰的最小的尺寸。 如果ABS_MT_TOUCH_MAJOR报告的是区域测量,则不使用。
    ABS_MT_WIDTH_MAJOR: (可选) 报告触碰本身的代表性区域,或者触碰本身的最大尺寸。 如果触碰的尺寸不知道则不使用。
    ABS_MT_WIDTH_MINOR: (可选) 报告触碰本身的最小尺寸。 如果触碰的尺寸不知道则不使用。
    ABS_MT_ORIENTATION: (可选) 报告触碰的方向。
    ABS_MT_DISTANCE: (可选) 报告触碰本身和表面的距离。
    ABS_MT_TOOL_TYPE: (可选) 报告触碰是MT_TOOL_FINGER还是MT_TOOL_PEN.
    ABS_MT_TRACKING_ID: (可选) 报告触碰的跟踪ID。跟踪ID是一个非负的任意整数,用来分辨多个同时的操作。例如,当多个手指触碰设备,在手指还在屏幕上时每个手指绑定一个独立的跟踪ID,当手指离开屏幕后,跟踪ID可能被重新使用。
    ABS_MT_SLOT: (可选) 报告触碰的slot ID,在使用Linux多点触碰协议B的情况下使用。查看Linnux多点触碰协议文档获取详细信息
    </pre>
    我的目标是实现一个能够点击任意坐标与长按的库

    单点点击event###

    单点点击相对容易,消息内容较少
    <pre>
    ABS_MT_TRACKING_ID
    ABS_MT_POSITION_X
    ABS_MT_POSITION_Y
    SYN_REPORT
    </pre>
    作为一次event提交,压力大小跟区域参数是可选的

    单点释放event###

    <pre>
    ABS_MT_TRACKING_ID -1
    SYN_REPOR
    </pre>
    单点释放只需要提交两个event即可,不需要提交坐标等信息

    多点点击event###

    多点触碰涉及到ABS_MT_SLOT的概念,这个并不是很难理解,在每次指定触碰的同时,增加一个ABS_MT_SLOT。在释放前,也要注明ABS_MT_SLOT。并且在默认情况下,ABS_MT_SLOT为0。这样也就能理解单点触碰所在的ABS_MT_SLOT一直为0,单点触碰只是多点触碰的一个特例。
    需要注意的是,ABS_MT_SLOT一定要再ABS_MT_TRACKING_ID前定义,不然会出问题,使用的是之前的slot。
    <pre>
    ABS_MT_SLOT
    ABS_MT_TRACKING_ID
    ABS_MT_POSITION_X
    ABS_MT_POSITION_Y
    SYN_REPORT
    </pre>

    多点释放event###

    ABS_MT_SLOT
    ABS_MT_TRACKING_ID -1
    SYN_REPORT
    多点释放需要依次释放ABS_MT_SLOT,范围为0-9,切记要全部释放

    滑动操作的event###

    滑动操作实际上是间隔取样某一个ABS_MT_TRACKING_ID的多次触碰。只需先指定ABS_MT_TRACKING_ID跟所在的ABS_MT_SLOT,再改变X,Y坐标(如果某个坐标没变则不需要改变),然后同步信号量即可

    总结###

    拥有了以上的知识,实现起来的步骤就很清晰了。具体各个常量名和数值的关系,可以调用getevent -lp和getevent -p进行对比查看。本文只是做了简单的总结与整理

    本文参考:http://azard.me/blog/2015/06/13/android-cross-app-touch-event-injection/

    相关文章

      网友评论

          本文标题:Android跨进程模拟触屏事件

          本文链接:https://www.haomeiwen.com/subject/anzllttx.html