[gnome-builder] libide-gui: add libadwaita recoloring support



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]