mirror of
https://github.com/swaywm/sway.git
synced 2024-12-28 07:56:31 +01:00
swaynag: split config into own file and fix optind
This commit is contained in:
parent
58f3fa74ae
commit
6124d0f9a2
6 changed files with 315 additions and 289 deletions
13
include/swaynag/config.h
Normal file
13
include/swaynag/config.h
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
#ifndef _SWAY_NAGBAR_CONFIG_H
|
||||||
|
#define _SWAY_NAGBAR_CONFIG_H
|
||||||
|
#include "swaynag/nagbar.h"
|
||||||
|
#include "list.h"
|
||||||
|
|
||||||
|
int nagbar_parse_options(int argc, char **argv, struct sway_nagbar *nagbar,
|
||||||
|
list_t *types, char **config, bool *debug);
|
||||||
|
|
||||||
|
char *nagbar_get_config_path(void);
|
||||||
|
|
||||||
|
int nagbar_load_config(char *path, struct sway_nagbar *nagbar, list_t *types);
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,6 +1,7 @@
|
||||||
#ifndef _SWAY_NAGBAR_NAGBAR_H
|
#ifndef _SWAY_NAGBAR_NAGBAR_H
|
||||||
#define _SWAY_NAGBAR_NAGBAR_H
|
#define _SWAY_NAGBAR_NAGBAR_H
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <strings.h>
|
||||||
#include "list.h"
|
#include "list.h"
|
||||||
#include "pool-buffer.h"
|
#include "pool-buffer.h"
|
||||||
#include "swaynag/types.h"
|
#include "swaynag/types.h"
|
||||||
|
|
292
swaynag/config.c
Normal file
292
swaynag/config.c
Normal file
|
@ -0,0 +1,292 @@
|
||||||
|
#define _XOPEN_SOURCE 700
|
||||||
|
#define _POSIX_C_SOURCE 200112L
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <wordexp.h>
|
||||||
|
#include "log.h"
|
||||||
|
#include "list.h"
|
||||||
|
#include "readline.h"
|
||||||
|
#include "swaynag/nagbar.h"
|
||||||
|
#include "swaynag/types.h"
|
||||||
|
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
||||||
|
|
||||||
|
static char *read_from_stdin() {
|
||||||
|
char *buffer = NULL;
|
||||||
|
while (!feof(stdin)) {
|
||||||
|
char *line = read_line(stdin);
|
||||||
|
if (!line) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!buffer) {
|
||||||
|
buffer = strdup(line);
|
||||||
|
} else {
|
||||||
|
buffer = realloc(buffer, strlen(buffer) + strlen(line) + 2);
|
||||||
|
strcat(buffer, line);
|
||||||
|
strcat(buffer, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buffer && buffer[strlen(buffer) - 1] == '\n') {
|
||||||
|
buffer[strlen(buffer) - 1] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
int nagbar_parse_options(int argc, char **argv, struct sway_nagbar *nagbar,
|
||||||
|
list_t *types, char **config, bool *debug) {
|
||||||
|
static struct option opts[] = {
|
||||||
|
{"button", required_argument, NULL, 'b'},
|
||||||
|
{"config", required_argument, NULL, 'c'},
|
||||||
|
{"debug", no_argument, NULL, 'd'},
|
||||||
|
{"edge", required_argument, NULL, 'e'},
|
||||||
|
{"font", required_argument, NULL, 'f'},
|
||||||
|
{"help", no_argument, NULL, 'h'},
|
||||||
|
{"detailed-message", no_argument, NULL, 'l'},
|
||||||
|
{"detailed-button", required_argument, NULL, 'L'},
|
||||||
|
{"message", required_argument, NULL, 'm'},
|
||||||
|
{"output", required_argument, NULL, 'o'},
|
||||||
|
{"dismiss-button", required_argument, NULL, 's'},
|
||||||
|
{"type", required_argument, NULL, 't'},
|
||||||
|
{"version", no_argument, NULL, 'v'},
|
||||||
|
{0, 0, 0, 0}
|
||||||
|
};
|
||||||
|
|
||||||
|
const char *usage =
|
||||||
|
"Usage: swaynag [options...]\n"
|
||||||
|
"\n"
|
||||||
|
" -b, --button <text> <action> Create a button with text that "
|
||||||
|
"executes action when pressed. Multiple buttons can be defined.\n"
|
||||||
|
" -c, --config <path> Path to config file.\n"
|
||||||
|
" -d, --debug Enable debugging.\n"
|
||||||
|
" -e, --edge top|bottom Set the edge to use.\n"
|
||||||
|
" -f, --font <font> Set the font to use.\n"
|
||||||
|
" -h, --help Show help message and quit.\n"
|
||||||
|
" -l, --detailed-message Read a detailed message from stdin.\n"
|
||||||
|
" -L, --detailed-button <text> Set the text of the detail button.\n"
|
||||||
|
" -m, --message <msg> Set the message text.\n"
|
||||||
|
" -o, --output <output> Set the output to use.\n"
|
||||||
|
" -s, --dismiss-button <text> Set the dismiss button text.\n"
|
||||||
|
" -t, --type <type> Set the message type.\n"
|
||||||
|
" -v, --version Show the version number and quit.\n";
|
||||||
|
|
||||||
|
optind = 1;
|
||||||
|
while (1) {
|
||||||
|
int c = getopt_long(argc, argv, "b:c:de:f:hlL:m:o:s:t:v", opts, NULL);
|
||||||
|
if (c == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
switch (c) {
|
||||||
|
case 'b': // Button
|
||||||
|
if (nagbar) {
|
||||||
|
if (optind >= argc) {
|
||||||
|
fprintf(stderr, "Missing action for button %s\n", optarg);
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
struct sway_nagbar_button *button;
|
||||||
|
button = calloc(sizeof(struct sway_nagbar_button), 1);
|
||||||
|
button->text = strdup(optarg);
|
||||||
|
button->type = NAGBAR_ACTION_COMMAND;
|
||||||
|
button->action = strdup(argv[optind]);
|
||||||
|
list_add(nagbar->buttons, button);
|
||||||
|
}
|
||||||
|
optind++;
|
||||||
|
break;
|
||||||
|
case 'c': // Config
|
||||||
|
if (config) {
|
||||||
|
*config = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'd': // Debug
|
||||||
|
if (debug) {
|
||||||
|
*debug = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'e': // Edge
|
||||||
|
if (nagbar) {
|
||||||
|
if (strcmp(optarg, "top") == 0) {
|
||||||
|
nagbar->anchors = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP
|
||||||
|
| ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT
|
||||||
|
| ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
||||||
|
} else if (strcmp(optarg, "bottom") == 0) {
|
||||||
|
nagbar->anchors = ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM
|
||||||
|
| ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT
|
||||||
|
| ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "Invalid edge: %s\n", optarg);
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'f': // Font
|
||||||
|
if (nagbar) {
|
||||||
|
free(nagbar->font);
|
||||||
|
nagbar->font = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'l': // Detailed Message
|
||||||
|
if (nagbar) {
|
||||||
|
free(nagbar->details.message);
|
||||||
|
nagbar->details.message = read_from_stdin();
|
||||||
|
nagbar->details.button_up.text = strdup("▲");
|
||||||
|
nagbar->details.button_down.text = strdup("▼");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'L': // Detailed Button Text
|
||||||
|
if (nagbar) {
|
||||||
|
free(nagbar->details.button_details.text);
|
||||||
|
nagbar->details.button_details.text = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'm': // Message
|
||||||
|
if (nagbar) {
|
||||||
|
free(nagbar->message);
|
||||||
|
nagbar->message = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'o': // Output
|
||||||
|
if (nagbar) {
|
||||||
|
free(nagbar->output.name);
|
||||||
|
nagbar->output.name = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 's': // Dismiss Button Text
|
||||||
|
if (nagbar) {
|
||||||
|
struct sway_nagbar_button *button_close;
|
||||||
|
button_close = nagbar->buttons->items[0];
|
||||||
|
free(button_close->text);
|
||||||
|
button_close->text = strdup(optarg);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 't': // Type
|
||||||
|
if (nagbar) {
|
||||||
|
nagbar->type = nagbar_type_get(types, optarg);
|
||||||
|
if (!nagbar->type) {
|
||||||
|
fprintf(stderr, "Unknown type %s\n", optarg);
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'v': // Version
|
||||||
|
fprintf(stdout, "swaynag version " SWAY_VERSION "\n");
|
||||||
|
return -1;
|
||||||
|
default: // Help or unknown flag
|
||||||
|
fprintf(c == 'h' ? stdout : stderr, "%s", usage);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool file_exists(const char *path) {
|
||||||
|
return path && access(path, R_OK) != -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *nagbar_get_config_path(void) {
|
||||||
|
static const char *config_paths[] = {
|
||||||
|
"$HOME/.swaynag/config",
|
||||||
|
"$XDG_CONFIG_HOME/swaynag/config",
|
||||||
|
SYSCONFDIR "/swaynag/config",
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!getenv("XDG_CONFIG_HOME")) {
|
||||||
|
char *home = getenv("HOME");
|
||||||
|
char *config_home = malloc(strlen(home) + strlen("/.config") + 1);
|
||||||
|
if (!config_home) {
|
||||||
|
wlr_log(WLR_ERROR, "Unable to allocate $HOME/.config");
|
||||||
|
} else {
|
||||||
|
strcpy(config_home, home);
|
||||||
|
strcat(config_home, "/.config");
|
||||||
|
setenv("XDG_CONFIG_HOME", config_home, 1);
|
||||||
|
wlr_log(WLR_DEBUG, "Set XDG_CONFIG_HOME to %s", config_home);
|
||||||
|
free(config_home);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wordexp_t p;
|
||||||
|
char *path;
|
||||||
|
for (size_t i = 0; i < sizeof(config_paths) / sizeof(char *); ++i) {
|
||||||
|
if (wordexp(config_paths[i], &p, 0) == 0) {
|
||||||
|
path = strdup(p.we_wordv[0]);
|
||||||
|
wordfree(&p);
|
||||||
|
if (file_exists(path)) {
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
free(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int nagbar_load_config(char *path, struct sway_nagbar *nagbar, list_t *types) {
|
||||||
|
FILE *config = fopen(path, "r");
|
||||||
|
if (!config) {
|
||||||
|
fprintf(stderr, "Failed to read config. Running without it.\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
struct sway_nagbar_type *type = NULL;
|
||||||
|
char *line;
|
||||||
|
int line_number = 0;
|
||||||
|
while (!feof(config)) {
|
||||||
|
line = read_line(config);
|
||||||
|
if (!line) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
line_number++;
|
||||||
|
if (line[0] == '#') {
|
||||||
|
free(line);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (strlen(line) == 0) {
|
||||||
|
free(line);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (line[0] == '[') {
|
||||||
|
char *close = strchr(line, ']');
|
||||||
|
if (!close) {
|
||||||
|
free(line);
|
||||||
|
fclose(config);
|
||||||
|
fprintf(stderr, "Closing bracket not found on line %d\n",
|
||||||
|
line_number);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
char *name = calloc(1, close - line);
|
||||||
|
strncat(name, line + 1, close - line - 1);
|
||||||
|
type = nagbar_type_get(types, name);
|
||||||
|
if (!type) {
|
||||||
|
type = calloc(1, sizeof(struct sway_nagbar_type));
|
||||||
|
type->name = strdup(name);
|
||||||
|
list_add(types, type);
|
||||||
|
}
|
||||||
|
free(name);
|
||||||
|
} else {
|
||||||
|
char flag[strlen(line) + 3];
|
||||||
|
sprintf(flag, "--%s", line);
|
||||||
|
char *argv[] = {"swaynag", flag};
|
||||||
|
int result;
|
||||||
|
if (type) {
|
||||||
|
result = nagbar_parse_type(2, argv, type);
|
||||||
|
} else {
|
||||||
|
result = nagbar_parse_options(2, argv, nagbar, types,
|
||||||
|
NULL, NULL);
|
||||||
|
}
|
||||||
|
if (result != 0) {
|
||||||
|
free(line);
|
||||||
|
fclose(config);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
}
|
||||||
|
fclose(config);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
296
swaynag/main.c
296
swaynag/main.c
|
@ -1,12 +1,8 @@
|
||||||
#define _XOPEN_SOURCE 700
|
#define _XOPEN_SOURCE 500
|
||||||
#define _POSIX_C_SOURCE 200112L
|
|
||||||
#include <getopt.h>
|
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <stdlib.h>
|
|
||||||
#include <wordexp.h>
|
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "list.h"
|
#include "list.h"
|
||||||
#include "readline.h"
|
#include "swaynag/config.h"
|
||||||
#include "swaynag/nagbar.h"
|
#include "swaynag/nagbar.h"
|
||||||
#include "swaynag/types.h"
|
#include "swaynag/types.h"
|
||||||
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
||||||
|
@ -23,285 +19,6 @@ void sway_terminate(int code) {
|
||||||
exit(code);
|
exit(code);
|
||||||
}
|
}
|
||||||
|
|
||||||
static char *read_from_stdin() {
|
|
||||||
char *buffer = NULL;
|
|
||||||
while (!feof(stdin)) {
|
|
||||||
char *line = read_line(stdin);
|
|
||||||
if (!line) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!buffer) {
|
|
||||||
buffer = strdup(line);
|
|
||||||
} else {
|
|
||||||
buffer = realloc(buffer, strlen(buffer) + strlen(line) + 2);
|
|
||||||
strcat(buffer, line);
|
|
||||||
strcat(buffer, "\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
free(line);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buffer && buffer[strlen(buffer) - 1] == '\n') {
|
|
||||||
buffer[strlen(buffer) - 1] = '\0';
|
|
||||||
}
|
|
||||||
|
|
||||||
return buffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int parse_options(int argc, char **argv, struct sway_nagbar *nagbar,
|
|
||||||
list_t *types, char **config, bool *debug) {
|
|
||||||
static struct option opts[] = {
|
|
||||||
{"button", required_argument, NULL, 'b'},
|
|
||||||
{"config", required_argument, NULL, 'c'},
|
|
||||||
{"debug", no_argument, NULL, 'd'},
|
|
||||||
{"edge", required_argument, NULL, 'e'},
|
|
||||||
{"font", required_argument, NULL, 'f'},
|
|
||||||
{"help", no_argument, NULL, 'h'},
|
|
||||||
{"detailed-message", no_argument, NULL, 'l'},
|
|
||||||
{"detailed-button", required_argument, NULL, 'L'},
|
|
||||||
{"message", required_argument, NULL, 'm'},
|
|
||||||
{"output", required_argument, NULL, 'o'},
|
|
||||||
{"dismiss-button", required_argument, NULL, 's'},
|
|
||||||
{"type", required_argument, NULL, 't'},
|
|
||||||
{"version", no_argument, NULL, 'v'},
|
|
||||||
{0, 0, 0, 0}
|
|
||||||
};
|
|
||||||
|
|
||||||
const char *usage =
|
|
||||||
"Usage: swaynag [options...]\n"
|
|
||||||
"\n"
|
|
||||||
" -b, --button <text> <action> Create a button with text that "
|
|
||||||
"executes action when pressed. Multiple buttons can be defined.\n"
|
|
||||||
" -c, --config <path> Path to config file.\n"
|
|
||||||
" -d, --debug Enable debugging.\n"
|
|
||||||
" -e, --edge top|bottom Set the edge to use.\n"
|
|
||||||
" -f, --font <font> Set the font to use.\n"
|
|
||||||
" -h, --help Show help message and quit.\n"
|
|
||||||
" -l, --detailed-message Read a detailed message from stdin.\n"
|
|
||||||
" -L, --detailed-button <text> Set the text of the detail button.\n"
|
|
||||||
" -m, --message <msg> Set the message text.\n"
|
|
||||||
" -o, --output <output> Set the output to use.\n"
|
|
||||||
" -s, --dismiss-button <text> Set the dismiss button text.\n"
|
|
||||||
" -t, --type <type> Set the message type.\n"
|
|
||||||
" -v, --version Show the version number and quit.\n";
|
|
||||||
|
|
||||||
optind = 1;
|
|
||||||
while (1) {
|
|
||||||
int c = getopt_long(argc, argv, "b:c:de:f:hlL:m:o:s:t:v", opts, NULL);
|
|
||||||
if (c == -1) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
switch (c) {
|
|
||||||
case 'b': // Button
|
|
||||||
if (nagbar) {
|
|
||||||
if (optind >= argc) {
|
|
||||||
fprintf(stderr, "Missing action for button %s\n", optarg);
|
|
||||||
return EXIT_FAILURE;
|
|
||||||
}
|
|
||||||
struct sway_nagbar_button *button;
|
|
||||||
button = calloc(sizeof(struct sway_nagbar_button), 1);
|
|
||||||
button->text = strdup(optarg);
|
|
||||||
button->type = NAGBAR_ACTION_COMMAND;
|
|
||||||
button->action = strdup(argv[optind]);
|
|
||||||
optind++;
|
|
||||||
list_add(nagbar->buttons, button);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'c': // Config
|
|
||||||
if (config) {
|
|
||||||
*config = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'd': // Debug
|
|
||||||
if (debug) {
|
|
||||||
*debug = true;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'e': // Edge
|
|
||||||
if (nagbar) {
|
|
||||||
if (strcmp(optarg, "top") == 0) {
|
|
||||||
nagbar->anchors = ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP
|
|
||||||
| ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT
|
|
||||||
| ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
|
||||||
} else if (strcmp(optarg, "bottom") == 0) {
|
|
||||||
nagbar->anchors = ZWLR_LAYER_SURFACE_V1_ANCHOR_BOTTOM
|
|
||||||
| ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT
|
|
||||||
| ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT;
|
|
||||||
} else {
|
|
||||||
fprintf(stderr, "Invalid edge: %s\n", optarg);
|
|
||||||
return EXIT_FAILURE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'f': // Font
|
|
||||||
if (nagbar) {
|
|
||||||
free(nagbar->font);
|
|
||||||
nagbar->font = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'l': // Detailed Message
|
|
||||||
if (nagbar) {
|
|
||||||
free(nagbar->details.message);
|
|
||||||
nagbar->details.message = read_from_stdin();
|
|
||||||
nagbar->details.button_up.text = strdup("▲");
|
|
||||||
nagbar->details.button_down.text = strdup("▼");
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'L': // Detailed Button Text
|
|
||||||
if (nagbar) {
|
|
||||||
free(nagbar->details.button_details.text);
|
|
||||||
nagbar->details.button_details.text = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'm': // Message
|
|
||||||
if (nagbar) {
|
|
||||||
free(nagbar->message);
|
|
||||||
nagbar->message = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'o': // Output
|
|
||||||
if (nagbar) {
|
|
||||||
free(nagbar->output.name);
|
|
||||||
nagbar->output.name = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 's': // Dismiss Button Text
|
|
||||||
if (nagbar) {
|
|
||||||
struct sway_nagbar_button *button_close;
|
|
||||||
button_close = nagbar->buttons->items[0];
|
|
||||||
free(button_close->text);
|
|
||||||
button_close->text = strdup(optarg);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 't': // Type
|
|
||||||
if (nagbar) {
|
|
||||||
nagbar->type = nagbar_type_get(types, optarg);
|
|
||||||
if (!nagbar->type) {
|
|
||||||
fprintf(stderr, "Unknown type %s\n", optarg);
|
|
||||||
return EXIT_FAILURE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case 'v': // Version
|
|
||||||
fprintf(stdout, "swaynag version " SWAY_VERSION "\n");
|
|
||||||
return -1;
|
|
||||||
default: // Help or unknown flag
|
|
||||||
fprintf(c == 'h' ? stdout : stderr, "%s", usage);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool file_exists(const char *path) {
|
|
||||||
return path && access(path, R_OK) != -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static char *get_config_path(void) {
|
|
||||||
static const char *config_paths[] = {
|
|
||||||
"$HOME/.swaynag/config",
|
|
||||||
"$XDG_CONFIG_HOME/swaynag/config",
|
|
||||||
SYSCONFDIR "/swaynag/config",
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!getenv("XDG_CONFIG_HOME")) {
|
|
||||||
char *home = getenv("HOME");
|
|
||||||
char *config_home = malloc(strlen(home) + strlen("/.config") + 1);
|
|
||||||
if (!config_home) {
|
|
||||||
wlr_log(WLR_ERROR, "Unable to allocate $HOME/.config");
|
|
||||||
} else {
|
|
||||||
strcpy(config_home, home);
|
|
||||||
strcat(config_home, "/.config");
|
|
||||||
setenv("XDG_CONFIG_HOME", config_home, 1);
|
|
||||||
wlr_log(WLR_DEBUG, "Set XDG_CONFIG_HOME to %s", config_home);
|
|
||||||
free(config_home);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
wordexp_t p;
|
|
||||||
char *path;
|
|
||||||
for (size_t i = 0; i < sizeof(config_paths) / sizeof(char *); ++i) {
|
|
||||||
if (wordexp(config_paths[i], &p, 0) == 0) {
|
|
||||||
path = strdup(p.we_wordv[0]);
|
|
||||||
wordfree(&p);
|
|
||||||
if (file_exists(path)) {
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
free(path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int load_config(char *path, struct sway_nagbar *nagbar, list_t *types) {
|
|
||||||
FILE *config = fopen(path, "r");
|
|
||||||
if (!config) {
|
|
||||||
fprintf(stderr, "Failed to read config. Running without it.\n");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
struct sway_nagbar_type *type = NULL;
|
|
||||||
char *line;
|
|
||||||
int line_number = 0;
|
|
||||||
while (!feof(config)) {
|
|
||||||
line = read_line(config);
|
|
||||||
if (!line) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
line_number++;
|
|
||||||
if (line[0] == '#') {
|
|
||||||
free(line);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (strlen(line) == 0) {
|
|
||||||
free(line);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (line[0] == '[') {
|
|
||||||
char *close = strchr(line, ']');
|
|
||||||
if (!close) {
|
|
||||||
free(line);
|
|
||||||
fclose(config);
|
|
||||||
fprintf(stderr, "Closing bracket not found on line %d\n",
|
|
||||||
line_number);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
char *name = calloc(1, close - line);
|
|
||||||
strncat(name, line + 1, close - line - 1);
|
|
||||||
type = nagbar_type_get(types, name);
|
|
||||||
if (!type) {
|
|
||||||
type = calloc(1, sizeof(struct sway_nagbar_type));
|
|
||||||
type->name = strdup(name);
|
|
||||||
list_add(types, type);
|
|
||||||
}
|
|
||||||
free(name);
|
|
||||||
} else {
|
|
||||||
char flag[strlen(line) + 3];
|
|
||||||
sprintf(flag, "--%s", line);
|
|
||||||
char *argv[] = {"swaynag", flag};
|
|
||||||
int result;
|
|
||||||
if (type) {
|
|
||||||
result = nagbar_parse_type(2, argv, type);
|
|
||||||
} else {
|
|
||||||
result = parse_options(2, argv, nagbar, types, NULL, NULL);
|
|
||||||
}
|
|
||||||
if (result != 0) {
|
|
||||||
free(line);
|
|
||||||
fclose(config);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
free(line);
|
|
||||||
}
|
|
||||||
fclose(config);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
int exit_code = EXIT_SUCCESS;
|
int exit_code = EXIT_SUCCESS;
|
||||||
|
|
||||||
|
@ -327,7 +44,7 @@ int main(int argc, char **argv) {
|
||||||
|
|
||||||
char *config_path = NULL;
|
char *config_path = NULL;
|
||||||
bool debug = false;
|
bool debug = false;
|
||||||
int launch_status = parse_options(argc, argv, NULL, NULL,
|
int launch_status = nagbar_parse_options(argc, argv, NULL, NULL,
|
||||||
&config_path, &debug);
|
&config_path, &debug);
|
||||||
if (launch_status != 0) {
|
if (launch_status != 0) {
|
||||||
exit_code = launch_status;
|
exit_code = launch_status;
|
||||||
|
@ -336,11 +53,11 @@ int main(int argc, char **argv) {
|
||||||
wlr_log_init(debug ? WLR_DEBUG : WLR_ERROR, NULL);
|
wlr_log_init(debug ? WLR_DEBUG : WLR_ERROR, NULL);
|
||||||
|
|
||||||
if (!config_path) {
|
if (!config_path) {
|
||||||
config_path = get_config_path();
|
config_path = nagbar_get_config_path();
|
||||||
}
|
}
|
||||||
if (config_path) {
|
if (config_path) {
|
||||||
wlr_log(WLR_DEBUG, "Loading config file: %s", config_path);
|
wlr_log(WLR_DEBUG, "Loading config file: %s", config_path);
|
||||||
int config_status = load_config(config_path, &nagbar, types);
|
int config_status = nagbar_load_config(config_path, &nagbar, types);
|
||||||
free(config_path);
|
free(config_path);
|
||||||
if (config_status != 0) {
|
if (config_status != 0) {
|
||||||
exit_code = config_status;
|
exit_code = config_status;
|
||||||
|
@ -349,7 +66,8 @@ int main(int argc, char **argv) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (argc > 1) {
|
if (argc > 1) {
|
||||||
int result = parse_options(argc, argv, &nagbar, types, NULL, NULL);
|
int result = nagbar_parse_options(argc, argv, &nagbar, types,
|
||||||
|
NULL, NULL);
|
||||||
if (result != 0) {
|
if (result != 0) {
|
||||||
exit_code = result;
|
exit_code = result;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
executable(
|
executable(
|
||||||
'swaynag', [
|
'swaynag', [
|
||||||
|
'config.c',
|
||||||
'main.c',
|
'main.c',
|
||||||
'nagbar.c',
|
'nagbar.c',
|
||||||
'render.c',
|
'render.c',
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <strings.h>
|
#include <strings.h>
|
||||||
#include "list.h"
|
#include "list.h"
|
||||||
|
#include "swaynag/config.h"
|
||||||
#include "swaynag/types.h"
|
#include "swaynag/types.h"
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue