/* Copyright 2011-2020 Bert Muennich * Copyright 2021-2023 nsxiv contributors * * This file is a part of nsxiv. * * nsxiv is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published * by the Free Software Foundation; either version 2 of the License, * or (at your option) any later version. * * nsxiv is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with nsxiv. If not, see . */ #include "nsxiv.h" #include "version.h" #define INCLUDE_OPTIONS_CONFIG #include "config.h" #include #include #include #include #include #include #define OPTPARSE_IMPLEMENTATION #define OPTPARSE_API static #pragma GCC diagnostic push /* also works on clang */ #pragma GCC diagnostic ignored "-Wshadow" #pragma GCC diagnostic ignored "-Wunused-function" #include "optparse.h" #pragma GCC diagnostic pop const opt_t *options; void print_usage(void) { printf("usage: %s [-abcfhiopqrtvZ0] [-A FRAMERATE] [-e WID] [-G GAMMA] " "[-g GEOMETRY] [-N NAME] [-n NUM] [-S DELAY] [-s MODE] " "[-z ZOOM] FILES...\n", progname); } static void print_version(void) { printf("%s %s\n", progname, VERSION); fputs("features: " #if HAVE_INOTIFY "+inotify " #endif #if HAVE_LIBFONTS "+statusbar " #endif #if HAVE_LIBEXIF "+exif " #endif #if HAVE_IMLIB2_MULTI_FRAME "+multiframe " #endif "\n", stdout); } void parse_options(int argc, char **argv) { enum { /* ensure these can't be represented in a single byte in order * to avoid conflicts with short opts */ OPT_START = UCHAR_MAX, OPT_AA, OPT_AL, OPT_BG }; static const struct optparse_long longopts[] = { { "framerate", 'A', OPTPARSE_REQUIRED }, { "animate", 'a', OPTPARSE_NONE }, { "no-bar", 'b', OPTPARSE_NONE }, { "clean-cache", 'c', OPTPARSE_NONE }, { "embed", 'e', OPTPARSE_REQUIRED }, { "fullscreen", 'f', OPTPARSE_NONE }, { "gamma", 'G', OPTPARSE_REQUIRED }, { "geometry", 'g', OPTPARSE_REQUIRED }, { "help", 'h', OPTPARSE_NONE }, { "stdin", 'i', OPTPARSE_NONE }, { "class", 'N', OPTPARSE_REQUIRED }, { "start-at", 'n', OPTPARSE_REQUIRED }, { "stdout", 'o', OPTPARSE_NONE }, { "private", 'p', OPTPARSE_NONE }, { "quiet", 'q', OPTPARSE_NONE }, { "recursive", 'r', OPTPARSE_NONE }, { "ss-delay", 'S', OPTPARSE_REQUIRED }, { "scale-mode", 's', OPTPARSE_REQUIRED }, { "thumbnail", 't', OPTPARSE_NONE }, { "version", 'v', OPTPARSE_NONE }, { "zoom-100", 'Z', OPTPARSE_NONE }, { "zoom", 'z', OPTPARSE_REQUIRED }, { "null", '0', OPTPARSE_NONE }, { "anti-alias", OPT_AA, OPTPARSE_OPTIONAL }, { "alpha-layer", OPT_AL, OPTPARSE_OPTIONAL }, /* TODO: document this when it's stable */ { "bg-cache", OPT_BG, OPTPARSE_OPTIONAL }, { 0 }, /* end */ }; long n, opt; float f; char *end, *s; struct optparse op; const char scalemodes[] = "dfFwh"; /* must be sorted according to scalemode_t */ static opt_t _options; options = &_options; _options.from_stdin = false; _options.to_stdout = false; _options.using_null = false; _options.recursive = false; _options.startnum = 0; _options.scalemode = SCALE_DOWN; _options.zoom = 1.0; _options.anti_alias = ANTI_ALIAS; _options.alpha_layer = ALPHA_LAYER; _options.animate = false; _options.gamma = 0; _options.slideshow = 0; _options.framerate = 0; _options.fullscreen = false; _options.embed = 0; _options.hide_bar = false; _options.geometry = NULL; _options.res_name = NULL; _options.quiet = false; _options.thumb_mode = false; _options.clean_cache = false; _options.private_mode = false; _options.background_cache = false; if (argc > 0) { s = strrchr(argv[0], '/'); progname = s != NULL && s[1] != '\0' ? s + 1 : argv[0]; } optparse_init(&op, argv); while ((opt = optparse_long(&op, longopts, NULL)) != -1) { for (n = 0; n < (int)ARRLEN(longopts); ++n) { /* clang-tidy finds some non-sensical branch and thinks optarg == NULL is possible */ if (opt == longopts[n].shortname && longopts[n].argtype == OPTPARSE_REQUIRED) assert(op.optarg != NULL); } switch (opt) { case '?': fprintf(stderr, "%s\n", op.errmsg); print_usage(); exit(EXIT_FAILURE); case 'A': n = strtol(op.optarg, &end, 0); if (*end != '\0' || n <= 0 || n > INT_MAX) error(EXIT_FAILURE, 0, "Invalid framerate: %s", op.optarg); _options.framerate = n; /* fall through */ case 'a': _options.animate = true; break; case 'b': _options.hide_bar = true; break; case 'c': _options.clean_cache = true; break; case 'e': n = strtol(op.optarg, &end, 0); if (*end != '\0') error(EXIT_FAILURE, 0, "Invalid window id: %s", op.optarg); _options.embed = n; break; case 'f': _options.fullscreen = true; break; case 'G': n = strtol(op.optarg, &end, 0); if (*end != '\0' || n < INT_MIN || n > INT_MAX) error(EXIT_FAILURE, 0, "Invalid gamma: %s", op.optarg); _options.gamma = n; break; case 'g': _options.geometry = op.optarg; break; case 'h': print_usage(); exit(EXIT_SUCCESS); case 'i': _options.from_stdin = true; break; case 'n': n = strtol(op.optarg, &end, 0); if (*end != '\0' || n <= 0 || n > INT_MAX) error(EXIT_FAILURE, 0, "Invalid starting number: %s", op.optarg); _options.startnum = n - 1; break; case 'N': _options.res_name = op.optarg; break; case 'o': _options.to_stdout = true; break; case 'p': _options.private_mode = true; break; case 'q': _options.quiet = true; break; case 'r': _options.recursive = true; break; case 'S': f = strtof(op.optarg, &end) * 10.0f; if (*end != '\0' || f <= 0 || f >= (float)UINT_MAX) error(EXIT_FAILURE, 0, "Invalid slideshow delay: %s", op.optarg); _options.slideshow = (unsigned int)f; break; case 's': s = strchr(scalemodes, op.optarg[0]); if (s == NULL || *s == '\0' || strlen(op.optarg) != 1) error(EXIT_FAILURE, 0, "Invalid scale mode: %s", op.optarg); _options.scalemode = s - scalemodes; break; case 't': _options.thumb_mode = true; break; case 'v': print_version(); exit(EXIT_SUCCESS); case 'Z': _options.scalemode = SCALE_ZOOM; _options.zoom = 1.0f; break; case 'z': n = strtol(op.optarg, &end, 0); if (*end != '\0' || n <= 0) error(EXIT_FAILURE, 0, "Invalid zoom level: %s", op.optarg); _options.scalemode = SCALE_ZOOM; _options.zoom = (float)n / 100.0f; break; case '0': _options.using_null = true; break; case OPT_AA: if (op.optarg != NULL && !STREQ(op.optarg, "no")) error(EXIT_FAILURE, 0, "Invalid argument for option --anti-alias: %s", op.optarg); _options.anti_alias = op.optarg == NULL; break; case OPT_AL: if (op.optarg != NULL && !STREQ(op.optarg, "no")) error(EXIT_FAILURE, 0, "Invalid argument for option --alpha-layer: %s", op.optarg); _options.alpha_layer = op.optarg == NULL; break; case OPT_BG: if (op.optarg != NULL && !STREQ(op.optarg, "no")) error(EXIT_FAILURE, 0, "Invalid argument for option --bg-cache: %s", op.optarg); _options.background_cache = op.optarg == NULL; break; } } _options.filenames = argv + op.optind; _options.filecnt = argc - op.optind; if (_options.filecnt == 1 && STREQ(_options.filenames[0], "-")) { _options.filenames++; _options.filecnt--; _options.from_stdin = true; } }