Android模拟快速刷卡

时间:2013-09-28 03:55:45

标签: android adb

我正在做一个通用的自动化脚本。

我需要将复杂的滑动事件发送到Android屏幕,而无需专门访问重点应用程序

到目前为止,我认为最好的方法是使用adb,使用sendevent命令创建文件,将其推送到设备上并从那里运行。即便如此,它也非常缓慢(与我用getevent录制它并将其重新安装后相比要慢得多)。

我设法优化文件,因为我发现每个sendevent块并不特别需要X和Y,但它仍然慢了几个数量级

部分文件示例(我正在尝试使用HTC One):

sendevent /dev/input/event5 3 57 49
sendevent /dev/input/event5 3 53 942
sendevent /dev/input/event5 3 54 2747
sendevent /dev/input/event5 0 0 0

sendevent /dev/input/event5 3 53 1207
sendevent /dev/input/event5 3 54 2483
sendevent /dev/input/event5 0 0 0

sendevent /dev/input/event5 3 53 1472
sendevent /dev/input/event5 0 0 0

sendevent /dev/input/event5 3 54 2218
sendevent /dev/input/event5 0 0 0

sendevent /dev/input/event5 3 53 1207
sendevent /dev/input/event5 3 54 2483
sendevent /dev/input/event5 0 0 0

sendevent /dev/input/event5 3 53 1472

所以我的重点是优化单个长复杂滑动的速度,而不是多个小滑动的速度。

任何人都知道更好的方法吗?


所以,Chris Stratton的想法原则上起作用(重新管道cat-ed输出会成功地生成相同的滑动),但我无法创建自己的代码来重新管道。我猜它是与发送事件命令之间的分隔符有关...但我仍然无法使它工作

我使用了对sendevent.c文件的修改来获取每行三元组并输出到另一个文件的文件。你碰巧知道可能是什么问题吗?转换看起来不错......


SOLLUTION:我设法解决了这个问题,主要归功于下面的答案。这是一个C脚本,它接收带有HEX值的文件并输出相应的二进制文件。

用法:(对我来说,触摸驱动文件是/ dev / input / event5 - HTC One - 对于其他设备,它可能是一个不同的文件!)

 $> adb shell getevent > tmp.in
 $> ./sendevent tmp.in tmp.out
 $> adb shell push tmp.out /mnt/sdcard/
 $> adb shell "cd /mnt/sdcard/ && cat tmp.out > /dev/input/event5"

和来源:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <errno.h>

typedef uint32_t        __u32;
typedef uint16_t        __u16;
typedef __signed__ int  __s32;

__attribute__((aligned(1),packed)) struct input_event {
    __u32 time_dummy_1;
    __u32 time_dummy_2;
    __u16 type;
    __u16 code;
    __s32 value;
};

int convert (char * str) {
    return (int) strtol(str, NULL, 16);
}

#define S_ALL (S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH | S_IWOTH | S_IXOTH)

int main (int argc, char *argv[]) {
    int i;
    int fd;
    int ret;

    if(argc < 3) {
        fprintf(stderr, "use: %s in-file out-file\n", argv[0]);
        return 1;
    }

    fd = open(argv[2], O_CREAT | O_WRONLY, S_ALL);
    if(fd < 0) {
        fprintf(stderr, "could not open %s, %s\n", argv[2], strerror(errno));
        return 1;
    }

    FILE * fd_in = fopen(argv[1], "r");
    if (fd_in == NULL) {
        fprintf(stderr, "Can't open input file: %s\n", argv[1]);
        return 1;
    }

    struct input_event event;
    char type[32];
    char code[32];
    char value[32];
    int count = 0;
    while (fscanf(fd_in, "%s %s %s", type, code, value) != EOF) {
        memset(&event, 0, sizeof(event));
        // printf("%d) %s %s %s\n", ++count, type, code, value);
        event.type = convert(type);
        event.code = convert(code);
        event.value = convert(value);
        memset(type, 0, sizeof(type));
        memset(code, 0, sizeof(code));
        memset(value, 0, sizeof(value));
        ret = write(fd, &event, sizeof(event));
        if(ret < sizeof(event)) {
            fprintf(stderr, "write event failed, %s\n", strerror(errno));
            return -1;
        }
    }

    return 0;
}

2 个答案:

答案 0 :(得分:13)

请注意,此答案适用于大约2013年版本的Android,可能不适用于当前版本。 Jellybean当时是当代的,Kitkat在提出问题几个星期后问世

您的延迟可能是由于无法重复启动新的sendevent进程,解析文本事件记录以及打开设备节点 - 导致每个单独的事件。如果您在一个进程中执行所有操作,只打开一次设备文件,则效率会更高。

如果我们使用问题日期(例如,https://android.googlesource.com/platform/system/core/+/jb-release/toolbox/sendevent.c)查看工具箱当前的sendevent源代码,我们会看到它所做的核心是编码事件到二进制记录

struct input_event {
    struct timeval time;
    __u16 type;
    __u16 code;
    __s32 value;
};

并将它们写入适当的设备

memset(&event, 0, sizeof(event));
event.type = atoi(argv[2]);
event.code = atoi(argv[3]);
event.value = atoi(argv[4]);
ret = write(fd, &event, sizeof(event));

如果您在shell unix组中执行input用户标识或其他内容,您应该能够完成sendevent从您自定义程序中执行的操作,或使用其他命令行工具,如cat,从而有效地推送事件记录的二进制文件。

例如

adb shell
cd /mnt/sdcard
cat /dev/input/event2 > events

做一些触摸屏事件,然后ctrl-C杀死猫

现在您可以播放捕获的二进制事件文件:

cat events > /dev/input/event2 

(注意:sendevent将每个记录的timeval部分归零;记录和播放可能不会这样做;你必须看到,如果它重要,那么你之前的文件中每个记录的那些部分都是零把它写回来)

答案 1 :(得分:6)

如果您只想生成线性滑动,可以在shell上使用input swipe命令。

$ adb shell input
usage: input ...
       input text <string>
       input keyevent <key code number or name>
       input [touchscreen|touchpad|touchnavigation] tap <x> <y>
       input [touchscreen|touchpad|touchnavigation] swipe <x1> <y1> <x2> <y2> [duration(ms)]
       input trackball press
       input trackball roll <dx> <dy>

下面的命令在绘图应用程序中为我绘制了一条很好的线条

$ adb shell input swipe 300 300 500 1000

和更快的

$ adb shell input touchscreen swipe 300 300 500 1000 100