mirror of
https://github.com/hyprwm/wlroots-hyprland.git
synced 2024-11-26 14:45:58 +01:00
Add per-seat cursor configuration
This commit is contained in:
parent
cd566ccd8e
commit
5b13f51dfc
5 changed files with 130 additions and 36 deletions
|
@ -3,6 +3,8 @@
|
||||||
#include <wlr/types/wlr_output_layout.h>
|
#include <wlr/types/wlr_output_layout.h>
|
||||||
#include <wlr/types/wlr_input_device.h>
|
#include <wlr/types/wlr_input_device.h>
|
||||||
|
|
||||||
|
#define ROOTS_CONFIG_DEFAULT_SEAT_NAME "seat0"
|
||||||
|
|
||||||
struct roots_output_config {
|
struct roots_output_config {
|
||||||
char *name;
|
char *name;
|
||||||
enum wl_output_transform transform;
|
enum wl_output_transform transform;
|
||||||
|
@ -17,9 +19,9 @@ struct roots_output_config {
|
||||||
|
|
||||||
struct roots_device_config {
|
struct roots_device_config {
|
||||||
char *name;
|
char *name;
|
||||||
|
char *seat;
|
||||||
char *mapped_output;
|
char *mapped_output;
|
||||||
struct wlr_box *mapped_box;
|
struct wlr_box *mapped_box;
|
||||||
char *seat;
|
|
||||||
struct wl_list link;
|
struct wl_list link;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -33,6 +35,7 @@ struct roots_binding_config {
|
||||||
|
|
||||||
struct roots_keyboard_config {
|
struct roots_keyboard_config {
|
||||||
char *name;
|
char *name;
|
||||||
|
char *seat;
|
||||||
uint32_t meta_key;
|
uint32_t meta_key;
|
||||||
char *rules;
|
char *rules;
|
||||||
char *model;
|
char *model;
|
||||||
|
@ -42,19 +45,22 @@ struct roots_keyboard_config {
|
||||||
struct wl_list link;
|
struct wl_list link;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct roots_config {
|
struct roots_cursor_config {
|
||||||
bool xwayland;
|
char *seat;
|
||||||
|
|
||||||
struct {
|
|
||||||
char *mapped_output;
|
char *mapped_output;
|
||||||
struct wlr_box *mapped_box;
|
struct wlr_box *mapped_box;
|
||||||
char *theme;
|
char *theme;
|
||||||
} cursor;
|
struct wl_list link;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct roots_config {
|
||||||
|
bool xwayland;
|
||||||
|
|
||||||
struct wl_list outputs;
|
struct wl_list outputs;
|
||||||
struct wl_list devices;
|
struct wl_list devices;
|
||||||
struct wl_list bindings;
|
struct wl_list bindings;
|
||||||
struct wl_list keyboards;
|
struct wl_list keyboards;
|
||||||
|
struct wl_list cursors;
|
||||||
char *config_path;
|
char *config_path;
|
||||||
char *startup_cmd;
|
char *startup_cmd;
|
||||||
};
|
};
|
||||||
|
@ -92,4 +98,11 @@ struct roots_device_config *roots_config_get_device(struct roots_config *config,
|
||||||
struct roots_keyboard_config *roots_config_get_keyboard(
|
struct roots_keyboard_config *roots_config_get_keyboard(
|
||||||
struct roots_config *config, struct wlr_input_device *device);
|
struct roots_config *config, struct wlr_input_device *device);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get configuration for the cursor. If the cursor is not configured, returns
|
||||||
|
* NULL. A NULL seat_name returns the default config for cursors.
|
||||||
|
*/
|
||||||
|
struct roots_cursor_config *roots_config_get_cursor(struct roots_config *config,
|
||||||
|
const char *seat_name);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -150,6 +150,37 @@ void add_binding_config(struct wl_list *bindings, const char* combination,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void config_handle_cursor(struct roots_config *config,
|
||||||
|
const char *seat_name, const char *name, const char *value) {
|
||||||
|
struct roots_cursor_config *cc;
|
||||||
|
bool found = false;
|
||||||
|
wl_list_for_each(cc, &config->cursors, link) {
|
||||||
|
if (strcmp(cc->seat, seat_name) == 0) {
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!found) {
|
||||||
|
cc = calloc(1, sizeof(struct roots_cursor_config));
|
||||||
|
cc->seat = strdup(seat_name);
|
||||||
|
wl_list_insert(&config->cursors, &cc->link);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(name, "map-to-output") == 0) {
|
||||||
|
free(cc->mapped_output);
|
||||||
|
cc->mapped_output = strdup(value);
|
||||||
|
} else if (strcmp(name, "geometry") == 0) {
|
||||||
|
free(cc->mapped_box);
|
||||||
|
cc->mapped_box = parse_geometry(value);
|
||||||
|
} else if (strcmp(name, "theme") == 0) {
|
||||||
|
free(cc->theme);
|
||||||
|
cc->theme = strdup(value);
|
||||||
|
} else {
|
||||||
|
wlr_log(L_ERROR, "got unknown cursor config: %s", name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void config_handle_keyboard(struct roots_config *config,
|
static void config_handle_keyboard(struct roots_config *config,
|
||||||
const char *device_name, const char *name, const char *value) {
|
const char *device_name, const char *name, const char *value) {
|
||||||
struct roots_keyboard_config *kc;
|
struct roots_keyboard_config *kc;
|
||||||
|
@ -190,6 +221,7 @@ static void config_handle_keyboard(struct roots_config *config,
|
||||||
static const char *output_prefix = "output:";
|
static const char *output_prefix = "output:";
|
||||||
static const char *device_prefix = "device:";
|
static const char *device_prefix = "device:";
|
||||||
static const char *keyboard_prefix = "keyboard:";
|
static const char *keyboard_prefix = "keyboard:";
|
||||||
|
static const char *cursor_prefix = "cursor:";
|
||||||
|
|
||||||
static int config_ini_handler(void *user, const char *section, const char *name,
|
static int config_ini_handler(void *user, const char *section, const char *name,
|
||||||
const char *value) {
|
const char *value) {
|
||||||
|
@ -269,19 +301,12 @@ static int config_ini_handler(void *user, const char *section, const char *name,
|
||||||
oc->name, oc->mode.width, oc->mode.height,
|
oc->name, oc->mode.width, oc->mode.height,
|
||||||
oc->mode.refresh_rate);
|
oc->mode.refresh_rate);
|
||||||
}
|
}
|
||||||
|
} else if (strncmp(cursor_prefix, section, strlen(cursor_prefix)) == 0) {
|
||||||
|
const char *seat_name = section + strlen(cursor_prefix);
|
||||||
|
config_handle_cursor(config, seat_name, name, value);
|
||||||
} else if (strcmp(section, "cursor") == 0) {
|
} else if (strcmp(section, "cursor") == 0) {
|
||||||
if (strcmp(name, "map-to-output") == 0) {
|
config_handle_cursor(config, ROOTS_CONFIG_DEFAULT_SEAT_NAME, name,
|
||||||
free(config->cursor.mapped_output);
|
value);
|
||||||
config->cursor.mapped_output = strdup(value);
|
|
||||||
} else if (strcmp(name, "geometry") == 0) {
|
|
||||||
free(config->cursor.mapped_box);
|
|
||||||
config->cursor.mapped_box = parse_geometry(value);
|
|
||||||
} else if (strcmp(name, "theme") == 0) {
|
|
||||||
free(config->cursor.theme);
|
|
||||||
config->cursor.theme = strdup(value);
|
|
||||||
} else {
|
|
||||||
wlr_log(L_ERROR, "got unknown cursor config: %s", name);
|
|
||||||
}
|
|
||||||
} else if (strncmp(device_prefix, section, strlen(device_prefix)) == 0) {
|
} else if (strncmp(device_prefix, section, strlen(device_prefix)) == 0) {
|
||||||
const char *device_name = section + strlen(device_prefix);
|
const char *device_name = section + strlen(device_prefix);
|
||||||
|
|
||||||
|
@ -297,7 +322,7 @@ static int config_ini_handler(void *user, const char *section, const char *name,
|
||||||
if (!found) {
|
if (!found) {
|
||||||
dc = calloc(1, sizeof(struct roots_device_config));
|
dc = calloc(1, sizeof(struct roots_device_config));
|
||||||
dc->name = strdup(device_name);
|
dc->name = strdup(device_name);
|
||||||
dc->seat = strdup("seat0");
|
dc->seat = strdup(ROOTS_CONFIG_DEFAULT_SEAT_NAME);
|
||||||
wl_list_insert(&config->devices, &dc->link);
|
wl_list_insert(&config->devices, &dc->link);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -338,6 +363,7 @@ struct roots_config *roots_config_create_from_args(int argc, char *argv[]) {
|
||||||
wl_list_init(&config->outputs);
|
wl_list_init(&config->outputs);
|
||||||
wl_list_init(&config->devices);
|
wl_list_init(&config->devices);
|
||||||
wl_list_init(&config->keyboards);
|
wl_list_init(&config->keyboards);
|
||||||
|
wl_list_init(&config->cursors);
|
||||||
wl_list_init(&config->bindings);
|
wl_list_init(&config->bindings);
|
||||||
|
|
||||||
int c;
|
int c;
|
||||||
|
@ -418,6 +444,15 @@ void roots_config_destroy(struct roots_config *config) {
|
||||||
free(kc);
|
free(kc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct roots_cursor_config *cc, *ctmp = NULL;
|
||||||
|
wl_list_for_each_safe(cc, ctmp, &config->cursors, link) {
|
||||||
|
free(cc->seat);
|
||||||
|
free(cc->mapped_output);
|
||||||
|
free(cc->mapped_box);
|
||||||
|
free(cc->theme);
|
||||||
|
free(cc);
|
||||||
|
}
|
||||||
|
|
||||||
struct roots_binding_config *bc, *btmp = NULL;
|
struct roots_binding_config *bc, *btmp = NULL;
|
||||||
wl_list_for_each_safe(bc, btmp, &config->bindings, link) {
|
wl_list_for_each_safe(bc, btmp, &config->bindings, link) {
|
||||||
free(bc->keysyms);
|
free(bc->keysyms);
|
||||||
|
@ -426,8 +461,6 @@ void roots_config_destroy(struct roots_config *config) {
|
||||||
}
|
}
|
||||||
|
|
||||||
free(config->config_path);
|
free(config->config_path);
|
||||||
free(config->cursor.mapped_output);
|
|
||||||
free(config->cursor.mapped_box);
|
|
||||||
free(config);
|
free(config);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -457,13 +490,33 @@ struct roots_device_config *roots_config_get_device(struct roots_config *config,
|
||||||
|
|
||||||
struct roots_keyboard_config *roots_config_get_keyboard(
|
struct roots_keyboard_config *roots_config_get_keyboard(
|
||||||
struct roots_config *config, struct wlr_input_device *device) {
|
struct roots_config *config, struct wlr_input_device *device) {
|
||||||
|
const char *device_name = "";
|
||||||
|
if (device != NULL) {
|
||||||
|
device_name = device->name;
|
||||||
|
}
|
||||||
|
|
||||||
struct roots_keyboard_config *kc;
|
struct roots_keyboard_config *kc;
|
||||||
wl_list_for_each(kc, &config->keyboards, link) {
|
wl_list_for_each(kc, &config->keyboards, link) {
|
||||||
if ((device != NULL && strcmp(kc->name, device->name) == 0) ||
|
if (strcmp(kc->name, device_name) == 0) {
|
||||||
(device == NULL && strcmp(kc->name, "") == 0)) {
|
|
||||||
return kc;
|
return kc;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct roots_cursor_config *roots_config_get_cursor(struct roots_config *config,
|
||||||
|
const char *seat_name) {
|
||||||
|
if (seat_name == NULL) {
|
||||||
|
seat_name = ROOTS_CONFIG_DEFAULT_SEAT_NAME;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct roots_cursor_config *cc;
|
||||||
|
wl_list_for_each(cc, &config->cursors, link) {
|
||||||
|
if (strcmp(cc->seat, seat_name) == 0) {
|
||||||
|
return cc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
|
@ -338,11 +338,18 @@ struct roots_desktop *desktop_create(struct roots_server *server,
|
||||||
desktop->server = server;
|
desktop->server = server;
|
||||||
desktop->config = config;
|
desktop->config = config;
|
||||||
|
|
||||||
desktop->xcursor_manager = wlr_xcursor_manager_create(config->cursor.theme,
|
const char *cursor_theme = NULL;
|
||||||
|
struct roots_cursor_config *cc =
|
||||||
|
roots_config_get_cursor(config, ROOTS_CONFIG_DEFAULT_SEAT_NAME);
|
||||||
|
if (cc != NULL) {
|
||||||
|
cursor_theme = cc->theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
desktop->xcursor_manager = wlr_xcursor_manager_create(cursor_theme,
|
||||||
ROOTS_XCURSOR_SIZE);
|
ROOTS_XCURSOR_SIZE);
|
||||||
if (desktop->xcursor_manager == NULL) {
|
if (desktop->xcursor_manager == NULL) {
|
||||||
wlr_log(L_ERROR, "Cannot create XCursor manager for theme %s",
|
wlr_log(L_ERROR, "Cannot create XCursor manager for theme %s",
|
||||||
config->cursor.theme);
|
cursor_theme);
|
||||||
wlr_list_free(desktop->views);
|
wlr_list_free(desktop->views);
|
||||||
free(desktop);
|
free(desktop);
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
|
@ -43,7 +43,7 @@ static void input_add_notify(struct wl_listener *listener, void *data) {
|
||||||
struct wlr_input_device *device = data;
|
struct wlr_input_device *device = data;
|
||||||
struct roots_input *input = wl_container_of(listener, input, input_add);
|
struct roots_input *input = wl_container_of(listener, input, input_add);
|
||||||
|
|
||||||
char *seat_name = "seat0";
|
char *seat_name = ROOTS_CONFIG_DEFAULT_SEAT_NAME;
|
||||||
struct roots_device_config *dc =
|
struct roots_device_config *dc =
|
||||||
roots_config_get_device(input->config, device);
|
roots_config_get_device(input->config, device);
|
||||||
if (dc) {
|
if (dc) {
|
||||||
|
|
|
@ -157,20 +157,29 @@ void roots_seat_configure_cursor(struct roots_seat *seat) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// configure device to output mappings
|
// configure device to output mappings
|
||||||
const char *mapped_output = config->cursor.mapped_output;
|
const char *mapped_output = NULL;
|
||||||
|
struct roots_cursor_config *cc =
|
||||||
|
roots_config_get_cursor(config, seat->seat->name);
|
||||||
|
if (cc != NULL) {
|
||||||
|
mapped_output = cc->mapped_output;
|
||||||
|
}
|
||||||
wl_list_for_each(output, &desktop->outputs, link) {
|
wl_list_for_each(output, &desktop->outputs, link) {
|
||||||
if (mapped_output && strcmp(mapped_output, output->wlr_output->name) == 0) {
|
if (mapped_output &&
|
||||||
|
strcmp(mapped_output, output->wlr_output->name) == 0) {
|
||||||
wlr_cursor_map_to_output(cursor, output->wlr_output);
|
wlr_cursor_map_to_output(cursor, output->wlr_output);
|
||||||
}
|
}
|
||||||
|
|
||||||
wl_list_for_each(pointer, &seat->pointers, link) {
|
wl_list_for_each(pointer, &seat->pointers, link) {
|
||||||
seat_set_device_output_mappings(seat, pointer->device, output->wlr_output);
|
seat_set_device_output_mappings(seat, pointer->device,
|
||||||
|
output->wlr_output);
|
||||||
}
|
}
|
||||||
wl_list_for_each(tablet_tool, &seat->tablet_tools, link) {
|
wl_list_for_each(tablet_tool, &seat->tablet_tools, link) {
|
||||||
seat_set_device_output_mappings(seat, tablet_tool->device, output->wlr_output);
|
seat_set_device_output_mappings(seat, tablet_tool->device,
|
||||||
|
output->wlr_output);
|
||||||
}
|
}
|
||||||
wl_list_for_each(touch, &seat->touch, link) {
|
wl_list_for_each(touch, &seat->touch, link) {
|
||||||
seat_set_device_output_mappings(seat, touch->device, output->wlr_output);
|
seat_set_device_output_mappings(seat, touch->device,
|
||||||
|
output->wlr_output);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -185,9 +194,6 @@ static void roots_seat_init_cursor(struct roots_seat *seat) {
|
||||||
struct roots_desktop *desktop = seat->input->server->desktop;
|
struct roots_desktop *desktop = seat->input->server->desktop;
|
||||||
wlr_cursor_attach_output_layout(wlr_cursor, desktop->layout);
|
wlr_cursor_attach_output_layout(wlr_cursor, desktop->layout);
|
||||||
|
|
||||||
// TODO: be able to configure per-seat cursor themes
|
|
||||||
seat->cursor->xcursor_manager = desktop->xcursor_manager;
|
|
||||||
|
|
||||||
wl_list_init(&seat->cursor->touch_points);
|
wl_list_init(&seat->cursor->touch_points);
|
||||||
|
|
||||||
roots_seat_configure_cursor(seat);
|
roots_seat_configure_cursor(seat);
|
||||||
|
@ -446,6 +452,21 @@ void roots_seat_remove_device(struct roots_seat *seat,
|
||||||
}
|
}
|
||||||
|
|
||||||
void roots_seat_configure_xcursor(struct roots_seat *seat) {
|
void roots_seat_configure_xcursor(struct roots_seat *seat) {
|
||||||
|
const char *cursor_theme = NULL;
|
||||||
|
struct roots_cursor_config *cc =
|
||||||
|
roots_config_get_cursor(seat->input->config, seat->seat->name);
|
||||||
|
if (cc != NULL) {
|
||||||
|
cursor_theme = cc->theme;
|
||||||
|
}
|
||||||
|
|
||||||
|
seat->cursor->xcursor_manager =
|
||||||
|
wlr_xcursor_manager_create(cursor_theme, ROOTS_XCURSOR_SIZE);
|
||||||
|
if (seat->cursor->xcursor_manager == NULL) {
|
||||||
|
wlr_log(L_ERROR, "Cannot create XCursor manager for theme %s",
|
||||||
|
cursor_theme);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
struct roots_output *output;
|
struct roots_output *output;
|
||||||
wl_list_for_each(output, &seat->input->server->desktop->outputs, link) {
|
wl_list_for_each(output, &seat->input->server->desktop->outputs, link) {
|
||||||
if (wlr_xcursor_manager_load(seat->cursor->xcursor_manager,
|
if (wlr_xcursor_manager_load(seat->cursor->xcursor_manager,
|
||||||
|
|
Loading…
Reference in a new issue