[gtk+/gtk-style-context: 222/540] GtkStyleContext: Add gtk_style_context_[add|remove]_provider_for_screen()
- From: Carlos Garnacho <carlosg src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gtk+/gtk-style-context: 222/540] GtkStyleContext: Add gtk_style_context_[add|remove]_provider_for_screen()
- Date: Fri, 3 Dec 2010 02:54:47 +0000 (UTC)
commit 1e0c0168f32d3a3d6b15e9872f3ef6af70d3b2b1
Author: Carlos Garnacho <carlosg gnome org>
Date: Sat Sep 11 12:42:35 2010 +0200
GtkStyleContext: Add gtk_style_context_[add|remove]_provider_for_screen()
These functions allow per-screen style providers. also, the changed signal
will be emitted whenever a style context changes.
gtk/gtkstylecontext.c | 233 ++++++++++++++++++++++++++++++++++++-------------
gtk/gtkstylecontext.h | 6 ++
2 files changed, 178 insertions(+), 61 deletions(-)
---
diff --git a/gtk/gtkstylecontext.c b/gtk/gtkstylecontext.c
index c738e91..106f78a 100644
--- a/gtk/gtkstylecontext.c
+++ b/gtk/gtkstylecontext.c
@@ -28,6 +28,7 @@
#include "gtkthemingengine.h"
#include "gtkintl.h"
#include "gtkwidget.h"
+#include "gtkwindow.h"
#include "gtkprivate.h"
#include "gtkanimationdescription.h"
#include "gtktimeline.h"
@@ -119,6 +120,8 @@ enum {
guint signals[LAST_SIGNAL] = { 0 };
+static GQuark provider_list_quark = 0;
+
static void gtk_style_context_finalize (GObject *object);
static void gtk_style_context_impl_set_property (GObject *object,
@@ -484,24 +487,55 @@ gtk_style_context_impl_get_property (GObject *object,
}
}
+GList *
+find_next_candidate (GList *local,
+ GList *global)
+{
+ if (local && global)
+ {
+ GtkStyleProviderData *local_data, *global_data;
+
+ local_data = local->data;
+ global_data = global->data;
+
+ if (local_data->priority >= global_data->priority)
+ return local;
+ else
+ return global;
+ }
+ else if (local)
+ return local;
+ else if (global)
+ return global;
+
+ return NULL;
+}
+
static void
rebuild_properties (GtkStyleContext *context)
{
GtkStyleContextPrivate *priv;
- GList *list;
+ GList *elem, *list, *global_list = NULL;
priv = context->priv;
list = priv->providers;
+ if (priv->screen)
+ global_list = g_object_get_qdata (G_OBJECT (priv->screen), provider_list_quark);
+
gtk_style_set_clear (priv->store);
- while (list)
+ while ((elem = find_next_candidate (list, global_list)) != NULL)
{
GtkStyleProviderData *data;
GtkStyleSet *provider_style;
- data = list->data;
- list = list->next;
+ data = elem->data;
+
+ if (elem == list)
+ list = list->next;
+ else
+ global_list = global_list->next;
provider_style = gtk_style_provider_get_style (data->provider,
priv->widget_path);
@@ -525,7 +559,7 @@ static void
rebuild_icon_factories (GtkStyleContext *context)
{
GtkStyleContextPrivate *priv;
- GList *providers;
+ GList *elem, *list, *global_list = NULL;
priv = context->priv;
@@ -533,12 +567,26 @@ rebuild_icon_factories (GtkStyleContext *context)
g_slist_free (priv->icon_factories);
priv->icon_factories = NULL;
- for (providers = priv->providers_last; providers; providers = providers->prev)
+ list = priv->providers_last;
+
+ if (priv->screen)
+ {
+ global_list = g_object_get_qdata (G_OBJECT (priv->screen), provider_list_quark);
+ global_list = g_list_last (global_list);
+ }
+
+ while ((elem = find_next_candidate (list, global_list)) != NULL)
{
GtkIconFactory *factory;
GtkStyleProviderData *data;
- data = providers->data;
+ data = elem->data;
+
+ if (elem == list)
+ list = list->prev;
+ else
+ global_list = global_list->prev;
+
factory = gtk_style_provider_get_icon_factory (data->provider,
priv->widget_path);
@@ -547,28 +595,22 @@ rebuild_icon_factories (GtkStyleContext *context)
}
}
-void
-gtk_style_context_add_provider (GtkStyleContext *context,
- GtkStyleProvider *provider,
- guint priority)
+static void
+style_provider_add (GList **list,
+ GtkStyleProvider *provider,
+ guint priority)
{
- GtkStyleContextPrivate *priv;
GtkStyleProviderData *new_data;
gboolean added = FALSE;
- GList *list;
+ GList *l = *list;
- g_return_if_fail (GTK_IS_STYLE_CONTEXT (context));
- g_return_if_fail (GTK_IS_STYLE_PROVIDER (provider));
-
- priv = context->priv;
new_data = style_provider_data_new (provider, priority);
- list = priv->providers;
- while (list)
+ while (l)
{
GtkStyleProviderData *data;
- data = list->data;
+ data = l->data;
/* Provider was already attached to the style
* context, remove in order to add the new data
@@ -577,11 +619,11 @@ gtk_style_context_add_provider (GtkStyleContext *context,
{
GList *link;
- link = list;
- list = list->next;
+ link = l;
+ l = l->next;
/* Remove and free link */
- priv->providers = g_list_remove_link (priv->providers, link);
+ *list = g_list_remove_link (*list, link);
style_provider_data_free (link->data);
g_list_free_1 (link);
@@ -591,24 +633,59 @@ gtk_style_context_add_provider (GtkStyleContext *context,
if (!added &&
data->priority > priority)
{
- priv->providers = g_list_insert_before (priv->providers, list, new_data);
+ *list = g_list_insert_before (*list, l, new_data);
added = TRUE;
}
- list = list->next;
+ l = l->next;
}
if (!added)
- priv->providers = g_list_append (priv->providers, new_data);
+ *list = g_list_append (*list, new_data);
+}
- priv->providers_last = g_list_last (priv->providers);
+static gboolean
+style_provider_remove (GList **list,
+ GtkStyleProvider *provider)
+{
+ GList *l = *list;
- if (priv->widget_path)
+ while (l)
{
- rebuild_properties (context);
- clear_property_cache (context);
- rebuild_icon_factories (context);
+ GtkStyleProviderData *data;
+
+ data = l->data;
+
+ if (data->provider == provider)
+ {
+ *list = g_list_remove_link (*list, l);
+ style_provider_data_free (l->data);
+ g_list_free_1 (l);
+
+ return TRUE;
+ }
+
+ l = l->next;
}
+
+ return FALSE;
+}
+
+void
+gtk_style_context_add_provider (GtkStyleContext *context,
+ GtkStyleProvider *provider,
+ guint priority)
+{
+ GtkStyleContextPrivate *priv;
+
+ g_return_if_fail (GTK_IS_STYLE_CONTEXT (context));
+ g_return_if_fail (GTK_IS_STYLE_PROVIDER (provider));
+
+ priv = context->priv;
+ style_provider_add (&priv->providers, provider, priority);
+ priv->providers_last = g_list_last (priv->providers);
+
+ gtk_style_context_invalidate (context);
}
void
@@ -616,45 +693,81 @@ gtk_style_context_remove_provider (GtkStyleContext *context,
GtkStyleProvider *provider)
{
GtkStyleContextPrivate *priv;
- gboolean removed = FALSE;
- GList *list;
g_return_if_fail (GTK_IS_STYLE_CONTEXT (context));
g_return_if_fail (GTK_IS_STYLE_PROVIDER (provider));
priv = context->priv;
- list = priv->providers;
- while (list)
+ if (style_provider_remove (&priv->providers, provider))
{
- GtkStyleProviderData *data;
+ priv->providers_last = g_list_last (priv->providers);
- data = list->data;
+ gtk_style_context_invalidate (context);
+ }
+}
- if (data->provider == provider)
- {
- priv->providers = g_list_remove_link (priv->providers, list);
- style_provider_data_free (list->data);
- g_list_free_1 (list);
+static void
+reset_toplevels (GdkScreen *screen)
+{
+ GList *list, *toplevels;
- removed = TRUE;
+ toplevels = gtk_window_list_toplevels ();
+ g_list_foreach (toplevels, (GFunc)g_object_ref, NULL);
- break;
- }
+ for (list = toplevels; list; list = list->next)
+ {
+ if (gtk_widget_get_screen (list->data) == screen)
+ gtk_widget_reset_style (list->data);
- list = list->next;
+ g_object_unref (list->data);
}
- if (removed)
+ g_list_free (toplevels);
+}
+
+void
+gtk_style_context_add_provider_for_screen (GdkScreen *screen,
+ GtkStyleProvider *provider,
+ guint priority)
+{
+ GList *providers, *list;
+
+ g_return_if_fail (GDK_IS_SCREEN (screen));
+ g_return_if_fail (GTK_IS_STYLE_PROVIDER (provider));
+
+ if (G_UNLIKELY (!provider_list_quark))
+ provider_list_quark = g_quark_from_static_string ("gtk-provider-list-quark");
+
+ list = providers = g_object_get_qdata (G_OBJECT (screen), provider_list_quark);
+ style_provider_add (&list, provider, priority);
+
+ if (list != providers)
+ g_object_set_qdata (G_OBJECT (screen), provider_list_quark, list);
+
+ reset_toplevels (screen);
+}
+
+void
+gtk_style_context_remove_provider_for_screen (GdkScreen *screen,
+ GtkStyleProvider *provider)
+{
+ GList *providers, *list;
+
+ g_return_if_fail (GDK_IS_SCREEN (screen));
+ g_return_if_fail (GTK_IS_STYLE_PROVIDER (provider));
+
+ if (G_UNLIKELY (!provider_list_quark))
+ return;
+
+ list = providers = g_object_get_qdata (G_OBJECT (screen), provider_list_quark);
+
+ if (style_provider_remove (&list, provider))
{
- priv->providers_last = g_list_last (priv->providers);
+ if (list != providers)
+ g_object_set_qdata (G_OBJECT (screen), provider_list_quark, list);
- if (priv->widget_path)
- {
- rebuild_properties (context);
- clear_property_cache (context);
- rebuild_icon_factories (context);
- }
+ reset_toplevels (screen);
}
}
@@ -833,9 +946,7 @@ gtk_style_context_set_path (GtkStyleContext *context,
if (path)
{
priv->widget_path = gtk_widget_path_copy (path);
- rebuild_properties (context);
- clear_property_cache (context);
- rebuild_icon_factories (context);
+ gtk_style_context_invalidate (context);
}
}
@@ -1040,7 +1151,7 @@ gtk_style_context_set_class (GtkStyleContext *context,
if (priv->widget_path)
{
gtk_widget_path_iter_add_class (priv->widget_path, 0, class_name);
- rebuild_properties (context);
+ gtk_style_context_invalidate (context);
}
}
}
@@ -1074,7 +1185,7 @@ gtk_style_context_unset_class (GtkStyleContext *context,
if (priv->widget_path)
{
gtk_widget_path_iter_remove_class (priv->widget_path, 0, class_name);
- rebuild_properties (context);
+ gtk_style_context_invalidate (context);
}
}
}
@@ -1192,7 +1303,7 @@ gtk_style_context_set_region (GtkStyleContext *context,
if (priv->widget_path)
{
gtk_widget_path_iter_add_region (priv->widget_path, 0, class_name, flags);
- rebuild_properties (context);
+ gtk_style_context_invalidate (context);
}
}
}
@@ -1226,7 +1337,7 @@ gtk_style_context_unset_region (GtkStyleContext *context,
if (priv->widget_path)
{
gtk_widget_path_iter_remove_region (priv->widget_path, 0, class_name);
- rebuild_properties (context);
+ gtk_style_context_invalidate (context);
}
}
}
diff --git a/gtk/gtkstylecontext.h b/gtk/gtkstylecontext.h
index fba6028..07cec5b 100644
--- a/gtk/gtkstylecontext.h
+++ b/gtk/gtkstylecontext.h
@@ -51,6 +51,12 @@ struct GtkStyleContextClass
GType gtk_style_context_get_type (void) G_GNUC_CONST;
+void gtk_style_context_add_provider_for_screen (GdkScreen *screen,
+ GtkStyleProvider *provider,
+ guint priority);
+void gtk_style_context_remove_provider_for_screen (GdkScreen *screen,
+ GtkStyleProvider *provider);
+
void gtk_style_context_add_provider (GtkStyleContext *context,
GtkStyleProvider *provider,
guint priority);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]