18CFGFUN(include,
const char *pattern) {
19 DLOG(
"include %s\n", pattern);
22 const int ret = wordexp(pattern, &p, 0);
24 ELOG(
"wordexp(%s): error %d\n", pattern, ret);
25 result->has_errors =
true;
28 char **w = p.we_wordv;
29 for (
size_t i = 0; i < p.we_wordc; i++) {
30 char resolved_path[PATH_MAX] = {
'\0'};
31 if (realpath(w[i], resolved_path) == NULL) {
32 LOG(
"Skipping %s: %s\n", w[i], strerror(errno));
39 if (strcmp(file->
path, resolved_path) == 0) {
45 LOG(
"Skipping file %s (already included)\n", resolved_path);
49 LOG(
"Including config file %s\n", resolved_path);
58 .use_nagbar = result->ctx->use_nagbar,
64 .variables = result->ctx->variables,
71 ELOG(
"including config file %s: %s\n", resolved_path, strerror(errno));
75 result->has_errors =
true;
106 DLOG(
"Initializing criteria, current_match = %p, state = %d\n",
current_match, _state);
120CFGFUN(criteria_add,
const char *ctype,
const char *cvalue) {
138 if (strstr(str,
"Mod1") != NULL)
139 result |= XCB_KEY_BUT_MASK_MOD_1;
140 if (strstr(str,
"Mod2") != NULL)
141 result |= XCB_KEY_BUT_MASK_MOD_2;
142 if (strstr(str,
"Mod3") != NULL)
143 result |= XCB_KEY_BUT_MASK_MOD_3;
144 if (strstr(str,
"Mod4") != NULL)
145 result |= XCB_KEY_BUT_MASK_MOD_4;
146 if (strstr(str,
"Mod5") != NULL)
147 result |= XCB_KEY_BUT_MASK_MOD_5;
148 if (strstr(str,
"Control") != NULL ||
149 strstr(str,
"Ctrl") != NULL)
150 result |= XCB_KEY_BUT_MASK_CONTROL;
151 if (strstr(str,
"Shift") != NULL)
152 result |= XCB_KEY_BUT_MASK_SHIFT;
154 if (strstr(str,
"Group1") != NULL)
156 if (strstr(str,
"Group2") != NULL ||
157 strstr(str,
"Mode_switch") != NULL)
159 if (strstr(str,
"Group3") != NULL)
161 if (strstr(str,
"Group4") != NULL)
171CFGFUN(binding,
const char *bindtype,
const char *modifiers,
const char *key,
const char *release,
const char *border,
const char *whole_window,
const char *exclude_titlebar,
const char *command) {
182CFGFUN(mode_binding,
const char *bindtype,
const char *modifiers,
const char *key,
const char *release,
const char *border,
const char *whole_window,
const char *exclude_titlebar,
const char *command) {
191CFGFUN(enter_mode,
const char *pango_markup,
const char *modename) {
199 if (strcmp(mode->
name, modename) == 0) {
200 ELOG(
"The binding mode with name \"%s\" is defined at least twice.\n", modename);
204 DLOG(
"\t now in mode %s\n", modename);
210CFGFUN(exec,
const char *exectype,
const char *no_startup_id,
const char *command) {
214 if (strcmp(exectype,
"exec") == 0) {
223 ELOG(
"Match is empty, ignoring this for_window statement\n");
226 DLOG(
"\t should execute command %s for the criteria mentioned above\n",
command);
228 assignment->
type = A_COMMAND;
260 DLOG(
"Setting gaps for workspace %s", workspace);
265 if (strcasecmp(assignment->
name, workspace) == 0) {
275 assignment->
output = NULL;
284 if (!strcmp(scope,
"inner")) {
286 }
else if (!strcmp(scope,
"outer")) {
288 }
else if (!strcmp(scope,
"vertical")) {
290 }
else if (!strcmp(scope,
"horizontal")) {
292 }
else if (!strcmp(scope,
"top")) {
294 }
else if (!strcmp(scope,
"right")) {
296 }
else if (!strcmp(scope,
"bottom")) {
298 }
else if (!strcmp(scope,
"left")) {
301 ELOG(
"Invalid command, cannot process scope %s", scope);
305CFGFUN(
gaps,
const char *workspace,
const char *scope,
const long value) {
309 if (workspace == NULL) {
316CFGFUN(smart_borders,
const char *enable) {
317 if (!strcmp(enable,
"no_gaps"))
324 if (!strcmp(enable,
"inverse_outer"))
330CFGFUN(floating_minimum_size,
const long width,
const long height) {
331 config.floating_minimum_width = width;
332 config.floating_minimum_height = height;
335CFGFUN(floating_maximum_size,
const long width,
const long height) {
336 config.floating_maximum_width = width;
337 config.floating_maximum_height = height;
340CFGFUN(floating_modifier,
const char *modifiers) {
344CFGFUN(default_orientation,
const char *orientation) {
345 if (strcmp(orientation,
"horizontal") == 0)
347 else if (strcmp(orientation,
"vertical") == 0)
353CFGFUN(workspace_layout,
const char *layout) {
354 if (strcmp(layout,
"default") == 0)
356 else if (strcmp(layout,
"stacking") == 0 ||
357 strcmp(layout,
"stacked") == 0)
363CFGFUN(default_border,
const char *windowtype,
const char *border,
const long width) {
367 if (strcmp(border,
"1pixel") == 0) {
370 }
else if (strcmp(border,
"none") == 0) {
373 }
else if (strcmp(border,
"pixel") == 0) {
375 border_width = width;
378 border_width = width;
381 if ((strcmp(windowtype,
"default_border") == 0) ||
382 (strcmp(windowtype,
"new_window") == 0)) {
383 DLOG(
"default tiled border style = %d and border width = %d (%d physical px)\n",
384 border_style, border_width,
logical_px(border_width));
385 config.default_border = border_style;
388 DLOG(
"default floating border style = %d and border width = %d (%d physical px)\n",
389 border_style, border_width,
logical_px(border_width));
390 config.default_floating_border = border_style;
395CFGFUN(hide_edge_borders,
const char *borders) {
396 if (strcmp(borders,
"smart_no_gaps") == 0)
398 else if (strcmp(borders,
"smart") == 0)
400 else if (strcmp(borders,
"vertical") == 0)
402 else if (strcmp(borders,
"horizontal") == 0)
404 else if (strcmp(borders,
"both") == 0)
406 else if (strcmp(borders,
"none") == 0)
414CFGFUN(focus_follows_mouse,
const char *value) {
418CFGFUN(mouse_warping,
const char *value) {
419 if (strcmp(value,
"none") == 0)
421 else if (strcmp(value,
"output") == 0)
429CFGFUN(disable_randr15,
const char *value) {
433CFGFUN(focus_wrapping,
const char *value) {
434 if (strcmp(value,
"force") == 0) {
436 }
else if (strcmp(value,
"workspace") == 0) {
445CFGFUN(force_focus_wrapping,
const char *value) {
463 free(
config.fake_outputs);
467CFGFUN(force_display_urgency_hint,
const long duration_ms) {
468 config.workspace_urgency_timer = duration_ms / 1000.0;
471CFGFUN(focus_on_window_activation,
const char *mode) {
472 if (strcmp(mode,
"smart") == 0)
473 config.focus_on_window_activation = FOWA_SMART;
474 else if (strcmp(mode,
"urgent") == 0)
475 config.focus_on_window_activation = FOWA_URGENT;
476 else if (strcmp(mode,
"focus") == 0)
477 config.focus_on_window_activation = FOWA_FOCUS;
478 else if (strcmp(mode,
"none") == 0)
479 config.focus_on_window_activation = FOWA_NONE;
481 ELOG(
"Unknown focus_on_window_activation mode \"%s\", ignoring it.\n", mode);
485 DLOG(
"Set new focus_on_window_activation mode = %i.\n",
config.focus_on_window_activation);
488CFGFUN(title_align,
const char *alignment) {
489 if (strcmp(alignment,
"left") == 0) {
490 config.title_align = ALIGN_LEFT;
491 }
else if (strcmp(alignment,
"center") == 0) {
492 config.title_align = ALIGN_CENTER;
493 }
else if (strcmp(alignment,
"right") == 0) {
494 config.title_align = ALIGN_RIGHT;
517 if (strcasecmp(assignment->
name, workspace) == 0) {
518 if (assignment->
output != NULL) {
519 ELOG(
"You have a duplicate workspace assignment for workspace \"%s\"\n",
529 DLOG(
"Both workspace and current_workspace are NULL, assuming we had an error before\n");
535 DLOG(
"Assigning workspace \"%s\" to output \"%s\"\n", workspace,
output);
544 free(
config.ipc_socket_path);
549 free(
config.restart_state_path);
553CFGFUN(popup_during_fullscreen,
const char *value) {
554 if (strcmp(value,
"ignore") == 0) {
555 config.popup_during_fullscreen = PDF_IGNORE;
556 }
else if (strcmp(value,
"leave_fullscreen") == 0) {
557 config.popup_during_fullscreen = PDF_LEAVE_FULLSCREEN;
559 config.popup_during_fullscreen = PDF_SMART;
563CFGFUN(color_single,
const char *colorclass,
const char *color) {
568CFGFUN(color,
const char *colorclass,
const char *qubelabel,
const char *border,
const char *background,
const char *text,
const char *indicator,
const char *child_border) {
569#define APPLY_COLORS(classname, label) \
571 if (strcmp(colorclass, "client." #classname) == 0) { \
572 if (strcmp("focused_tab_title", #classname) == 0) { \
573 config.client[label].got_focused_tab_title = true; \
574 if (indicator || child_border) { \
575 ELOG("indicator and child_border colors have no effect for client.focused_tab_title\n"); \
578 config.client[label].classname.border = draw_util_hex_to_color(border); \
579 config.client[label].classname.background = draw_util_hex_to_color(background); \
580 config.client[label].classname.text = draw_util_hex_to_color(text); \
581 if (indicator != NULL) { \
582 config.client[label].classname.indicator = draw_util_hex_to_color(indicator); \
584 if (child_border != NULL) { \
585 config.client[label].classname.child_border = draw_util_hex_to_color(child_border); \
587 config.client[label].classname.child_border = config.client[label].classname.background; \
595 if (strcmp(qubelabel,
"dom0") == 0) {
597 }
else if (strcmp(qubelabel,
"red") == 0) {
599 }
else if (strcmp(qubelabel,
"orange") == 0) {
601 }
else if (strcmp(qubelabel,
"yellow") == 0) {
603 }
else if (strcmp(qubelabel,
"green") == 0) {
605 }
else if (strcmp(qubelabel,
"gray") == 0) {
607 }
else if (strcmp(qubelabel,
"blue") == 0) {
609 }
else if (strcmp(qubelabel,
"purple") == 0) {
611 }
else if (strcmp(qubelabel,
"black") == 0) {
630 ELOG(
"Match is empty, ignoring this assignment\n");
635 ELOG(
"Assignments using window mode (floating/tiling) is not supported\n");
639 DLOG(
"New assignment, using above criteria, to output \"%s\".\n",
output);
642 assignment->
type = A_TO_OUTPUT;
647CFGFUN(assign,
const char *workspace,
bool is_number) {
649 ELOG(
"Match is empty, ignoring this assignment\n");
654 ELOG(
"Assignments using window mode (floating/tiling) is not supported\n");
659 ELOG(
"Could not parse initial part of \"%s\" as a number.\n", workspace);
663 DLOG(
"New assignment, using above criteria, to workspace \"%s\".\n", workspace);
666 assignment->
type = is_number ? A_TO_WORKSPACE_NUMBER : A_TO_WORKSPACE;
673 ELOG(
"Match is empty, ignoring this assignment\n");
677 DLOG(
"New assignment, using above criteria, to ignore focus on manage.\n");
680 assignment->
type = A_NO_FOCUS;
684CFGFUN(ipc_kill_timeout,
const long timeout_ms) {
689 if (strcmp(value,
"modifier") == 0) {
691 }
else if (strcmp(value,
"titlebar") == 0) {
693 }
else if (strcmp(value,
"modifier,titlebar") == 0 ||
694 strcmp(value,
"titlebar,modifier") == 0) {
713CFGFUN(bar_separator_symbol,
const char *separator) {
719 current_bar->mode = (strcmp(mode,
"dock") == 0 ? M_DOCK : (strcmp(mode,
"hide") == 0 ? M_HIDE : M_INVISIBLE));
722CFGFUN(bar_hidden_state,
const char *hidden_state) {
723 current_bar->hidden_state = (strcmp(hidden_state,
"hide") == 0 ? S_HIDE : S_SHOW);
746 DLOG(
"padding now: x=%d, y=%d, w=%d, h=%d\n",
761CFGFUN(bar_padding_two,
const long top_and_bottom,
const long right_and_left) {
764 current_bar->padding.width = (uint32_t)right_and_left;
765 current_bar->padding.height = (uint32_t)top_and_bottom;
769CFGFUN(bar_padding_three,
const long top,
const long right_and_left,
const long bottom) {
772 current_bar->padding.width = (uint32_t)right_and_left;
777CFGFUN(bar_padding_four,
const long top,
const long right,
const long bottom,
const long left) {
785CFGFUN(bar_modifier,
const char *modifiers) {
790 if (strncasecmp(button,
"button", strlen(
"button")) != 0) {
791 ELOG(
"Bindings for a bar can only be mouse bindings, not \"%s\", ignoring.\n", button);
795 int input_code = atoi(button + strlen(
"button"));
796 if (input_code < 1) {
797 ELOG(
"Button \"%s\" does not seem to be in format 'buttonX'.\n", button);
800 const bool release_bool = release != NULL;
805 ELOG(
"command for button %s was already specified, ignoring.\n", button);
811 new_binding->
release = release_bool;
818 ELOG(
"'wheel_up_cmd' is deprecated. Please us 'bindsym button4 %s' instead.\n",
command);
823 ELOG(
"'wheel_down_cmd' is deprecated. Please us 'bindsym button5 %s' instead.\n",
command);
831CFGFUN(bar_position,
const char *position) {
832 current_bar->position = (strcmp(position,
"top") == 0 ? P_TOP : P_BOTTOM);
835CFGFUN(bar_i3bar_command,
const char *i3bar_command) {
840CFGFUN(bar_color,
const char *colorclass,
const char *border,
const char *background,
const char *text) {
841#define APPLY_COLORS(classname) \
843 if (strcmp(colorclass, #classname) == 0) { \
844 if (text != NULL) { \
846 current_bar->colors.classname##_border = sstrdup(border); \
847 current_bar->colors.classname##_bg = sstrdup(background); \
848 current_bar->colors.classname##_text = sstrdup(text); \
851 current_bar->colors.classname##_bg = sstrdup(background); \
852 current_bar->colors.classname##_text = sstrdup(border); \
866CFGFUN(bar_socket_path,
const char *socket_path) {
871CFGFUN(bar_tray_output,
const char *output) {
877CFGFUN(bar_tray_padding,
const long padding_px) {
881CFGFUN(bar_color_single,
const char *colorclass,
const char *color) {
882 if (strcmp(colorclass,
"background") == 0)
884 else if (strcmp(colorclass,
"separator") == 0)
886 else if (strcmp(colorclass,
"statusline") == 0)
888 else if (strcmp(colorclass,
"focused_background") == 0)
890 else if (strcmp(colorclass,
"focused_separator") == 0)
896CFGFUN(bar_status_command,
const char *command) {
901CFGFUN(bar_workspace_command,
const char *command) {
906CFGFUN(bar_binding_mode_indicator,
const char *value) {
910CFGFUN(bar_workspace_buttons,
const char *value) {
914CFGFUN(bar_workspace_min_width,
const long width) {
918CFGFUN(bar_strip_workspace_numbers,
const char *value) {
922CFGFUN(bar_strip_workspace_name,
const char *value) {
935 DLOG(
"\t new bar configuration finished, saving.\n");
940 config.number_barconfigs++;
bool match_is_empty(Match *match)
Check if a match is empty.
void match_init(Match *match)
Initializes the Match data structure.
void match_copy(Match *dest, Match *src)
Copies the data of a match from src to dest.
void match_free(Match *match)
Frees the given match.
void match_parse_property(Match *match, const char *ctype, const char *cvalue)
Interprets a ctype=cvalue pair and adds it to the given match specification.
struct autostarts_always_head autostarts_always
struct autostarts_head autostarts
struct assignments_head assignments
struct ws_assignments_head ws_assignments
struct bindings_head * bindings
void tiling_drag(Con *con, xcb_button_press_event_t *event, bool use_threshold)
Initiates a mouse drag operation on a tiled window.
struct outputs_head outputs
struct includedfiles_head included_files
struct barconfig_head barconfigs
parse_file_result_t parse_file(struct parser_ctx *ctx, const char *f, IncludedFile *included_file)
Parses the given file by first replacing the variables, then calling parse_config and launching i3-na...
static Match current_match
const char * DEFAULT_BINDING_MODE
The name of the default mode.
Binding * configure_binding(const char *bindtype, const char *modifiers, const char *input_code, const char *release, const char *border, const char *whole_window, const char *exclude_titlebar, const char *command, const char *modename, bool pango_markup)
Adds a binding from config parameters given as strings and returns a pointer to the binding structure...
void workspace_back_and_forth(void)
Focuses the previously focused workspace.
static xcb_cursor_context_t * ctx
int ws_name_to_number(const char *name)
Parses the workspace name as a number.
#define APPLY_COLORS(classname, label)
static void apply_gaps(gaps_t *gaps, gaps_mask_t mask, int value)
static bool current_mode_pango_markup
static void bar_configure_binding(const char *button, const char *release, const char *command)
static void create_gaps_assignment(const char *workspace, const gaps_mask_t mask, const int pixels)
static char * current_workspace
static void dlog_padding(void)
static int criteria_next_state
static Barconfig * current_bar
static char * current_mode
static gaps_mask_t gaps_scope_to_mask(const char *scope)
i3_event_state_mask_t event_state_from_str(const char *str)
A utility function to convert a string containing the group and modifiers to the corresponding bit ma...
#define SLIST_FOREACH(var, head, field)
#define TAILQ_FOREACH(var, head, field)
#define TAILQ_INSERT_TAIL(head, elm, field)
#define TAILQ_REMOVE(head, elm, field)
@ SMART_GAPS_INVERSE_OUTER
uint32_t i3_event_state_mask_t
The lower 16 bits contain a xcb_key_but_mask_t, the higher 16 bits contain an i3_xkb_group_mask_t.
@ FOCUS_WRAPPING_WORKSPACE
qube_label_t
Qubes colors.
int logical_px(const int logical)
Convert a logical amount of pixels (e.g.
char * sstrdup(const char *str)
Safe-wrapper around strdup which exits if malloc returns NULL (meaning that there is no more memory a...
void * scalloc(size_t num, size_t size)
Safe-wrapper around calloc which exits if malloc returns NULL (meaning that there is no more memory a...
color_t draw_util_hex_to_color(const char *color)
Parses the given color in hex format to an internal color representation.
int sasprintf(char **strp, const char *fmt,...)
Safe-wrapper around asprintf which exits if it returns -1 (meaning that there is no more memory avail...
void * srealloc(void *ptr, size_t size)
Safe-wrapper around realloc which exits if realloc returns NULL (meaning that there is no more memory...
bool boolstr(const char *str)
Reports whether str represents the enabled state (1, yes, true, …).
void set_font(i3Font *font)
Defines the font to be used for the forthcoming calls.
i3Font load_font(const char *pattern, const bool fallback)
Loads a font for usage, also getting its height.
void * smalloc(size_t size)
Safe-wrapper around malloc which exits if malloc returns NULL (meaning that there is no more memory a...
@ TILING_DRAG_MODIFIER_OR_TITLEBAR
@ PARSE_FILE_CONFIG_ERRORS
void ipc_set_kill_timeout(ev_tstamp new)
Set the maximum duration that we allow for a connection with an unwriteable socket.
List entry struct for an included file.
char * variable_replaced_contents
The configuration file can contain multiple sets of bindings.
Holds the status bar configuration (i3bar).
Defines a mouse command to be executed instead of the default behavior when clicking on the non-statu...
bool release
If true, the command will be executed after the button is released.
int input_code
The button to be used (e.g., 1 for "button1").
char * command
The command which is to be executed for this button.
Stores which workspace (by name or number) goes to which output and its gaps config.
Holds a command specified by either an:
bool no_startup_id
no_startup_id flag for start_application().
char * command
Command, like in command mode.
An Assignment makes specific windows go to a specific workspace/output or run a command for that wind...
union Assignment::@222204053252343114274042165165310271133332337156 dest
destination workspace/command/output, depending on the type
Match match
the criteria to check if a window matches
enum Assignment::@362164307203113023013316042237071156073024074140 type
type of this assignment: