swayfx/sway/layer_criteria.c
William McKinnon 82fe097b31
feat: sway 1.10 rebase (#367)
Co-authored-by: Erik Reider <35975961+ErikReider@users.noreply.github.com>
2025-02-16 15:33:52 -05:00

155 lines
4.4 KiB
C

#include <ctype.h>
#include <string.h>
#include "list.h"
#include "log.h"
#include "stringop.h"
#include "sway/commands.h"
#include "sway/layer_criteria.h"
#include "util.h"
static void init_criteria_effects(struct layer_criteria *criteria) {
criteria->corner_radius = 0;
criteria->blur_enabled = false;
criteria->blur_xray = false;
criteria->blur_ignore_transparent = false;
criteria->shadow_enabled = false;
}
static void copy_criteria_effects(struct layer_criteria *dst, struct layer_criteria *src) {
dst->corner_radius = src->corner_radius;
dst->blur_enabled = src->blur_enabled;
dst->blur_xray = src->blur_xray;
dst->blur_ignore_transparent = src->blur_ignore_transparent;
dst->shadow_enabled = src->shadow_enabled;
}
static bool layer_criteria_find(char *namespace,
struct layer_criteria **criteria, int *index) {
*index = -1;
*criteria = NULL;
list_t *criterias = config->layer_criteria;
for (int i = 0; i < criterias->length; ++i) {
struct layer_criteria *existing = criterias->items[i];
if (strcmp(namespace, existing->namespace) == 0) {
*index = i;
*criteria = existing;
return true;
}
}
return false;
}
static bool layer_criteria_parse(struct layer_criteria *criteria) {
char matched_delim = ';';
char *head = malloc(strlen(criteria->cmdlist) + 1);
strcpy(head, criteria->cmdlist);
do {
// Trim leading whitespaces
for (; isspace(*head); ++head) {}
// Split command list
char *cmd = argsep(&head, ";,", &matched_delim);
for (; isspace(*cmd); ++cmd) {}
if (strcmp(cmd, "") == 0) {
sway_log(SWAY_INFO, "Ignoring empty layer effect.");
continue;
}
sway_log(SWAY_INFO, "Handling layer effect '%s'", cmd);
int argc;
char **argv = split_args(cmd, &argc);
// Strip all quotes from each token
for (int i = 1; i < argc; ++i) {
if (*argv[i] == '\"' || *argv[i] == '\'') {
strip_quotes(argv[i]);
}
}
if (strcmp(argv[0], "reset") == 0) {
// Reset all args
init_criteria_effects(criteria);
return true;
} else if (strcmp(argv[0], "blur") == 0) {
criteria->blur_enabled = parse_boolean(argv[1], true);
continue;
} if (strcmp(argv[0], "blur_xray") == 0) {
criteria->blur_xray = parse_boolean(argv[1], true);
continue;
} if (strcmp(argv[0], "blur_ignore_transparent") == 0) {
criteria->blur_ignore_transparent = parse_boolean(argv[1], true);
continue;
} else if (strcmp(argv[0], "shadows") == 0) {
criteria->shadow_enabled = parse_boolean(argv[1], true);
continue;
} else if (strcmp(argv[0], "corner_radius") == 0) {
int value;
if (cmd_corner_radius_parse_value(argv[1], &value)) {
criteria->corner_radius = value;
continue;
}
sway_log(SWAY_ERROR,
"Invalid layer_effects corner_radius size! Got \"%s\"",
argv[1]);
return false;
} else {
sway_log(SWAY_ERROR, "Invalid layer_effects effect! Got \"%s\"", cmd);
return false;
}
} while(head);
return true;
}
struct layer_criteria *layer_criteria_add(char *namespace, char *cmdlist) {
struct layer_criteria *criteria = calloc(1, sizeof(*criteria));
init_criteria_effects(criteria);
// Check if the rule already exists
int existing_index;
struct layer_criteria *existing = NULL;
if (layer_criteria_find(namespace, &existing, &existing_index)) {
sway_log(SWAY_DEBUG, "layer_effect '%s' already exists: '%s'. Replacing with: '%s'",
namespace, existing->cmdlist, cmdlist);
// Clone the effects into the new criteria
copy_criteria_effects(criteria, existing);
layer_criteria_destroy(existing);
list_del(config->layer_criteria, existing_index);
}
// Create a new criteria
if (!criteria) {
sway_log(SWAY_ERROR, "Could not allocate new layer_criteria");
return NULL;
}
criteria->namespace = malloc(strlen(namespace) + 1);
strcpy(criteria->namespace, namespace);
criteria->cmdlist = cmdlist;
// Parsing
if (!layer_criteria_parse(criteria)) {
// Failed to parse
layer_criteria_destroy(criteria);
return NULL;
}
list_add(config->layer_criteria, criteria);
return criteria;
}
struct layer_criteria *layer_criteria_for_namespace(char *namespace) {
list_t *criterias = config->layer_criteria;
for (int i = criterias->length - 1; i >= 0; i--) {
struct layer_criteria *criteria = criterias->items[i];
if (strcmp(criteria->namespace, namespace) == 0) {
return criteria;
}
}
return NULL;
}
void layer_criteria_destroy(struct layer_criteria *criteria) {
free(criteria->namespace);
free(criteria->cmdlist);
free(criteria);
}