3 * Copyright (C) 2010 Openismus GmbH
6 * Tristan Van Berkom <tristanvb@openismus.com>
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Library General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Library General Public License for more details.
18 * You should have received a copy of the GNU Library General Public
19 * License along with this library; if not, write to the
20 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21 * Boston, MA 02111-1307, USA.
26 #include "gtktreemenu.h"
27 #include "gtkmarshalers.h"
28 #include "gtkmenuitem.h"
29 #include "gtkseparatormenuitem.h"
30 #include "gtkcellareabox.h"
31 #include "gtkcellareacontext.h"
32 #include "gtkcelllayout.h"
33 #include "gtkcellview.h"
34 #include "gtkprivate.h"
38 static GObject *gtk_tree_menu_constructor (GType type,
39 guint n_construct_properties,
40 GObjectConstructParam *construct_properties);
41 static void gtk_tree_menu_dispose (GObject *object);
42 static void gtk_tree_menu_finalize (GObject *object);
43 static void gtk_tree_menu_set_property (GObject *object,
47 static void gtk_tree_menu_get_property (GObject *object,
53 static void gtk_tree_menu_get_preferred_width (GtkWidget *widget,
56 static void gtk_tree_menu_get_preferred_height (GtkWidget *widget,
59 static void gtk_tree_menu_size_allocate (GtkWidget *widget,
60 GtkAllocation *allocation);
62 /* GtkCellLayoutIface */
63 static void gtk_tree_menu_cell_layout_init (GtkCellLayoutIface *iface);
64 static void gtk_tree_menu_cell_layout_pack_start (GtkCellLayout *layout,
65 GtkCellRenderer *cell,
67 static void gtk_tree_menu_cell_layout_pack_end (GtkCellLayout *layout,
68 GtkCellRenderer *cell,
70 static GList *gtk_tree_menu_cell_layout_get_cells (GtkCellLayout *layout);
71 static void gtk_tree_menu_cell_layout_clear (GtkCellLayout *layout);
72 static void gtk_tree_menu_cell_layout_add_attribute (GtkCellLayout *layout,
73 GtkCellRenderer *cell,
74 const gchar *attribute,
76 static void gtk_tree_menu_cell_layout_set_cell_data_func (GtkCellLayout *layout,
77 GtkCellRenderer *cell,
78 GtkCellLayoutDataFunc func,
80 GDestroyNotify destroy);
81 static void gtk_tree_menu_cell_layout_clear_attributes (GtkCellLayout *layout,
82 GtkCellRenderer *cell);
83 static void gtk_tree_menu_cell_layout_reorder (GtkCellLayout *layout,
84 GtkCellRenderer *cell,
86 static GtkCellArea *gtk_tree_menu_cell_layout_get_area (GtkCellLayout *layout);
89 /* TreeModel/DrawingArea callbacks and building menus/submenus */
90 static void gtk_tree_menu_populate (GtkTreeMenu *menu);
91 static GtkWidget *gtk_tree_menu_create_item (GtkTreeMenu *menu,
93 gboolean header_item);
94 static void gtk_tree_menu_set_area (GtkTreeMenu *menu,
96 static GtkWidget *gtk_tree_menu_get_path_item (GtkTreeMenu *menu,
99 static void context_size_changed_cb (GtkCellAreaContext *context,
102 static void item_activated_cb (GtkMenuItem *item,
104 static void submenu_activated_cb (GtkTreeMenu *submenu,
108 struct _GtkTreeMenuPrivate
110 /* TreeModel and parent for this menu */
112 GtkTreeRowReference *root;
114 /* CellArea and context for this menu */
116 GtkCellAreaContext *context;
119 gulong size_changed_id;
120 gulong row_inserted_id;
121 gulong row_deleted_id;
122 gulong row_reordered_id;
125 GtkTreeViewRowSeparatorFunc row_separator_func;
126 gpointer row_separator_data;
127 GDestroyNotify row_separator_destroy;
129 /* Submenu headers */
130 GtkTreeMenuHeaderFunc header_func;
131 gpointer header_data;
132 GDestroyNotify header_destroy;
134 guint32 menu_with_header : 1;
145 SIGNAL_MENU_ACTIVATE,
149 static guint tree_menu_signals[N_SIGNALS] = { 0 };
150 static GQuark tree_menu_path_quark = 0;
152 G_DEFINE_TYPE_WITH_CODE (GtkTreeMenu, gtk_tree_menu, GTK_TYPE_MENU,
153 G_IMPLEMENT_INTERFACE (GTK_TYPE_CELL_LAYOUT,
154 gtk_tree_menu_cell_layout_init));
157 gtk_tree_menu_init (GtkTreeMenu *menu)
159 GtkTreeMenuPrivate *priv;
161 menu->priv = G_TYPE_INSTANCE_GET_PRIVATE (menu,
166 gtk_menu_set_reserve_toggle_size (GTK_MENU (menu), FALSE);
170 gtk_tree_menu_class_init (GtkTreeMenuClass *class)
172 GObjectClass *object_class = G_OBJECT_CLASS (class);
173 GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (class);
175 tree_menu_path_quark = g_quark_from_static_string ("gtk-tree-menu-path");
177 object_class->constructor = gtk_tree_menu_constructor;
178 object_class->dispose = gtk_tree_menu_dispose;
179 object_class->finalize = gtk_tree_menu_finalize;
180 object_class->set_property = gtk_tree_menu_set_property;
181 object_class->get_property = gtk_tree_menu_get_property;
183 widget_class->get_preferred_width = gtk_tree_menu_get_preferred_width;
184 widget_class->get_preferred_height = gtk_tree_menu_get_preferred_height;
185 widget_class->size_allocate = gtk_tree_menu_size_allocate;
187 tree_menu_signals[SIGNAL_MENU_ACTIVATE] =
188 g_signal_new (I_("menu-activate"),
189 G_OBJECT_CLASS_TYPE (object_class),
191 0, /* No class closure here */
193 _gtk_marshal_VOID__STRING,
194 G_TYPE_NONE, 1, G_TYPE_STRING);
196 g_object_class_install_property (object_class,
198 g_param_spec_object ("model",
199 P_("TreeMenu model"),
200 P_("The model for the tree menu"),
202 GTK_PARAM_READWRITE));
204 g_object_class_install_property (object_class,
206 g_param_spec_boxed ("root",
207 P_("TreeMenu root row"),
208 P_("The TreeMenu will display children of the "
211 GTK_PARAM_READWRITE));
213 g_object_class_install_property (object_class,
215 g_param_spec_object ("cell-area",
217 P_("The GtkCellArea used to layout cells"),
219 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
222 g_type_class_add_private (object_class, sizeof (GtkTreeMenuPrivate));
225 /****************************************************************
227 ****************************************************************/
229 gtk_tree_menu_constructor (GType type,
230 guint n_construct_properties,
231 GObjectConstructParam *construct_properties)
235 GtkTreeMenuPrivate *priv;
237 object = G_OBJECT_CLASS (gtk_tree_menu_parent_class)->constructor
238 (type, n_construct_properties, construct_properties);
240 menu = GTK_TREE_MENU (object);
245 GtkCellArea *area = gtk_cell_area_box_new ();
247 gtk_tree_menu_set_area (menu, area);
250 priv->context = gtk_cell_area_create_context (priv->area);
251 priv->size_changed_id =
252 g_signal_connect (priv->context, "notify",
253 G_CALLBACK (context_size_changed_cb), menu);
260 gtk_tree_menu_dispose (GObject *object)
263 GtkTreeMenuPrivate *priv;
265 menu = GTK_TREE_MENU (object);
268 gtk_tree_menu_set_model (menu, NULL);
269 gtk_tree_menu_set_area (menu, NULL);
273 /* Disconnect signals */
274 g_signal_handler_disconnect (priv->context, priv->size_changed_id);
276 g_object_unref (priv->context);
277 priv->context = NULL;
278 priv->size_changed_id = 0;
281 G_OBJECT_CLASS (gtk_tree_menu_parent_class)->dispose (object);
285 gtk_tree_menu_finalize (GObject *object)
288 GtkTreeMenuPrivate *priv;
290 menu = GTK_TREE_MENU (object);
293 gtk_tree_menu_set_row_separator_func (menu, NULL, NULL, NULL);
294 gtk_tree_menu_set_header_func (menu, NULL, NULL, NULL);
297 gtk_tree_row_reference_free (priv->root);
299 G_OBJECT_CLASS (gtk_tree_menu_parent_class)->finalize (object);
303 gtk_tree_menu_set_property (GObject *object,
308 GtkTreeMenu *menu = GTK_TREE_MENU (object);
313 gtk_tree_menu_set_model (menu, g_value_get_object (value));
317 gtk_tree_menu_set_root (menu, g_value_get_boxed (value));
321 /* Construct-only, can only be assigned once */
322 gtk_tree_menu_set_area (menu, (GtkCellArea *)g_value_get_object (value));
326 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
332 gtk_tree_menu_get_property (GObject *object,
337 GtkTreeMenu *menu = GTK_TREE_MENU (object);
338 GtkTreeMenuPrivate *priv = menu->priv;
343 g_value_set_object (value, priv->model);
347 g_value_set_boxed (value, priv->root);
351 g_value_set_object (value, priv->area);
355 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
360 /****************************************************************
362 ****************************************************************/
364 /* We tell all the menu items to reserve space for the submenu
365 * indicator if there is at least one submenu, this way we ensure
366 * that every internal cell area gets allocated the
367 * same width (and requested height for the same appropriate width).
370 sync_reserve_submenu_size (GtkTreeMenu *menu)
373 gboolean has_submenu = FALSE;
375 children = gtk_container_get_children (GTK_CONTAINER (menu));
376 for (l = children; l; l = l->next)
378 GtkMenuItem *item = l->data;
380 if (gtk_menu_item_get_submenu (item) != NULL)
387 for (l = children; l; l = l->next)
389 GtkMenuItem *item = l->data;
391 gtk_menu_item_set_reserve_indicator (item, has_submenu);
394 g_list_free (children);
398 gtk_tree_menu_get_preferred_width (GtkWidget *widget,
402 GtkTreeMenu *menu = GTK_TREE_MENU (widget);
403 GtkTreeMenuPrivate *priv = menu->priv;
405 /* We leave the requesting work up to the cellviews which operate in the same
406 * context, reserving space for the submenu indicator if any of the items have
407 * submenus ensures that every cellview will receive the same allocated width.
409 * Since GtkMenu does hieght-for-width correctly, we know that the width of
410 * every cell will be requested before the height-for-widths are requested.
412 g_signal_handler_block (priv->context, priv->size_changed_id);
414 sync_reserve_submenu_size (menu);
415 gtk_cell_area_context_flush_preferred_width (priv->context);
417 GTK_WIDGET_CLASS (gtk_tree_menu_parent_class)->get_preferred_width (widget, minimum_size, natural_size);
419 g_signal_handler_unblock (priv->context, priv->size_changed_id);
423 gtk_tree_menu_get_preferred_height (GtkWidget *widget,
427 GtkTreeMenu *menu = GTK_TREE_MENU (widget);
428 GtkTreeMenuPrivate *priv = menu->priv;
430 g_signal_handler_block (priv->context, priv->size_changed_id);
432 sync_reserve_submenu_size (menu);
433 gtk_cell_area_context_flush_preferred_height (priv->context);
435 GTK_WIDGET_CLASS (gtk_tree_menu_parent_class)->get_preferred_height (widget, minimum_size, natural_size);
437 g_signal_handler_unblock (priv->context, priv->size_changed_id);
441 gtk_tree_menu_size_allocate (GtkWidget *widget,
442 GtkAllocation *allocation)
444 GtkTreeMenu *menu = GTK_TREE_MENU (widget);
445 GtkTreeMenuPrivate *priv = menu->priv;
447 /* flush the context allocation */
448 gtk_cell_area_context_flush_allocation (priv->context);
450 /* Leave it to the first cell area to allocate the size of priv->context, since
451 * we configure the menu to allocate all children the same width this works fine
453 GTK_WIDGET_CLASS (gtk_tree_menu_parent_class)->size_allocate (widget, allocation);
456 /****************************************************************
457 * GtkCellLayoutIface *
458 ****************************************************************/
459 /* Just forward all the GtkCellLayoutIface methods to the
460 * underlying GtkCellArea
463 gtk_tree_menu_cell_layout_init (GtkCellLayoutIface *iface)
465 iface->pack_start = gtk_tree_menu_cell_layout_pack_start;
466 iface->pack_end = gtk_tree_menu_cell_layout_pack_end;
467 iface->get_cells = gtk_tree_menu_cell_layout_get_cells;
468 iface->clear = gtk_tree_menu_cell_layout_clear;
469 iface->add_attribute = gtk_tree_menu_cell_layout_add_attribute;
470 iface->set_cell_data_func = gtk_tree_menu_cell_layout_set_cell_data_func;
471 iface->clear_attributes = gtk_tree_menu_cell_layout_clear_attributes;
472 iface->reorder = gtk_tree_menu_cell_layout_reorder;
473 iface->get_area = gtk_tree_menu_cell_layout_get_area;
477 gtk_tree_menu_cell_layout_pack_start (GtkCellLayout *layout,
478 GtkCellRenderer *cell,
481 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
482 GtkTreeMenuPrivate *priv = menu->priv;
484 gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (priv->area), cell, expand);
488 gtk_tree_menu_cell_layout_pack_end (GtkCellLayout *layout,
489 GtkCellRenderer *cell,
492 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
493 GtkTreeMenuPrivate *priv = menu->priv;
495 gtk_cell_layout_pack_end (GTK_CELL_LAYOUT (priv->area), cell, expand);
499 gtk_tree_menu_cell_layout_get_cells (GtkCellLayout *layout)
501 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
502 GtkTreeMenuPrivate *priv = menu->priv;
504 return gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (priv->area));
508 gtk_tree_menu_cell_layout_clear (GtkCellLayout *layout)
510 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
511 GtkTreeMenuPrivate *priv = menu->priv;
513 gtk_cell_layout_clear (GTK_CELL_LAYOUT (priv->area));
517 gtk_tree_menu_cell_layout_add_attribute (GtkCellLayout *layout,
518 GtkCellRenderer *cell,
519 const gchar *attribute,
522 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
523 GtkTreeMenuPrivate *priv = menu->priv;
525 gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (priv->area), cell, attribute, column);
529 gtk_tree_menu_cell_layout_set_cell_data_func (GtkCellLayout *layout,
530 GtkCellRenderer *cell,
531 GtkCellLayoutDataFunc func,
533 GDestroyNotify destroy)
535 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
536 GtkTreeMenuPrivate *priv = menu->priv;
538 gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (priv->area), cell, func, func_data, destroy);
542 gtk_tree_menu_cell_layout_clear_attributes (GtkCellLayout *layout,
543 GtkCellRenderer *cell)
545 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
546 GtkTreeMenuPrivate *priv = menu->priv;
548 gtk_cell_layout_clear_attributes (GTK_CELL_LAYOUT (priv->area), cell);
552 gtk_tree_menu_cell_layout_reorder (GtkCellLayout *layout,
553 GtkCellRenderer *cell,
556 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
557 GtkTreeMenuPrivate *priv = menu->priv;
559 gtk_cell_layout_reorder (GTK_CELL_LAYOUT (priv->area), cell, position);
563 gtk_tree_menu_cell_layout_get_area (GtkCellLayout *layout)
565 GtkTreeMenu *menu = GTK_TREE_MENU (layout);
566 GtkTreeMenuPrivate *priv = menu->priv;
572 /****************************************************************
573 * TreeModel callbacks/populating menus *
574 ****************************************************************/
576 gtk_tree_menu_get_path_item (GtkTreeMenu *menu,
579 GtkWidget *item = NULL;
582 children = gtk_container_get_children (GTK_CONTAINER (menu));
584 for (l = children; item == NULL && l != NULL; l = l->next)
586 GtkWidget *child = l->data;
587 GtkTreePath *path = NULL;
589 if (GTK_IS_SEPARATOR_MENU_ITEM (child))
591 GtkTreeRowReference *row =
592 g_object_get_qdata (G_OBJECT (child), tree_menu_path_quark);
594 if (row && gtk_tree_row_reference_valid (row))
595 path = gtk_tree_row_reference_get_path (row);
599 GtkWidget *view = gtk_bin_get_child (GTK_BIN (child));
601 /* It's always a cellview */
602 if (GTK_IS_CELL_VIEW (view))
603 path = gtk_cell_view_get_displayed_row (GTK_CELL_VIEW (view));
608 if (gtk_tree_path_compare (search, path) == 0)
611 gtk_tree_path_free (path);
615 g_list_free (children);
621 row_inserted_cb (GtkTreeModel *model,
626 GtkTreeMenuPrivate *priv = menu->priv;
627 GtkTreePath *parent_path;
628 gboolean this_menu = FALSE;
629 gint *indices, index, depth;
631 parent_path = gtk_tree_path_copy (path);
633 /* Check if the menu and the added iter are in root of the model */
634 if (!gtk_tree_path_up (parent_path))
639 /* If we are a submenu, compare the path */
642 GtkTreePath *root_path =
643 gtk_tree_row_reference_get_path (priv->root);
645 if (gtk_tree_path_compare (root_path, parent_path) == 0)
648 gtk_tree_path_free (root_path);
651 gtk_tree_path_free (parent_path);
653 /* If the iter should be in this menu then go ahead and insert it */
658 /* Get the index of the path for this depth */
659 indices = gtk_tree_path_get_indices (path);
660 depth = gtk_tree_path_get_depth (path);
661 index = indices[depth -1];
663 /* Menus with a header include a menuitem for it's root node
664 * and a separator menu item */
665 if (priv->menu_with_header)
668 item = gtk_tree_menu_create_item (menu, iter, FALSE);
669 gtk_menu_shell_insert (GTK_MENU_SHELL (menu), item, index);
671 /* Resize everything */
672 gtk_cell_area_context_flush (menu->priv->context);
677 row_deleted_cb (GtkTreeModel *model,
681 GtkTreeMenuPrivate *priv = menu->priv;
682 GtkTreePath *root_path;
685 /* If it's the root node we leave it to the parent menu to remove us
689 root_path = gtk_tree_row_reference_get_path (priv->root);
691 if (gtk_tree_path_compare (root_path, path) == 0)
693 gtk_tree_path_free (root_path);
698 /* Get rid of the deleted item */
699 item = gtk_tree_menu_get_path_item (menu, path);
702 gtk_widget_destroy (item);
704 /* Resize everything */
705 gtk_cell_area_context_flush (menu->priv->context);
710 row_reordered_cb (GtkTreeModel *model,
716 GtkTreeMenuPrivate *priv = menu->priv;
717 gboolean this_menu = FALSE;
719 if (iter == NULL && priv->root == NULL)
723 GtkTreePath *root_path =
724 gtk_tree_row_reference_get_path (priv->root);
726 if (gtk_tree_path_compare (root_path, path) == 0)
729 gtk_tree_path_free (root_path);
734 /* Destroy and repopulate the menu at the level where the order changed */
735 gtk_container_foreach (GTK_CONTAINER (menu),
736 (GtkCallback) gtk_widget_destroy, NULL);
738 gtk_tree_menu_populate (menu);
740 /* Resize everything */
741 gtk_cell_area_context_flush (menu->priv->context);
746 context_size_changed_cb (GtkCellAreaContext *context,
750 if (!strcmp (pspec->name, "minimum-width") ||
751 !strcmp (pspec->name, "natural-width") ||
752 !strcmp (pspec->name, "minimum-height") ||
753 !strcmp (pspec->name, "natural-height"))
754 gtk_widget_queue_resize (menu);
758 gtk_tree_menu_set_area (GtkTreeMenu *menu,
761 GtkTreeMenuPrivate *priv = menu->priv;
764 g_object_unref (priv->area);
769 g_object_ref_sink (priv->area);
773 gtk_tree_menu_create_item (GtkTreeMenu *menu,
775 gboolean header_item)
777 GtkTreeMenuPrivate *priv = menu->priv;
778 GtkWidget *item, *view;
780 gboolean is_separator = FALSE;
782 path = gtk_tree_model_get_path (priv->model, iter);
784 if (priv->row_separator_func)
786 priv->row_separator_func (priv->model, iter,
787 priv->row_separator_data);
791 item = gtk_separator_menu_item_new ();
793 g_object_set_qdata_full (G_OBJECT (item),
794 tree_menu_path_quark,
795 gtk_tree_row_reference_new (priv->model, path),
796 (GDestroyNotify)gtk_tree_row_reference_free);
800 view = gtk_cell_view_new_with_context (priv->area, priv->context);
801 item = gtk_menu_item_new ();
802 gtk_widget_show (view);
803 gtk_widget_show (item);
805 gtk_cell_view_set_model (GTK_CELL_VIEW (view), priv->model);
806 gtk_cell_view_set_displayed_row (GTK_CELL_VIEW (view), path);
808 gtk_widget_show (view);
809 gtk_container_add (GTK_CONTAINER (item), view);
811 g_signal_connect (item, "activate", G_CALLBACK (item_activated_cb), menu);
813 /* Add a GtkTreeMenu submenu to render the children of this row */
814 if (header_item == FALSE &&
815 gtk_tree_model_iter_has_child (priv->model, iter))
819 submenu = gtk_tree_menu_new_with_area (priv->area);
821 gtk_tree_menu_set_row_separator_func (GTK_TREE_MENU (submenu),
822 priv->row_separator_func,
823 priv->row_separator_data,
824 priv->row_separator_destroy);
825 gtk_tree_menu_set_header_func (GTK_TREE_MENU (submenu),
828 priv->header_destroy);
830 gtk_tree_menu_set_model (GTK_TREE_MENU (submenu), priv->model);
831 gtk_tree_menu_set_root (GTK_TREE_MENU (submenu), path);
833 gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), submenu);
835 g_signal_connect (submenu, "menu-activate",
836 G_CALLBACK (submenu_activated_cb), menu);
840 gtk_tree_path_free (path);
846 gtk_tree_menu_populate (GtkTreeMenu *menu)
848 GtkTreeMenuPrivate *priv = menu->priv;
849 GtkTreePath *path = NULL;
852 gboolean valid = FALSE;
853 GtkWidget *menu_item;
859 path = gtk_tree_row_reference_get_path (priv->root);
863 if (gtk_tree_model_get_iter (priv->model, &parent, path))
865 valid = gtk_tree_model_iter_children (priv->model, &iter, &parent);
867 if (priv->header_func &&
868 priv->header_func (priv->model, &parent, priv->header_data))
870 /* Add a submenu header for rows which desire one, used for
871 * combo boxes to allow all rows to be activatable/selectable
873 menu_item = gtk_tree_menu_create_item (menu, &parent, TRUE);
874 gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
876 menu_item = gtk_separator_menu_item_new ();
877 gtk_widget_show (menu_item);
878 gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
880 priv->menu_with_header = TRUE;
883 gtk_tree_path_free (path);
886 valid = gtk_tree_model_iter_children (priv->model, &iter, NULL);
888 /* Create a menu item for every row at the current depth, add a GtkTreeMenu
889 * submenu for iters/items that have children */
892 menu_item = gtk_tree_menu_create_item (menu, &iter, FALSE);
893 gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_item);
895 valid = gtk_tree_model_iter_next (priv->model, &iter);
900 item_activated_cb (GtkMenuItem *item,
907 /* Only activate leafs, not parents */
908 if (!gtk_menu_item_get_submenu (item))
910 view = GTK_CELL_VIEW (gtk_bin_get_child (GTK_BIN (item)));
911 path = gtk_cell_view_get_displayed_row (view);
912 path_str = gtk_tree_path_to_string (path);
914 g_signal_emit (menu, tree_menu_signals[SIGNAL_MENU_ACTIVATE], 0, path_str);
917 gtk_tree_path_free (path);
922 submenu_activated_cb (GtkTreeMenu *submenu,
926 g_signal_emit (menu, tree_menu_signals[SIGNAL_MENU_ACTIVATE], 0, path);
929 /****************************************************************
931 ****************************************************************/
933 gtk_tree_menu_new (void)
935 return (GtkWidget *)g_object_new (GTK_TYPE_TREE_MENU, NULL);
939 gtk_tree_menu_new_with_area (GtkCellArea *area)
941 return (GtkWidget *)g_object_new (GTK_TYPE_TREE_MENU,
947 gtk_tree_menu_new_full (GtkCellArea *area,
951 return (GtkWidget *)g_object_new (GTK_TYPE_TREE_MENU,
959 gtk_tree_menu_set_model (GtkTreeMenu *menu,
962 GtkTreeMenuPrivate *priv;
964 g_return_if_fail (GTK_IS_TREE_MENU (menu));
965 g_return_if_fail (model == NULL || GTK_IS_TREE_MODEL (model));
969 if (priv->model != model)
973 /* Disconnect signals */
974 g_signal_handler_disconnect (priv->model,
975 priv->row_inserted_id);
976 g_signal_handler_disconnect (priv->model,
977 priv->row_deleted_id);
978 g_signal_handler_disconnect (priv->model,
979 priv->row_reordered_id);
980 priv->row_inserted_id = 0;
981 priv->row_deleted_id = 0;
982 priv->row_reordered_id = 0;
984 g_object_unref (priv->model);
991 g_object_ref (priv->model);
993 /* Connect signals */
994 priv->row_inserted_id = g_signal_connect (priv->model, "row-inserted",
995 G_CALLBACK (row_inserted_cb), menu);
996 priv->row_deleted_id = g_signal_connect (priv->model, "row-deleted",
997 G_CALLBACK (row_deleted_cb), menu);
998 priv->row_reordered_id = g_signal_connect (priv->model, "rows-reordered",
999 G_CALLBACK (row_reordered_cb), menu);
1005 gtk_tree_menu_get_model (GtkTreeMenu *menu)
1007 GtkTreeMenuPrivate *priv;
1009 g_return_val_if_fail (GTK_IS_TREE_MENU (menu), NULL);
1017 gtk_tree_menu_set_root (GtkTreeMenu *menu,
1020 GtkTreeMenuPrivate *priv;
1022 g_return_if_fail (GTK_IS_TREE_MENU (menu));
1023 g_return_if_fail (menu->priv->model != NULL || path == NULL);
1028 gtk_tree_row_reference_free (priv->root);
1031 priv->root = gtk_tree_row_reference_new (priv->model, path);
1035 /* Destroy all the menu items for the previous root */
1036 gtk_container_foreach (GTK_CONTAINER (menu),
1037 (GtkCallback) gtk_widget_destroy, NULL);
1039 /* Populate for the new root */
1041 gtk_tree_menu_populate (menu);
1045 gtk_tree_menu_get_root (GtkTreeMenu *menu)
1047 GtkTreeMenuPrivate *priv;
1049 g_return_val_if_fail (GTK_IS_TREE_MENU (menu), NULL);
1054 return gtk_tree_row_reference_get_path (priv->root);
1060 gtk_tree_menu_set_row_separator_func (GtkTreeMenu *menu,
1061 GtkTreeViewRowSeparatorFunc func,
1063 GDestroyNotify destroy)
1065 GtkTreeMenuPrivate *priv;
1067 g_return_if_fail (GTK_IS_TREE_MENU (menu));
1071 if (priv->row_separator_destroy)
1072 priv->row_separator_destroy (priv->row_separator_data);
1074 priv->row_separator_func = func;
1075 priv->row_separator_data = data;
1076 priv->row_separator_destroy = destroy;
1078 /* Destroy all the menu items */
1079 gtk_container_foreach (GTK_CONTAINER (menu),
1080 (GtkCallback) gtk_widget_destroy, NULL);
1082 /* Populate again */
1084 gtk_tree_menu_populate (menu);
1087 GtkTreeViewRowSeparatorFunc
1088 gtk_tree_menu_get_row_separator_func (GtkTreeMenu *menu)
1090 GtkTreeMenuPrivate *priv;
1092 g_return_val_if_fail (GTK_IS_TREE_MENU (menu), NULL);
1096 return priv->row_separator_func;
1100 gtk_tree_menu_set_header_func (GtkTreeMenu *menu,
1101 GtkTreeMenuHeaderFunc func,
1103 GDestroyNotify destroy)
1105 GtkTreeMenuPrivate *priv;
1107 g_return_if_fail (GTK_IS_TREE_MENU (menu));
1111 if (priv->header_destroy)
1112 priv->header_destroy (priv->header_data);
1114 priv->header_func = func;
1115 priv->header_data = data;
1116 priv->header_destroy = destroy;
1118 /* Destroy all the menu items */
1119 gtk_container_foreach (GTK_CONTAINER (menu),
1120 (GtkCallback) gtk_widget_destroy, NULL);
1122 /* Populate again */
1124 gtk_tree_menu_populate (menu);
1127 GtkTreeMenuHeaderFunc
1128 gtk_tree_menu_get_header_func (GtkTreeMenu *menu)
1130 GtkTreeMenuPrivate *priv;
1132 g_return_val_if_fail (GTK_IS_TREE_MENU (menu), NULL);
1136 return priv->header_func;