18 CFGFUN(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);
120 CFGFUN(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)
177 CFGFUN(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) {
188 CFGFUN(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) {
192 CFGFUN(enter_mode,
const char *pango_markup,
const char *modename) {
200 if (strcmp(mode->
name, modename) == 0) {
201 ELOG(
"The binding mode with name \"%s\" is defined at least twice.\n", modename);
205 DLOG(
"\t now in mode %s\n", modename);
211 CFGFUN(exec,
const char *exectype,
const char *no_startup_id,
const char *command) {
215 if (strcmp(exectype,
"exec") == 0) {
224 ELOG(
"Match is empty, ignoring this for_window statement\n");
227 DLOG(
"\t should execute command %s for the criteria mentioned above\n",
command);
229 assignment->
type = A_COMMAND;
235 CFGFUN(floating_minimum_size,
const long width,
const long height) {
240 CFGFUN(floating_maximum_size,
const long width,
const long height) {
245 CFGFUN(floating_modifier,
const char *modifiers) {
249 CFGFUN(default_orientation,
const char *orientation) {
250 if (strcmp(orientation,
"horizontal") == 0)
252 else if (strcmp(orientation,
"vertical") == 0)
258 CFGFUN(workspace_layout,
const char *layout) {
259 if (strcmp(layout,
"default") == 0)
261 else if (strcmp(layout,
"stacking") == 0 ||
262 strcmp(layout,
"stacked") == 0)
268 CFGFUN(default_border,
const char *windowtype,
const char *border,
const long width) {
272 if (strcmp(border,
"1pixel") == 0) {
275 }
else if (strcmp(border,
"none") == 0) {
278 }
else if (strcmp(border,
"pixel") == 0) {
280 border_width = width;
283 border_width = width;
286 if ((strcmp(windowtype,
"default_border") == 0) ||
287 (strcmp(windowtype,
"new_window") == 0)) {
288 DLOG(
"default tiled border style = %d and border width = %d (%d physical px)\n",
289 border_style, border_width,
logical_px(border_width));
293 DLOG(
"default floating border style = %d and border width = %d (%d physical px)\n",
294 border_style, border_width,
logical_px(border_width));
300 CFGFUN(hide_edge_borders,
const char *borders) {
301 if (strcmp(borders,
"smart") == 0)
303 else if (strcmp(borders,
"vertical") == 0)
305 else if (strcmp(borders,
"horizontal") == 0)
307 else if (strcmp(borders,
"both") == 0)
309 else if (strcmp(borders,
"none") == 0)
317 CFGFUN(focus_follows_mouse,
const char *value) {
321 CFGFUN(mouse_warping,
const char *value) {
322 if (strcmp(value,
"none") == 0)
324 else if (strcmp(value,
"output") == 0)
332 CFGFUN(disable_randr15,
const char *value) {
336 CFGFUN(focus_wrapping,
const char *value) {
337 if (strcmp(value,
"force") == 0) {
339 }
else if (strcmp(value,
"workspace") == 0) {
348 CFGFUN(force_focus_wrapping,
const char *value) {
370 CFGFUN(force_display_urgency_hint,
const long duration_ms) {
374 CFGFUN(focus_on_window_activation,
const char *mode) {
375 if (strcmp(mode,
"smart") == 0)
377 else if (strcmp(mode,
"urgent") == 0)
379 else if (strcmp(mode,
"focus") == 0)
381 else if (strcmp(mode,
"none") == 0)
384 ELOG(
"Unknown focus_on_window_activation mode \"%s\", ignoring it.\n", mode);
391 CFGFUN(title_align,
const char *alignment) {
392 if (strcmp(alignment,
"left") == 0) {
394 }
else if (strcmp(alignment,
"center") == 0) {
396 }
else if (strcmp(alignment,
"right") == 0) {
409 CFGFUN(workspace,
const char *workspace,
const char *output) {
420 if (strcasecmp(assignment->
name, workspace) == 0) {
421 ELOG(
"You have a duplicate workspace assignment for workspace \"%s\"\n",
430 DLOG(
"Both workspace and current_workspace are NULL, assuming we had an error before\n");
436 DLOG(
"Assigning workspace \"%s\" to output \"%s\"\n", workspace,
output);
449 CFGFUN(restart_state,
const char *path) {
454 CFGFUN(popup_during_fullscreen,
const char *value) {
455 if (strcmp(value,
"ignore") == 0) {
457 }
else if (strcmp(value,
"leave_fullscreen") == 0) {
464 CFGFUN(color_single,
const char *colorclass,
const char *color) {
469 CFGFUN(color,
const char *colorclass,
const char *border,
const char *background,
const char *text,
const char *indicator,
const char *child_border) {
470 #define APPLY_COLORS(classname) \
472 if (strcmp(colorclass, "client." #classname) == 0) { \
473 config.client.classname.border = draw_util_hex_to_color(border); \
474 config.client.classname.background = draw_util_hex_to_color(background); \
475 config.client.classname.text = draw_util_hex_to_color(text); \
476 if (indicator != NULL) { \
477 config.client.classname.indicator = draw_util_hex_to_color(indicator); \
479 if (child_border != NULL) { \
480 config.client.classname.child_border = draw_util_hex_to_color(child_border); \
482 config.client.classname.child_border = config.client.classname.background; \
498 ELOG(
"Match is empty, ignoring this assignment\n");
503 ELOG(
"Assignments using window mode (floating/tiling) is not supported\n");
507 DLOG(
"New assignment, using above criteria, to output \"%s\".\n",
output);
510 assignment->
type = A_TO_OUTPUT;
515 CFGFUN(assign,
const char *workspace,
bool is_number) {
517 ELOG(
"Match is empty, ignoring this assignment\n");
522 ELOG(
"Assignments using window mode (floating/tiling) is not supported\n");
527 ELOG(
"Could not parse initial part of \"%s\" as a number.\n", workspace);
531 DLOG(
"New assignment, using above criteria, to workspace \"%s\".\n", workspace);
534 assignment->
type = is_number ? A_TO_WORKSPACE_NUMBER : A_TO_WORKSPACE;
541 ELOG(
"Match is empty, ignoring this assignment\n");
545 DLOG(
"New assignment, using above criteria, to ignore focus on manage.\n");
548 assignment->
type = A_NO_FOCUS;
552 CFGFUN(ipc_kill_timeout,
const long timeout_ms) {
567 CFGFUN(bar_separator_symbol,
const char *separator) {
573 current_bar->
mode = (strcmp(mode,
"dock") == 0 ? M_DOCK : (strcmp(mode,
"hide") == 0 ? M_HIDE : M_INVISIBLE));
576 CFGFUN(bar_hidden_state,
const char *hidden_state) {
595 CFGFUN(bar_modifier,
const char *modifiers) {
600 if (strncasecmp(button,
"button", strlen(
"button")) != 0) {
601 ELOG(
"Bindings for a bar can only be mouse bindings, not \"%s\", ignoring.\n", button);
605 int input_code = atoi(button + strlen(
"button"));
606 if (input_code < 1) {
607 ELOG(
"Button \"%s\" does not seem to be in format 'buttonX'.\n", button);
610 const bool release_bool = release != NULL;
615 ELOG(
"command for button %s was already specified, ignoring.\n", button);
621 new_binding->
release = release_bool;
628 ELOG(
"'wheel_up_cmd' is deprecated. Please us 'bindsym button4 %s' instead.\n",
command);
633 ELOG(
"'wheel_down_cmd' is deprecated. Please us 'bindsym button5 %s' instead.\n",
command);
641 CFGFUN(bar_position,
const char *position) {
645 CFGFUN(bar_i3bar_command,
const char *i3bar_command) {
650 CFGFUN(bar_color,
const char *colorclass,
const char *border,
const char *background,
const char *text) {
651 #define APPLY_COLORS(classname) \
653 if (strcmp(colorclass, #classname) == 0) { \
654 if (text != NULL) { \
656 current_bar->colors.classname##_border = sstrdup(border); \
657 current_bar->colors.classname##_bg = sstrdup(background); \
658 current_bar->colors.classname##_text = sstrdup(text); \
661 current_bar->colors.classname##_bg = sstrdup(background); \
662 current_bar->colors.classname##_text = sstrdup(border); \
676 CFGFUN(bar_socket_path,
const char *socket_path) {
681 CFGFUN(bar_tray_output,
const char *output) {
687 CFGFUN(bar_tray_padding,
const long padding_px) {
691 CFGFUN(bar_color_single,
const char *colorclass,
const char *color) {
692 if (strcmp(colorclass,
"background") == 0)
694 else if (strcmp(colorclass,
"separator") == 0)
696 else if (strcmp(colorclass,
"statusline") == 0)
698 else if (strcmp(colorclass,
"focused_background") == 0)
700 else if (strcmp(colorclass,
"focused_separator") == 0)
706 CFGFUN(bar_status_command,
const char *command) {
711 CFGFUN(bar_binding_mode_indicator,
const char *value) {
715 CFGFUN(bar_workspace_buttons,
const char *value) {
719 CFGFUN(bar_workspace_min_width,
const long width) {
723 CFGFUN(bar_strip_workspace_numbers,
const char *value) {
727 CFGFUN(bar_strip_workspace_name,
const char *value) {
740 DLOG(
"\t new bar configuration finished, saving.\n");
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...
static struct stack stack
static Match current_match
struct includedfiles_head included_files
struct barconfig_head barconfigs
static char * font_pattern
static bool current_mode_pango_markup
static void bar_configure_binding(const char *button, const char *release, const char *command)
static char * current_workspace
static int criteria_next_state
CFGFUN(include, const char *pattern)
static Barconfig * current_bar
#define APPLY_COLORS(classname)
static char * current_mode
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...
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...
struct autostarts_always_head autostarts_always
struct autostarts_head autostarts
struct assignments_head assignments
struct ws_assignments_head ws_assignments
struct bindings_head * bindings
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 outputs_head outputs
int ws_name_to_number(const char *name)
Parses the workspace name as a number.
void workspace_back_and_forth(void)
Focuses the previously focused workspace.
static xcb_cursor_context_t * ctx
@ PARSE_FILE_CONFIG_ERRORS
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
void ipc_set_kill_timeout(ev_tstamp new)
Set the maximum duration that we allow for a connection with an unwriteable socket.
char * sstrdup(const char *str)
Safe-wrapper around strdup which exits if malloc returns NULL (meaning that there is no more memory a...
int logical_px(const int logical)
Convert a logical amount of pixels (e.g.
void * smalloc(size_t size)
Safe-wrapper around malloc 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 * 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...
#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)
List entry struct for an included file.
char * variable_replaced_contents
The configuration file can contain multiple sets of bindings.
enum Config::@6 popup_during_fullscreen
What should happen when a new popup is opened during fullscreen mode.
focus_wrapping_t focus_wrapping
When focus wrapping is enabled (the default), attempting to move focus past the edge of the screen (i...
char * restart_state_path
bool workspace_auto_back_and_forth
Automatic workspace back and forth switching.
int32_t floating_minimum_width
enum Config::@5 title_align
Title alignment options.
hide_edge_borders_mode_t hide_edge_borders
Remove borders if they are adjacent to the screen edge.
int32_t floating_minimum_height
bool disable_focus_follows_mouse
By default, focus follows mouse.
struct Config::config_client client
warping_t mouse_warping
By default, when switching focus to a window on a different output (e.g.
char * fake_outputs
Overwrites output detection (for testing), see src/fake_outputs.c.
int default_floating_border_width
int default_orientation
Default orientation for new containers.
uint32_t floating_modifier
The modifier which needs to be pressed in combination with your mouse buttons to do things with float...
bool show_marks
Specifies whether or not marks should be displayed in the window decoration.
float workspace_urgency_timer
By default, urgency is cleared immediately when switching to another workspace leads to focusing the ...
bool disable_randr15
Don’t use RandR 1.5 for querying outputs.
enum Config::@4 focus_on_window_activation
Behavior when a window sends a NET_ACTIVE_WINDOW message.
int32_t floating_maximum_width
Maximum and minimum dimensions of a floating window.
bool force_xinerama
By default, use the RandR API for multi-monitor setups.
int32_t floating_maximum_height
border_style_t default_border
The default border style for new windows.
border_style_t default_floating_border
The default border style for new floating windows.
Holds the status bar configuration (i3bar).
bool hide_workspace_buttons
Hide workspace buttons? Configuration option is 'workspace_buttons no' but we invert the bool to get ...
char * separator_symbol
A custom separator to use instead of a vertical line.
int workspace_min_width
The minimal width for workspace buttons.
struct Barconfig::bar_colors colors
uint32_t modifier
Bar modifier (to show bar when in hide mode).
char * i3bar_command
Command that should be run to execute i3bar, give a full path if i3bar is not in your $PATH.
int num_outputs
Number of outputs in the outputs array.
enum Barconfig::@8 hidden_state
char * font
Font specification for all text rendered on the bar.
char * id
Automatically generated ID for this bar config.
enum Barconfig::@7 mode
Bar display mode (hide unless modifier is pressed or show in dock mode or always hide in invisible mo...
bool hide_binding_mode_indicator
Hide mode button? Configuration option is 'binding_mode_indicator no' but we invert the bool for the ...
char * status_command
Command that should be run to get a statusline, for example 'i3status'.
bool strip_workspace_numbers
Strip workspace numbers? Configuration option is 'strip_workspace_numbers yes'.
bool strip_workspace_name
Strip workspace name? Configuration option is 'strip_workspace_name yes'.
char ** outputs
Outputs on which this bar should show up on.
enum Barconfig::@9 position
Bar position (bottom by default).
char * socket_path
Path to the i3 IPC socket.
bool verbose
Enable verbose mode? Useful for debugging purposes.
char * focused_background
char * focused_statusline
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.
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.
enum Match::@14 window_mode
An Assignment makes specific windows go to a specific workspace/output or run a command for that wind...
Match match
the criteria to check if a window matches
union Assignment::@17 dest
destination workspace/command/output, depending on the type
enum Assignment::@16 type
type of this assignment: