1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
|
#include <math.h>
#include "indicator-image-helper.h"
const gchar * INDICATOR_NAMES_DATA = "indicator-names-data";
static void
refresh_image (GtkImage * image)
{
g_return_if_fail(GTK_IS_IMAGE(image));
GIcon * icon_names = (GIcon *)g_object_get_data(G_OBJECT(image), INDICATOR_NAMES_DATA);
g_return_if_fail(icon_names != NULL);
/* Get the default theme */
GtkIconTheme * default_theme = gtk_icon_theme_get_default();
g_return_if_fail(default_theme != NULL);
gint icon_size = 22;
GtkStyle * style = gtk_widget_get_style(GTK_WIDGET(image));
GValue styleprop = {0};
gtk_style_get_style_property(style, GTK_TYPE_IMAGE, "x-ayatana-indicator-dynamic", &styleprop);
if (G_VALUE_HOLDS_BOOLEAN(&styleprop) && g_value_get_boolean(&styleprop)) {
PangoContext * context = gtk_widget_get_pango_context(GTK_WIDGET(image));
PangoFontMetrics * metrics = pango_context_get_metrics(context, style->font_desc, pango_context_get_language(context));
icon_size = PANGO_PIXELS(pango_font_metrics_get_ascent(metrics)) + PANGO_PIXELS(pango_font_metrics_get_descent(metrics));
g_debug("Looking for icon size %d", icon_size);
pango_font_metrics_unref(metrics);
}
/* Look through the themes for that icon */
GtkIconInfo * icon_info = gtk_icon_theme_lookup_by_gicon(default_theme, icon_names, icon_size, 0);
if (icon_info == NULL) {
g_warning("Unable to find icon in theme.");
return;
}
/* Grab the filename */
const gchar * icon_filename = gtk_icon_info_get_filename(icon_info);
g_return_if_fail(icon_filename != NULL); /* An error because we shouldn't get info without a filename */
/* Build a pixbuf */
GError * error = NULL;
GdkPixbuf * pixbuf = gdk_pixbuf_new_from_file(icon_filename, &error);
gtk_icon_info_free(icon_info);
if (pixbuf == NULL) {
g_error("Unable to load icon from file '%s' because: %s", icon_filename, error == NULL ? "I don't know" : error->message);
return;
}
/* Scale icon if all we get is something too big. */
if (gdk_pixbuf_get_height(pixbuf) > icon_size) {
gfloat scale = (gfloat)icon_size / (gfloat)gdk_pixbuf_get_height(pixbuf);
gint width = round(gdk_pixbuf_get_width(pixbuf) * scale);
GdkPixbuf * scaled = gdk_pixbuf_scale_simple(pixbuf, width, icon_size, GDK_INTERP_BILINEAR);
g_object_unref(G_OBJECT(pixbuf));
pixbuf = scaled;
}
/* Put the pixbuf on the image */
gtk_image_set_from_pixbuf(image, pixbuf);
g_object_unref(G_OBJECT(pixbuf));
return;
}
/* Handles the theme changed signal to refresh the icon to make
sure that it changes appropriately */
static void
theme_changed_cb (GtkIconTheme * theme, gpointer user_data)
{
GtkImage * image = GTK_IMAGE(user_data);
refresh_image(image);
return;
}
/* Removes the signal on the theme that was calling update on this
image. */
static void
image_destroyed_cb (GtkImage * image, gpointer user_data)
{
g_signal_handlers_disconnect_by_func(gtk_icon_theme_get_default(), theme_changed_cb, image);
return;
}
GtkImage *
indicator_image_helper (const gchar * name)
{
g_return_val_if_fail(name != NULL, NULL);
g_return_val_if_fail(name[0] != '\0', NULL);
/* Build us a GIcon */
GIcon * icon_names = g_themed_icon_new_with_default_fallbacks(name);
g_return_val_if_fail(icon_names != NULL, NULL);
/* Build us an image */
GtkImage * image = GTK_IMAGE(gtk_image_new());
if (image == NULL) {
g_error("Unable to create image from pixbuf on icon name '%s'", name);
g_object_unref(icon_names);
return NULL;
}
/* Attach our names to the image */
g_object_set_data_full(G_OBJECT(image), INDICATOR_NAMES_DATA, icon_names, g_object_unref);
/* Put the pixbuf in */
refresh_image(image);
/* Connect to all changes */
g_signal_connect(G_OBJECT(gtk_icon_theme_get_default()), "changed", G_CALLBACK(theme_changed_cb), image);
g_signal_connect(G_OBJECT(image), "destroy", G_CALLBACK(image_destroyed_cb), NULL);
/* Return our built image */
return image;
}
|