[gnome-builder/wip/gtk4-port: 254/343] libide/gui: add recoloring helpers
- From: Christian Hergert <chergert src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-builder/wip/gtk4-port: 254/343] libide/gui: add recoloring helpers
- Date: Mon, 4 Apr 2022 20:02:16 +0000 (UTC)
commit 8c63855607a09b6d4ac9bca5ecf703f26f935a20
Author: Christian Hergert <chergert redhat com>
Date: Thu Mar 31 03:59:46 2022 -0700
libide/gui: add recoloring helpers
From text editor, to do the recoloring generation from schemes.
src/libide/gui/ide-recoloring-private.h | 32 +++
src/libide/gui/ide-recoloring.c | 368 ++++++++++++++++++++++++++++++++
src/libide/gui/meson.build | 4 +
3 files changed, 404 insertions(+)
---
diff --git a/src/libide/gui/ide-recoloring-private.h b/src/libide/gui/ide-recoloring-private.h
new file mode 100644
index 000000000..39e344fe8
--- /dev/null
+++ b/src/libide/gui/ide-recoloring-private.h
@@ -0,0 +1,32 @@
+/* ide-recoloring-private.h
+ *
+ * Copyright 2020 Christian Hergert <chergert redhat com>
+ *
+ * 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 <http://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#pragma once
+
+#include <gtksourceview/gtksource.h>
+
+G_BEGIN_DECLS
+
+char *_ide_recoloring_generate_css (GtkSourceStyleScheme *style_scheme);
+gboolean _ide_source_style_scheme_is_dark (GtkSourceStyleScheme *style_scheme);
+GtkSourceStyleScheme *_ide_source_style_scheme_get_variant (GtkSourceStyleScheme *style_scheme,
+ const char *variant);
+
+G_END_DECLS
diff --git a/src/libide/gui/ide-recoloring.c b/src/libide/gui/ide-recoloring.c
new file mode 100644
index 000000000..ff2bbd485
--- /dev/null
+++ b/src/libide/gui/ide-recoloring.c
@@ -0,0 +1,368 @@
+/* ide-recoloring.c
+ *
+ * Copyright 2021 Christian Hergert <chergert redhat com>
+ *
+ * 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 <http://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#include "config.h"
+
+#include <math.h>
+
+#include "ide-recoloring-private.h"
+
+#define SHARED_CSS \
+ "@define-color card_fg_color @window_fg_color;\n" \
+ "@define-color headerbar_fg_color @window_fg_color;\n" \
+ "@define-color headerbar_border_color @window_fg_color;\n" \
+ "@define-color popover_fg_color @window_fg_color;\n" \
+ "@define-color dark_fill_bg_color @headerbar_bg_color;\n" \
+ "@define-color view_bg_color @card_bg_color;\n" \
+ "@define-color view_fg_color @window_fg_color;\n"
+#define LIGHT_CSS_SUFFIX \
+ "@define-color popover_bg_color mix(@window_bg_color, white, .1);\n" \
+ "@define-color card_bg_color alpha(white, .6);\n"
+#define DARK_CSS_SUFFIX \
+ "@define-color popover_bg_color mix(@window_bg_color, white, 0.07);\n" \
+ "@define-color card_bg_color @popover_bg_color;\n" \
+ "@define-color view_bg_color darker(@window_bg_color);\n"
+
+enum {
+ FOREGROUND,
+ BACKGROUND,
+};
+
+static gboolean
+get_color (GtkSourceStyleScheme *scheme,
+ const char *style_name,
+ GdkRGBA *color,
+ int kind)
+{
+ GtkSourceStyle *style;
+ g_autofree char *fg = NULL;
+ g_autofree char *bg = NULL;
+ gboolean fg_set = FALSE;
+ gboolean bg_set = FALSE;
+
+ g_assert (GTK_SOURCE_IS_STYLE_SCHEME (scheme));
+ g_assert (style_name != NULL);
+
+ if (!(style = gtk_source_style_scheme_get_style (scheme, style_name)))
+ return FALSE;
+
+ g_object_get (style,
+ "foreground", &fg,
+ "foreground-set", &fg_set,
+ "background", &bg,
+ "background-set", &bg_set,
+ NULL);
+
+ if (kind == FOREGROUND && fg && fg_set)
+ gdk_rgba_parse (color, fg);
+ else if (kind == BACKGROUND && bg && bg_set)
+ gdk_rgba_parse (color, bg);
+ else
+ return FALSE;
+
+ return color->alpha >= .1;
+}
+
+static inline gboolean
+get_foreground (GtkSourceStyleScheme *scheme,
+ const char *style_name,
+ GdkRGBA *fg)
+{
+ return get_color (scheme, style_name, fg, FOREGROUND);
+}
+
+static inline gboolean
+get_background (GtkSourceStyleScheme *scheme,
+ const char *style_name,
+ GdkRGBA *bg)
+{
+ return get_color (scheme, style_name, bg, BACKGROUND);
+}
+
+static gboolean
+get_metadata_color (GtkSourceStyleScheme *scheme,
+ const char *key,
+ GdkRGBA *color)
+{
+ const char *str;
+
+ if ((str = gtk_source_style_scheme_get_metadata (scheme, key)))
+ return gdk_rgba_parse (color, str);
+
+ return FALSE;
+}
+
+static void
+define_color (GString *str,
+ const char *name,
+ const GdkRGBA *color)
+{
+ g_autofree char *color_str = NULL;
+ GdkRGBA opaque;
+
+ g_assert (str != NULL);
+ g_assert (name != NULL);
+ g_assert (color != NULL);
+
+ opaque = *color;
+ opaque.alpha = 1.0f;
+
+ color_str = gdk_rgba_to_string (&opaque);
+ g_string_append_printf (str, "@define-color %s %s;\n", name, color_str);
+}
+
+static void
+define_color_mixed (GString *str,
+ const char *name,
+ const GdkRGBA *a,
+ const GdkRGBA *b,
+ double level)
+{
+ g_autofree char *a_str = NULL;
+ g_autofree char *b_str = NULL;
+ char levelstr[G_ASCII_DTOSTR_BUF_SIZE];
+
+ g_assert (str != NULL);
+ g_assert (name != NULL);
+ g_assert (a != NULL);
+ g_assert (b != NULL);
+
+ a_str = gdk_rgba_to_string (a);
+ b_str = gdk_rgba_to_string (b);
+
+ g_ascii_dtostr (levelstr, sizeof levelstr, level);
+
+ /* truncate */
+ levelstr[6] = 0;
+
+ g_string_append_printf (str, "@define-color %s mix(%s,%s,%s);\n", name, a_str, b_str, levelstr);
+}
+
+#if 0
+static inline void
+premix_colors (GdkRGBA *dest,
+ const GdkRGBA *fg,
+ const GdkRGBA *bg,
+ gboolean bg_set,
+ double alpha)
+{
+ g_assert (dest != NULL);
+ g_assert (fg != NULL);
+ g_assert (bg != NULL || bg_set == FALSE);
+ g_assert (alpha >= 0.0 && alpha <= 1.0);
+
+ if (bg_set)
+ {
+ dest->red = ((1 - alpha) * bg->red) + (alpha * fg->red);
+ dest->green = ((1 - alpha) * bg->green) + (alpha * fg->green);
+ dest->blue = ((1 - alpha) * bg->blue) + (alpha * fg->blue);
+ dest->alpha = 1.0;
+ }
+ else
+ {
+ *dest = *fg;
+ dest->alpha = alpha;
+ }
+}
+#endif
+
+gboolean
+_ide_source_style_scheme_is_dark (GtkSourceStyleScheme *scheme)
+{
+ const char *id = gtk_source_style_scheme_get_id (scheme);
+ const char *variant = gtk_source_style_scheme_get_metadata (scheme, "variant");
+ GdkRGBA text_bg;
+
+ if (g_strcmp0 (variant, "light") == 0)
+ return FALSE;
+ else if (g_strcmp0 (variant, "dark") == 0)
+ return TRUE;
+ else if (strstr (id, "-dark") != NULL)
+ return TRUE;
+
+ if (get_background (scheme, "text", &text_bg))
+ {
+ /* http://alienryderflex.com/hsp.html */
+ double r = text_bg.red * 255.0;
+ double g = text_bg.green * 255.0;
+ double b = text_bg.blue * 255.0;
+ double hsp = sqrt (0.299 * (r * r) +
+ 0.587 * (g * g) +
+ 0.114 * (b * b));
+
+ return hsp <= 127.5;
+ }
+
+ return FALSE;
+}
+
+char *
+_ide_recoloring_generate_css (GtkSourceStyleScheme *style_scheme)
+{
+ static const GdkRGBA black = {0,0,0,1};
+ static const GdkRGBA white = {1,1,1,1};
+ const GdkRGBA *alt;
+ GdkRGBA text_bg;
+ GdkRGBA text_fg;
+ GdkRGBA right_margin;
+ const char *id;
+ const char *name;
+ GString *str;
+ GdkRGBA color;
+ gboolean is_dark;
+ gboolean has_fg;
+ gboolean has_bg;
+
+ g_return_val_if_fail (GTK_SOURCE_IS_STYLE_SCHEME (style_scheme), NULL);
+
+ /* Don't restyle Adwaita as we already have it */
+ id = gtk_source_style_scheme_get_name (style_scheme);
+ if (g_str_has_prefix (id, "Adwaita"))
+ return NULL;
+
+ name = gtk_source_style_scheme_get_name (style_scheme);
+ is_dark = _ide_source_style_scheme_is_dark (style_scheme);
+ alt = is_dark ? &white : &black;
+
+ str = g_string_new (SHARED_CSS);
+ g_string_append_printf (str, "/* %s */\n", name);
+
+ /* TODO: Improve error checking and fallbacks */
+
+ has_bg = get_background (style_scheme, "text", &text_bg);
+ has_fg = get_foreground (style_scheme, "text", &text_fg);
+ get_background (style_scheme, "right-margin", &right_margin);
+ right_margin.alpha = 1;
+
+ if (get_metadata_color (style_scheme, "window_bg_color", &color))
+ define_color (str, "window_bg_color", &color);
+ else if (has_bg && has_fg)
+ define_color (str, "window_bg_color", &text_bg);
+ else if (is_dark)
+ define_color_mixed (str, "window_bg_color", &text_bg, alt, .025);
+ else
+ define_color_mixed (str, "window_bg_color", &text_bg, &white, .1);
+
+ if (get_metadata_color (style_scheme, "window_fg_color", &color))
+ define_color (str, "window_fg_color", &color);
+ else if (has_bg && has_fg)
+ define_color (str, "window_fg_color", &text_fg);
+ else if (is_dark)
+ define_color_mixed (str, "window_fg_color", &text_bg, alt, .05);
+ else
+ define_color_mixed (str, "window_fg_color", &text_bg, alt, .025);
+
+ if (get_metadata_color (style_scheme, "headerbar_bg_color", &color))
+ define_color (str, "headerbar_bg_color", &color);
+ else if (has_bg && has_fg)
+ define_color_mixed (str, "headerbar_bg_color", &text_bg, &text_fg, .05);
+ else if (is_dark)
+ define_color_mixed (str, "headerbar_bg_color", &text_bg, alt, .025);
+ else
+ define_color_mixed (str, "headerbar_bg_color", &text_bg, &white, .1);
+
+ if (get_metadata_color (style_scheme, "headerbar_fg_color", &color))
+ define_color (str, "headerbar_fg_color", &color);
+ else if (has_bg && has_fg)
+ define_color (str, "headerbar_fg_color", &text_fg);
+ else if (is_dark)
+ define_color_mixed (str, "headerbar_fg_color", &text_bg, alt, .05);
+ else
+ define_color_mixed (str, "headerbar_fg_color", &text_bg, alt, .025);
+
+ define_color_mixed (str, "view_bg_color", &text_bg, &white, is_dark ? .1 : .3);
+ define_color (str, "view_fg_color", &text_fg);
+
+ if (get_metadata_color (style_scheme, "accent_bg_color", &color) ||
+ get_background (style_scheme, "selection", &color))
+ define_color (str, "accent_bg_color", &color);
+
+ if (get_metadata_color (style_scheme, "accent_fg_color", &color) ||
+ get_foreground (style_scheme, "selection", &color))
+ define_color (str, "accent_fg_color", &color);
+
+ if (get_metadata_color (style_scheme, "accent_color", &color))
+ {
+ define_color (str, "accent_color", &color);
+ }
+ else if (get_metadata_color (style_scheme, "accent_bg_color", &color) ||
+ get_background (style_scheme, "selection", &color))
+ {
+ color.alpha = 1;
+ define_color_mixed (str, "accent_color", &color, alt, .1);
+ }
+
+ if (is_dark)
+ g_string_append (str, DARK_CSS_SUFFIX);
+ else
+ g_string_append (str, LIGHT_CSS_SUFFIX);
+
+ return g_string_free (str, FALSE);
+}
+
+GtkSourceStyleScheme *
+_ide_source_style_scheme_get_variant (GtkSourceStyleScheme *scheme,
+ const char *variant)
+{
+ GtkSourceStyleSchemeManager *style_scheme_manager;
+ GtkSourceStyleScheme *ret;
+ g_autoptr(GString) str = NULL;
+ g_autofree char *key = NULL;
+ const char *mapping;
+
+ g_return_val_if_fail (GTK_SOURCE_IS_STYLE_SCHEME (scheme), NULL);
+ g_return_val_if_fail (g_strcmp0 (variant, "light") == 0 ||
+ g_strcmp0 (variant, "dark") == 0, NULL);
+
+ style_scheme_manager = gtk_source_style_scheme_manager_get_default ();
+
+ /* If the scheme provides "light-variant" or "dark-variant" metadata,
+ * we will prefer those if the variant is available.
+ */
+ key = g_strdup_printf ("%s-variant", variant);
+ if ((mapping = gtk_source_style_scheme_get_metadata (scheme, key)))
+ {
+ if ((ret = gtk_source_style_scheme_manager_get_scheme (style_scheme_manager, mapping)))
+ return ret;
+ }
+
+ /* Try to find a match by replacing -light/-dark with @variant */
+ str = g_string_new (gtk_source_style_scheme_get_id (scheme));
+
+ if (g_str_has_suffix (str->str, "-light"))
+ g_string_truncate (str, str->len - strlen ("-light"));
+ else if (g_str_has_suffix (str->str, "-dark"))
+ g_string_truncate (str, str->len - strlen ("-dark"));
+
+ g_string_append_printf (str, "-%s", variant);
+
+ /* Look for "Foo-variant" directly */
+ if ((ret = gtk_source_style_scheme_manager_get_scheme (style_scheme_manager, str->str)))
+ return ret;
+
+ /* Look for "Foo" */
+ g_string_truncate (str, str->len - strlen (variant) - 1);
+ if ((ret = gtk_source_style_scheme_manager_get_scheme (style_scheme_manager, str->str)))
+ return ret;
+
+ /* Fallback to what we were provided */
+ return ret;
+}
+
diff --git a/src/libide/gui/meson.build b/src/libide/gui/meson.build
index fb800f21f..d138c6a45 100644
--- a/src/libide/gui/meson.build
+++ b/src/libide/gui/meson.build
@@ -51,6 +51,7 @@ libide_gui_private_headers = [
'ide-notification-view-private.h',
'ide-preferences-builtin-private.h',
'ide-primary-workspace-private.h',
+ 'ide-recoloring-private.h',
'ide-session-private.h',
]
@@ -64,6 +65,7 @@ libide_gui_private_sources = [
'ide-notification-view.c',
'ide-preferences-builtin.c',
'ide-primary-workspace-actions.c',
+ 'ide-recoloring.c',
'ide-session.c',
'ide-workspace-actions.c',
]
@@ -137,6 +139,8 @@ libide_gui_sources += [libide_gui_enums[0]]
#
libide_gui_deps = [
+ cc.find_library('m', required: false),
+
libgio_dep,
libgtk_dep,
libgtksource_dep,
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]