Put function name in the changelog.
[hildon] / src / hildon-bread-crumb-widget.c
1 /*
2  * This file is a part of hildon
3  *
4  * Copyright (C) 2007 Nokia Corporation, all rights reserved.
5  *
6  * Contact: Michael Dominic Kostrzewa <michael.kostrzewa@nokia.com>
7  * Author: Xan Lopez <xan.lopez@nokia.com>
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Lesser General Public License
11  * as published by the Free Software Foundation; version 2.1 of
12  * the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful, but
15  * WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17  * Lesser General Public License for more details.
18  *
19  * You should have received a copy of the GNU Lesser General Public
20  * License along with this library; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
22  * 02110-1301 USA
23  *
24  */
25
26 #include "hildon-bread-crumb-widget.h"
27 #include "hildon-defines.h"
28
29 #define HILDON_BREAD_CRUMB_WIDGET_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), HILDON_TYPE_BREAD_CRUMB_WIDGET, HildonBreadCrumbWidgetPrivate))
30
31 struct _HildonBreadCrumbWidgetPrivate
32 {
33   GtkWidget *hbox;
34   GtkWidget *label;
35   GtkWidget *icon;
36   GtkWidget *arrow;
37   gchar *text;
38
39   GtkPositionType icon_position;
40   gboolean constructed;
41   gboolean show_separator;
42 };
43
44 /* Signals */
45
46 enum {
47   LAST_SIGNAL
48 };
49
50 /* Properties */
51
52 enum {
53   PROP_0,
54   PROP_TEXT,
55   PROP_ICON,
56   PROP_ICON_POSITION,
57   PROP_SHOW_SEPARATOR
58 };
59
60 /*
61 static guint bread_crumb_item_signals[LAST_SIGNAL] = { 0 };
62 */
63
64 /* GType methods */
65
66 static void hildon_bread_crumb_widget_finalize (GObject *object);
67 static void hildon_bread_crumb_widget_set_property (GObject *object, guint prop_id,
68                                              const GValue *value, GParamSpec *pspec);
69 static void hildon_bread_crumb_widget_get_property (GObject *object, guint prop_id,
70                                              GValue *value, GParamSpec *pspec);
71 static GObject* hildon_bread_crumb_widget_constructor (GType                  type,
72                                                        guint                  n_construct_properties,
73                                                        GObjectConstructParam *construct_params);
74 static void hildon_bread_crumb_widget_set_contents (HildonBreadCrumbWidget *bread_crumb);
75
76 static void hildon_bread_crumb_widget_clicked (GtkButton *button);
77
78 static void hildon_bread_crumb_widget_bread_crumb_init (HildonBreadCrumbIface *iface);
79
80 static void hildon_bread_crumb_widget_get_natural_size (HildonBreadCrumb *bread_crumb,
81                                                         gint *width,
82                                                         gint *height);
83
84 G_DEFINE_TYPE_WITH_CODE (HildonBreadCrumbWidget, hildon_bread_crumb_widget, GTK_TYPE_BUTTON,
85                          G_IMPLEMENT_INTERFACE (HILDON_TYPE_BREAD_CRUMB,
86                                                 hildon_bread_crumb_widget_bread_crumb_init))
87
88 static void
89 hildon_bread_crumb_widget_class_init (HildonBreadCrumbWidgetClass *klass)
90 {
91   GObjectClass *gobject_class = (GObjectClass*)klass;
92   GtkButtonClass *button_class = (GtkButtonClass*)klass;
93
94     /* GObject signals */
95   gobject_class->constructor = hildon_bread_crumb_widget_constructor;
96   gobject_class->finalize = hildon_bread_crumb_widget_finalize;
97   gobject_class->set_property = hildon_bread_crumb_widget_set_property;
98   gobject_class->get_property = hildon_bread_crumb_widget_get_property;
99
100   /* GtkButton signals */
101   button_class->clicked = hildon_bread_crumb_widget_clicked;
102
103   /* Properties */
104   g_object_class_install_property (gobject_class,
105                                    PROP_TEXT,
106                                    g_param_spec_string ("text",
107                                                         "Text",
108                                                         "Text of the label widget inside the bread crumb",
109                                                         NULL,
110                                                         G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
111
112   g_object_class_install_property (gobject_class,
113                                    PROP_ICON,
114                                    g_param_spec_object ("icon",
115                                                         "Icon",
116                                                         "Image that will appear next to the bread crumb text",
117                                                         GTK_TYPE_WIDGET,
118                                                         G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
119
120   g_object_class_install_property (gobject_class,
121                                    PROP_ICON_POSITION,
122                                    g_param_spec_enum ("icon-position",
123                                                       "Icon position",
124                                                       "The position of the image relative to the text",
125                                                       GTK_TYPE_POSITION_TYPE,
126                                                       GTK_POS_LEFT,
127                                                       G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
128
129   g_object_class_install_property (gobject_class,
130                                    PROP_SHOW_SEPARATOR,
131                                    g_param_spec_boolean ("show-separator",
132                                                          "Show separator",
133                                                          "Show the separator attached to the item",
134                                                          TRUE,
135                                                          G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
136
137   /* Private data */
138   g_type_class_add_private (gobject_class, sizeof (HildonBreadCrumbWidgetPrivate));
139 }
140
141 static void
142 hildon_bread_crumb_widget_bread_crumb_init (HildonBreadCrumbIface *iface)
143 {
144   iface->get_natural_size = hildon_bread_crumb_widget_get_natural_size;
145 }
146
147 static GObject*
148 hildon_bread_crumb_widget_constructor (GType type,
149                                        guint n_construct_properties,
150                                        GObjectConstructParam *construct_params)
151 {
152   GObject *object;
153   HildonBreadCrumbWidget *bread_crumb;
154   HildonBreadCrumbWidgetPrivate *priv;
155
156   object = (* G_OBJECT_CLASS (hildon_bread_crumb_widget_parent_class)->constructor) (type,
157                                                                               n_construct_properties,
158                                                                               construct_params);
159
160   bread_crumb = HILDON_BREAD_CRUMB_WIDGET (object);
161   priv = bread_crumb->priv;
162   priv->constructed = TRUE;
163
164   priv->hbox = gtk_hbox_new (FALSE, 6);
165   gtk_container_add (GTK_CONTAINER (bread_crumb), priv->hbox);
166
167   /* Separator */
168   priv->arrow = gtk_arrow_new (GTK_ARROW_RIGHT, GTK_SHADOW_NONE);
169   gtk_widget_set_name (priv->arrow, "hildon-bread-crumb-separator-arrow");
170   gtk_box_pack_start (GTK_BOX (priv->hbox), priv->arrow, FALSE, FALSE, 0);
171
172   if (priv->show_separator)
173     gtk_widget_show (priv->arrow);
174
175   /* Contents base container */
176   bread_crumb->contents = gtk_hbox_new (FALSE, HILDON_MARGIN_DEFAULT);
177   gtk_container_add (GTK_CONTAINER (priv->hbox), bread_crumb->contents);
178   gtk_widget_show (bread_crumb->contents);
179
180   if (priv->text || priv->icon)
181     hildon_bread_crumb_widget_set_contents (bread_crumb);
182
183   /* Show everything */
184   gtk_widget_show (priv->hbox);
185   
186   return object;
187 }
188
189 static void
190 hildon_bread_crumb_widget_init (HildonBreadCrumbWidget *item)
191 {
192   HildonBreadCrumbWidgetPrivate *priv = HILDON_BREAD_CRUMB_WIDGET_GET_PRIVATE (item);
193
194   item->priv = priv;
195
196   item->contents = NULL;
197
198   priv->constructed = FALSE;
199   priv->text = NULL;
200   priv->icon = NULL;
201   priv->icon_position = GTK_POS_LEFT;
202   priv->show_separator = TRUE;
203 }
204
205 static void
206 hildon_bread_crumb_widget_finalize (GObject *object)
207 {
208   HildonBreadCrumbWidgetPrivate *priv = HILDON_BREAD_CRUMB_WIDGET (object)->priv;
209
210   g_free (priv->text);
211
212   G_OBJECT_CLASS (hildon_bread_crumb_widget_parent_class)->finalize (object);
213 }
214
215 static void
216 hildon_bread_crumb_widget_clicked (GtkButton *button)
217 {
218   hildon_bread_crumb_activated (HILDON_BREAD_CRUMB (button));
219 }
220
221 static void
222 hildon_bread_crumb_widget_set_contents (HildonBreadCrumbWidget *bread_crumb)
223 {
224   GtkWidget *icon = NULL;
225   HildonBreadCrumbWidgetPrivate *priv = bread_crumb->priv;
226
227   if (!priv->constructed)
228     return;
229
230   if (!priv->text && !priv->icon)
231     return;
232
233   /* If the icon exists, keep it */
234   if (priv->icon)
235     {
236       icon = g_object_ref (priv->icon);
237       if (icon->parent)
238         gtk_container_remove (GTK_CONTAINER (icon->parent), icon);
239       priv->icon = NULL;
240     }
241
242   /* Reset contents */
243   if (bread_crumb->contents)
244     gtk_container_remove (GTK_CONTAINER (priv->hbox),
245                           bread_crumb->contents);
246
247   if (icon)
248     {
249       priv->icon = icon;
250       if (priv->icon_position == GTK_POS_LEFT ||
251           priv->icon_position == GTK_POS_RIGHT)
252           bread_crumb->contents = gtk_hbox_new (FALSE, HILDON_MARGIN_DEFAULT);
253       else
254           bread_crumb->contents = gtk_vbox_new (FALSE, HILDON_MARGIN_DEFAULT);
255
256       if (priv->icon_position == GTK_POS_LEFT ||
257           priv->icon_position == GTK_POS_TOP)
258         gtk_box_pack_start (GTK_BOX (bread_crumb->contents), priv->icon,
259                             FALSE, FALSE, 0);
260       else
261         gtk_box_pack_end (GTK_BOX (bread_crumb->contents), priv->icon,
262                           FALSE, FALSE, 0);
263         
264       if (priv->text)
265         {
266           priv->label = gtk_label_new (priv->text);
267           g_object_set (G_OBJECT (priv->label), "xalign", 0.0, NULL);
268           gtk_label_set_ellipsize (GTK_LABEL (priv->label),
269                                    PANGO_ELLIPSIZE_END);
270
271           if (priv->icon_position == GTK_POS_RIGHT ||
272               priv->icon_position == GTK_POS_BOTTOM)
273             gtk_box_pack_start (GTK_BOX (bread_crumb->contents), priv->label,
274                                 TRUE, TRUE, 0);
275           else
276             gtk_box_pack_end (GTK_BOX (bread_crumb->contents), priv->label,
277                               TRUE, TRUE, 0);
278           
279         }
280
281       gtk_box_pack_start (GTK_BOX (priv->hbox), bread_crumb->contents,
282                           TRUE, TRUE, 0);
283       gtk_widget_show_all (bread_crumb->contents);
284
285       g_object_unref (icon);
286     }
287   else
288     {
289       /* Only text */
290       bread_crumb->contents = gtk_hbox_new (FALSE, 0);
291       gtk_box_pack_start (GTK_BOX (priv->hbox), bread_crumb->contents,
292                           TRUE, TRUE, 0);
293
294       priv->label = gtk_label_new (priv->text);
295       g_object_set (G_OBJECT (priv->label), "xalign", 0.0, NULL);
296       gtk_label_set_ellipsize (GTK_LABEL (priv->label),
297                                PANGO_ELLIPSIZE_END);
298       gtk_box_pack_start (GTK_BOX (bread_crumb->contents), priv->label, TRUE, TRUE, 0);
299
300       gtk_widget_show_all (bread_crumb->contents);
301     }
302 }
303
304 static void
305 hildon_bread_crumb_widget_set_property (GObject *object, guint prop_id,
306                                         const GValue *value, GParamSpec *pspec)
307 {
308   HildonBreadCrumbWidget *item = HILDON_BREAD_CRUMB_WIDGET (object);
309
310   switch (prop_id)
311     {
312     case PROP_TEXT:
313       _hildon_bread_crumb_widget_set_text (item, g_value_get_string (value));
314       break;
315     case PROP_ICON:
316       _hildon_bread_crumb_widget_set_icon (item, (GtkWidget*)g_value_get_object (value));
317       break;
318     case PROP_ICON_POSITION:
319       _hildon_bread_crumb_widget_set_icon_position (item, g_value_get_enum (value));
320       break;
321     case PROP_SHOW_SEPARATOR:
322       _hildon_bread_crumb_widget_set_show_separator (item, g_value_get_boolean (value));
323       break;
324     default:
325       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
326       break;
327     }
328 }
329
330 static void
331 hildon_bread_crumb_widget_get_property (GObject *object, guint prop_id,
332                                         GValue *value, GParamSpec *pspec)
333 {
334   HildonBreadCrumbWidget *item = HILDON_BREAD_CRUMB_WIDGET (object);
335
336   switch (prop_id)
337     {
338     case PROP_TEXT:
339       g_value_set_string (value, item->priv->text);
340       break;
341     case PROP_ICON:
342       g_value_set_object (value, (GObject *)item->priv->icon);
343       break;
344     case PROP_ICON_POSITION:
345       g_value_set_enum (value, item->priv->icon_position);
346       break;
347     case PROP_SHOW_SEPARATOR:
348       g_value_set_boolean (value, item->priv->show_separator);
349       break;
350     default:
351       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
352       break;
353     }
354 }
355
356 void
357 _hildon_bread_crumb_widget_set_text (HildonBreadCrumbWidget *item,
358                                      const gchar *text)
359 {
360   HildonBreadCrumbWidgetPrivate *priv;
361
362   g_return_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (item));
363
364   priv = item->priv;
365
366   if (priv->text)
367     g_free (priv->text);
368
369   priv->text = g_strdup (text);
370
371   hildon_bread_crumb_widget_set_contents (item);
372
373   g_object_notify (G_OBJECT (item), "text");
374 }
375
376 const gchar*
377 _hildon_bread_crumb_widget_get_text (HildonBreadCrumbWidget *item)
378 {
379   HildonBreadCrumbWidgetPrivate *priv;
380
381   g_return_val_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (item), NULL);
382
383   priv = item->priv;
384
385   return priv->text;
386 }
387
388 void
389 _hildon_bread_crumb_widget_set_show_separator (HildonBreadCrumbWidget *item,
390                                                gboolean show_separator)
391 {
392   HildonBreadCrumbWidgetPrivate *priv;
393
394   g_return_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (item));
395
396   priv = item->priv;
397
398   if (priv->show_separator == show_separator)
399     return;
400   
401   priv->show_separator = show_separator;
402
403   if (!priv->constructed)
404     return;
405
406   if (show_separator)
407     gtk_widget_show (priv->arrow);
408   else
409     gtk_widget_hide (priv->arrow);
410
411   g_object_notify (G_OBJECT (item), "show-separator");
412 }
413
414 static void
415 hildon_bread_crumb_widget_get_natural_size (HildonBreadCrumb *bread_crumb,
416                                             gint *natural_width,
417                                             gint *natural_height)
418 {
419   GtkRequisition widget_req, label_req;
420   gint width, height;
421   PangoLayout *layout;
422   HildonBreadCrumbWidget *item;
423   HildonBreadCrumbWidgetPrivate *priv;
424
425   g_return_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (bread_crumb));
426
427   item = HILDON_BREAD_CRUMB_WIDGET (bread_crumb);
428   priv = item->priv;
429
430   gtk_widget_size_request (GTK_WIDGET (item), &widget_req);
431
432   layout = gtk_widget_create_pango_layout (priv->label, priv->text);
433   pango_layout_get_pixel_size (layout, &width, &height);
434   g_object_unref (layout);
435
436   if (natural_width)
437     {
438       *natural_width = widget_req.width;
439       /* Substract the size request of the label */
440       gtk_widget_size_request (priv->label, &label_req);
441       *natural_width -= label_req.width;
442
443       /* Add the "natural" width for the label */
444       *natural_width += width;
445       *natural_width += GTK_CONTAINER (item)->border_width * 2;
446     }
447
448   if (natural_height)
449     {
450       *natural_height = widget_req.height;
451       *natural_height += GTK_CONTAINER (item)->border_width * 2;
452     }
453 }
454
455 GtkWidget*
456 _hildon_bread_crumb_widget_new ()
457 {
458   return GTK_WIDGET (g_object_new (HILDON_TYPE_BREAD_CRUMB_WIDGET, NULL));
459 }
460
461 GtkWidget*
462 _hildon_bread_crumb_widget_new_with_text (const gchar *text)
463 {
464   return GTK_WIDGET (g_object_new (HILDON_TYPE_BREAD_CRUMB_WIDGET,
465                                    "text", text,
466                                    NULL));
467 }
468
469 GtkWidget*
470 _hildon_bread_crumb_widget_new_with_icon (GtkWidget *icon, const gchar *text)
471 {
472   return GTK_WIDGET (g_object_new (HILDON_TYPE_BREAD_CRUMB_WIDGET,
473                                    "icon", icon,
474                                    "text", text,
475                                    NULL));
476 }
477
478 void
479 _hildon_bread_crumb_widget_set_icon (HildonBreadCrumbWidget *bread_crumb,
480                                      GtkWidget *icon)
481 {
482   HildonBreadCrumbWidgetPrivate *priv;
483
484   g_return_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (bread_crumb));
485
486   priv = bread_crumb->priv;
487
488   priv->icon = icon;
489
490   hildon_bread_crumb_widget_set_contents (bread_crumb);
491
492   g_object_notify (G_OBJECT (bread_crumb), "icon");
493 }
494
495 void
496 _hildon_bread_crumb_widget_set_icon_position (HildonBreadCrumbWidget *bread_crumb,
497                                               GtkPositionType icon_position)
498 {
499   HildonBreadCrumbWidgetPrivate *priv;
500
501   g_return_if_fail (HILDON_IS_BREAD_CRUMB_WIDGET (bread_crumb));
502
503   priv = bread_crumb->priv;
504
505   if (priv->icon_position == icon_position)
506     return;
507
508   priv->icon_position = icon_position;
509
510   hildon_bread_crumb_widget_set_contents (bread_crumb);
511
512   g_object_notify (G_OBJECT (bread_crumb), "icon-position");
513 }