2 * This file is a part of hildon
4 * Copyright (C) 2007 Nokia Corporation, all rights reserved.
6 * Contact: Michael Dominic Kostrzewa <michael.kostrzewa@nokia.com>
7 * Author: Xan Lopez <xan.lopez@nokia.com>
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.
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.
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
28 * SECTION:hildon-bread-crumb-trail
29 * @short_description: Widget used to represent a specific path in a hierarchical tree.
32 * HildonBreadCrumbTrail is a GTK widget used to represent the currently active path in
33 * some kind of hierarchical structure (file system, media library, structured document, etc).
35 * It has built-in support for text and icon bread crumbs, but the trail only requires a very
36 * simple interface to be implemented for its children and thus new types of items can be
37 * implemented if needed. See #HildonBreadCrumb for more details.
40 #include "hildon-marshalers.h"
41 #include "hildon-bread-crumb-trail.h"
42 #include "hildon-bread-crumb-widget.h"
44 #define HILDON_BREAD_CRUMB_TRAIL_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), HILDON_TYPE_BREAD_CRUMB_TRAIL, HildonBreadCrumbTrailPrivate))
46 struct _HildonBreadCrumbTrailPrivate
48 GtkWidget *back_button;
66 static void hildon_bread_crumb_trail_size_request (GtkWidget *widget,
67 GtkRequisition *requisition);
68 static void hildon_bread_crumb_trail_size_allocate (GtkWidget *widget,
69 GtkAllocation *allocation);
70 static void hildon_bread_crumb_trail_add (GtkContainer *container,
72 static void hildon_bread_crumb_trail_forall (GtkContainer *container,
73 gboolean include_internals,
75 gpointer callback_data);
76 static void hildon_bread_crumb_trail_remove (GtkContainer *container,
78 static void hildon_bread_crumb_trail_finalize (GObject *object);
79 static void hildon_bread_crumb_trail_scroll_back (GtkWidget *button,
80 HildonBreadCrumbTrail *bct);
81 static void hildon_bread_crumb_trail_update_back_button_sensitivity (HildonBreadCrumbTrail *bct);
83 static guint bread_crumb_trail_signals[LAST_SIGNAL] = { 0 };
87 G_DEFINE_TYPE (HildonBreadCrumbTrail, hildon_bread_crumb_trail, GTK_TYPE_CONTAINER)
90 hildon_bread_crumb_trail_class_init (HildonBreadCrumbTrailClass *klass)
92 GObjectClass *gobject_class = (GObjectClass*)klass;
93 GtkObjectClass *object_class = (GtkObjectClass*)klass;
94 GtkWidgetClass *widget_class = (GtkWidgetClass*)klass;
95 GtkContainerClass *container_class = (GtkContainerClass*)klass;
98 gobject_class->finalize = hildon_bread_crumb_trail_finalize;
100 /* GtkWidget signals */
101 widget_class->size_request = hildon_bread_crumb_trail_size_request;
102 widget_class->size_allocate = hildon_bread_crumb_trail_size_allocate;
104 /* GtkContainer signals */
105 container_class->add = hildon_bread_crumb_trail_add;
106 container_class->forall = hildon_bread_crumb_trail_forall;
107 container_class->remove = hildon_bread_crumb_trail_remove;
109 /* Style properties */
111 #define _BREAD_CRUMB_TRAIL_MINIMUM_WIDTH 10
113 /* FIXME: is this the best way to do it? */
114 gtk_widget_class_install_style_property (widget_class,
115 g_param_spec_int ("minimum-width",
117 "The minimum width in characters the children widgets will request",
120 _BREAD_CRUMB_TRAIL_MINIMUM_WIDTH,
123 #define _BREAD_CRUMB_TRAIL_ARROW_SIZE 34
125 gtk_widget_class_install_style_property (widget_class,
126 g_param_spec_int ("arrow-size",
128 "Size of the back button arrow",
131 _BREAD_CRUMB_TRAIL_ARROW_SIZE,
134 bread_crumb_trail_signals[CRUMB_CLICKED] =
135 g_signal_new ("crumb-clicked",
136 G_OBJECT_CLASS_TYPE (object_class),
138 G_STRUCT_OFFSET (HildonBreadCrumbTrailClass, crumb_clicked),
139 g_signal_accumulator_true_handled, NULL,
140 _hildon_marshal_BOOLEAN__POINTER,
145 g_type_class_add_private (gobject_class, sizeof (HildonBreadCrumbTrailPrivate));
149 hildon_bread_crumb_trail_finalize (GObject *object)
151 HildonBreadCrumbTrailPrivate *priv = HILDON_BREAD_CRUMB_TRAIL (object)->priv;
153 g_list_free (priv->item_list);
155 G_OBJECT_CLASS (hildon_bread_crumb_trail_parent_class)->finalize (object);
159 hildon_bread_crumb_trail_size_request (GtkWidget *widget,
160 GtkRequisition *requisition)
163 GtkRequisition child_requisition;
164 HildonBreadCrumbTrail *bct;
165 HildonBreadCrumbTrailPrivate *priv;
166 gint minimum_width, width = 0;
170 bct= HILDON_BREAD_CRUMB_TRAIL (widget);
173 requisition->height = 0;
174 requisition->width = 0;
176 gtk_widget_size_request (priv->back_button, &child_requisition);
177 requisition->width = child_requisition.width;
178 requisition->height = child_requisition.height;
182 /* Add minimum width for one item */
183 /* TODO: this can be probably cached */
184 gtk_widget_style_get (widget,
185 "minimum-width", &minimum_width,
188 tmp = g_strnfill ((gsize)minimum_width, 'm');
189 layout = gtk_widget_create_pango_layout (widget, tmp);
191 pango_layout_get_size (layout, &width, NULL);
192 requisition->width += PANGO_PIXELS (width);
193 g_object_unref (layout);
196 /* Button requisitions */
197 for (p = priv->item_list; p; p = p->next)
199 GtkWidget *child = GTK_WIDGET (p->data);
201 if (GTK_WIDGET_VISIBLE (child))
202 gtk_widget_size_request (child, &child_requisition);
206 requisition->width += GTK_CONTAINER (widget)->border_width * 2;
207 requisition->height += GTK_CONTAINER (widget)->border_width * 2;
209 widget->requisition = *requisition;
212 /* Document me please */
215 hildon_bread_crumb_trail_size_allocate (GtkWidget *widget,
216 GtkAllocation *allocation)
219 gint natural_width, natural_height;
220 HildonBreadCrumb *item;
221 GtkAllocation child_allocation;
222 GtkRequisition child_requisition;
224 gint allocation_width;
225 gint border_width, width;
227 GList *p, *first_show, *first_hide;
228 gint back_button_size;
229 HildonBreadCrumbTrailPrivate *priv = HILDON_BREAD_CRUMB_TRAIL (widget)->priv;
231 widget->allocation = *allocation;
233 border_width = (gint) GTK_CONTAINER (widget)->border_width;
234 allocation_width = allocation->width - 2 * border_width;
236 /* Allocate the back button */
237 child_allocation.x = allocation->x + border_width;
238 child_allocation.y = allocation->y + border_width;
239 gtk_widget_get_child_requisition (priv->back_button, &child_requisition);
240 /* We want the back button to be a square */
241 back_button_size = MAX (child_requisition.width, child_requisition.height);
242 child_allocation.width = child_allocation.height = back_button_size;
243 gtk_widget_size_allocate (priv->back_button, &child_allocation);
244 child_allocation.x += back_button_size;
246 /* If there are no buttons there's nothing else to do */
247 if (priv->item_list == NULL)
250 /* We find out how many buttons can we show, starting from the
251 the last one in the logical path (the first item in the list) */
253 width = back_button_size;
259 for (p = priv->item_list; p; p = p->next)
261 item = HILDON_BREAD_CRUMB (p->data);
262 child = GTK_WIDGET (item);
264 /* Does the widget fit with its natural size? */
265 hildon_bread_crumb_get_natural_size (item,
269 if (width + natural_width <= allocation_width)
273 first_hide = p->next;
274 width += natural_width;
278 /* No, it doesn't. Allocate as much as possible
280 child_allocation.width = allocation_width - width;
282 gtk_widget_size_request (child, &req);
284 if (child_allocation.width > req.width)
286 first_hide = p->next;
287 gtk_widget_size_allocate (child, &child_allocation);
288 gtk_widget_set_child_visible (child, TRUE);
289 child_allocation.x += child_allocation.width;
293 extra_space = child_allocation.width;
300 /* Not enough items to fill the breadcrumb? */
301 if (p == NULL && width < allocation_width)
303 extra_space = allocation_width - width;
306 /* Allocate the other buttons */
307 for (p = first_show; p; p = p->prev)
309 item = HILDON_BREAD_CRUMB (p->data);
310 child = GTK_WIDGET (item);
312 /* Does the widget fit with its natural size? */
313 hildon_bread_crumb_get_natural_size (item,
317 /* If I'm the last and there's extra space, use it */
318 if (p->prev == NULL && extra_space != 0)
320 natural_width += extra_space;
323 child_allocation.width = natural_width;
324 gtk_widget_size_allocate (child, &child_allocation);
325 gtk_widget_set_child_visible (child, TRUE);
326 child_allocation.x += child_allocation.width;
329 for (p = first_hide; p; p = p->next)
331 item = HILDON_BREAD_CRUMB (p->data);
332 child = GTK_WIDGET (item);
334 gtk_widget_set_child_visible (GTK_WIDGET (item), FALSE);
339 get_bread_crumb_id (HildonBreadCrumb *item)
341 return g_object_get_data (G_OBJECT (item), "bread-crumb-id");
345 crumb_activated_cb (GtkWidget *button,
346 HildonBreadCrumbTrail *bct)
348 gboolean signal_handled = FALSE;
350 g_signal_emit (bct, bread_crumb_trail_signals[CRUMB_CLICKED], 0,
351 get_bread_crumb_id (HILDON_BREAD_CRUMB (button)),
354 if (signal_handled == FALSE)
357 HildonBreadCrumbTrailPrivate *priv;
361 child = GTK_WIDGET (priv->item_list->data);
363 /* We remove the tip of the list until we hit the clicked button */
364 while (child != button)
366 gtk_container_remove (GTK_CONTAINER (bct), child);
368 child = GTK_WIDGET (priv->item_list->data);
374 hildon_bread_crumb_trail_add (GtkContainer *container,
377 gtk_widget_set_parent (widget, GTK_WIDGET (container));
379 if (HILDON_IS_BREAD_CRUMB (widget))
381 HildonBreadCrumbTrail *bct = HILDON_BREAD_CRUMB_TRAIL (container);
383 g_signal_connect (G_OBJECT (widget), "crumb-activated",
384 G_CALLBACK (crumb_activated_cb), container);
386 bct->priv->item_list = g_list_prepend (bct->priv->item_list, widget);
388 hildon_bread_crumb_trail_update_back_button_sensitivity (bct);
393 hildon_bread_crumb_trail_forall (GtkContainer *container,
394 gboolean include_internals,
395 GtkCallback callback,
396 gpointer callback_data)
398 g_return_if_fail (callback != NULL);
399 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (container));
402 HildonBreadCrumbTrailPrivate *priv = HILDON_BREAD_CRUMB_TRAIL (container)->priv;
404 children = priv->item_list;
409 child = GTK_WIDGET (children->data);
410 children = children->next;
412 (*callback) (child, callback_data);
415 if (include_internals && priv->back_button)
417 (*callback) (priv->back_button, callback_data);
422 hildon_bread_crumb_trail_remove (GtkContainer *container,
426 HildonBreadCrumbTrailPrivate *priv;
427 gboolean was_visible = GTK_WIDGET_VISIBLE (widget);
429 priv = HILDON_BREAD_CRUMB_TRAIL (container)->priv;
435 if (widget == GTK_WIDGET (p->data))
437 g_signal_handlers_disconnect_by_func (widget, G_CALLBACK (crumb_activated_cb),
438 HILDON_BREAD_CRUMB_TRAIL (container));
439 gtk_widget_unparent (widget);
441 priv->item_list = g_list_remove_link (priv->item_list, p);
444 hildon_bread_crumb_trail_update_back_button_sensitivity (HILDON_BREAD_CRUMB_TRAIL (container));
448 gtk_widget_queue_resize (GTK_WIDGET (container));
457 hildon_bread_crumb_trail_update_back_button_sensitivity (HildonBreadCrumbTrail *bct)
460 HildonBreadCrumbTrailPrivate *priv = bct->priv;
462 list_length = g_list_length (priv->item_list);
464 if (list_length <= 1)
466 gtk_widget_set_sensitive (priv->back_button, FALSE);
470 gtk_widget_set_sensitive (priv->back_button, TRUE);
475 create_back_button (HildonBreadCrumbTrail *bct)
481 gtk_widget_push_composite_child ();
483 button = gtk_button_new ();
484 gtk_widget_set_name (button, "hildon-bread-crumb-back-button");
485 gtk_widget_set_sensitive (button, FALSE);
487 arrow = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE);
488 bct->priv->arrow = arrow;
489 gtk_widget_style_get (GTK_WIDGET (bct),
490 "arrow-size", &arrow_size,
492 gtk_widget_set_size_request (arrow, arrow_size, arrow_size);
494 gtk_container_add (GTK_CONTAINER (button), arrow);
495 gtk_container_add (GTK_CONTAINER (bct), button);
496 gtk_widget_show_all (button);
498 gtk_widget_pop_composite_child ();
504 hildon_bread_crumb_trail_init (HildonBreadCrumbTrail *bct)
506 HildonBreadCrumbTrailPrivate *priv = HILDON_BREAD_CRUMB_TRAIL_GET_PRIVATE (bct);
508 GTK_WIDGET_SET_FLAGS (bct, GTK_NO_WINDOW);
509 gtk_widget_set_redraw_on_allocate (GTK_WIDGET (bct), FALSE);
512 priv->item_list = NULL;
514 priv->back_button = create_back_button (bct);
515 g_signal_connect (priv->back_button, "clicked",
516 G_CALLBACK (hildon_bread_crumb_trail_scroll_back),
521 hildon_bread_crumb_trail_scroll_back (GtkWidget *button,
522 HildonBreadCrumbTrail *bct)
524 HildonBreadCrumb *item;
526 hildon_bread_crumb_trail_pop (bct);
528 item = HILDON_BREAD_CRUMB (bct->priv->item_list->data);
530 g_signal_emit (bct, bread_crumb_trail_signals[CRUMB_CLICKED], 0,
531 get_bread_crumb_id (item));
535 attach_bread_crumb (HildonBreadCrumbTrail *bct,
536 GtkWidget *bread_crumb,
538 GDestroyNotify destroy)
540 g_object_set_data_full (G_OBJECT (bread_crumb), "bread-crumb-id", id, destroy);
542 gtk_container_add (GTK_CONTAINER (bct), bread_crumb);
544 gtk_widget_show (bread_crumb);
550 * hildon_bread_crumb_trail_new:
552 * Creates a new #HildonBreadCrumbTrail widget.
554 * Returns: a #GtkWidget pointer of newly created bread crumb trail
557 * Stability: Unstable
561 hildon_bread_crumb_trail_new (void)
563 return GTK_WIDGET (g_object_new (HILDON_TYPE_BREAD_CRUMB_TRAIL, NULL));
567 * hildon_bread_crumb_trail_push:
568 * @bct: pointer to #HildonBreadCrumbTrail
569 * @item: the #HildonBreadCrumb to be added to the trail
570 * @id: optional id for the bread crumb
571 * @destroy: GDestroyNotify callback to be called when the bread crumb is destroyed
573 * Adds a new bread crumb to the end of the trail.
575 * Stability: Unstable
579 hildon_bread_crumb_trail_push (HildonBreadCrumbTrail *bct,
580 HildonBreadCrumb *item,
582 GDestroyNotify destroy)
586 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (bct));
587 g_return_if_fail (HILDON_IS_BREAD_CRUMB (item));
589 widget = GTK_WIDGET (item);
591 attach_bread_crumb (bct, widget, id, destroy);
595 * hildon_bread_crumb_trail_push_text:
596 * @bct: pointer to #HildonBreadCrumbTrail
597 * @text: content of the new bread crumb
598 * @id: optional id for the bread crumb
599 * @destroy: GDestroyNotify callback to be called when the bread crumb is destroyed
601 * Adds a new bread crumb to the end of the trail containing the specified text.
603 * Stability: Unstable
607 hildon_bread_crumb_trail_push_text (HildonBreadCrumbTrail *bct,
610 GDestroyNotify destroy)
614 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (bct));
615 g_return_if_fail (text != NULL);
617 widget = _hildon_bread_crumb_widget_new_with_text (text);
618 attach_bread_crumb (bct, widget, id, destroy);
622 * hildon_bread_crumb_trail_push_text:
623 * @bct: pointer to #HildonBreadCrumbTrail
624 * @text: content of the new bread crumb
625 * @icon: a widget to set as the icon in the bread crumb
626 * @id: optional id for the bread crumb
627 * @destroy: GDestroyNotify callback to be called when the bread crumb is destroyed
629 * Adds a new bread crumb to the end of the trail containing the specified text and
632 * Stability: Unstable
636 hildon_bread_crumb_trail_push_icon (HildonBreadCrumbTrail *bct,
640 GDestroyNotify destroy)
644 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (bct));
645 g_return_if_fail (text != NULL);
646 g_return_if_fail (GTK_IS_WIDGET (icon));
648 widget = _hildon_bread_crumb_widget_new_with_icon (icon, text);
649 attach_bread_crumb (bct, widget, id, destroy);
653 * hildon_bread_crumb_trail_pop:
654 * @bct: pointer to #HildonBreadCrumbTrail
656 * Removes the last bread crumb from the trail.
658 * Stability: Unstable
662 hildon_bread_crumb_trail_pop (HildonBreadCrumbTrail *bct)
665 HildonBreadCrumbTrailPrivate *priv;
667 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (bct));
671 if (priv->item_list == NULL)
676 child = GTK_WIDGET (priv->item_list->data);
677 gtk_container_remove (GTK_CONTAINER (bct), child);
680 hildon_bread_crumb_trail_update_back_button_sensitivity (bct);
684 * hildon_bread_crumb_trail_clear:
685 * @bct: pointer to #HildonBreadCrumbTrail
687 * Removes all the bread crumbs from the bread crumb trail.
689 * Stability: Unstable
693 hildon_bread_crumb_trail_clear (HildonBreadCrumbTrail *bct)
695 HildonBreadCrumbTrailPrivate *priv;
697 g_return_if_fail (HILDON_IS_BREAD_CRUMB_TRAIL (bct));
701 while (priv->item_list)
703 hildon_bread_crumb_trail_pop (bct);