Android & iOS · 2014-05-22

android测试之getevent/sendevent

关于在android平台上捕获事件的资料互联网上已经铺天盖地,但个人觉得可用性都不太大,大部仅

针对特定设备,而对于其他设备引导性不强,故整理处本文,望对初学者有个有力的帮助:

getevent 旨在获取android设备的事件信息,具体参考详细用法(本人亦初学者一枚,无法深入解释)
sendevent 则可以向设备发送模拟事件,其中包括touch和keypress

详细用法如下:

  1. Usage: getevent [-t] [-n] [-s switchmask] [-S] [-v [mask]] [-d] [-p] [-i] [-l] [-q] [-c count] [-r] [device]
  2.     -t: show time stamps
  3.     -n: don’t print newlines
  4.     -s: print switch states for given bits
  5.     -S: print all switch states
  6.     -v: verbosity mask (errs=1, dev=2, name=4, info=8, vers=16, pos. events=32, props=64)
  7.     -d: show HID descriptor, if available
  8.     -p: show possible events (errs, dev, name, pos. events)
  9.     -i: show all device info and possible events
  10.     -l: label event types and names in plain text
  11.     -q: quiet (clear verbosity mask)
  12.     -c: print given number of events then exit
  13.     -r: print rate events are received
  14. Usage: sendevent <device> <type> <code> <value>

首先,adb shell进入android设备,运行getevent命令得到如下信息,为各类事件的驱动设备,每部

硬件可能都不一样,可以说无规律可循,个人觉得掠过吧

  1. add device 1: /dev/input/event1
  2.   name:     “pmic8xxx_pwrkey”
  3. add device 2: /dev/input/event3
  4.   name:     “apq8064-tabla-snd-card Headset Jack”
  5. add device 3: /dev/input/event2
  6.   name:     “apq8064-tabla-snd-card Button Jack”
  7. add device 4: /dev/input/event5
  8.   name:     “gpio-keys”
  9. could not get driver version for /dev/input/mice, Not a typewriter
  10. add device 5: /dev/input/event0
  11.   name:     “atmel_mxt_ts”
  12. add device 6: /dev/input/event4
  13.   name:     “mhl_rcp”

在点击设备屏幕之后得到:

  1. /dev/input/event0: 0003 0039 000002a5
  2. /dev/input/event0: 0003 0030 00000004
  3. /dev/input/event0: 0003 0035 0000017b
  4. /dev/input/event0: 0003 0036 000001cf
  5. /dev/input/event0: 0003 003a 0000001c
  6. /dev/input/event0: 0000 0000 00000000
  7. /dev/input/event0: 0003 0039 ffffffff
  8. /dev/input/event0: 0000 0000 00000000

很难辨认,加-l参数后得到 ( getevent -l )

  1. /dev/input/event0: EV_ABS       ABS_MT_TRACKING_ID     000002a6
  2. /dev/input/event0: EV_ABS       ABS_MT_TOUCH_MAJOR   00000004
  3. /dev/input/event0: EV_ABS       ABS_MT_POSITION_X      0000017b
  4. /dev/input/event0: EV_ABS       ABS_MT_POSITION_Y      000001cf
  5. /dev/input/event0: EV_ABS       ABS_MT_PRESSURE         0000001c
  6. /dev/input/event0: EV_SYN       SYN_REPORT                  00000000
  7. /dev/input/event0: EV_ABS       ABS_MT_TRACKING_ID     ffffffff
  8. /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行,即看起
来可能是这样子:

  1. /dev/input/event0: EV_ABS       ABS_MT_TRACKING_ID     000002a6
  2. /dev/input/event0: EV_ABS       ABS_MT_TOUCH_MAJOR   00000004
  3. /dev/input/event0: EV_ABS       ABS_MT_POSITION_X      0000017b
  4. /dev/input/event0: EV_ABS       ABS_MT_POSITION_Y      000001cf
  5. /dev/input/event0: EV_ABS       ABS_MT_PRESSURE         0000001c
  6. /dev/input/event0: EV_SYN       SYN_REPORT                  00000000
  7. /dev/input/event0: EV_ABS       ABS_MT_TOUCH_MAJOR   00000004
  8. /dev/input/event0: EV_ABS       ABS_MT_POSITION_X      0000017b
  9. /dev/input/event0: EV_ABS       ABS_MT_POSITION_Y      000001cf
  10. /dev/input/event0: EV_ABS       ABS_MT_PRESSURE         0000001c
  11. /dev/input/event0: EV_SYN       SYN_REPORT                  00000000
  12. /dev/input/event0: EV_ABS       ABS_MT_TOUCH_MAJOR   00000004
  13. /dev/input/event0: EV_ABS       ABS_MT_POSITION_X      0000017b
  14. /dev/input/event0: EV_ABS       ABS_MT_POSITION_Y      000001cf
  15. /dev/input/event0: EV_ABS       ABS_MT_PRESSURE         0000001c
  16. /dev/input/event0: EV_SYN       SYN_REPORT                  00000000
  17. ……
  18. /dev/input/event0: EV_ABS       ABS_MT_TRACKING_ID     ffffffff
  19. /dev/input/event0: EV_SYN       SYN_REPORT                  00000000

如果是drag呢,在longTouch的基础上,xy坐标从起点到终点是渐变的,其他可认为一样。

对于按键,HOME为例,一目了然:

  1. /dev/input/event0: 0001 0066 00000001
  2. /dev/input/event0: 0000 0000 00000000
  3. /dev/input/event0: 0001 0066 00000000
  4. /dev/input/event0: 0000 0000 00000000
  5. /dev/input/event0: EV_KEY       KEY_HOME             DOWN
  6. /dev/input/event0: EV_SYN       SYN_REPORT          00000000
  7. /dev/input/event0: EV_KEY       KEY_HOME             UP
  8. /dev/input/event0: EV_SYN       SYN_REPORT          00000000

所以清楚了之后,使用sendevent进行事件模拟就很轻松了,关键是从input.h中获取按键对应的扫

描码,也许每个版本的系统中input.h的内容都稍有不同,这个很头疼。

值得注意的是使用getevent获取的数值都是16进制的,而sendevent使用的是10进制的,需要进行

转换。

如点击坐标: 120,254

  1. sendevent /dev/input/event1 0003 0057 00000000      <—事件开始
  2. sendevent /dev/input/event1 0003 0048 00000010      <—点击开始
  3. sendevent /dev/input/event1 0003 0058 00000070      <—  触摸范围
  4. sendevent /dev/input/event1 0003 0053 00000120      <—  x坐标
  5. sendevent /dev/input/event1 0003 0054 00000254      <—  y坐标
  6. sendevent /dev/input/event1 0000 0000 00000000      <—点击结束(同步)
  7. sendevent /dev/input/event1 0003 0057 4294967295        <—事件结束
  8. 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

  1. sendevent /dev/input/event0 0001 0102 0000000001
  2. sendevent /dev/input/event0 0000 0000 0000000000
  3. sendevent /dev/input/event0 0001 0102 0000000000
  4. sendevent /dev/input/event0 0000 0000 0000000000
  5. EV_KEY       KEY_MENU             DOWN
  6. EV_SYN       SYN_REPORT         00000000
  7. EV_KEY       KEY_MENU             UP
  8. EV_SYN       SYN_REPORT         00000000

此外还可以模拟可见字符按键。

到这里是否觉得getevent好用好理解,但sendevent却超级麻烦。没关系,如果真不想用sendevent

模拟事件的话,可以转向input命令。

个人觉得这个input命令是个重量级的东东,调用时间非常长,貌似低系统版本和高系统版本的input提

供的功能还不一样,但它毕竟好用啊,如下:

  1. usage: input …
  2.        input text <string>
  3.        input keyevent <key code number or name>
  4.        input tap <x> <y>
  5.        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的情况这个公式是错误的,请各位同学补充。