2a684cad5f
Patch tested by compiling with `__attribute__ ((format (printf, 2, 3)))` applied to `cmd_results_new`. String usage constants have been converted from pointers to arrays when encountered. General handler format strings were sometimes modified to include the old input string, especially for unknown command errors.
225 lines
6.4 KiB
C
225 lines
6.4 KiB
C
#include <string.h>
|
|
#include <strings.h>
|
|
#include "sway/commands.h"
|
|
#include "sway/config.h"
|
|
#include "sway/tree/arrange.h"
|
|
#include "sway/tree/workspace.h"
|
|
#include "log.h"
|
|
#include "stringop.h"
|
|
#include <math.h>
|
|
|
|
enum gaps_op {
|
|
GAPS_OP_SET,
|
|
GAPS_OP_ADD,
|
|
GAPS_OP_SUBTRACT
|
|
};
|
|
|
|
struct gaps_data {
|
|
bool inner;
|
|
struct {
|
|
bool top;
|
|
bool right;
|
|
bool bottom;
|
|
bool left;
|
|
} outer;
|
|
enum gaps_op operation;
|
|
int amount;
|
|
};
|
|
|
|
// Prevent negative outer gaps from moving windows out of the workspace.
|
|
static void prevent_invalid_outer_gaps(void) {
|
|
if (config->gaps_outer.top < -config->gaps_inner) {
|
|
config->gaps_outer.top = -config->gaps_inner;
|
|
}
|
|
if (config->gaps_outer.right < -config->gaps_inner) {
|
|
config->gaps_outer.right = -config->gaps_inner;
|
|
}
|
|
if (config->gaps_outer.bottom < -config->gaps_inner) {
|
|
config->gaps_outer.bottom = -config->gaps_inner;
|
|
}
|
|
if (config->gaps_outer.left < -config->gaps_inner) {
|
|
config->gaps_outer.left = -config->gaps_inner;
|
|
}
|
|
}
|
|
|
|
// gaps inner|outer|horizontal|vertical|top|right|bottom|left <px>
|
|
static const char expected_defaults[] =
|
|
"'gaps inner|outer|horizontal|vertical|top|right|bottom|left <px>'";
|
|
static struct cmd_results *gaps_set_defaults(int argc, char **argv) {
|
|
struct cmd_results *error = checkarg(argc, "gaps", EXPECTED_EQUAL_TO, 2);
|
|
if (error) {
|
|
return error;
|
|
}
|
|
|
|
char *end;
|
|
int amount = strtol(argv[1], &end, 10);
|
|
if (strlen(end) && strcasecmp(end, "px") != 0) {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_defaults);
|
|
}
|
|
|
|
bool valid = false;
|
|
if (!strcasecmp(argv[0], "inner")) {
|
|
valid = true;
|
|
config->gaps_inner = (amount >= 0) ? amount : 0;
|
|
} else {
|
|
if (!strcasecmp(argv[0], "outer") || !strcasecmp(argv[0], "vertical")
|
|
|| !strcasecmp(argv[0], "top")) {
|
|
valid = true;
|
|
config->gaps_outer.top = amount;
|
|
}
|
|
if (!strcasecmp(argv[0], "outer") || !strcasecmp(argv[0], "horizontal")
|
|
|| !strcasecmp(argv[0], "right")) {
|
|
valid = true;
|
|
config->gaps_outer.right = amount;
|
|
}
|
|
if (!strcasecmp(argv[0], "outer") || !strcasecmp(argv[0], "vertical")
|
|
|| !strcasecmp(argv[0], "bottom")) {
|
|
valid = true;
|
|
config->gaps_outer.bottom = amount;
|
|
}
|
|
if (!strcasecmp(argv[0], "outer") || !strcasecmp(argv[0], "horizontal")
|
|
|| !strcasecmp(argv[0], "left")) {
|
|
valid = true;
|
|
config->gaps_outer.left = amount;
|
|
}
|
|
}
|
|
if (!valid) {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_defaults);
|
|
}
|
|
|
|
prevent_invalid_outer_gaps();
|
|
return cmd_results_new(CMD_SUCCESS, NULL);
|
|
}
|
|
|
|
static void apply_gaps_op(int *prop, enum gaps_op op, int amount) {
|
|
switch (op) {
|
|
case GAPS_OP_SET:
|
|
*prop = amount;
|
|
break;
|
|
case GAPS_OP_ADD:
|
|
*prop += amount;
|
|
break;
|
|
case GAPS_OP_SUBTRACT:
|
|
*prop -= amount;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void configure_gaps(struct sway_workspace *ws, void *_data) {
|
|
// Apply operation to gaps
|
|
struct gaps_data *data = _data;
|
|
if (data->inner) {
|
|
apply_gaps_op(&ws->gaps_inner, data->operation, data->amount);
|
|
}
|
|
if (data->outer.top) {
|
|
apply_gaps_op(&(ws->gaps_outer.top), data->operation, data->amount);
|
|
}
|
|
if (data->outer.right) {
|
|
apply_gaps_op(&(ws->gaps_outer.right), data->operation, data->amount);
|
|
}
|
|
if (data->outer.bottom) {
|
|
apply_gaps_op(&(ws->gaps_outer.bottom), data->operation, data->amount);
|
|
}
|
|
if (data->outer.left) {
|
|
apply_gaps_op(&(ws->gaps_outer.left), data->operation, data->amount);
|
|
}
|
|
|
|
// Prevent invalid gaps configurations.
|
|
if (ws->gaps_inner < 0) {
|
|
ws->gaps_inner = 0;
|
|
}
|
|
prevent_invalid_outer_gaps();
|
|
arrange_workspace(ws);
|
|
}
|
|
|
|
// gaps inner|outer|horizontal|vertical|top|right|bottom|left current|all
|
|
// set|plus|minus <px>
|
|
static const char expected_runtime[] = "'gaps inner|outer|horizontal|vertical|"
|
|
"top|right|bottom|left current|all set|plus|minus <px>'";
|
|
static struct cmd_results *gaps_set_runtime(int argc, char **argv) {
|
|
struct cmd_results *error = checkarg(argc, "gaps", EXPECTED_EQUAL_TO, 4);
|
|
if (error) {
|
|
return error;
|
|
}
|
|
if (!root->outputs->length) {
|
|
return cmd_results_new(CMD_INVALID,
|
|
"Can't run this command while there's no outputs connected.");
|
|
}
|
|
|
|
struct gaps_data data = {0};
|
|
|
|
if (strcasecmp(argv[0], "inner") == 0) {
|
|
data.inner = true;
|
|
} else {
|
|
data.outer.top = !strcasecmp(argv[0], "outer") ||
|
|
!strcasecmp(argv[0], "vertical") || !strcasecmp(argv[0], "top");
|
|
data.outer.right = !strcasecmp(argv[0], "outer") ||
|
|
!strcasecmp(argv[0], "horizontal") || !strcasecmp(argv[0], "right");
|
|
data.outer.bottom = !strcasecmp(argv[0], "outer") ||
|
|
!strcasecmp(argv[0], "vertical") || !strcasecmp(argv[0], "bottom");
|
|
data.outer.left = !strcasecmp(argv[0], "outer") ||
|
|
!strcasecmp(argv[0], "horizontal") || !strcasecmp(argv[0], "left");
|
|
}
|
|
if (!data.inner && !data.outer.top && !data.outer.right &&
|
|
!data.outer.bottom && !data.outer.left) {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_runtime);
|
|
}
|
|
|
|
bool all;
|
|
if (strcasecmp(argv[1], "current") == 0) {
|
|
all = false;
|
|
} else if (strcasecmp(argv[1], "all") == 0) {
|
|
all = true;
|
|
} else {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_runtime);
|
|
}
|
|
|
|
if (strcasecmp(argv[2], "set") == 0) {
|
|
data.operation = GAPS_OP_SET;
|
|
} else if (strcasecmp(argv[2], "plus") == 0) {
|
|
data.operation = GAPS_OP_ADD;
|
|
} else if (strcasecmp(argv[2], "minus") == 0) {
|
|
data.operation = GAPS_OP_SUBTRACT;
|
|
} else {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_runtime);
|
|
}
|
|
|
|
char *end;
|
|
data.amount = strtol(argv[3], &end, 10);
|
|
if (strlen(end) && strcasecmp(end, "px") != 0) {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_runtime);
|
|
}
|
|
|
|
if (all) {
|
|
root_for_each_workspace(configure_gaps, &data);
|
|
} else {
|
|
configure_gaps(config->handler_context.workspace, &data);
|
|
}
|
|
|
|
return cmd_results_new(CMD_SUCCESS, NULL);
|
|
}
|
|
|
|
// gaps inner|outer|<dir>|<side> <px> - sets defaults for workspaces
|
|
// gaps inner|outer|<dir>|<side> current|all set|plus|minus <px> - runtime only
|
|
// <dir> = horizontal|vertical
|
|
// <side> = top|right|bottom|left
|
|
struct cmd_results *cmd_gaps(int argc, char **argv) {
|
|
struct cmd_results *error = checkarg(argc, "gaps", EXPECTED_AT_LEAST, 2);
|
|
if (error) {
|
|
return error;
|
|
}
|
|
|
|
bool config_loading = !config->active || config->reloading;
|
|
|
|
if (argc == 2) {
|
|
return gaps_set_defaults(argc, argv);
|
|
}
|
|
if (argc == 4 && !config_loading) {
|
|
return gaps_set_runtime(argc, argv);
|
|
}
|
|
if (config_loading) {
|
|
return cmd_results_new(CMD_INVALID, "Expected %s", expected_defaults);
|
|
}
|
|
return cmd_results_new(CMD_INVALID, "Expected %s or %s",
|
|
expected_runtime, expected_defaults);
|
|
}
|