/* ide-application-color.c * * Copyright 2017-2019 Christian Hergert * * This program 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 3 of the License, or * (at your option) any later version. * * This program 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 this program. If not, see . * * SPDX-License-Identifier: GPL-3.0-or-later */ #define G_LOG_DOMAIN "ide-application-color" #include "config.h" #include #include #include "ide-application.h" #include "ide-application-private.h" static void add_style_name (GPtrArray *ar, const gchar *base, gboolean dark) { g_ptr_array_add (ar, g_strdup_printf ("%s-%s", base, dark ? "dark" : "light")); } static gchar * find_similar_style_scheme (const gchar *name, gboolean is_dark_mode) { g_autoptr(GPtrArray) attempts = NULL; GtkSourceStyleSchemeManager *mgr; const gchar * const *scheme_ids; const gchar *dash; g_assert (name != NULL); attempts = g_ptr_array_new_with_free_func (g_free); mgr = gtk_source_style_scheme_manager_get_default (); scheme_ids = gtk_source_style_scheme_manager_get_scheme_ids (mgr); add_style_name (attempts, name, is_dark_mode); if ((dash = strrchr (name, '-'))) { if (g_str_equal (dash, "-light") || g_str_equal (dash, "-dark")) { g_autofree gchar *base = NULL; base = g_strndup (name, dash - name); add_style_name (attempts, base, is_dark_mode); /* Add the base name last so light/dark matches first */ g_ptr_array_add (attempts, g_steal_pointer (&base)); } } /* * Instead of using gtk_source_style_scheme_manager_get_scheme(), we get the * IDs and look using case insensitive search so that its more likely we get * a match when something is named Dark or Light in the id. */ for (guint i = 0; i < attempts->len; i++) { const gchar *attempt = g_ptr_array_index (attempts, i); for (guint j = 0; scheme_ids[j] != NULL; j++) { if (strcasecmp (attempt, scheme_ids[j]) == 0) return g_strdup (scheme_ids[j]); } } return NULL; } static void _ide_application_update_color (IdeApplication *self) { static gboolean ignore_reentrant = FALSE; HdyStyleManager *manager; g_autofree char *style_variant = NULL; g_assert (IDE_IS_APPLICATION (self)); if (ignore_reentrant) return; if (self->settings == NULL) return; ignore_reentrant = TRUE; g_assert (G_IS_SETTINGS (self->settings)); style_variant = g_settings_get_string (self->settings, "style-variant"); manager = hdy_style_manager_get_default (); if (!g_strcmp0 (style_variant, "follow")) hdy_style_manager_set_color_scheme (manager, HDY_COLOR_SCHEME_PREFER_LIGHT); else if (!g_strcmp0 (style_variant, "dark")) hdy_style_manager_set_color_scheme (manager, HDY_COLOR_SCHEME_FORCE_DARK); else hdy_style_manager_set_color_scheme (manager, HDY_COLOR_SCHEME_FORCE_LIGHT); ignore_reentrant = FALSE; } static void _ide_application_update_style_scheme (IdeApplication *self) { HdyStyleManager *manager; g_autoptr(GSettings) editor_settings = NULL; g_autofree gchar *old_name = NULL; g_autofree gchar *new_name = NULL; manager = hdy_style_manager_get_default (); /* * Now that we have our color up to date, we need to possibly update the * color scheme to match the setting. We always do this (and not just when * the style-variant changes) so that we pick up changes at startup. * * Try to locate a corresponding style-scheme for the light/dark switch * based on some naming conventions. If found, switch the current style * scheme to match. */ editor_settings = g_settings_new ("org.gnome.builder.editor"); old_name = g_settings_get_string (editor_settings, "style-scheme-name"); new_name = find_similar_style_scheme (old_name, hdy_style_manager_get_dark (manager)); if (new_name != NULL) g_settings_set_string (editor_settings, "style-scheme-name", new_name); } void _ide_application_init_color (IdeApplication *self) { g_return_if_fail (IDE_IS_APPLICATION (self)); g_return_if_fail (G_IS_SETTINGS (self->settings)); if (g_getenv ("GTK_THEME") == NULL) { g_signal_connect_object (self->settings, "changed::style-variant", G_CALLBACK (_ide_application_update_color), self, G_CONNECT_SWAPPED); g_signal_connect_object (hdy_style_manager_get_default (), "notify::dark", G_CALLBACK (_ide_application_update_style_scheme), self, G_CONNECT_SWAPPED); } _ide_application_update_color (self); _ide_application_update_style_scheme (self); }