我需要捕捉屏幕(作为打印屏幕)以便我可以访问像素颜色数据,进行一些图像识别,之后我需要在屏幕上生成鼠标事件,如左键单击,拖放(按下按钮时移动鼠标,然后松开)。完成后,图像将被删除。
注意:我需要捕获整个屏幕用户可以看到的所有内容,我需要模拟程序窗口外的点击(如果它有任何区别)
规范:Linux ubuntu 语言:C ++
性能不是很重要,“打印屏幕”功能每隔约10秒执行一次。 该过程的持续时间可长达24小时,因此方法需要稳定且内存泄漏(如通常所示)
我能够在Windows中赢得GDI和一些Windows事件,但我不知道如何在Linux中实现它。
非常感谢
答案 0 :(得分:15)
//sg
//Solution using Xlib for those who use Linux
#include <X11/Xlib.h>
#include<stdio.h>
#include<unistd.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
void mouseClick(int button)
{
Display *display = XOpenDisplay(NULL);
XEvent event;
if(display == NULL)
{
fprintf(stderr, "Cannot initialize the display\n");
exit(EXIT_FAILURE);
}
memset(&event, 0x00, sizeof(event));
event.type = ButtonPress;
event.xbutton.button = button;
event.xbutton.same_screen = True;
XQueryPointer(display, RootWindow(display, DefaultScreen(display)), &event.xbutton.root, &event.xbutton.window, &event.xbutton.x_root, &event.xbutton.y_root, &event.xbutton.x, &event.xbutton.y, &event.xbutton.state);
event.xbutton.subwindow = event.xbutton.window;
while(event.xbutton.subwindow)
{
event.xbutton.window = event.xbutton.subwindow;
XQueryPointer(display, event.xbutton.window, &event.xbutton.root, &event.xbutton.subwindow, &event.xbutton.x_root, &event.xbutton.y_root, &event.xbutton.x, &event.xbutton.y, &event.xbutton.state);
}
if(XSendEvent(display, PointerWindow, True, 0xfff, &event) == 0) fprintf(stderr, "Error\n");
XFlush(display);
usleep(100000);
event.type = ButtonRelease;
event.xbutton.state = 0x100;
if(XSendEvent(display, PointerWindow, True, 0xfff, &event) == 0) fprintf(stderr, "Error\n");
XFlush(display);
XCloseDisplay(display);
}
int main(int argc,char * argv[]) {
int x , y;
x=atoi(argv[1]);
y=atoi(argv[2]);
Display *display = XOpenDisplay(0);
Window root = DefaultRootWindow(display);
XWarpPointer(display, None, root, 0, 0, 0, 0, x, y);
mouseClick(Button1);
XFlush(display);
XCloseDisplay(display);
return 0;
}
构建它然后模拟x上的点击,执行:
$ ./a.out x y
即
$ g ++ -lX11 sgmousesim2.cpp
$ ./a.out 123 13
以防你仍然感兴趣。
答案 1 :(得分:1)
Swinput是模拟鼠标/键事件的解决方案。你需要为你的内核编译它。 Xorg提供了一些用于录制鼠标/键事件的标题,但我认为它目前已被打破。有C
代码evtest,可用于从/dev/input/eventX
,/dev/input/mice
文件中捕获事件。它可能会有所帮助。
修改强>
Xorg记录扩展中的bug was fixed,因此它也可以正常工作。
答案 2 :(得分:0)
为使其正常工作,必须在扭曲指针后调用XFlush
在Linux Mint19(Cinamon 3.8)上测试 XWarpPointer(display,None,root,0,0,0,0,x,y); XFlush(显示);
答案 3 :(得分:0)
我无法使用所使用的@axiom方法进行点击,只能移动指针。我改用这个:(Ubuntu 18.04)。
使用以下命令编译:g ++ mouse_click.cpp -lX11 -lXtst -lstdc ++
#include<stdio.h>
#include<unistd.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/extensions/XTest.h>
void mouseClick(int button)
{
Display *display = XOpenDisplay(NULL);
// click left button
XTestFakeButtonEvent(display, Button1, true, 0);
XFlush(display);
usleep(10000);
// release left mouse
XTestFakeButtonEvent(display, Button1, false, 0);
XFlush(display);
XCloseDisplay(display);
}
int main(int argc,char * argv[]) {
int x , y;
x=atoi(argv[1]);
y=atoi(argv[2]);
Display *display = XOpenDisplay(0);
Window root = DefaultRootWindow(display);
XTestFakeMotionEvent(display, root, x, y, 0);
XFlush(display);
mouseClick(Button1);
XFlush(display);
XCloseDisplay(display);
return 0;
}