mirror of
https://github.com/labwc/labwc.git
synced 2025-11-04 13:30:07 -05: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
This commit is contained in:
parent
d0aff49c81
commit
698c7ace07
14 changed files with 330 additions and 240 deletions
|
|
@ -7,10 +7,12 @@
|
|||
#include <sys/stat.h>
|
||||
#include <wlr/util/log.h>
|
||||
#include "common/buf.h"
|
||||
#include "common/dir.h"
|
||||
#include "common/file-helpers.h"
|
||||
#include "common/spawn.h"
|
||||
#include "common/string-helpers.h"
|
||||
#include "config/session.h"
|
||||
#include "labwc.h"
|
||||
|
||||
static bool
|
||||
string_empty(const char *s)
|
||||
|
|
@ -45,14 +47,15 @@ error:
|
|||
free(value.buf);
|
||||
}
|
||||
|
||||
static void
|
||||
/* return true on successful read */
|
||||
static bool
|
||||
read_environment_file(const char *filename)
|
||||
{
|
||||
char *line = NULL;
|
||||
size_t len = 0;
|
||||
FILE *stream = fopen(filename, "r");
|
||||
if (!stream) {
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
wlr_log(WLR_INFO, "read environment file %s", filename);
|
||||
while (getline(&line, &len, stream) != -1) {
|
||||
|
|
@ -64,15 +67,7 @@ read_environment_file(const char *filename)
|
|||
}
|
||||
free(line);
|
||||
fclose(stream);
|
||||
}
|
||||
|
||||
static char *
|
||||
build_path(const char *dir, const char *filename)
|
||||
{
|
||||
if (string_empty(dir) || string_empty(filename)) {
|
||||
return NULL;
|
||||
}
|
||||
return strdup_printf("%s/%s", dir, filename);
|
||||
return true;
|
||||
}
|
||||
|
||||
static void
|
||||
|
|
@ -96,7 +91,7 @@ update_activation_env(const char *env_keys)
|
|||
}
|
||||
|
||||
void
|
||||
session_environment_init(const char *dir)
|
||||
session_environment_init(void)
|
||||
{
|
||||
/*
|
||||
* Set default for XDG_CURRENT_DESKTOP so xdg-desktop-portal-wlr is happy.
|
||||
|
|
@ -114,32 +109,49 @@ session_environment_init(const char *dir)
|
|||
*/
|
||||
setenv("_JAVA_AWT_WM_NONREPARENTING", "1", 0);
|
||||
|
||||
char *environment = build_path(dir, "environment");
|
||||
if (!environment) {
|
||||
return;
|
||||
struct wl_list paths;
|
||||
paths_config_create(&paths, "environment");
|
||||
|
||||
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;
|
||||
|
||||
for (struct wl_list *elm = iter(&paths); elm != &paths; elm = iter(elm)) {
|
||||
struct path *path = wl_container_of(elm, path, link);
|
||||
bool success = read_environment_file(path->string);
|
||||
if (success && !should_merge_config) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
read_environment_file(environment);
|
||||
free(environment);
|
||||
paths_destroy(&paths);
|
||||
}
|
||||
|
||||
void
|
||||
session_autostart_init(const char *dir)
|
||||
session_autostart_init(void)
|
||||
{
|
||||
/* Update dbus and systemd user environment, each may fail gracefully */
|
||||
update_activation_env("DISPLAY WAYLAND_DISPLAY XDG_CURRENT_DESKTOP");
|
||||
|
||||
char *autostart = build_path(dir, "autostart");
|
||||
if (!autostart) {
|
||||
return;
|
||||
struct wl_list paths;
|
||||
paths_config_create(&paths, "autostart");
|
||||
|
||||
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;
|
||||
|
||||
for (struct wl_list *elm = iter(&paths); elm != &paths; elm = iter(elm)) {
|
||||
struct path *path = wl_container_of(elm, path, link);
|
||||
if (!file_exists(path->string)) {
|
||||
continue;
|
||||
}
|
||||
wlr_log(WLR_INFO, "run autostart file %s", path->string);
|
||||
char *cmd = strdup_printf("sh %s", path->string);
|
||||
spawn_async_no_shell(cmd);
|
||||
free(cmd);
|
||||
|
||||
if (!should_merge_config) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!file_exists(autostart)) {
|
||||
wlr_log(WLR_ERROR, "no autostart file");
|
||||
goto out;
|
||||
}
|
||||
wlr_log(WLR_INFO, "run autostart file %s", autostart);
|
||||
char *cmd = strdup_printf("sh %s", autostart);
|
||||
spawn_async_no_shell(cmd);
|
||||
free(cmd);
|
||||
out:
|
||||
free(autostart);
|
||||
paths_destroy(&paths);
|
||||
}
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue