[gnome-builder] libide-gui: add libadwaita recoloring support
- From: Christian Hergert <chergert src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-builder] libide-gui: add libadwaita recoloring support
- Date: Tue, 12 Jul 2022 06:39:12 +0000 (UTC)
commit cd8855beffaa00dab279b6ee08a45408a7caf716
Author: Christian Hergert <chergert redhat com>
Date: Mon Jul 11 21:37:30 2022 -0700
libide-gui: add libadwaita recoloring support
src/libide/gui/ide-recoloring-private.h | 32 ++++
src/libide/gui/ide-recoloring.c | 290 ++++++++++++++++++++++++++++++++
src/libide/gui/meson.build | 2 +
3 files changed, 324 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..897d3531b
--- /dev/null
+++ b/src/libide/gui/ide-recoloring.c
@@ -0,0 +1,290 @@
+/* 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 <libide-sourceview.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, .025);\n" \
+ "@define-color card_bg_color mix(@window_bg_color, white, .1);\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
+
+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);
+}
diff --git a/src/libide/gui/meson.build b/src/libide/gui/meson.build
index 77f16c7a1..f79caf0fc 100644
--- a/src/libide/gui/meson.build
+++ b/src/libide/gui/meson.build
@@ -47,6 +47,7 @@ libide_gui_private_headers = [
'ide-notification-stack-private.h',
'ide-notification-view-private.h',
'ide-preferences-builtin-private.h',
+ 'ide-recoloring-private.h',
'ide-session-private.h',
]
@@ -60,6 +61,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',
]
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]