Merge pull request #1543 from emersion/output-config-by-identifier
Allow to configure outputs by their identifier
This commit is contained in:
commit
bc7011db3c
|
@ -397,6 +397,8 @@ struct seat_attachment_config *seat_config_get_attachment(
|
||||||
void apply_seat_config(struct seat_config *seat);
|
void apply_seat_config(struct seat_config *seat);
|
||||||
|
|
||||||
int output_name_cmp(const void *item, const void *data);
|
int output_name_cmp(const void *item, const void *data);
|
||||||
|
void output_get_identifier(char *identifier, size_t len,
|
||||||
|
struct sway_output *output);
|
||||||
struct output_config *new_output_config(const char *name);
|
struct output_config *new_output_config(const char *name);
|
||||||
void merge_output_config(struct output_config *dst, struct output_config *src);
|
void merge_output_config(struct output_config *dst, struct output_config *src);
|
||||||
void apply_output_config(struct output_config *oc, swayc_t *output);
|
void apply_output_config(struct output_config *oc, swayc_t *output);
|
||||||
|
|
|
@ -231,8 +231,8 @@ static struct cmd_results *cmd_output_background(struct output_config *output,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct cmd_results *cmd_output(int argc, char **argv) {
|
struct cmd_results *cmd_output(int argc, char **argv) {
|
||||||
struct cmd_results *error = NULL;
|
struct cmd_results *error = checkarg(argc, "output", EXPECTED_AT_LEAST, 1);
|
||||||
if ((error = checkarg(argc, "output", EXPECTED_AT_LEAST, 1))) {
|
if (error != NULL) {
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -275,11 +275,11 @@ struct cmd_results *cmd_output(int argc, char **argv) {
|
||||||
|
|
||||||
int i = list_seq_find(config->output_configs, output_name_cmp, output->name);
|
int i = list_seq_find(config->output_configs, output_name_cmp, output->name);
|
||||||
if (i >= 0) {
|
if (i >= 0) {
|
||||||
// merge existing config
|
// Merge existing config
|
||||||
struct output_config *oc = config->output_configs->items[i];
|
struct output_config *current = config->output_configs->items[i];
|
||||||
merge_output_config(oc, output);
|
merge_output_config(current, output);
|
||||||
free_output_config(output);
|
free_output_config(output);
|
||||||
output = oc;
|
output = current;
|
||||||
} else {
|
} else {
|
||||||
list_add(config->output_configs, output);
|
list_add(config->output_configs, output);
|
||||||
}
|
}
|
||||||
|
@ -290,22 +290,26 @@ struct cmd_results *cmd_output(int argc, char **argv) {
|
||||||
output->refresh_rate, output->x, output->y, output->scale,
|
output->refresh_rate, output->x, output->y, output->scale,
|
||||||
output->transform, output->background, output->background_option);
|
output->transform, output->background, output->background_option);
|
||||||
|
|
||||||
if (output->name) {
|
// Try to find the output container and apply configuration now. If
|
||||||
// Try to find the output container and apply configuration now. If
|
// this is during startup then there will be no container and config
|
||||||
// this is during startup then there will be no container and config
|
// will be applied during normal "new output" event from wlroots.
|
||||||
// will be applied during normal "new output" event from wlroots.
|
char identifier[128];
|
||||||
swayc_t *cont = NULL;
|
bool all = strcmp(output->name, "*") == 0;
|
||||||
for (int i = 0; i < root_container.children->length; ++i) {
|
for (int i = 0; i < root_container.children->length; ++i) {
|
||||||
cont = root_container.children->items[i];
|
swayc_t *cont = root_container.children->items[i];
|
||||||
if (cont->name && ((strcmp(cont->name, output->name) == 0) ||
|
if (cont->type != C_OUTPUT) {
|
||||||
(strcmp(output->name, "*") == 0))) {
|
continue;
|
||||||
apply_output_config(output, cont);
|
}
|
||||||
|
|
||||||
if (strcmp(output->name, "*") != 0) {
|
output_get_identifier(identifier, sizeof(identifier), cont->sway_output);
|
||||||
// Stop looking if the output config isn't applicable to all
|
if (all || strcmp(cont->name, output->name) == 0 ||
|
||||||
// outputs
|
strcmp(identifier, output->name) == 0) {
|
||||||
break;
|
apply_output_config(output, cont);
|
||||||
}
|
|
||||||
|
if (!all) {
|
||||||
|
// Stop looking if the output config isn't applicable to all
|
||||||
|
// outputs
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#define _XOPEN_SOURCE 700
|
#define _XOPEN_SOURCE 700
|
||||||
|
#include <stdbool.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <wlr/types/wlr_output.h>
|
#include <wlr/types/wlr_output.h>
|
||||||
|
@ -14,6 +15,13 @@ int output_name_cmp(const void *item, const void *data) {
|
||||||
return strcmp(output->name, name);
|
return strcmp(output->name, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void output_get_identifier(char *identifier, size_t len,
|
||||||
|
struct sway_output *output) {
|
||||||
|
struct wlr_output *wlr_output = output->wlr_output;
|
||||||
|
snprintf(identifier, len, "%s %s %s", wlr_output->make, wlr_output->model,
|
||||||
|
wlr_output->serial);
|
||||||
|
}
|
||||||
|
|
||||||
struct output_config *new_output_config(const char *name) {
|
struct output_config *new_output_config(const char *name) {
|
||||||
struct output_config *oc = calloc(1, sizeof(struct output_config));
|
struct output_config *oc = calloc(1, sizeof(struct output_config));
|
||||||
if (oc == NULL) {
|
if (oc == NULL) {
|
||||||
|
|
|
@ -49,16 +49,49 @@ static swayc_t *new_swayc(enum swayc_types type) {
|
||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void free_swayc(swayc_t *cont) {
|
||||||
|
if (!sway_assert(cont, "free_swayc passed NULL")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wl_signal_emit(&cont->events.destroy, cont);
|
||||||
|
|
||||||
|
if (cont->children) {
|
||||||
|
// remove children until there are no more, free_swayc calls
|
||||||
|
// remove_child, which removes child from this container
|
||||||
|
while (cont->children->length) {
|
||||||
|
free_swayc(cont->children->items[0]);
|
||||||
|
}
|
||||||
|
list_free(cont->children);
|
||||||
|
}
|
||||||
|
if (cont->marks) {
|
||||||
|
list_foreach(cont->marks, free);
|
||||||
|
list_free(cont->marks);
|
||||||
|
}
|
||||||
|
if (cont->parent) {
|
||||||
|
remove_child(cont);
|
||||||
|
}
|
||||||
|
if (cont->name) {
|
||||||
|
free(cont->name);
|
||||||
|
}
|
||||||
|
free(cont);
|
||||||
|
}
|
||||||
|
|
||||||
swayc_t *new_output(struct sway_output *sway_output) {
|
swayc_t *new_output(struct sway_output *sway_output) {
|
||||||
struct wlr_box size;
|
struct wlr_box size;
|
||||||
wlr_output_effective_resolution(sway_output->wlr_output, &size.width,
|
wlr_output_effective_resolution(sway_output->wlr_output, &size.width,
|
||||||
&size.height);
|
&size.height);
|
||||||
|
|
||||||
const char *name = sway_output->wlr_output->name;
|
const char *name = sway_output->wlr_output->name;
|
||||||
|
char identifier[128];
|
||||||
|
output_get_identifier(identifier, sizeof(identifier), sway_output);
|
||||||
|
|
||||||
struct output_config *oc = NULL, *all = NULL;
|
struct output_config *oc = NULL, *all = NULL;
|
||||||
for (int i = 0; i < config->output_configs->length; ++i) {
|
for (int i = 0; i < config->output_configs->length; ++i) {
|
||||||
struct output_config *cur = config->output_configs->items[i];
|
struct output_config *cur = config->output_configs->items[i];
|
||||||
if (strcasecmp(name, cur->name) == 0) {
|
|
||||||
|
if (strcasecmp(name, cur->name) == 0 ||
|
||||||
|
strcasecmp(identifier, cur->name) == 0) {
|
||||||
sway_log(L_DEBUG, "Matched output config for %s", name);
|
sway_log(L_DEBUG, "Matched output config for %s", name);
|
||||||
oc = cur;
|
oc = cur;
|
||||||
}
|
}
|
||||||
|
@ -74,13 +107,18 @@ swayc_t *new_output(struct sway_output *sway_output) {
|
||||||
if (!oc) {
|
if (!oc) {
|
||||||
oc = all;
|
oc = all;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (oc && !oc->enabled) {
|
if (oc && !oc->enabled) {
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
swayc_t *output = new_swayc(C_OUTPUT);
|
swayc_t *output = new_swayc(C_OUTPUT);
|
||||||
output->sway_output = sway_output;
|
output->sway_output = sway_output;
|
||||||
output->name = name ? strdup(name) : NULL;
|
output->name = strdup(name);
|
||||||
|
if (output->name == NULL) {
|
||||||
|
free_swayc(output);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
apply_output_config(oc, output);
|
apply_output_config(oc, output);
|
||||||
|
|
||||||
|
@ -140,34 +178,6 @@ swayc_t *new_view(swayc_t *sibling, struct sway_view *sway_view) {
|
||||||
return swayc;
|
return swayc;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void free_swayc(swayc_t *cont) {
|
|
||||||
if (!sway_assert(cont, "free_swayc passed NULL")) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
wl_signal_emit(&cont->events.destroy, cont);
|
|
||||||
|
|
||||||
if (cont->children) {
|
|
||||||
// remove children until there are no more, free_swayc calls
|
|
||||||
// remove_child, which removes child from this container
|
|
||||||
while (cont->children->length) {
|
|
||||||
free_swayc(cont->children->items[0]);
|
|
||||||
}
|
|
||||||
list_free(cont->children);
|
|
||||||
}
|
|
||||||
if (cont->marks) {
|
|
||||||
list_foreach(cont->marks, free);
|
|
||||||
list_free(cont->marks);
|
|
||||||
}
|
|
||||||
if (cont->parent) {
|
|
||||||
remove_child(cont);
|
|
||||||
}
|
|
||||||
if (cont->name) {
|
|
||||||
free(cont->name);
|
|
||||||
}
|
|
||||||
free(cont);
|
|
||||||
}
|
|
||||||
|
|
||||||
swayc_t *destroy_output(swayc_t *output) {
|
swayc_t *destroy_output(swayc_t *output) {
|
||||||
if (!sway_assert(output, "null output passed to destroy_output")) {
|
if (!sway_assert(output, "null output passed to destroy_output")) {
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
Loading…
Reference in a new issue