2021-04-12 21:16:16 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/extensions/XTest.h>
|
|
|
|
#include <unistd.h>
|
2021-04-12 23:01:43 +00:00
|
|
|
#include <X11/extensions/XInput2.h>
|
|
|
|
#include <pthread.h>
|
2021-04-12 21:16:16 +00:00
|
|
|
|
|
|
|
static Display *dpy;
|
|
|
|
static int scr, sw, sh;
|
2021-04-12 23:01:43 +00:00
|
|
|
static int sx, sy;
|
2021-04-12 21:16:16 +00:00
|
|
|
static Window root;
|
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
static void select_events(Display *dpy, Window win) {
|
|
|
|
XIEventMask evmasks[1];
|
|
|
|
unsigned char mask1[(XI_LASTEVENT + 7)/8];
|
|
|
|
|
|
|
|
memset(mask1, 0, sizeof(mask1));
|
|
|
|
|
|
|
|
/* select for button and key events from all master devices */
|
|
|
|
XISetMask(mask1, XI_RawButtonPress);
|
|
|
|
XISetMask(mask1, XI_RawButtonRelease);
|
|
|
|
|
|
|
|
evmasks[0].deviceid = XIAllMasterDevices;
|
|
|
|
evmasks[0].mask_len = sizeof(mask1);
|
|
|
|
evmasks[0].mask = mask1;
|
|
|
|
|
|
|
|
XISelectEvents(dpy, win, evmasks, 1);
|
|
|
|
XFlush(dpy);
|
2021-04-12 21:16:16 +00:00
|
|
|
}
|
2021-04-12 23:01:43 +00:00
|
|
|
|
|
|
|
static void getmousepos(int *px, int *py) {
|
|
|
|
Window r, child;
|
|
|
|
int rx, ry;
|
|
|
|
unsigned int mask;
|
|
|
|
XQueryPointer(dpy, root, &r, &child, px, py, &rx, &ry, &mask);
|
2021-04-12 21:16:16 +00:00
|
|
|
}
|
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
static void* loop() {
|
|
|
|
int px, py, lx, ly, dx, dy;
|
|
|
|
while (1) {
|
|
|
|
lx = px, ly = py;
|
|
|
|
getmousepos(&px, &py);
|
|
|
|
dx = px - lx;
|
|
|
|
dy = py - ly;
|
|
|
|
printf("(%d, %d) from (%d, %d)\n", px, py, sx, sy);
|
|
|
|
sleep(1);
|
2021-04-12 21:16:16 +00:00
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mouse_down(XIRawEvent *xev) {
|
|
|
|
getmousepos(&sx, &sy);
|
2021-04-12 21:16:16 +00:00
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
printf("Button pressed %d @ %d, %d\n", xev->detail, sx, sy);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mouse_up(XIRawEvent *xev) {
|
|
|
|
sx = -1;
|
|
|
|
sy = -1;
|
|
|
|
printf("Button released %d %d\n", xev->detail);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, const char **argv) {
|
|
|
|
XInitThreads();
|
2021-04-12 21:16:16 +00:00
|
|
|
for (int i = 1; i < argc; i++) {
|
|
|
|
if (argv[i][1] == 'h') {
|
|
|
|
fprintf(stdout, "there is no help to be given\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(dpy = XOpenDisplay(0x0))) return 1;
|
|
|
|
|
|
|
|
scr = DefaultScreen(dpy);
|
|
|
|
root = XDefaultRootWindow(dpy);
|
|
|
|
sw = DisplayWidth(dpy, scr);
|
|
|
|
sh = DisplayHeight(dpy, scr);
|
|
|
|
|
|
|
|
fprintf(stdout, "waiting for events\n");
|
|
|
|
|
|
|
|
XEvent ev;
|
2021-04-12 23:01:43 +00:00
|
|
|
XIEvent *xi_event;
|
|
|
|
XIRawEvent *xev;
|
|
|
|
XGenericEventCookie *cookie = &ev.xcookie;
|
|
|
|
|
|
|
|
select_events(dpy, root);
|
2021-04-12 21:16:16 +00:00
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
pthread_t id[2];
|
|
|
|
pthread_create(&id[0], NULL, loop, &argv);
|
|
|
|
|
2021-04-12 21:16:16 +00:00
|
|
|
for (;;) {
|
2021-04-12 23:01:43 +00:00
|
|
|
if (XCheckTypedEvent(dpy, GenericEvent ,&ev)) {
|
|
|
|
if (cookie->type != GenericEvent || !XGetEventData(dpy, cookie)) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-04-12 21:16:16 +00:00
|
|
|
|
2021-04-12 23:01:43 +00:00
|
|
|
|
|
|
|
xi_event = (XIEvent *) cookie->data;
|
|
|
|
xev = (XIRawEvent *) xi_event;
|
|
|
|
switch (cookie->evtype) {
|
|
|
|
case XI_RawButtonPress:
|
|
|
|
if (xev->detail == 2) mouse_down(xev);
|
|
|
|
break;
|
|
|
|
case XI_RawButtonRelease:
|
|
|
|
if (xev->detail == 2) mouse_up(xev);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
XFreeEventData(dpy, cookie);
|
|
|
|
}
|
2021-04-12 21:16:16 +00:00
|
|
|
}
|
|
|
|
XCloseDisplay(dpy);
|
|
|
|
return 0;
|
|
|
|
}
|