2021-09-24 21:45:48 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2020-06-09 21:40:46 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
2020-09-28 20:53:59 +01:00
|
|
|
#include <assert.h>
|
2020-09-28 20:41:41 +01:00
|
|
|
#include <fcntl.h>
|
2024-03-20 11:31:42 +08:00
|
|
|
#include <glib.h>
|
2020-06-05 23:04:54 +01:00
|
|
|
#include <libxml/parser.h>
|
|
|
|
|
#include <libxml/tree.h>
|
2020-09-28 20:41:41 +01:00
|
|
|
#include <stdbool.h>
|
|
|
|
|
#include <stdio.h>
|
2023-06-19 08:46:56 +08:00
|
|
|
#include <stdlib.h>
|
2020-09-28 20:41:41 +01:00
|
|
|
#include <string.h>
|
|
|
|
|
#include <strings.h>
|
2020-06-05 23:04:54 +01:00
|
|
|
#include <unistd.h>
|
2020-06-16 07:21:53 +01:00
|
|
|
#include <wayland-server-core.h>
|
2022-07-06 17:21:02 +02:00
|
|
|
#include <wlr/util/box.h>
|
2021-07-22 21:30:17 +01:00
|
|
|
#include <wlr/util/log.h>
|
2022-01-05 09:11:24 +01:00
|
|
|
#include "action.h"
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
#include "common/dir.h"
|
2022-10-05 08:43:56 +02:00
|
|
|
#include "common/list.h"
|
2023-10-20 20:24:29 -04:00
|
|
|
#include "common/macros.h"
|
2022-09-16 18:41:02 -04:00
|
|
|
#include "common/mem.h"
|
2021-09-06 21:54:00 +01:00
|
|
|
#include "common/nodename.h"
|
2023-03-26 22:34:44 +01:00
|
|
|
#include "common/parse-bool.h"
|
2024-03-27 22:13:50 -04:00
|
|
|
#include "common/parse-double.h"
|
2021-09-06 21:54:00 +01:00
|
|
|
#include "common/string-helpers.h"
|
2024-03-10 14:28:41 -04:00
|
|
|
#include "config/default-bindings.h"
|
2021-09-06 21:54:00 +01:00
|
|
|
#include "config/keybind.h"
|
2021-10-15 10:26:00 -04:00
|
|
|
#include "config/libinput.h"
|
2021-09-06 21:54:00 +01:00
|
|
|
#include "config/mousebind.h"
|
2023-12-29 10:06:17 +01:00
|
|
|
#include "config/tablet.h"
|
2021-09-06 21:54:00 +01:00
|
|
|
#include "config/rcxml.h"
|
2023-06-19 08:46:56 +08:00
|
|
|
#include "labwc.h"
|
2024-04-10 17:39:31 -05:00
|
|
|
#include "osd.h"
|
2022-07-06 17:21:02 +02:00
|
|
|
#include "regions.h"
|
2023-08-28 19:14:04 +03:00
|
|
|
#include "view.h"
|
2023-04-28 21:41:41 +01:00
|
|
|
#include "window-rules.h"
|
2022-06-15 02:07:22 +02:00
|
|
|
#include "workspaces.h"
|
2020-06-05 23:04:54 +01:00
|
|
|
|
2022-07-06 17:21:02 +02:00
|
|
|
static bool in_regions;
|
2023-06-06 19:52:57 +01:00
|
|
|
static bool in_usable_area_override;
|
2021-09-24 22:04:03 +01:00
|
|
|
static bool in_keybind;
|
|
|
|
|
static bool in_mousebind;
|
2024-01-19 10:06:53 +01:00
|
|
|
static bool in_touch;
|
2021-10-15 10:26:00 -04:00
|
|
|
static bool in_libinput_category;
|
2023-04-19 14:44:41 +01:00
|
|
|
static bool in_window_switcher_field;
|
2023-04-28 21:41:41 +01:00
|
|
|
static bool in_window_rules;
|
2023-08-28 19:14:04 +03:00
|
|
|
static bool in_action_query;
|
|
|
|
|
static bool in_action_then_branch;
|
|
|
|
|
static bool in_action_else_branch;
|
2024-04-13 21:24:04 +02:00
|
|
|
static bool in_action_none_branch;
|
2023-04-19 14:44:41 +01:00
|
|
|
|
2023-06-06 19:52:57 +01:00
|
|
|
static struct usable_area_override *current_usable_area_override;
|
2020-06-19 22:00:22 +01:00
|
|
|
static struct keybind *current_keybind;
|
2021-09-06 21:54:00 +01:00
|
|
|
static struct mousebind *current_mousebind;
|
2024-01-19 10:06:53 +01:00
|
|
|
static struct touch_config_entry *current_touch;
|
2021-10-15 10:26:00 -04:00
|
|
|
static struct libinput_category *current_libinput_category;
|
2021-09-06 21:54:00 +01:00
|
|
|
static const char *current_mouse_context;
|
2022-01-05 09:11:24 +01:00
|
|
|
static struct action *current_keybind_action;
|
|
|
|
|
static struct action *current_mousebind_action;
|
2022-07-06 17:21:02 +02:00
|
|
|
static struct region *current_region;
|
2023-04-19 14:44:41 +01:00
|
|
|
static struct window_switcher_field *current_field;
|
2023-04-28 21:41:41 +01:00
|
|
|
static struct window_rule *current_window_rule;
|
|
|
|
|
static struct action *current_window_rule_action;
|
2023-08-28 19:14:04 +03:00
|
|
|
static struct view_query *current_view_query;
|
|
|
|
|
static struct action *current_child_action;
|
2020-06-05 23:04:54 +01:00
|
|
|
|
2021-09-24 22:04:03 +01:00
|
|
|
enum font_place {
|
2022-11-09 22:41:31 +00:00
|
|
|
FONT_PLACE_NONE = 0,
|
|
|
|
|
FONT_PLACE_UNKNOWN,
|
2020-07-31 11:11:50 +01:00
|
|
|
FONT_PLACE_ACTIVEWINDOW,
|
2023-12-07 09:28:27 +01:00
|
|
|
FONT_PLACE_INACTIVEWINDOW,
|
2021-08-20 20:27:52 +01:00
|
|
|
FONT_PLACE_MENUITEM,
|
2021-12-06 17:44:28 -05:00
|
|
|
FONT_PLACE_OSD,
|
2020-07-31 11:11:50 +01:00
|
|
|
/* TODO: Add all places based on Openbox's rc.xml */
|
|
|
|
|
};
|
|
|
|
|
|
2021-09-06 21:54:00 +01:00
|
|
|
static void load_default_key_bindings(void);
|
2022-08-29 23:43:13 +02:00
|
|
|
static void load_default_mouse_bindings(void);
|
2021-08-22 17:00:22 +01:00
|
|
|
|
2024-04-19 20:15:49 +02:00
|
|
|
static int
|
|
|
|
|
parse_window_type(const char *type)
|
|
|
|
|
{
|
|
|
|
|
if (!type) {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
if (!strcasecmp(type, "desktop")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_DESKTOP;
|
|
|
|
|
} else if (!strcasecmp(type, "dock")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_DOCK;
|
|
|
|
|
} else if (!strcasecmp(type, "toolbar")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_TOOLBAR;
|
|
|
|
|
} else if (!strcasecmp(type, "menu")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_MENU;
|
|
|
|
|
} else if (!strcasecmp(type, "utility")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_UTILITY;
|
|
|
|
|
} else if (!strcasecmp(type, "splash")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_SPLASH;
|
|
|
|
|
} else if (!strcasecmp(type, "dialog")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_DIALOG;
|
|
|
|
|
} else if (!strcasecmp(type, "dropdown_menu")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_DROPDOWN_MENU;
|
|
|
|
|
} else if (!strcasecmp(type, "popup_menu")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_POPUP_MENU;
|
|
|
|
|
} else if (!strcasecmp(type, "tooltip")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_TOOLTIP;
|
|
|
|
|
} else if (!strcasecmp(type, "notification")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_NOTIFICATION;
|
|
|
|
|
} else if (!strcasecmp(type, "combo")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_COMBO;
|
|
|
|
|
} else if (!strcasecmp(type, "dnd")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_DND;
|
|
|
|
|
} else if (!strcasecmp(type, "normal")) {
|
|
|
|
|
return NET_WM_WINDOW_TYPE_NORMAL;
|
|
|
|
|
} else {
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-06 19:52:57 +01:00
|
|
|
static void
|
|
|
|
|
fill_usable_area_override(char *nodename, char *content)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(nodename, "margin")) {
|
|
|
|
|
current_usable_area_override = znew(*current_usable_area_override);
|
|
|
|
|
wl_list_append(&rc.usable_area_overrides, ¤t_usable_area_override->link);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
string_truncate_at_pattern(nodename, ".margin");
|
|
|
|
|
if (!content) {
|
|
|
|
|
/* nop */
|
|
|
|
|
} else if (!current_usable_area_override) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no usable-area-override object");
|
|
|
|
|
} else if (!strcmp(nodename, "output")) {
|
|
|
|
|
free(current_usable_area_override->output);
|
|
|
|
|
current_usable_area_override->output = xstrdup(content);
|
|
|
|
|
} else if (!strcmp(nodename, "left")) {
|
|
|
|
|
current_usable_area_override->margin.left = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "right")) {
|
|
|
|
|
current_usable_area_override->margin.right = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "top")) {
|
|
|
|
|
current_usable_area_override->margin.top = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "bottom")) {
|
|
|
|
|
current_usable_area_override->margin.bottom = atoi(content);
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "Unexpected data usable-area-override parser: %s=\"%s\"",
|
|
|
|
|
nodename, content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-28 21:41:41 +01:00
|
|
|
/* Does a boolean-parse but also allows 'default' */
|
|
|
|
|
static void
|
|
|
|
|
set_property(const char *str, enum property *variable)
|
|
|
|
|
{
|
|
|
|
|
if (!str || !strcasecmp(str, "default")) {
|
|
|
|
|
*variable = LAB_PROP_UNSET;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
int ret = parse_bool(str, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
*variable = ret ? LAB_PROP_TRUE : LAB_PROP_FALSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fill_window_rule(char *nodename, char *content)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(nodename, "windowRule.windowRules")) {
|
|
|
|
|
current_window_rule = znew(*current_window_rule);
|
2024-04-19 20:15:49 +02:00
|
|
|
current_window_rule->window_type = -1; // Window types are >= 0
|
2023-04-28 21:41:41 +01:00
|
|
|
wl_list_append(&rc.window_rules, ¤t_window_rule->link);
|
|
|
|
|
wl_list_init(¤t_window_rule->actions);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string_truncate_at_pattern(nodename, ".windowrule.windowrules");
|
|
|
|
|
if (!content) {
|
|
|
|
|
/* nop */
|
|
|
|
|
} else if (!current_window_rule) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no window-rule");
|
2023-05-09 21:20:05 +01:00
|
|
|
|
|
|
|
|
/* Criteria */
|
2023-04-28 21:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "identifier")) {
|
|
|
|
|
free(current_window_rule->identifier);
|
|
|
|
|
current_window_rule->identifier = xstrdup(content);
|
2023-05-09 21:20:05 +01:00
|
|
|
} else if (!strcmp(nodename, "title")) {
|
|
|
|
|
free(current_window_rule->title);
|
|
|
|
|
current_window_rule->title = xstrdup(content);
|
2024-04-19 20:15:49 +02:00
|
|
|
} else if (!strcmp(nodename, "type")) {
|
|
|
|
|
current_window_rule->window_type = parse_window_type(content);
|
2023-06-07 21:44:38 +01:00
|
|
|
} else if (!strcasecmp(nodename, "matchOnce")) {
|
|
|
|
|
set_bool(content, ¤t_window_rule->match_once);
|
2024-05-16 12:52:47 +03:00
|
|
|
} else if (!strcasecmp(nodename, "sandboxEngine")) {
|
|
|
|
|
free(current_window_rule->sandbox_engine);
|
|
|
|
|
current_window_rule->sandbox_engine = xstrdup(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "sandboxAppId")) {
|
|
|
|
|
free(current_window_rule->sandbox_app_id);
|
|
|
|
|
current_window_rule->sandbox_app_id = xstrdup(content);
|
2023-05-09 21:20:05 +01:00
|
|
|
|
|
|
|
|
/* Event */
|
2023-04-28 21:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "event")) {
|
|
|
|
|
/*
|
|
|
|
|
* This is just in readiness for adding any other types of
|
|
|
|
|
* events in the future. We default to onFirstMap anyway.
|
|
|
|
|
*/
|
|
|
|
|
if (!strcasecmp(content, "onFirstMap")) {
|
|
|
|
|
current_window_rule->event = LAB_WINDOW_RULE_EVENT_ON_FIRST_MAP;
|
|
|
|
|
}
|
2023-05-09 21:20:05 +01:00
|
|
|
|
|
|
|
|
/* Properties */
|
2023-04-28 21:41:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "serverDecoration")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->server_decoration);
|
2023-05-20 10:20:36 +01:00
|
|
|
} else if (!strcasecmp(nodename, "skipTaskbar")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->skip_taskbar);
|
|
|
|
|
} else if (!strcasecmp(nodename, "skipWindowSwitcher")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->skip_window_switcher);
|
2023-09-03 14:01:40 +02:00
|
|
|
} else if (!strcasecmp(nodename, "ignoreFocusRequest")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->ignore_focus_request);
|
2024-04-13 20:36:09 +02:00
|
|
|
} else if (!strcasecmp(nodename, "ignoreConfigureRequest")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->ignore_configure_request);
|
2023-11-09 21:44:51 +00:00
|
|
|
} else if (!strcasecmp(nodename, "fixedPosition")) {
|
|
|
|
|
set_property(content, ¤t_window_rule->fixed_position);
|
2023-05-09 21:20:05 +01:00
|
|
|
|
|
|
|
|
/* Actions */
|
2023-04-28 21:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "name.action")) {
|
|
|
|
|
current_window_rule_action = action_create(content);
|
|
|
|
|
if (current_window_rule_action) {
|
|
|
|
|
wl_list_append(¤t_window_rule->actions,
|
|
|
|
|
¤t_window_rule_action->link);
|
|
|
|
|
}
|
|
|
|
|
} else if (!current_window_rule_action) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <action name=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
|
|
|
|
} else {
|
|
|
|
|
action_arg_from_xml_node(current_window_rule_action, nodename, content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-19 14:44:41 +01:00
|
|
|
static void
|
|
|
|
|
fill_window_switcher_field(char *nodename, char *content)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(nodename, "field.fields.windowswitcher")) {
|
2024-04-10 17:39:31 -05:00
|
|
|
current_field = osd_field_create();
|
2023-04-19 14:44:41 +01:00
|
|
|
wl_list_append(&rc.window_switcher.fields, ¤t_field->link);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string_truncate_at_pattern(nodename, ".field.fields.windowswitcher");
|
|
|
|
|
if (!content) {
|
|
|
|
|
/* intentionally left empty */
|
|
|
|
|
} else if (!current_field) {
|
|
|
|
|
wlr_log(WLR_ERROR, "no <field>");
|
|
|
|
|
} else {
|
2024-04-10 17:39:31 -05:00
|
|
|
osd_field_arg_from_xml_node(current_field, nodename, content);
|
2023-04-19 14:44:41 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-06 17:21:02 +02:00
|
|
|
static void
|
|
|
|
|
fill_region(char *nodename, char *content)
|
|
|
|
|
{
|
|
|
|
|
string_truncate_at_pattern(nodename, ".region.regions");
|
|
|
|
|
|
|
|
|
|
if (!strcasecmp(nodename, "region.regions")) {
|
|
|
|
|
current_region = znew(*current_region);
|
|
|
|
|
wl_list_append(&rc.regions, ¤t_region->link);
|
|
|
|
|
} else if (!content) {
|
|
|
|
|
/* intentionally left empty */
|
|
|
|
|
} else if (!current_region) {
|
|
|
|
|
wlr_log(WLR_ERROR, "Expecting <region name=\"\" before %s='%s'",
|
|
|
|
|
nodename, content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "name")) {
|
|
|
|
|
/* Prevent leaking memory if config contains multiple names */
|
|
|
|
|
if (!current_region->name) {
|
|
|
|
|
current_region->name = xstrdup(content);
|
|
|
|
|
}
|
|
|
|
|
} else if (strstr("xywidtheight", nodename) && !strchr(content, '%')) {
|
|
|
|
|
wlr_log(WLR_ERROR, "Removing invalid region '%s': %s='%s' misses"
|
|
|
|
|
" a trailing %%", current_region->name, nodename, content);
|
|
|
|
|
wl_list_remove(¤t_region->link);
|
|
|
|
|
zfree(current_region->name);
|
|
|
|
|
zfree(current_region);
|
|
|
|
|
} else if (!strcmp(nodename, "x")) {
|
|
|
|
|
current_region->percentage.x = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "y")) {
|
|
|
|
|
current_region->percentage.y = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "width")) {
|
|
|
|
|
current_region->percentage.width = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "height")) {
|
|
|
|
|
current_region->percentage.height = atoi(content);
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "Unexpected data in region parser: %s=\"%s\"",
|
|
|
|
|
nodename, content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-28 19:14:04 +03:00
|
|
|
static void
|
|
|
|
|
fill_action_query(char *nodename, char *content, struct action *action)
|
|
|
|
|
{
|
2024-05-22 03:36:33 +02:00
|
|
|
if (!action) {
|
|
|
|
|
wlr_log(WLR_ERROR, "No parent action for query: %s=%s", nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-28 19:14:04 +03:00
|
|
|
string_truncate_at_pattern(nodename, ".keybind.keyboard");
|
|
|
|
|
string_truncate_at_pattern(nodename, ".mousebind.context.mouse");
|
|
|
|
|
|
|
|
|
|
if (!strcasecmp(nodename, "query.action")) {
|
|
|
|
|
current_view_query = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string_truncate_at_pattern(nodename, ".query.action");
|
|
|
|
|
|
|
|
|
|
if (!content) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!current_view_query) {
|
|
|
|
|
struct wl_list *queries = action_get_querylist(action, "query");
|
|
|
|
|
if (!queries) {
|
|
|
|
|
action_arg_add_querylist(action, "query");
|
|
|
|
|
queries = action_get_querylist(action, "query");
|
|
|
|
|
}
|
2024-05-27 19:55:16 +02:00
|
|
|
current_view_query = view_query_create();
|
2023-08-28 19:14:04 +03:00
|
|
|
wl_list_append(queries, ¤t_view_query->link);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!strcasecmp(nodename, "identifier")) {
|
|
|
|
|
current_view_query->identifier = xstrdup(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "title")) {
|
|
|
|
|
current_view_query->title = xstrdup(content);
|
2024-04-20 07:57:35 +02:00
|
|
|
} else if (!strcmp(nodename, "type")) {
|
|
|
|
|
current_view_query->window_type = parse_window_type(content);
|
2024-05-16 12:52:47 +03:00
|
|
|
} else if (!strcasecmp(nodename, "sandboxEngine")) {
|
|
|
|
|
current_view_query->sandbox_engine = xstrdup(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "sandboxAppId")) {
|
|
|
|
|
current_view_query->sandbox_app_id = xstrdup(content);
|
2023-08-28 19:14:04 +03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fill_child_action(char *nodename, char *content, struct action *parent,
|
|
|
|
|
const char *branch_name)
|
|
|
|
|
{
|
2024-05-22 03:36:33 +02:00
|
|
|
if (!parent) {
|
|
|
|
|
wlr_log(WLR_ERROR, "No parent action for branch: %s=%s", nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-28 19:14:04 +03:00
|
|
|
string_truncate_at_pattern(nodename, ".keybind.keyboard");
|
|
|
|
|
string_truncate_at_pattern(nodename, ".mousebind.context.mouse");
|
|
|
|
|
string_truncate_at_pattern(nodename, ".then.action");
|
|
|
|
|
string_truncate_at_pattern(nodename, ".else.action");
|
2024-04-13 21:24:04 +02:00
|
|
|
string_truncate_at_pattern(nodename, ".none.action");
|
2023-08-28 19:14:04 +03:00
|
|
|
|
|
|
|
|
if (!strcasecmp(nodename, "action")) {
|
|
|
|
|
current_child_action = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!content) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct wl_list *siblings = action_get_actionlist(parent, branch_name);
|
|
|
|
|
if (!siblings) {
|
|
|
|
|
action_arg_add_actionlist(parent, branch_name);
|
|
|
|
|
siblings = action_get_actionlist(parent, branch_name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!strcasecmp(nodename, "name.action")) {
|
|
|
|
|
if (!strcasecmp(content, "If") || !strcasecmp(content, "ForEach")) {
|
|
|
|
|
wlr_log(WLR_ERROR, "action '%s' cannot be a child action", content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
current_child_action = action_create(content);
|
|
|
|
|
if (current_child_action) {
|
|
|
|
|
wl_list_append(siblings, ¤t_child_action->link);
|
|
|
|
|
}
|
|
|
|
|
} else if (!current_child_action) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <action name=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
|
|
|
|
} else {
|
|
|
|
|
action_arg_from_xml_node(current_child_action, nodename, content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
fill_keybind(char *nodename, char *content)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!content) {
|
2020-06-19 22:00:22 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2021-02-16 21:03:38 +00:00
|
|
|
string_truncate_at_pattern(nodename, ".keybind.keyboard");
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!strcmp(nodename, "key")) {
|
2020-09-02 21:05:28 +01:00
|
|
|
current_keybind = keybind_create(content);
|
2022-01-06 03:24:32 +01:00
|
|
|
current_keybind_action = NULL;
|
|
|
|
|
/*
|
|
|
|
|
* If an invalid keybind has been provided,
|
|
|
|
|
* keybind_create() complains.
|
|
|
|
|
*/
|
|
|
|
|
if (!current_keybind) {
|
|
|
|
|
wlr_log(WLR_ERROR, "Invalid keybind: %s", content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else if (!current_keybind) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <keybind key=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
2024-04-26 11:05:12 +01:00
|
|
|
} else if (!strcasecmp(nodename, "onRelease")) {
|
|
|
|
|
set_bool(content, ¤t_keybind->on_release);
|
2023-09-11 15:15:37 +02:00
|
|
|
} else if (!strcasecmp(nodename, "layoutDependent")) {
|
|
|
|
|
set_bool(content, ¤t_keybind->use_syms_only);
|
2022-01-06 03:24:32 +01:00
|
|
|
} else if (!strcmp(nodename, "name.action")) {
|
2022-01-05 09:11:24 +01:00
|
|
|
current_keybind_action = action_create(content);
|
2023-01-29 04:06:46 +01:00
|
|
|
if (current_keybind_action) {
|
|
|
|
|
wl_list_append(¤t_keybind->actions,
|
|
|
|
|
¤t_keybind_action->link);
|
|
|
|
|
}
|
2022-01-06 03:24:32 +01:00
|
|
|
} else if (!current_keybind_action) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <action name=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
2022-10-30 14:26:57 +00:00
|
|
|
} else {
|
2023-03-26 10:57:53 +01:00
|
|
|
/*
|
|
|
|
|
* Here we deal with action sub-elements such as <to>, <output>,
|
|
|
|
|
* <region>, <direction> and so on. This is common to key- and
|
|
|
|
|
* mousebinds.
|
|
|
|
|
*/
|
2022-12-10 15:28:25 +01:00
|
|
|
action_arg_from_xml_node(current_keybind_action, nodename, content);
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-30 22:03:41 -04:00
|
|
|
static void
|
2021-08-31 17:43:49 -04:00
|
|
|
fill_mousebind(char *nodename, char *content)
|
2021-08-30 22:03:41 -04:00
|
|
|
{
|
|
|
|
|
/*
|
2021-09-06 21:54:00 +01:00
|
|
|
* Example of what we are parsing:
|
2021-08-30 22:03:41 -04:00
|
|
|
* <mousebind button="Left" action="DoubleClick">
|
2021-12-23 05:37:57 +01:00
|
|
|
* <action name="Focus"/>
|
|
|
|
|
* <action name="Raise"/>
|
2021-08-30 22:03:41 -04:00
|
|
|
* <action name="ToggleMaximize"/>
|
|
|
|
|
* </mousebind>
|
|
|
|
|
*/
|
2021-09-06 21:54:00 +01:00
|
|
|
|
2022-01-06 03:24:32 +01:00
|
|
|
if (!current_mouse_context) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <context name=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
} else if (!strcmp(nodename, "mousebind.context.mouse")) {
|
2021-09-06 21:54:00 +01:00
|
|
|
wlr_log(WLR_INFO, "create mousebind for %s",
|
|
|
|
|
current_mouse_context);
|
|
|
|
|
current_mousebind = mousebind_create(current_mouse_context);
|
2022-01-06 03:24:32 +01:00
|
|
|
current_mousebind_action = NULL;
|
|
|
|
|
return;
|
|
|
|
|
} else if (!content) {
|
2021-08-30 22:03:41 -04:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-06 03:24:32 +01:00
|
|
|
string_truncate_at_pattern(nodename, ".mousebind.context.mouse");
|
|
|
|
|
if (!current_mousebind) {
|
|
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
|
"expect <mousebind button=\"\" action=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
|
|
|
|
} else if (!strcmp(nodename, "button")) {
|
2021-12-01 02:32:24 +00:00
|
|
|
current_mousebind->button = mousebind_button_from_str(content,
|
|
|
|
|
¤t_mousebind->modifiers);
|
2022-11-09 05:18:14 +00:00
|
|
|
} else if (!strcmp(nodename, "direction")) {
|
|
|
|
|
current_mousebind->direction = mousebind_direction_from_str(content,
|
|
|
|
|
¤t_mousebind->modifiers);
|
2021-08-31 17:43:49 -04:00
|
|
|
} else if (!strcmp(nodename, "action")) {
|
2022-01-06 03:24:32 +01:00
|
|
|
/* <mousebind button="" action="EVENT"> */
|
2021-09-06 21:54:00 +01:00
|
|
|
current_mousebind->mouse_event =
|
|
|
|
|
mousebind_event_from_str(content);
|
|
|
|
|
} else if (!strcmp(nodename, "name.action")) {
|
2022-01-05 09:11:24 +01:00
|
|
|
current_mousebind_action = action_create(content);
|
2023-01-29 04:06:46 +01:00
|
|
|
if (current_mousebind_action) {
|
|
|
|
|
wl_list_append(¤t_mousebind->actions,
|
|
|
|
|
¤t_mousebind_action->link);
|
|
|
|
|
}
|
2022-01-06 03:24:32 +01:00
|
|
|
} else if (!current_mousebind_action) {
|
|
|
|
|
wlr_log(WLR_ERROR, "expect <action name=\"\"> element first. "
|
|
|
|
|
"nodename: '%s' content: '%s'", nodename, content);
|
2022-10-30 14:26:57 +00:00
|
|
|
} else {
|
2022-12-10 15:28:25 +01:00
|
|
|
action_arg_from_xml_node(current_mousebind_action, nodename, content);
|
2021-08-30 22:03:41 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-19 10:06:53 +01:00
|
|
|
static void
|
|
|
|
|
fill_touch(char *nodename, char *content)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(nodename, "touch")) {
|
|
|
|
|
current_touch = znew(*current_touch);
|
|
|
|
|
wl_list_append(&rc.touch_configs, ¤t_touch->link);
|
|
|
|
|
} else if (!strcasecmp(nodename, "deviceName.touch")) {
|
|
|
|
|
current_touch->device_name = xstrdup(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "mapToOutput.touch")) {
|
|
|
|
|
current_touch->output_name = xstrdup(content);
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "Unexpected data in touch parser: %s=\"%s\"",
|
|
|
|
|
nodename, content);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-07 23:43:58 +09:00
|
|
|
static int
|
2021-10-09 09:10:26 -04:00
|
|
|
get_accel_profile(const char *s)
|
|
|
|
|
{
|
|
|
|
|
if (!s) {
|
2023-09-07 23:43:58 +09:00
|
|
|
return -1;
|
2021-10-09 09:10:26 -04:00
|
|
|
}
|
|
|
|
|
if (!strcasecmp(s, "flat")) {
|
|
|
|
|
return LIBINPUT_CONFIG_ACCEL_PROFILE_FLAT;
|
|
|
|
|
}
|
|
|
|
|
if (!strcasecmp(s, "adaptive")) {
|
|
|
|
|
return LIBINPUT_CONFIG_ACCEL_PROFILE_ADAPTIVE;
|
|
|
|
|
}
|
2023-09-07 23:43:58 +09:00
|
|
|
return -1;
|
2021-10-09 09:10:26 -04:00
|
|
|
}
|
|
|
|
|
|
2023-06-26 07:36:54 +08:00
|
|
|
static int
|
|
|
|
|
get_send_events_mode(const char *s)
|
|
|
|
|
{
|
|
|
|
|
if (!s) {
|
|
|
|
|
goto err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int ret = parse_bool(s, -1);
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
|
return ret
|
|
|
|
|
? LIBINPUT_CONFIG_SEND_EVENTS_ENABLED
|
|
|
|
|
: LIBINPUT_CONFIG_SEND_EVENTS_DISABLED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!strcasecmp(s, "disabledOnExternalMouse")) {
|
|
|
|
|
return LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err:
|
|
|
|
|
wlr_log(WLR_INFO, "Not a recognised send events mode");
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-15 10:26:00 -04:00
|
|
|
static void
|
|
|
|
|
fill_libinput_category(char *nodename, char *content)
|
|
|
|
|
{
|
2024-01-01 17:08:20 +00:00
|
|
|
/*
|
|
|
|
|
* Create a new profile (libinput-category) on `<libinput><device>`
|
|
|
|
|
* so that the 'default' profile can be created without even providing a
|
|
|
|
|
* category="" attribute (same as <device category="default">...)
|
|
|
|
|
*/
|
|
|
|
|
if (!strcmp(nodename, "device.libinput")) {
|
2021-10-15 10:26:00 -04:00
|
|
|
current_libinput_category = libinput_category_create();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!content) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!current_libinput_category) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string_truncate_at_pattern(nodename, ".device.libinput");
|
|
|
|
|
|
|
|
|
|
if (!strcmp(nodename, "category")) {
|
2023-12-29 19:50:37 +00:00
|
|
|
/*
|
|
|
|
|
* First we try to get a type based on a number of pre-defined
|
|
|
|
|
* terms, for example: 'default', 'touch', 'touchpad' and
|
|
|
|
|
* 'non-touch'
|
|
|
|
|
*/
|
|
|
|
|
current_libinput_category->type = get_device_type(content);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If we couldn't match against any of those terms, we use the
|
|
|
|
|
* provided value to define the device name that the settings
|
|
|
|
|
* should be applicable to.
|
|
|
|
|
*/
|
|
|
|
|
if (current_libinput_category->type == LAB_LIBINPUT_DEVICE_NONE) {
|
2022-09-16 18:41:02 -04:00
|
|
|
current_libinput_category->name = xstrdup(content);
|
2021-10-15 10:26:00 -04:00
|
|
|
}
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "naturalScroll")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool_as_int(content, ¤t_libinput_category->natural_scroll);
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "leftHanded")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool_as_int(content, ¤t_libinput_category->left_handed);
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "pointerSpeed")) {
|
2024-03-27 22:13:50 -04:00
|
|
|
set_float(content, ¤t_libinput_category->pointer_speed);
|
2021-10-15 10:26:00 -04:00
|
|
|
if (current_libinput_category->pointer_speed < -1) {
|
|
|
|
|
current_libinput_category->pointer_speed = -1;
|
|
|
|
|
} else if (current_libinput_category->pointer_speed > 1) {
|
|
|
|
|
current_libinput_category->pointer_speed = 1;
|
|
|
|
|
}
|
|
|
|
|
} else if (!strcasecmp(nodename, "tap")) {
|
2023-03-26 22:34:44 +01:00
|
|
|
int ret = parse_bool(content, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-04-26 20:34:48 +01:00
|
|
|
current_libinput_category->tap = ret
|
|
|
|
|
? LIBINPUT_CONFIG_TAP_ENABLED
|
2023-03-26 22:34:44 +01:00
|
|
|
: LIBINPUT_CONFIG_TAP_DISABLED;
|
2021-12-12 22:12:30 +00:00
|
|
|
} else if (!strcasecmp(nodename, "tapButtonMap")) {
|
|
|
|
|
if (!strcmp(content, "lrm")) {
|
2022-04-04 20:53:36 +01:00
|
|
|
current_libinput_category->tap_button_map =
|
|
|
|
|
LIBINPUT_CONFIG_TAP_MAP_LRM;
|
2021-12-12 22:12:30 +00:00
|
|
|
} else if (!strcmp(content, "lmr")) {
|
2022-04-04 20:53:36 +01:00
|
|
|
current_libinput_category->tap_button_map =
|
|
|
|
|
LIBINPUT_CONFIG_TAP_MAP_LMR;
|
2021-12-12 22:12:30 +00:00
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid tapButtonMap");
|
|
|
|
|
}
|
2023-09-02 17:32:48 +09:00
|
|
|
} else if (!strcasecmp(nodename, "tapAndDrag")) {
|
|
|
|
|
int ret = parse_bool(content, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
current_libinput_category->tap_and_drag = ret
|
|
|
|
|
? LIBINPUT_CONFIG_DRAG_ENABLED
|
|
|
|
|
: LIBINPUT_CONFIG_DRAG_DISABLED;
|
|
|
|
|
} else if (!strcasecmp(nodename, "dragLock")) {
|
|
|
|
|
int ret = parse_bool(content, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
current_libinput_category->drag_lock = ret
|
|
|
|
|
? LIBINPUT_CONFIG_DRAG_LOCK_ENABLED
|
|
|
|
|
: LIBINPUT_CONFIG_DRAG_LOCK_DISABLED;
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "accelProfile")) {
|
2022-04-04 20:53:36 +01:00
|
|
|
current_libinput_category->accel_profile =
|
|
|
|
|
get_accel_profile(content);
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "middleEmulation")) {
|
2023-03-26 22:34:44 +01:00
|
|
|
int ret = parse_bool(content, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
current_libinput_category->middle_emu = ret
|
|
|
|
|
? LIBINPUT_CONFIG_MIDDLE_EMULATION_ENABLED
|
|
|
|
|
: LIBINPUT_CONFIG_MIDDLE_EMULATION_DISABLED;
|
2021-10-19 21:44:44 +01:00
|
|
|
} else if (!strcasecmp(nodename, "disableWhileTyping")) {
|
2023-03-26 22:34:44 +01:00
|
|
|
int ret = parse_bool(content, -1);
|
|
|
|
|
if (ret < 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-04-26 20:34:48 +01:00
|
|
|
current_libinput_category->dwt = ret
|
|
|
|
|
? LIBINPUT_CONFIG_DWT_ENABLED
|
2023-03-26 22:34:44 +01:00
|
|
|
: LIBINPUT_CONFIG_DWT_DISABLED;
|
2024-02-21 18:19:48 +01:00
|
|
|
} else if (!strcasecmp(nodename, "clickMethod")) {
|
|
|
|
|
if (!strcasecmp(content, "none")) {
|
|
|
|
|
current_libinput_category->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_NONE;
|
|
|
|
|
} else if (!strcasecmp(content, "clickfinger")) {
|
|
|
|
|
current_libinput_category->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_CLICKFINGER;
|
|
|
|
|
} else if (!strcasecmp(content, "buttonAreas")) {
|
|
|
|
|
current_libinput_category->click_method =
|
|
|
|
|
LIBINPUT_CONFIG_CLICK_METHOD_BUTTON_AREAS;
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid clickMethod");
|
|
|
|
|
}
|
2023-06-26 07:36:54 +08:00
|
|
|
} else if (!strcasecmp(nodename, "sendEventsMode")) {
|
|
|
|
|
current_libinput_category->send_events_mode =
|
|
|
|
|
get_send_events_mode(content);
|
2024-03-20 03:32:41 +08:00
|
|
|
} else if (!strcasecmp(nodename, "calibrationMatrix")) {
|
2024-03-20 11:31:42 +08:00
|
|
|
errno = 0;
|
|
|
|
|
current_libinput_category->have_calibration_matrix = true;
|
|
|
|
|
float *mat = current_libinput_category->calibration_matrix;
|
|
|
|
|
gchar **elements = g_strsplit(content, " ", -1);
|
2024-03-21 16:46:44 +08:00
|
|
|
guint i = 0;
|
|
|
|
|
for (; elements[i]; ++i) {
|
2024-03-20 11:31:42 +08:00
|
|
|
char *end_str = NULL;
|
|
|
|
|
mat[i] = strtof(elements[i], &end_str);
|
2024-03-20 16:19:49 +08:00
|
|
|
if (errno == ERANGE || *end_str != '\0' || i == 6 || *elements[i] == '\0') {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid calibration matrix element"
|
|
|
|
|
" %s (index %d), expect six floats",
|
|
|
|
|
elements[i], i);
|
2024-03-20 11:31:42 +08:00
|
|
|
current_libinput_category->have_calibration_matrix = false;
|
|
|
|
|
errno = 0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2024-03-20 03:32:41 +08:00
|
|
|
}
|
2024-03-21 16:46:44 +08:00
|
|
|
if (i != 6 && current_libinput_category->have_calibration_matrix) {
|
|
|
|
|
wlr_log(WLR_ERROR, "wrong number of calibration matrix elements,"
|
|
|
|
|
" expected 6, got %d", i);
|
|
|
|
|
current_libinput_category->have_calibration_matrix = false;
|
|
|
|
|
}
|
2024-03-20 11:31:42 +08:00
|
|
|
g_strfreev(elements);
|
2021-10-15 10:26:00 -04:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-15 10:53:49 -04:00
|
|
|
static void
|
|
|
|
|
set_font_attr(struct font *font, const char *nodename, const char *content)
|
|
|
|
|
{
|
|
|
|
|
if (!strcmp(nodename, "name")) {
|
|
|
|
|
zfree(font->name);
|
2022-09-16 18:41:02 -04:00
|
|
|
font->name = xstrdup(content);
|
2022-09-15 10:53:49 -04:00
|
|
|
} else if (!strcmp(nodename, "size")) {
|
|
|
|
|
font->size = atoi(content);
|
|
|
|
|
} else if (!strcmp(nodename, "slant")) {
|
|
|
|
|
font->slant = !strcasecmp(content, "italic") ?
|
|
|
|
|
FONT_SLANT_ITALIC : FONT_SLANT_NORMAL;
|
|
|
|
|
} else if (!strcmp(nodename, "weight")) {
|
|
|
|
|
font->weight = !strcasecmp(content, "bold") ?
|
|
|
|
|
FONT_WEIGHT_BOLD : FONT_WEIGHT_NORMAL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
fill_font(char *nodename, char *content, enum font_place place)
|
2020-07-31 11:11:50 +01:00
|
|
|
{
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!content) {
|
2020-07-31 11:11:50 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2021-02-16 21:03:38 +00:00
|
|
|
string_truncate_at_pattern(nodename, ".font.theme");
|
2020-07-31 11:11:50 +01:00
|
|
|
|
2021-07-19 20:46:32 +01:00
|
|
|
switch (place) {
|
2022-11-09 22:41:31 +00:00
|
|
|
case FONT_PLACE_NONE:
|
2021-07-19 20:46:32 +01:00
|
|
|
/*
|
|
|
|
|
* If <theme><font></font></theme> is used without a place=""
|
|
|
|
|
* attribute, we set all font variables
|
|
|
|
|
*/
|
2022-09-15 10:53:49 -04:00
|
|
|
set_font_attr(&rc.font_activewindow, nodename, content);
|
2023-12-07 09:28:27 +01:00
|
|
|
set_font_attr(&rc.font_inactivewindow, nodename, content);
|
2022-09-15 10:53:49 -04:00
|
|
|
set_font_attr(&rc.font_menuitem, nodename, content);
|
|
|
|
|
set_font_attr(&rc.font_osd, nodename, content);
|
2021-07-19 20:46:32 +01:00
|
|
|
break;
|
|
|
|
|
case FONT_PLACE_ACTIVEWINDOW:
|
2022-09-15 10:53:49 -04:00
|
|
|
set_font_attr(&rc.font_activewindow, nodename, content);
|
2021-07-19 20:46:32 +01:00
|
|
|
break;
|
2023-12-07 09:28:27 +01:00
|
|
|
case FONT_PLACE_INACTIVEWINDOW:
|
|
|
|
|
set_font_attr(&rc.font_inactivewindow, nodename, content);
|
|
|
|
|
break;
|
2021-08-20 20:27:52 +01:00
|
|
|
case FONT_PLACE_MENUITEM:
|
2022-09-15 10:53:49 -04:00
|
|
|
set_font_attr(&rc.font_menuitem, nodename, content);
|
2021-08-20 20:27:52 +01:00
|
|
|
break;
|
2021-12-06 17:44:28 -05:00
|
|
|
case FONT_PLACE_OSD:
|
2022-09-15 10:53:49 -04:00
|
|
|
set_font_attr(&rc.font_osd, nodename, content);
|
2021-12-06 17:44:28 -05:00
|
|
|
break;
|
2021-07-19 20:46:32 +01:00
|
|
|
|
2021-08-31 17:43:49 -04:00
|
|
|
/* TODO: implement for all font places */
|
2021-07-19 20:46:32 +01:00
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-07-31 11:11:50 +01:00
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static enum font_place
|
|
|
|
|
enum_font_place(const char *place)
|
2020-07-31 11:11:50 +01:00
|
|
|
{
|
2022-11-09 22:41:31 +00:00
|
|
|
if (!place || place[0] == '\0') {
|
|
|
|
|
return FONT_PLACE_NONE;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
|
|
|
|
if (!strcasecmp(place, "ActiveWindow")) {
|
2020-07-31 11:11:50 +01:00
|
|
|
return FONT_PLACE_ACTIVEWINDOW;
|
2023-12-07 09:28:27 +01:00
|
|
|
} else if (!strcasecmp(place, "InactiveWindow")) {
|
|
|
|
|
return FONT_PLACE_INACTIVEWINDOW;
|
2021-08-20 20:27:52 +01:00
|
|
|
} else if (!strcasecmp(place, "MenuItem")) {
|
|
|
|
|
return FONT_PLACE_MENUITEM;
|
2022-09-15 22:03:52 +01:00
|
|
|
} else if (!strcasecmp(place, "OnScreenDisplay")
|
|
|
|
|
|| !strcasecmp(place, "OSD")) {
|
2021-12-06 17:44:28 -05:00
|
|
|
return FONT_PLACE_OSD;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-07-31 11:11:50 +01:00
|
|
|
return FONT_PLACE_UNKNOWN;
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-30 17:35:41 +02:00
|
|
|
static void
|
|
|
|
|
set_adaptive_sync_mode(const char *str, enum adaptive_sync_mode *variable)
|
|
|
|
|
{
|
|
|
|
|
if (!strcasecmp(str, "fullscreen")) {
|
|
|
|
|
*variable = LAB_ADAPTIVE_SYNC_FULLSCREEN;
|
|
|
|
|
} else {
|
|
|
|
|
int ret = parse_bool(str, -1);
|
|
|
|
|
if (ret == 1) {
|
|
|
|
|
*variable = LAB_ADAPTIVE_SYNC_ENABLED;
|
|
|
|
|
} else {
|
|
|
|
|
*variable = LAB_ADAPTIVE_SYNC_DISABLED;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
entry(xmlNode *node, char *nodename, char *content)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
2021-07-19 20:46:32 +01:00
|
|
|
/* current <theme><font place=""></font></theme> */
|
2022-11-09 22:41:31 +00:00
|
|
|
static enum font_place font_place = FONT_PLACE_NONE;
|
2020-07-31 11:11:50 +01:00
|
|
|
|
2023-12-29 10:06:17 +01:00
|
|
|
static uint32_t button_map_from;
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!nodename) {
|
2020-06-05 23:04:54 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2021-02-16 21:03:38 +00:00
|
|
|
string_truncate_at_pattern(nodename, ".openbox_config");
|
2021-07-16 17:30:07 +01:00
|
|
|
string_truncate_at_pattern(nodename, ".labwc_config");
|
2020-07-31 11:11:50 +01:00
|
|
|
|
2021-03-30 22:40:41 +01:00
|
|
|
if (getenv("LABWC_DEBUG_CONFIG_NODENAMES")) {
|
|
|
|
|
printf("%s: %s\n", nodename, content);
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
2020-07-31 11:11:50 +01:00
|
|
|
|
2023-06-06 19:52:57 +01:00
|
|
|
if (in_usable_area_override) {
|
|
|
|
|
fill_usable_area_override(nodename, content);
|
|
|
|
|
}
|
2021-09-06 21:54:00 +01:00
|
|
|
if (in_keybind) {
|
2023-08-28 19:14:04 +03:00
|
|
|
if (in_action_query) {
|
|
|
|
|
fill_action_query(nodename, content,
|
|
|
|
|
current_keybind_action);
|
|
|
|
|
} else if (in_action_then_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_keybind_action, "then");
|
|
|
|
|
} else if (in_action_else_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_keybind_action, "else");
|
2024-04-13 21:24:04 +02:00
|
|
|
} else if (in_action_none_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_keybind_action, "none");
|
2023-08-28 19:14:04 +03:00
|
|
|
} else {
|
|
|
|
|
fill_keybind(nodename, content);
|
|
|
|
|
}
|
2021-09-06 21:54:00 +01:00
|
|
|
}
|
|
|
|
|
if (in_mousebind) {
|
2023-08-28 19:14:04 +03:00
|
|
|
if (in_action_query) {
|
|
|
|
|
fill_action_query(nodename, content,
|
|
|
|
|
current_mousebind_action);
|
|
|
|
|
} else if (in_action_then_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_mousebind_action, "then");
|
|
|
|
|
} else if (in_action_else_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_mousebind_action, "else");
|
2024-04-13 21:24:04 +02:00
|
|
|
} else if (in_action_none_branch) {
|
|
|
|
|
fill_child_action(nodename, content,
|
|
|
|
|
current_mousebind_action, "none");
|
2023-08-28 19:14:04 +03:00
|
|
|
} else {
|
|
|
|
|
fill_mousebind(nodename, content);
|
|
|
|
|
}
|
2021-09-06 21:54:00 +01:00
|
|
|
}
|
2024-01-19 10:06:53 +01:00
|
|
|
if (in_touch) {
|
|
|
|
|
fill_touch(nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-10-15 10:26:00 -04:00
|
|
|
if (in_libinput_category) {
|
|
|
|
|
fill_libinput_category(nodename, content);
|
2024-01-01 17:11:51 +00:00
|
|
|
return;
|
2021-10-15 10:26:00 -04:00
|
|
|
}
|
2022-07-06 17:21:02 +02:00
|
|
|
if (in_regions) {
|
|
|
|
|
fill_region(nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-04-19 14:44:41 +01:00
|
|
|
if (in_window_switcher_field) {
|
|
|
|
|
fill_window_switcher_field(nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-04-28 21:41:41 +01:00
|
|
|
if (in_window_rules) {
|
|
|
|
|
fill_window_rule(nodename, content);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-09-06 21:54:00 +01:00
|
|
|
|
2021-08-22 19:09:31 +01:00
|
|
|
/* handle nodes without content, e.g. <keyboard><default /> */
|
|
|
|
|
if (!strcmp(nodename, "default.keyboard")) {
|
|
|
|
|
load_default_key_bindings();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2022-08-29 23:43:13 +02:00
|
|
|
if (!strcmp(nodename, "devault.mouse")
|
|
|
|
|
|| !strcmp(nodename, "default.mouse")) {
|
|
|
|
|
load_default_mouse_bindings();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-08-22 19:09:31 +01:00
|
|
|
|
2023-12-29 10:06:17 +01:00
|
|
|
if (!strcasecmp(nodename, "map.tablet")) {
|
|
|
|
|
button_map_from = UINT32_MAX;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-22 19:09:31 +01:00
|
|
|
/* handle the rest */
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!content) {
|
2020-06-05 23:04:54 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2021-09-22 20:25:57 +01:00
|
|
|
if (!strcmp(nodename, "place.font.theme")) {
|
2020-07-31 11:11:50 +01:00
|
|
|
font_place = enum_font_place(content);
|
2022-11-09 22:41:31 +00:00
|
|
|
if (font_place == FONT_PLACE_UNKNOWN) {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid font place %s", content);
|
|
|
|
|
}
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-07-31 11:11:50 +01:00
|
|
|
|
2021-07-16 17:38:17 +01:00
|
|
|
if (!strcmp(nodename, "decoration.core")) {
|
|
|
|
|
if (!strcmp(content, "client")) {
|
|
|
|
|
rc.xdg_shell_server_side_deco = false;
|
|
|
|
|
} else {
|
|
|
|
|
rc.xdg_shell_server_side_deco = true;
|
|
|
|
|
}
|
2021-08-22 14:32:19 +01:00
|
|
|
} else if (!strcmp(nodename, "gap.core")) {
|
|
|
|
|
rc.gap = atoi(content);
|
2022-07-14 01:01:10 +02:00
|
|
|
} else if (!strcasecmp(nodename, "adaptiveSync.core")) {
|
2023-12-30 17:35:41 +02:00
|
|
|
set_adaptive_sync_mode(content, &rc.adaptive_sync);
|
2024-01-08 22:58:58 +02:00
|
|
|
} else if (!strcasecmp(nodename, "allowTearing.core")) {
|
|
|
|
|
set_bool(content, &rc.allow_tearing);
|
2024-01-08 18:19:12 -05:00
|
|
|
if (rc.allow_tearing) {
|
|
|
|
|
char *no_atomic_env = getenv("WLR_DRM_NO_ATOMIC");
|
|
|
|
|
if (!no_atomic_env || strcmp(no_atomic_env, "1") != 0) {
|
|
|
|
|
rc.allow_tearing = false;
|
|
|
|
|
wlr_log(WLR_ERROR, "tearing requires WLR_DRM_NO_ATOMIC=1");
|
|
|
|
|
}
|
2024-01-08 22:58:58 +02:00
|
|
|
}
|
2023-01-23 07:15:55 +01:00
|
|
|
} else if (!strcasecmp(nodename, "reuseOutputMode.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.reuse_output_mode);
|
2023-12-10 21:00:05 -05:00
|
|
|
} else if (!strcmp(nodename, "policy.placement")) {
|
2024-05-07 09:46:05 -04:00
|
|
|
rc.placement_policy = view_placement_parse(content);
|
|
|
|
|
if (rc.placement_policy == LAB_PLACE_INVALID) {
|
2023-12-10 21:00:05 -05:00
|
|
|
rc.placement_policy = LAB_PLACE_CENTER;
|
|
|
|
|
}
|
2024-07-03 13:43:04 -04:00
|
|
|
} else if (!strcasecmp(nodename, "xwaylandPersistence.core")) {
|
|
|
|
|
set_bool(content, &rc.xwayland_persistence);
|
2020-09-28 20:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "name.theme")) {
|
2022-09-16 18:41:02 -04:00
|
|
|
rc.theme_name = xstrdup(content);
|
2021-03-30 22:39:52 +01:00
|
|
|
} else if (!strcmp(nodename, "cornerradius.theme")) {
|
|
|
|
|
rc.corner_radius = atoi(content);
|
2023-04-14 09:01:13 +02:00
|
|
|
} else if (!strcasecmp(nodename, "keepBorder.theme")) {
|
|
|
|
|
set_bool(content, &rc.ssd_keep_border);
|
2024-04-22 19:27:53 +01:00
|
|
|
} else if (!strcasecmp(nodename, "dropShadows.theme")) {
|
|
|
|
|
set_bool(content, &rc.shadows_enabled);
|
2020-09-28 20:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "name.font.theme")) {
|
2020-07-31 11:11:50 +01:00
|
|
|
fill_font(nodename, content, font_place);
|
2020-09-28 20:41:41 +01:00
|
|
|
} else if (!strcmp(nodename, "size.font.theme")) {
|
2020-07-31 11:11:50 +01:00
|
|
|
fill_font(nodename, content, font_place);
|
2022-09-15 10:53:49 -04:00
|
|
|
} else if (!strcmp(nodename, "slant.font.theme")) {
|
|
|
|
|
fill_font(nodename, content, font_place);
|
|
|
|
|
} else if (!strcmp(nodename, "weight.font.theme")) {
|
|
|
|
|
fill_font(nodename, content, font_place);
|
2021-10-21 20:40:14 +01:00
|
|
|
} else if (!strcasecmp(nodename, "followMouse.focus")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.focus_follow_mouse);
|
2023-04-02 02:18:25 +02:00
|
|
|
} else if (!strcasecmp(nodename, "followMouseRequiresMovement.focus")) {
|
|
|
|
|
set_bool(content, &rc.focus_follow_mouse_requires_movement);
|
2021-10-21 20:40:14 +01:00
|
|
|
} else if (!strcasecmp(nodename, "raiseOnFocus.focus")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.raise_on_focus);
|
2021-08-31 17:43:49 -04:00
|
|
|
} else if (!strcasecmp(nodename, "doubleClickTime.mouse")) {
|
2021-08-30 22:03:41 -04:00
|
|
|
long doubleclick_time_parsed = strtol(content, NULL, 10);
|
2021-09-22 20:36:55 +01:00
|
|
|
if (doubleclick_time_parsed > 0) {
|
2021-08-30 22:03:41 -04:00
|
|
|
rc.doubleclick_time = doubleclick_time_parsed;
|
2021-09-22 20:36:55 +01:00
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "invalid doubleClickTime");
|
2021-08-30 22:03:41 -04:00
|
|
|
}
|
2023-03-25 15:57:47 +00:00
|
|
|
} else if (!strcasecmp(nodename, "scrollFactor.mouse")) {
|
2024-03-27 22:13:50 -04:00
|
|
|
set_double(content, &rc.scroll_factor);
|
2021-09-22 20:25:57 +01:00
|
|
|
} else if (!strcasecmp(nodename, "name.context.mouse")) {
|
2021-08-30 22:03:41 -04:00
|
|
|
current_mouse_context = content;
|
2022-01-06 03:24:32 +01:00
|
|
|
current_mousebind = NULL;
|
2023-06-06 19:52:57 +01:00
|
|
|
|
2021-10-21 20:40:14 +01:00
|
|
|
} else if (!strcasecmp(nodename, "repeatRate.keyboard")) {
|
2021-10-09 15:16:02 -04:00
|
|
|
rc.repeat_rate = atoi(content);
|
2021-10-21 20:40:14 +01:00
|
|
|
} else if (!strcasecmp(nodename, "repeatDelay.keyboard")) {
|
2021-10-09 15:16:02 -04:00
|
|
|
rc.repeat_delay = atoi(content);
|
2023-10-08 22:26:26 -04:00
|
|
|
} else if (!strcasecmp(nodename, "numlock.keyboard")) {
|
|
|
|
|
set_bool(content, &rc.kb_numlock_enable);
|
2023-09-07 16:41:47 +02:00
|
|
|
} else if (!strcasecmp(nodename, "layoutScope.keyboard")) {
|
|
|
|
|
/*
|
|
|
|
|
* This can be changed to an enum later on
|
|
|
|
|
* if we decide to also support "application".
|
|
|
|
|
*/
|
|
|
|
|
rc.kb_layout_per_window = !strcasecmp(content, "window");
|
2021-10-24 12:08:30 -04:00
|
|
|
} else if (!strcasecmp(nodename, "screenEdgeStrength.resistance")) {
|
2021-10-23 22:31:39 -04:00
|
|
|
rc.screen_edge_strength = atoi(content);
|
2024-01-20 21:59:46 -05:00
|
|
|
} else if (!strcasecmp(nodename, "windowEdgeStrength.resistance")) {
|
|
|
|
|
rc.window_edge_strength = atoi(content);
|
2022-01-08 11:25:18 +01:00
|
|
|
} else if (!strcasecmp(nodename, "range.snapping")) {
|
|
|
|
|
rc.snap_edge_range = atoi(content);
|
2024-04-21 02:22:28 +09:00
|
|
|
} else if (!strcasecmp(nodename, "enabled.overlay.snapping")) {
|
|
|
|
|
set_bool(content, &rc.snap_overlay_enabled);
|
|
|
|
|
} else if (!strcasecmp(nodename, "inner.delay.overlay.snapping")) {
|
|
|
|
|
rc.snap_overlay_delay_inner = atoi(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "outer.delay.overlay.snapping")) {
|
|
|
|
|
rc.snap_overlay_delay_outer = atoi(content);
|
2022-01-08 11:25:18 +01:00
|
|
|
} else if (!strcasecmp(nodename, "topMaximize.snapping")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.snap_top_maximize);
|
2024-01-22 22:11:53 -05:00
|
|
|
} else if (!strcasecmp(nodename, "notifyClient.snapping")) {
|
|
|
|
|
if (!strcasecmp(content, "always")) {
|
|
|
|
|
rc.snap_tiling_events_mode = LAB_TILING_EVENTS_ALWAYS;
|
|
|
|
|
} else if (!strcasecmp(content, "region")) {
|
|
|
|
|
rc.snap_tiling_events_mode = LAB_TILING_EVENTS_REGION;
|
|
|
|
|
} else if (!strcasecmp(content, "edge")) {
|
|
|
|
|
rc.snap_tiling_events_mode = LAB_TILING_EVENTS_EDGE;
|
|
|
|
|
} else if (!strcasecmp(content, "never")) {
|
|
|
|
|
rc.snap_tiling_events_mode = LAB_TILING_EVENTS_NEVER;
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "ignoring invalid value for notifyClient");
|
|
|
|
|
}
|
2023-03-11 22:15:32 +00:00
|
|
|
|
|
|
|
|
/* <windowSwitcher show="" preview="" outlines="" /> */
|
2023-04-19 14:34:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "show.windowSwitcher")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.show);
|
2023-04-19 14:34:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "preview.windowSwitcher")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.preview);
|
2023-04-19 14:34:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "outlines.windowSwitcher")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.outlines);
|
2024-03-03 20:03:32 +00:00
|
|
|
} else if (!strcasecmp(nodename, "allWorkspaces.windowSwitcher")) {
|
|
|
|
|
if (parse_bool(content, -1) == true) {
|
|
|
|
|
rc.window_switcher.criteria &=
|
|
|
|
|
~LAB_VIEW_CRITERIA_CURRENT_WORKSPACE;
|
|
|
|
|
}
|
2023-03-11 22:15:32 +00:00
|
|
|
|
2023-04-19 14:34:41 +01:00
|
|
|
/* Remove this long term - just a friendly warning for now */
|
|
|
|
|
} else if (strstr(nodename, "windowswitcher.core")) {
|
|
|
|
|
wlr_log(WLR_ERROR, "<windowSwitcher> should not be child of <core>");
|
|
|
|
|
|
2023-04-19 14:44:41 +01:00
|
|
|
/* The following three are for backward compatibility only */
|
|
|
|
|
} else if (!strcasecmp(nodename, "show.windowSwitcher.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.show);
|
2023-04-19 14:44:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "preview.windowSwitcher.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.preview);
|
2023-04-19 14:44:41 +01:00
|
|
|
} else if (!strcasecmp(nodename, "outlines.windowSwitcher.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.outlines);
|
2023-04-19 14:44:41 +01:00
|
|
|
|
2023-03-11 22:15:32 +00:00
|
|
|
/* The following three are for backward compatibility only */
|
2023-03-08 15:26:49 +01:00
|
|
|
} else if (!strcasecmp(nodename, "cycleViewOSD.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.show);
|
2023-03-11 22:15:32 +00:00
|
|
|
wlr_log(WLR_ERROR, "<cycleViewOSD> is deprecated."
|
|
|
|
|
" Use <windowSwitcher show=\"\" />");
|
2022-01-20 11:38:23 -08:00
|
|
|
} else if (!strcasecmp(nodename, "cycleViewPreview.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.preview);
|
2023-03-11 22:15:32 +00:00
|
|
|
wlr_log(WLR_ERROR, "<cycleViewPreview> is deprecated."
|
|
|
|
|
" Use <windowSwitcher preview=\"\" />");
|
2022-08-24 20:27:08 +02:00
|
|
|
} else if (!strcasecmp(nodename, "cycleViewOutlines.core")) {
|
2023-04-26 20:34:48 +01:00
|
|
|
set_bool(content, &rc.window_switcher.outlines);
|
2023-03-11 22:15:32 +00:00
|
|
|
wlr_log(WLR_ERROR, "<cycleViewOutlines> is deprecated."
|
|
|
|
|
" Use <windowSwitcher outlines=\"\" />");
|
|
|
|
|
|
2022-06-15 02:07:22 +02:00
|
|
|
} else if (!strcasecmp(nodename, "name.names.desktops")) {
|
2022-09-18 15:22:26 -04:00
|
|
|
struct workspace *workspace = znew(*workspace);
|
2022-09-16 18:41:02 -04:00
|
|
|
workspace->name = xstrdup(content);
|
2022-10-05 08:43:56 +02:00
|
|
|
wl_list_append(&rc.workspace_config.workspaces, &workspace->link);
|
2022-06-15 02:07:22 +02:00
|
|
|
} else if (!strcasecmp(nodename, "popupTime.desktops")) {
|
|
|
|
|
rc.workspace_config.popuptime = atoi(content);
|
2023-06-19 08:46:56 +08:00
|
|
|
} else if (!strcasecmp(nodename, "number.desktops")) {
|
2023-06-24 20:36:06 +08:00
|
|
|
rc.workspace_config.min_nr_workspaces = MAX(1, atoi(content));
|
2024-03-11 17:19:03 -05:00
|
|
|
} else if (!strcasecmp(nodename, "prefix.desktops")) {
|
|
|
|
|
rc.workspace_config.prefix = xstrdup(content);
|
2023-08-17 18:59:29 +02:00
|
|
|
} else if (!strcasecmp(nodename, "popupShow.resize")) {
|
|
|
|
|
if (!strcasecmp(content, "Always")) {
|
|
|
|
|
rc.resize_indicator = LAB_RESIZE_INDICATOR_ALWAYS;
|
|
|
|
|
} else if (!strcasecmp(content, "Never")) {
|
|
|
|
|
rc.resize_indicator = LAB_RESIZE_INDICATOR_NEVER;
|
|
|
|
|
} else if (!strcasecmp(content, "Nonpixel")) {
|
|
|
|
|
rc.resize_indicator = LAB_RESIZE_INDICATOR_NON_PIXEL;
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "Invalid value for <resize popupShow />");
|
|
|
|
|
}
|
2024-05-29 11:06:39 +09:00
|
|
|
} else if (!strcasecmp(nodename, "drawContents.resize")) {
|
|
|
|
|
set_bool(content, &rc.resize_draw_contents);
|
2024-04-22 17:21:55 +02:00
|
|
|
} else if (!strcasecmp(nodename, "mouseEmulation.tablet")) {
|
|
|
|
|
set_bool(content, &rc.tablet.force_mouse_emulation);
|
2024-01-07 22:19:05 +01:00
|
|
|
} else if (!strcasecmp(nodename, "mapToOutput.tablet")) {
|
|
|
|
|
rc.tablet.output_name = xstrdup(content);
|
2024-01-07 22:18:06 +01:00
|
|
|
} else if (!strcasecmp(nodename, "rotate.tablet")) {
|
|
|
|
|
rc.tablet.rotation = tablet_parse_rotation(atoi(content));
|
2024-01-01 18:26:38 +01:00
|
|
|
} else if (!strcasecmp(nodename, "left.area.tablet")) {
|
|
|
|
|
rc.tablet.box.x = tablet_get_dbl_if_positive(content, "left");
|
|
|
|
|
} else if (!strcasecmp(nodename, "top.area.tablet")) {
|
|
|
|
|
rc.tablet.box.y = tablet_get_dbl_if_positive(content, "top");
|
|
|
|
|
} else if (!strcasecmp(nodename, "width.area.tablet")) {
|
|
|
|
|
rc.tablet.box.width = tablet_get_dbl_if_positive(content, "width");
|
|
|
|
|
} else if (!strcasecmp(nodename, "height.area.tablet")) {
|
|
|
|
|
rc.tablet.box.height = tablet_get_dbl_if_positive(content, "height");
|
2023-12-29 10:06:17 +01:00
|
|
|
} else if (!strcasecmp(nodename, "button.map.tablet")) {
|
|
|
|
|
button_map_from = tablet_button_from_str(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "to.map.tablet")) {
|
|
|
|
|
if (button_map_from != UINT32_MAX) {
|
2024-01-06 17:56:32 +01:00
|
|
|
uint32_t button_map_to = mousebind_button_from_str(content, NULL);
|
2023-12-29 10:06:17 +01:00
|
|
|
if (button_map_to != UINT32_MAX) {
|
|
|
|
|
tablet_button_mapping_add(button_map_from, button_map_to);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
wlr_log(WLR_ERROR, "Missing 'button' argument for tablet button mapping");
|
|
|
|
|
}
|
2024-07-05 06:17:29 +02:00
|
|
|
} else if (!strcasecmp(nodename, "motion.tabletTool")) {
|
|
|
|
|
rc.tablet_tool.motion = tablet_parse_motion(content);
|
2024-04-26 09:03:17 +09:00
|
|
|
} else if (!strcasecmp(nodename, "ignoreButtonReleasePeriod.menu")) {
|
|
|
|
|
rc.menu_ignore_button_release_period = atoi(content);
|
2024-05-15 23:07:23 +01:00
|
|
|
} else if (!strcasecmp(nodename, "width.magnifier")) {
|
|
|
|
|
rc.mag_width = atoi(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "height.magnifier")) {
|
|
|
|
|
rc.mag_height = atoi(content);
|
|
|
|
|
} else if (!strcasecmp(nodename, "initScale.magnifier")) {
|
|
|
|
|
set_float(content, &rc.mag_scale);
|
|
|
|
|
} else if (!strcasecmp(nodename, "increment.magnifier")) {
|
|
|
|
|
set_float(content, &rc.mag_increment);
|
|
|
|
|
} else if (!strcasecmp(nodename, "useFilter.magnifier")) {
|
|
|
|
|
set_bool(content, &rc.mag_filter);
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
process_node(xmlNode *node)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
|
|
|
|
char *content;
|
|
|
|
|
static char buffer[256];
|
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
|
|
content = (char *)node->content;
|
2020-09-28 20:41:41 +01:00
|
|
|
if (xmlIsBlankNode(node)) {
|
2020-06-05 23:04:54 +01:00
|
|
|
return;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
name = nodename(node, buffer, sizeof(buffer));
|
|
|
|
|
entry(node, name, content);
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-06 21:54:00 +01:00
|
|
|
static void xml_tree_walk(xmlNode *node);
|
2020-06-05 23:04:54 +01:00
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
traverse(xmlNode *n)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
2021-10-15 20:52:36 +01:00
|
|
|
xmlAttr *attr;
|
|
|
|
|
|
2020-06-05 23:04:54 +01:00
|
|
|
process_node(n);
|
2021-10-15 20:52:36 +01:00
|
|
|
for (attr = n->properties; attr; attr = attr->next) {
|
2020-06-05 23:04:54 +01:00
|
|
|
xml_tree_walk(attr->children);
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
xml_tree_walk(n->children);
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
xml_tree_walk(xmlNode *node)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
|
|
|
|
for (xmlNode *n = node; n && n->name; n = n->next) {
|
2020-09-28 20:41:41 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "comment")) {
|
2020-06-05 23:04:54 +01:00
|
|
|
continue;
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2023-06-06 19:52:57 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "margin")) {
|
|
|
|
|
in_usable_area_override = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_usable_area_override = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "keybind")) {
|
2020-06-19 22:00:22 +01:00
|
|
|
in_keybind = true;
|
2020-06-05 23:04:54 +01:00
|
|
|
traverse(n);
|
2020-06-19 22:00:22 +01:00
|
|
|
in_keybind = false;
|
2020-06-05 23:04:54 +01:00
|
|
|
continue;
|
2021-08-31 17:43:49 -04:00
|
|
|
}
|
|
|
|
|
if (!strcasecmp((char *)n->name, "mousebind")) {
|
2021-08-30 22:03:41 -04:00
|
|
|
in_mousebind = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_mousebind = false;
|
2021-08-29 14:22:49 -04:00
|
|
|
continue;
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
2024-01-19 10:06:53 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "touch")) {
|
|
|
|
|
in_touch = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_touch = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2021-10-15 10:26:00 -04:00
|
|
|
if (!strcasecmp((char *)n->name, "device")) {
|
|
|
|
|
in_libinput_category = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_libinput_category = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2022-07-06 17:21:02 +02:00
|
|
|
if (!strcasecmp((char *)n->name, "regions")) {
|
|
|
|
|
in_regions = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_regions = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2023-04-19 14:44:41 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "fields")) {
|
|
|
|
|
in_window_switcher_field = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_window_switcher_field = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2023-04-28 21:41:41 +01:00
|
|
|
if (!strcasecmp((char *)n->name, "windowRules")) {
|
|
|
|
|
in_window_rules = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_window_rules = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2023-08-28 19:14:04 +03:00
|
|
|
if (!strcasecmp((char *)n->name, "query")) {
|
|
|
|
|
in_action_query = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_action_query = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!strcasecmp((char *)n->name, "then")) {
|
|
|
|
|
in_action_then_branch = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_action_then_branch = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!strcasecmp((char *)n->name, "else")) {
|
|
|
|
|
in_action_else_branch = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_action_else_branch = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2024-04-13 21:24:04 +02:00
|
|
|
if (!strcasecmp((char *)n->name, "none")) {
|
|
|
|
|
in_action_none_branch = true;
|
|
|
|
|
traverse(n);
|
|
|
|
|
in_action_none_branch = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
traverse(n);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-09 21:40:46 +01:00
|
|
|
/* Exposed in header file to allow unit tests to parse buffers */
|
2020-09-28 20:41:41 +01:00
|
|
|
void
|
|
|
|
|
rcxml_parse_xml(struct buf *b)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
2024-04-16 23:36:32 -04:00
|
|
|
xmlDoc *d = xmlParseMemory(b->data, b->len);
|
2020-06-05 23:04:54 +01:00
|
|
|
if (!d) {
|
2021-10-15 20:33:55 +01:00
|
|
|
wlr_log(WLR_ERROR, "error parsing config file");
|
|
|
|
|
return;
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
|
|
|
|
xml_tree_walk(xmlDocGetRootElement(d));
|
|
|
|
|
xmlFreeDoc(d);
|
|
|
|
|
xmlCleanupParser();
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-08 23:53:34 +01:00
|
|
|
static void
|
|
|
|
|
init_font_defaults(struct font *font)
|
|
|
|
|
{
|
|
|
|
|
font->size = 10;
|
|
|
|
|
font->slant = FONT_SLANT_NORMAL;
|
|
|
|
|
font->weight = FONT_WEIGHT_NORMAL;
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
2023-01-31 03:35:13 +01:00
|
|
|
rcxml_init(void)
|
2020-06-05 23:04:54 +01:00
|
|
|
{
|
2020-09-25 19:42:40 +01:00
|
|
|
static bool has_run;
|
|
|
|
|
|
2023-03-08 23:53:34 +01:00
|
|
|
if (!has_run) {
|
2023-06-06 19:52:57 +01:00
|
|
|
wl_list_init(&rc.usable_area_overrides);
|
2023-03-08 23:53:34 +01:00
|
|
|
wl_list_init(&rc.keybinds);
|
|
|
|
|
wl_list_init(&rc.mousebinds);
|
|
|
|
|
wl_list_init(&rc.libinput_categories);
|
|
|
|
|
wl_list_init(&rc.workspace_config.workspaces);
|
|
|
|
|
wl_list_init(&rc.regions);
|
2023-04-19 14:44:41 +01:00
|
|
|
wl_list_init(&rc.window_switcher.fields);
|
2023-04-28 21:41:41 +01:00
|
|
|
wl_list_init(&rc.window_rules);
|
2024-01-19 10:06:53 +01:00
|
|
|
wl_list_init(&rc.touch_configs);
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2020-09-25 19:42:40 +01:00
|
|
|
has_run = true;
|
2023-03-08 23:53:34 +01:00
|
|
|
|
2023-12-10 21:00:05 -05:00
|
|
|
rc.placement_policy = LAB_PLACE_CENTER;
|
|
|
|
|
|
2021-03-30 22:39:52 +01:00
|
|
|
rc.xdg_shell_server_side_deco = true;
|
2023-04-14 09:01:13 +02:00
|
|
|
rc.ssd_keep_border = true;
|
2021-03-30 22:39:52 +01:00
|
|
|
rc.corner_radius = 8;
|
2024-04-22 19:27:53 +01:00
|
|
|
rc.shadows_enabled = false;
|
2023-03-08 23:53:34 +01:00
|
|
|
|
2024-07-17 18:01:15 -07:00
|
|
|
rc.gap = 0;
|
|
|
|
|
rc.adaptive_sync = LAB_ADAPTIVE_SYNC_DISABLED;
|
|
|
|
|
rc.allow_tearing = false;
|
|
|
|
|
rc.reuse_output_mode = false;
|
|
|
|
|
rc.xwayland_persistence = false;
|
|
|
|
|
|
2023-03-08 23:53:34 +01:00
|
|
|
init_font_defaults(&rc.font_activewindow);
|
2023-12-07 09:28:27 +01:00
|
|
|
init_font_defaults(&rc.font_inactivewindow);
|
2023-03-08 23:53:34 +01:00
|
|
|
init_font_defaults(&rc.font_menuitem);
|
|
|
|
|
init_font_defaults(&rc.font_osd);
|
|
|
|
|
|
2023-04-02 02:18:25 +02:00
|
|
|
rc.focus_follow_mouse = false;
|
|
|
|
|
rc.focus_follow_mouse_requires_movement = true;
|
|
|
|
|
rc.raise_on_focus = false;
|
|
|
|
|
|
2021-08-29 18:31:57 -04:00
|
|
|
rc.doubleclick_time = 500;
|
2023-03-25 15:57:47 +00:00
|
|
|
rc.scroll_factor = 1.0;
|
2023-12-29 10:06:17 +01:00
|
|
|
|
2024-04-22 17:21:55 +02:00
|
|
|
rc.tablet.force_mouse_emulation = false;
|
2024-01-07 22:19:05 +01:00
|
|
|
rc.tablet.output_name = NULL;
|
|
|
|
|
rc.tablet.rotation = 0;
|
2024-01-01 18:26:38 +01:00
|
|
|
rc.tablet.box = (struct wlr_fbox){0};
|
2023-12-29 10:06:17 +01:00
|
|
|
tablet_load_default_button_mappings();
|
2024-07-05 06:17:29 +02:00
|
|
|
rc.tablet_tool.motion = LAB_TABLET_MOTION_ABSOLUTE;
|
2023-12-29 10:06:17 +01:00
|
|
|
|
2021-10-09 15:16:02 -04:00
|
|
|
rc.repeat_rate = 25;
|
|
|
|
|
rc.repeat_delay = 600;
|
2023-10-08 22:26:26 -04:00
|
|
|
rc.kb_numlock_enable = true;
|
2023-09-07 16:41:47 +02:00
|
|
|
rc.kb_layout_per_window = false;
|
2021-10-25 17:04:22 -04:00
|
|
|
rc.screen_edge_strength = 20;
|
2024-01-20 21:59:46 -05:00
|
|
|
rc.window_edge_strength = 20;
|
2023-04-02 02:18:25 +02:00
|
|
|
|
2022-01-08 11:25:18 +01:00
|
|
|
rc.snap_edge_range = 1;
|
2024-04-21 02:22:28 +09:00
|
|
|
rc.snap_overlay_enabled = true;
|
|
|
|
|
rc.snap_overlay_delay_inner = 500;
|
|
|
|
|
rc.snap_overlay_delay_outer = 500;
|
2022-01-08 11:25:18 +01:00
|
|
|
rc.snap_top_maximize = true;
|
2024-01-22 22:11:53 -05:00
|
|
|
rc.snap_tiling_events_mode = LAB_TILING_EVENTS_ALWAYS;
|
2023-04-20 22:31:26 +01:00
|
|
|
|
|
|
|
|
rc.window_switcher.show = true;
|
|
|
|
|
rc.window_switcher.preview = true;
|
|
|
|
|
rc.window_switcher.outlines = true;
|
2024-03-03 20:03:32 +00:00
|
|
|
rc.window_switcher.criteria = LAB_VIEW_CRITERIA_CURRENT_WORKSPACE
|
|
|
|
|
| LAB_VIEW_CRITERIA_ROOT_TOPLEVEL
|
|
|
|
|
| LAB_VIEW_CRITERIA_NO_SKIP_WINDOW_SWITCHER;
|
2023-04-20 22:31:26 +01:00
|
|
|
|
2023-08-17 18:59:29 +02:00
|
|
|
rc.resize_indicator = LAB_RESIZE_INDICATOR_NEVER;
|
2024-05-29 11:06:39 +09:00
|
|
|
rc.resize_draw_contents = true;
|
2023-08-17 18:59:29 +02:00
|
|
|
|
2022-06-15 02:07:22 +02:00
|
|
|
rc.workspace_config.popuptime = INT_MIN;
|
2023-06-24 20:36:06 +08:00
|
|
|
rc.workspace_config.min_nr_workspaces = 1;
|
2024-04-26 09:03:17 +09:00
|
|
|
|
|
|
|
|
rc.menu_ignore_button_release_period = 250;
|
2024-05-15 23:07:23 +01:00
|
|
|
|
|
|
|
|
rc.mag_width = 400;
|
|
|
|
|
rc.mag_height = 400;
|
|
|
|
|
rc.mag_scale = 2.0;
|
|
|
|
|
rc.mag_increment = 0.2;
|
|
|
|
|
rc.mag_filter = true;
|
2020-06-19 22:00:22 +01:00
|
|
|
}
|
|
|
|
|
|
2021-08-22 17:00:22 +01:00
|
|
|
static void
|
|
|
|
|
load_default_key_bindings(void)
|
|
|
|
|
{
|
2022-01-05 09:11:24 +01:00
|
|
|
struct keybind *k;
|
|
|
|
|
struct action *action;
|
2021-09-06 22:04:56 +01:00
|
|
|
for (int i = 0; key_combos[i].binding; i++) {
|
2024-03-10 14:28:41 -04:00
|
|
|
struct key_combos *current = &key_combos[i];
|
|
|
|
|
k = keybind_create(current->binding);
|
2021-09-06 22:04:56 +01:00
|
|
|
if (!k) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2022-01-05 09:11:24 +01:00
|
|
|
|
2024-03-10 14:28:41 -04:00
|
|
|
action = action_create(current->action);
|
2022-10-05 08:43:56 +02:00
|
|
|
wl_list_append(&k->actions, &action->link);
|
2022-01-05 09:11:24 +01:00
|
|
|
|
2024-03-10 14:28:41 -04:00
|
|
|
for (size_t j = 0; j < ARRAY_SIZE(current->attributes); j++) {
|
|
|
|
|
if (!current->attributes[j].name
|
|
|
|
|
|| !current->attributes[j].value) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
2023-08-06 10:26:04 +02:00
|
|
|
action_arg_from_xml_node(action,
|
2024-03-10 14:28:41 -04:00
|
|
|
current->attributes[j].name,
|
|
|
|
|
current->attributes[j].value);
|
2021-09-06 22:04:56 +01:00
|
|
|
}
|
|
|
|
|
}
|
2021-08-22 17:00:22 +01:00
|
|
|
}
|
|
|
|
|
|
2021-09-06 22:02:02 +01:00
|
|
|
static void
|
|
|
|
|
load_default_mouse_bindings(void)
|
|
|
|
|
{
|
2022-08-29 23:42:40 +02:00
|
|
|
uint32_t count = 0;
|
2022-01-05 09:11:24 +01:00
|
|
|
struct mousebind *m;
|
|
|
|
|
struct action *action;
|
2021-09-06 22:02:02 +01:00
|
|
|
for (int i = 0; mouse_combos[i].context; i++) {
|
2024-03-10 14:28:41 -04:00
|
|
|
struct mouse_combos *current = &mouse_combos[i];
|
2022-08-29 22:47:34 +02:00
|
|
|
if (i == 0
|
|
|
|
|
|| strcmp(current->context, mouse_combos[i - 1].context)
|
|
|
|
|
|| strcmp(current->button, mouse_combos[i - 1].button)
|
|
|
|
|
|| strcmp(current->event, mouse_combos[i - 1].event)) {
|
|
|
|
|
/* Create new mousebind */
|
|
|
|
|
m = mousebind_create(current->context);
|
|
|
|
|
m->mouse_event = mousebind_event_from_str(current->event);
|
2022-11-09 05:18:14 +00:00
|
|
|
if (m->mouse_event == MOUSE_ACTION_SCROLL) {
|
|
|
|
|
m->direction = mousebind_direction_from_str(current->button,
|
|
|
|
|
&m->modifiers);
|
|
|
|
|
} else {
|
|
|
|
|
m->button = mousebind_button_from_str(current->button,
|
|
|
|
|
&m->modifiers);
|
|
|
|
|
}
|
2022-08-29 23:42:40 +02:00
|
|
|
count++;
|
2022-08-29 22:47:34 +02:00
|
|
|
}
|
2022-01-05 09:11:24 +01:00
|
|
|
|
2022-08-29 22:47:34 +02:00
|
|
|
action = action_create(current->action);
|
2022-10-05 08:43:56 +02:00
|
|
|
wl_list_append(&m->actions, &action->link);
|
2022-01-05 09:11:24 +01:00
|
|
|
|
2024-03-10 14:28:41 -04:00
|
|
|
for (size_t j = 0; j < ARRAY_SIZE(current->attributes); j++) {
|
|
|
|
|
if (!current->attributes[j].name
|
|
|
|
|
|| !current->attributes[j].value) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
2023-08-06 10:26:04 +02:00
|
|
|
action_arg_from_xml_node(action,
|
2024-03-10 14:28:41 -04:00
|
|
|
current->attributes[j].name,
|
|
|
|
|
current->attributes[j].value);
|
2021-09-06 22:02:02 +01:00
|
|
|
}
|
|
|
|
|
}
|
2022-08-29 23:42:40 +02:00
|
|
|
wlr_log(WLR_DEBUG, "Loaded %u merged mousebinds", count);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2023-01-29 04:06:46 +01:00
|
|
|
deduplicate_mouse_bindings(void)
|
2022-08-29 23:42:40 +02:00
|
|
|
{
|
|
|
|
|
uint32_t replaced = 0;
|
2023-01-29 04:06:46 +01:00
|
|
|
uint32_t cleared = 0;
|
2022-08-29 23:42:40 +02:00
|
|
|
struct mousebind *current, *tmp, *existing;
|
|
|
|
|
wl_list_for_each_safe(existing, tmp, &rc.mousebinds, link) {
|
|
|
|
|
wl_list_for_each_reverse(current, &rc.mousebinds, link) {
|
|
|
|
|
if (existing == current) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
2023-01-29 04:06:46 +01:00
|
|
|
if (mousebind_the_same(existing, current)) {
|
2022-08-29 23:42:40 +02:00
|
|
|
wl_list_remove(&existing->link);
|
|
|
|
|
action_list_free(&existing->actions);
|
|
|
|
|
free(existing);
|
|
|
|
|
replaced++;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-01-29 04:06:46 +01:00
|
|
|
wl_list_for_each_safe(current, tmp, &rc.mousebinds, link) {
|
|
|
|
|
if (wl_list_empty(¤t->actions)) {
|
|
|
|
|
wl_list_remove(¤t->link);
|
|
|
|
|
free(current);
|
|
|
|
|
cleared++;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-08-29 23:42:40 +02:00
|
|
|
if (replaced) {
|
|
|
|
|
wlr_log(WLR_DEBUG, "Replaced %u mousebinds", replaced);
|
|
|
|
|
}
|
2023-01-29 04:06:46 +01:00
|
|
|
if (cleared) {
|
|
|
|
|
wlr_log(WLR_DEBUG, "Cleared %u mousebinds", cleared);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
deduplicate_key_bindings(void)
|
|
|
|
|
{
|
|
|
|
|
uint32_t replaced = 0;
|
|
|
|
|
uint32_t cleared = 0;
|
|
|
|
|
struct keybind *current, *tmp, *existing;
|
|
|
|
|
wl_list_for_each_safe(existing, tmp, &rc.keybinds, link) {
|
|
|
|
|
wl_list_for_each_reverse(current, &rc.keybinds, link) {
|
|
|
|
|
if (existing == current) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (keybind_the_same(existing, current)) {
|
|
|
|
|
wl_list_remove(&existing->link);
|
|
|
|
|
action_list_free(&existing->actions);
|
|
|
|
|
free(existing);
|
|
|
|
|
replaced++;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
wl_list_for_each_safe(current, tmp, &rc.keybinds, link) {
|
|
|
|
|
if (wl_list_empty(¤t->actions)) {
|
|
|
|
|
wl_list_remove(¤t->link);
|
|
|
|
|
free(current);
|
|
|
|
|
cleared++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (replaced) {
|
|
|
|
|
wlr_log(WLR_DEBUG, "Replaced %u keybinds", replaced);
|
|
|
|
|
}
|
|
|
|
|
if (cleared) {
|
|
|
|
|
wlr_log(WLR_DEBUG, "Cleared %u keybinds", cleared);
|
|
|
|
|
}
|
2021-09-06 22:02:02 +01:00
|
|
|
}
|
|
|
|
|
|
2023-04-19 14:44:41 +01:00
|
|
|
static struct {
|
|
|
|
|
enum window_switcher_field_content content;
|
|
|
|
|
int width;
|
|
|
|
|
} fields[] = {
|
|
|
|
|
{ LAB_FIELD_TYPE, 25 },
|
2023-12-27 13:55:49 +03:00
|
|
|
{ LAB_FIELD_TRIMMED_IDENTIFIER, 25 },
|
2023-04-19 14:44:41 +01:00
|
|
|
{ LAB_FIELD_TITLE, 50 },
|
|
|
|
|
{ LAB_FIELD_NONE, 0 },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
load_default_window_switcher_fields(void)
|
|
|
|
|
{
|
|
|
|
|
struct window_switcher_field *field;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; fields[i].content != LAB_FIELD_NONE; i++) {
|
|
|
|
|
field = znew(*field);
|
|
|
|
|
field->content = fields[i].content;
|
|
|
|
|
field->width = fields[i].width;
|
|
|
|
|
wl_list_append(&rc.window_switcher.fields, &field->link);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
static void
|
|
|
|
|
post_processing(void)
|
2020-06-19 22:00:22 +01:00
|
|
|
{
|
|
|
|
|
if (!wl_list_length(&rc.keybinds)) {
|
2021-07-22 21:30:17 +01:00
|
|
|
wlr_log(WLR_INFO, "load default key bindings");
|
2021-08-22 17:00:22 +01:00
|
|
|
load_default_key_bindings();
|
2020-06-19 22:00:22 +01:00
|
|
|
}
|
2022-08-29 23:42:40 +02:00
|
|
|
|
2021-09-06 22:02:02 +01:00
|
|
|
if (!wl_list_length(&rc.mousebinds)) {
|
|
|
|
|
wlr_log(WLR_INFO, "load default mouse bindings");
|
|
|
|
|
load_default_mouse_bindings();
|
|
|
|
|
}
|
2020-08-05 20:14:17 +01:00
|
|
|
|
2023-01-29 04:06:46 +01:00
|
|
|
/*
|
|
|
|
|
* Replace all earlier bindings by later ones
|
|
|
|
|
* and clear the ones with an empty action list.
|
|
|
|
|
*
|
|
|
|
|
* This is required so users are able to remove
|
|
|
|
|
* a default binding by using the "None" action.
|
|
|
|
|
*/
|
|
|
|
|
deduplicate_key_bindings();
|
|
|
|
|
deduplicate_mouse_bindings();
|
2022-08-29 23:42:40 +02:00
|
|
|
|
2022-09-15 10:53:49 -04:00
|
|
|
if (!rc.font_activewindow.name) {
|
2022-09-16 18:41:02 -04:00
|
|
|
rc.font_activewindow.name = xstrdup("sans");
|
2020-09-28 20:41:41 +01:00
|
|
|
}
|
2023-12-07 09:28:27 +01:00
|
|
|
if (!rc.font_inactivewindow.name) {
|
|
|
|
|
rc.font_inactivewindow.name = xstrdup("sans");
|
|
|
|
|
}
|
2022-09-15 10:53:49 -04:00
|
|
|
if (!rc.font_menuitem.name) {
|
2022-09-16 18:41:02 -04:00
|
|
|
rc.font_menuitem.name = xstrdup("sans");
|
2021-08-20 20:27:52 +01:00
|
|
|
}
|
2022-09-15 10:53:49 -04:00
|
|
|
if (!rc.font_osd.name) {
|
2022-09-16 18:41:02 -04:00
|
|
|
rc.font_osd.name = xstrdup("sans");
|
2021-12-06 17:44:28 -05:00
|
|
|
}
|
2022-12-22 16:44:08 -05:00
|
|
|
if (!libinput_category_get_default()) {
|
2021-10-15 10:26:00 -04:00
|
|
|
/* So we still allow tap to click by default */
|
|
|
|
|
struct libinput_category *l = libinput_category_create();
|
2024-02-10 18:02:30 +01:00
|
|
|
/* Prevents unused variable warning when compiled without asserts */
|
|
|
|
|
(void)l;
|
2022-12-22 16:44:08 -05:00
|
|
|
assert(l && libinput_category_get_default() == l);
|
2021-10-15 10:26:00 -04:00
|
|
|
}
|
2023-06-19 08:46:56 +08:00
|
|
|
|
2023-06-24 20:36:06 +08:00
|
|
|
int nr_workspaces = wl_list_length(&rc.workspace_config.workspaces);
|
|
|
|
|
if (nr_workspaces < rc.workspace_config.min_nr_workspaces) {
|
2024-03-11 17:19:03 -05:00
|
|
|
if (!rc.workspace_config.prefix) {
|
2024-03-17 15:23:41 +10:00
|
|
|
rc.workspace_config.prefix = xstrdup(_("Workspace"));
|
2024-03-11 17:19:03 -05:00
|
|
|
}
|
2023-06-19 08:46:56 +08:00
|
|
|
struct workspace *workspace;
|
2023-06-24 20:36:06 +08:00
|
|
|
for (int i = nr_workspaces; i < rc.workspace_config.min_nr_workspaces; i++) {
|
2023-06-19 08:46:56 +08:00
|
|
|
workspace = znew(*workspace);
|
2024-03-11 17:19:03 -05:00
|
|
|
workspace->name = strdup_printf("%s %d",
|
|
|
|
|
rc.workspace_config.prefix, i + 1);
|
2023-06-19 08:46:56 +08:00
|
|
|
wl_list_append(&rc.workspace_config.workspaces, &workspace->link);
|
|
|
|
|
}
|
2022-06-15 02:07:22 +02:00
|
|
|
}
|
|
|
|
|
if (rc.workspace_config.popuptime == INT_MIN) {
|
|
|
|
|
rc.workspace_config.popuptime = 1000;
|
|
|
|
|
}
|
2023-04-19 14:44:41 +01:00
|
|
|
if (!wl_list_length(&rc.window_switcher.fields)) {
|
|
|
|
|
wlr_log(WLR_INFO, "load default window switcher fields");
|
|
|
|
|
load_default_window_switcher_fields();
|
|
|
|
|
}
|
2024-05-15 23:07:23 +01:00
|
|
|
|
|
|
|
|
if (rc.mag_scale <= 0.0) {
|
|
|
|
|
rc.mag_scale = 1.0;
|
|
|
|
|
}
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
|
|
|
|
|
2023-05-09 21:20:05 +01:00
|
|
|
static void
|
|
|
|
|
rule_destroy(struct window_rule *rule)
|
|
|
|
|
{
|
|
|
|
|
wl_list_remove(&rule->link);
|
|
|
|
|
zfree(rule->identifier);
|
|
|
|
|
zfree(rule->title);
|
2024-05-16 12:52:47 +03:00
|
|
|
zfree(rule->sandbox_engine);
|
|
|
|
|
zfree(rule->sandbox_app_id);
|
2023-05-09 21:20:05 +01:00
|
|
|
action_list_free(&rule->actions);
|
|
|
|
|
zfree(rule);
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-13 16:27:46 +02:00
|
|
|
static void
|
|
|
|
|
validate_actions(void)
|
|
|
|
|
{
|
|
|
|
|
struct action *action, *action_tmp;
|
|
|
|
|
|
|
|
|
|
struct keybind *keybind;
|
|
|
|
|
wl_list_for_each(keybind, &rc.keybinds, link) {
|
|
|
|
|
wl_list_for_each_safe(action, action_tmp, &keybind->actions, link) {
|
|
|
|
|
if (!action_is_valid(action)) {
|
|
|
|
|
wl_list_remove(&action->link);
|
|
|
|
|
action_free(action);
|
|
|
|
|
wlr_log(WLR_ERROR, "Removed invalid keybind action");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct mousebind *mousebind;
|
|
|
|
|
wl_list_for_each(mousebind, &rc.mousebinds, link) {
|
|
|
|
|
wl_list_for_each_safe(action, action_tmp, &mousebind->actions, link) {
|
|
|
|
|
if (!action_is_valid(action)) {
|
|
|
|
|
wl_list_remove(&action->link);
|
|
|
|
|
action_free(action);
|
|
|
|
|
wlr_log(WLR_ERROR, "Removed invalid mousebind action");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct window_rule *rule;
|
|
|
|
|
wl_list_for_each(rule, &rc.window_rules, link) {
|
|
|
|
|
wl_list_for_each_safe(action, action_tmp, &rule->actions, link) {
|
|
|
|
|
if (!action_is_valid(action)) {
|
|
|
|
|
wl_list_remove(&action->link);
|
|
|
|
|
action_free(action);
|
|
|
|
|
wlr_log(WLR_ERROR, "Removed invalid window rule action");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-09 21:20:05 +01:00
|
|
|
static void
|
|
|
|
|
validate(void)
|
|
|
|
|
{
|
2023-05-12 13:53:50 +02:00
|
|
|
/* Regions */
|
|
|
|
|
struct region *region, *region_tmp;
|
|
|
|
|
wl_list_for_each_safe(region, region_tmp, &rc.regions, link) {
|
|
|
|
|
struct wlr_box box = region->percentage;
|
|
|
|
|
bool invalid = !region->name
|
|
|
|
|
|| box.x < 0 || box.x > 100
|
|
|
|
|
|| box.y < 0 || box.y > 100
|
|
|
|
|
|| box.width <= 0 || box.width > 100
|
|
|
|
|
|| box.height <= 0 || box.height > 100;
|
|
|
|
|
if (invalid) {
|
|
|
|
|
wlr_log(WLR_ERROR,
|
|
|
|
|
"Removing invalid region '%s': %d%% x %d%% @ %d%%,%d%%",
|
|
|
|
|
region->name, box.width, box.height, box.x, box.y);
|
|
|
|
|
wl_list_remove(®ion->link);
|
|
|
|
|
zfree(region->name);
|
|
|
|
|
free(region);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-09 21:20:05 +01:00
|
|
|
/* Window-rule criteria */
|
2023-05-13 16:27:46 +02:00
|
|
|
struct window_rule *rule, *rule_tmp;
|
|
|
|
|
wl_list_for_each_safe(rule, rule_tmp, &rc.window_rules, link) {
|
2024-05-16 12:52:47 +03:00
|
|
|
if (!rule->identifier && !rule->title && rule->window_type < 0
|
|
|
|
|
&& !rule->sandbox_engine && !rule->sandbox_app_id) {
|
2023-05-12 13:53:50 +02:00
|
|
|
wlr_log(WLR_ERROR, "Deleting rule %p as it has no criteria", rule);
|
2023-05-09 21:20:05 +01:00
|
|
|
rule_destroy(rule);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-05-13 16:27:46 +02:00
|
|
|
|
|
|
|
|
validate_actions();
|
2024-04-10 17:39:31 -05:00
|
|
|
|
|
|
|
|
/* OSD fields */
|
|
|
|
|
struct window_switcher_field *field, *field_tmp;
|
|
|
|
|
wl_list_for_each_safe(field, field_tmp, &rc.window_switcher.fields, link) {
|
|
|
|
|
if (!osd_field_validate(field)) {
|
|
|
|
|
wlr_log(WLR_ERROR, "Deleting invalid window switcher field %p", field);
|
|
|
|
|
wl_list_remove(&field->link);
|
|
|
|
|
osd_field_free(field);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-05-09 21:20:05 +01:00
|
|
|
}
|
|
|
|
|
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
void
|
|
|
|
|
rcxml_read(const char *filename)
|
2020-07-18 11:28:39 +01:00
|
|
|
{
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
rcxml_init();
|
|
|
|
|
|
|
|
|
|
struct wl_list paths;
|
2020-07-18 11:28:39 +01:00
|
|
|
|
2020-09-25 19:42:40 +01:00
|
|
|
if (filename) {
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
/* Honour command line argument -c <filename> */
|
|
|
|
|
wl_list_init(&paths);
|
|
|
|
|
struct path *path = znew(*path);
|
|
|
|
|
path->string = xstrdup(filename);
|
|
|
|
|
wl_list_append(&paths, &path->link);
|
|
|
|
|
} else {
|
|
|
|
|
paths_config_create(&paths, "rc.xml");
|
2020-09-25 19:42:40 +01:00
|
|
|
}
|
|
|
|
|
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
/* Reading file into buffer before parsing - better for unit tests */
|
|
|
|
|
bool should_merge_config = rc.merge_config;
|
|
|
|
|
struct wl_list *(*iter)(struct wl_list *list);
|
|
|
|
|
iter = should_merge_config ? paths_get_prev : paths_get_next;
|
2020-06-19 22:00:22 +01:00
|
|
|
|
2020-07-18 11:28:39 +01:00
|
|
|
/*
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
* This is the equivalent of a wl_list_for_each() which optionally
|
|
|
|
|
* iterates in reverse depending on 'should_merge_config'
|
|
|
|
|
*
|
|
|
|
|
* If not merging, we iterate forwards and break after the first
|
|
|
|
|
* iteration.
|
|
|
|
|
*
|
|
|
|
|
* If merging, we iterate backwards (least important XDG Base Dir first)
|
|
|
|
|
* and keep going.
|
2020-07-18 11:28:39 +01:00
|
|
|
*/
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
for (struct wl_list *elm = iter(&paths); elm != &paths; elm = iter(elm)) {
|
|
|
|
|
struct path *path = wl_container_of(elm, path, link);
|
|
|
|
|
FILE *stream = fopen(path->string, "r");
|
|
|
|
|
if (!stream) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2020-09-25 19:42:40 +01:00
|
|
|
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
wlr_log(WLR_INFO, "read config file %s", path->string);
|
|
|
|
|
|
2024-04-14 14:20:57 -04:00
|
|
|
struct buf b = BUF_INIT;
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
char *line = NULL;
|
|
|
|
|
size_t len = 0;
|
|
|
|
|
while (getline(&line, &len, stream) != -1) {
|
|
|
|
|
char *p = strrchr(line, '\n');
|
|
|
|
|
if (p) {
|
|
|
|
|
*p = '\0';
|
|
|
|
|
}
|
|
|
|
|
buf_add(&b, line);
|
2022-01-09 05:48:27 +01:00
|
|
|
}
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
zfree(line);
|
|
|
|
|
fclose(stream);
|
|
|
|
|
rcxml_parse_xml(&b);
|
2024-04-14 14:20:57 -04:00
|
|
|
buf_reset(&b);
|
config: support merging multiple config files
Add the -m|--merge-config command line option to iterate backwards over
XDG Base Dir paths and read config/theme files multiple times.
For example if both ~/.config/labwc/rc.xml and /etc/xdg/labwc/rc.xml
exist, the latter will be read first and then the former (if
--merge-config is enabled).
When $XDG_CONFIG_HOME is defined, make it replace (not augment)
$HOME/.config. Similarly, make $XDG_CONFIG_DIRS replace /etc/xdg when
defined.
XDG Base Dir Spec does not specify whether or not an application (or a
compositor!) should (a) define that only the file under the most important
base directory should be used, or (b) define rules for merging the
information from the different files.
ref: https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html
In the case of labwc there is a use-case for both positions, just to be
clear, the default behaviour, described by position (a) above, does NOT
change.
This change affects the following config/theme files:
- rc.xml
- menu.xml
- autostart
- environment
- themerc
- themerc-override
- Theme buttons, for example max.xbm
Instead of caching global config/theme directories, create lists of paths
(e.g. '/home/foo/.config/labwc/rc.xml', '/etc/xdg/labwc/rc.xml', etc).
This creates more common parsing logic and just reversing the direction
of iteration and breaks early if config-merge is not wanted.
Enable better fallback for themes. For example if a particular theme does
not exist in $HOME/.local/share/themes, it will be searched for in
~/.themes/ and so on. This also applies to theme buttons which now
fallback on an individual basis.
Avoid using stat() in most situations and just go straight to fopen().
Fixes #1406
2024-01-09 22:00:45 +00:00
|
|
|
if (!should_merge_config) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
paths_destroy(&paths);
|
2020-06-19 22:00:22 +01:00
|
|
|
post_processing();
|
2023-05-09 21:20:05 +01:00
|
|
|
validate();
|
2020-06-05 23:04:54 +01:00
|
|
|
}
|
|
|
|
|
|
2020-09-28 20:41:41 +01:00
|
|
|
void
|
|
|
|
|
rcxml_finish(void)
|
2020-08-13 20:18:48 +01:00
|
|
|
{
|
2022-09-15 10:53:49 -04:00
|
|
|
zfree(rc.font_activewindow.name);
|
2023-12-07 09:28:27 +01:00
|
|
|
zfree(rc.font_inactivewindow.name);
|
2022-09-15 10:53:49 -04:00
|
|
|
zfree(rc.font_menuitem.name);
|
|
|
|
|
zfree(rc.font_osd.name);
|
2021-02-21 21:59:53 +00:00
|
|
|
zfree(rc.theme_name);
|
2024-03-11 17:19:03 -05:00
|
|
|
zfree(rc.workspace_config.prefix);
|
2020-08-13 20:18:48 +01:00
|
|
|
|
2023-06-06 19:52:57 +01:00
|
|
|
struct usable_area_override *area, *area_tmp;
|
|
|
|
|
wl_list_for_each_safe(area, area_tmp, &rc.usable_area_overrides, link) {
|
|
|
|
|
wl_list_remove(&area->link);
|
|
|
|
|
zfree(area->output);
|
|
|
|
|
zfree(area);
|
|
|
|
|
}
|
|
|
|
|
|
2020-08-13 20:18:48 +01:00
|
|
|
struct keybind *k, *k_tmp;
|
2021-09-06 21:54:00 +01:00
|
|
|
wl_list_for_each_safe(k, k_tmp, &rc.keybinds, link) {
|
2020-08-13 20:18:48 +01:00
|
|
|
wl_list_remove(&k->link);
|
2022-01-23 13:46:46 +01:00
|
|
|
action_list_free(&k->actions);
|
2021-02-21 21:59:53 +00:00
|
|
|
zfree(k->keysyms);
|
|
|
|
|
zfree(k);
|
2020-08-13 20:18:48 +01:00
|
|
|
}
|
2021-08-29 14:22:49 -04:00
|
|
|
|
|
|
|
|
struct mousebind *m, *m_tmp;
|
2021-09-06 21:54:00 +01:00
|
|
|
wl_list_for_each_safe(m, m_tmp, &rc.mousebinds, link) {
|
2021-08-29 14:22:49 -04:00
|
|
|
wl_list_remove(&m->link);
|
2022-01-23 13:46:46 +01:00
|
|
|
action_list_free(&m->actions);
|
2021-08-29 14:22:49 -04:00
|
|
|
zfree(m);
|
|
|
|
|
}
|
2021-10-15 10:26:00 -04:00
|
|
|
|
2024-01-19 10:06:53 +01:00
|
|
|
struct touch_config_entry *touch_config, *touch_config_tmp;
|
|
|
|
|
wl_list_for_each_safe(touch_config, touch_config_tmp, &rc.touch_configs, link) {
|
|
|
|
|
wl_list_remove(&touch_config->link);
|
|
|
|
|
zfree(touch_config->device_name);
|
|
|
|
|
zfree(touch_config->output_name);
|
|
|
|
|
zfree(touch_config);
|
|
|
|
|
}
|
2024-01-13 00:19:44 +01:00
|
|
|
|
2024-01-07 22:19:05 +01:00
|
|
|
zfree(rc.tablet.output_name);
|
|
|
|
|
|
2021-10-15 10:26:00 -04:00
|
|
|
struct libinput_category *l, *l_tmp;
|
|
|
|
|
wl_list_for_each_safe(l, l_tmp, &rc.libinput_categories, link) {
|
|
|
|
|
wl_list_remove(&l->link);
|
|
|
|
|
zfree(l->name);
|
|
|
|
|
zfree(l);
|
|
|
|
|
}
|
2022-01-06 03:24:32 +01:00
|
|
|
|
2022-06-15 02:07:22 +02:00
|
|
|
struct workspace *w, *w_tmp;
|
|
|
|
|
wl_list_for_each_safe(w, w_tmp, &rc.workspace_config.workspaces, link) {
|
|
|
|
|
wl_list_remove(&w->link);
|
|
|
|
|
zfree(w->name);
|
|
|
|
|
zfree(w);
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-13 08:15:33 +02:00
|
|
|
regions_destroy(NULL, &rc.regions);
|
2022-07-06 17:21:02 +02:00
|
|
|
|
2023-04-19 14:44:41 +01:00
|
|
|
struct window_switcher_field *field, *field_tmp;
|
|
|
|
|
wl_list_for_each_safe(field, field_tmp, &rc.window_switcher.fields, link) {
|
|
|
|
|
wl_list_remove(&field->link);
|
2024-04-10 17:39:31 -05:00
|
|
|
osd_field_free(field);
|
2023-04-19 14:44:41 +01:00
|
|
|
}
|
|
|
|
|
|
2023-04-28 21:41:41 +01:00
|
|
|
struct window_rule *rule, *rule_tmp;
|
|
|
|
|
wl_list_for_each_safe(rule, rule_tmp, &rc.window_rules, link) {
|
2023-05-09 21:20:05 +01:00
|
|
|
rule_destroy(rule);
|
2023-04-28 21:41:41 +01:00
|
|
|
}
|
|
|
|
|
|
2022-01-06 03:24:32 +01:00
|
|
|
/* Reset state vars for starting fresh when Reload is triggered */
|
2023-06-06 19:52:57 +01:00
|
|
|
current_usable_area_override = NULL;
|
2022-01-06 03:24:32 +01:00
|
|
|
current_keybind = NULL;
|
|
|
|
|
current_mousebind = NULL;
|
2024-01-19 10:06:53 +01:00
|
|
|
current_touch = NULL;
|
2022-01-06 03:24:32 +01:00
|
|
|
current_libinput_category = NULL;
|
|
|
|
|
current_mouse_context = NULL;
|
|
|
|
|
current_keybind_action = NULL;
|
|
|
|
|
current_mousebind_action = NULL;
|
2023-08-28 19:14:04 +03:00
|
|
|
current_child_action = NULL;
|
|
|
|
|
current_view_query = NULL;
|
2022-07-06 17:21:02 +02:00
|
|
|
current_region = NULL;
|
2023-04-19 14:44:41 +01:00
|
|
|
current_field = NULL;
|
2023-04-28 21:41:41 +01:00
|
|
|
current_window_rule = NULL;
|
|
|
|
|
current_window_rule_action = NULL;
|
2020-08-13 20:18:48 +01:00
|
|
|
}
|