2 * Copyright (C) 2000 Red Hat, Inc., Jonathan Blandford <jrb@redhat.com>
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
20 #include "gtktreemodel.h"
21 #include "gtktreestore.h"
22 #include "gtktreedatalist.h"
23 #include "gtktreednd.h"
24 #include "gtksignal.h"
26 #include <gobject/gvaluecollector.h>
28 #define G_NODE(node) ((GNode *)node)
38 static guint tree_store_signals[LAST_SIGNAL] = { 0 };
41 static void gtk_tree_store_init (GtkTreeStore *tree_store);
42 static void gtk_tree_store_class_init (GtkTreeStoreClass *tree_store_class);
43 static void gtk_tree_store_tree_model_init (GtkTreeModelIface *iface);
44 static void gtk_tree_store_drag_source_init(GtkTreeDragSourceIface *iface);
45 static void gtk_tree_store_drag_dest_init (GtkTreeDragDestIface *iface);
46 static guint gtk_tree_store_get_flags (GtkTreeModel *tree_model);
47 static gint gtk_tree_store_get_n_columns (GtkTreeModel *tree_model);
48 static GType gtk_tree_store_get_column_type (GtkTreeModel *tree_model,
50 static GtkTreePath *gtk_tree_store_get_path (GtkTreeModel *tree_model,
52 static void gtk_tree_store_get_value (GtkTreeModel *tree_model,
56 static gboolean gtk_tree_store_iter_next (GtkTreeModel *tree_model,
58 static gboolean gtk_tree_store_iter_children (GtkTreeModel *tree_model,
61 static gboolean gtk_tree_store_iter_has_child (GtkTreeModel *tree_model,
63 static gint gtk_tree_store_iter_n_children (GtkTreeModel *tree_model,
65 static gboolean gtk_tree_store_iter_nth_child (GtkTreeModel *tree_model,
69 static gboolean gtk_tree_store_iter_parent (GtkTreeModel *tree_model,
74 static gboolean gtk_tree_store_drag_data_delete (GtkTreeDragSource *drag_source,
76 static gboolean gtk_tree_store_drag_data_get (GtkTreeDragSource *drag_source,
78 GtkSelectionData *selection_data);
79 static gboolean gtk_tree_store_drag_data_received (GtkTreeDragDest *drag_dest,
81 GtkSelectionData *selection_data);
82 static gboolean gtk_tree_store_row_drop_possible (GtkTreeDragDest *drag_dest,
83 GtkTreeModel *src_model,
84 GtkTreePath *src_path,
85 GtkTreePath *dest_path);
87 static void validate_gnode (GNode *node);
90 validate_tree (GtkTreeStore *tree_store)
92 if (gtk_debug_flags & GTK_DEBUG_TREE)
94 g_assert (G_NODE (tree_store->root)->parent == NULL);
96 validate_gnode (G_NODE (tree_store->root));
101 gtk_tree_store_get_type (void)
103 static GtkType tree_store_type = 0;
105 if (!tree_store_type)
107 static const GTypeInfo tree_store_info =
109 sizeof (GtkTreeStoreClass),
110 NULL, /* base_init */
111 NULL, /* base_finalize */
112 (GClassInitFunc) gtk_tree_store_class_init,
113 NULL, /* class_finalize */
114 NULL, /* class_data */
115 sizeof (GtkTreeStore),
117 (GInstanceInitFunc) gtk_tree_store_init
120 static const GInterfaceInfo tree_model_info =
122 (GInterfaceInitFunc) gtk_tree_store_tree_model_init,
127 static const GInterfaceInfo drag_source_info =
129 (GInterfaceInitFunc) gtk_tree_store_drag_source_init,
134 static const GInterfaceInfo drag_dest_info =
136 (GInterfaceInitFunc) gtk_tree_store_drag_dest_init,
141 tree_store_type = g_type_register_static (GTK_TYPE_OBJECT, "GtkTreeStore", &tree_store_info, 0);
143 g_type_add_interface_static (tree_store_type,
147 g_type_add_interface_static (tree_store_type,
148 GTK_TYPE_TREE_DRAG_SOURCE,
150 g_type_add_interface_static (tree_store_type,
151 GTK_TYPE_TREE_DRAG_DEST,
157 return tree_store_type;
161 gtk_tree_store_class_init (GtkTreeStoreClass *tree_store_class)
163 GtkObjectClass *object_class;
165 object_class = (GtkObjectClass *) tree_store_class;
167 tree_store_signals[CHANGED] =
168 gtk_signal_new ("changed",
170 GTK_CLASS_TYPE (object_class),
171 GTK_SIGNAL_OFFSET (GtkTreeStoreClass, changed),
172 gtk_marshal_VOID__BOXED_BOXED,
176 tree_store_signals[INSERTED] =
177 gtk_signal_new ("inserted",
179 GTK_CLASS_TYPE (object_class),
180 GTK_SIGNAL_OFFSET (GtkTreeStoreClass, inserted),
181 gtk_marshal_VOID__BOXED_BOXED,
185 tree_store_signals[CHILD_TOGGLED] =
186 gtk_signal_new ("child_toggled",
188 GTK_CLASS_TYPE (object_class),
189 GTK_SIGNAL_OFFSET (GtkTreeStoreClass, child_toggled),
190 gtk_marshal_VOID__BOXED_BOXED,
194 tree_store_signals[DELETED] =
195 gtk_signal_new ("deleted",
197 GTK_CLASS_TYPE (object_class),
198 GTK_SIGNAL_OFFSET (GtkTreeStoreClass, deleted),
199 gtk_marshal_VOID__BOXED,
205 gtk_tree_store_tree_model_init (GtkTreeModelIface *iface)
207 iface->get_flags = gtk_tree_store_get_flags;
208 iface->get_n_columns = gtk_tree_store_get_n_columns;
209 iface->get_column_type = gtk_tree_store_get_column_type;
210 iface->get_path = gtk_tree_store_get_path;
211 iface->get_value = gtk_tree_store_get_value;
212 iface->iter_next = gtk_tree_store_iter_next;
213 iface->iter_children = gtk_tree_store_iter_children;
214 iface->iter_has_child = gtk_tree_store_iter_has_child;
215 iface->iter_n_children = gtk_tree_store_iter_n_children;
216 iface->iter_nth_child = gtk_tree_store_iter_nth_child;
217 iface->iter_parent = gtk_tree_store_iter_parent;
221 gtk_tree_store_drag_source_init (GtkTreeDragSourceIface *iface)
223 iface->drag_data_delete = gtk_tree_store_drag_data_delete;
224 iface->drag_data_get = gtk_tree_store_drag_data_get;
228 gtk_tree_store_drag_dest_init (GtkTreeDragDestIface *iface)
230 iface->drag_data_received = gtk_tree_store_drag_data_received;
231 iface->row_drop_possible = gtk_tree_store_row_drop_possible;
235 gtk_tree_store_init (GtkTreeStore *tree_store)
237 tree_store->root = g_node_new (NULL);
238 tree_store->stamp = g_random_int ();
242 gtk_tree_store_new (void)
244 return GTK_TREE_STORE (gtk_type_new (gtk_tree_store_get_type ()));
248 gtk_tree_store_new_with_types (gint n_columns,
251 GtkTreeStore *retval;
255 g_return_val_if_fail (n_columns > 0, NULL);
257 retval = gtk_tree_store_new ();
258 gtk_tree_store_set_n_columns (retval, n_columns);
260 va_start (args, n_columns);
262 for (i = 0; i < n_columns; i++)
263 gtk_tree_store_set_column_type (retval, i, va_arg (args, GType));
271 gtk_tree_store_set_n_columns (GtkTreeStore *tree_store,
276 g_return_if_fail (tree_store != NULL);
277 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
279 if (tree_store->n_columns == n_columns)
282 new_columns = g_new0 (GType, n_columns);
283 if (tree_store->column_headers)
285 /* copy the old header orders over */
286 if (n_columns >= tree_store->n_columns)
287 memcpy (new_columns, tree_store->column_headers, tree_store->n_columns * sizeof (gchar *));
289 memcpy (new_columns, tree_store->column_headers, n_columns * sizeof (GType));
291 g_free (tree_store->column_headers);
294 tree_store->column_headers = new_columns;
295 tree_store->n_columns = n_columns;
299 gtk_tree_store_set_column_type (GtkTreeStore *tree_store,
303 g_return_if_fail (tree_store != NULL);
304 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
305 g_return_if_fail (column >=0 && column < tree_store->n_columns);
307 tree_store->column_headers[column] = type;
310 /* fulfill the GtkTreeModel requirements */
311 /* NOTE: GtkTreeStore::root is a GNode, that acts as the parent node. However,
312 * it is not visible to the tree or to the user., and the path "0" refers to the
313 * first child of GtkTreeStore::root.
318 gtk_tree_store_get_flags (GtkTreeModel *tree_model)
320 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), 0);
322 return GTK_TREE_MODEL_ITERS_PERSIST;
326 gtk_tree_store_get_n_columns (GtkTreeModel *tree_model)
328 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), 0);
330 return GTK_TREE_STORE (tree_model)->n_columns;
334 gtk_tree_store_get_column_type (GtkTreeModel *tree_model,
337 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), G_TYPE_INVALID);
338 g_return_val_if_fail (index < GTK_TREE_STORE (tree_model)->n_columns &&
339 index >= 0, G_TYPE_INVALID);
341 return GTK_TREE_STORE (tree_model)->column_headers[index];
345 gtk_tree_store_get_path (GtkTreeModel *tree_model,
352 g_return_val_if_fail (tree_model != NULL, NULL);
353 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), NULL);
354 g_return_val_if_fail (iter != NULL, NULL);
355 g_return_val_if_fail (iter->user_data != NULL, NULL);
357 validate_tree ((GtkTreeStore*)tree_model);
359 g_assert (G_NODE (iter->user_data)->parent != NULL);
361 if (G_NODE (iter->user_data)->parent == G_NODE (GTK_TREE_STORE (tree_model)->root))
363 retval = gtk_tree_path_new ();
364 tmp_node = G_NODE (GTK_TREE_STORE (tree_model)->root)->children;
368 GtkTreeIter tmp_iter = *iter;
370 tmp_iter.user_data = G_NODE (iter->user_data)->parent;
372 retval = gtk_tree_store_get_path (tree_model,
374 tmp_node = G_NODE (iter->user_data)->parent->children;
380 if (tmp_node == NULL)
382 gtk_tree_path_free (retval);
386 for (; tmp_node; tmp_node = tmp_node->next)
388 if (tmp_node == G_NODE (iter->user_data))
393 if (tmp_node == NULL)
395 /* We couldn't find node, meaning it's prolly not ours */
396 /* Perhaps I should do a g_return_if_fail here. */
397 gtk_tree_path_free (retval);
401 gtk_tree_path_append_index (retval, i);
408 gtk_tree_store_get_value (GtkTreeModel *tree_model,
413 GtkTreeDataList *list;
414 gint tmp_column = column;
416 g_return_if_fail (tree_model != NULL);
417 g_return_if_fail (GTK_IS_TREE_STORE (tree_model));
418 g_return_if_fail (iter != NULL);
419 g_return_if_fail (column < GTK_TREE_STORE (tree_model)->n_columns);
421 list = G_NODE (iter->user_data)->data;
423 while (tmp_column-- > 0 && list)
428 _gtk_tree_data_list_node_to_value (list,
429 GTK_TREE_STORE (tree_model)->column_headers[column],
434 /* We want to return an initialized but empty (default) value */
435 g_value_init (value, GTK_TREE_STORE (tree_model)->column_headers[column]);
440 gtk_tree_store_iter_next (GtkTreeModel *tree_model,
443 g_return_val_if_fail (iter->user_data != NULL, FALSE);
445 if (G_NODE (iter->user_data)->next)
447 iter->user_data = G_NODE (iter->user_data)->next;
455 gtk_tree_store_iter_children (GtkTreeModel *tree_model,
461 g_return_val_if_fail (parent == NULL || parent->user_data != NULL, FALSE);
464 children = G_NODE (parent->user_data)->children;
466 children = G_NODE (GTK_TREE_STORE (tree_model)->root)->children;
470 iter->stamp = GTK_TREE_STORE (tree_model)->stamp;
471 iter->user_data = children;
479 gtk_tree_store_iter_has_child (GtkTreeModel *tree_model,
482 g_return_val_if_fail (tree_model != NULL, FALSE);
483 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), FALSE);
484 g_return_val_if_fail (iter != NULL, FALSE);
485 g_return_val_if_fail (iter->user_data != NULL, FALSE);
487 return G_NODE (iter->user_data)->children != NULL;
491 gtk_tree_store_iter_n_children (GtkTreeModel *tree_model,
497 g_return_val_if_fail (tree_model != NULL, 0);
498 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), 0);
499 g_return_val_if_fail (iter != NULL, FALSE);
500 g_return_val_if_fail (iter->user_data != NULL, FALSE);
503 node = G_NODE (GTK_TREE_STORE (tree_model)->root)->children;
505 node = G_NODE (iter->user_data)->children;
517 gtk_tree_store_iter_nth_child (GtkTreeModel *tree_model,
525 g_return_val_if_fail (tree_model != NULL, FALSE);
526 g_return_val_if_fail (GTK_IS_TREE_STORE (tree_model), FALSE);
527 g_return_val_if_fail (parent == NULL || parent->user_data != NULL, FALSE);
530 parent_node = GTK_TREE_STORE (tree_model)->root;
532 parent_node = parent->user_data;
534 child = g_node_nth_child (parent_node, n);
538 iter->user_data = child;
539 iter->stamp = GTK_TREE_STORE (tree_model)->stamp;
547 gtk_tree_store_iter_parent (GtkTreeModel *tree_model,
553 g_return_val_if_fail (iter != NULL, FALSE);
554 g_return_val_if_fail (iter->user_data != NULL, FALSE);
556 parent = G_NODE (child->user_data)->parent;
558 g_assert (parent != NULL);
560 if (parent != GTK_TREE_STORE (tree_model)->root)
562 iter->user_data = parent;
563 iter->stamp = GTK_TREE_STORE (tree_model)->stamp;
571 * This is a somewhat inelegant function that does a lot of list
572 * manipulations on it's own.
575 gtk_tree_store_set_cell (GtkTreeStore *tree_store,
580 GtkTreeDataList *list;
581 GtkTreeDataList *prev;
583 g_return_if_fail (tree_store != NULL);
584 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
585 g_return_if_fail (column >= 0 && column < tree_store->n_columns);
587 prev = list = G_NODE (iter->user_data)->data;
593 _gtk_tree_data_list_value_to_node (list, value);
594 gtk_signal_emit_by_name (GTK_OBJECT (tree_store),
605 if (G_NODE (iter->user_data)->data == NULL)
607 G_NODE (iter->user_data)->data = list = _gtk_tree_data_list_alloc ();
612 list = prev->next = _gtk_tree_data_list_alloc ();
618 list->next = _gtk_tree_data_list_alloc ();
623 _gtk_tree_data_list_value_to_node (list, value);
624 gtk_signal_emit_by_name (GTK_OBJECT (tree_store),
630 gtk_tree_store_set_valist (GtkTreeStore *tree_store,
636 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
638 column = va_arg (var_args, gint);
642 GValue value = { 0, };
645 if (column >= tree_store->n_columns)
647 g_warning ("%s: Invalid column number %d added to iter (remember to end your list of columns with a -1)", G_STRLOC, column);
650 g_value_init (&value, tree_store->column_headers[column]);
652 G_VALUE_COLLECT (&value, var_args, &error);
655 g_warning ("%s: %s", G_STRLOC, error);
658 /* we purposely leak the value here, it might not be
659 * in a sane state if an error condition occoured
664 gtk_tree_store_set_cell (tree_store,
669 g_value_unset (&value);
671 column = va_arg (var_args, gint);
676 * gtk_tree_store_set:
677 * @tree_store: a #GtkTreeStore
678 * @iter: row iterator
679 * @Varargs: pairs of column number and value, terminated with -1
681 * Sets the value of one or more cells in the row referenced by @iter.
682 * The variable argument list should contain integer column numbers,
683 * each column number followed by the value to be set. For example,
684 * The list is terminated by a -1. For example, to set column 0 with type
685 * %G_TYPE_STRING to "Foo", you would write gtk_tree_store_set (store, iter,
689 gtk_tree_store_set (GtkTreeStore *tree_store,
695 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
697 va_start (var_args, iter);
698 gtk_tree_store_set_valist (tree_store, iter, var_args);
703 gtk_tree_store_get_valist (GtkTreeStore *tree_store,
709 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
711 column = va_arg (var_args, gint);
715 GValue value = { 0, };
718 if (column >= tree_store->n_columns)
720 g_warning ("%s: Invalid column number %d accessed (remember to end your list of columns with a -1)", G_STRLOC, column);
724 gtk_tree_store_get_value (GTK_TREE_MODEL (tree_store), iter, column, &value);
726 G_VALUE_LCOPY (&value, var_args, &error);
729 g_warning ("%s: %s", G_STRLOC, error);
732 /* we purposely leak the value here, it might not be
733 * in a sane state if an error condition occoured
738 g_value_unset (&value);
740 column = va_arg (var_args, gint);
745 gtk_tree_store_get (GtkTreeStore *tree_store,
751 g_return_if_fail (GTK_IS_TREE_STORE (tree_store));
753 va_start (var_args, iter);
754 gtk_tree_store_get_valist (tree_store, iter, var_args);
759 gtk_tree_store_remove (GtkTreeStore *model,
766 g_return_if_fail (model != NULL);
767 g_return_if_fail (GTK_IS_TREE_STORE (model));
769 parent = G_NODE (iter->user_data)->parent;
771 g_assert (parent != NULL);
773 if (G_NODE (iter->user_data)->data)
774 _gtk_tree_data_list_free ((GtkTreeDataList *) G_NODE (iter->user_data)->data,
775 model->column_headers);
777 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
778 g_node_destroy (G_NODE (iter->user_data));
781 gtk_signal_emit_by_name (GTK_OBJECT (model),
784 if (parent != G_NODE (model->root) && parent->children == NULL)
786 gtk_tree_path_up (path);
788 gtk_signal_emit_by_name (GTK_OBJECT (model),
793 gtk_tree_path_free (path);
797 gtk_tree_store_insert (GtkTreeStore *model,
805 g_return_if_fail (model != NULL);
806 g_return_if_fail (GTK_IS_TREE_STORE (model));
809 parent_node = parent->user_data;
811 parent_node = model->root;
813 iter->stamp = model->stamp;
814 iter->user_data = g_node_new (NULL);
815 g_node_insert (parent_node, position, G_NODE (iter->user_data));
817 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
818 gtk_signal_emit_by_name (GTK_OBJECT (model),
821 gtk_tree_path_free (path);
823 validate_tree ((GtkTreeStore*)model);
827 gtk_tree_store_insert_before (GtkTreeStore *model,
830 GtkTreeIter *sibling)
833 GNode *parent_node = NULL;
836 g_return_if_fail (model != NULL);
837 g_return_if_fail (GTK_IS_TREE_STORE (model));
838 g_return_if_fail (iter != NULL);
840 new_node = g_node_new (NULL);
842 if (parent == NULL && sibling == NULL)
843 parent_node = model->root;
844 else if (parent == NULL)
845 parent_node = G_NODE (sibling->user_data)->parent;
846 else if (sibling == NULL)
847 parent_node = G_NODE (parent->user_data);
850 g_return_if_fail (G_NODE (sibling->user_data)->parent ==
851 G_NODE (parent->user_data));
852 parent_node = G_NODE (parent->user_data);
855 g_node_insert_before (parent_node,
856 sibling ? G_NODE (sibling->user_data) : NULL,
859 iter->stamp = model->stamp;
860 iter->user_data = new_node;
862 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
863 gtk_signal_emit_by_name (GTK_OBJECT (model),
866 gtk_tree_path_free (path);
868 validate_tree ((GtkTreeStore*)model);
872 gtk_tree_store_insert_after (GtkTreeStore *model,
875 GtkTreeIter *sibling)
881 g_return_if_fail (model != NULL);
882 g_return_if_fail (GTK_IS_TREE_STORE (model));
883 g_return_if_fail (iter != NULL);
885 new_node = g_node_new (NULL);
887 if (parent == NULL && sibling == NULL)
888 parent_node = model->root;
889 else if (parent == NULL)
890 parent_node = G_NODE (sibling->user_data)->parent;
891 else if (sibling == NULL)
892 parent_node = G_NODE (parent->user_data);
895 g_return_if_fail (G_NODE (sibling->user_data)->parent ==
896 G_NODE (parent->user_data));
897 parent_node = G_NODE (parent->user_data);
901 g_node_insert_after (parent_node,
902 sibling ? G_NODE (sibling->user_data) : NULL,
905 iter->stamp = model->stamp;
906 iter->user_data = new_node;
908 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
909 gtk_signal_emit_by_name (GTK_OBJECT (model),
912 gtk_tree_path_free (path);
914 validate_tree ((GtkTreeStore*)model);
918 gtk_tree_store_prepend (GtkTreeStore *model,
924 g_return_if_fail (model != NULL);
925 g_return_if_fail (GTK_IS_TREE_STORE (model));
926 g_return_if_fail (iter != NULL);
929 parent_node = model->root;
931 parent_node = parent->user_data;
933 if (parent_node->children == NULL)
937 iter->stamp = model->stamp;
938 iter->user_data = g_node_new (NULL);
940 g_node_prepend (parent_node, iter->user_data);
942 if (parent_node != model->root)
944 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), parent);
945 gtk_signal_emit_by_name (GTK_OBJECT (model),
949 gtk_tree_path_append_index (path, 0);
953 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
955 gtk_signal_emit_by_name (GTK_OBJECT (model),
959 gtk_tree_path_free (path);
963 gtk_tree_store_insert_after (model, iter, parent, NULL);
966 validate_tree ((GtkTreeStore*)model);
970 gtk_tree_store_append (GtkTreeStore *model,
976 g_return_if_fail (model != NULL);
977 g_return_if_fail (GTK_IS_TREE_STORE (model));
978 g_return_if_fail (iter != NULL);
981 parent_node = model->root;
983 parent_node = parent->user_data;
985 if (parent_node->children == NULL)
989 iter->stamp = model->stamp;
990 iter->user_data = g_node_new (NULL);
992 g_node_append (parent_node, G_NODE (iter->user_data));
994 if (parent_node != model->root)
996 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), parent);
997 gtk_signal_emit_by_name (GTK_OBJECT (model),
1001 gtk_tree_path_append_index (path, 0);
1005 path = gtk_tree_store_get_path (GTK_TREE_MODEL (model), iter);
1008 gtk_signal_emit_by_name (GTK_OBJECT (model),
1012 gtk_tree_path_free (path);
1016 gtk_tree_store_insert_before (model, iter, parent, NULL);
1019 validate_tree ((GtkTreeStore*)model);
1023 gtk_tree_store_is_ancestor (GtkTreeStore *model,
1025 GtkTreeIter *descendant)
1027 g_return_val_if_fail (model != NULL, FALSE);
1028 g_return_val_if_fail (GTK_IS_TREE_STORE (model), FALSE);
1029 g_return_val_if_fail (iter != NULL, FALSE);
1030 g_return_val_if_fail (descendant != NULL, FALSE);
1032 return g_node_is_ancestor (G_NODE (iter->user_data),
1033 G_NODE (descendant->user_data));
1038 gtk_tree_store_iter_depth (GtkTreeStore *model,
1041 g_return_val_if_fail (model != NULL, 0);
1042 g_return_val_if_fail (GTK_IS_TREE_STORE (model), 0);
1043 g_return_val_if_fail (iter != NULL, 0);
1045 return g_node_depth (G_NODE (iter->user_data)) - 1;
1053 gtk_tree_store_drag_data_delete (GtkTreeDragSource *drag_source,
1058 g_return_val_if_fail (GTK_IS_TREE_STORE (drag_source), FALSE);
1060 if (gtk_tree_model_get_iter (GTK_TREE_MODEL (drag_source),
1064 gtk_tree_store_remove (GTK_TREE_STORE (drag_source),
1075 gtk_tree_store_drag_data_get (GtkTreeDragSource *drag_source,
1077 GtkSelectionData *selection_data)
1079 g_return_val_if_fail (GTK_IS_TREE_STORE (drag_source), FALSE);
1081 /* Note that we don't need to handle the GTK_TREE_MODEL_ROW
1082 * target, because the default handler does it for us, but
1083 * we do anyway for the convenience of someone maybe overriding the
1087 if (gtk_selection_data_set_tree_row (selection_data,
1088 GTK_TREE_MODEL (drag_source),
1095 /* FIXME handle text targets at least. */
1102 copy_node_data (GtkTreeStore *tree_store,
1103 GtkTreeIter *src_iter,
1104 GtkTreeIter *dest_iter)
1106 GtkTreeDataList *dl = G_NODE (src_iter->user_data)->data;
1107 GtkTreeDataList *copy_head = NULL;
1108 GtkTreeDataList *copy_prev = NULL;
1109 GtkTreeDataList *copy_iter = NULL;
1115 copy_iter = _gtk_tree_data_list_node_copy (dl,
1116 tree_store->column_headers[col]);
1118 if (copy_head == NULL)
1119 copy_head = copy_iter;
1122 copy_prev->next = copy_iter;
1124 copy_prev = copy_iter;
1130 G_NODE (dest_iter->user_data)->data = copy_head;
1132 gtk_signal_emit_by_name (GTK_OBJECT (tree_store),
1138 recursive_node_copy (GtkTreeStore *tree_store,
1139 GtkTreeIter *src_iter,
1140 GtkTreeIter *dest_iter)
1143 GtkTreeModel *model;
1145 model = GTK_TREE_MODEL (tree_store);
1147 copy_node_data (tree_store, src_iter, dest_iter);
1149 if (gtk_tree_model_iter_children (model,
1153 /* Need to create children and recurse. Note our
1154 * dependence on persistent iterators here.
1160 /* Gee, a really slow algorithm... ;-) FIXME */
1161 gtk_tree_store_append (tree_store,
1165 recursive_node_copy (tree_store, &child, ©);
1167 while (gtk_tree_model_iter_next (model, &child));
1172 gtk_tree_store_drag_data_received (GtkTreeDragDest *drag_dest,
1174 GtkSelectionData *selection_data)
1176 GtkTreeModel *tree_model;
1177 GtkTreeStore *tree_store;
1178 GtkTreeModel *src_model = NULL;
1179 GtkTreePath *src_path = NULL;
1180 gboolean retval = FALSE;
1182 g_return_val_if_fail (GTK_IS_TREE_STORE (drag_dest), FALSE);
1184 tree_model = GTK_TREE_MODEL (drag_dest);
1185 tree_store = GTK_TREE_STORE (drag_dest);
1187 validate_tree (tree_store);
1189 if (gtk_selection_data_get_tree_row (selection_data,
1192 src_model == tree_model)
1194 /* Copy the given row to a new position */
1195 GtkTreeIter src_iter;
1196 GtkTreeIter dest_iter;
1199 if (!gtk_tree_model_get_iter (src_model,
1206 /* Get the path to insert _after_ (dest is the path to insert _before_) */
1207 prev = gtk_tree_path_copy (dest);
1209 if (!gtk_tree_path_prev (prev))
1211 GtkTreeIter dest_parent;
1212 GtkTreePath *parent;
1213 GtkTreeIter *dest_parent_p;
1215 /* dest was the first spot at the current depth; which means
1216 * we are supposed to prepend.
1219 /* Get the parent, NULL if parent is the root */
1220 dest_parent_p = NULL;
1221 parent = gtk_tree_path_copy (dest);
1222 if (gtk_tree_path_up (parent))
1224 gtk_tree_model_get_iter (tree_model,
1227 dest_parent_p = &dest_parent;
1229 gtk_tree_path_free (parent);
1232 gtk_tree_store_prepend (GTK_TREE_STORE (tree_model),
1240 if (gtk_tree_model_get_iter (GTK_TREE_MODEL (tree_model),
1244 GtkTreeIter tmp_iter = dest_iter;
1245 gtk_tree_store_insert_after (GTK_TREE_STORE (tree_model),
1254 gtk_tree_path_free (prev);
1256 /* If we succeeded in creating dest_iter, walk src_iter tree branch,
1257 * duplicating it below dest_iter.
1262 recursive_node_copy (tree_store,
1269 /* FIXME maybe add some data targets eventually, or handle text
1270 * targets in the simple case.
1278 gtk_tree_path_free (src_path);
1284 gtk_tree_store_row_drop_possible (GtkTreeDragDest *drag_dest,
1285 GtkTreeModel *src_model,
1286 GtkTreePath *src_path,
1287 GtkTreePath *dest_path)
1289 /* can only drag to ourselves */
1290 if (src_model != GTK_TREE_MODEL (drag_dest))
1293 /* Can't drop into ourself. */
1294 if (gtk_tree_path_is_ancestor (src_path,
1298 /* Can't drop if dest_path's parent doesn't exist */
1301 GtkTreePath *tmp = gtk_tree_path_copy (dest_path);
1303 /* if we can't go up, we know the parent exists, the root
1306 if (gtk_tree_path_up (tmp))
1308 if (!gtk_tree_model_get_iter (GTK_TREE_MODEL (drag_dest),
1312 gtk_tree_path_free (tmp);
1318 gtk_tree_path_free (tmp);
1321 /* Can otherwise drop anywhere. */
1326 validate_gnode (GNode* node)
1330 iter = node->children;
1331 while (iter != NULL)
1333 g_assert (iter->parent == node);
1335 g_assert (iter->prev->next == iter);
1336 validate_gnode (iter);