-
Notifications
You must be signed in to change notification settings - Fork 318
/
cmd_behave.c
153 lines (134 loc) · 4.55 KB
/
cmd_behave.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
#include "xdo_cmd.h"
#include <string.h>
struct events {
const char * const name;
int mask;
} events[] = {
{ "mouse-enter", EnterWindowMask },
{ "mouse-leave", LeaveWindowMask },
{ "focus", FocusChangeMask },
{ "blur", FocusChangeMask },
{ "mouse-click", ButtonReleaseMask },
{ NULL, 0 },
};
int cmd_behave(context_t *context) {
int ret = 0;
char *cmd = *context->argv;
int c;
enum { opt_unused, opt_help };
static struct option longopts[] = {
{ "help", no_argument, NULL, opt_help },
{ 0, 0, 0, 0 },
};
static const char *usage =
"Usage: %s window event action [args...]\n"
"The event is a window event, such as mouse-enter, resize, etc.\n"
"The action is any valid xdotool command (chains OK here)\n"
"\n"
"Events: \n"
" mouse-enter - When the mouse moves into the window\n"
" mouse-leave - When the mouse leaves a window\n"
" mouse-click - Fired when the mouse button is released\n"
" focus - When the window gets focus\n"
" blur - When the window loses focus\n";
int option_index;
while ((c = getopt_long_only(context->argc, context->argv, "+h",
longopts, &option_index)) != -1) {
switch (c) {
case 'h':
case opt_help:
printf(usage, cmd);
consume_args(context, context->argc);
return EXIT_SUCCESS;
break;
default:
fprintf(stderr, usage, cmd);
return EXIT_FAILURE;
}
}
consume_args(context, optind);
if (context->argc < 3) {
fprintf(stderr, "Invalid number of arguments (minimum is 3)\n");
fprintf(stderr, usage, cmd);
return EXIT_FAILURE;
}
/* Can't use consume_args since 'behave' eats the rest of the line. */
/* TODO(sissel): make it work. */
const char *window_arg = context->argv[0];
consume_args(context, 1);
const char *event = context->argv[0];
consume_args(context, 1);
/* The remainder of args are supposed to be what to run on the action */
long selectmask = 0;
int i;
for (i = 0; events[i].name != NULL; i++) {
//printf("%s vs %s\n", events[i].name, event);
if (!strcmp(events[i].name, event)) {
xdotool_debug(context, "Adding mask for event '%s': 0x%lx", event, events[i].mask);
selectmask |= events[i].mask;
}
}
if (selectmask == 0) {
fprintf(stderr, "Unknown event '%s'\n", event);
return EXIT_FAILURE;
}
xdotool_debug(context, "Selectmask: 0x%lx", selectmask);
window_each(context, window_arg, {
xdotool_debug(context, "Selecting events on window %ld: mask = 0x%lx",
window, selectmask);
ret = XSelectInput(context->xdo->xdpy, window, selectmask);
xdotool_debug(context, "Select: %d", ret);
if (ret != True) {
fprintf(stderr, "XSelectInput reported an error\n");
}
}); /* window_each(...) */
while (True) {
XEvent e;
xdotool_debug(context, "Waiting for next event...");
XNextEvent(context->xdo->xdpy, &e);
xdotool_debug(context, "Got event type %d", e.type);
// Copy context
context_t tmpcontext = *context;
tmpcontext.nwindows = 1;
Window hover; /* for LeaveNotify */
switch (e.type) {
case LeaveNotify:
/* LeaveNotify is confusing.
* It is sometimes fired when you are actually entering the window
* especially at the screen edges. not sure why or what causes it.
* Work around: Query the window the mouse is over. If it is not
* us, then we can fire leave. */
/* allow some time to pass to let the mouse really leave if we are on our way out */
/* TODO(sissel): allow this delay to be tunable */
usleep(100000); /* 100ms */
xdo_get_window_at_mouse(context->xdo, &hover);
if (hover == e.xcrossing.window) {
//printf("Ignoring Leave, we're still in the window\n");
break;
}
//printf("Window: %ld\n", e.xcrossing.window);
//printf("Hover: %ld\n", hover);
/* fall through */
case EnterNotify:
tmpcontext.windows = &(e.xcrossing.window);
ret = context_execute(&tmpcontext);
break;
case FocusIn:
case FocusOut:
tmpcontext.windows = &(e.xfocus.window);
ret = context_execute(&tmpcontext);
break;
case ButtonRelease:
tmpcontext.windows = &(e.xbutton.window);
ret = context_execute(&tmpcontext);
break;
default:
printf("Unexpected event: %d\n", e.type);
break;
}
if (ret != XDO_SUCCESS) {
xdotool_output(context, "Command failed.");
}
}
return ret;
}