2 * geanywraplabel.c - this file is part of Geany, a fast and lightweight IDE
4 * Copyright 2009-2012 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
5 * Copyright 2009-2012 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 * A GtkLabel subclass that can wrap to any width, unlike GtkLabel which has a fixed wrap point.
24 * (inspired by libview's WrapLabel, http://view.sourceforge.net)
31 #include "geanywraplabel.h"
34 struct _GeanyWrapLabelClass
36 GtkLabelClass parent_class
;
43 } GeanyWrapLabelPrivate
;
45 struct _GeanyWrapLabel
48 GeanyWrapLabelPrivate
*priv
;
52 #if GTK_CHECK_VERSION(3, 0, 0)
53 static gboolean
geany_wrap_label_draw(GtkWidget
*widget
, cairo_t
*cr
);
54 static void geany_wrap_label_get_preferred_width (GtkWidget
*widget
,
55 gint
*minimal_width
, gint
*natural_width
);
56 static void geany_wrap_label_get_preferred_height (GtkWidget
*widget
,
57 gint
*minimal_height
, gint
*natural_height
);
58 static void geany_wrap_label_get_preferred_width_for_height (GtkWidget
*widget
,
59 gint height
, gint
*minimal_width
, gint
*natural_width
);
60 static void geany_wrap_label_get_preferred_height_for_width (GtkWidget
*widget
,
61 gint width
, gint
*minimal_height
, gint
*natural_height
);
62 static GtkSizeRequestMode
geany_wrap_label_get_request_mode(GtkWidget
*widget
);
64 static gboolean
geany_wrap_label_expose (GtkWidget
*widget
, GdkEventExpose
*event
);
65 static void geany_wrap_label_size_request (GtkWidget
*widget
, GtkRequisition
*req
);
67 static void geany_wrap_label_size_allocate (GtkWidget
*widget
, GtkAllocation
*alloc
);
68 static void geany_wrap_label_set_wrap_width (GtkWidget
*widget
, gint width
);
69 static void geany_wrap_label_label_notify (GObject
*object
, GParamSpec
*pspec
, gpointer data
);
71 G_DEFINE_TYPE(GeanyWrapLabel
, geany_wrap_label
, GTK_TYPE_LABEL
)
74 static void geany_wrap_label_class_init(GeanyWrapLabelClass
*klass
)
76 GtkWidgetClass
*widget_class
= GTK_WIDGET_CLASS(klass
);
78 widget_class
->size_allocate
= geany_wrap_label_size_allocate
;
79 #if GTK_CHECK_VERSION(3, 0, 0)
80 widget_class
->draw
= geany_wrap_label_draw
;
81 widget_class
->get_preferred_width
= geany_wrap_label_get_preferred_width
;
82 widget_class
->get_preferred_width_for_height
= geany_wrap_label_get_preferred_width_for_height
;
83 widget_class
->get_preferred_height
= geany_wrap_label_get_preferred_height
;
84 widget_class
->get_preferred_height_for_width
= geany_wrap_label_get_preferred_height_for_width
;
85 widget_class
->get_request_mode
= geany_wrap_label_get_request_mode
;
87 widget_class
->size_request
= geany_wrap_label_size_request
;
88 widget_class
->expose_event
= geany_wrap_label_expose
;
91 g_type_class_add_private(klass
, sizeof (GeanyWrapLabelPrivate
));
95 static void geany_wrap_label_init(GeanyWrapLabel
*self
)
97 self
->priv
= G_TYPE_INSTANCE_GET_PRIVATE(self
,
98 GEANY_WRAP_LABEL_TYPE
, GeanyWrapLabelPrivate
);
100 self
->priv
->wrap_width
= 0;
101 self
->priv
->wrap_height
= 0;
103 g_signal_connect(self
, "notify::label", G_CALLBACK(geany_wrap_label_label_notify
), NULL
);
104 gtk_misc_set_alignment(GTK_MISC(self
), 0.0, 0.0);
108 /* Sets the point at which the text should wrap. */
109 static void geany_wrap_label_set_wrap_width(GtkWidget
*widget
, gint width
)
111 GeanyWrapLabel
*self
= GEANY_WRAP_LABEL(widget
);
117 layout
= gtk_label_get_layout(GTK_LABEL(widget
));
120 * We may need to reset the wrap width, so do this regardless of whether
121 * or not we've changed the width.
123 pango_layout_set_width(layout
, width
* PANGO_SCALE
);
124 pango_layout_set_wrap(layout
, PANGO_WRAP_WORD_CHAR
);
125 pango_layout_get_pixel_size(layout
, NULL
, &self
->priv
->wrap_height
);
127 if (self
->priv
->wrap_width
!= width
)
129 self
->priv
->wrap_width
= width
;
130 gtk_widget_queue_resize(widget
);
135 /* updates the wrap width when the label text changes */
136 static void geany_wrap_label_label_notify(GObject
*object
, GParamSpec
*pspec
, gpointer data
)
138 GeanyWrapLabel
*self
= GEANY_WRAP_LABEL(object
);
140 geany_wrap_label_set_wrap_width(GTK_WIDGET(object
), self
->priv
->wrap_width
);
144 #if GTK_CHECK_VERSION(3, 0, 0)
145 /* makes sure the layout is setup for rendering and chains to parent renderer */
146 static gboolean
geany_wrap_label_draw(GtkWidget
*widget
, cairo_t
*cr
)
148 GeanyWrapLabel
*self
= GEANY_WRAP_LABEL(widget
);
149 PangoLayout
*layout
= gtk_label_get_layout(GTK_LABEL(widget
));
151 pango_layout_set_width(layout
, self
->priv
->wrap_width
* PANGO_SCALE
);
152 pango_layout_set_wrap(layout
, PANGO_WRAP_WORD_CHAR
);
154 return (* GTK_WIDGET_CLASS(geany_wrap_label_parent_class
)->draw
)(widget
, cr
);
158 static void geany_wrap_label_get_preferred_width (GtkWidget
*widget
,
159 gint
*minimal_width
, gint
*natural_width
)
161 *minimal_width
= *natural_width
= 0;
165 static void geany_wrap_label_get_preferred_width_for_height (GtkWidget
*widget
,
166 gint height
, gint
*minimal_width
, gint
*natural_width
)
168 PangoLayout
*layout
= gtk_label_get_layout(GTK_LABEL(widget
));;
170 pango_layout_set_height(layout
, height
* PANGO_SCALE
);
171 pango_layout_set_wrap(layout
, PANGO_WRAP_WORD_CHAR
);
172 pango_layout_get_pixel_size(layout
, natural_width
, NULL
);
178 static void geany_wrap_label_get_preferred_height (GtkWidget
*widget
,
179 gint
*minimal_height
, gint
*natural_height
)
181 *minimal_height
= *natural_height
= GEANY_WRAP_LABEL(widget
)->priv
->wrap_height
;
185 static void geany_wrap_label_get_preferred_height_for_width (GtkWidget
*widget
,
186 gint width
, gint
*minimal_height
, gint
*natural_height
)
188 PangoLayout
*layout
= gtk_label_get_layout(GTK_LABEL(widget
));
190 pango_layout_set_width(layout
, width
* PANGO_SCALE
);
191 pango_layout_set_wrap(layout
, PANGO_WRAP_WORD_CHAR
);
192 pango_layout_get_pixel_size(layout
, NULL
, natural_height
);
194 *minimal_height
= *natural_height
;
198 static GtkSizeRequestMode
geany_wrap_label_get_request_mode(GtkWidget
*widget
)
200 return GTK_SIZE_REQUEST_WIDTH_FOR_HEIGHT
;
205 /* makes sure the layout is setup for rendering and chains to parent renderer */
206 static gboolean
geany_wrap_label_expose(GtkWidget
*widget
, GdkEventExpose
*event
)
208 GeanyWrapLabel
*self
= GEANY_WRAP_LABEL(widget
);
209 PangoLayout
*layout
= gtk_label_get_layout(GTK_LABEL(widget
));
211 pango_layout_set_width(layout
, self
->priv
->wrap_width
* PANGO_SCALE
);
212 pango_layout_set_wrap(layout
, PANGO_WRAP_WORD_CHAR
);
214 return (* GTK_WIDGET_CLASS(geany_wrap_label_parent_class
)->expose_event
)(widget
, event
);
218 /* Forces the height to be the size necessary for the Pango layout, while allowing the
219 * width to be flexible. */
220 static void geany_wrap_label_size_request(GtkWidget
*widget
, GtkRequisition
*req
)
223 req
->height
= GEANY_WRAP_LABEL(widget
)->priv
->wrap_height
;
228 /* Sets the wrap width to the width allocated to us. */
229 static void geany_wrap_label_size_allocate(GtkWidget
*widget
, GtkAllocation
*alloc
)
231 (* GTK_WIDGET_CLASS(geany_wrap_label_parent_class
)->size_allocate
)(widget
, alloc
);
233 geany_wrap_label_set_wrap_width(widget
, alloc
->width
);
235 #if GTK_CHECK_VERSION(3, 0, 0)
237 /* ask the parent to recompute our size, because it seems GTK3 size
238 * caching is too aggressive */
239 GtkWidget
*parent
= gtk_widget_get_parent(widget
);
240 if (GTK_IS_CONTAINER(parent
))
241 gtk_container_check_resize(GTK_CONTAINER(parent
));
247 GtkWidget
*geany_wrap_label_new(const gchar
*text
)
249 return g_object_new(GEANY_WRAP_LABEL_TYPE
, "label", text
, NULL
);