17 #define _i3_timercmp(a, b, CMP) \
18 (((a).tv_sec == (b).tv_sec) ? ((a).tv_usec CMP(b).tv_usec) : ((a).tv_sec CMP(b).tv_sec))
27 memset(match, 0,
sizeof(
Match));
28 match->
urgent = U_DONTCHECK;
43 return (match->
title == NULL &&
44 match->
mark == NULL &&
46 match->
class == NULL &&
50 match->
urgent == U_DONTCHECK &&
51 match->
id == XCB_NONE &&
54 match->
dock == M_NODOCK &&
63 memcpy(dest, src,
sizeof(
Match));
68 #define DUPLICATE_REGEX(field) \
70 if (src->field != NULL) \
71 dest->field = regex_new(src->field->pattern); \
90 #define GET_FIELD_str(field) (field)
91 #define GET_FIELD_i3string(field) (i3string_as_utf8(field))
92 #define CHECK_WINDOW_FIELD(match_field, window_field, type) \
94 if (match->match_field != NULL) { \
95 const char *window_field_str = window->window_field == NULL \
97 : GET_FIELD_##type(window->window_field); \
98 if (strcmp(match->match_field->pattern, "__focused__") == 0 && \
99 focused && focused->window && focused->window->window_field && \
100 strcmp(window_field_str, GET_FIELD_##type(focused->window->window_field)) == 0) { \
101 LOG("window " #match_field " matches focused window\n"); \
102 } else if (regex_matches(match->match_field, window_field_str)) { \
103 LOG("window " #match_field " matches (%s)\n", window_field_str); \
113 if (match->
id != XCB_NONE) {
114 if (window->
id == match->
id) {
115 LOG(
"match made by window id (%d)\n", window->
id);
117 LOG(
"window id does not match\n");
134 if (match->
urgent == U_LATEST) {
136 if (window->
urgent.tv_sec == 0) {
141 if ((con->
window != NULL) &&
146 LOG(
"urgent matches latest\n");
149 if (match->
urgent == U_OLDEST) {
151 if (window->
urgent.tv_sec == 0) {
156 if ((con->
window != NULL) &&
162 LOG(
"urgent matches oldest\n");
175 LOG(
"workspace matches focused workspace\n");
177 LOG(
"workspace matches (%s)\n", ws->
name);
183 if (match->
dock != M_DONTCHECK) {
184 if ((window->
dock == W_DOCK_TOP && match->
dock == M_DOCK_TOP) ||
185 (window->
dock == W_DOCK_BOTTOM && match->
dock == M_DOCK_BOTTOM) ||
186 ((window->
dock == W_DOCK_TOP || window->
dock == W_DOCK_BOTTOM) &&
187 match->
dock == M_DOCK_ANY) ||
188 (window->
dock == W_NODOCK && match->
dock == M_NODOCK)) {
189 LOG(
"dock status matches\n");
191 LOG(
"dock status does not match\n");
196 if (match->
mark != NULL) {
200 bool matched =
false;
210 LOG(
"mark matches\n");
212 LOG(
"mark does not match\n");
224 if (con->
floating != FLOATING_AUTO_OFF) {
229 if (con->
floating != FLOATING_USER_OFF) {
238 case WM_FLOATING_AUTO:
239 if (con->
floating != FLOATING_AUTO_ON) {
243 case WM_FLOATING_USER:
244 if (con->
floating != FLOATING_USER_ON) {
257 LOG(
"window_mode matches\n");
283 assert(match != NULL);
284 DLOG(
"ctype=*%s*, cvalue=*%s*\n", ctype, cvalue);
286 if (strcmp(ctype,
"class") == 0) {
292 if (strcmp(ctype,
"instance") == 0) {
298 if (strcmp(ctype,
"window_role") == 0) {
304 if (strcmp(ctype,
"con_id") == 0) {
305 if (strcmp(cvalue,
"__focused__") == 0) {
312 ELOG(
"Could not parse con id \"%s\"\n", cvalue);
321 if (strcmp(ctype,
"id") == 0) {
324 ELOG(
"Could not parse window id \"%s\"\n", cvalue);
328 DLOG(
"window id as int = %d\n", match->
id);
333 if (strcmp(ctype,
"window_type") == 0) {
334 if (strcasecmp(cvalue,
"normal") == 0) {
336 }
else if (strcasecmp(cvalue,
"dialog") == 0) {
338 }
else if (strcasecmp(cvalue,
"utility") == 0) {
339 match->
window_type = A__NET_WM_WINDOW_TYPE_UTILITY;
340 }
else if (strcasecmp(cvalue,
"toolbar") == 0) {
341 match->
window_type = A__NET_WM_WINDOW_TYPE_TOOLBAR;
342 }
else if (strcasecmp(cvalue,
"splash") == 0) {
344 }
else if (strcasecmp(cvalue,
"menu") == 0) {
346 }
else if (strcasecmp(cvalue,
"dropdown_menu") == 0) {
347 match->
window_type = A__NET_WM_WINDOW_TYPE_DROPDOWN_MENU;
348 }
else if (strcasecmp(cvalue,
"popup_menu") == 0) {
349 match->
window_type = A__NET_WM_WINDOW_TYPE_POPUP_MENU;
350 }
else if (strcasecmp(cvalue,
"tooltip") == 0) {
351 match->
window_type = A__NET_WM_WINDOW_TYPE_TOOLTIP;
352 }
else if (strcasecmp(cvalue,
"notification") == 0) {
353 match->
window_type = A__NET_WM_WINDOW_TYPE_NOTIFICATION;
355 ELOG(
"unknown window_type value \"%s\"\n", cvalue);
362 if (strcmp(ctype,
"con_mark") == 0) {
368 if (strcmp(ctype,
"title") == 0) {
374 if (strcmp(ctype,
"urgent") == 0) {
375 if (strcasecmp(cvalue,
"latest") == 0 ||
376 strcasecmp(cvalue,
"newest") == 0 ||
377 strcasecmp(cvalue,
"recent") == 0 ||
378 strcasecmp(cvalue,
"last") == 0) {
380 }
else if (strcasecmp(cvalue,
"oldest") == 0 ||
381 strcasecmp(cvalue,
"first") == 0) {
387 if (strcmp(ctype,
"workspace") == 0) {
393 if (strcmp(ctype,
"tiling") == 0) {
398 if (strcmp(ctype,
"tiling_from") == 0 &&
400 strcmp(cvalue,
"auto") == 0) {
405 if (strcmp(ctype,
"tiling_from") == 0 &&
407 strcmp(cvalue,
"user") == 0) {
412 if (strcmp(ctype,
"floating") == 0) {
417 if (strcmp(ctype,
"floating_from") == 0 &&
419 strcmp(cvalue,
"auto") == 0) {
424 if (strcmp(ctype,
"floating_from") == 0 &&
426 strcmp(cvalue,
"user") == 0) {
431 ELOG(
"Unknown criterion: %s\n", ctype);
Con * con_get_workspace(Con *con)
Gets the workspace container this node is on.
Con * con_by_window_id(xcb_window_t window)
Returns the container with the given client window ID or NULL if no such container exists.
Con * con_inside_floating(Con *con)
Checks if the given container is either floating or inside some floating container.
struct pending_marks * marks
#define _i3_timercmp(a, b, CMP)
#define DUPLICATE_REGEX(field)
bool match_is_empty(Match *match)
Check if a match is empty.
void match_init(Match *match)
Initializes the Match data structure.
void match_copy(Match *dest, Match *src)
Copies the data of a match from src to dest.
void match_free(Match *match)
Frees the given match.
bool match_matches_window(Match *match, i3Window *window)
Check if a match data structure matches the given window.
void match_parse_property(Match *match, const char *ctype, const char *cvalue)
Interprets a ctype=cvalue pair and adds it to the given match specification.
#define CHECK_WINDOW_FIELD(match_field, window_field, type)
struct regex * regex_new(const char *pattern)
Creates a new 'regex' struct containing the given pattern and a PCRE compiled regular expression.
bool regex_matches(struct regex *regex, const char *input)
Checks if the given regular expression matches the given input and returns true if it does.
void regex_free(struct regex *regex)
Frees the given regular expression.
struct all_cons_head all_cons
bool parse_long(const char *str, long *out, int base)
Converts a string into a long using strtol().
char * sstrdup(const char *str)
Safe-wrapper around strdup which exits if malloc returns NULL (meaning that there is no more memory a...
#define TAILQ_FOREACH(var, head, field)
A 'Window' is a type which contains an xcb_window_t and all the related information (hints like _NET_...
struct timeval urgent
When this window was marked urgent.
xcb_atom_t window_type
The _NET_WM_WINDOW_TYPE for this window.
enum Window::@13 dock
Whether the window says it is a dock window.
A "match" is a data structure which acts like a mask or expression to match certain windows or not.
struct regex * window_role
struct regex * application
enum Match::@16 window_mode
A 'Con' represents everything from the X11 root window down to a single X11 window.
enum Con::@21 floating
floating? (= not in tiling layout) This cannot be simply a bool because we want to keep track of whet...