关于在android平台上捕获事件的资料互联网上已经铺天盖地,但个人觉得可用性都不太大,大部仅
针对特定设备,而对于其他设备引导性不强,故整理处本文,望对初学者有个有力的帮助:
getevent 旨在获取android设备的事件信息,具体参考详细用法(本人亦初学者一枚,无法深入解释)
sendevent 则可以向设备发送模拟事件,其中包括touch和keypress
详细用法如下:
- Usage: getevent [-t] [-n] [-s switchmask] [-S] [-v [mask]] [-d] [-p] [-i] [-l] [-q] [-c count] [-r] [device]
- -t: show time stamps
- -n: don’t print newlines
- -s: print switch states for given bits
- -S: print all switch states
- -v: verbosity mask (errs=1, dev=2, name=4, info=8, vers=16, pos. events=32, props=64)
- -d: show HID descriptor, if available
- -p: show possible events (errs, dev, name, pos. events)
- -i: show all device info and possible events
- -l: label event types and names in plain text
- -q: quiet (clear verbosity mask)
- -c: print given number of events then exit
- -r: print rate events are received
- Usage: sendevent <device> <type> <code> <value>
首先,adb shell进入android设备,运行getevent命令得到如下信息,为各类事件的驱动设备,每部
硬件可能都不一样,可以说无规律可循,个人觉得掠过吧
- add device 1: /dev/input/event1
- name: “pmic8xxx_pwrkey”
- add device 2: /dev/input/event3
- name: “apq8064-tabla-snd-card Headset Jack”
- add device 3: /dev/input/event2
- name: “apq8064-tabla-snd-card Button Jack”
- add device 4: /dev/input/event5
- name: “gpio-keys”
- could not get driver version for /dev/input/mice, Not a typewriter
- add device 5: /dev/input/event0
- name: “atmel_mxt_ts”
- add device 6: /dev/input/event4
- name: “mhl_rcp”
在点击设备屏幕之后得到:
- /dev/input/event0: 0003 0039 000002a5
- /dev/input/event0: 0003 0030 00000004
- /dev/input/event0: 0003 0035 0000017b
- /dev/input/event0: 0003 0036 000001cf
- /dev/input/event0: 0003 003a 0000001c
- /dev/input/event0: 0000 0000 00000000
- /dev/input/event0: 0003 0039 ffffffff
- /dev/input/event0: 0000 0000 00000000
很难辨认,加-l参数后得到 ( getevent -l )
- /dev/input/event0: EV_ABS ABS_MT_TRACKING_ID 000002a6
- /dev/input/event0: EV_ABS ABS_MT_TOUCH_MAJOR 00000004
- /dev/input/event0: EV_ABS ABS_MT_POSITION_X 0000017b
- /dev/input/event0: EV_ABS ABS_MT_POSITION_Y 000001cf
- /dev/input/event0: EV_ABS ABS_MT_PRESSURE 0000001c
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
- /dev/input/event0: EV_ABS ABS_MT_TRACKING_ID ffffffff
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
格式为 device: type code value,即 设备、输入设备类型、按键扫描码、附加码,具体定义可从kernel/include/linux/input.h中获得,至于这个头文件,途径之一是从google官网源码中获取:
type: 输入设备类型,在手机系统中经常使用的键盘(keyboard)和小键盘(kaypad)属于按键设
备EV_KEY,轨迹球属于相对设备EV_REL,触摸屏属于绝对设备EV_ABS
code: 按键扫描码,区别于ASCII码和SDK中KeyEvent的键码
value: 附加码,1/0 down/up
第一行:可理解为一次touch的开始
第2行:可理解为点击开始
第3行:触摸点x坐标
第4行:触摸点y坐标
第5行:可理解为触摸点大小
第6行:事件同步(点击结束)
第7行:一次touch结束
第8行:事件同步(事件结束)
一次touch此8行是必须的,如果是longTouch呢,在touch的基础上,重复若干次第2~6行,即看起
来可能是这样子:
- /dev/input/event0: EV_ABS ABS_MT_TRACKING_ID 000002a6
- /dev/input/event0: EV_ABS ABS_MT_TOUCH_MAJOR 00000004
- /dev/input/event0: EV_ABS ABS_MT_POSITION_X 0000017b
- /dev/input/event0: EV_ABS ABS_MT_POSITION_Y 000001cf
- /dev/input/event0: EV_ABS ABS_MT_PRESSURE 0000001c
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
- /dev/input/event0: EV_ABS ABS_MT_TOUCH_MAJOR 00000004
- /dev/input/event0: EV_ABS ABS_MT_POSITION_X 0000017b
- /dev/input/event0: EV_ABS ABS_MT_POSITION_Y 000001cf
- /dev/input/event0: EV_ABS ABS_MT_PRESSURE 0000001c
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
- /dev/input/event0: EV_ABS ABS_MT_TOUCH_MAJOR 00000004
- /dev/input/event0: EV_ABS ABS_MT_POSITION_X 0000017b
- /dev/input/event0: EV_ABS ABS_MT_POSITION_Y 000001cf
- /dev/input/event0: EV_ABS ABS_MT_PRESSURE 0000001c
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
- ……
- /dev/input/event0: EV_ABS ABS_MT_TRACKING_ID ffffffff
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
如果是drag呢,在longTouch的基础上,xy坐标从起点到终点是渐变的,其他可认为一样。
对于按键,HOME为例,一目了然:
- /dev/input/event0: 0001 0066 00000001
- /dev/input/event0: 0000 0000 00000000
- /dev/input/event0: 0001 0066 00000000
- /dev/input/event0: 0000 0000 00000000
- /dev/input/event0: EV_KEY KEY_HOME DOWN
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
- /dev/input/event0: EV_KEY KEY_HOME UP
- /dev/input/event0: EV_SYN SYN_REPORT 00000000
所以清楚了之后,使用sendevent进行事件模拟就很轻松了,关键是从input.h中获取按键对应的扫
描码,也许每个版本的系统中input.h的内容都稍有不同,这个很头疼。
值得注意的是使用getevent获取的数值都是16进制的,而sendevent使用的是10进制的,需要进行
转换。
如点击坐标: 120,254
- sendevent /dev/input/event1 0003 0057 00000000 <—事件开始
- sendevent /dev/input/event1 0003 0048 00000010 <—点击开始
- sendevent /dev/input/event1 0003 0058 00000070 <— 触摸范围
- sendevent /dev/input/event1 0003 0053 00000120 <— x坐标
- sendevent /dev/input/event1 0003 0054 00000254 <— y坐标
- sendevent /dev/input/event1 0000 0000 00000000 <—点击结束(同步)
- sendevent /dev/input/event1 0003 0057 4294967295 <—事件结束
- sendevent /dev/input/event1 0000 0000 00000000 <—事件同步
翻译为:
type code value
EV_ABS ABS_MT_TRACKING_ID 00000000 <—事件开始
EV_ABS ABS_MT_TOUCH_MAJOR 00000010 <—点击开始
EV_ABS ABS_MT_PRESSURE 00000070 <— 触摸范围
EV_ABS ABS_MT_POSITION_X 00000120 <— x坐标
EV_ABS ABS_MT_POSITION_Y 00000254 <— y坐标
EV_SYN SYN_REPORT 00000000 <—点击结束(同步)
EV_ABS ABS_MT_TRACKING_ID 4294967295 <—事件结束
EV_SYN SYN_REPORT 00000000 <—事件同步
对于longTouch,重复几次上面提到的步骤,而drag再弄个坐标渐变。
按键方面(MENU) DOWN:1 UP:0
- sendevent /dev/input/event0 0001 0102 0000000001
- sendevent /dev/input/event0 0000 0000 0000000000
- sendevent /dev/input/event0 0001 0102 0000000000
- sendevent /dev/input/event0 0000 0000 0000000000
- EV_KEY KEY_MENU DOWN
- EV_SYN SYN_REPORT 00000000
- EV_KEY KEY_MENU UP
- EV_SYN SYN_REPORT 00000000
此外还可以模拟可见字符按键。
到这里是否觉得getevent好用好理解,但sendevent却超级麻烦。没关系,如果真不想用sendevent
模拟事件的话,可以转向input命令。
个人觉得这个input命令是个重量级的东东,调用时间非常长,貌似低系统版本和高系统版本的input提
供的功能还不一样,但它毕竟好用啊,如下:
- usage: input …
- input text <string>
- input keyevent <key code number or name>
- input tap <x> <y>
- input swipe <x1> <y1> <x2> <y2>
可见除了longTouch无法模拟之外,其他的都很简便了
输入文本: input text abcdefg
按键 : input keyevent KEYCODE_MENU
点击 : input tap 100 300
拖拽 : input swipe 100 600 500 600
①对于文本输入,以下字符需要加 \ 进行转义:
` ‘ ” ~ # & ( ) | \ ; < >
②对于longTouch,使用input如何模拟?
通过使用adb shell getevent,可以得到如下设备操作后的指令,具体表示的含义,可以参考网上很多的文章,这里就不再叙述
这里的0035和0036分别表示屏幕上的X坐标和Y坐标,后面的值表示具体的坐标点,通过16进制转换可以得到值1142和1527,而我手机的屏幕分辨率为800*480的,很明显两个点是完全不匹配,那如何转换呢?
在cmd中输入adb shell getevent -p
找到您的屏幕设备中0035和0036的含义,我的手机中可以看到x的值min为0,max为1172,y的值min为0,max为1900。
这样就找到您的设备的坐标具体大小值,这里计算公式如下:
x=(x-xmin)*480/(xmax-xmin) ;
y=(y-ymin)*800/(ymax-ymin);
这样算出的坐标值就跟您手机的屏幕分辨率相匹配了。
该公式是通过查看androidvncserver源码,然后反向推出的公式,目前碰到的手机min都为0,未测试过不过为0的情况,如果不为0的情况这个公式是错误的,请各位同学补充。