forked from AuroraMiddleware/gtk
503e8f40b9
yields more visually stable appearance
740 lines
22 KiB
C
740 lines
22 KiB
C
/* gtkcellrendererprogress.c
|
||
* Copyright (C) 2002 Naba Kumar <kh_naba@users.sourceforge.net>
|
||
* heavily modified by Jörgen Scheibengruber <mfcn@gmx.de>
|
||
* heavily modified by Marco Pesenti Gritti <marco@gnome.org>
|
||
*
|
||
* 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 this library. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2007. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
#include "config.h"
|
||
#include <stdlib.h>
|
||
|
||
#include "gtkcellrendererprogress.h"
|
||
#include "gtkintl.h"
|
||
#include "gtkorientable.h"
|
||
#include "gtkprivate.h"
|
||
#include "gtkrender.h"
|
||
|
||
|
||
/**
|
||
* SECTION:gtkcellrendererprogress
|
||
* @Short_description: Renders numbers as progress bars
|
||
* @Title: GtkCellRendererProgress
|
||
*
|
||
* #GtkCellRendererProgress renders a numeric value as a progress par in a cell.
|
||
* Additionally, it can display a text on top of the progress bar.
|
||
*
|
||
* The #GtkCellRendererProgress cell renderer was added in GTK+ 2.6.
|
||
*/
|
||
|
||
|
||
enum
|
||
{
|
||
PROP_0,
|
||
PROP_VALUE,
|
||
PROP_TEXT,
|
||
PROP_PULSE,
|
||
PROP_TEXT_XALIGN,
|
||
PROP_TEXT_YALIGN,
|
||
PROP_ORIENTATION,
|
||
PROP_INVERTED
|
||
};
|
||
|
||
struct _GtkCellRendererProgressPrivate
|
||
{
|
||
gint value;
|
||
gchar *text;
|
||
gchar *label;
|
||
gint min_h;
|
||
gint min_w;
|
||
gint pulse;
|
||
gint offset;
|
||
gfloat text_xalign;
|
||
gfloat text_yalign;
|
||
GtkOrientation orientation;
|
||
gboolean inverted;
|
||
};
|
||
|
||
static void gtk_cell_renderer_progress_finalize (GObject *object);
|
||
static void gtk_cell_renderer_progress_get_property (GObject *object,
|
||
guint param_id,
|
||
GValue *value,
|
||
GParamSpec *pspec);
|
||
static void gtk_cell_renderer_progress_set_property (GObject *object,
|
||
guint param_id,
|
||
const GValue *value,
|
||
GParamSpec *pspec);
|
||
static void gtk_cell_renderer_progress_set_value (GtkCellRendererProgress *cellprogress,
|
||
gint value);
|
||
static void gtk_cell_renderer_progress_set_text (GtkCellRendererProgress *cellprogress,
|
||
const gchar *text);
|
||
static void gtk_cell_renderer_progress_set_pulse (GtkCellRendererProgress *cellprogress,
|
||
gint pulse);
|
||
static void compute_dimensions (GtkCellRenderer *cell,
|
||
GtkWidget *widget,
|
||
const gchar *text,
|
||
gint *width,
|
||
gint *height);
|
||
static void gtk_cell_renderer_progress_get_size (GtkCellRenderer *cell,
|
||
GtkWidget *widget,
|
||
const GdkRectangle *cell_area,
|
||
gint *x_offset,
|
||
gint *y_offset,
|
||
gint *width,
|
||
gint *height);
|
||
static void gtk_cell_renderer_progress_render (GtkCellRenderer *cell,
|
||
cairo_t *cr,
|
||
GtkWidget *widget,
|
||
const GdkRectangle *background_area,
|
||
const GdkRectangle *cell_area,
|
||
GtkCellRendererState flags);
|
||
|
||
|
||
G_DEFINE_TYPE_WITH_CODE (GtkCellRendererProgress, gtk_cell_renderer_progress, GTK_TYPE_CELL_RENDERER,
|
||
G_ADD_PRIVATE (GtkCellRendererProgress)
|
||
G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL))
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_class_init (GtkCellRendererProgressClass *klass)
|
||
{
|
||
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||
GtkCellRendererClass *cell_class = GTK_CELL_RENDERER_CLASS (klass);
|
||
|
||
object_class->finalize = gtk_cell_renderer_progress_finalize;
|
||
object_class->get_property = gtk_cell_renderer_progress_get_property;
|
||
object_class->set_property = gtk_cell_renderer_progress_set_property;
|
||
|
||
cell_class->get_size = gtk_cell_renderer_progress_get_size;
|
||
cell_class->render = gtk_cell_renderer_progress_render;
|
||
|
||
/**
|
||
* GtkCellRendererProgress:value:
|
||
*
|
||
* The "value" property determines the percentage to which the
|
||
* progress bar will be "filled in".
|
||
*
|
||
* Since: 2.6
|
||
**/
|
||
g_object_class_install_property (object_class,
|
||
PROP_VALUE,
|
||
g_param_spec_int ("value",
|
||
P_("Value"),
|
||
P_("Value of the progress bar"),
|
||
0, 100, 0,
|
||
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY));
|
||
|
||
/**
|
||
* GtkCellRendererProgress:text:
|
||
*
|
||
* The "text" property determines the label which will be drawn
|
||
* over the progress bar. Setting this property to %NULL causes the default
|
||
* label to be displayed. Setting this property to an empty string causes
|
||
* no label to be displayed.
|
||
*
|
||
* Since: 2.6
|
||
**/
|
||
g_object_class_install_property (object_class,
|
||
PROP_TEXT,
|
||
g_param_spec_string ("text",
|
||
P_("Text"),
|
||
P_("Text on the progress bar"),
|
||
NULL,
|
||
GTK_PARAM_READWRITE));
|
||
|
||
/**
|
||
* GtkCellRendererProgress:pulse:
|
||
*
|
||
* Setting this to a non-negative value causes the cell renderer to
|
||
* enter "activity mode", where a block bounces back and forth to
|
||
* indicate that some progress is made, without specifying exactly how
|
||
* much.
|
||
*
|
||
* Each increment of the property causes the block to move by a little
|
||
* bit.
|
||
*
|
||
* To indicate that the activity has not started yet, set the property
|
||
* to zero. To indicate completion, set the property to %G_MAXINT.
|
||
*
|
||
* Since: 2.12
|
||
*/
|
||
g_object_class_install_property (object_class,
|
||
PROP_PULSE,
|
||
g_param_spec_int ("pulse",
|
||
P_("Pulse"),
|
||
P_("Set this to positive values to indicate that some progress is made, but you don't know how much."),
|
||
-1, G_MAXINT, -1,
|
||
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY));
|
||
|
||
/**
|
||
* GtkCellRendererProgress:text-xalign:
|
||
*
|
||
* The "text-xalign" property controls the horizontal alignment of the
|
||
* text in the progress bar. Valid values range from 0 (left) to 1
|
||
* (right). Reserved for RTL layouts.
|
||
*
|
||
* Since: 2.12
|
||
*/
|
||
g_object_class_install_property (object_class,
|
||
PROP_TEXT_XALIGN,
|
||
g_param_spec_float ("text-xalign",
|
||
P_("Text x alignment"),
|
||
P_("The horizontal text alignment, from 0 (left) to 1 (right). Reversed for RTL layouts."),
|
||
0.0, 1.0, 0.5,
|
||
GTK_PARAM_READWRITE));
|
||
|
||
/**
|
||
* GtkCellRendererProgress:text-yalign:
|
||
*
|
||
* The "text-yalign" property controls the vertical alignment of the
|
||
* text in the progress bar. Valid values range from 0 (top) to 1
|
||
* (bottom).
|
||
*
|
||
* Since: 2.12
|
||
*/
|
||
g_object_class_install_property (object_class,
|
||
PROP_TEXT_YALIGN,
|
||
g_param_spec_float ("text-yalign",
|
||
P_("Text y alignment"),
|
||
P_("The vertical text alignment, from 0 (top) to 1 (bottom)."),
|
||
0.0, 1.0, 0.5,
|
||
GTK_PARAM_READWRITE));
|
||
|
||
g_object_class_override_property (object_class,
|
||
PROP_ORIENTATION,
|
||
"orientation");
|
||
|
||
g_object_class_install_property (object_class,
|
||
PROP_INVERTED,
|
||
g_param_spec_boolean ("inverted",
|
||
P_("Inverted"),
|
||
P_("Invert the direction in which the progress bar grows"),
|
||
FALSE,
|
||
GTK_PARAM_READWRITE|G_PARAM_EXPLICIT_NOTIFY));
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_init (GtkCellRendererProgress *cellprogress)
|
||
{
|
||
GtkCellRendererProgressPrivate *priv;
|
||
|
||
cellprogress->priv = gtk_cell_renderer_progress_get_instance_private (cellprogress);
|
||
priv = cellprogress->priv;
|
||
|
||
priv->value = 0;
|
||
priv->text = NULL;
|
||
priv->label = NULL;
|
||
priv->min_w = -1;
|
||
priv->min_h = -1;
|
||
priv->pulse = -1;
|
||
priv->offset = 0;
|
||
|
||
priv->text_xalign = 0.5;
|
||
priv->text_yalign = 0.5;
|
||
|
||
priv->orientation = GTK_ORIENTATION_HORIZONTAL,
|
||
priv->inverted = FALSE;
|
||
}
|
||
|
||
|
||
/**
|
||
* gtk_cell_renderer_progress_new:
|
||
*
|
||
* Creates a new #GtkCellRendererProgress.
|
||
*
|
||
* Returns: the new cell renderer
|
||
*
|
||
* Since: 2.6
|
||
**/
|
||
GtkCellRenderer*
|
||
gtk_cell_renderer_progress_new (void)
|
||
{
|
||
return g_object_new (GTK_TYPE_CELL_RENDERER_PROGRESS, NULL);
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_finalize (GObject *object)
|
||
{
|
||
GtkCellRendererProgress *cellprogress = GTK_CELL_RENDERER_PROGRESS (object);
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
|
||
g_free (priv->text);
|
||
g_free (priv->label);
|
||
|
||
G_OBJECT_CLASS (gtk_cell_renderer_progress_parent_class)->finalize (object);
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_get_property (GObject *object,
|
||
guint param_id,
|
||
GValue *value,
|
||
GParamSpec *pspec)
|
||
{
|
||
GtkCellRendererProgress *cellprogress = GTK_CELL_RENDERER_PROGRESS (object);
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
|
||
switch (param_id)
|
||
{
|
||
case PROP_VALUE:
|
||
g_value_set_int (value, priv->value);
|
||
break;
|
||
case PROP_TEXT:
|
||
g_value_set_string (value, priv->text);
|
||
break;
|
||
case PROP_PULSE:
|
||
g_value_set_int (value, priv->pulse);
|
||
break;
|
||
case PROP_TEXT_XALIGN:
|
||
g_value_set_float (value, priv->text_xalign);
|
||
break;
|
||
case PROP_TEXT_YALIGN:
|
||
g_value_set_float (value, priv->text_yalign);
|
||
break;
|
||
case PROP_ORIENTATION:
|
||
g_value_set_enum (value, priv->orientation);
|
||
break;
|
||
case PROP_INVERTED:
|
||
g_value_set_boolean (value, priv->inverted);
|
||
break;
|
||
default:
|
||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_set_property (GObject *object,
|
||
guint param_id,
|
||
const GValue *value,
|
||
GParamSpec *pspec)
|
||
{
|
||
GtkCellRendererProgress *cellprogress = GTK_CELL_RENDERER_PROGRESS (object);
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
|
||
switch (param_id)
|
||
{
|
||
case PROP_VALUE:
|
||
gtk_cell_renderer_progress_set_value (cellprogress,
|
||
g_value_get_int (value));
|
||
break;
|
||
case PROP_TEXT:
|
||
gtk_cell_renderer_progress_set_text (cellprogress,
|
||
g_value_get_string (value));
|
||
break;
|
||
case PROP_PULSE:
|
||
gtk_cell_renderer_progress_set_pulse (cellprogress,
|
||
g_value_get_int (value));
|
||
break;
|
||
case PROP_TEXT_XALIGN:
|
||
priv->text_xalign = g_value_get_float (value);
|
||
break;
|
||
case PROP_TEXT_YALIGN:
|
||
priv->text_yalign = g_value_get_float (value);
|
||
break;
|
||
case PROP_ORIENTATION:
|
||
if (priv->orientation != g_value_get_enum (value))
|
||
{
|
||
priv->orientation = g_value_get_enum (value);
|
||
g_object_notify_by_pspec (object, pspec);
|
||
}
|
||
break;
|
||
case PROP_INVERTED:
|
||
if (priv->inverted != g_value_get_boolean (value))
|
||
{
|
||
priv->inverted = g_value_get_boolean (value);
|
||
g_object_notify_by_pspec (object, pspec);
|
||
}
|
||
break;
|
||
default:
|
||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
|
||
}
|
||
}
|
||
|
||
static void
|
||
recompute_label (GtkCellRendererProgress *cellprogress)
|
||
{
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
gchar *label;
|
||
|
||
if (priv->text)
|
||
label = g_strdup (priv->text);
|
||
else if (priv->pulse < 0)
|
||
label = g_strdup_printf (C_("progress bar label", "%d %%"), priv->value);
|
||
else
|
||
label = NULL;
|
||
|
||
g_free (priv->label);
|
||
priv->label = label;
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_set_value (GtkCellRendererProgress *cellprogress,
|
||
gint value)
|
||
{
|
||
if (cellprogress->priv->value != value)
|
||
{
|
||
cellprogress->priv->value = value;
|
||
recompute_label (cellprogress);
|
||
g_object_notify (G_OBJECT (cellprogress), "value");
|
||
}
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_set_text (GtkCellRendererProgress *cellprogress,
|
||
const gchar *text)
|
||
{
|
||
gchar *new_text;
|
||
|
||
new_text = g_strdup (text);
|
||
g_free (cellprogress->priv->text);
|
||
cellprogress->priv->text = new_text;
|
||
recompute_label (cellprogress);
|
||
g_object_notify (G_OBJECT (cellprogress), "text");
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_set_pulse (GtkCellRendererProgress *cellprogress,
|
||
gint pulse)
|
||
{
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
|
||
if (pulse != priv->pulse)
|
||
{
|
||
if (pulse <= 0)
|
||
priv->offset = 0;
|
||
else
|
||
priv->offset = pulse;
|
||
g_object_notify (G_OBJECT (cellprogress), "pulse");
|
||
}
|
||
|
||
priv->pulse = pulse;
|
||
recompute_label (cellprogress);
|
||
}
|
||
|
||
static void
|
||
set_layout_tnum (PangoLayout *layout)
|
||
{
|
||
PangoAttrList* attrs = pango_attr_list_new ();
|
||
PangoAttribute* attribute_font_features = pango_attr_font_features_new ("tnum 1");
|
||
pango_attr_list_insert (attrs, attribute_font_features);
|
||
pango_layout_set_attributes (layout, attrs);
|
||
pango_attr_list_unref (attrs);
|
||
}
|
||
|
||
static void
|
||
compute_dimensions (GtkCellRenderer *cell,
|
||
GtkWidget *widget,
|
||
const gchar *text,
|
||
gint *width,
|
||
gint *height)
|
||
{
|
||
PangoRectangle logical_rect;
|
||
PangoLayout *layout;
|
||
gint xpad, ypad;
|
||
|
||
layout = gtk_widget_create_pango_layout (widget, text);
|
||
set_layout_tnum (layout);
|
||
pango_layout_get_pixel_extents (layout, NULL, &logical_rect);
|
||
|
||
gtk_cell_renderer_get_padding (cell, &xpad, &ypad);
|
||
|
||
if (width)
|
||
*width = logical_rect.width + xpad * 2;
|
||
|
||
if (height)
|
||
*height = logical_rect.height + ypad * 2;
|
||
|
||
g_object_unref (layout);
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_get_size (GtkCellRenderer *cell,
|
||
GtkWidget *widget,
|
||
const GdkRectangle *cell_area,
|
||
gint *x_offset,
|
||
gint *y_offset,
|
||
gint *width,
|
||
gint *height)
|
||
{
|
||
GtkCellRendererProgress *cellprogress = GTK_CELL_RENDERER_PROGRESS (cell);
|
||
GtkCellRendererProgressPrivate *priv = cellprogress->priv;
|
||
gint w, h;
|
||
gchar *text;
|
||
|
||
if (priv->min_w < 0)
|
||
{
|
||
text = g_strdup_printf (C_("progress bar label", "%d %%"), 100);
|
||
compute_dimensions (cell, widget, text,
|
||
&priv->min_w,
|
||
&priv->min_h);
|
||
g_free (text);
|
||
}
|
||
|
||
compute_dimensions (cell, widget, priv->label, &w, &h);
|
||
|
||
if (width)
|
||
*width = MAX (priv->min_w, w);
|
||
|
||
if (height)
|
||
*height = MIN (priv->min_h, h);
|
||
|
||
/* FIXME: at the moment cell_area is only set when we are requesting
|
||
* the size for drawing the focus rectangle. We now just return
|
||
* the last size we used for drawing the progress bar, which will
|
||
* work for now. Not a really nice solution though.
|
||
*/
|
||
if (cell_area)
|
||
{
|
||
if (width)
|
||
*width = cell_area->width;
|
||
if (height)
|
||
*height = cell_area->height;
|
||
}
|
||
|
||
if (x_offset) *x_offset = 0;
|
||
if (y_offset) *y_offset = 0;
|
||
}
|
||
|
||
static inline gint
|
||
get_bar_size (gint pulse,
|
||
gint value,
|
||
gint full_size)
|
||
{
|
||
gint bar_size;
|
||
|
||
if (pulse < 0)
|
||
bar_size = full_size * MAX (0, value) / 100;
|
||
else if (pulse == 0)
|
||
bar_size = 0;
|
||
else if (pulse == G_MAXINT)
|
||
bar_size = full_size;
|
||
else
|
||
bar_size = MAX (2, full_size / 5);
|
||
|
||
return bar_size;
|
||
}
|
||
|
||
static inline gint
|
||
get_bar_position (gint start,
|
||
gint full_size,
|
||
gint bar_size,
|
||
gint pulse,
|
||
gint offset,
|
||
gboolean is_rtl)
|
||
{
|
||
gint position;
|
||
|
||
if (pulse < 0 || pulse == 0 || pulse == G_MAXINT)
|
||
{
|
||
position = is_rtl ? (start + full_size - bar_size) : start;
|
||
}
|
||
else
|
||
{
|
||
position = (is_rtl ? offset + 12 : offset) % 24;
|
||
if (position > 12)
|
||
position = 24 - position;
|
||
position = start + full_size * position / 15;
|
||
}
|
||
|
||
return position;
|
||
}
|
||
|
||
static void
|
||
gtk_cell_renderer_progress_render (GtkCellRenderer *cell,
|
||
cairo_t *cr,
|
||
GtkWidget *widget,
|
||
const GdkRectangle *background_area,
|
||
const GdkRectangle *cell_area,
|
||
GtkCellRendererState flags)
|
||
{
|
||
GtkCellRendererProgress *cellprogress = GTK_CELL_RENDERER_PROGRESS (cell);
|
||
GtkCellRendererProgressPrivate *priv= cellprogress->priv;
|
||
GtkStyleContext *context;
|
||
GtkBorder padding;
|
||
PangoLayout *layout;
|
||
PangoRectangle logical_rect;
|
||
gint x, y, w, h, x_pos, y_pos, bar_position, bar_size, start, full_size;
|
||
gint xpad, ypad;
|
||
GdkRectangle clip;
|
||
gboolean is_rtl;
|
||
|
||
context = gtk_widget_get_style_context (widget);
|
||
is_rtl = gtk_widget_get_direction (widget) == GTK_TEXT_DIR_RTL;
|
||
|
||
gtk_cell_renderer_get_padding (cell, &xpad, &ypad);
|
||
x = cell_area->x + xpad;
|
||
y = cell_area->y + ypad;
|
||
w = cell_area->width - xpad * 2;
|
||
h = cell_area->height - ypad * 2;
|
||
|
||
gtk_style_context_save (context);
|
||
gtk_style_context_add_class (context, GTK_STYLE_CLASS_TROUGH);
|
||
|
||
gtk_render_background (context, cr, x, y, w, h);
|
||
gtk_render_frame (context, cr, x, y, w, h);
|
||
|
||
gtk_style_context_get_padding (context, gtk_style_context_get_state (context), &padding);
|
||
|
||
x += padding.left;
|
||
y += padding.top;
|
||
w -= padding.left + padding.right;
|
||
h -= padding.top + padding.bottom;
|
||
|
||
gtk_style_context_restore (context);
|
||
|
||
if (priv->orientation == GTK_ORIENTATION_HORIZONTAL)
|
||
{
|
||
clip.y = y;
|
||
clip.height = h;
|
||
|
||
start = x;
|
||
full_size = w;
|
||
|
||
bar_size = get_bar_size (priv->pulse, priv->value, full_size);
|
||
|
||
if (!priv->inverted)
|
||
bar_position = get_bar_position (start, full_size, bar_size,
|
||
priv->pulse, priv->offset, is_rtl);
|
||
else
|
||
bar_position = get_bar_position (start, full_size, bar_size,
|
||
priv->pulse, priv->offset, !is_rtl);
|
||
|
||
clip.width = bar_size;
|
||
clip.x = bar_position;
|
||
}
|
||
else
|
||
{
|
||
clip.x = x;
|
||
clip.width = w;
|
||
|
||
start = y;
|
||
full_size = h;
|
||
|
||
bar_size = get_bar_size (priv->pulse, priv->value, full_size);
|
||
|
||
if (priv->inverted)
|
||
bar_position = get_bar_position (start, full_size, bar_size,
|
||
priv->pulse, priv->offset, TRUE);
|
||
else
|
||
bar_position = get_bar_position (start, full_size, bar_size,
|
||
priv->pulse, priv->offset, FALSE);
|
||
|
||
clip.height = bar_size;
|
||
clip.y = bar_position;
|
||
}
|
||
|
||
if (bar_size > 0)
|
||
{
|
||
gtk_style_context_save (context);
|
||
gtk_style_context_add_class (context, GTK_STYLE_CLASS_PROGRESSBAR);
|
||
|
||
gtk_render_background (context, cr, clip.x, clip.y, clip.width, clip.height);
|
||
gtk_render_frame (context, cr, clip.x, clip.y, clip.width, clip.height);
|
||
|
||
gtk_style_context_restore (context);
|
||
}
|
||
|
||
if (priv->label)
|
||
{
|
||
gfloat text_xalign;
|
||
|
||
layout = gtk_widget_create_pango_layout (widget, priv->label);
|
||
set_layout_tnum (layout);
|
||
pango_layout_get_pixel_extents (layout, NULL, &logical_rect);
|
||
|
||
if (gtk_widget_get_direction (widget) != GTK_TEXT_DIR_LTR)
|
||
text_xalign = 1.0 - priv->text_xalign;
|
||
else
|
||
text_xalign = priv->text_xalign;
|
||
|
||
x_pos = x + padding.left + text_xalign *
|
||
(w - padding.left - padding.right - logical_rect.width);
|
||
|
||
y_pos = y + padding.top + priv->text_yalign *
|
||
(h - padding.top - padding.bottom - logical_rect.height);
|
||
|
||
cairo_save (cr);
|
||
gdk_cairo_rectangle (cr, &clip);
|
||
cairo_clip (cr);
|
||
|
||
gtk_style_context_save (context);
|
||
gtk_style_context_add_class (context, GTK_STYLE_CLASS_PROGRESSBAR);
|
||
|
||
gtk_render_layout (context, cr,
|
||
x_pos, y_pos,
|
||
layout);
|
||
|
||
gtk_style_context_restore (context);
|
||
cairo_restore (cr);
|
||
|
||
gtk_style_context_save (context);
|
||
gtk_style_context_add_class (context, GTK_STYLE_CLASS_TROUGH);
|
||
|
||
if (bar_position > start)
|
||
{
|
||
if (priv->orientation == GTK_ORIENTATION_HORIZONTAL)
|
||
{
|
||
clip.x = x;
|
||
clip.width = bar_position - x;
|
||
}
|
||
else
|
||
{
|
||
clip.y = y;
|
||
clip.height = bar_position - y;
|
||
}
|
||
|
||
cairo_save (cr);
|
||
gdk_cairo_rectangle (cr, &clip);
|
||
cairo_clip (cr);
|
||
|
||
gtk_render_layout (context, cr,
|
||
x_pos, y_pos,
|
||
layout);
|
||
|
||
cairo_restore (cr);
|
||
}
|
||
|
||
if (bar_position + bar_size < start + full_size)
|
||
{
|
||
if (priv->orientation == GTK_ORIENTATION_HORIZONTAL)
|
||
{
|
||
clip.x = bar_position + bar_size;
|
||
clip.width = x + w - (bar_position + bar_size);
|
||
}
|
||
else
|
||
{
|
||
clip.y = bar_position + bar_size;
|
||
clip.height = y + h - (bar_position + bar_size);
|
||
}
|
||
|
||
cairo_save (cr);
|
||
gdk_cairo_rectangle (cr, &clip);
|
||
cairo_clip (cr);
|
||
|
||
gtk_render_layout (context, cr,
|
||
x_pos, y_pos,
|
||
layout);
|
||
|
||
cairo_restore (cr);
|
||
}
|
||
|
||
gtk_style_context_restore (context);
|
||
g_object_unref (layout);
|
||
}
|
||
}
|