mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-14 14:20:21 +00:00
7799fbf239
This is a subclass of GEmblemedIcon that can show a number or short string as an emblem, overlayed on top of another emblem. Written by Cosimo Cecchi https://bugzilla.gnome.org/show_bug.cgi?id=637169
1049 lines
29 KiB
C
1049 lines
29 KiB
C
/*
|
|
* gtknumerableicon.c: an emblemed icon with number emblems
|
|
*
|
|
* Copyright (C) 2010 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with the Gnome Library; see the file COPYING.LIB. If not,
|
|
* write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Authors: Cosimo Cecchi <cosimoc@redhat.com>
|
|
*/
|
|
|
|
/**
|
|
* SECTION:gtknumerableicon
|
|
* @Title: GtkNumerableIcon
|
|
* @Short_description: A GIcon that allows numbered emblems
|
|
*
|
|
* GtkNumerableIcon is a subclass of #GEmblemedIcon that can
|
|
* show a number or short string as an emblem. The number can
|
|
* be overlayed on top of another emblem, if desired.
|
|
*
|
|
* It supports theming by taking font and color information
|
|
* from a provided #GtkStyleContext; see
|
|
* gtk_numerable_icon_set_style_context().
|
|
*
|
|
* <example>
|
|
* <title>Typical numerable icons</title>
|
|
* <inlinegraphic fileref="numerableicon.png" format="PNG"/>
|
|
* <inlinegraphic fileref="numerableicon2.png" format="PNG"/>
|
|
* </example>
|
|
*/
|
|
#include <config.h>
|
|
|
|
#include "gtknumerableicon.h"
|
|
|
|
#include "gtkicontheme.h"
|
|
#include "gtkintl.h"
|
|
#include "gtkwidget.h"
|
|
#include "gtkwindow.h"
|
|
|
|
#include <gdk/gdk.h>
|
|
#include <pango/pango.h>
|
|
#include <math.h>
|
|
|
|
struct _GtkNumerableIconPrivate {
|
|
gint count;
|
|
gchar *label;
|
|
|
|
GIcon *background_icon;
|
|
gchar *background_icon_name;
|
|
gint icon_size;
|
|
|
|
GdkRGBA *background;
|
|
GdkRGBA *foreground;
|
|
|
|
PangoFontDescription *font;
|
|
cairo_pattern_t *background_image;
|
|
gint border_size;
|
|
|
|
GtkStyleContext *style;
|
|
gulong style_changed_id;
|
|
|
|
gchar *rendered_string;
|
|
};
|
|
|
|
enum {
|
|
PROP_COUNT = 1,
|
|
PROP_LABEL,
|
|
PROP_STYLE,
|
|
PROP_BACKGROUND_ICON,
|
|
PROP_BACKGROUND_ICON_NAME,
|
|
NUM_PROPERTIES
|
|
};
|
|
|
|
#define DEFAULT_SURFACE_SIZE 256
|
|
#define DEFAULT_BORDER_SIZE DEFAULT_SURFACE_SIZE * 0.06
|
|
#define DEFAULT_RADIUS DEFAULT_SURFACE_SIZE / 2
|
|
|
|
#define DEFAULT_BACKGROUND "#000000"
|
|
#define DEFAULT_FOREGROUND "#ffffff"
|
|
|
|
static GParamSpec *properties[NUM_PROPERTIES] = { NULL, };
|
|
|
|
G_DEFINE_TYPE (GtkNumerableIcon, gtk_numerable_icon, G_TYPE_EMBLEMED_ICON);
|
|
|
|
static gint
|
|
get_surface_size (cairo_surface_t *surface)
|
|
{
|
|
return MAX (cairo_image_surface_get_width (surface), cairo_image_surface_get_height (surface));
|
|
}
|
|
|
|
static gdouble
|
|
get_border_size (GtkNumerableIcon *self)
|
|
{
|
|
return self->priv->border_size;
|
|
}
|
|
|
|
static cairo_surface_t *
|
|
draw_default_surface (GtkNumerableIcon *self)
|
|
{
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
|
|
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
|
|
DEFAULT_SURFACE_SIZE, DEFAULT_SURFACE_SIZE);
|
|
|
|
cr = cairo_create (surface);
|
|
|
|
cairo_arc (cr, DEFAULT_SURFACE_SIZE / 2., DEFAULT_SURFACE_SIZE / 2.,
|
|
DEFAULT_RADIUS, 0., 2 * M_PI);
|
|
|
|
gdk_cairo_set_source_rgba (cr, self->priv->background);
|
|
cairo_fill (cr);
|
|
|
|
cairo_arc (cr, DEFAULT_SURFACE_SIZE / 2., DEFAULT_SURFACE_SIZE / 2.,
|
|
DEFAULT_RADIUS - DEFAULT_BORDER_SIZE, 0., 2 * M_PI);
|
|
gdk_cairo_set_source_rgba (cr, self->priv->foreground);
|
|
cairo_fill (cr);
|
|
|
|
cairo_arc (cr, DEFAULT_SURFACE_SIZE / 2., DEFAULT_SURFACE_SIZE / 2.,
|
|
DEFAULT_RADIUS - 2 * DEFAULT_BORDER_SIZE, 0., 2 * M_PI);
|
|
gdk_cairo_set_source_rgba (cr, self->priv->background);
|
|
cairo_fill (cr);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
return surface;
|
|
}
|
|
|
|
static cairo_surface_t *
|
|
draw_from_gradient (cairo_pattern_t *pattern)
|
|
{
|
|
cairo_surface_t *surface;
|
|
cairo_matrix_t matrix;
|
|
cairo_t *cr;
|
|
|
|
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
|
|
DEFAULT_SURFACE_SIZE, DEFAULT_SURFACE_SIZE);
|
|
|
|
cr = cairo_create (surface);
|
|
|
|
/* scale the gradient points to the user space coordinates */
|
|
cairo_matrix_init_scale (&matrix,
|
|
1. / (double) DEFAULT_SURFACE_SIZE,
|
|
1. / (double) DEFAULT_SURFACE_SIZE);
|
|
cairo_pattern_set_matrix (pattern, &matrix);
|
|
|
|
cairo_arc (cr, DEFAULT_SURFACE_SIZE / 2., DEFAULT_SURFACE_SIZE / 2.,
|
|
DEFAULT_RADIUS, 0., 2 * M_PI);
|
|
|
|
cairo_set_source (cr, pattern);
|
|
cairo_fill (cr);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
return surface;
|
|
}
|
|
|
|
/* copy the surface */
|
|
static cairo_surface_t *
|
|
draw_from_image (cairo_surface_t *image)
|
|
{
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
|
|
surface = cairo_surface_create_similar (image, CAIRO_CONTENT_COLOR_ALPHA,
|
|
cairo_image_surface_get_width (image),
|
|
cairo_image_surface_get_height (image));
|
|
cr = cairo_create (surface);
|
|
|
|
cairo_set_source_surface (cr, image, 0, 0);
|
|
cairo_paint (cr);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
return surface;
|
|
}
|
|
|
|
static cairo_surface_t *
|
|
draw_from_gicon (GtkNumerableIcon *self)
|
|
{
|
|
GtkIconTheme *theme;
|
|
GdkScreen *screen;
|
|
GtkIconInfo *info;
|
|
GdkPixbuf *pixbuf;
|
|
cairo_surface_t *surface;
|
|
cairo_t *cr;
|
|
|
|
if (self->priv->style != NULL)
|
|
{
|
|
screen = gtk_style_context_get_screen (self->priv->style);
|
|
theme = gtk_icon_theme_get_for_screen (screen);
|
|
}
|
|
else
|
|
{
|
|
theme = gtk_icon_theme_get_default ();
|
|
}
|
|
|
|
info = gtk_icon_theme_lookup_by_gicon (theme, self->priv->background_icon,
|
|
self->priv->icon_size,
|
|
GTK_ICON_LOOKUP_GENERIC_FALLBACK);
|
|
if (info == NULL)
|
|
return NULL;
|
|
|
|
pixbuf = gtk_icon_info_load_icon (info, NULL);
|
|
gtk_icon_info_free (info);
|
|
|
|
if (pixbuf == NULL)
|
|
return NULL;
|
|
|
|
surface = cairo_image_surface_create (CAIRO_FORMAT_ARGB32,
|
|
gdk_pixbuf_get_width (pixbuf),
|
|
gdk_pixbuf_get_height (pixbuf));
|
|
|
|
cr = cairo_create (surface);
|
|
|
|
gdk_cairo_set_source_pixbuf (cr, pixbuf, 0, 0);
|
|
cairo_paint (cr);
|
|
|
|
cairo_destroy (cr);
|
|
g_object_unref (pixbuf);
|
|
|
|
return surface;
|
|
}
|
|
|
|
static cairo_surface_t *
|
|
get_image_surface (GtkNumerableIcon *self)
|
|
{
|
|
cairo_surface_t *retval = NULL, *image;
|
|
|
|
if (self->priv->background_icon != NULL)
|
|
{
|
|
retval = draw_from_gicon (self);
|
|
self->priv->border_size = 0;
|
|
}
|
|
else if (self->priv->background_image != NULL)
|
|
{
|
|
if (cairo_pattern_get_surface (self->priv->background_image, &image) == CAIRO_STATUS_SUCCESS)
|
|
retval = draw_from_image (image);
|
|
else
|
|
retval = draw_from_gradient (self->priv->background_image);
|
|
|
|
self->priv->border_size = 0;
|
|
}
|
|
|
|
if (retval == NULL)
|
|
{
|
|
retval = draw_default_surface (self);
|
|
self->priv->border_size = DEFAULT_BORDER_SIZE;
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
static PangoLayout *
|
|
get_pango_layout (GtkNumerableIcon *self)
|
|
{
|
|
PangoContext *context;
|
|
GdkScreen *screen;
|
|
PangoLayout *layout;
|
|
|
|
if (self->priv->style != NULL)
|
|
{
|
|
screen = gtk_style_context_get_screen (self->priv->style);
|
|
context = gdk_pango_context_get_for_screen (screen);
|
|
layout = pango_layout_new (context);
|
|
|
|
if (self->priv->font != NULL)
|
|
pango_layout_set_font_description (layout, self->priv->font);
|
|
|
|
pango_layout_set_text (layout, self->priv->rendered_string, -1);
|
|
|
|
g_object_unref (context);
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *fake;
|
|
|
|
/* steal gtk text settings from the window */
|
|
fake = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
layout = gtk_widget_create_pango_layout (fake, self->priv->rendered_string);
|
|
gtk_widget_destroy (fake);
|
|
}
|
|
|
|
return layout;
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_ensure_emblem (GtkNumerableIcon *self)
|
|
{
|
|
cairo_t *cr;
|
|
cairo_surface_t *surface;
|
|
PangoLayout *layout;
|
|
GEmblem *emblem;
|
|
gint width, height;
|
|
gdouble scale;
|
|
PangoAttrList *attr_list;
|
|
PangoAttribute *attr;
|
|
GdkPixbuf *pixbuf;
|
|
|
|
/* don't draw anything if the count is zero */
|
|
if (self->priv->rendered_string == NULL)
|
|
{
|
|
g_emblemed_icon_clear_emblems (G_EMBLEMED_ICON (self));
|
|
return;
|
|
}
|
|
|
|
surface = get_image_surface (self);
|
|
cr = cairo_create (surface);
|
|
|
|
layout = get_pango_layout (self);
|
|
pango_layout_get_pixel_size (layout, &width, &height);
|
|
|
|
/* scale the layout to be 0.75 of the size still available for drawing */
|
|
scale = ((get_surface_size (surface) - 2 * get_border_size (self)) * 0.75) / (MAX (height, width));
|
|
attr_list = pango_attr_list_new ();
|
|
|
|
attr = pango_attr_scale_new (scale);
|
|
pango_attr_list_insert (attr_list, attr);
|
|
|
|
attr = pango_attr_weight_new (PANGO_WEIGHT_BOLD);
|
|
pango_attr_list_insert (attr_list, attr);
|
|
|
|
pango_layout_set_attributes (layout, attr_list);
|
|
|
|
/* update these values */
|
|
pango_layout_get_pixel_size (layout, &width, &height);
|
|
|
|
/* move to the center */
|
|
cairo_move_to (cr,
|
|
get_surface_size (surface) / 2. - (gdouble) width / 2.,
|
|
get_surface_size (surface) / 2. - (gdouble) height / 2.);
|
|
|
|
gdk_cairo_set_source_rgba (cr, self->priv->foreground);
|
|
pango_cairo_show_layout (cr, layout);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
pixbuf =
|
|
gdk_pixbuf_get_from_surface (surface, 0, 0,
|
|
get_surface_size (surface), get_surface_size (surface));
|
|
|
|
emblem = g_emblem_new (G_ICON (pixbuf));
|
|
g_emblemed_icon_clear_emblems (G_EMBLEMED_ICON (self));
|
|
g_emblemed_icon_add_emblem (G_EMBLEMED_ICON (self), emblem);
|
|
|
|
g_object_unref (layout);
|
|
g_object_unref (emblem);
|
|
g_object_unref (pixbuf);
|
|
|
|
cairo_surface_destroy (surface);
|
|
pango_attr_list_unref (attr_list);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_update_properties_from_style (GtkNumerableIcon *self)
|
|
{
|
|
GtkStyleContext *style = self->priv->style;
|
|
GtkWidgetPath *path, *saved;
|
|
cairo_pattern_t *pattern = NULL;
|
|
GdkRGBA background, foreground;
|
|
PangoFontDescription *font = NULL;
|
|
|
|
/* save an unmodified copy of the original widget path, in order
|
|
* to restore it later */
|
|
path = gtk_widget_path_copy (gtk_style_context_get_path (style));
|
|
saved = gtk_widget_path_copy (path);
|
|
|
|
if (!gtk_widget_path_is_type (path, GTK_TYPE_NUMERABLE_ICON))
|
|
{
|
|
/* append our GType to the style context to fetch appropriate colors */
|
|
gtk_widget_path_append_type (path, GTK_TYPE_NUMERABLE_ICON);
|
|
gtk_style_context_set_path (style, path);
|
|
}
|
|
|
|
gtk_style_context_get_background_color (style, gtk_style_context_get_state (style),
|
|
&background);
|
|
gtk_style_context_get_color (style, gtk_style_context_get_state (style),
|
|
&foreground);
|
|
|
|
if (self->priv->background != NULL)
|
|
gdk_rgba_free (self->priv->background);
|
|
|
|
self->priv->background = gdk_rgba_copy (&background);
|
|
|
|
if (self->priv->foreground != NULL)
|
|
gdk_rgba_free (self->priv->foreground);
|
|
|
|
self->priv->foreground = gdk_rgba_copy (&foreground);
|
|
|
|
gtk_style_context_get (style, gtk_style_context_get_state (style),
|
|
GTK_STYLE_PROPERTY_BACKGROUND_IMAGE, &pattern,
|
|
NULL);
|
|
|
|
if (pattern != NULL)
|
|
{
|
|
if (self->priv->background_image != NULL)
|
|
cairo_pattern_destroy (self->priv->background_image);
|
|
|
|
self->priv->background_image = pattern;
|
|
}
|
|
|
|
gtk_style_context_get (style, gtk_style_context_get_state (style),
|
|
GTK_STYLE_PROPERTY_FONT, &font,
|
|
NULL);
|
|
|
|
if (font != NULL)
|
|
{
|
|
if (self->priv->font != NULL)
|
|
pango_font_description_free (self->priv->font);
|
|
|
|
self->priv->font = font;
|
|
}
|
|
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
|
|
/* restore original widget path */
|
|
gtk_style_context_set_path (style, saved);
|
|
|
|
gtk_widget_path_free (path);
|
|
gtk_widget_path_free (saved);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_init_style (GtkNumerableIcon *self)
|
|
{
|
|
GtkStyleContext *style = self->priv->style;
|
|
|
|
if (style == NULL)
|
|
return;
|
|
|
|
gtk_numerable_icon_update_properties_from_style (self);
|
|
|
|
self->priv->style_changed_id =
|
|
g_signal_connect_swapped (style, "changed",
|
|
G_CALLBACK (gtk_numerable_icon_update_properties_from_style), self);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_ensure_and_replace_label (GtkNumerableIcon *self,
|
|
gint count,
|
|
const gchar *label)
|
|
{
|
|
g_assert (!(label != NULL && count != 0));
|
|
|
|
g_free (self->priv->rendered_string);
|
|
self->priv->rendered_string = NULL;
|
|
|
|
if (count != 0)
|
|
{
|
|
if (self->priv->label != NULL)
|
|
{
|
|
g_free (self->priv->label);
|
|
self->priv->label = NULL;
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_LABEL]);
|
|
}
|
|
|
|
if (count > 99)
|
|
count = 99;
|
|
|
|
if (count < -99)
|
|
count = -99;
|
|
|
|
self->priv->count = count;
|
|
|
|
/* Translators: the format here is used to build the string that will be rendered
|
|
* in the number emblem.
|
|
*/
|
|
self->priv->rendered_string = g_strdup_printf (C_("Number format", "%d"), count);
|
|
|
|
return;
|
|
}
|
|
|
|
if (label != NULL)
|
|
{
|
|
if (self->priv->count != 0)
|
|
{
|
|
self->priv->count = 0;
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_COUNT]);
|
|
}
|
|
|
|
g_free (self->priv->label);
|
|
|
|
if (g_strcmp0 (label, "") == 0)
|
|
{
|
|
self->priv->label = NULL;
|
|
return;
|
|
}
|
|
|
|
self->priv->label = g_strdup (label);
|
|
self->priv->rendered_string = g_strdup (label);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
real_set_background_icon (GtkNumerableIcon *self,
|
|
GIcon *icon)
|
|
{
|
|
if (!g_icon_equal (self->priv->background_icon, icon))
|
|
{
|
|
g_clear_object (&self->priv->background_icon);
|
|
|
|
if (icon != NULL)
|
|
self->priv->background_icon = g_object_ref (icon);
|
|
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_constructed (GObject *object)
|
|
{
|
|
GtkNumerableIcon *self = GTK_NUMERABLE_ICON (object);
|
|
|
|
if (G_OBJECT_CLASS (gtk_numerable_icon_parent_class)->constructed != NULL)
|
|
G_OBJECT_CLASS (gtk_numerable_icon_parent_class)->constructed (object);
|
|
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_finalize (GObject *object)
|
|
{
|
|
GtkNumerableIcon *self = GTK_NUMERABLE_ICON (object);
|
|
|
|
g_free (self->priv->label);
|
|
g_free (self->priv->rendered_string);
|
|
|
|
gdk_rgba_free (self->priv->background);
|
|
gdk_rgba_free (self->priv->foreground);
|
|
|
|
pango_font_description_free (self->priv->font);
|
|
|
|
cairo_pattern_destroy (self->priv->background_image);
|
|
|
|
G_OBJECT_CLASS (gtk_numerable_icon_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_dispose (GObject *object)
|
|
{
|
|
GtkNumerableIcon *self = GTK_NUMERABLE_ICON (object);
|
|
|
|
if (self->priv->style_changed_id != 0)
|
|
{
|
|
g_signal_handler_disconnect (self->priv->style,
|
|
self->priv->style_changed_id);
|
|
self->priv->style_changed_id = 0;
|
|
}
|
|
|
|
g_clear_object (&self->priv->style);
|
|
g_clear_object (&self->priv->background_icon);
|
|
|
|
G_OBJECT_CLASS (gtk_numerable_icon_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkNumerableIcon *self = GTK_NUMERABLE_ICON (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_COUNT:
|
|
gtk_numerable_icon_set_count (self, g_value_get_int (value));
|
|
break;
|
|
case PROP_LABEL:
|
|
gtk_numerable_icon_set_label (self, g_value_get_string (value));
|
|
break;
|
|
case PROP_STYLE:
|
|
gtk_numerable_icon_set_style_context (self, g_value_get_object (value));
|
|
break;
|
|
case PROP_BACKGROUND_ICON:
|
|
gtk_numerable_icon_set_background_gicon (self, g_value_get_object (value));
|
|
break;
|
|
case PROP_BACKGROUND_ICON_NAME:
|
|
gtk_numerable_icon_set_background_icon_name (self, g_value_get_string (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkNumerableIcon *self = GTK_NUMERABLE_ICON (object);
|
|
|
|
switch (property_id)
|
|
{
|
|
case PROP_COUNT:
|
|
g_value_set_int (value, self->priv->count);
|
|
break;
|
|
case PROP_LABEL:
|
|
g_value_set_string (value, self->priv->label);
|
|
break;
|
|
case PROP_STYLE:
|
|
g_value_set_object (value, self->priv->style);
|
|
break;
|
|
case PROP_BACKGROUND_ICON:
|
|
if (self->priv->background_icon != NULL)
|
|
g_value_set_object (value, self->priv->background_icon);
|
|
break;
|
|
case PROP_BACKGROUND_ICON_NAME:
|
|
g_value_set_string (value, self->priv->background_icon_name);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_class_init (GtkNumerableIconClass *klass)
|
|
{
|
|
GObjectClass *oclass = G_OBJECT_CLASS (klass);
|
|
|
|
oclass->get_property = gtk_numerable_icon_get_property;
|
|
oclass->set_property = gtk_numerable_icon_set_property;
|
|
oclass->constructed = gtk_numerable_icon_constructed;
|
|
oclass->dispose = gtk_numerable_icon_dispose;
|
|
oclass->finalize = gtk_numerable_icon_finalize;
|
|
|
|
g_type_class_add_private (klass, sizeof (GtkNumerableIconPrivate));
|
|
|
|
properties[PROP_COUNT] =
|
|
g_param_spec_int ("count",
|
|
P_("Icon's count"),
|
|
P_("The count of the emblem currently displayed"),
|
|
-99, 99, 0,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
properties[PROP_LABEL] =
|
|
g_param_spec_string ("label",
|
|
P_("Icon's label"),
|
|
P_("The label to be displayed over the icon"),
|
|
NULL,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
properties[PROP_STYLE] =
|
|
g_param_spec_object ("style-context",
|
|
P_("Icon's style context"),
|
|
P_("The style context to theme the icon appearance"),
|
|
GTK_TYPE_STYLE_CONTEXT,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
properties[PROP_BACKGROUND_ICON] =
|
|
g_param_spec_object ("background-icon",
|
|
P_("Background icon"),
|
|
P_("The icon for the number emblem background"),
|
|
G_TYPE_ICON,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
properties[PROP_BACKGROUND_ICON_NAME] =
|
|
g_param_spec_string ("background-icon-name",
|
|
P_("Background icon name"),
|
|
P_("The icon name for the number emblem background"),
|
|
NULL,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (oclass, NUM_PROPERTIES, properties);
|
|
}
|
|
|
|
static void
|
|
gtk_numerable_icon_init (GtkNumerableIcon *self)
|
|
{
|
|
GdkRGBA bg;
|
|
GdkRGBA fg;
|
|
|
|
self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self,
|
|
GTK_TYPE_NUMERABLE_ICON,
|
|
GtkNumerableIconPrivate);
|
|
|
|
gdk_rgba_parse (&bg, DEFAULT_BACKGROUND);
|
|
gdk_rgba_parse (&fg, DEFAULT_FOREGROUND);
|
|
|
|
self->priv->background = gdk_rgba_copy (&bg);
|
|
self->priv->foreground = gdk_rgba_copy (&fg);
|
|
|
|
self->priv->icon_size = 48;
|
|
}
|
|
|
|
/* private */
|
|
void
|
|
_gtk_numerable_icon_set_background_icon_size (GtkNumerableIcon *self,
|
|
gint icon_size)
|
|
{
|
|
if (self->priv->background_icon == NULL)
|
|
return;
|
|
|
|
if (self->priv->icon_size != icon_size)
|
|
{
|
|
self->priv->icon_size = icon_size;
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_get_label:
|
|
* @self: a #GtkNumerableIcon
|
|
*
|
|
* Returns the currently displayed label of the icon, or %NULL.
|
|
*
|
|
* Returns: the currently displayed label
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
const gchar *
|
|
gtk_numerable_icon_get_label (GtkNumerableIcon *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_NUMERABLE_ICON (self), NULL);
|
|
|
|
return self->priv->label;
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_set_label:
|
|
* @self: a #GtkNumerableIcon
|
|
* @label: (allow-none): a short label, or %NULL
|
|
*
|
|
* Sets the currently displayed value of @self to the string
|
|
* in @label. Setting an empty label removes the emblem.
|
|
*
|
|
* Note that this is meant for displaying short labels, such as
|
|
* roman numbers, or single letters. For roman numbers, consider
|
|
* using the Unicode characters U+2160 - U+217F. Strings longer
|
|
* than two characters will likely not be rendered very well.
|
|
*
|
|
* If this method is called, and a number was already set on the
|
|
* icon, it will automatically be reset to zero before rendering
|
|
* the label, i.e. the last method called between
|
|
* gtk_numerable_icon_set_label() and gtk_numerable_icon_set_count()
|
|
* has always priority.
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
void
|
|
gtk_numerable_icon_set_label (GtkNumerableIcon *self,
|
|
const gchar *label)
|
|
{
|
|
g_return_if_fail (GTK_IS_NUMERABLE_ICON (self));
|
|
|
|
if (g_strcmp0 (label, self->priv->label) != 0)
|
|
{
|
|
gtk_numerable_icon_ensure_and_replace_label (self, 0, label);
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_LABEL]);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_get_count:
|
|
* @self: a #GtkNumerableIcon
|
|
*
|
|
* Returns the value currently displayed by @self.
|
|
*
|
|
* Returns: the currently displayed value
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
gint
|
|
gtk_numerable_icon_get_count (GtkNumerableIcon *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_NUMERABLE_ICON (self), 0);
|
|
|
|
return self->priv->count;
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_set_count:
|
|
* @self: a #GtkNumerableIcon
|
|
* @count: a number between -99 and 99
|
|
*
|
|
* Sets the currently displayed value of @self to @count.
|
|
*
|
|
* The numeric value is always clamped to make it two digits, i.e.
|
|
* between -99 and 99. Setting a count of zero removes the emblem.
|
|
* If this method is called, and a label was already set on the icon,
|
|
* it will automatically be reset to %NULL before rendering the number,
|
|
* i.e. the last method called between gtk_numerable_icon_set_count()
|
|
* and gtk_numerable_icon_set_label() has always priority.
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
void
|
|
gtk_numerable_icon_set_count (GtkNumerableIcon *self,
|
|
gint count)
|
|
{
|
|
g_return_if_fail (GTK_IS_NUMERABLE_ICON (self));
|
|
|
|
if (count != self->priv->count)
|
|
{
|
|
gtk_numerable_icon_ensure_and_replace_label (self, count, NULL);
|
|
gtk_numerable_icon_ensure_emblem (self);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_COUNT]);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_get_style_context:
|
|
* @self: a #GtkNumerableIcon
|
|
*
|
|
* Returns the #GtkStyleContext used by the icon for theming,
|
|
* or %NULL if there's none.
|
|
*
|
|
* Returns: (transfer none): a #GtkStyleContext, or %NULL.
|
|
* This object is internal to GTK+ and should not be unreffed.
|
|
* Use g_object_ref() if you want to keep it around
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
GtkStyleContext *
|
|
gtk_numerable_icon_get_style_context (GtkNumerableIcon *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_NUMERABLE_ICON (self), NULL);
|
|
|
|
return self->priv->style;
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_set_style_context:
|
|
* @self: a #GtkNumerableIcon
|
|
* @style: a #GtkStyleContext
|
|
*
|
|
* Updates the icon to fetch theme information from the
|
|
* given #GtkStyleContext.
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
void
|
|
gtk_numerable_icon_set_style_context (GtkNumerableIcon *self,
|
|
GtkStyleContext *style)
|
|
{
|
|
g_return_if_fail (GTK_IS_NUMERABLE_ICON (self));
|
|
g_return_if_fail (GTK_IS_STYLE_CONTEXT (style));
|
|
|
|
if (style != self->priv->style)
|
|
{
|
|
if (self->priv->style_changed_id != 0)
|
|
g_signal_handler_disconnect (self->priv->style,
|
|
self->priv->style_changed_id);
|
|
|
|
if (self->priv->style != NULL)
|
|
g_object_unref (self->priv->style);
|
|
|
|
self->priv->style = g_object_ref (style);
|
|
|
|
gtk_numerable_icon_init_style (self);
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_STYLE]);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_set_background_gicon:
|
|
* @self: a #GtkNumerableIcon
|
|
* @icon: (allow-none): a #GIcon, or %NULL
|
|
*
|
|
* Updates the icon to use @icon as the base background image.
|
|
* If @icon is %NULL, @self will go back using style information
|
|
* or default theming for its background image.
|
|
*
|
|
* If this method is called and an icon name was already set as
|
|
* background for the icon, @icon will be used, i.e. the last method
|
|
* called between gtk_numerable_icon_set_background_gicon() and
|
|
* #gtk_numerable_icon_set_background_icon_name() has always priority.
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
void
|
|
gtk_numerable_icon_set_background_gicon (GtkNumerableIcon *self,
|
|
GIcon *icon)
|
|
{
|
|
gboolean res;
|
|
|
|
g_return_if_fail (GTK_IS_NUMERABLE_ICON (self));
|
|
|
|
if (self->priv->background_icon_name != NULL)
|
|
{
|
|
g_free (self->priv->background_icon_name);
|
|
self->priv->background_icon_name = NULL;
|
|
}
|
|
|
|
res = real_set_background_icon (self, icon);
|
|
|
|
if (res)
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_BACKGROUND_ICON]);
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_get_background_gicon:
|
|
* @self: a #GtkNumerableIcon
|
|
*
|
|
* Returns the #GIcon that was set as the base background image, or
|
|
* %NULL if there's none. The caller of this function does not own
|
|
* a reference to the returned #GIcon.
|
|
*
|
|
* Returns: (transfer none): a #GIcon, or %NULL
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
GIcon *
|
|
gtk_numerable_icon_get_background_gicon (GtkNumerableIcon *self)
|
|
{
|
|
GIcon *retval = NULL;
|
|
|
|
g_return_val_if_fail (GTK_IS_NUMERABLE_ICON (self), NULL);
|
|
|
|
/* return the GIcon only if it wasn't created from an icon name */
|
|
if (self->priv->background_icon_name == NULL)
|
|
retval = self->priv->background_icon;
|
|
|
|
return retval;
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_set_background_icon_name:
|
|
* @self: a #GtkNumerableIcon
|
|
* @icon_name: (allow-none): an icon name, or %NULL
|
|
*
|
|
* Updates the icon to use the icon named @icon_name from the
|
|
* current icon theme as the base background image. If @icon_name
|
|
* is %NULL, @self will go back using style information or default
|
|
* theming for its background image.
|
|
*
|
|
* If this method is called and a #GIcon was already set as
|
|
* background for the icon, @icon_name will be used, i.e. the
|
|
* last method called between gtk_numerable_icon_set_background_icon_name()
|
|
* and gtk_numerable_icon_set_background_gicon() has always priority.
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
void
|
|
gtk_numerable_icon_set_background_icon_name (GtkNumerableIcon *self,
|
|
const gchar *icon_name)
|
|
{
|
|
GIcon *icon = NULL;
|
|
gboolean res;
|
|
|
|
g_return_if_fail (GTK_IS_NUMERABLE_ICON (self));
|
|
|
|
if (g_strcmp0 (icon_name, self->priv->background_icon_name) != 0)
|
|
{
|
|
g_free (self->priv->background_icon_name);
|
|
self->priv->background_icon_name = g_strdup (icon_name);
|
|
}
|
|
|
|
if (icon_name != NULL)
|
|
icon = g_themed_icon_new_with_default_fallbacks (icon_name);
|
|
|
|
res = real_set_background_icon (self, icon);
|
|
|
|
if (res)
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_BACKGROUND_ICON_NAME]);
|
|
|
|
if (icon != NULL)
|
|
g_object_unref (icon);
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_get_background_icon_name:
|
|
* @self: a #GtkNumerableIcon
|
|
*
|
|
* Returns the icon name used as the base background image,
|
|
* or %NULL if there's none.
|
|
*
|
|
* Returns: an icon name, or %NULL
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
const gchar *
|
|
gtk_numerable_icon_get_background_icon_name (GtkNumerableIcon *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_NUMERABLE_ICON (self), NULL);
|
|
|
|
return self->priv->background_icon_name;
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_new:
|
|
* @base_icon: a #GIcon to overlay on
|
|
*
|
|
* Creates a new unthemed #GtkNumerableIcon.
|
|
*
|
|
* Returns: a new #GIcon
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
GIcon *
|
|
gtk_numerable_icon_new (GIcon *base_icon)
|
|
{
|
|
g_return_val_if_fail (G_IS_ICON (base_icon), NULL);
|
|
|
|
return g_object_new (GTK_TYPE_NUMERABLE_ICON,
|
|
"gicon", base_icon,
|
|
NULL);
|
|
}
|
|
|
|
/**
|
|
* gtk_numerable_icon_new_with_style_context:
|
|
* @base_icon: a #GIcon to overlay on
|
|
* @context: a #GtkStyleContext
|
|
*
|
|
* Creates a new #GtkNumerableIcon which will themed according
|
|
* to the passed #GtkStyleContext. This is a convenience constructor
|
|
* that calls gtk_numerable_icon_set_style_context() internally.
|
|
*
|
|
* Returns: a new #GIcon
|
|
*
|
|
* Since: 3.0
|
|
*/
|
|
GIcon *
|
|
gtk_numerable_icon_new_with_style_context (GIcon *base_icon,
|
|
GtkStyleContext *context)
|
|
{
|
|
g_return_val_if_fail (G_IS_ICON (base_icon), NULL);
|
|
|
|
return g_object_new (GTK_TYPE_NUMERABLE_ICON,
|
|
"gicon", base_icon,
|
|
"style-context", context,
|
|
NULL);
|
|
}
|