diff options
Diffstat (limited to 'src/redshift.c')
-rw-r--r-- | src/redshift.c | 657 |
1 files changed, 138 insertions, 519 deletions
diff --git a/src/redshift.c b/src/redshift.c index d8969af..9a44b0b 100644 --- a/src/redshift.c +++ b/src/redshift.c @@ -1,5 +1,7 @@ -/* redshift.c -- Main program source - * This file is part of redshift-ng. +/* redshift-ng - Automatically adjust display colour temperature according the Sun + * + * Copyright (c) 2009-2018 Jon Lund Steffensen <jonlst@gmail.com> + * Copyright (c) 2014-2016, 2025 Mattias Andrée <m@maandree.se> * * redshift-ng is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by @@ -13,9 +15,6 @@ * * You should have received a copy of the GNU General Public License * along with redshift-ng. If not, see <http://www.gnu.org/licenses/>. - * - * Copyright (c) 2009-2017 Jon Lund Steffensen <jonlst@gmail.com> - * Copyright (c) 2025 Mattias Andrée <m@maandree.se> */ #include "common.h" @@ -43,42 +42,6 @@ int poll(struct pollfd *fds, int nfds, int timeout) { abort(); } #define FADE_LENGTH 40 -const struct gamma_method *gamma_methods[] = { -#ifdef ENABLE_COOPGAMMA - &coopgamma_gamma_method, -#endif -#ifdef ENABLE_DRM - &drm_gamma_method, -#endif -#ifdef ENABLE_RANDR - &randr_gamma_method, -#endif -#ifdef ENABLE_VIDMODE - &vidmode_gamma_method, -#endif -#ifdef ENABLE_QUARTZ - &quartz_gamma_method, -#endif -#ifdef ENABLE_WINGDI - &w32gdi_gamma_method, -#endif - &dummy_gamma_method, - NULL -}; - - -const struct location_provider *location_providers[] = { -#ifdef ENABLE_GEOCLUE2 - &geoclue2_location_provider, -#endif -#ifdef ENABLE_CORELOCATION - &corelocation_location_provider, -#endif - &manual_location_provider, - NULL -}; - - /* Names of periods of day */ static const char *period_names[] = { /* TRANSLATORS: Name printed when period of day is unknown */ @@ -103,72 +66,47 @@ exact_eq(double a, double b) #endif -/* Determine which period we are currently in based on time offset. */ -static enum period -get_period_from_time(const struct transition_scheme *transition, int time_offset) -{ - if (time_offset < transition->dawn.start || time_offset >= transition->dusk.end) - return PERIOD_NIGHT; - else if (time_offset >= transition->dawn.end && time_offset < transition->dusk.start) - return PERIOD_DAYTIME; - else - return PERIOD_TRANSITION; -} - -/* Determine which period we are currently in based on solar elevation. */ -static enum period -get_period_from_elevation(const struct transition_scheme *transition, double elevation) -{ - if (elevation < transition->low) - return PERIOD_NIGHT; - else if (elevation < transition->high) - return PERIOD_TRANSITION; - else - return PERIOD_DAYTIME; -} - -/* Determine how far through the transition we are based on time offset. */ -static double -get_transition_progress_from_time(const struct transition_scheme *transition, int time_offset) +/** + * Suspend the process for a short time + * + * The process may be resumed earily, specifically + * if it receives a signal + * + * @param msecs The number of milliseconds to sleep + */ +static void +millisleep(unsigned int msecs) { - if (time_offset < transition->dawn.start || time_offset >= transition->dusk.end) - return 0.0; - else if (time_offset < transition->dawn.end) - return (transition->dawn.start - time_offset) / (double)(transition->dawn.start - transition->dawn.end); - else if (time_offset > transition->dusk.start) - return (transition->dusk.end - time_offset) / (double)(transition->dusk.end - transition->dusk.start); - else - return 1.0; +#ifdef WINDOWS + Sleep(msecs); /* TODO [Windows] not interruptible */ +#else + struct timespec ts; + ts.tv_sec = (time_t)(msecs / 1000U); + ts.tv_nsec = (long)(msecs % 1000U) * 1000000L; + nanosleep(&ts, NULL); +#endif } -/* Determine how far through the transition we are based on elevation. */ -static double -get_transition_progress_from_elevation(const struct transition_scheme *transition, double elevation) -{ - if (elevation < transition->low) - return 0.0; - else if (elevation < transition->high) - return (transition->low - elevation) / (transition->low - transition->high); - else - return 1.0; -} /* Return number of seconds since midnight from timestamp. */ -static int -get_seconds_since_midnight(double timestamp) +static time_t +get_time_since_midnight(void) { - time_t t = (time_t)timestamp; + time_t t = time(NULL); struct tm tm; localtime_r(&t, &tm); - return tm.tm_sec + tm.tm_min * 60 + tm.tm_hour * 3600; + t = (time_t)tm.tm_sec; + t += (time_t)tm.tm_min * 60; + t += (time_t)tm.tm_hour * 3600; + return t; } /* Print verbose description of the given period. */ static void -print_period(enum period period, double transition) +print_period(enum period period, double day_level) { if (period == PERIOD_TRANSITION) - printf(_("Period: %s (%.2f%% day)\n"), gettext(period_names[period]), transition * 100); + printf(_("Period: %s (%.2f%% day)\n"), gettext(period_names[period]), day_level * 100); else printf(_("Period: %s\n"), gettext(period_names[period])); } @@ -207,13 +145,6 @@ interpolate_colour_settings(const struct colour_setting *first, const struct col result->gamma[i] = (1.0 - alpha) * first->gamma[i] + alpha * second->gamma[i]; } -/* Interpolate colour setting structs transition scheme. */ -static void -interpolate_transition_scheme(const struct transition_scheme *transition, double alpha, struct colour_setting *result) -{ - interpolate_colour_settings(&transition->night, &transition->day, alpha, result); -} - /* Return 1 if colour settings have major differences, otherwise 0. Used to determine if a fade should be applied in continual mode. */ static int @@ -226,153 +157,69 @@ colour_setting_diff_is_major(const struct colour_setting *first, const struct co fabs(first->gamma[2] - second->gamma[2]) > 0.1; } - -static int -provider_try_start(const struct location_provider *provider, LOCATION_STATE **state, - struct config_ini_state *config, char *args) -{ - const char *manual_keys[] = {"lat", "lon"}; - struct config_ini_section *section; - struct config_ini_setting *setting; - char *next_arg, *value; - const char *key; - int i; - - if (provider->init(state) < 0) { - weprintf(_("Initialization of %s failed."), provider->name); - return -1; - } - - /* Set provider options from config file. */ - if ((section = config_ini_get_section(config, provider->name))) { - for (setting = section->settings; setting; setting = setting->next) { - if (provider->set_option(*state, setting->name, setting->value) < 0) { - provider->free(*state); - weprintf(_("Failed to set %s option."), provider->name); - /* TRANSLATORS: `help' must not be translated. */ - weprintf(_("Try `-l %s:help' for more information."), provider->name); - return -1; - } - } - } - - /* Set provider options from command line. */ - for (i = 0; args; i++) { - next_arg = strchr(args, ':'); - if (next_arg) - *next_arg++ = '\0'; - - key = args; - value = strchr(args, '='); - if (!value) { - /* The options for the "manual" method can be set - without keys on the command line for convencience - and for backwards compatability. We add the proper - keys here before calling set_option(). */ - if (!strcmp(provider->name, "manual") && i < ELEMSOF(manual_keys)) { - key = manual_keys[i]; - value = args; - } else { - weprintf(_("Failed to parse option `%s'."), args); - return -1; - } - } else { - *value++ = '\0'; - } - - if (provider->set_option(*state, key, value) < 0) { - provider->free(*state); - weprintf(_("Failed to set %s option."), provider->name); - /* TRANSLATORS: `help' must not be translated. */ - weprintf(_("Try `-l %s:help' for more information."), provider->name); - return -1; - } - - args = next_arg; - } - - /* Start provider. */ - if (provider->start(*state) < 0) { - provider->free(*state); - weprintf(_("Failed to start provider %s.\n"), provider->name); - return -1; - } - - return 0; -} - -static int -method_try_start(const struct gamma_method *method, GAMMA_STATE **state, - enum program_mode mode, struct config_ini_state *config, char *args) +/** + * Get the current period of day and the colour settings + * applicable to the current time of the day + * + * @param location Geographical user location + * @param colour_out Output parameter for the colour settings + * @param period_out Output parameter for the period of the day + * @param day_level_out Output parameter for the dayness level + * @param verbose Whether the application is running in verbose mode + */ +static void +get_colour_settings(const struct location *location, struct colour_setting *colour_out, + enum period *period_out, double *day_level_out, int verbose) { - struct config_ini_section *section; - struct config_ini_setting *setting; - char *next_arg, *value; - const char *key; - - if (method->init(state) < 0) { - weprintf(_("Initialization of %s failed."), method->name); - return -1; - } - - /* Set method options from config file. */ - if ((section = config_ini_get_section(config, method->name))) { - for (setting = section->settings; setting; setting = setting->next) { - if (method->set_option(*state, setting->name, setting->value) < 0) { - method->free(*state); - weprintf(_("Failed to set %s option."), method->name); - /* TRANSLATORS: `help' must not be translated. */ - weprintf(_("Try `-m %s:help' for more information.\n"), method->name); /* TODO \n */ - return -1; - } + time_t time_offset; + double t, elevation; + + /* Get dayness level */ + if (scheme.type == CLOCK_SCHEME) { + time_offset = get_time_since_midnight(); + while (time_offset >= scheme.time.periods->next->start) + scheme.time.periods = scheme.time.periods->next; + time_offset -= scheme.time.periods->start; + if (time_offset < 0) + time_offset += ONE_DAY; + t = (double)time_offset; + *day_level_out = fma(t, scheme.time.periods->diff_over_duration, scheme.time.periods->day_level); + + } else if (scheme.type == SOLAR_SCHEME) { + if (libred_solar_elevation(location->latitude, location->longitude, &elevation)) + eprintf("libred_solar_elevation:"); + if (verbose) { + /* TRANSLATORS: Append degree symbol if possible. */ + printf(_("Solar elevation: %f\n"), elevation); } + *day_level_out = (elevation - scheme.elevation.low) / scheme.elevation.range; + + } else { + /* Static scheme, no dayness-level or peroid; day_settings == nigh_settings, use either */ + *day_level_out = NAN; + *period_out = PERIOD_NONE; + *colour_out = day_settings; + return; } - /* Set method options from command line. */ - while (args) { - next_arg = strchr(args, ':'); - if (next_arg) - *next_arg++ = '\0'; - - key = args; - value = strchr(args, '='); - if (!value) { - weprintf(_("Failed to parse option `%s'."), args); - return -1; - } - *value++ = '\0'; - - if (method->set_option(*state, key, value) < 0) { - method->free(*state); - weprintf(_("Failed to set %s option."), method->name); - /* TRANSLATORS: `help' must not be translated. */ - weprintf(_("Try -m %s:help' for more information.\n"), method->name); /* TODO missing ` and \n */ - return -1; - } + /* Clamp dayness level and get colour */ + if (*day_level_out <= 0.0) { + *day_level_out = 0.0; + *period_out = PERIOD_NIGHT; + *colour_out = night_settings; - args = next_arg; - } + } else if (*day_level_out >= 1.0) { + *day_level_out = 1.0; + *period_out = PERIOD_DAYTIME; + *colour_out = day_settings; - /* Start method. */ - if (method->start(*state, mode) < 0) { - method->free(*state); - weprintf(_("Failed to start adjustment method %s.\n"), method->name); /* TODO \n */ - return -1; + } else { + *period_out = PERIOD_TRANSITION; + interpolate_colour_settings(&night_settings, &day_settings, *day_level_out, colour_out); } - - return 0; } -/* Check whether gamma is within allowed levels. */ -static int -gamma_is_valid(const double gamma[3]) -{ - return WITHIN(MIN_GAMMA, gamma[0], MAX_GAMMA) && - WITHIN(MIN_GAMMA, gamma[1], MAX_GAMMA) && - WITHIN(MIN_GAMMA, gamma[2], MAX_GAMMA); -} - /* Check whether location is valid. Prints error message on stderr and returns 0 if invalid, otherwise returns 1. */ @@ -381,62 +228,17 @@ location_is_valid(const struct location *location) { if (!WITHIN(MIN_LATITUDE, location->latitude, MAX_LATITUDE)) { /* TRANSLATORS: Append degree symbols if possible. */ - weprintf(_("Latitude must be between %.1f and %.1f.\n"), MIN_LATITUDE, MAX_LATITUDE); /* TODO \n */ + weprintf(_("Latitude must be between %.1f and %.1f."), MIN_LATITUDE, MAX_LATITUDE); return 0; } if (!WITHIN(MIN_LONGITUDE, location->longitude, MAX_LONGITUDE)) { /* TRANSLATORS: Append degree symbols if possible. */ - weprintf(_("Longitude must be between %.1f and %.1f.\n"), MIN_LONGITUDE, MAX_LONGITUDE); /* TODO \n */ + weprintf(_("Longitude must be between %.1f and %.1f."), MIN_LONGITUDE, MAX_LONGITUDE); return 0; } return 1; } -/* Wait for location to become available from provider. - Waits until timeout (milliseconds) has elapsed or forever if timeout - is -1. Writes location to loc. Returns -1 on error, - 0 if timeout was reached, 1 if location became available. */ -static int -provider_get_location(const struct location_provider *provider, LOCATION_STATE *state, int timeout, struct location *loc) -{ - int r, available, loc_fd; - struct pollfd pollfds[1]; - double now, later; - - do { - loc_fd = provider->get_fd(state); - if (loc_fd >= 0) { - /* Provider is dynamic. */ - /* TODO: This should use a monotonic time source. */ - now = systemtime_get_time(); - - /* Poll on file descriptor until ready. */ - pollfds[0].fd = loc_fd; - pollfds[0].events = POLLIN; - r = poll(pollfds, 1, timeout); - if (r < 0) { - weprintf("poll {{.fd=<location provider>, .events=EPOLLIN}} 1 %i:", timeout); - return -1; - } else if (r == 0) { - return 0; - } - - later = systemtime_get_time(); - - /* Adjust timeout by elapsed time */ - if (timeout >= 0) { - timeout -= (later - now) * 1000; - timeout = MAX(timeout, 0); - } - } - - if (provider->handle(state, loc, &available) < 0) - return -1; - } while (!available); - - return 1; -} - /* Easing function for fade. See https://github.com/mietek/ease-tween */ static double @@ -455,8 +257,8 @@ ease_fade(double t) colour temperature. */ static void run_continual_mode(const struct location_provider *provider, LOCATION_STATE *location_state, - const struct transition_scheme *scheme, const struct gamma_method *method, - GAMMA_STATE *method_state, int use_fade, int preserve_gamma, int verbose) + const struct gamma_method *method, GAMMA_STATE *method_state, int use_fade, + int preserve_gamma, int verbose) { int done = 0; int prev_disabled = 1; @@ -466,7 +268,6 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc struct colour_setting prev_target_interp; struct colour_setting interp; struct location loc; - int need_location; /* Short fade parameters */ int fade_length = 0; @@ -476,7 +277,7 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc the values did not change. */ enum period prev_period = PERIOD_NONE; - signals_install_handlers(); + install_signal_handlers(); /* Previous target colour setting and current actual colour setting. Actual colour setting takes into account the current colour fade. */ @@ -485,16 +286,15 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc interp = COLOUR_SETTING_NEUTRAL; loc = (struct location){NAN, NAN}; - need_location = !scheme->use_time; - if (need_location) { - weprintf(_("Waiting for initial location to become available...\n")); /* TODO \n */ + if (scheme.type == SOLAR_SCHEME) { + weprintf(_("Waiting for initial location to become available...")); /* Get initial location from provider */ - if (provider_get_location(provider, location_state, -1, &loc) < 0) + if (get_location(provider, location_state, -1, &loc) < 0) eprintf(_("Unable to get location from provider.")); if (!location_is_valid(&loc)) - eprintf(_("Invalid location returned from provider.\n")); /* TODO \n */ + eprintf(_("Invalid location returned from provider.")); print_location(&loc); } @@ -506,9 +306,8 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc /* Continuously adjust colour temperature */ for (;;) { - double now; enum period period; - double transition_prog; + double day_level; struct colour_setting target_interp; int delay, loc_fd; @@ -533,26 +332,7 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc prev_disabled = disabled; - /* Read timestamp */ - now = systemtime_get_time(); - - if (scheme->use_time) { - int time_offset = get_seconds_since_midnight(now); - - period = get_period_from_time(scheme, time_offset); - transition_prog = get_transition_progress_from_time(scheme, time_offset); - } else { - /* Current angular elevation of the sun */ - double elevation; - if (libred_solar_elevation(loc.latitude, loc.longitude, &elevation)) - eprintf("libred_solar_elevation:"); - - period = get_period_from_elevation(scheme, elevation); - transition_prog = get_transition_progress_from_elevation(scheme, elevation); - } - - /* Use transition progress to get target colour temperature. */ - interpolate_transition_scheme(scheme, transition_prog, &target_interp); + get_colour_settings(&loc, &target_interp, &period, &day_level, verbose); if (disabled) { period = PERIOD_NONE; @@ -567,21 +347,18 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc print the progress, so we always print it in that case. */ if (verbose && (period != prev_period || period == PERIOD_TRANSITION)) - print_period(period, transition_prog); + print_period(period, day_level); /* Activate hooks if period changed */ if (period != prev_period) - hooks_signal_period_change(prev_period, period); + run_period_change_hooks(prev_period, period); /* Start fade if the parameter differences are too big to apply instantly. */ - if (use_fade) { - if (fade_length ? colour_setting_diff_is_major(&target_interp, &prev_target_interp) - : colour_setting_diff_is_major(&interp, &target_interp)) { - fade_length = FADE_LENGTH; - fade_time = 0; - fade_start_interp = interp; - } + if (use_fade && colour_setting_diff_is_major(&target_interp, fade_length ? &prev_target_interp : &interp)) { + fade_length = FADE_LENGTH; + fade_time = 0; + fade_start_interp = interp; } /* Handle ongoing fade */ @@ -622,7 +399,7 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc delay = fade_length ? SLEEP_DURATION_SHORT : SLEEP_DURATION; /* Update location. */ - loc_fd = need_location ? provider->get_fd(location_state) : -1; + loc_fd = scheme.type == SOLAR_SCHEME ? provider->get_fd(location_state) : -1; if (loc_fd >= 0) { struct pollfd pollfds[1]; @@ -634,22 +411,23 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc pollfds[0].events = POLLIN; r = poll(pollfds, 1, delay); if (r < 0) { +#ifndef WINDOWS if (errno == EINTR) continue; +#endif weprintf("poll:"); eprintf(_("Unable to get location from provider.")); } else if (!r) { continue; } - /* Get new location and availability - information. */ + /* Get new location and availability information. */ if (provider->handle(location_state, &new_loc, &new_available) < 0) eprintf(_("Unable to get location from provider.")); if (!new_available && new_available != location_available) { - weprintf(_("Location is temporarily unavailable; Using previous" /* TODO captial U efter ; and \n*/ - " location until it becomes available...\n")); + weprintf(_("Location is temporarily unavailable; using previous" + " location until it becomes available...")); } if (new_available && @@ -665,7 +443,7 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc if (!location_is_valid(&loc)) eprintf(_("Invalid location returned from provider.")); } else { - systemtime_msleep(delay); + millisleep(delay); } } @@ -677,15 +455,11 @@ run_continual_mode(const struct location_provider *provider, LOCATION_STATE *loc int main(int argc, char *argv[]) { - struct options options; - struct config_ini_state config_state; - struct transition_scheme *scheme; + struct settings settings; GAMMA_STATE *method_state; LOCATION_STATE *location_state; - int need_location; - size_t i; - struct location loc = { NAN, NAN }; - double now, transition_prog; + struct location loc = {NAN, NAN}; + double day_level; enum period period; struct colour_setting colour; @@ -698,167 +472,32 @@ main(int argc, char *argv[]) textdomain(PACKAGE); #endif - options_init(&options); - options_parse_args(&options, argc, argv); - - /* Load settings from config file. */ - config_ini_init(&config_state, options.config_filepath); - - free(options.config_filepath); - - options_parse_config_file(&options, &config_state); - - options_set_defaults(&options); - - if (options.scheme.dawn.start >= 0 || options.scheme.dawn.end >= 0 || - options.scheme.dusk.start >= 0 || options.scheme.dusk.end >= 0) { - if (options.scheme.dawn.start < 0 || options.scheme.dawn.end < 0 || - options.scheme.dusk.start < 0 || options.scheme.dusk.end < 0) - eprintf(_("Partial time-configuration not supported!")); - - if (options.scheme.dawn.start > options.scheme.dawn.end || - options.scheme.dawn.end > options.scheme.dusk.start || - options.scheme.dusk.start > options.scheme.dusk.end) - eprintf(_("Invalid dawn/dusk time configuration!")); - - options.scheme.use_time = 1; - } + load_settings(&settings, argc, argv); /* Initialize location provider if needed. If provider is NULL try all providers until one that works is found. */ /* Location is not needed for reset mode and manual mode. */ - need_location = options.mode != PROGRAM_MODE_RESET && - options.mode != PROGRAM_MODE_MANUAL && - !options.scheme.use_time; - if (need_location) { - if (options.provider) { - /* Use provider specified on command line. */ - if (provider_try_start(options.provider, &location_state, &config_state, options.provider_args) < 0) - exit(1); - } else { - /* Try all providers, use the first that works. */ - for (i = 0; location_providers[i]; i++) { - const struct location_provider *p = location_providers[i]; - weprintf(_("Trying location provider `%s'..."), p->name); - if (provider_try_start(p, &location_state, &config_state, NULL) < 0) { - weprintf(_("Trying next provider...")); - continue; - } - - /* Found provider that works. */ - printf(_("Using provider `%s'.\n"), p->name); - options.provider = p; - break; - } - - /* Failure if no providers were successful at this point. */ - if (!options.provider) - eprintf(_("No more location providers to try.")); - } - - /* Solar elevations */ - if (options.scheme.high < options.scheme.low) { - eprintf(_("High transition elevation cannot be lower than the low transition elevation.")); - } - - if (options.verbose) { - /* TRANSLATORS: Append degree symbols if possible. */ - printf(_("Solar elevations: day above %.1f, night below %.1f\n"), - options.scheme.high, options.scheme.low); - } - } - - if (options.mode != PROGRAM_MODE_RESET && - options.mode != PROGRAM_MODE_MANUAL) { - if (options.verbose) { - printf(_("Temperatures: %luK at day, %luK at night\n"), - options.scheme.day.temperature, options.scheme.night.temperature); - } - - /* Colour temperature */ - if (!WITHIN(MIN_TEMPERATURE, options.scheme.day.temperature, MAX_TEMPERATURE) || - !WITHIN(MIN_TEMPERATURE, options.scheme.night.temperature, MAX_TEMPERATURE)) - eprintf(_("Temperature must be between %luK and %luK."), MIN_TEMPERATURE, MAX_TEMPERATURE); - } - - if (options.mode == PROGRAM_MODE_MANUAL) { - /* Check colour temperature to be set */ - if (!WITHIN(MIN_TEMPERATURE, options.temp_set, MAX_TEMPERATURE)) - eprintf(_("Temperature must be between %luK and %luK."), MIN_TEMPERATURE, MAX_TEMPERATURE); - } - - /* Brightness */ - if (!WITHIN(MIN_BRIGHTNESS, options.scheme.day.brightness, MAX_BRIGHTNESS) || - !WITHIN(MIN_BRIGHTNESS, options.scheme.night.brightness, MAX_BRIGHTNESS)) - eprintf(_("Brightness values must be between %.1f and %.1f."), MIN_BRIGHTNESS, MAX_BRIGHTNESS); - - if (options.verbose) - printf(_("Brightness: %.2f:%.2f\n"), options.scheme.day.brightness, options.scheme.night.brightness); - - /* Gamma */ - if (!gamma_is_valid(options.scheme.day.gamma) || - !gamma_is_valid(options.scheme.night.gamma)) - eprintf(_("Gamma value must be between %.1f and %.1f."), MIN_GAMMA, MAX_GAMMA); - - if (options.verbose) { - /* TRANSLATORS: The string in parenthesis is either - Daytime or Night (translated). */ - printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"), - _("Daytime"), options.scheme.day.gamma[0], - options.scheme.day.gamma[1], - options.scheme.day.gamma[2]); - printf(_("Gamma (%s): %.3f, %.3f, %.3f\n"), - _("Night"), options.scheme.night.gamma[0], - options.scheme.night.gamma[1], - options.scheme.night.gamma[2]); - } - - scheme = &options.scheme; + if (settings.scheme_type == SOLAR_SCHEME) + acquire_location_provider(&settings, &location_state); /* Initialize gamma adjustment method. If method is NULL try all methods until one that works is found. */ /* Gamma adjustment not needed for print mode */ - if (options.mode != PROGRAM_MODE_PRINT) { - if (options.method) { - /* Use method specified on command line. */ - if (method_try_start(options.method, &method_state, options.mode, &config_state, options.method_args) < 0) - exit(1); - } else { - /* Try all methods, use the first that works. */ - for (i = 0; gamma_methods[i]; i++) { - const struct gamma_method *m = gamma_methods[i]; - if (!m->autostart) - continue; + if (settings.mode != PROGRAM_MODE_PRINT) + acquire_adjustment_method(&settings, &method_state); - if (method_try_start(m, &method_state, options.mode, &config_state, NULL) < 0) { - weprintf(_("Trying next method...")); - continue; - } - - /* Found method that works. */ - printf(_("Using method `%s'.\n"), m->name); - options.method = m; - break; - } + config_ini_free(&settings.config); - /* Failure if no methods were successful at this point. */ - if (!options.method) - eprintf(_("No more methods to try.")); - } - } - - config_ini_free(&config_state); - - switch (options.mode) { + switch (settings.mode) { case PROGRAM_MODE_ONE_SHOT: case PROGRAM_MODE_PRINT: - if (need_location) { + if (settings.scheme_type == SOLAR_SCHEME) { weprintf(_("Waiting for current location to become available...")); /* Wait for location provider. */ - if (provider_get_location(options.provider, location_state, -1, &loc) < 0) + if (get_location(settings.provider, location_state, -1, &loc) < 0) eprintf(_("Unable to get location from provider.")); if (!location_is_valid(&loc)) @@ -867,46 +506,26 @@ main(int argc, char *argv[]) print_location(&loc); } - now = systemtime_get_time(); - - if (options.scheme.use_time) { - int time_offset = get_seconds_since_midnight(now); - period = get_period_from_time(scheme, time_offset); - transition_prog = get_transition_progress_from_time(scheme, time_offset); - } else { - /* Current angular elevation of the sun */ - double elevation; - if (libred_solar_elevation(loc.latitude, loc.longitude, &elevation)) - eprintf("libred_solar_elevation:"); - if (options.verbose) { - /* TRANSLATORS: Append degree symbol if possible. */ - printf(_("Solar elevation: %f\n"), elevation); - } + get_colour_settings(&loc, &colour, &period, &day_level, settings.verbose); - period = get_period_from_elevation(scheme, elevation); - transition_prog = get_transition_progress_from_elevation(scheme, elevation); - } - - /* Use transition progress to set colour temperature */ - interpolate_transition_scheme(scheme, transition_prog, &colour); - - if (options.verbose || options.mode == PROGRAM_MODE_PRINT) { - print_period(period, transition_prog); + if (settings.verbose || settings.mode == PROGRAM_MODE_PRINT) { + if (settings.scheme_type != STATIC_SCHEME) + print_period(period, day_level); printf(_("Color temperature: %luK\n"), colour.temperature); printf(_("Brightness: %.2f\n"), colour.brightness); } - if (options.mode == PROGRAM_MODE_PRINT) + if (settings.mode == PROGRAM_MODE_PRINT) break; apply: - if (options.method->set_temperature(method_state, &colour, options.preserve_gamma) < 0) + if (settings.method->set_temperature(method_state, &colour, settings.preserve_gamma.value) < 0) eprintf(_("Temperature adjustment failed.")); #ifndef WINDOWS /* In Quartz (OSX) the gamma adjustments will automatically revert when * the process exits. Therefore, we have to loop until CTRL-C is received. */ - if (!strcmp(options.method->name, "quartz")) { + if (!strcmp(settings.method->name, "quartz")) { weprintf(_("Press ctrl-c to stop...")); while (!exiting) pause(); @@ -915,26 +534,26 @@ main(int argc, char *argv[]) break; case PROGRAM_MODE_MANUAL: - if (options.verbose) - printf(_("Color temperature: %luK\n"), options.temp_set); - colour = scheme->day; - colour.temperature = options.temp_set; + /* TODO interpolate for current time if a value range has been specified */ + colour = day_settings; + if (settings.verbose) + printf(_("Color temperature: %luK\n"), colour.temperature); goto apply; case PROGRAM_MODE_RESET: colour = COLOUR_SETTING_NEUTRAL; - options.preserve_gamma = 0; + settings.preserve_gamma.value = 0; goto apply; case PROGRAM_MODE_CONTINUAL: - run_continual_mode(options.provider, location_state, scheme, options.method, method_state, - options.use_fade, options.preserve_gamma, options.verbose); + run_continual_mode(settings.provider, location_state, settings.method, method_state, + settings.use_fade.value, settings.preserve_gamma.value, settings.verbose); break; } - if (options.mode != PROGRAM_MODE_PRINT) - options.method->free(method_state); - if (need_location) - options.provider->free(location_state); + if (settings.mode != PROGRAM_MODE_PRINT) + settings.method->free(method_state); + if (scheme.type == SOLAR_SCHEME) + settings.provider->free(location_state); return 0; } |