NEDM/parse.c

1905 lines
58 KiB
C

// Copyright 2020 - 2025, project-repo and the NEDM contributors
// SPDX-License-Identifier: MIT
#define _POSIX_C_SOURCE 200812L
#include <float.h>
#include <libinput.h>
#include <limits.h>
#include <string.h>
#include <wlr/util/log.h>
#include "input_manager.h"
#include "keybinding.h"
#include "message.h"
#include "output.h"
#include "parse.h"
#include "server.h"
#include "status_bar.h"
#include "util.h"
#include "wallpaper.h"
char *
log_error(const char *fmt, ...) {
va_list args;
va_start(args, fmt);
char *ret = malloc_vsprintf_va_list(fmt, args);
if(ret != NULL) {
wlr_log(WLR_ERROR, "%s", ret);
}
va_end(args);
return ret;
}
/* parses a key definition (e.g. "S-Tab") and sets key and modifiers in
* keybinding respectivly */
int
parse_key(struct keybinding *keybinding, const char *key_def, char **errstr) {
if(key_def == NULL) {
*errstr = log_error("Expected key definition, got nothing.");
return -1;
}
keybinding->modifiers = 0;
while(strlen(key_def) > 1 && key_def[1] == '-') {
switch(key_def[0]) {
case 'S':
keybinding->modifiers |= WLR_MODIFIER_SHIFT;
break;
case 'A':
keybinding->modifiers |= WLR_MODIFIER_ALT;
break;
case 'C':
keybinding->modifiers |= WLR_MODIFIER_CTRL;
break;
case 'L':
keybinding->modifiers |= WLR_MODIFIER_LOGO;
break;
case '2':
keybinding->modifiers |= WLR_MODIFIER_MOD2;
break;
case '3':
keybinding->modifiers |= WLR_MODIFIER_MOD3;
break;
case '5':
keybinding->modifiers |= WLR_MODIFIER_MOD5;
break;
default:
*errstr = log_error("Unknown modifier \"%c\"", key_def[0]);
return -1;
}
key_def += 2;
}
xkb_keysym_t keysym = xkb_keysym_from_name(key_def, XKB_KEYSYM_NO_FLAGS);
if(keysym == XKB_KEY_NoSymbol) {
*errstr = log_error("Could not convert key \"%s\" to keysym.", key_def);
return -1;
}
keybinding->key = keysym;
return 0;
}
int
parse_command(struct nedm_server *server, struct keybinding *keybinding,
char *saveptr, char **errstr, int nesting_level);
/* Parse a keybinding definition and return it if successful, else return NULL
*/
struct keybinding *
parse_keybinding(struct nedm_server *server, char **saveptr, char **errstr,
int nesting_level) {
struct keybinding *keybinding = malloc(sizeof(struct keybinding));
if(keybinding == NULL) {
*errstr = log_error(
"Failed to allocate memory for keybinding in parse_keybinding");
return NULL;
}
char *key = strtok_r(NULL, " ", saveptr);
if(parse_key(keybinding, key, errstr) != 0) {
wlr_log(WLR_ERROR, "Could not parse key definition \"%s\"", key);
free(keybinding);
return NULL;
}
if(parse_command(server, keybinding, *saveptr, errstr, nesting_level + 1) !=
0) {
free(keybinding);
return NULL;
}
return keybinding;
}
float
parse_float(char **saveptr, const char *delim) {
char *uint_str = strtok_r(NULL, delim, saveptr);
if(uint_str == NULL) {
wlr_log(WLR_ERROR, "Expected a float, got nothing");
return -1;
}
float ufloat = strtof(uint_str, NULL);
if(ufloat != NAN && ufloat != INFINITY && errno != ERANGE) {
return ufloat;
} else {
wlr_log(WLR_ERROR, "Error parsing float");
return FLT_MIN;
}
}
int
parse_uint(char **saveptr, const char *delim) {
char *uint_str = strtok_r(NULL, delim, saveptr);
if(uint_str == NULL) {
wlr_log(WLR_ERROR, "Expected a non-negative integer, got nothing");
return -1;
}
long uint = strtol(uint_str, NULL, 10);
if(uint >= 0 && uint <= INT_MAX) {
return uint;
} else {
wlr_log(WLR_ERROR,
"Error parsing non-negative integer. Must be a number larger "
"or equal to 0 and less or equal to %d",
INT_MAX);
return -1;
}
}
struct nedm_input_config *
parse_input_config(char **saveptr, char **errstr) {
struct nedm_input_config *cfg = input_manager_create_empty_input_config();
char *value = NULL;
char *ident = NULL;
if(cfg == NULL) {
*errstr =
log_error("Failed to allocate memory for input configuration");
goto error;
}
ident = strtok_r(NULL, " ", saveptr);
if(ident == NULL) {
*errstr = log_error(
"Expected identifier of input device to configure, got none");
goto error;
}
cfg->identifier = strdup(ident);
char *setting = strtok_r(NULL, " ", saveptr);
if(setting == NULL) {
*errstr =
log_error("Expected setting to be set on input device, got none");
goto error;
}
value = strdup(*saveptr);
if(value == NULL) {
*errstr = log_error("Failed to obtain value for input device "
"configuration of device \"%s\"",
ident);
goto error;
}
if(strcmp(setting, "accel_profile") == 0) {
if(strcmp(value, "adaptive") == 0) {
cfg->accel_profile = LIBINPUT_CONFIG_ACCEL_PROFILE_ADAPTIVE;
} else if(strcmp(value, "flat") == 0) {
cfg->accel_profile = LIBINPUT_CONFIG_ACCEL_PROFILE_FLAT;
} else {
*errstr = log_error(
"Invalid profile \"%s\" for accel_profile configuration",
value);
goto error;
}
} else if(strcmp(setting, "calibration_matrix") == 0) {
cfg->calibration_matrix.configured = true;
for(int i = 0; i < 6; ++i) {
cfg->calibration_matrix.matrix[i] = parse_float(saveptr, " ");
if(cfg->calibration_matrix.matrix[i] == FLT_MIN) {
*errstr =
log_error("Failed to read calibration matrix, expected 6 "
"floating point values separated by spaces");
goto error;
}
}
} else if(strcmp(setting, "click_method") == 0) {
if(strcmp(value, "none")) {
cfg->click_method = LIBINPUT_CONFIG_CLICK_METHOD_NONE;
} else if(strcmp(value, "button_areas")) {
cfg->click_method = LIBINPUT_CONFIG_CLICK_METHOD_BUTTON_AREAS;
} else if(strcmp(value, "clickfinger")) {
cfg->click_method = LIBINPUT_CONFIG_CLICK_METHOD_CLICKFINGER;
} else {
*errstr = log_error(
"Invalid method \"%s\" for click_method configuration", value);
goto error;
}
} else if(strcmp(setting, "drag") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->drag = LIBINPUT_CONFIG_DRAG_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->drag = LIBINPUT_CONFIG_DRAG_DISABLED;
} else {
*errstr =
log_error("Invalid option \"%s\" to setting \"drag\"", value);
goto error;
}
} else if(strcmp(setting, "drag_lock") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->drag_lock = LIBINPUT_CONFIG_DRAG_LOCK_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->drag_lock = LIBINPUT_CONFIG_DRAG_LOCK_DISABLED;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"drag_lock\"", value);
goto error;
}
} else if(strcmp(setting, "dwt") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->dwt = LIBINPUT_CONFIG_DWT_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->dwt = LIBINPUT_CONFIG_DWT_DISABLED;
} else {
*errstr =
log_error("Invalid option \"%s\" to setting \"dwt\"", value);
goto error;
}
} else if(strcmp(setting, "events") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->send_events = LIBINPUT_CONFIG_SEND_EVENTS_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->send_events = LIBINPUT_CONFIG_SEND_EVENTS_DISABLED;
} else if(strcmp(value, "disabled_on_external_mouse") == 0) {
cfg->send_events =
LIBINPUT_CONFIG_SEND_EVENTS_DISABLED_ON_EXTERNAL_MOUSE;
} else {
*errstr =
log_error("Invalid option \"%s\" to setting \"events\"", value);
goto error;
}
} else if(strcmp(setting, "left_handed") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->left_handed = true;
} else if(strcmp(value, "disabled") == 0) {
cfg->left_handed = false;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"left_handed\"", value);
goto error;
}
} else if(strcmp(setting, "middle_emulation") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->middle_emulation = LIBINPUT_CONFIG_MIDDLE_EMULATION_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->middle_emulation = LIBINPUT_CONFIG_MIDDLE_EMULATION_DISABLED;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"middle_emulation\"", value);
goto error;
}
} else if(strcmp(setting, "natural_scroll") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->natural_scroll = true;
} else if(strcmp(value, "disabled") == 0) {
cfg->natural_scroll = false;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"natural_scroll\"", value);
goto error;
}
} else if(strcmp(setting, "pointer_accel") == 0) {
cfg->pointer_accel = parse_float(saveptr, " ");
if(cfg->pointer_accel == FLT_MIN) {
*errstr = log_error("Invalid option \"%s\" to setting "
"\"pointer_accel\", expected float value",
value);
goto error;
}
} else if(strcmp(setting, "scroll_button") == 0) {
char *err = NULL;
cfg->scroll_button = input_manager_get_mouse_button(value, &err);
if(err) {
*errstr = log_error("Error parsing button for \"scroll_button\" "
"setting. Returned error \"%s\"",
err);
free(err);
goto error;
}
} else if(strcmp(setting, "scroll_factor") == 0) {
cfg->scroll_factor = parse_float(saveptr, " ");
if(cfg->scroll_factor == FLT_MIN) {
*errstr = log_error("Invalid option \"%s\" to setting "
"\"scroll_factor\", expected float value",
value);
goto error;
}
} else if(strcmp(setting, "scroll_method") == 0) {
if(strcmp(value, "none") == 0) {
cfg->scroll_method = LIBINPUT_CONFIG_SCROLL_NO_SCROLL;
} else if(strcmp(value, "two_finger") == 0) {
cfg->scroll_method = LIBINPUT_CONFIG_SCROLL_2FG;
} else if(strcmp(value, "edge") == 0) {
cfg->scroll_method = LIBINPUT_CONFIG_SCROLL_EDGE;
} else if(strcmp(value, "on_button_down") == 0) {
cfg->scroll_method = LIBINPUT_CONFIG_SCROLL_ON_BUTTON_DOWN;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"scroll_method\"", value);
goto error;
}
} else if(strcmp(setting, "tap") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->tap = LIBINPUT_CONFIG_TAP_ENABLED;
} else if(strcmp(value, "disabled") == 0) {
cfg->tap = LIBINPUT_CONFIG_TAP_DISABLED;
} else {
*errstr =
log_error("Invalid option \"%s\" to setting \"tap\"", value);
goto error;
}
} else if(strcmp(setting, "tap_button_map") == 0) {
if(strcmp(value, "lrm") == 0) {
cfg->tap = LIBINPUT_CONFIG_TAP_MAP_LRM;
} else if(strcmp(value, "lmr") == 0) {
cfg->tap = LIBINPUT_CONFIG_TAP_MAP_LMR;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"tap_button_map\"", value);
goto error;
}
} else if(strcmp(setting, "keybindings") == 0) {
if(strcmp(value, "enabled") == 0) {
cfg->enable_keybindings = true;
} else if(strcmp(value, "disabled") == 0) {
cfg->enable_keybindings = false;
} else {
*errstr = log_error(
"Invalid option \"%s\" to setting \"keybindings\"", value);
goto error;
}
} else if(strcmp(setting, "repeat_delay") == 0) {
cfg->repeat_delay = parse_uint(saveptr, " ");
if(cfg->repeat_delay < 0) {
*errstr = log_error("Invalid option \"%s\" to setting "
"\"repeat_delay\", expected positive integer",
value);
goto error;
}
} else if(strcmp(setting, "repeat_rate") == 0) {
cfg->repeat_rate = parse_uint(saveptr, " ");
if(cfg->repeat_rate < 0) {
*errstr = log_error("Invalid option \"%s\" to setting "
"\"repeat_rate\", expected positive integer",
value);
goto error;
}
} else {
*errstr = log_error("Invalid option to command \"input\"");
goto error;
}
free(value);
return cfg;
error:
if(cfg) {
if(cfg->identifier) {
free(cfg->identifier);
}
free(cfg);
}
if(value) {
free(value);
}
wlr_log(WLR_ERROR, "Input configuration must be of the form 'input "
"\"<ident>\" <setting> <value>'");
return NULL;
}
struct keybinding *
parse_bind(struct nedm_server *server, char **saveptr, char **errstr,
int nesting_level) {
struct keybinding *keybinding =
parse_keybinding(server, saveptr, errstr, nesting_level);
if(keybinding == NULL) {
wlr_log(WLR_ERROR, "Could not parse keybinding for \"bind\".");
return NULL;
}
keybinding->mode = 1;
return keybinding;
}
struct keybinding *
parse_definekey(struct nedm_server *server, char **saveptr, char **errstr,
int nesting_level) {
char *mode = strtok_r(NULL, " ", saveptr);
if(mode == NULL) {
*errstr =
log_error("Too few arguments to \"definekey\". Expected mode");
return NULL;
}
int mode_idx = get_mode_index_from_name(server->modes, mode);
if(mode_idx == -1) {
*errstr = log_error("Unknown mode \"%s\"", mode);
return NULL;
}
struct keybinding *keybinding =
parse_keybinding(server, saveptr, errstr, nesting_level);
if(keybinding == NULL) {
wlr_log(WLR_ERROR, "Could not parse keybinding for \"definekey\"");
return NULL;
}
keybinding->mode = mode_idx;
return keybinding;
}
int
parse_background(float *color, char **saveptr, char **errstr) {
/* Read rgb numbers */
for(unsigned int i = 0; i < 3; ++i) {
char *nstr = strtok_r(NULL, " \n", saveptr);
int nstrlen;
if(nstr == NULL || (nstrlen = strlen(nstr)) == 0) {
*errstr = log_error("Expected three space-separated numbers (rgb) "
"for background color setting. Got %d.",
i);
return -1;
}
if(nstr[nstrlen - 1] == '\n') {
nstr[nstrlen - 1] = '\0';
--nstrlen;
}
char *endptr = NULL;
float nval = strtof(nstr, &endptr);
if(endptr != nstr + nstrlen) {
*errstr = log_error(
"Could not parse number \"%s\" for background color setting.",
nstr);
return -1;
}
if(nval < 0 || nval > 1) {
*errstr = log_error("Expected a number between 0 and 1 for setting "
"of background color. Got %f.",
nval);
return -1;
}
color[i] = nval;
}
return 0;
}
struct keybinding *
parse_escape(char **saveptr, char **errstr) {
struct keybinding *keybinding = malloc(sizeof(struct keybinding));
if(keybinding == NULL) {
*errstr = log_error(
"Failed to allocate memory for keybinding in parse_escape");
return NULL;
}
char *key = strtok_r(NULL, " ", saveptr);
if(parse_key(keybinding, key, errstr) != 0) {
wlr_log(WLR_ERROR,
"Could not parse key definition \"%s\" for \"escape\"", key);
free(keybinding);
return NULL;
}
keybinding->mode = 0; //"top" mode
keybinding->action = KEYBINDING_SWITCH_MODE;
keybinding->data.u = 1; //"root" mode
return keybinding;
}
int
parse_cursor(char **saveptr, char **errstr) {
if(strcmp(*saveptr, "enable") == 0) {
return 1;
} else if(strcmp(*saveptr, "disable") == 0) {
return 0;
} else {
*errstr = log_error(
"Invalid option \"%s\" for \"cursor\". Expected \"enable\" or "
"\"disable\".",
*saveptr);
return -1;
}
}
char *
parse_definemode(char **saveptr, char **errstr) {
char *mode = strtok_r(NULL, " ", saveptr);
if(mode == NULL) {
*errstr = log_error("Expected mode to succeed \"definemode\" keyword.");
return NULL;
}
return strdup(mode);
}
int
parse_workspaces(char **saveptr, char **errstr) {
char *nws_str = strtok_r(NULL, " ", saveptr);
if(nws_str == NULL) {
*errstr = log_error(
"Expected argument for \"workspaces\" command, got none.");
return -1;
}
long nws = strtol(nws_str, NULL, 10);
if(!(1 <= nws && nws <= 30)) {
*errstr = log_error(
"More than 30 workspaces are not supported. Received %li", nws);
return -1;
}
return nws;
}
int
parse_output_config_keyword(char *key_str, enum output_status *status) {
if(key_str == NULL) {
return -1;
}
if(strcmp(key_str, "pos") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "prio") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "rotate") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "scale") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "enable") == 0) {
*status = OUTPUT_ENABLE;
} else if(strcmp(key_str, "permanent") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "peripheral") == 0) {
*status = OUTPUT_DEFAULT;
} else if(strcmp(key_str, "disable") == 0) {
*status = OUTPUT_DISABLE;
} else {
return -1;
}
return 0;
}
struct nedm_output_config *
parse_output_config(char **saveptr, char **errstr) {
struct nedm_output_config *cfg = malloc(sizeof(struct nedm_output_config));
if(cfg == NULL) {
*errstr =
log_error("Failed to allocate memory for output configuration");
goto error;
}
cfg->status = OUTPUT_DEFAULT;
cfg->pos.x = -1;
cfg->output_name = NULL;
cfg->refresh_rate = 0;
cfg->priority = -1;
cfg->scale = -1;
cfg->angle = -1;
cfg->role = OUTPUT_ROLE_DEFAULT;
char *name = strtok_r(NULL, " ", saveptr);
if(name == NULL) {
*errstr =
log_error("Expected name of output to be configured, got none");
goto error;
}
char *key_str = strtok_r(NULL, " ", saveptr);
if(parse_output_config_keyword(key_str, &(cfg->status)) != 0) {
*errstr = log_error("Expected keyword \"pos\", \"prio\", \"enable\", "
"\"disable\", \"permanent\" or \"peripheral\" in "
"output configuration for output %s",
name);
goto error;
}
if(cfg->status == OUTPUT_ENABLE || cfg->status == OUTPUT_DISABLE) {
cfg->output_name = strdup(name);
return cfg;
}
if(strcmp(key_str, "rotate") == 0) {
uint32_t angle = parse_uint(saveptr, " ");
// 360 degrees is equivalent to 0 degrees
cfg->angle = angle % 4;
if(cfg->angle < 0) {
*errstr =
log_error("Error parsing option rotate for output %s", name);
goto error;
}
cfg->output_name = strdup(name);
return cfg;
}
if(strcmp(key_str, "prio") == 0) {
cfg->priority = parse_uint(saveptr, " ");
if(cfg->priority < 0) {
*errstr = log_error(
"Error parsing priority of output configuration for output %s",
name);
goto error;
}
cfg->output_name = strdup(name);
return cfg;
}
if(strcmp(key_str, "scale") == 0) {
cfg->scale = parse_float(saveptr, " ");
if(cfg->scale <= 0.0) {
*errstr =
log_error("Error parsing scale of output configuration for "
"output %s, expected positive float",
name);
goto error;
}
cfg->output_name = strdup(name);
return cfg;
}
if(strcmp(key_str, "peripheral") == 0) {
cfg->output_name = strdup(name);
cfg->role = OUTPUT_ROLE_PERIPHERAL;
return cfg;
}
if(strcmp(key_str, "permanent") == 0) {
cfg->output_name = strdup(name);
cfg->role = OUTPUT_ROLE_PERMANENT;
return cfg;
}
cfg->pos.x = parse_uint(saveptr, " ");
if(cfg->pos.x < 0) {
*errstr = log_error(
"Error parsing x coordinate of output configuration for output %s",
name);
goto error;
}
cfg->pos.y = parse_uint(saveptr, " ");
if(cfg->pos.y < 0) {
*errstr = log_error(
"Error parsing y coordinate of output configuration for output %s",
name);
goto error;
}
char *res_str = strtok_r(NULL, " ", saveptr);
if(res_str == NULL || strcmp(res_str, "res") != 0) {
*errstr = log_error(
"Expected keyword \"res\" in output configuration for output %s",
name);
goto error;
}
cfg->pos.width = parse_uint(saveptr, "x");
if(cfg->pos.width <= 0) {
*errstr = log_error("Error parsing width of output configuration for "
"output %s (hint: width must be larger than 0)",
name);
goto error;
}
cfg->pos.height = parse_uint(saveptr, " ");
if(cfg->pos.height <= 0) {
*errstr = log_error("Error parsing height of output configuration for "
"output %s (hint: height must e larger than 0)",
name);
goto error;
}
char *rate_str = strtok_r(NULL, " ", saveptr);
if(rate_str == NULL || strcmp(rate_str, "rate") != 0) {
*errstr = log_error(
"Expected keyword \"rate\" in output configuration for output %s",
name);
goto error;
}
cfg->refresh_rate = parse_float(saveptr, " ");
if(cfg->refresh_rate <= 0.0) {
*errstr =
log_error("Error parsing refresh rate of output configuration for "
"output %s, expected positive float",
name);
goto error;
}
cfg->output_name = strdup(name);
return cfg;
error:
free(cfg);
wlr_log(WLR_ERROR,
"Output configuration must be of the form \"output <name> pos <x> "
"<y> res <width>x<height> rate <refresh_rate>");
return NULL;
}
struct nedm_message_config *
parse_message_config(char **saveptr, char **errstr) {
struct nedm_message_config *cfg = calloc(1, sizeof(struct nedm_message_config));
if(cfg == NULL) {
*errstr =
log_error("Failed to allocate memory for message configuration");
goto error;
}
cfg->bg_color[0] = -1;
cfg->fg_color[0] = -1;
cfg->display_time = -1;
cfg->font = NULL;
cfg->anchor = NEDM_MESSAGE_NOPT;
cfg->enabled = -1;
char *setting = strtok_r(NULL, " ", saveptr);
if(setting == NULL) {
*errstr = log_error(
"Expected setting to be set for message configuration, got none");
goto error;
}
if(strcmp(setting, "font") == 0) {
cfg->font = strdup(*saveptr);
if(cfg->font == NULL) {
*errstr = log_error("Unable to allocate memory for font descrition "
"in command \"message\"");
goto error;
}
} else if(strcmp(setting, "display_time") == 0) {
cfg->display_time = parse_uint(saveptr, " ");
if(cfg->display_time < 0) {
*errstr =
log_error("Error parsing command \"configure_message "
"display_time\", expected a non-negative integer");
goto error;
}
} else if(strcmp(setting, "bg_color") == 0) {
for(int i = 0; i < 4; ++i) {
cfg->bg_color[i] = parse_float(saveptr, " ");
if(cfg->bg_color[i] == FLT_MIN) {
*errstr = log_error(
"Error parsing command \"configure_message bg_color\", "
"expected 4 float values separated by spaces");
goto error;
}
}
} else if(strcmp(setting, "fg_color") == 0) {
for(int i = 0; i < 4; ++i) {
cfg->fg_color[i] = parse_float(saveptr, " ");
if(cfg->fg_color[i] == FLT_MIN) {
*errstr = log_error(
"Error parsing command \"configure_message bg_color\", "
"expected 4 float values separated by spaces");
goto error;
}
}
} else if(strcmp(setting, "anchor") == 0) {
char *anchors[] = {"top_left", "top_center", "top_right",
"bottom_left", "bottom_center", "bottom_right",
"center"};
for(int i = 0; i < 7; ++i) {
if(strcmp(*saveptr, anchors[i]) == 0) {
cfg->anchor = i;
break;
}
}
if(cfg->anchor == NEDM_MESSAGE_NOPT) {
*errstr =
log_error("Error parsing command \"configure_message anchor\", "
"the given anchor value is not a valid option");
goto error;
}
} else if(strcmp(setting, "enable") == 0) {
cfg->enabled = 1;
} else if(strcmp(setting, "disable") == 0) {
cfg->enabled = 0;
} else {
*errstr = log_error("Invalid option to command \"configure_message\"");
goto error;
}
return cfg;
error:
if(cfg != NULL) {
free(cfg);
}
wlr_log(WLR_ERROR, "Message configuration must be of the form "
"'configure_message <setting> <value>'");
return NULL;
}
static struct nedm_status_bar_config *
parse_status_bar_config(char **saveptr, char **errstr) {
struct nedm_status_bar_config *cfg = calloc(1, sizeof(struct nedm_status_bar_config));
if(cfg == NULL) {
*errstr = log_error("Failed to allocate memory for status bar configuration");
goto error;
}
// Set defaults
cfg->position = NEDM_STATUS_BAR_TOP_RIGHT;
cfg->height = 24;
cfg->width_percent = 20;
cfg->update_interval = 1000;
cfg->bg_color[0] = 0.1f; cfg->bg_color[1] = 0.1f; cfg->bg_color[2] = 0.1f; cfg->bg_color[3] = 0.9f;
cfg->text_color[0] = 1.0f; cfg->text_color[1] = 1.0f; cfg->text_color[2] = 1.0f; cfg->text_color[3] = 1.0f;
cfg->font = strdup("monospace 10");
cfg->show_time = true;
cfg->show_date = true;
cfg->show_battery = true;
cfg->show_volume = true;
cfg->show_wifi = true;
cfg->show_workspace = true;
char *setting = strtok_r(NULL, " ", saveptr);
if(setting == NULL) {
*errstr = log_error("Expected setting to be set for status bar configuration, got none");
goto error;
}
if(strcmp(setting, "position") == 0) {
char *pos_str = strtok_r(NULL, " ", saveptr);
if(pos_str == NULL) {
*errstr = log_error("Expected position for status bar configuration, got none");
goto error;
}
if(strcmp(pos_str, "top_left") == 0) {
cfg->position = NEDM_STATUS_BAR_TOP_LEFT;
} else if(strcmp(pos_str, "top_right") == 0) {
cfg->position = NEDM_STATUS_BAR_TOP_RIGHT;
} else if(strcmp(pos_str, "bottom_left") == 0) {
cfg->position = NEDM_STATUS_BAR_BOTTOM_LEFT;
} else if(strcmp(pos_str, "bottom_right") == 0) {
cfg->position = NEDM_STATUS_BAR_BOTTOM_RIGHT;
} else {
*errstr = log_error("Invalid position \"%s\" for status bar", pos_str);
goto error;
}
} else if(strcmp(setting, "height") == 0) {
cfg->height = parse_uint(saveptr, " ");
if(cfg->height == 0) {
*errstr = log_error("Invalid height for status bar");
goto error;
}
} else if(strcmp(setting, "width_percent") == 0) {
cfg->width_percent = parse_uint(saveptr, " ");
if(cfg->width_percent == 0 || cfg->width_percent > 100) {
*errstr = log_error("Invalid width_percent for status bar (must be 1-100)");
goto error;
}
} else if(strcmp(setting, "update_interval") == 0) {
cfg->update_interval = parse_uint(saveptr, " ");
if(cfg->update_interval == 0) {
*errstr = log_error("Invalid update_interval for status bar");
goto error;
}
} else if(strcmp(setting, "font") == 0) {
free(cfg->font);
cfg->font = strdup(*saveptr);
if(cfg->font == NULL) {
*errstr = log_error("Unable to allocate memory for font in status bar config");
goto error;
}
} else if(strcmp(setting, "bg_color") == 0) {
if(parse_background(cfg->bg_color, saveptr, errstr) != 0) {
goto error;
}
} else if(strcmp(setting, "text_color") == 0) {
if(parse_background(cfg->text_color, saveptr, errstr) != 0) {
goto error;
}
} else {
*errstr = log_error("Unknown status bar setting: \"%s\"", setting);
goto error;
}
return cfg;
error:
if(cfg) {
free(cfg->font);
free(cfg);
}
return NULL;
}
static struct nedm_wallpaper_config *
parse_wallpaper_config(char **saveptr, char **errstr) {
struct nedm_wallpaper_config *cfg = calloc(1, sizeof(struct nedm_wallpaper_config));
if(cfg == NULL) {
*errstr = log_error("Failed to allocate memory for wallpaper configuration");
goto error;
}
// Set defaults
cfg->image_path = strdup("assets/nedm.png");
cfg->mode = NEDM_WALLPAPER_FILL;
cfg->bg_color[0] = 0.2f; cfg->bg_color[1] = 0.2f; cfg->bg_color[2] = 0.3f; cfg->bg_color[3] = 1.0f;
char *setting = strtok_r(NULL, " ", saveptr);
if(setting == NULL) {
*errstr = log_error("Expected setting to be set for wallpaper configuration, got none");
goto error;
}
if(strcmp(setting, "image_path") == 0) {
free(cfg->image_path);
cfg->image_path = strdup(*saveptr);
if(cfg->image_path == NULL) {
*errstr = log_error("Unable to allocate memory for image path in wallpaper config");
goto error;
}
} else if(strcmp(setting, "mode") == 0) {
char *mode_str = strtok_r(NULL, " ", saveptr);
if(mode_str == NULL) {
*errstr = log_error("Expected mode for wallpaper configuration, got none");
goto error;
}
if(strcmp(mode_str, "fill") == 0) {
cfg->mode = NEDM_WALLPAPER_FILL;
} else if(strcmp(mode_str, "fit") == 0) {
cfg->mode = NEDM_WALLPAPER_FIT;
} else if(strcmp(mode_str, "stretch") == 0) {
cfg->mode = NEDM_WALLPAPER_STRETCH;
} else if(strcmp(mode_str, "center") == 0) {
cfg->mode = NEDM_WALLPAPER_CENTER;
} else if(strcmp(mode_str, "tile") == 0) {
cfg->mode = NEDM_WALLPAPER_TILE;
} else {
*errstr = log_error("Invalid mode \"%s\" for wallpaper", mode_str);
goto error;
}
} else if(strcmp(setting, "bg_color") == 0) {
if(parse_background(cfg->bg_color, saveptr, errstr) != 0) {
goto error;
}
} else {
*errstr = log_error("Unknown wallpaper setting: \"%s\"", setting);
goto error;
}
return cfg;
error:
if(cfg) {
free(cfg->image_path);
free(cfg);
}
return NULL;
}
int
parse_command(struct nedm_server *server, struct keybinding *keybinding,
char *saveptr, char **errstr, int nesting_level) {
char *action = strtok_r(NULL, " ", &saveptr);
if(nesting_level >= MAX_NESTING_LEVEL) {
*errstr =
log_error("Nesting level of commands is too deep. Giving up.");
return -1;
}
if(action == NULL) {
*errstr = log_error("Expexted an action to parse, got none.");
return -1;
}
keybinding->data = (union keybinding_params){.c = NULL};
if(strcmp(action, "vsplit") == 0) {
keybinding->action = KEYBINDING_SPLIT_VERTICAL;
char *percentage_string = strtok_r(NULL, " ", &saveptr);
if(percentage_string == NULL) {
keybinding->data.f = 0.5;
} else {
float percentage = strtof(percentage_string, NULL);
if(percentage == NAN || percentage == INFINITY || errno == ERANGE) {
wlr_log(WLR_ERROR, "Error parsing float.");
return -1;
}
if(percentage >= 1.0 || percentage <= 0.0) {
wlr_log(WLR_ERROR, "Expected a float between 0 and 1, got %f.",
percentage);
return -1;
}
keybinding->data.f = percentage;
}
} else if(strcmp(action, "hsplit") == 0) {
keybinding->action = KEYBINDING_SPLIT_HORIZONTAL;
char *percentage_string = strtok_r(NULL, " ", &saveptr);
if(percentage_string == NULL) {
keybinding->data.f = 0.5;
} else {
float percentage = strtof(percentage_string, NULL);
if(percentage == NAN || percentage == INFINITY || errno == ERANGE) {
wlr_log(WLR_ERROR, "Error parsing float.");
return -1;
}
if(percentage >= 1.0 || percentage <= 0.0) {
wlr_log(WLR_ERROR, "Expected a float between 0 and 1, got %f.",
percentage);
return -1;
}
keybinding->data.f = percentage;
}
} else if(strcmp(action, "quit") == 0) {
keybinding->action = KEYBINDING_QUIT;
} else if(strcmp(action, "dump") == 0) {
keybinding->action = KEYBINDING_DUMP;
} else if(strcmp(action, "show_info") == 0) {
keybinding->action = KEYBINDING_SHOW_INFO;
} else if(strcmp(action, "close") == 0) {
keybinding->action = KEYBINDING_CLOSE_VIEW;
} else if(strcmp(action, "focus") == 0) {
keybinding->action = KEYBINDING_CYCLE_TILES;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 0;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("Tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.us[1] = tile_id;
}
} else if(strcmp(action, "focusprev") == 0) {
keybinding->action = KEYBINDING_CYCLE_TILES;
keybinding->data.us[0] = 1;
keybinding->data.us[1] = 0;
} else if(strcmp(action, "next") == 0) {
keybinding->action = KEYBINDING_CYCLE_VIEWS;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 0;
char *view_str = strtok_r(NULL, " ", &saveptr);
if(view_str != NULL) {
long view_id = strtol(view_str, NULL, 10);
if(view_id < 1) {
*errstr = log_error("View id must be an integer number "
"larger or equal to 1. Got %ld",
view_id);
return -1;
}
keybinding->data.us[1] = view_id;
}
} else if(strcmp(action, "prev") == 0) {
keybinding->action = KEYBINDING_CYCLE_VIEWS;
keybinding->data.us[0] = 1;
keybinding->data.us[1] = 0;
} else if(strcmp(action, "only") == 0) {
keybinding->action = KEYBINDING_LAYOUT_FULLSCREEN;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 0;
char *screen_str = strtok_r(NULL, " ", &saveptr);
if(screen_str != NULL) {
long screen = strtol(screen_str, NULL, 10);
if(screen < 1) {
*errstr = log_error("Screen must be an integer number "
"larger or equal to 1. Got %ld",
screen);
return -1;
}
keybinding->data.us[0] = screen;
char *workspace_str = strtok_r(NULL, " ", &saveptr);
if(workspace_str == NULL) {
*errstr = log_error(
"\"only\" requires either none or two arguments, got one");
return -1;
}
long workspace = strtol(workspace_str, NULL, 10);
if(workspace < 1) {
*errstr = log_error("Workspace must be an integer number "
"larger or equal to 1. Got %ld",
workspace);
return -1;
}
keybinding->data.us[1] = workspace - 1;
}
} else if(strcmp(action, "abort") == 0) {
keybinding->action = KEYBINDING_NOOP;
} else if(strcmp(action, "message") == 0) {
keybinding->action = KEYBINDING_DISPLAY_MESSAGE;
if(saveptr == NULL) {
*errstr =
log_error("Not enough paramaters to \"message\". Expected "
"string to display.");
return -1;
}
keybinding->data.c = strdup(saveptr);
} else if(strcmp(action, "custom_event") == 0) {
keybinding->action = KEYBINDING_SEND_CUSTOM_EVENT;
if(saveptr == NULL) {
*errstr =
log_error("Not enough paramaters to \"custom_event\". Expected "
"string to send.");
return -1;
}
keybinding->data.c = strdup(saveptr);
} else if(strcmp(action, "time") == 0) {
keybinding->action = KEYBINDING_SHOW_TIME;
} else if(strcmp(action, "nextscreen") == 0) {
keybinding->action = KEYBINDING_CYCLE_OUTPUT;
keybinding->data.b = false;
} else if(strcmp(action, "prevscreen") == 0) {
keybinding->action = KEYBINDING_CYCLE_OUTPUT;
keybinding->data.b = true;
} else if(strcmp(action, "exec") == 0) {
keybinding->action = KEYBINDING_RUN_COMMAND;
if(saveptr == NULL) {
*errstr = log_error("Not enough paramaters to \"exec\". Expected "
"string to execute.");
return -1;
}
keybinding->data.c = strdup(saveptr);
} else if(strcmp(action, "resizeleft") == 0) {
keybinding->action = KEYBINDING_RESIZE_TILE_HORIZONTAL;
keybinding->data.is[0] = -10;
keybinding->data.is[1] = 0;
char *str = strtok_r(NULL, " ", &saveptr);
if(str != NULL) {
long n_pixels = strtol(str, NULL, 10);
if(n_pixels < 1) {
*errstr =
log_error("Number of pixels must be an integer number "
"larger or equal to 1. Got %ld",
n_pixels);
return -1;
}
keybinding->data.is[0] = -n_pixels;
}
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.is[1] = tile_id;
}
} else if(strcmp(action, "resizeright") == 0) {
keybinding->action = KEYBINDING_RESIZE_TILE_HORIZONTAL;
keybinding->data.is[0] = 10;
keybinding->data.is[1] = 0;
char *str = strtok_r(NULL, " ", &saveptr);
if(str != NULL) {
long n_pixels = strtol(str, NULL, 10);
if(n_pixels < 1) {
*errstr =
log_error("Number of pixels must be an integer number "
"larger or equal to 1. Got %ld",
n_pixels);
return -1;
}
keybinding->data.is[0] = n_pixels;
}
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.is[1] = tile_id;
}
} else if(strcmp(action, "resizedown") == 0) {
keybinding->action = KEYBINDING_RESIZE_TILE_VERTICAL;
keybinding->data.is[0] = 10;
keybinding->data.is[1] = 0;
char *str = strtok_r(NULL, " ", &saveptr);
if(str != NULL) {
long n_pixels = strtol(str, NULL, 10);
if(n_pixels < 1) {
*errstr =
log_error("Number of pixels must be an integer number "
"larger or equal to 1. Got %ld",
n_pixels);
return -1;
}
keybinding->data.is[0] = n_pixels;
}
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.is[1] = tile_id;
}
} else if(strcmp(action, "resizeup") == 0) {
keybinding->action = KEYBINDING_RESIZE_TILE_VERTICAL;
keybinding->data.is[0] = -10;
keybinding->data.is[1] = 0;
char *str = strtok_r(NULL, " ", &saveptr);
if(str != NULL) {
long n_pixels = strtol(str, NULL, 10);
if(n_pixels < 1) {
*errstr =
log_error("Number of pixels must be an integer number "
"larger or equal to 1. Got %ld",
n_pixels);
return -1;
}
keybinding->data.is[0] = -n_pixels;
}
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.is[1] = tile_id;
}
} else if(strcmp(action, "screen") == 0) {
keybinding->action = KEYBINDING_SWITCH_OUTPUT;
char *noutp_str = strtok_r(NULL, " ", &saveptr);
if(noutp_str == NULL) {
*errstr =
log_error("Expected argument for \"output\" action, got none.");
return -1;
}
long outp = strtol(noutp_str, NULL, 10);
if(outp < 1) {
*errstr = log_error("Workspace number must be an integer number "
"larger or equal to 1. Got %ld",
outp);
return -1;
}
keybinding->data.u = outp;
} else if(strcmp(action, "workspace") == 0) {
keybinding->action = KEYBINDING_SWITCH_WORKSPACE;
char *nws_str = strtok_r(NULL, " ", &saveptr);
if(nws_str == NULL) {
*errstr = log_error(
"Expected argument for \"workspace\" action, got none.");
return -1;
}
long ws = strtol(nws_str, NULL, 10);
if(ws < 1) {
*errstr = log_error("Workspace number must be an integer number "
"larger or equal to 1. Got %ld",
ws);
return -1;
}
keybinding->data.u = ws - 1;
} else if(strcmp(action, "moveviewtoscreen") == 0) {
keybinding->action = KEYBINDING_MOVE_VIEW_TO_OUTPUT;
char *view_id_str = strtok_r(NULL, " ", &saveptr);
if(view_id_str == NULL) {
*errstr = log_error(
"Expected argument for \"moveviewtoscreen\" action, got none.");
return -1;
}
long view_id = strtol(view_id_str, NULL, 10);
if(view_id < 1) {
*errstr = log_error("View id must be an integer larger or "
"equal to 1. Got %ld",
view_id);
return -1;
}
char *noutp_str = strtok_r(NULL, " ", &saveptr);
if(noutp_str == NULL) {
*errstr = log_error("Expected two arguments for "
"\"moveviewtoscreen\" action, got one.");
return -1;
}
long outp = strtol(noutp_str, NULL, 10);
if(outp < 1) {
*errstr = log_error("Output number must be an integer larger or "
"equal to 1. Got %ld",
outp);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = view_id;
keybinding->data.us[1] = outp;
keybinding->data.us[2] = follow;
} else if(strcmp(action, "moveviewtoworkspace") == 0) {
keybinding->action = KEYBINDING_MOVE_VIEW_TO_WORKSPACE;
char *view_id_str = strtok_r(NULL, " ", &saveptr);
if(view_id_str == NULL) {
*errstr = log_error("Expected argument for \"moveviewtoworkspace\" "
"action, got none.");
return -1;
}
long view_id = strtol(view_id_str, NULL, 10);
if(view_id < 1) {
*errstr = log_error("View id number must be an integer larger or "
"equal to 1. Got %ld",
view_id);
return -1;
}
char *nws_str = strtok_r(NULL, " ", &saveptr);
if(nws_str == NULL) {
*errstr = log_error("Expected argument for \"moveviewtoworkspace\" "
"action, got none.");
return -1;
}
long ws = strtol(nws_str, NULL, 10);
if(ws < 1) {
*errstr = log_error("Workspace number must be an integer larger or "
"equal to 1. Got %ld",
ws);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = view_id;
keybinding->data.us[1] = ws - 1;
keybinding->data.us[2] = follow;
} else if(strcmp(action, "moveviewtotile") == 0) {
keybinding->action = KEYBINDING_MOVE_VIEW_TO_TILE;
char *view_id_str = strtok_r(NULL, " ", &saveptr);
if(view_id_str == NULL) {
*errstr = log_error(
"Expected argument for \"moveviewtotile\" action, got none.");
return -1;
}
long view_id = strtol(view_id_str, NULL, 10);
if(view_id < 1) {
*errstr = log_error("View id number must be an integer larger or "
"equal to 1. Got %ld",
view_id);
return -1;
}
char *tile_id_str = strtok_r(NULL, " ", &saveptr);
if(tile_id_str == NULL) {
*errstr = log_error("Expected argument for \"moveviewtoworkspace\" "
"action, got none.");
return -1;
}
long tile_id = strtol(tile_id_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("Tile id must be an integer larger or "
"equal to 1. Got %ld",
tile_id);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = view_id;
keybinding->data.us[1] = tile_id;
keybinding->data.us[2] = follow;
} else if(strcmp(action, "movetoscreen") == 0) {
keybinding->action = KEYBINDING_MOVE_TO_OUTPUT;
char *noutp_str = strtok_r(NULL, " ", &saveptr);
if(noutp_str == NULL) {
*errstr = log_error(
"Expected argument for \"movetoscreen\" action, got none.");
return -1;
}
long outp = strtol(noutp_str, NULL, 10);
if(outp < 1) {
*errstr = log_error("Output number must be an integer larger or "
"equal to 1. Got %ld",
outp);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = outp;
keybinding->data.us[1] = follow;
} else if(strcmp(action, "movetoworkspace") == 0) {
keybinding->action = KEYBINDING_MOVE_TO_WORKSPACE;
char *nws_str = strtok_r(NULL, " ", &saveptr);
if(nws_str == NULL) {
*errstr = log_error(
"Expected argument for \"movetoworkspace\" action, got none.");
return -1;
}
long ws = strtol(nws_str, NULL, 10);
if(ws < 1) {
*errstr = log_error("Workspace number must be an integer larger or "
"equal to 1. Got %ld",
ws);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = ws - 1;
keybinding->data.us[1] = follow;
} else if(strcmp(action, "movetotile") == 0) {
keybinding->action = KEYBINDING_MOVE_TO_TILE;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str == NULL) {
*errstr = log_error(
"Expected argument for \"movetotile\" action, got none.");
return -1;
}
long tile = strtol(tile_str, NULL, 10);
if(tile < 1) {
*errstr = log_error("Tile number must be an integer larger or "
"equal to 1. Got %ld",
tile);
return -1;
}
long follow = 1;
char *follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
follow = 1;
} else if(strcmp(follow_str, "false") == 0) {
follow = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
keybinding->data.us[0] = tile;
keybinding->data.us[1] = follow;
} else if(strcmp(action, "mergeleft") == 0) {
keybinding->action = KEYBINDING_MERGE_LEFT;
keybinding->data.u = 0;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.u = tile_id;
}
} else if(strcmp(action, "mergeright") == 0) {
keybinding->action = KEYBINDING_MERGE_RIGHT;
keybinding->data.u = 0;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.u = tile_id;
}
} else if(strcmp(action, "mergeup") == 0) {
keybinding->action = KEYBINDING_MERGE_TOP;
keybinding->data.u = 0;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.u = tile_id;
}
} else if(strcmp(action, "mergedown") == 0) {
keybinding->action = KEYBINDING_MERGE_BOTTOM;
keybinding->data.u = 0;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.u = tile_id;
}
} else if(strcmp(action, "exchangeleft") == 0) {
keybinding->action = KEYBINDING_SWAP_LEFT;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 1;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
if(strcmp(tile_str, "true") != 0 &&
strcmp(tile_str, "false") != 0) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.us[0] = tile_id;
tile_str = strtok_r(NULL, " ", &saveptr);
}
if(tile_str != NULL) {
if(strcmp(tile_str, "true") == 0) {
keybinding->data.us[1] = 1;
} else if(strcmp(tile_str, "false") == 0) {
keybinding->data.us[1] = 0;
} else {
*errstr = log_error("The value of \"follow\" must be "
"\"true\" or \"false\", got %s",
tile_str);
return -1;
}
}
}
} else if(strcmp(action, "exchangeright") == 0) {
keybinding->action = KEYBINDING_SWAP_RIGHT;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 1;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
if(strcmp(tile_str, "true") != 0 &&
strcmp(tile_str, "false") != 0) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.us[0] = tile_id;
tile_str = strtok_r(NULL, " ", &saveptr);
}
if(tile_str != NULL) {
if(strcmp(tile_str, "true") == 0) {
keybinding->data.us[1] = 1;
} else if(strcmp(tile_str, "false") == 0) {
keybinding->data.us[1] = 0;
} else {
*errstr = log_error("The value of \"follow\" must be "
"\"true\" or \"false\", got %s",
tile_str);
return -1;
}
}
}
} else if(strcmp(action, "exchangeup") == 0) {
keybinding->action = KEYBINDING_SWAP_TOP;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 1;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
if(strcmp(tile_str, "true") != 0 &&
strcmp(tile_str, "false") != 0) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.us[0] = tile_id;
tile_str = strtok_r(NULL, " ", &saveptr);
}
if(tile_str != NULL) {
if(strcmp(tile_str, "true") == 0) {
keybinding->data.us[1] = 1;
} else if(strcmp(tile_str, "false") == 0) {
keybinding->data.us[1] = 0;
} else {
*errstr = log_error("The value of \"follow\" must be "
"\"true\" or \"false\", got %s",
tile_str);
return -1;
}
}
}
} else if(strcmp(action, "exchangedown") == 0) {
keybinding->action = KEYBINDING_SWAP_BOTTOM;
keybinding->data.us[0] = 0;
keybinding->data.us[1] = 1;
char *tile_str = strtok_r(NULL, " ", &saveptr);
if(tile_str != NULL) {
if(strcmp(tile_str, "true") != 0 &&
strcmp(tile_str, "false") != 0) {
long tile_id = strtol(tile_str, NULL, 10);
if(tile_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile_id);
return -1;
}
keybinding->data.us[0] = tile_id;
tile_str = strtok_r(NULL, " ", &saveptr);
}
if(tile_str != NULL) {
if(strcmp(tile_str, "true") == 0) {
keybinding->data.us[1] = 1;
} else if(strcmp(tile_str, "false") == 0) {
keybinding->data.us[1] = 0;
} else {
*errstr = log_error("The value of \"follow\" must be "
"\"true\" or \"false\", got %s",
tile_str);
return -1;
}
}
}
} else if(strcmp(action, "exchange") == 0) {
keybinding->action = KEYBINDING_SWAP;
char *tile1_str = strtok_r(NULL, " ", &saveptr);
keybinding->data.us[2] = 1;
if(tile1_str == NULL) {
*errstr = log_error(
"\"exchange\" requires two arguments, but none were provided");
return -1;
}
long tile1_id = strtol(tile1_str, NULL, 10);
if(tile1_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile1_id);
return -1;
}
keybinding->data.us[0] = tile1_id;
char *tile2_str = strtok_r(NULL, " ", &saveptr);
if(tile2_str == NULL) {
*errstr = log_error("\"exchange\" requires two arguments, but only "
"one was provided");
return -1;
}
long tile2_id = strtol(tile2_str, NULL, 10);
if(tile2_id < 1) {
*errstr = log_error("The tile id must be an integer number "
"larger or equal to 1. Got %ld",
tile2_id);
return -1;
}
keybinding->data.us[1] = tile2_id;
char *follow_str = NULL;
follow_str = strtok_r(NULL, " ", &saveptr);
if(follow_str != NULL) {
if(strcmp(follow_str, "true") == 0) {
keybinding->data.us[2] = 1;
} else if(strcmp(follow_str, "false") == 0) {
keybinding->data.us[2] = 0;
} else {
*errstr = log_error("The value of \"follow\" must be \"true\" "
"or \"false\", got %s",
follow_str);
return -1;
}
}
} else if(strcmp(action, "focusleft") == 0) {
keybinding->action = KEYBINDING_FOCUS_LEFT;
} else if(strcmp(action, "focusright") == 0) {
keybinding->action = KEYBINDING_FOCUS_RIGHT;
} else if(strcmp(action, "focusup") == 0) {
keybinding->action = KEYBINDING_FOCUS_TOP;
} else if(strcmp(action, "focusdown") == 0) {
keybinding->action = KEYBINDING_FOCUS_BOTTOM;
} else if(strcmp(action, "movetonextscreen") == 0) {
keybinding->action = KEYBINDING_MOVE_VIEW_TO_CYCLE_OUTPUT;
keybinding->data.b = false;
} else if(strcmp(action, "movetoprevscreen") == 0) {
keybinding->action = KEYBINDING_MOVE_VIEW_TO_CYCLE_OUTPUT;
keybinding->data.b = true;
} else if(strcmp(action, "switchvt") == 0) {
keybinding->action = KEYBINDING_CHANGE_TTY;
char *ntty = strtok_r(NULL, " ", &saveptr);
if(ntty == NULL) {
*errstr = log_error(
"Expected argument for \"switchvt\" command, got none.");
return -1;
}
long tty = strtol(ntty, NULL, 10);
keybinding->data.u = tty;
} else if(strcmp(action, "mode") == 0) {
keybinding->action = KEYBINDING_SWITCH_MODE;
char *mode = strtok_r(NULL, " ", &saveptr);
if(mode == NULL) {
*errstr = log_error("Expected mode after \"mode\". Got nothing.");
return -1;
}
int mode_idx = get_mode_index_from_name(server->modes, mode);
if(mode_idx == -1) {
*errstr = log_error("Unknown mode \"%s\" for \"mode\"", mode);
return -1;
}
keybinding->data.u = (unsigned int)mode_idx;
} else if(strcmp(action, "setmode") == 0) {
keybinding->action = KEYBINDING_SWITCH_DEFAULT_MODE;
char *mode = strtok_r(NULL, " ", &saveptr);
if(mode == NULL) {
*errstr = log_error(
"Expected mode after \"switch_default_mode\". Got nothing.");
return -1;
}
int mode_idx = get_mode_index_from_name(server->modes, mode);
if(mode_idx == -1) {
*errstr =
log_error("Unknown mode \"%s\" for switch_default_mode", mode);
return -1;
}
keybinding->data.u = (unsigned int)mode_idx;
} else if(strcmp(action, "setmodecursor") == 0) {
keybinding->action = KEYBINDING_SETMODECURSOR;
char *mode = strtok_r(NULL, " ", &saveptr);
if(mode == NULL) {
*errstr = log_error("Expected mode name and cursor name after "
"\"setmodecursor\". Got nothing.");
return -1;
}
char *cursor = strtok_r(NULL, " ", &saveptr);
if(cursor == NULL) {
*errstr = log_error("Expected mode name and cursor name after "
"\"setmodecursor\". Got nothing.");
return -1;
}
keybinding->data.cs[0] = strdup(mode);
keybinding->data.cs[1] = strdup(cursor);
} else if(strcmp(action, "bind") == 0) {
keybinding->action = KEYBINDING_DEFINEKEY;
keybinding->data.kb =
parse_bind(server, &saveptr, errstr, nesting_level);
if(keybinding->data.kb == NULL) {
return -1;
}
} else if(strcmp(action, "definekey") == 0) {
keybinding->action = KEYBINDING_DEFINEKEY;
keybinding->data.kb =
parse_definekey(server, &saveptr, errstr, nesting_level);
if(keybinding->data.kb == NULL) {
return -1;
}
} else if(strcmp(action, "background") == 0) {
keybinding->action = KEYBINDING_BACKGROUND;
if(parse_background(keybinding->data.color, &saveptr, errstr) != 0) {
return -1;
}
} else if(strcmp(action, "escape") == 0) {
keybinding->action = KEYBINDING_DEFINEKEY;
keybinding->data.kb = parse_escape(&saveptr, errstr);
if(keybinding->data.kb == NULL) {
return -1;
}
} else if(strcmp(action, "cursor") == 0) {
keybinding->action = KEYBINDING_CURSOR;
keybinding->data.i = parse_cursor(&saveptr, errstr);
if(keybinding->data.i < 0) {
return -1;
}
} else if(strcmp(action, "definemode") == 0) {
keybinding->action = KEYBINDING_DEFINEMODE;
keybinding->data.c = parse_definemode(&saveptr, errstr);
if(keybinding->data.c == NULL) {
return -1;
}
} else if(strcmp(action, "workspaces") == 0) {
keybinding->action = KEYBINDING_WORKSPACES;
keybinding->data.i = parse_workspaces(&saveptr, errstr);
if(keybinding->data.i < 0) {
return -1;
}
} else if(strcmp(action, "output") == 0) {
keybinding->action = KEYBINDING_CONFIGURE_OUTPUT;
keybinding->data.o_cfg = parse_output_config(&saveptr, errstr);
if(keybinding->data.o_cfg == NULL) {
return -1;
}
} else if(strcmp(action, "input") == 0) {
keybinding->action = KEYBINDING_CONFIGURE_INPUT;
keybinding->data.i_cfg = parse_input_config(&saveptr, errstr);
if(keybinding->data.i_cfg == NULL) {
return -1;
}
} else if(strcmp(action, "configure_message") == 0) {
keybinding->action = KEYBINDING_CONFIGURE_MESSAGE;
keybinding->data.m_cfg = parse_message_config(&saveptr, errstr);
if(keybinding->data.m_cfg == NULL) {
return -1;
}
} else if(strcmp(action, "configure_status_bar") == 0) {
keybinding->action = KEYBINDING_CONFIGURE_STATUS_BAR;
keybinding->data.sb_cfg = parse_status_bar_config(&saveptr, errstr);
if(keybinding->data.sb_cfg == NULL) {
return -1;
}
} else if(strcmp(action, "configure_wallpaper") == 0) {
keybinding->action = KEYBINDING_CONFIGURE_WALLPAPER;
keybinding->data.wp_cfg = parse_wallpaper_config(&saveptr, errstr);
if(keybinding->data.wp_cfg == NULL) {
return -1;
}
} else {
*errstr = log_error("Error, unsupported action \"%s\".", action);
return -1;
}
return 0;
}
int
parse_rc_line(struct nedm_server *server, char *line, char **errstr) {
char *saveptr = strdup(line); // Used internally by strtok_r
struct keybinding *keybinding = malloc(sizeof(struct keybinding));
if(keybinding == NULL) {
*errstr = log_error(
"Failed to allocate memory for temporary keybinding struct.");
free(keybinding);
free(saveptr);
return -1;
}
if(parse_command(server, keybinding, saveptr, errstr, 1) != 0) {
wlr_log(WLR_ERROR, "Error parsing command.");
free(keybinding);
free(saveptr);
return -1;
}
run_action(keybinding->action, server, keybinding->data);
keybinding_free(keybinding, false);
free(saveptr);
return 0;
}