mirror of
https://github.com/swaywm/sway.git
synced 2025-01-16 16:11:11 +01:00
Merge pull request #2550 from RyanDwyer/window-type-criteria
Implement window_type criteria token
This commit is contained in:
commit
4ac67d6408
4 changed files with 42 additions and 7 deletions
|
@ -31,7 +31,7 @@ struct criteria {
|
||||||
uint32_t id; // X11 window ID
|
uint32_t id; // X11 window ID
|
||||||
#endif
|
#endif
|
||||||
pcre *window_role;
|
pcre *window_role;
|
||||||
uint32_t window_type;
|
enum atom_name window_type;
|
||||||
bool floating;
|
bool floating;
|
||||||
bool tiling;
|
bool tiling;
|
||||||
char urgent; // 'l' for latest or 'o' for oldest
|
char urgent; // 'l' for latest or 'o' for oldest
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
#include <xcb/xproto.h>
|
#include <xcb/xproto.h>
|
||||||
|
|
||||||
enum atom_name {
|
enum atom_name {
|
||||||
|
NET_WM_WINDOW_TYPE_NORMAL,
|
||||||
NET_WM_WINDOW_TYPE_DIALOG,
|
NET_WM_WINDOW_TYPE_DIALOG,
|
||||||
NET_WM_WINDOW_TYPE_UTILITY,
|
NET_WM_WINDOW_TYPE_UTILITY,
|
||||||
NET_WM_WINDOW_TYPE_TOOLBAR,
|
NET_WM_WINDOW_TYPE_TOOLBAR,
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
#include <strings.h>
|
||||||
#include <pcre.h>
|
#include <pcre.h>
|
||||||
#include "sway/criteria.h"
|
#include "sway/criteria.h"
|
||||||
#include "sway/tree/container.h"
|
#include "sway/tree/container.h"
|
||||||
|
@ -25,7 +26,7 @@ bool criteria_is_empty(struct criteria *criteria) {
|
||||||
&& !criteria->id
|
&& !criteria->id
|
||||||
#endif
|
#endif
|
||||||
&& !criteria->window_role
|
&& !criteria->window_role
|
||||||
&& !criteria->window_type
|
&& criteria->window_type == ATOM_LAST
|
||||||
&& !criteria->floating
|
&& !criteria->floating
|
||||||
&& !criteria->tiling
|
&& !criteria->tiling
|
||||||
&& !criteria->urgent
|
&& !criteria->urgent
|
||||||
|
@ -50,6 +51,23 @@ static int regex_cmp(const char *item, const pcre *regex) {
|
||||||
return pcre_exec(regex, NULL, item, strlen(item), 0, 0, NULL, 0);
|
return pcre_exec(regex, NULL, item, strlen(item), 0, 0, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool view_has_window_type(struct sway_view *view, enum atom_name name) {
|
||||||
|
#ifdef HAVE_XWAYLAND
|
||||||
|
if (view->type != SWAY_VIEW_XWAYLAND) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
struct wlr_xwayland_surface *surface = view->wlr_xwayland_surface;
|
||||||
|
struct sway_xwayland *xwayland = &server.xwayland;
|
||||||
|
xcb_atom_t desired_atom = xwayland->atoms[name];
|
||||||
|
for (size_t i = 0; i < surface->window_type_len; ++i) {
|
||||||
|
if (surface->window_type[i] == desired_atom) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
static int cmp_urgent(const void *_a, const void *_b) {
|
static int cmp_urgent(const void *_a, const void *_b) {
|
||||||
struct sway_view *a = *(void **)_a;
|
struct sway_view *a = *(void **)_a;
|
||||||
struct sway_view *b = *(void **)_b;
|
struct sway_view *b = *(void **)_b;
|
||||||
|
@ -144,9 +162,8 @@ static bool criteria_matches_view(struct criteria *criteria,
|
||||||
// TODO
|
// TODO
|
||||||
}
|
}
|
||||||
|
|
||||||
if (criteria->window_type) {
|
if (criteria->window_type != ATOM_LAST) {
|
||||||
uint32_t type = view_get_window_type(view);
|
if (!view_has_window_type(view, criteria->window_type)) {
|
||||||
if (!type || type != criteria->window_type) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -254,6 +271,21 @@ static bool generate_regex(pcre **regex, char *value) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static enum atom_name parse_window_type(const char *type) {
|
||||||
|
if (strcasecmp(type, "normal") == 0) {
|
||||||
|
return NET_WM_WINDOW_TYPE_NORMAL;
|
||||||
|
} else if (strcasecmp(type, "dialog") == 0) {
|
||||||
|
return NET_WM_WINDOW_TYPE_DIALOG;
|
||||||
|
} else if (strcasecmp(type, "utility") == 0) {
|
||||||
|
return NET_WM_WINDOW_TYPE_UTILITY;
|
||||||
|
} else if (strcasecmp(type, "toolbar") == 0) {
|
||||||
|
return NET_WM_WINDOW_TYPE_TOOLBAR;
|
||||||
|
} else if (strcasecmp(type, "splash") == 0) {
|
||||||
|
return NET_WM_WINDOW_TYPE_SPLASH;
|
||||||
|
}
|
||||||
|
return ATOM_LAST; // ie. invalid
|
||||||
|
}
|
||||||
|
|
||||||
enum criteria_token {
|
enum criteria_token {
|
||||||
T_APP_ID,
|
T_APP_ID,
|
||||||
T_CLASS,
|
T_CLASS,
|
||||||
|
@ -434,7 +466,7 @@ static bool parse_token(struct criteria *criteria, char *name, char *value) {
|
||||||
generate_regex(&criteria->window_role, effective_value);
|
generate_regex(&criteria->window_role, effective_value);
|
||||||
break;
|
break;
|
||||||
case T_WINDOW_TYPE:
|
case T_WINDOW_TYPE:
|
||||||
// TODO: This is a string but will be stored as an enum or integer
|
criteria->window_type = parse_window_type(effective_value);
|
||||||
break;
|
break;
|
||||||
#ifdef HAVE_XWAYLAND
|
#ifdef HAVE_XWAYLAND
|
||||||
case T_ID:
|
case T_ID:
|
||||||
|
@ -526,7 +558,8 @@ struct criteria *criteria_parse(char *raw, char **error_arg) {
|
||||||
}
|
}
|
||||||
++head;
|
++head;
|
||||||
|
|
||||||
struct criteria *criteria = calloc(sizeof(struct criteria), 1);
|
struct criteria *criteria = calloc(1, sizeof(struct criteria));
|
||||||
|
criteria->window_type = ATOM_LAST; // default value
|
||||||
char *name = NULL, *value = NULL;
|
char *name = NULL, *value = NULL;
|
||||||
bool in_quotes = false;
|
bool in_quotes = false;
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
#include "sway/tree/view.h"
|
#include "sway/tree/view.h"
|
||||||
|
|
||||||
static const char *atom_map[ATOM_LAST] = {
|
static const char *atom_map[ATOM_LAST] = {
|
||||||
|
"_NET_WM_WINDOW_TYPE_NORMAL",
|
||||||
"_NET_WM_WINDOW_TYPE_DIALOG",
|
"_NET_WM_WINDOW_TYPE_DIALOG",
|
||||||
"_NET_WM_WINDOW_TYPE_UTILITY",
|
"_NET_WM_WINDOW_TYPE_UTILITY",
|
||||||
"_NET_WM_WINDOW_TYPE_TOOLBAR",
|
"_NET_WM_WINDOW_TYPE_TOOLBAR",
|
||||||
|
|
Loading…
Reference in a new issue