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.
21 #include "gtktreeview.h"
22 #include "gtkrbtree.h"
23 #include "gtktreeprivate.h"
24 #include "gtkcellrenderer.h"
25 #include "gtksignal.h"
27 #include "gtkbutton.h"
28 #include "gtkalignment.h"
31 #include <gdk/gdkkeysyms.h>
34 /* the width of the column resize windows */
35 #define TREE_VIEW_DRAG_WIDTH 6
36 #define TREE_VIEW_EXPANDER_WIDTH 14
37 #define TREE_VIEW_EXPANDER_HEIGHT 14
38 #define TREE_VIEW_VERTICAL_SEPARATOR 2
39 #define TREE_VIEW_HORIZONTAL_SEPARATOR 0
42 typedef struct _GtkTreeViewChild GtkTreeViewChild;
44 struct _GtkTreeViewChild
52 static void gtk_tree_view_init (GtkTreeView *tree_view);
53 static void gtk_tree_view_class_init (GtkTreeViewClass *klass);
56 static void gtk_tree_view_finalize (GObject *object);
59 static void gtk_tree_view_setup_model (GtkTreeView *tree_view);
60 static void gtk_tree_view_realize (GtkWidget *widget);
61 static void gtk_tree_view_unrealize (GtkWidget *widget);
62 static void gtk_tree_view_map (GtkWidget *widget);
63 static void gtk_tree_view_size_request (GtkWidget *widget,
64 GtkRequisition *requisition);
65 static void gtk_tree_view_size_allocate (GtkWidget *widget,
66 GtkAllocation *allocation);
67 static gboolean gtk_tree_view_expose (GtkWidget *widget,
68 GdkEventExpose *event);
69 static gboolean gtk_tree_view_motion (GtkWidget *widget,
70 GdkEventMotion *event);
71 static gboolean gtk_tree_view_enter_notify (GtkWidget *widget,
72 GdkEventCrossing *event);
73 static gboolean gtk_tree_view_leave_notify (GtkWidget *widget,
74 GdkEventCrossing *event);
75 static gboolean gtk_tree_view_button_press (GtkWidget *widget,
76 GdkEventButton *event);
77 static gboolean gtk_tree_view_button_release (GtkWidget *widget,
78 GdkEventButton *event);
79 static void gtk_tree_view_draw_focus (GtkWidget *widget);
80 static gint gtk_tree_view_focus_in (GtkWidget *widget,
81 GdkEventFocus *event);
82 static gint gtk_tree_view_focus_out (GtkWidget *widget,
83 GdkEventFocus *event);
84 static gint gtk_tree_view_focus (GtkContainer *container,
85 GtkDirectionType direction);
87 /* container signals */
88 static void gtk_tree_view_remove (GtkContainer *container,
90 static void gtk_tree_view_forall (GtkContainer *container,
91 gboolean include_internals,
93 gpointer callback_data);
95 /* Source side drag signals */
96 static void gtk_tree_view_drag_begin (GtkWidget *widget,
97 GdkDragContext *context);
98 static void gtk_tree_view_drag_end (GtkWidget *widget,
99 GdkDragContext *context);
100 static void gtk_tree_view_drag_data_get (GtkWidget *widget,
101 GdkDragContext *context,
102 GtkSelectionData *selection_data,
105 static void gtk_tree_view_drag_data_delete (GtkWidget *widget,
106 GdkDragContext *context);
108 /* Target side drag signals */
109 static void gtk_tree_view_drag_leave (GtkWidget *widget,
110 GdkDragContext *context,
112 static gboolean gtk_tree_view_drag_motion (GtkWidget *widget,
113 GdkDragContext *context,
117 static gboolean gtk_tree_view_drag_drop (GtkWidget *widget,
118 GdkDragContext *context,
122 static void gtk_tree_view_drag_data_received (GtkWidget *widget,
123 GdkDragContext *context,
126 GtkSelectionData *selection_data,
130 /* tree_model signals */
131 static void gtk_tree_view_set_adjustments (GtkTreeView *tree_view,
133 GtkAdjustment *vadj);
134 static void gtk_tree_view_changed (GtkTreeModel *model,
138 static void gtk_tree_view_inserted (GtkTreeModel *model,
142 static void gtk_tree_view_child_toggled (GtkTreeModel *model,
146 static void gtk_tree_view_deleted (GtkTreeModel *model,
150 /* Internal functions */
151 static void gtk_tree_view_queue_draw_node (GtkTreeView *tree_view,
154 GdkRectangle *clip_rect);
155 static void gtk_tree_view_draw_arrow (GtkTreeView *tree_view,
160 static void gtk_tree_view_get_arrow_range (GtkTreeView *tree_view,
163 static gint gtk_tree_view_new_column_width (GtkTreeView *tree_view,
166 static void gtk_tree_view_adjustment_changed (GtkAdjustment *adjustment,
167 GtkTreeView *tree_view);
168 static gint gtk_tree_view_insert_iter_height (GtkTreeView *tree_view,
172 static void gtk_tree_view_build_tree (GtkTreeView *tree_view,
177 gboolean calc_bounds);
178 static void gtk_tree_view_calc_size (GtkTreeView *priv,
182 static gboolean gtk_tree_view_discover_dirty_iter (GtkTreeView *tree_view,
186 static void gtk_tree_view_discover_dirty (GtkTreeView *tree_view,
190 static void gtk_tree_view_check_dirty (GtkTreeView *tree_view);
191 static void gtk_tree_view_create_button (GtkTreeView *tree_view,
193 static void gtk_tree_view_create_buttons (GtkTreeView *tree_view);
194 static void gtk_tree_view_button_clicked (GtkWidget *widget,
196 static void gtk_tree_view_clamp_node_visible (GtkTreeView *tree_view,
199 static gboolean gtk_tree_view_maybe_begin_dragging_row (GtkTreeView *tree_view,
200 GdkEventMotion *event);
202 static GtkContainerClass *parent_class = NULL;
205 /* Class Functions */
207 gtk_tree_view_get_type (void)
209 static GtkType tree_view_type = 0;
213 static const GTypeInfo tree_view_info =
215 sizeof (GtkTreeViewClass),
216 NULL, /* base_init */
217 NULL, /* base_finalize */
218 (GClassInitFunc) gtk_tree_view_class_init,
219 NULL, /* class_finalize */
220 NULL, /* class_data */
221 sizeof (GtkTreeView),
223 (GInstanceInitFunc) gtk_tree_view_init
226 tree_view_type = g_type_register_static (GTK_TYPE_CONTAINER, "GtkTreeView", &tree_view_info, 0);
229 return tree_view_type;
233 gtk_tree_view_class_init (GtkTreeViewClass *class)
235 GObjectClass *o_class;
236 GtkObjectClass *object_class;
237 GtkWidgetClass *widget_class;
238 GtkContainerClass *container_class;
240 o_class = (GObjectClass *) class;
241 object_class = (GtkObjectClass *) class;
242 widget_class = (GtkWidgetClass *) class;
243 container_class = (GtkContainerClass *) class;
245 parent_class = g_type_class_peek_parent (class);
247 o_class->finalize = gtk_tree_view_finalize;
249 widget_class->realize = gtk_tree_view_realize;
250 widget_class->unrealize = gtk_tree_view_unrealize;
251 widget_class->map = gtk_tree_view_map;
252 widget_class->size_request = gtk_tree_view_size_request;
253 widget_class->size_allocate = gtk_tree_view_size_allocate;
254 widget_class->expose_event = gtk_tree_view_expose;
255 widget_class->motion_notify_event = gtk_tree_view_motion;
256 widget_class->enter_notify_event = gtk_tree_view_enter_notify;
257 widget_class->leave_notify_event = gtk_tree_view_leave_notify;
258 widget_class->button_press_event = gtk_tree_view_button_press;
259 widget_class->button_release_event = gtk_tree_view_button_release;
260 widget_class->draw_focus = gtk_tree_view_draw_focus;
261 widget_class->focus_in_event = gtk_tree_view_focus_in;
262 widget_class->focus_out_event = gtk_tree_view_focus_out;
264 widget_class->drag_begin = gtk_tree_view_drag_begin;
265 widget_class->drag_end = gtk_tree_view_drag_end;
266 widget_class->drag_data_get = gtk_tree_view_drag_data_get;
267 widget_class->drag_data_delete = gtk_tree_view_drag_data_delete;
269 widget_class->drag_leave = gtk_tree_view_drag_leave;
270 widget_class->drag_motion = gtk_tree_view_drag_motion;
271 widget_class->drag_drop = gtk_tree_view_drag_drop;
272 widget_class->drag_data_received = gtk_tree_view_drag_data_received;
274 container_class->forall = gtk_tree_view_forall;
275 container_class->remove = gtk_tree_view_remove;
276 container_class->focus = gtk_tree_view_focus;
278 class->set_scroll_adjustments = gtk_tree_view_set_adjustments;
280 widget_class->set_scroll_adjustments_signal =
281 gtk_signal_new ("set_scroll_adjustments",
283 GTK_CLASS_TYPE (object_class),
284 GTK_SIGNAL_OFFSET (GtkTreeViewClass, set_scroll_adjustments),
285 gtk_marshal_VOID__POINTER_POINTER,
287 GTK_TYPE_ADJUSTMENT, GTK_TYPE_ADJUSTMENT);
291 gtk_tree_view_init (GtkTreeView *tree_view)
293 tree_view->priv = g_new0 (GtkTreeViewPrivate, 1);
295 GTK_WIDGET_SET_FLAGS (tree_view, GTK_CAN_FOCUS);
297 tree_view->priv->flags = GTK_TREE_VIEW_IS_LIST | GTK_TREE_VIEW_SHOW_EXPANDERS | GTK_TREE_VIEW_DRAW_KEYFOCUS | GTK_TREE_VIEW_HEADERS_VISIBLE;
298 tree_view->priv->tab_offset = TREE_VIEW_EXPANDER_WIDTH;
299 tree_view->priv->n_columns = 0;
300 tree_view->priv->columns = NULL;
301 tree_view->priv->button_pressed_node = NULL;
302 tree_view->priv->button_pressed_tree = NULL;
303 tree_view->priv->prelight_node = NULL;
304 tree_view->priv->header_height = 1;
305 tree_view->priv->x_drag = 0;
306 tree_view->priv->drag_pos = -1;
307 tree_view->priv->selection = NULL;
308 tree_view->priv->anchor = NULL;
309 tree_view->priv->cursor = NULL;
311 tree_view->priv->pressed_button = -1;
312 tree_view->priv->press_start_x = -1;
313 tree_view->priv->press_start_y = -1;
315 gtk_tree_view_set_adjustments (tree_view, NULL, NULL);
316 _gtk_tree_view_set_size (tree_view, 0, 0);
324 gtk_tree_view_finalize (GObject *object)
326 GtkTreeView *tree_view = (GtkTreeView *) object;
328 if (tree_view->priv->tree)
329 _gtk_rbtree_free (tree_view->priv->tree);
331 if (tree_view->priv->scroll_to_path != NULL)
332 gtk_tree_path_free (tree_view->priv->scroll_to_path);
334 if (tree_view->priv->drag_dest_row)
335 gtk_tree_path_free (tree_view->priv->drag_dest_row);
337 g_free (tree_view->priv);
338 if (G_OBJECT_CLASS (parent_class)->finalize)
339 (* G_OBJECT_CLASS (parent_class)->finalize) (object);
346 gtk_tree_view_realize_buttons (GtkTreeView *tree_view)
349 GtkTreeViewColumn *column;
351 guint attributes_mask;
353 g_return_if_fail (GTK_WIDGET_REALIZED (tree_view));
354 g_return_if_fail (tree_view->priv->header_window != NULL);
356 attr.window_type = GDK_WINDOW_CHILD;
357 attr.wclass = GDK_INPUT_ONLY;
358 attr.visual = gtk_widget_get_visual (GTK_WIDGET (tree_view));
359 attr.colormap = gtk_widget_get_colormap (GTK_WIDGET (tree_view));
360 attr.event_mask = gtk_widget_get_events (GTK_WIDGET (tree_view));
361 attr.event_mask = (GDK_BUTTON_PRESS_MASK |
362 GDK_BUTTON_RELEASE_MASK |
363 GDK_POINTER_MOTION_MASK |
364 GDK_POINTER_MOTION_HINT_MASK |
366 attributes_mask = GDK_WA_CURSOR | GDK_WA_X | GDK_WA_Y;
367 attr.cursor = gdk_cursor_new (GDK_SB_H_DOUBLE_ARROW);
368 tree_view->priv->cursor_drag = attr.cursor;
371 attr.width = TREE_VIEW_DRAG_WIDTH;
372 attr.height = tree_view->priv->header_height;
374 for (list = tree_view->priv->columns; list; list = list->next)
379 if (column->visible == FALSE)
381 if (column->window != NULL)
384 gtk_widget_set_parent_window (column->button,
385 tree_view->priv->header_window);
387 attr.x = (column->button->allocation.x + column->button->allocation.width) - 3;
389 column->window = gdk_window_new (tree_view->priv->header_window,
390 &attr, attributes_mask);
391 gdk_window_set_user_data (column->window, tree_view);
397 gtk_tree_view_realize (GtkWidget *widget)
400 GtkTreeView *tree_view;
402 GdkWindowAttr attributes;
403 gint attributes_mask;
405 g_return_if_fail (widget != NULL);
406 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
408 tree_view = GTK_TREE_VIEW (widget);
410 if (!GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_MODEL_SETUP) &&
411 tree_view->priv->model)
412 gtk_tree_view_setup_model (tree_view);
414 gtk_tree_view_check_dirty (GTK_TREE_VIEW (widget));
415 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
417 /* Make the main, clipping window */
418 attributes.window_type = GDK_WINDOW_CHILD;
419 attributes.x = widget->allocation.x;
420 attributes.y = widget->allocation.y;
421 attributes.width = widget->allocation.width;
422 attributes.height = widget->allocation.height;
423 attributes.wclass = GDK_INPUT_OUTPUT;
424 attributes.visual = gtk_widget_get_visual (widget);
425 attributes.colormap = gtk_widget_get_colormap (widget);
426 attributes.event_mask = GDK_VISIBILITY_NOTIFY_MASK;
428 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
430 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget),
431 &attributes, attributes_mask);
432 gdk_window_set_user_data (widget->window, widget);
434 /* Make the window for the tree */
437 attributes.width = tree_view->priv->width;
438 attributes.height = tree_view->priv->height + TREE_VIEW_HEADER_HEIGHT (tree_view);
439 attributes.event_mask = GDK_EXPOSURE_MASK |
441 GDK_POINTER_MOTION_MASK |
442 GDK_ENTER_NOTIFY_MASK |
443 GDK_LEAVE_NOTIFY_MASK |
444 GDK_BUTTON_PRESS_MASK |
445 GDK_BUTTON_RELEASE_MASK |
446 gtk_widget_get_events (widget);
448 tree_view->priv->bin_window = gdk_window_new (widget->window,
449 &attributes, attributes_mask);
450 gdk_window_set_user_data (tree_view->priv->bin_window, widget);
452 /* Make the column header window */
455 attributes.width = MAX (tree_view->priv->width, widget->allocation.width);
456 attributes.height = tree_view->priv->header_height;
457 attributes.event_mask = (GDK_EXPOSURE_MASK |
459 GDK_BUTTON_PRESS_MASK |
460 GDK_BUTTON_RELEASE_MASK |
462 GDK_KEY_RELEASE_MASK) |
463 gtk_widget_get_events (widget);
465 tree_view->priv->header_window = gdk_window_new (widget->window,
466 &attributes, attributes_mask);
467 gdk_window_set_user_data (tree_view->priv->header_window, widget);
470 values.foreground = (widget->style->white.pixel==0 ?
471 widget->style->black:widget->style->white);
472 values.function = GDK_XOR;
473 values.subwindow_mode = GDK_INCLUDE_INFERIORS;
474 tree_view->priv->xor_gc = gdk_gc_new_with_values (widget->window,
479 /* Add them all up. */
480 widget->style = gtk_style_attach (widget->style, widget->window);
481 gdk_window_set_background (widget->window, &widget->style->base[widget->state]);
482 gdk_window_set_background (tree_view->priv->bin_window, &widget->style->base[widget->state]);
483 gtk_style_set_background (widget->style, tree_view->priv->header_window, GTK_STATE_NORMAL);
485 tmp_list = tree_view->priv->children;
488 GtkTreeViewChild *child = tmp_list->data;
489 tmp_list = tmp_list->next;
491 gtk_widget_set_parent_window (child->widget, tree_view->priv->bin_window);
493 gtk_tree_view_realize_buttons (GTK_TREE_VIEW (widget));
494 _gtk_tree_view_set_size (GTK_TREE_VIEW (widget), -1, -1);
496 if (tree_view->priv->scroll_to_path != NULL ||
497 tree_view->priv->scroll_to_column != NULL)
499 gtk_tree_view_scroll_to_cell (tree_view,
500 tree_view->priv->scroll_to_path,
501 tree_view->priv->scroll_to_column,
502 tree_view->priv->scroll_to_row_align,
503 tree_view->priv->scroll_to_col_align);
504 if (tree_view->priv->scroll_to_path)
506 gtk_tree_path_free (tree_view->priv->scroll_to_path);
507 tree_view->priv->scroll_to_path = NULL;
509 tree_view->priv->scroll_to_column = NULL;
514 gtk_tree_view_unrealize (GtkWidget *widget)
516 GtkTreeView *tree_view;
518 g_return_if_fail (widget != NULL);
519 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
521 tree_view = GTK_TREE_VIEW (widget);
523 if (tree_view->priv->scroll_timeout != 0)
525 gtk_timeout_remove (tree_view->priv->scroll_timeout);
526 tree_view->priv->scroll_timeout = 0;
529 if (tree_view->priv->open_dest_timeout != 0)
531 gtk_timeout_remove (tree_view->priv->open_dest_timeout);
532 tree_view->priv->open_dest_timeout = 0;
535 /* FIXME where do we clear column->window for each column? */
537 gdk_window_set_user_data (tree_view->priv->bin_window, NULL);
538 gdk_window_destroy (tree_view->priv->bin_window);
539 tree_view->priv->bin_window = NULL;
541 gdk_window_set_user_data (tree_view->priv->header_window, NULL);
542 gdk_window_destroy (tree_view->priv->header_window);
543 tree_view->priv->header_window = NULL;
545 gdk_cursor_destroy (tree_view->priv->cursor_drag);
546 gdk_gc_destroy (tree_view->priv->xor_gc);
548 /* GtkWidget::unrealize destroys children and widget->window */
550 if (GTK_WIDGET_CLASS (parent_class)->unrealize)
551 (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget);
555 gtk_tree_view_map_buttons (GtkTreeView *tree_view)
559 g_return_if_fail (GTK_WIDGET_MAPPED (tree_view));
561 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE))
563 GtkTreeViewColumn *column;
565 for (list = tree_view->priv->columns; list; list = list->next)
568 if (GTK_WIDGET_VISIBLE (column->button) &&
569 !GTK_WIDGET_MAPPED (column->button))
570 gtk_widget_map (column->button);
572 for (list = tree_view->priv->columns; list; list = list->next)
575 if (column->visible == FALSE)
577 if (column->column_type == GTK_TREE_VIEW_COLUMN_RESIZEABLE)
579 gdk_window_raise (column->window);
580 gdk_window_show (column->window);
583 gdk_window_hide (column->window);
585 gdk_window_show (tree_view->priv->header_window);
590 gtk_tree_view_map (GtkWidget *widget)
593 GtkTreeView *tree_view;
595 g_return_if_fail (widget != NULL);
596 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
598 tree_view = GTK_TREE_VIEW (widget);
600 GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED);
602 tmp_list = tree_view->priv->children;
605 GtkTreeViewChild *child = tmp_list->data;
606 tmp_list = tmp_list->next;
608 if (GTK_WIDGET_VISIBLE (child->widget))
610 if (!GTK_WIDGET_MAPPED (child->widget))
611 gtk_widget_map (child->widget);
614 gdk_window_show (tree_view->priv->bin_window);
616 gtk_tree_view_map_buttons (tree_view);
618 gdk_window_show (widget->window);
622 gtk_tree_view_size_request_buttons (GtkTreeView *tree_view)
626 tree_view->priv->header_height = 1;
628 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_MODEL_SETUP))
630 for (list = tree_view->priv->columns; list; list = list->next)
632 GtkRequisition requisition;
633 GtkTreeViewColumn *column;
637 gtk_widget_size_request (column->button, &requisition);
639 gtk_tree_view_column_set_width (column,
640 MAX (column->width, requisition.width));
641 tree_view->priv->header_height = MAX (tree_view->priv->header_height, requisition.height);
647 gtk_tree_view_size_request (GtkWidget *widget,
648 GtkRequisition *requisition)
650 GtkTreeView *tree_view;
653 g_return_if_fail (widget != NULL);
654 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
656 tree_view = GTK_TREE_VIEW (widget);
658 requisition->width = 200;
659 requisition->height = 200;
661 tmp_list = tree_view->priv->children;
665 GtkTreeViewChild *child = tmp_list->data;
666 GtkRequisition child_requisition;
668 tmp_list = tmp_list->next;
670 if (GTK_WIDGET_VISIBLE (child->widget))
671 gtk_widget_size_request (child->widget, &child_requisition);
674 gtk_tree_view_size_request_buttons (tree_view);
678 gtk_tree_view_size_allocate_buttons (GtkWidget *widget)
680 GtkTreeView *tree_view;
683 GtkTreeViewColumn *column;
684 GtkAllocation allocation;
687 tree_view = GTK_TREE_VIEW (widget);
690 allocation.height = tree_view->priv->header_height;
692 for (last_column = g_list_last (tree_view->priv->columns);
693 last_column && !(GTK_TREE_VIEW_COLUMN (last_column->data)->visible);
694 last_column = last_column->prev)
697 if (last_column == NULL)
700 for (list = tree_view->priv->columns; list != last_column; list = list->next)
704 if (!column->visible)
707 allocation.x = width;
708 allocation.width = column->width;
709 width += column->width;
710 gtk_widget_size_allocate (column->button, &allocation);
713 gdk_window_move_resize (column->window,
714 width - TREE_VIEW_DRAG_WIDTH/2, allocation.y,
715 TREE_VIEW_DRAG_WIDTH, allocation.height);
718 allocation.x = width;
719 allocation.width = MAX (widget->allocation.width, tree_view->priv->width) - width;
720 gtk_widget_size_allocate (column->button, &allocation);
722 gdk_window_move_resize (column->window,
723 allocation.x + allocation.width - TREE_VIEW_DRAG_WIDTH/2,
725 TREE_VIEW_DRAG_WIDTH, allocation.height);
729 gtk_tree_view_size_allocate (GtkWidget *widget,
730 GtkAllocation *allocation)
733 GtkTreeView *tree_view;
735 g_return_if_fail (widget != NULL);
736 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
738 widget->allocation = *allocation;
740 tree_view = GTK_TREE_VIEW (widget);
742 tmp_list = tree_view->priv->children;
746 GtkAllocation allocation;
747 GtkRequisition requisition;
749 GtkTreeViewChild *child = tmp_list->data;
750 tmp_list = tmp_list->next;
752 allocation.x = child->x;
753 allocation.y = child->y;
754 gtk_widget_get_child_requisition (child->widget, &requisition);
755 allocation.width = requisition.width;
756 allocation.height = requisition.height;
758 gtk_widget_size_allocate (child->widget, &allocation);
761 if (GTK_WIDGET_REALIZED (widget))
763 gdk_window_move_resize (widget->window,
764 allocation->x, allocation->y,
765 allocation->width, allocation->height);
766 gdk_window_move_resize (tree_view->priv->header_window,
768 MAX (tree_view->priv->width, allocation->width),
769 tree_view->priv->header_height);
772 /* FIXME I don't think the invariant that the model must be setup
773 * before touching the buttons is maintained in most of the
774 * rest of the code, e.g. in realize, so something is wrong
776 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_MODEL_SETUP))
777 gtk_tree_view_size_allocate_buttons (widget);
779 tree_view->priv->hadjustment->page_size = allocation->width;
780 tree_view->priv->hadjustment->page_increment = allocation->width / 2;
781 tree_view->priv->hadjustment->lower = 0;
782 tree_view->priv->hadjustment->upper = tree_view->priv->width;
783 if (tree_view->priv->hadjustment->value + allocation->width > tree_view->priv->width)
784 tree_view->priv->hadjustment->value = MAX (tree_view->priv->width - allocation->width, 0);
785 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->hadjustment), "changed");
787 tree_view->priv->vadjustment->page_size = allocation->height - TREE_VIEW_HEADER_HEIGHT (tree_view);
788 tree_view->priv->vadjustment->page_increment = (allocation->height - TREE_VIEW_HEADER_HEIGHT (tree_view)) / 2;
789 tree_view->priv->vadjustment->lower = 0;
790 tree_view->priv->vadjustment->upper = tree_view->priv->height;
791 if (tree_view->priv->vadjustment->value + allocation->height > tree_view->priv->height)
792 gtk_adjustment_set_value (tree_view->priv->vadjustment,
793 (gfloat) MAX (tree_view->priv->height - allocation->height, 0));
794 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->vadjustment), "changed");
798 gtk_tree_view_draw_node_focus_rect (GtkWidget *widget,
801 GtkTreeView *tree_view;
802 GtkRBTree *tree = NULL;
803 GtkRBNode *node = NULL;
804 gint bin_window_width = 0;
806 g_return_if_fail (widget != NULL);
807 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
809 tree_view = GTK_TREE_VIEW (widget);
811 _gtk_tree_view_find_node (tree_view, path, &tree, &node);
816 gdk_drawable_get_size (tree_view->priv->bin_window,
817 &bin_window_width, NULL);
819 /* FIXME need a style function appropriate for this */
820 gdk_draw_rectangle (tree_view->priv->bin_window,
821 widget->style->fg_gc[GTK_STATE_NORMAL],
824 _gtk_rbtree_node_find_offset (tree, node) + TREE_VIEW_HEADER_HEIGHT (tree_view),
825 bin_window_width - 2,
826 GTK_RBNODE_GET_HEIGHT (node));
830 gtk_tree_view_create_row_drag_icon (GtkTreeView *tree_view,
836 GtkCellRenderer *cell;
841 GdkRectangle background_area;
844 /* start drawing inside the black outline */
846 GdkDrawable *drawable;
847 gint bin_window_width;
849 widget = GTK_WIDGET (tree_view);
851 depth = gtk_tree_path_get_depth (path);
853 if (_gtk_tree_view_find_node (tree_view,
859 if (!gtk_tree_model_get_iter (tree_view->priv->model,
864 max_height = GTK_RBNODE_GET_HEIGHT (node);
868 background_area.y = y + TREE_VIEW_VERTICAL_SEPARATOR;
869 background_area.height = max_height - TREE_VIEW_VERTICAL_SEPARATOR;
871 gdk_drawable_get_size (tree_view->priv->bin_window,
872 &bin_window_width, NULL);
874 drawable = gdk_pixmap_new (tree_view->priv->bin_window,
875 bin_window_width + 2,
879 gdk_draw_rectangle (drawable,
880 widget->style->base_gc[GTK_WIDGET_STATE (widget)],
883 bin_window_width + 2,
886 gdk_draw_rectangle (drawable,
887 widget->style->black_gc,
890 bin_window_width + 1,
893 for (i = 0, list = tree_view->priv->columns; i < tree_view->priv->n_columns; i++, list = list->next)
895 GtkTreeViewColumn *column = list->data;
896 GdkRectangle cell_area;
898 if (!column->visible)
902 gtk_tree_view_column_set_cell_data (column,
903 tree_view->priv->model,
906 background_area.x = cell_offset;
907 background_area.width = TREE_VIEW_COLUMN_WIDTH (column);
909 cell_area = background_area;
911 if (i == tree_view->priv->expander_column &&
912 TREE_VIEW_DRAW_EXPANDERS(tree_view))
914 cell_area.x += depth * tree_view->priv->tab_offset;
915 cell_area.width -= depth * tree_view->priv->tab_offset;
918 gtk_cell_renderer_render (cell,
926 cell_offset += TREE_VIEW_COLUMN_WIDTH (column);
932 /* Warning: Very scary function.
933 * Modify at your own risk
936 gtk_tree_view_bin_expose (GtkWidget *widget,
937 GdkEventExpose *event)
939 GtkTreeView *tree_view;
943 GtkRBNode *node, *last_node = NULL;
944 GtkRBNode *cursor = NULL;
945 GtkRBTree *cursor_tree = NULL, *last_tree = NULL;
946 GtkRBNode *drag_highlight = NULL;
947 GtkRBTree *drag_highlight_tree = NULL;
949 GtkCellRenderer *cell;
951 gint y_offset, x_offset, cell_offset;
954 GdkRectangle background_area;
955 GdkRectangle cell_area;
957 gboolean last_selected;
959 gint bin_window_width;
961 g_return_val_if_fail (widget != NULL, FALSE);
962 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
964 tree_view = GTK_TREE_VIEW (widget);
966 if (tree_view->priv->tree == NULL)
969 gtk_tree_view_check_dirty (GTK_TREE_VIEW (widget));
970 /* we want to account for a potential HEADER offset.
971 * That is, if the header exists, we want to offset our event by its
972 * height to find the right node.
974 new_y = (event->area.y<TREE_VIEW_HEADER_HEIGHT (tree_view))?TREE_VIEW_HEADER_HEIGHT (tree_view):event->area.y;
975 y_offset = -_gtk_rbtree_find_offset (tree_view->priv->tree,
976 new_y - TREE_VIEW_HEADER_HEIGHT (tree_view),
978 &node) + new_y - event->area.y;
982 /* See if the last node was selected */
983 _gtk_rbtree_prev_full (tree, node, &last_tree, &last_node);
984 last_selected = (last_node && GTK_RBNODE_FLAG_SET (last_node, GTK_RBNODE_IS_SELECTED));
986 /* find the path for the node */
987 path = _gtk_tree_view_find_path ((GtkTreeView *)widget,
990 gtk_tree_model_get_iter (tree_view->priv->model,
993 depth = gtk_tree_path_get_depth (path);
994 gtk_tree_path_free (path);
996 if (tree_view->priv->cursor)
997 _gtk_tree_view_find_node (tree_view, tree_view->priv->cursor, &cursor_tree, &cursor);
999 if (tree_view->priv->drag_dest_row)
1000 _gtk_tree_view_find_node (tree_view, tree_view->priv->drag_dest_row,
1001 &drag_highlight_tree, &drag_highlight);
1003 gdk_drawable_get_size (tree_view->priv->bin_window,
1004 &bin_window_width, NULL);
1006 /* Actually process the expose event. To do this, we want to
1007 * start at the first node of the event, and walk the tree in
1008 * order, drawing each successive node.
1013 /* Need to think about this more.
1014 if (tree_view->priv->show_expanders)
1015 max_height = MAX (TREE_VIEW_EXPANDER_MIN_HEIGHT, GTK_RBNODE_GET_HEIGHT (node));
1018 max_height = GTK_RBNODE_GET_HEIGHT (node);
1020 x_offset = -event->area.x;
1022 highlight_x = 0; /* should match x coord of first cell */
1024 background_area.y = y_offset + event->area.y + TREE_VIEW_VERTICAL_SEPARATOR;
1025 background_area.height = max_height - TREE_VIEW_VERTICAL_SEPARATOR;
1028 if (GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_PRELIT))
1029 flags |= GTK_CELL_RENDERER_PRELIT;
1031 if (GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_SELECTED))
1033 flags |= GTK_CELL_RENDERER_SELECTED;
1035 /* Draw the selection */
1036 gdk_draw_rectangle (event->window,
1037 GTK_WIDGET (tree_view)->style->bg_gc [GTK_STATE_SELECTED],
1040 background_area.y - (last_selected?TREE_VIEW_VERTICAL_SEPARATOR:0),
1042 background_area.height + (last_selected?TREE_VIEW_VERTICAL_SEPARATOR:0));
1043 last_selected = TRUE;
1047 last_selected = FALSE;
1050 for (i = 0, list = tree_view->priv->columns; i < tree_view->priv->n_columns; i++, list = list->next)
1052 GtkTreeViewColumn *column = list->data;
1054 if (!column->visible)
1057 cell = column->cell;
1058 gtk_tree_view_column_set_cell_data (column,
1059 tree_view->priv->model,
1062 background_area.x = cell_offset;
1063 background_area.width = TREE_VIEW_COLUMN_WIDTH (column);
1064 if (i == tree_view->priv->expander_column &&
1065 TREE_VIEW_DRAW_EXPANDERS(tree_view))
1067 cell_area = background_area;
1068 cell_area.x += depth*tree_view->priv->tab_offset;
1069 cell_area.width -= depth*tree_view->priv->tab_offset;
1071 /* If we have an expander column, the highlight underline
1072 * starts with that column, so that it indicates which
1073 * level of the tree we're dropping at.
1075 highlight_x = cell_area.x;
1077 gtk_cell_renderer_render (cell,
1084 if ((node->flags & GTK_RBNODE_IS_PARENT) == GTK_RBNODE_IS_PARENT)
1087 gdk_window_get_pointer (tree_view->priv->bin_window, &x, &y, 0);
1088 gtk_tree_view_draw_arrow (GTK_TREE_VIEW (widget),
1096 cell_area = background_area;
1097 gtk_cell_renderer_render (cell,
1105 cell_offset += TREE_VIEW_COLUMN_WIDTH (column);
1108 if (node == cursor &&
1109 GTK_WIDGET_HAS_FOCUS (widget))
1110 gtk_tree_view_draw_focus (widget);
1112 if (node == drag_highlight)
1114 /* Draw indicator for the drop
1116 gint highlight_y = -1;
1118 switch (tree_view->priv->drag_dest_pos)
1120 case GTK_TREE_VIEW_DROP_BEFORE:
1121 highlight_y = background_area.y - TREE_VIEW_VERTICAL_SEPARATOR/2;
1124 case GTK_TREE_VIEW_DROP_AFTER:
1125 highlight_y = background_area.y + background_area.height + TREE_VIEW_VERTICAL_SEPARATOR/2;
1128 case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
1129 case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
1130 gtk_tree_view_draw_node_focus_rect (widget,
1131 tree_view->priv->drag_dest_row);
1135 if (highlight_y >= 0)
1137 gdk_draw_line (event->window,
1138 widget->style->black_gc,
1141 bin_window_width - highlight_x,
1146 y_offset += max_height;
1149 GtkTreeIter parent = iter;
1152 tree = node->children;
1154 while (node->left != tree->nil)
1156 has_child = gtk_tree_model_iter_children (tree_view->priv->model,
1159 cell = gtk_tree_view_get_column (tree_view, 0)->cell;
1163 TREE_VIEW_INTERNAL_ASSERT (has_child, FALSE);
1167 gboolean done = FALSE;
1170 node = _gtk_rbtree_next (tree, node);
1173 gboolean has_next = gtk_tree_model_iter_next (tree_view->priv->model, &iter);
1174 cell = gtk_tree_view_get_column (tree_view, 0)->cell;
1178 TREE_VIEW_INTERNAL_ASSERT (has_next, FALSE);
1182 GtkTreeIter parent_iter = iter;
1183 gboolean has_parent;
1185 node = tree->parent_node;
1186 tree = tree->parent_tree;
1188 /* we've run out of tree. It's okay to return though, as
1189 * we'd only break out of the while loop below. */
1191 has_parent = gtk_tree_model_iter_parent (tree_view->priv->model,
1197 TREE_VIEW_INTERNAL_ASSERT (has_parent, FALSE);
1203 while (y_offset < event->area.height);
1209 gtk_tree_view_expose (GtkWidget *widget,
1210 GdkEventExpose *event)
1212 GtkTreeView *tree_view;
1214 g_return_val_if_fail (widget != NULL, FALSE);
1215 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1217 tree_view = GTK_TREE_VIEW (widget);
1219 if (event->window == tree_view->priv->bin_window)
1220 return gtk_tree_view_bin_expose (widget, event);
1226 coords_are_over_arrow (GtkTreeView *tree_view,
1229 /* these are in tree window coords */
1236 if ((node->flags & GTK_RBNODE_IS_PARENT) == 0)
1239 arrow.y = _gtk_rbtree_node_find_offset (tree, node) + TREE_VIEW_HEADER_HEIGHT (tree_view);
1241 arrow.height = GTK_RBNODE_GET_HEIGHT (node);
1243 gtk_tree_view_get_arrow_range (tree_view, &arrow.x, &x2);
1245 arrow.width = x2 - arrow.x;
1247 return (x >= arrow.x &&
1248 x < (arrow.x + arrow.height) &&
1250 y < (arrow.y + arrow.height));
1254 do_unprelight (GtkTreeView *tree_view,
1255 /* these are in tree window coords */
1261 if (tree_view->priv->prelight_node == NULL)
1264 y1 = _gtk_rbtree_node_find_offset (tree_view->priv->prelight_tree,
1265 tree_view->priv->prelight_node) +
1266 TREE_VIEW_HEADER_HEIGHT (tree_view);
1268 y2 = y1 + GTK_RBNODE_GET_HEIGHT (tree_view->priv->prelight_node);
1270 if (tree_view->priv->prelight_node)
1271 GTK_RBNODE_UNSET_FLAG (tree_view->priv->prelight_node, GTK_RBNODE_IS_PRELIT);
1273 /* FIXME queue draw on y1-y2 range */
1275 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_ARROW_PRELIT) &&
1276 !coords_are_over_arrow (tree_view,
1277 tree_view->priv->prelight_tree,
1278 tree_view->priv->prelight_node,
1281 /* We need to unprelight the old arrow. */
1283 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_ARROW_PRELIT);
1285 gtk_tree_view_draw_arrow (tree_view,
1286 tree_view->priv->prelight_tree,
1287 tree_view->priv->prelight_node,
1293 tree_view->priv->prelight_node = NULL;
1294 tree_view->priv->prelight_tree = NULL;
1297 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
1301 do_prelight (GtkTreeView *tree_view,
1304 /* these are in tree window coords */
1308 if (coords_are_over_arrow (tree_view, tree, node, x, y))
1309 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_ARROW_PRELIT);
1311 tree_view->priv->prelight_node = node;
1312 tree_view->priv->prelight_tree = tree;
1314 GTK_RBNODE_SET_FLAG (node, GTK_RBNODE_IS_PRELIT);
1316 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
1320 gtk_tree_view_motion (GtkWidget *widget,
1321 GdkEventMotion *event)
1323 GtkTreeView *tree_view;
1328 tree_view = (GtkTreeView *) widget;
1330 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_IN_COLUMN_RESIZE))
1335 if (event->is_hint || event->window != widget->window)
1336 gtk_widget_get_pointer (widget, &x, NULL);
1340 new_width = gtk_tree_view_new_column_width (GTK_TREE_VIEW (widget), tree_view->priv->drag_pos, &x);
1341 if (x != tree_view->priv->x_drag)
1343 gtk_tree_view_column_set_width (gtk_tree_view_get_column (GTK_TREE_VIEW (widget), tree_view->priv->drag_pos), new_width);
1346 /* FIXME: Do we need to scroll */
1347 _gtk_tree_view_set_size (GTK_TREE_VIEW (widget), -1, tree_view->priv->height);
1351 /* Sanity check it */
1352 if (event->window != tree_view->priv->bin_window)
1355 if (tree_view->priv->tree == NULL)
1358 gtk_tree_view_maybe_begin_dragging_row (tree_view, event);
1360 do_unprelight (tree_view, event->x, event->y);
1362 new_y = ((gint)event->y<TREE_VIEW_HEADER_HEIGHT (tree_view))?TREE_VIEW_HEADER_HEIGHT (tree_view):(gint)event->y;
1364 _gtk_rbtree_find_offset (tree_view->priv->tree, new_y - TREE_VIEW_HEADER_HEIGHT (tree_view),
1371 /* If we are currently pressing down a button, we don't want to prelight anything else. */
1372 if ((tree_view->priv->button_pressed_node != NULL) &&
1373 (tree_view->priv->button_pressed_node != node))
1377 do_prelight (tree_view, tree, node, event->x, new_y);
1382 /* FIXME Is this function necessary? Can I get an enter_notify event
1383 * w/o either an expose event or a mouse motion event?
1386 gtk_tree_view_enter_notify (GtkWidget *widget,
1387 GdkEventCrossing *event)
1389 GtkTreeView *tree_view;
1394 g_return_val_if_fail (widget != NULL, FALSE);
1395 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1397 tree_view = GTK_TREE_VIEW (widget);
1399 /* Sanity check it */
1400 if (event->window != tree_view->priv->bin_window)
1403 if (tree_view->priv->tree == NULL)
1406 if ((tree_view->priv->button_pressed_node != NULL) &&
1407 (tree_view->priv->button_pressed_node != node))
1410 /* find the node internally */
1411 new_y = ((gint)event->y<TREE_VIEW_HEADER_HEIGHT (tree_view))?TREE_VIEW_HEADER_HEIGHT (tree_view):(gint)event->y;
1413 _gtk_rbtree_find_offset (tree_view->priv->tree,
1414 new_y - TREE_VIEW_HEADER_HEIGHT (tree_view),
1421 do_prelight (tree_view, tree, node, event->x, new_y);
1427 gtk_tree_view_leave_notify (GtkWidget *widget,
1428 GdkEventCrossing *event)
1430 GtkTreeView *tree_view;
1432 g_return_val_if_fail (widget != NULL, FALSE);
1433 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1435 tree_view = GTK_TREE_VIEW (widget);
1437 do_unprelight (tree_view, -1000, -1000); /* coords not possibly over an arrow */
1443 gtk_tree_view_button_press (GtkWidget *widget,
1444 GdkEventButton *event)
1446 GtkTreeView *tree_view;
1448 GtkTreeViewColumn *column;
1450 GdkRectangle background_area;
1451 GdkRectangle cell_area;
1453 g_return_val_if_fail (widget != NULL, FALSE);
1454 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1455 g_return_val_if_fail (event != NULL, FALSE);
1457 tree_view = GTK_TREE_VIEW (widget);
1459 if (event->window == tree_view->priv->bin_window)
1469 if (!GTK_WIDGET_HAS_FOCUS (widget))
1470 gtk_widget_grab_focus (widget);
1471 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_DRAW_KEYFOCUS);
1473 /* are we in an arrow? */
1474 if (tree_view->priv->prelight_node &&
1475 GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_ARROW_PRELIT))
1477 if (event->button == 1)
1479 gtk_grab_add (widget);
1480 tree_view->priv->button_pressed_node = tree_view->priv->prelight_node;
1481 tree_view->priv->button_pressed_tree = tree_view->priv->prelight_tree;
1482 gtk_tree_view_draw_arrow (GTK_TREE_VIEW (widget),
1483 tree_view->priv->prelight_tree,
1484 tree_view->priv->prelight_node,
1491 /* find the node that was clicked */
1492 new_y = ((gint)event->y<TREE_VIEW_HEADER_HEIGHT (tree_view))?TREE_VIEW_HEADER_HEIGHT (tree_view):(gint)event->y;
1493 y_offset = -_gtk_rbtree_find_offset (tree_view->priv->tree,
1494 new_y - TREE_VIEW_HEADER_HEIGHT (tree_view),
1496 &node) + new_y - (gint)event->y;
1499 /* We clicked in dead space */
1502 /* Get the path and the node */
1503 path = _gtk_tree_view_find_path (tree_view, tree, node);
1504 depth = gtk_tree_path_get_depth (path);
1505 background_area.y = y_offset + event->y + TREE_VIEW_VERTICAL_SEPARATOR;
1506 background_area.height = GTK_RBNODE_GET_HEIGHT (node) - TREE_VIEW_VERTICAL_SEPARATOR;
1507 background_area.x = 0;
1508 /* Let the cell have a chance at selecting it. */
1510 for (i = 0, list = tree_view->priv->columns; i < tree_view->priv->n_columns; i++, list = list->next)
1512 GtkTreeViewColumn *column = list->data;
1513 GtkCellRenderer *cell;
1516 if (!column->visible)
1519 background_area.width = TREE_VIEW_COLUMN_WIDTH (column);
1520 if (i == tree_view->priv->expander_column &&
1521 TREE_VIEW_DRAW_EXPANDERS(tree_view))
1523 cell_area = background_area;
1524 cell_area.x += depth*tree_view->priv->tab_offset;
1525 cell_area.width -= depth*tree_view->priv->tab_offset;
1529 cell_area = background_area;
1532 cell = column->cell;
1534 if ((background_area.x > (gint) event->x) ||
1535 (background_area.y > (gint) event->y) ||
1536 (background_area.x + background_area.width <= (gint) event->x) ||
1537 (background_area.y + background_area.height <= (gint) event->y))
1539 background_area.x += background_area.width;
1543 gtk_tree_model_get_iter (tree_view->priv->model,
1546 gtk_tree_view_column_set_cell_data (column,
1547 tree_view->priv->model,
1550 path_string = gtk_tree_path_to_string (path);
1551 if (gtk_cell_renderer_event (cell,
1560 g_free (path_string);
1561 gtk_tree_path_free (path);
1566 g_free (path_string);
1571 /* Save press to possibly begin a drag
1573 if (tree_view->priv->pressed_button < 0)
1575 tree_view->priv->pressed_button = event->button;
1576 tree_view->priv->press_start_x = event->x;
1577 tree_view->priv->press_start_y = event->y;
1580 /* Handle the selection */
1581 if (tree_view->priv->selection == NULL)
1582 tree_view->priv->selection =
1583 _gtk_tree_selection_new_with_tree_view (tree_view);
1585 _gtk_tree_selection_internal_select_node (tree_view->priv->selection,
1590 gtk_tree_path_free (path);
1594 for (i = 0, list = tree_view->priv->columns; list; list = list->next, i++)
1596 column = list->data;
1597 if (event->window == column->window &&
1598 column->column_type == GTK_TREE_VIEW_COLUMN_RESIZEABLE &&
1603 if (gdk_pointer_grab (column->window, FALSE,
1604 GDK_POINTER_MOTION_HINT_MASK |
1605 GDK_BUTTON1_MOTION_MASK |
1606 GDK_BUTTON_RELEASE_MASK,
1607 NULL, NULL, event->time))
1610 gtk_grab_add (widget);
1611 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_IN_COLUMN_RESIZE);
1613 /* block attached dnd signal handler */
1614 drag_data = gtk_object_get_data (GTK_OBJECT (widget), "gtk-site-data");
1616 gtk_signal_handler_block_by_data (GTK_OBJECT (widget), drag_data);
1618 if (!GTK_WIDGET_HAS_FOCUS (widget))
1619 gtk_widget_grab_focus (widget);
1621 tree_view->priv->drag_pos = i;
1622 tree_view->priv->x_drag = (column->button->allocation.x + column->button->allocation.width);
1629 gtk_tree_view_button_release (GtkWidget *widget,
1630 GdkEventButton *event)
1632 GtkTreeView *tree_view;
1634 g_return_val_if_fail (widget != NULL, FALSE);
1635 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1636 g_return_val_if_fail (event != NULL, FALSE);
1638 tree_view = GTK_TREE_VIEW (widget);
1640 if (tree_view->priv->pressed_button == event->button)
1641 tree_view->priv->pressed_button = -1;
1643 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_IN_COLUMN_RESIZE))
1650 i = tree_view->priv->drag_pos;
1651 tree_view->priv->drag_pos = -1;
1653 /* unblock attached dnd signal handler */
1654 drag_data = gtk_object_get_data (GTK_OBJECT (widget), "gtk-site-data");
1656 gtk_signal_handler_unblock_by_data (GTK_OBJECT (widget), drag_data);
1658 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_IN_COLUMN_RESIZE);
1659 gtk_widget_get_pointer (widget, &x, NULL);
1660 gtk_grab_remove (widget);
1661 gdk_pointer_ungrab (event->time);
1663 width = gtk_tree_view_new_column_width (GTK_TREE_VIEW (widget), i, &x);
1664 gtk_tree_view_column_set_width (gtk_tree_view_get_column (GTK_TREE_VIEW (widget), i), width);
1668 if (tree_view->priv->button_pressed_node == NULL)
1671 if (event->button == 1)
1673 gtk_grab_remove (widget);
1674 if (tree_view->priv->button_pressed_node == tree_view->priv->prelight_node &&
1675 GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_ARROW_PRELIT))
1680 /* Actually activate the node */
1681 if (tree_view->priv->button_pressed_node->children == NULL)
1684 path = _gtk_tree_view_find_path (GTK_TREE_VIEW (widget),
1685 tree_view->priv->button_pressed_tree,
1686 tree_view->priv->button_pressed_node);
1687 tree_view->priv->button_pressed_node->children = _gtk_rbtree_new ();
1688 tree_view->priv->button_pressed_node->children->parent_tree = tree_view->priv->button_pressed_tree;
1689 tree_view->priv->button_pressed_node->children->parent_node = tree_view->priv->button_pressed_node;
1690 gtk_tree_model_get_iter (tree_view->priv->model, &iter, path);
1691 gtk_tree_model_iter_children (tree_view->priv->model, &child, &iter);
1693 gtk_tree_view_build_tree (tree_view,
1694 tree_view->priv->button_pressed_node->children,
1696 gtk_tree_path_get_depth (path) + 1,
1698 GTK_WIDGET_REALIZED (widget));
1702 path = _gtk_tree_view_find_path (GTK_TREE_VIEW (widget),
1703 tree_view->priv->button_pressed_node->children,
1704 tree_view->priv->button_pressed_node->children->root);
1705 gtk_tree_model_get_iter (tree_view->priv->model,
1709 gtk_tree_view_discover_dirty (GTK_TREE_VIEW (widget),
1710 tree_view->priv->button_pressed_node->children,
1712 gtk_tree_path_get_depth (path));
1713 _gtk_rbtree_remove (tree_view->priv->button_pressed_node->children);
1715 gtk_tree_path_free (path);
1717 _gtk_tree_view_set_size (GTK_TREE_VIEW (widget), -1, -1);
1720 tree_view->priv->button_pressed_node = NULL;
1728 gtk_tree_view_draw_focus (GtkWidget *widget)
1730 GtkTreeView *tree_view;
1732 g_return_if_fail (widget != NULL);
1733 g_return_if_fail (GTK_IS_TREE_VIEW (widget));
1735 tree_view = GTK_TREE_VIEW (widget);
1737 if (! GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_DRAW_KEYFOCUS))
1739 if (tree_view->priv->cursor == NULL)
1742 gtk_tree_view_draw_node_focus_rect (widget, tree_view->priv->cursor);
1747 gtk_tree_view_focus_in (GtkWidget *widget,
1748 GdkEventFocus *event)
1750 GtkTreeView *tree_view;
1752 g_return_val_if_fail (widget != NULL, FALSE);
1753 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1754 g_return_val_if_fail (event != NULL, FALSE);
1756 tree_view = GTK_TREE_VIEW (widget);
1758 GTK_WIDGET_SET_FLAGS (widget, GTK_HAS_FOCUS);
1760 /* FIXME don't redraw so much */
1761 gtk_widget_queue_draw (widget);
1768 gtk_tree_view_focus_out (GtkWidget *widget,
1769 GdkEventFocus *event)
1771 g_return_val_if_fail (widget != NULL, FALSE);
1772 g_return_val_if_fail (GTK_IS_TREE_VIEW (widget), FALSE);
1773 g_return_val_if_fail (event != NULL, FALSE);
1775 GTK_WIDGET_UNSET_FLAGS (widget, GTK_HAS_FOCUS);
1777 /* FIXME don't redraw so much */
1778 gtk_widget_queue_draw (widget);
1783 /* FIXME: It would be neat to someday make the headers a seperate widget that
1784 * can be shared between various apps. Wishful thinking, though...
1786 /* Returns TRUE if the focus is within the headers, after the focus operation is
1790 gtk_tree_view_header_focus (GtkTreeView *tree_view,
1791 GtkDirectionType dir)
1793 GtkWidget *focus_child;
1794 GtkContainer *container;
1796 GList *last_column, *first_column;
1799 if (! GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE))
1802 focus_child = GTK_CONTAINER (tree_view)->focus_child;
1803 container = GTK_CONTAINER (tree_view);
1805 for (last_column = g_list_last (tree_view->priv->columns);
1807 !(GTK_TREE_VIEW_COLUMN (last_column->data)->visible) &&
1808 GTK_WIDGET_CAN_FOCUS (GTK_TREE_VIEW_COLUMN (last_column->data)->button);
1809 last_column = last_column->prev)
1812 for (first_column = tree_view->priv->columns;
1814 !(GTK_TREE_VIEW_COLUMN (first_column->data)->visible) &&
1815 GTK_WIDGET_CAN_FOCUS (GTK_TREE_VIEW_COLUMN (first_column->data)->button);
1816 first_column = first_column->next)
1819 /* no headers are visible, or are focussable. We can't focus in or out.
1820 * I wonder if focussable is a real word...
1822 if (last_column == NULL)
1825 /* First thing we want to handle is entering and leaving the headers.
1829 case GTK_DIR_TAB_BACKWARD:
1832 focus_child = GTK_TREE_VIEW_COLUMN (last_column->data)->button;
1833 gtk_widget_grab_focus (focus_child);
1836 if (focus_child == GTK_TREE_VIEW_COLUMN (first_column->data)->button)
1843 case GTK_DIR_TAB_FORWARD:
1846 focus_child = GTK_TREE_VIEW_COLUMN (first_column->data)->button;
1847 gtk_widget_grab_focus (focus_child);
1850 if (focus_child == GTK_TREE_VIEW_COLUMN (last_column->data)->button)
1860 focus_child = GTK_TREE_VIEW_COLUMN (last_column->data)->button;
1861 gtk_widget_grab_focus (focus_child);
1864 if (focus_child == GTK_TREE_VIEW_COLUMN (first_column->data)->button)
1874 focus_child = GTK_TREE_VIEW_COLUMN (first_column->data)->button;
1875 gtk_widget_grab_focus (focus_child);
1878 if (focus_child == GTK_TREE_VIEW_COLUMN (last_column->data)->button)
1888 focus_child = GTK_TREE_VIEW_COLUMN (first_column->data)->button;
1889 gtk_widget_grab_focus (focus_child);
1900 focus_child = GTK_TREE_VIEW_COLUMN (first_column->data)->button;
1901 gtk_widget_grab_focus (focus_child);
1910 /* We need to move the focus to the next button. */
1913 for (tmp_list = tree_view->priv->columns; tmp_list; tmp_list = tmp_list->next)
1914 if (GTK_TREE_VIEW_COLUMN (tmp_list->data)->button == focus_child)
1916 if (gtk_container_focus (GTK_CONTAINER (GTK_TREE_VIEW_COLUMN (tmp_list->data)->button), dir))
1918 /* The focus moves inside the button. */
1919 /* This is probably a great example of bad UI */
1925 /* We need to move the focus among the row of buttons. */
1928 GtkTreeViewColumn *column;
1930 if (dir == GTK_DIR_RIGHT || dir == GTK_DIR_TAB_FORWARD)
1931 tmp_list = tmp_list->next;
1933 tmp_list = tmp_list->prev;
1935 if (tmp_list == NULL)
1937 g_warning ("Internal button not found");
1940 column = tmp_list->data;
1941 if (column->button &&
1943 GTK_WIDGET_CAN_FOCUS (column->button))
1945 focus_child = column->button;
1946 gtk_widget_grab_focus (column->button);
1953 /* if focus child is non-null, we assume it's been set to the current focus child
1957 /* If the following isn't true, then the view is smaller then the scrollpane.
1959 if ((focus_child->allocation.x + focus_child->allocation.width) <=
1960 (tree_view->priv->hadjustment->upper))
1962 /* Scroll to the button, if needed */
1963 if ((tree_view->priv->hadjustment->value + tree_view->priv->hadjustment->page_size) <
1964 (focus_child->allocation.x + focus_child->allocation.width))
1965 gtk_adjustment_set_value (tree_view->priv->hadjustment,
1966 focus_child->allocation.x + focus_child->allocation.width -
1967 tree_view->priv->hadjustment->page_size);
1968 else if (tree_view->priv->hadjustment->value > focus_child->allocation.x)
1969 gtk_adjustment_set_value (tree_view->priv->hadjustment,
1970 focus_child->allocation.x);
1974 return (focus_child != NULL);
1977 /* WARNING: Scary function */
1979 gtk_tree_view_focus (GtkContainer *container,
1980 GtkDirectionType direction)
1982 GtkTreeView *tree_view;
1983 GtkWidget *focus_child;
1985 GtkRBTree *cursor_tree;
1986 GtkRBNode *cursor_node;
1988 g_return_val_if_fail (container != NULL, FALSE);
1989 g_return_val_if_fail (GTK_IS_TREE_VIEW (container), FALSE);
1990 g_return_val_if_fail (GTK_WIDGET_VISIBLE (container), FALSE);
1992 tree_view = GTK_TREE_VIEW (container);
1994 if (!GTK_WIDGET_IS_SENSITIVE (container))
1996 if (tree_view->priv->tree == NULL)
1999 focus_child = container->focus_child;
2001 /* Case 1. Headers have focus. */
2007 case GTK_DIR_TAB_BACKWARD:
2008 return (gtk_tree_view_header_focus (tree_view, direction));
2011 case GTK_DIR_TAB_FORWARD:
2014 if (direction != GTK_DIR_DOWN)
2016 if (gtk_tree_view_header_focus (tree_view, direction))
2019 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_DRAW_KEYFOCUS);
2020 gtk_widget_grab_focus (GTK_WIDGET (container));
2022 if (tree_view->priv->selection == NULL)
2023 tree_view->priv->selection =
2024 _gtk_tree_selection_new_with_tree_view (tree_view);
2026 /* if there is no keyboard focus yet, we select the first node
2028 if (tree_view->priv->cursor == NULL)
2029 tree_view->priv->cursor = gtk_tree_path_new_root ();
2031 gtk_tree_selection_select_path (tree_view->priv->selection,
2032 tree_view->priv->cursor);
2033 /* FIXME make this more efficient */
2034 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
2039 /* Case 2. We don't have focus at all. */
2040 if (!GTK_WIDGET_HAS_FOCUS (container))
2042 if ((direction == GTK_DIR_TAB_FORWARD) ||
2043 (direction == GTK_DIR_RIGHT) ||
2044 (direction == GTK_DIR_DOWN))
2046 if (gtk_tree_view_header_focus (tree_view, direction))
2050 /* The headers didn't want the focus, so we take it. */
2051 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_DRAW_KEYFOCUS);
2052 gtk_widget_grab_focus (GTK_WIDGET (container));
2054 if (tree_view->priv->selection == NULL)
2055 tree_view->priv->selection =
2056 _gtk_tree_selection_new_with_tree_view (tree_view);
2058 if (tree_view->priv->cursor == NULL)
2059 tree_view->priv->cursor = gtk_tree_path_new_root ();
2061 gtk_tree_selection_select_path (tree_view->priv->selection,
2062 tree_view->priv->cursor);
2063 /* FIXME make this more efficient */
2064 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
2068 /* Case 3. We have focus already, but no cursor. We pick the first one
2069 * and run with it. */
2070 if (tree_view->priv->cursor == NULL)
2072 /* We lost our cursor somehow. Arbitrarily select the first node, and
2075 tree_view->priv->cursor = gtk_tree_path_new_root ();
2077 gtk_tree_selection_select_path (tree_view->priv->selection,
2078 tree_view->priv->cursor);
2079 gtk_adjustment_set_value (GTK_ADJUSTMENT (tree_view->priv->vadjustment),
2081 /* FIXME make this more efficient */
2082 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
2087 /* Case 3. We have focus already. Move the cursor. */
2088 if (direction == GTK_DIR_LEFT)
2091 val = tree_view->priv->hadjustment->value - tree_view->priv->hadjustment->page_size/2;
2092 val = MAX (val, 0.0);
2093 gtk_adjustment_set_value (GTK_ADJUSTMENT (tree_view->priv->hadjustment), val);
2094 gtk_widget_grab_focus (GTK_WIDGET (tree_view));
2097 if (direction == GTK_DIR_RIGHT)
2100 val = tree_view->priv->hadjustment->value + tree_view->priv->hadjustment->page_size/2;
2101 val = MIN (tree_view->priv->hadjustment->upper - tree_view->priv->hadjustment->page_size, val);
2102 gtk_adjustment_set_value (GTK_ADJUSTMENT (tree_view->priv->hadjustment), val);
2103 gtk_widget_grab_focus (GTK_WIDGET (tree_view));
2109 _gtk_tree_view_find_node (tree_view, tree_view->priv->cursor,
2114 case GTK_DIR_TAB_BACKWARD:
2116 _gtk_rbtree_prev_full (cursor_tree,
2121 case GTK_DIR_TAB_FORWARD:
2123 _gtk_rbtree_next_full (cursor_tree,
2134 GdkModifierType state = 0;
2136 event = gtk_get_current_event ();
2138 gdk_event_get_state (event, &state);
2141 gdk_event_free (event);
2142 gtk_tree_path_free (tree_view->priv->cursor);
2144 tree_view->priv->cursor = _gtk_tree_view_find_path (tree_view,
2147 if (tree_view->priv->cursor)
2148 _gtk_tree_selection_internal_select_node (tree_view->priv->selection,
2151 tree_view->priv->cursor,
2153 gtk_tree_view_clamp_node_visible (tree_view, cursor_tree, cursor_node);
2154 gtk_widget_grab_focus (GTK_WIDGET (tree_view));
2155 /* FIXME make this more efficient */
2156 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
2160 /* At this point, we've progressed beyond the edge of the rows. */
2162 if ((direction == GTK_DIR_LEFT) ||
2163 (direction == GTK_DIR_TAB_BACKWARD) ||
2164 (direction == GTK_DIR_UP))
2165 /* We can't go back anymore. Try the headers */
2166 return (gtk_tree_view_header_focus (tree_view, direction));
2168 /* we've reached the end of the tree. Go on. */
2175 gtk_tree_view_remove (GtkContainer *container,
2178 GtkTreeView *tree_view;
2179 GtkTreeViewChild *child = NULL;
2182 g_return_if_fail (container != NULL);
2183 g_return_if_fail (GTK_IS_TREE_VIEW (container));
2185 tree_view = GTK_TREE_VIEW (container);
2187 tmp_list = tree_view->priv->children;
2190 child = tmp_list->data;
2191 if (child->widget == widget)
2193 tmp_list = tmp_list->next;
2198 gtk_widget_unparent (widget);
2200 tree_view->priv->children = g_list_remove_link (tree_view->priv->children, tmp_list);
2201 g_list_free_1 (tmp_list);
2207 gtk_tree_view_forall (GtkContainer *container,
2208 gboolean include_internals,
2209 GtkCallback callback,
2210 gpointer callback_data)
2212 GtkTreeView *tree_view;
2213 GtkTreeViewChild *child = NULL;
2214 GtkTreeViewColumn *column;
2217 g_return_if_fail (container != NULL);
2218 g_return_if_fail (GTK_IS_TREE_VIEW (container));
2219 g_return_if_fail (callback != NULL);
2221 tree_view = GTK_TREE_VIEW (container);
2223 tmp_list = tree_view->priv->children;
2226 child = tmp_list->data;
2227 tmp_list = tmp_list->next;
2229 (* callback) (child->widget, callback_data);
2231 if (include_internals == FALSE)
2234 for (tmp_list = tree_view->priv->columns; tmp_list; tmp_list = tmp_list->next)
2236 column = tmp_list->data;
2238 (* callback) (column->button, callback_data);
2242 /* TreeModel Callbacks
2246 gtk_tree_view_changed (GtkTreeModel *model,
2251 GtkTreeView *tree_view = (GtkTreeView *)data;
2255 gboolean dirty_marked;
2257 g_return_if_fail (path != NULL || iter != NULL);
2260 path = gtk_tree_model_get_path (model, iter);
2261 else if (iter == NULL)
2262 gtk_tree_model_get_iter (model, iter, path);
2264 if (_gtk_tree_view_find_node (tree_view,
2268 /* We aren't actually showing the node */
2271 dirty_marked = gtk_tree_view_discover_dirty_iter (tree_view,
2273 gtk_tree_path_get_depth (path),
2276 if (GTK_RBNODE_GET_HEIGHT (node) != height + TREE_VIEW_VERTICAL_SEPARATOR)
2278 _gtk_rbtree_node_set_height (tree, node, height + TREE_VIEW_VERTICAL_SEPARATOR);
2279 gtk_widget_queue_resize (GTK_WIDGET (data));
2283 gtk_widget_queue_resize (GTK_WIDGET (data));
2286 /* FIXME: just redraw the node */
2287 gtk_widget_queue_draw (GTK_WIDGET (data));
2292 gtk_tree_view_inserted (GtkTreeModel *model,
2297 GtkTreeView *tree_view = (GtkTreeView *) data;
2299 GtkRBTree *tmptree, *tree;
2300 GtkRBNode *tmpnode = NULL;
2305 tmptree = tree = tree_view->priv->tree;
2306 g_return_if_fail (path != NULL || iter != NULL);
2309 path = gtk_tree_model_get_path (model, iter);
2310 else if (iter == NULL)
2311 gtk_tree_model_get_iter (model, iter, path);
2313 depth = gtk_tree_path_get_depth (path);
2314 indices = gtk_tree_path_get_indices (path);
2316 /* First, find the parent tree */
2317 while (i < depth - 1)
2319 if (tmptree == NULL)
2321 /* We aren't showing the node */
2325 tmpnode = _gtk_rbtree_find_count (tmptree, indices[i] + 1);
2326 if (tmpnode == NULL)
2328 g_warning ("A node was inserted with a parent that's not in the tree.\n" \
2329 "This possibly means that a GtkTreeModel inserted a child node\n" \
2330 "before the parent was inserted.");
2333 else if (!GTK_RBNODE_FLAG_SET (tmpnode, GTK_RBNODE_IS_PARENT))
2335 /* FIXME enforce correct behavior on model, probably */
2336 /* In theory, the model should have emitted child_toggled here. We
2337 * try to catch it anyway, just to be safe, in case the model hasn't.
2339 GtkTreePath *tmppath = _gtk_tree_view_find_path (tree_view,
2342 gtk_tree_view_child_toggled (model, tmppath, NULL, data);
2343 gtk_tree_path_free (tmppath);
2347 tmptree = tmpnode->children;
2355 /* next, update the selection */
2356 if (tree_view->priv->anchor)
2358 gint *select_indices = gtk_tree_path_get_indices (tree_view->priv->anchor);
2359 gint select_depth = gtk_tree_path_get_depth (tree_view->priv->anchor);
2361 for (i = 0; i < depth && i < select_depth; i++)
2363 if (indices[i] < select_indices[i])
2365 select_indices[i]++;
2368 else if (indices[i] > select_indices[i])
2370 else if (i == depth - 1)
2372 select_indices[i]++;
2379 gtk_tree_model_ref_iter (tree_view->priv->model, iter);
2380 max_height = gtk_tree_view_insert_iter_height (tree_view,
2384 if (indices[depth - 1] == 0)
2386 tmpnode = _gtk_rbtree_find_count (tree, 1);
2387 _gtk_rbtree_insert_before (tree, tmpnode, max_height);
2391 tmpnode = _gtk_rbtree_find_count (tree, indices[depth - 1]);
2392 _gtk_rbtree_insert_after (tree, tmpnode, max_height);
2395 _gtk_tree_view_set_size (tree_view, -1, tree_view->priv->height + max_height);
2399 gtk_tree_view_child_toggled (GtkTreeModel *model,
2404 GtkTreeView *tree_view = (GtkTreeView *)data;
2405 GtkTreeIter real_iter;
2410 g_return_if_fail (path != NULL || iter != NULL);
2416 path = gtk_tree_model_get_path (model, iter);
2417 else if (iter == NULL)
2418 gtk_tree_model_get_iter (model, &real_iter, path);
2420 if (_gtk_tree_view_find_node (tree_view,
2424 /* We aren't actually showing the node */
2427 has_child = gtk_tree_model_iter_has_child (model, &real_iter);
2430 if (GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_PARENT) == has_child)
2434 GTK_RBNODE_SET_FLAG (node, GTK_RBNODE_IS_PARENT);
2436 GTK_RBNODE_UNSET_FLAG (node, GTK_RBNODE_IS_PARENT);
2438 if (has_child && GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_IS_LIST))
2440 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_IS_LIST);
2441 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_SHOW_EXPANDERS))
2444 for (list = tree_view->priv->columns; list; list = list->next)
2445 if (GTK_TREE_VIEW_COLUMN (list->data)->visible)
2447 GTK_TREE_VIEW_COLUMN (list->data)->dirty = TRUE;
2451 gtk_widget_queue_resize (GTK_WIDGET (tree_view));
2455 /* FIXME: Just redraw the node */
2456 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
2461 gtk_tree_view_deleted (GtkTreeModel *model,
2465 GtkTreeView *tree_view = (GtkTreeView *)data;
2470 g_return_if_fail (path != NULL);
2472 if (_gtk_tree_view_find_node (tree_view, path, &tree, &node))
2475 /* next, update the selection */
2476 if (tree_view->priv->anchor)
2479 gint depth = gtk_tree_path_get_depth (path);
2480 gint *indices = gtk_tree_path_get_indices (path);
2481 gint select_depth = gtk_tree_path_get_depth (tree_view->priv->anchor);
2482 gint *select_indices = gtk_tree_path_get_indices (tree_view->priv->anchor);
2484 if (gtk_tree_path_compare (path, tree_view->priv->anchor) == 0)
2486 if (GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_SELECTED) &&
2487 tree_view->priv->selection)
2488 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->selection),
2489 "selection_changed");
2493 for (i = 0; i < depth && i < select_depth; i++)
2495 if (indices[i] < select_indices[i])
2497 select_indices[i] = MAX (select_indices[i], 0);
2500 else if (indices[i] > select_indices[i])
2502 else if (i == depth - 1)
2504 select_indices[i] = MAX (select_indices[i], 0);
2511 for (list = tree_view->priv->columns; list; list = list->next)
2512 if (((GtkTreeViewColumn *)list->data)->visible &&
2513 ((GtkTreeViewColumn *)list->data)->column_type == GTK_TREE_VIEW_COLUMN_AUTOSIZE)
2514 ((GtkTreeViewColumn *)list->data)->dirty = TRUE;
2516 if (tree->root->count == 1)
2517 _gtk_rbtree_remove (tree);
2519 _gtk_rbtree_remove_node (tree, node);
2521 _gtk_tree_view_set_size (GTK_TREE_VIEW (data), -1, -1);
2524 /* Internal tree functions */
2526 gtk_tree_view_insert_iter_height (GtkTreeView *tree_view,
2531 GtkTreeViewColumn *column;
2532 GtkCellRenderer *cell;
2534 gint max_height = 0;
2539 /* do stuff with node */
2540 for (list = tree_view->priv->columns; list; list = list->next)
2542 gint height = 0, width = 0;
2543 column = list->data;
2545 if (!column->visible)
2548 if (column->column_type == GTK_TREE_VIEW_COLUMN_FIXED)
2554 cell = column->cell;
2555 gtk_tree_view_column_set_cell_data (column, tree_view->priv->model, iter);
2557 gtk_cell_renderer_get_size (cell, GTK_WIDGET (tree_view), &width, &height);
2558 max_height = MAX (max_height, TREE_VIEW_VERTICAL_SEPARATOR + height);
2560 if (i == tree_view->priv->expander_column &&
2561 TREE_VIEW_DRAW_EXPANDERS (tree_view))
2562 gtk_tree_view_column_set_width (column,
2563 MAX (column->width, depth * tree_view->priv->tab_offset + width));
2565 gtk_tree_view_column_set_width (column,
2566 MAX (column->width, width));
2574 gtk_tree_view_build_tree (GtkTreeView *tree_view,
2579 gboolean calc_bounds)
2581 GtkRBNode *temp = NULL;
2588 max_height = gtk_tree_view_insert_iter_height (tree_view,
2593 gtk_tree_model_ref_iter (tree_view->priv->model, iter);
2594 temp = _gtk_rbtree_insert_after (tree, temp, max_height);
2599 if (gtk_tree_model_iter_children (tree_view->priv->model, &child, iter))
2601 temp->children = _gtk_rbtree_new ();
2602 temp->children->parent_tree = tree;
2603 temp->children->parent_node = temp;
2604 gtk_tree_view_build_tree (tree_view, temp->children, &child, depth + 1, recurse, calc_bounds);
2607 if (gtk_tree_model_iter_has_child (tree_view->priv->model, iter))
2609 if ((temp->flags>K_RBNODE_IS_PARENT) != GTK_RBNODE_IS_PARENT)
2610 temp->flags ^= GTK_RBNODE_IS_PARENT;
2611 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_IS_LIST);
2614 while (gtk_tree_model_iter_next (tree_view->priv->model, iter));
2618 gtk_tree_view_calc_size (GtkTreeView *tree_view,
2625 GtkCellRenderer *cell;
2627 GtkTreeViewColumn *column;
2631 TREE_VIEW_INTERNAL_ASSERT_VOID (tree != NULL);
2634 while (temp->left != tree->nil)
2640 /* Do stuff with node */
2641 for (list = tree_view->priv->columns, i = 0; i < tree_view->priv->n_columns; list = list->next, i++)
2643 gint height = 0, width = 0;
2644 column = list->data;
2646 if (!column->visible)
2649 gtk_tree_view_column_set_cell_data (column, tree_view->priv->model, iter);
2650 cell = column->cell;
2651 gtk_cell_renderer_get_size (cell, GTK_WIDGET (tree_view), &width, &height);
2652 max_height = MAX (max_height, TREE_VIEW_VERTICAL_SEPARATOR + height);
2654 /* FIXME: I'm getting the width of all nodes here. )-: */
2655 if (column->dirty == FALSE || column->column_type == GTK_TREE_VIEW_COLUMN_FIXED)
2658 if (i == tree_view->priv->expander_column &&
2659 TREE_VIEW_DRAW_EXPANDERS (tree_view))
2660 gtk_tree_view_column_set_width (column,
2661 MAX (column->width, depth * tree_view->priv->tab_offset + width));
2663 gtk_tree_view_column_set_width (column, MAX (column->width, width));
2665 _gtk_rbtree_node_set_height (tree, temp, max_height);
2666 if (temp->children != NULL &&
2667 gtk_tree_model_iter_children (tree_view->priv->model, &child, iter))
2668 gtk_tree_view_calc_size (tree_view, temp->children, &child, depth + 1);
2669 temp = _gtk_rbtree_next (tree, temp);
2671 while (gtk_tree_model_iter_next (tree_view->priv->model, iter));
2675 gtk_tree_view_discover_dirty_iter (GtkTreeView *tree_view,
2680 GtkCellRenderer *cell;
2681 GtkTreeViewColumn *column;
2684 gint retval = FALSE;
2690 for (i = 0, list = tree_view->priv->columns; list; list = list->next, i++)
2693 column = list->data;
2694 if (column->dirty == TRUE || column->column_type == GTK_TREE_VIEW_COLUMN_FIXED)
2696 if (!column->visible)
2699 cell = column->cell;
2700 gtk_tree_view_column_set_cell_data (column, tree_view->priv->model, iter);
2704 gtk_cell_renderer_get_size (cell, GTK_WIDGET (tree_view), &width, &tmpheight);
2705 *height = MAX (*height, tmpheight);
2709 gtk_cell_renderer_get_size (cell, GTK_WIDGET (tree_view), &width, NULL);
2711 if (i == tree_view->priv->expander_column &&
2712 TREE_VIEW_DRAW_EXPANDERS (tree_view))
2714 if (depth * tree_view->priv->tab_offset + width > column->width)
2716 column->dirty = TRUE;
2722 if (width > column->width)
2724 column->dirty = TRUE;
2734 gtk_tree_view_discover_dirty (GtkTreeView *tree_view,
2739 GtkRBNode *temp = tree->root;
2740 GtkTreeViewColumn *column;
2743 gboolean is_all_dirty;
2745 TREE_VIEW_INTERNAL_ASSERT_VOID (tree != NULL);
2747 while (temp->left != tree->nil)
2752 is_all_dirty = TRUE;
2753 for (list = tree_view->priv->columns; list; list = list->next)
2755 column = list->data;
2756 if (column->dirty == FALSE)
2758 is_all_dirty = FALSE;
2766 gtk_tree_view_discover_dirty_iter (tree_view,
2770 if (gtk_tree_model_iter_children (tree_view->priv->model, &child, iter) &&
2771 temp->children != NULL)
2772 gtk_tree_view_discover_dirty (tree_view, temp->children, &child, depth + 1);
2773 temp = _gtk_rbtree_next (tree, temp);
2775 while (gtk_tree_model_iter_next (tree_view->priv->model, iter));
2780 gtk_tree_view_check_dirty (GtkTreeView *tree_view)
2783 gboolean dirty = FALSE;
2785 GtkTreeViewColumn *column;
2788 for (list = tree_view->priv->columns; list; list = list->next)
2790 column = list->data;
2794 if (column->column_type == GTK_TREE_VIEW_COLUMN_AUTOSIZE)
2796 gtk_tree_view_column_set_width (column, MAX (column->button->requisition.width, 1));
2803 path = gtk_tree_path_new_root ();
2804 if (gtk_tree_model_get_iter (tree_view->priv->model, &iter, path))
2806 gtk_tree_view_calc_size (tree_view, tree_view->priv->tree, &iter, 1);
2807 _gtk_tree_view_set_size (tree_view, -1, -1);
2810 gtk_tree_path_free (path);
2812 for (list = tree_view->priv->columns; list; list = list->next)
2814 column = list->data;
2815 column->dirty = FALSE;
2820 gtk_tree_view_create_button (GtkTreeView *tree_view,
2824 GtkTreeViewColumn *column;
2826 column = g_list_nth (tree_view->priv->columns, i)->data;
2827 gtk_widget_push_composite_child ();
2828 button = column->button = gtk_button_new ();
2829 gtk_widget_pop_composite_child ();
2831 gtk_widget_set_parent (button, GTK_WIDGET (tree_view));
2833 gtk_signal_connect (GTK_OBJECT (button), "clicked",
2834 (GtkSignalFunc) gtk_tree_view_button_clicked,
2835 (gpointer) tree_view);
2837 gtk_widget_show (button);
2841 gtk_tree_view_create_buttons (GtkTreeView *tree_view)
2843 GtkWidget *alignment;
2846 GtkTreeViewColumn *column;
2849 for (list = tree_view->priv->columns, i = 0; list; list = list->next, i++)
2851 column = list->data;
2853 if (column->button != NULL)
2856 gtk_tree_view_create_button (tree_view, i);
2857 switch (column->justification)
2859 case GTK_JUSTIFY_LEFT:
2860 alignment = gtk_alignment_new (0.0, 0.5, 0.0, 0.0);
2862 case GTK_JUSTIFY_RIGHT:
2863 alignment = gtk_alignment_new (1.0, 0.5, 0.0, 0.0);
2865 case GTK_JUSTIFY_CENTER:
2866 alignment = gtk_alignment_new (0.5, 0.5, 0.0, 0.0);
2868 case GTK_JUSTIFY_FILL:
2870 alignment = gtk_alignment_new (0.5, 0.5, 0.0, 0.0);
2875 label = column->child;
2878 label = gtk_label_new (column->title);
2879 gtk_widget_show (label);
2882 gtk_container_add (GTK_CONTAINER (alignment), label);
2883 gtk_container_add (GTK_CONTAINER (column->button), alignment);
2885 gtk_widget_show (alignment);
2888 gtk_tree_view_size_request_buttons (tree_view);
2890 if (GTK_WIDGET_REALIZED (tree_view))
2891 gtk_tree_view_realize_buttons (tree_view);
2893 if (GTK_WIDGET_MAPPED (tree_view))
2894 gtk_tree_view_map_buttons (tree_view);
2898 gtk_tree_view_button_clicked (GtkWidget *widget,
2902 GtkTreeView *tree_view;
2904 g_return_if_fail (widget != NULL);
2905 g_return_if_fail (GTK_IS_TREE_VIEW (data));
2907 tree_view = GTK_TREE_VIEW (data);
2909 /* find the column whose button was pressed */
2910 for (list = tree_view->priv->columns; list; list = list->next)
2911 if (GTK_TREE_VIEW_COLUMN (list->data)->button == widget)
2915 gtk_tree_view_column_clicked (GTK_TREE_VIEW_COLUMN (list->data));
2918 /* Make sure the node is visible vertically */
2920 gtk_tree_view_clamp_node_visible (GtkTreeView *tree_view,
2926 offset = _gtk_rbtree_node_find_offset (tree, node);
2928 /* we reverse the order, b/c in the unusual case of the
2929 * node's height being taller then the visible area, we'd rather
2930 * have the node flush to the top
2932 if (offset + GTK_RBNODE_GET_HEIGHT (node) >
2933 tree_view->priv->vadjustment->value + tree_view->priv->vadjustment->page_size)
2934 gtk_adjustment_set_value (GTK_ADJUSTMENT (tree_view->priv->vadjustment),
2935 offset + GTK_RBNODE_GET_HEIGHT (node) -
2936 tree_view->priv->vadjustment->page_size);
2937 if (offset < tree_view->priv->vadjustment->value)
2938 gtk_adjustment_set_value (GTK_ADJUSTMENT (tree_view->priv->vadjustment),
2942 /* This function could be more efficient.
2943 * I'll optimize it if profiling seems to imply that
2947 _gtk_tree_view_find_path (GtkTreeView *tree_view,
2952 GtkRBTree *tmp_tree;
2953 GtkRBNode *tmp_node, *last;
2956 path = gtk_tree_path_new ();
2958 g_return_val_if_fail (node != NULL, path);
2959 g_return_val_if_fail (node != tree->nil, path);
2961 count = 1 + node->left->count;
2964 tmp_node = node->parent;
2968 while (tmp_node != tmp_tree->nil)
2970 if (tmp_node->right == last)
2971 count += 1 + tmp_node->left->count;
2973 tmp_node = tmp_node->parent;
2975 gtk_tree_path_prepend_index (path, count - 1);
2976 last = tmp_tree->parent_node;
2977 tmp_tree = tmp_tree->parent_tree;
2980 count = 1 + last->left->count;
2981 tmp_node = last->parent;
2987 /* Returns whether or not it's a parent, or not */
2989 _gtk_tree_view_find_node (GtkTreeView *tree_view,
2994 GtkRBNode *tmpnode = NULL;
2995 GtkRBTree *tmptree = tree_view->priv->tree;
2996 gint *indices = gtk_tree_path_get_indices (path);
2997 gint depth = gtk_tree_path_get_depth (path);
3005 if (tmptree == NULL)
3011 tmpnode = _gtk_rbtree_find_count (tmptree, indices[i] + 1);
3018 tmptree = tmpnode->children;
3024 gtk_tree_view_get_arrow_range (GtkTreeView *tree_view,
3030 GtkTreeViewColumn *tmp_column = NULL;
3036 for (list = tree_view->priv->columns; list; list = list->next)
3038 tmp_column = list->data;
3040 if (i == tree_view->priv->expander_column)
3042 x_offset = total_width;
3046 if (tmp_column->visible)
3047 total_width += tmp_column->width;
3055 if (tmp_column && tmp_column->visible)
3057 /* +1 because x2 isn't included in the range. */
3059 *x2 = x_offset + tree_view->priv->tab_offset + 1;
3063 /* return an empty range, the expander column is hidden */
3070 gtk_tree_view_queue_draw_node (GtkTreeView *tree_view,
3073 GdkRectangle *clip_rect)
3078 rect.width = tree_view->priv->width;
3079 rect.y = _gtk_rbtree_node_find_offset (tree, node) +
3080 TREE_VIEW_VERTICAL_SEPARATOR/2 +
3081 TREE_VIEW_HEADER_HEIGHT (tree_view);
3082 rect.height = GTK_RBNODE_GET_HEIGHT (node) + TREE_VIEW_VERTICAL_SEPARATOR;
3085 GdkRectangle new_rect;
3086 gdk_rectangle_intersect (clip_rect, &rect, &new_rect);
3087 gtk_widget_queue_draw_area (GTK_WIDGET (tree_view),
3088 new_rect.x, new_rect.y,
3089 new_rect.width, new_rect.height);
3093 gtk_widget_queue_draw_area (GTK_WIDGET (tree_view),
3095 rect.width, rect.height);
3099 /* x and y are the mouse position
3102 gtk_tree_view_draw_arrow (GtkTreeView *tree_view,
3114 if (! GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_PARENT))
3117 widget = GTK_WIDGET (tree_view);
3119 y_offset = _gtk_rbtree_node_find_offset (tree, node) + TREE_VIEW_HEADER_HEIGHT (tree_view);
3121 gtk_tree_view_get_arrow_range (tree_view, &x_offset, NULL);
3124 area.y = y_offset + TREE_VIEW_VERTICAL_SEPARATOR;
3125 area.width = tree_view->priv->tab_offset - 2;
3126 area.height = GTK_RBNODE_GET_HEIGHT (node) - TREE_VIEW_VERTICAL_SEPARATOR;
3128 if (node == tree_view->priv->button_pressed_node)
3130 if (x >= area.x && x <= (area.x + area.width) &&
3131 y >= area.y && y <= (area.y + area.height))
3132 state = GTK_STATE_ACTIVE;
3134 state = GTK_STATE_NORMAL;
3138 state = (node==tree_view->priv->prelight_node&>K_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_ARROW_PRELIT)?GTK_STATE_PRELIGHT:GTK_STATE_NORMAL);
3141 /* FIXME expander size should come from a style property */
3142 #define EXPANDER_SIZE 8
3143 gtk_paint_expander (widget->style,
3144 tree_view->priv->bin_window,
3150 (area.y + (area.height - EXPANDER_SIZE) / 2 - (area.height + 1) % 2),
3151 node->children != NULL);
3152 #undef EXPANDER_SIZE
3156 _gtk_tree_view_set_size (GtkTreeView *tree_view,
3161 GtkTreeViewColumn *column;
3164 if (width == tree_view->priv->width &&
3165 height == tree_view->priv->height)
3168 if (tree_view->priv->model == NULL)
3170 tree_view->priv->width = width;
3171 tree_view->priv->height = height;
3172 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
3178 for (list = tree_view->priv->columns, i = 0; list; list = list->next, i++)
3180 column = list->data;
3181 if (!column->visible)
3183 width += TREE_VIEW_COLUMN_WIDTH (column);
3187 height = tree_view->priv->tree->root->offset + TREE_VIEW_VERTICAL_SEPARATOR;
3189 tree_view->priv->width = width;
3190 tree_view->priv->height = height;
3192 if (tree_view->priv->hadjustment->upper != tree_view->priv->width)
3194 tree_view->priv->hadjustment->upper = tree_view->priv->width;
3195 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->hadjustment), "changed");
3198 if (tree_view->priv->vadjustment->upper != tree_view->priv->height)
3200 tree_view->priv->vadjustment->upper = tree_view->priv->height;
3201 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->vadjustment), "changed");
3204 if (GTK_WIDGET_REALIZED (tree_view))
3206 gdk_window_resize (tree_view->priv->bin_window, MAX (width, GTK_WIDGET (tree_view)->allocation.width), height + TREE_VIEW_HEADER_HEIGHT (tree_view));
3207 gdk_window_resize (tree_view->priv->header_window, MAX (width, GTK_WIDGET (tree_view)->allocation.width), tree_view->priv->header_height);
3209 gtk_widget_queue_resize (GTK_WIDGET (tree_view));
3212 /* this function returns the new width of the column being resized given
3213 * the column and x position of the cursor; the x cursor position is passed
3214 * in as a pointer and automagicly corrected if it's beyond min/max limits
3217 gtk_tree_view_new_column_width (GtkTreeView *tree_view,
3221 GtkTreeViewColumn *column;
3224 /* first translate the x position from widget->window
3225 * to clist->clist_window
3228 column = g_list_nth (tree_view->priv->columns, i)->data;
3229 width = *x - column->button->allocation.x;
3231 /* Clamp down the value */
3232 if (column->min_width == -1)
3233 width = MAX (column->button->requisition.width,
3236 width = MAX (column->min_width,
3238 if (column->max_width != -1)
3239 width = MIN (width, column->max_width != -1);
3240 *x = column->button->allocation.x + width;
3247 gtk_tree_view_adjustment_changed (GtkAdjustment *adjustment,
3248 GtkTreeView *tree_view)
3250 if (GTK_WIDGET_REALIZED (tree_view))
3252 gdk_window_move (tree_view->priv->bin_window,
3253 - tree_view->priv->hadjustment->value,
3254 - tree_view->priv->vadjustment->value);
3255 gdk_window_move (tree_view->priv->header_window,
3256 - tree_view->priv->hadjustment->value,
3259 gdk_window_process_updates (tree_view->priv->bin_window, TRUE);
3260 gdk_window_process_updates (tree_view->priv->header_window, TRUE);
3270 * gtk_tree_view_new:
3272 * Creates a new #GtkTreeView widget.
3274 * Return value: A newly created #GtkTreeView widget.
3277 gtk_tree_view_new (void)
3279 GtkTreeView *tree_view;
3281 tree_view = GTK_TREE_VIEW (gtk_type_new (gtk_tree_view_get_type ()));
3283 return GTK_WIDGET (tree_view);
3287 * gtk_tree_view_new_with_model:
3288 * @model: the model.
3290 * Creates a new #GtkTreeView widget with the model initialized to @model.
3292 * Return value: A newly created #GtkTreeView widget.
3295 gtk_tree_view_new_with_model (GtkTreeModel *model)
3297 GtkTreeView *tree_view;
3299 tree_view = GTK_TREE_VIEW (gtk_type_new (gtk_tree_view_get_type ()));
3300 gtk_tree_view_set_model (tree_view, model);
3302 return GTK_WIDGET (tree_view);
3306 * gtk_tree_view_get_model:
3307 * @tree_view: a #GtkTreeView
3309 * Returns the model the the #GtkTreeView is based on. Returns NULL if the
3312 * Return value: A #GtkTreeModel, or NULL if none is currently being used.
3315 gtk_tree_view_get_model (GtkTreeView *tree_view)
3317 g_return_val_if_fail (tree_view != NULL, NULL);
3318 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), NULL);
3320 return tree_view->priv->model;
3324 gtk_tree_view_setup_model (GtkTreeView *tree_view)
3329 tree_view->priv->tree = _gtk_rbtree_new ();
3331 gtk_signal_connect (GTK_OBJECT (tree_view->priv->model),
3333 gtk_tree_view_changed,
3335 gtk_signal_connect (GTK_OBJECT (tree_view->priv->model),
3337 gtk_tree_view_inserted,
3339 gtk_signal_connect (GTK_OBJECT (tree_view->priv->model),
3341 gtk_tree_view_child_toggled,
3343 gtk_signal_connect (GTK_OBJECT (tree_view->priv->model),
3345 gtk_tree_view_deleted,
3348 if (tree_view->priv->columns == NULL)
3351 path = gtk_tree_path_new_root ();
3353 if (gtk_tree_model_get_iter (tree_view->priv->model, &iter, path))
3355 gtk_tree_view_build_tree (tree_view, tree_view->priv->tree, &iter, 1, FALSE, GTK_WIDGET_REALIZED (tree_view));
3358 gtk_tree_path_free (path);
3360 gtk_tree_view_create_buttons (tree_view);
3362 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_MODEL_SETUP);
3366 * gtk_tree_view_set_model:
3367 * @tree_view: A #GtkTreeNode.
3368 * @model: The model.
3370 * Sets the model for a #GtkTreeView. If the @tree_view already has a model
3371 * set, it will remove it before setting the new model. If @model is NULL, then
3372 * it will unset the old model.
3375 gtk_tree_view_set_model (GtkTreeView *tree_view,
3376 GtkTreeModel *model)
3378 g_return_if_fail (tree_view != NULL);
3379 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3381 if (tree_view->priv->model != NULL)
3384 /* No longer do this. */
3386 for (list = tree_view->priv->columns; list; list = list->next)
3388 column = list->data;
3391 gtk_widget_unparent (column->button);
3392 gdk_window_set_user_data (column->window, NULL);
3393 gdk_window_destroy (column->window);
3397 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_MODEL_SETUP))
3399 gtk_signal_disconnect_by_func (GTK_OBJECT (tree_view->priv->model),
3400 gtk_tree_view_changed,
3402 gtk_signal_disconnect_by_func (GTK_OBJECT (tree_view->priv->model),
3403 gtk_tree_view_inserted,
3405 gtk_signal_disconnect_by_func (GTK_OBJECT (tree_view->priv->model),
3406 gtk_tree_view_child_toggled,
3408 gtk_signal_disconnect_by_func (GTK_OBJECT (tree_view->priv->model),
3409 gtk_tree_view_deleted,
3411 _gtk_rbtree_free (tree_view->priv->tree);
3414 g_list_free (tree_view->priv->columns);
3415 tree_view->priv->columns = NULL;
3418 if (tree_view->priv->drag_dest_row)
3419 gtk_tree_path_free (tree_view->priv->drag_dest_row);
3421 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_MODEL_SETUP);
3424 tree_view->priv->model = model;
3427 tree_view->priv->tree = NULL;
3428 if (GTK_WIDGET_REALIZED (tree_view))
3429 _gtk_tree_view_set_size (tree_view, 0, 0);
3432 else if (GTK_WIDGET_REALIZED (tree_view))
3434 gtk_tree_view_setup_model (tree_view);
3435 _gtk_tree_view_set_size (tree_view, -1, -1);
3440 * gtk_tree_view_get_selection:
3441 * @tree_view: A #GtkTreeView.
3443 * Gets the #GtkTreeSelection associated with @tree_view.
3445 * Return value: A #GtkTreeSelection object.
3448 gtk_tree_view_get_selection (GtkTreeView *tree_view)
3450 g_return_val_if_fail (tree_view != NULL, NULL);
3451 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), NULL);
3453 if (tree_view->priv->selection == NULL)
3454 tree_view->priv->selection =
3455 _gtk_tree_selection_new_with_tree_view (tree_view);
3457 return tree_view->priv->selection;
3461 * gtk_tree_view_get_hadjustment:
3462 * @tree_view: A #GtkTreeView
3464 * Gets the #GtkAdjustment currently being used for the horizontal aspect.
3466 * Return value: A #GtkAdjustment object, or NULL if none is currently being
3470 gtk_tree_view_get_hadjustment (GtkTreeView *tree_view)
3472 g_return_val_if_fail (tree_view != NULL, NULL);
3473 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), NULL);
3475 if (tree_view->priv->hadjustment == NULL)
3476 gtk_tree_view_set_hadjustment (tree_view, NULL);
3478 return tree_view->priv->hadjustment;
3482 * gtk_tree_view_set_hadjustment:
3483 * @tree_view: A #GtkTreeView
3484 * @adjustment: The #GtkAdjustment to set, or NULL
3486 * Sets the #GtkAdjustment for the current horizontal aspect.
3489 gtk_tree_view_set_hadjustment (GtkTreeView *tree_view,
3490 GtkAdjustment *adjustment)
3492 g_return_if_fail (tree_view != NULL);
3493 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3495 gtk_tree_view_set_adjustments (tree_view,
3497 tree_view->priv->vadjustment);
3501 * gtk_tree_view_get_vadjustment:
3502 * @tree_view: A #GtkTreeView
3504 * Gets the #GtkAdjustment currently being used for the vertical aspect.
3506 * Return value: A #GtkAdjustment object, or NULL if none is currently being
3510 gtk_tree_view_get_vadjustment (GtkTreeView *tree_view)
3512 g_return_val_if_fail (tree_view != NULL, NULL);
3513 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), NULL);
3515 if (tree_view->priv->vadjustment == NULL)
3516 gtk_tree_view_set_vadjustment (tree_view, NULL);
3518 return tree_view->priv->vadjustment;
3522 * gtk_tree_view_set_vadjustment:
3523 * @tree_view: A #GtkTreeView
3524 * @adjustment: The #GtkAdjustment to set, or NULL
3526 * Sets the #GtkAdjustment for the current vertical aspect.
3529 gtk_tree_view_set_vadjustment (GtkTreeView *tree_view,
3530 GtkAdjustment *adjustment)
3532 g_return_if_fail (tree_view != NULL);
3533 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3535 gtk_tree_view_set_adjustments (tree_view,
3536 tree_view->priv->hadjustment,
3541 * gtk_tree_view_set_adjustments:
3542 * @tree_view: A #GtkTreeView
3543 * @hadj: The horizontal #GtkAdjustment to set, or NULL
3544 * @vadj: The vertical #GtkAdjustment to set, or NULL
3546 * Sets the horizonal and or vertical #GtkAdjustment.
3549 gtk_tree_view_set_adjustments (GtkTreeView *tree_view,
3550 GtkAdjustment *hadj,
3551 GtkAdjustment *vadj)
3553 gboolean need_adjust = FALSE;
3555 g_return_if_fail (tree_view != NULL);
3556 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3559 g_return_if_fail (GTK_IS_ADJUSTMENT (hadj));
3561 hadj = GTK_ADJUSTMENT (gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0));
3563 g_return_if_fail (GTK_IS_ADJUSTMENT (vadj));
3565 vadj = GTK_ADJUSTMENT (gtk_adjustment_new (0.0, 0.0, 0.0, 0.0, 0.0, 0.0));
3567 if (tree_view->priv->hadjustment && (tree_view->priv->hadjustment != hadj))
3569 gtk_signal_disconnect_by_data (GTK_OBJECT (tree_view->priv->hadjustment), tree_view);
3570 gtk_object_unref (GTK_OBJECT (tree_view->priv->hadjustment));
3573 if (tree_view->priv->vadjustment && (tree_view->priv->vadjustment != vadj))
3575 gtk_signal_disconnect_by_data (GTK_OBJECT (tree_view->priv->vadjustment), tree_view);
3576 gtk_object_unref (GTK_OBJECT (tree_view->priv->vadjustment));
3579 if (tree_view->priv->hadjustment != hadj)
3581 tree_view->priv->hadjustment = hadj;
3582 gtk_object_ref (GTK_OBJECT (tree_view->priv->hadjustment));
3583 gtk_object_sink (GTK_OBJECT (tree_view->priv->hadjustment));
3585 gtk_signal_connect (GTK_OBJECT (tree_view->priv->hadjustment), "value_changed",
3586 (GtkSignalFunc) gtk_tree_view_adjustment_changed,
3591 if (tree_view->priv->vadjustment != vadj)
3593 tree_view->priv->vadjustment = vadj;
3594 gtk_object_ref (GTK_OBJECT (tree_view->priv->vadjustment));
3595 gtk_object_sink (GTK_OBJECT (tree_view->priv->vadjustment));
3597 gtk_signal_connect (GTK_OBJECT (tree_view->priv->vadjustment), "value_changed",
3598 (GtkSignalFunc) gtk_tree_view_adjustment_changed,
3604 gtk_tree_view_adjustment_changed (NULL, tree_view);
3608 /* Column and header operations */
3611 * gtk_tree_view_get_headers_visible:
3612 * @tree_view: A #GtkTreeView.
3614 * Returns TRUE if the headers on the @tree_view are visible.
3616 * Return value: Whether the headers are visible or not.
3619 gtk_tree_view_get_headers_visible (GtkTreeView *tree_view)
3621 g_return_val_if_fail (tree_view != NULL, FALSE);
3622 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), FALSE);
3624 return GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE);
3628 * gtk_tree_view_set_headers_visible:
3629 * @tree_view: A #GtkTreeView.
3630 * @headers_visible: TRUE if the headers are visible
3632 * Sets the the visibility state of the headers.
3635 gtk_tree_view_set_headers_visible (GtkTreeView *tree_view,
3636 gboolean headers_visible)
3640 GtkTreeViewColumn *column;
3642 g_return_if_fail (tree_view != NULL);
3643 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3645 headers_visible = !! headers_visible;
3647 if (GTK_TREE_VIEW_FLAG_SET (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE) == headers_visible)
3650 if (headers_visible)
3651 GTK_TREE_VIEW_SET_FLAG (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE);
3653 GTK_TREE_VIEW_UNSET_FLAG (tree_view, GTK_TREE_VIEW_HEADERS_VISIBLE);
3655 if (GTK_WIDGET_REALIZED (tree_view))
3657 gdk_window_get_position (tree_view->priv->bin_window, &x, &y);
3658 if (headers_visible)
3660 gdk_window_move_resize (tree_view->priv->bin_window, x, y, tree_view->priv->width, tree_view->priv->height + TREE_VIEW_HEADER_HEIGHT (tree_view));
3662 if (GTK_WIDGET_MAPPED (tree_view))
3663 gtk_tree_view_map_buttons (tree_view);
3667 gdk_window_move_resize (tree_view->priv->bin_window, x, y, tree_view->priv->width, tree_view->priv->height);
3669 for (list = tree_view->priv->columns; list; list = list->next)
3671 column = list->data;
3672 gtk_widget_unmap (column->button);
3674 gdk_window_hide (tree_view->priv->header_window);
3678 tree_view->priv->vadjustment->page_size = GTK_WIDGET (tree_view)->allocation.height - TREE_VIEW_HEADER_HEIGHT (tree_view);
3679 tree_view->priv->vadjustment->page_increment = (GTK_WIDGET (tree_view)->allocation.height - TREE_VIEW_HEADER_HEIGHT (tree_view)) / 2;
3680 tree_view->priv->vadjustment->lower = 0;
3681 tree_view->priv->vadjustment->upper = tree_view->priv->height;
3682 gtk_signal_emit_by_name (GTK_OBJECT (tree_view->priv->vadjustment), "changed");
3684 gtk_widget_queue_resize (GTK_WIDGET (tree_view));
3689 * gtk_tree_view_columns_autosize:
3690 * @tree_view: A #GtkTreeView.
3692 * Resizes all columns to their optimal width.
3695 gtk_tree_view_columns_autosize (GtkTreeView *tree_view)
3697 gboolean dirty = FALSE;
3699 GtkTreeViewColumn *column;
3701 g_return_if_fail (tree_view != NULL);
3702 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3704 for (list = tree_view->priv->columns; list; list = list->next)
3706 column = list->data;
3707 if (column->column_type == GTK_TREE_VIEW_COLUMN_AUTOSIZE)
3709 column->dirty = TRUE;
3714 gtk_widget_queue_resize (GTK_WIDGET (tree_view));
3718 * gtk_tree_view_set_headers_clickable:
3719 * @tree_view: A #GtkTreeView.
3720 * @setting: TRUE if the columns are clickable.
3722 * Allow the column title buttons to be clicked.
3725 gtk_tree_view_set_headers_clickable (GtkTreeView *tree_view,
3730 g_return_if_fail (tree_view != NULL);
3731 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3732 g_return_if_fail (tree_view->priv->model != NULL);
3734 for (list = tree_view->priv->columns; list; list = list->next)
3735 gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (list->data), setting);
3739 * gtk_tree_view_append_column:
3740 * @tree_view: A #GtkTreeView.
3741 * @column: The #GtkTreeViewColumn to add.
3743 * Appends @column to the list of columns.
3745 * Return value: The number of columns in @tree_view after appending.
3748 gtk_tree_view_append_column (GtkTreeView *tree_view,
3749 GtkTreeViewColumn *column)
3751 g_return_val_if_fail (tree_view != NULL, -1);
3752 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), -1);
3753 g_return_val_if_fail (column != NULL, -1);
3754 g_return_val_if_fail (GTK_IS_TREE_VIEW_COLUMN (column), -1);
3755 g_return_val_if_fail (column->tree_view == NULL, -1);
3757 g_object_ref (G_OBJECT (column));
3758 gtk_object_sink (GTK_OBJECT (column));
3759 tree_view->priv->columns = g_list_append (tree_view->priv->columns, column);
3760 column->tree_view = GTK_WIDGET (tree_view);
3762 tree_view->priv->n_columns++;
3764 /* FIXME create header for the new column! */
3766 return tree_view->priv->n_columns;
3771 * gtk_tree_view_remove_column:
3772 * @tree_view: A #GtkTreeView.
3773 * @column: The #GtkTreeViewColumn to remove.
3775 * Removes @column from @tree_view.
3777 * Return value: The number of columns in @tree_view after removing.
3780 gtk_tree_view_remove_column (GtkTreeView *tree_view,
3781 GtkTreeViewColumn *column)
3783 g_return_val_if_fail (tree_view != NULL, -1);
3784 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), -1);
3785 g_return_val_if_fail (column != NULL, -1);
3786 g_return_val_if_fail (GTK_IS_TREE_VIEW_COLUMN (column), -1);
3787 g_return_val_if_fail (column->tree_view == GTK_WIDGET (tree_view), -1);
3789 tree_view->priv->columns = g_list_remove (tree_view->priv->columns,
3791 column->tree_view = NULL;
3792 g_object_unref (G_OBJECT (column));
3794 tree_view->priv->n_columns--;
3796 /* FIXME destroy header for the column! */
3798 return tree_view->priv->n_columns;
3802 * gtk_tree_view_insert_column:
3803 * @tree_view: A #GtkTreeView.
3804 * @column: The #GtkTreeViewColumn to be inserted.
3805 * @position: The position to insert @column in.
3807 * This inserts the @column into the @tree_view at @position.
3809 * Return value: The number of columns in @tree_view after insertion.
3812 gtk_tree_view_insert_column (GtkTreeView *tree_view,
3813 GtkTreeViewColumn *column,
3816 g_return_val_if_fail (tree_view != NULL, -1);
3817 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), -1);
3818 g_return_val_if_fail (column != NULL, -1);
3819 g_return_val_if_fail (GTK_IS_TREE_VIEW_COLUMN (column), -1);
3820 g_return_val_if_fail (column->tree_view == NULL, -1);
3822 g_object_ref (G_OBJECT (column));
3823 gtk_object_sink (GTK_OBJECT (column));
3824 tree_view->priv->columns = g_list_insert (tree_view->priv->columns,
3826 column->tree_view = GTK_WIDGET (tree_view);
3828 tree_view->priv->n_columns++;
3830 /* FIXME create header for the column! */
3832 return tree_view->priv->n_columns;
3836 * gtk_tree_view_get_column:
3837 * @tree_view: A #GtkTreeView.
3838 * @n: The position of the column, counting from 0.
3840 * Gets the #GtkTreeViewColumn at the given position in the #tree_view.
3842 * Return value: The #GtkTreeViewColumn, or NULL if the position is outside the
3846 gtk_tree_view_get_column (GtkTreeView *tree_view,
3849 g_return_val_if_fail (tree_view != NULL, NULL);
3850 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), NULL);
3851 g_return_val_if_fail (tree_view->priv->model != NULL, NULL);
3853 if (n < 0 || n >= tree_view->priv->n_columns)
3856 if (tree_view->priv->columns == NULL)
3859 return GTK_TREE_VIEW_COLUMN (g_list_nth (tree_view->priv->columns, n)->data);
3863 gtk_tree_view_set_expander_column (GtkTreeView *tree_view,
3866 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3868 if (tree_view->priv->expander_column != col)
3870 tree_view->priv->expander_column = col;
3872 /* g_object_notify (G_OBJECT (tree_view), "expander_column"); */
3877 gtk_tree_view_get_expander_column (GtkTreeView *tree_view)
3879 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), -1);
3881 return tree_view->priv->expander_column;
3885 * gtk_tree_view_scroll_to_point:
3886 * @tree_view: a #GtkTreeView
3887 * @tree_x: X coordinate of new top-left pixel of visible area
3888 * @tree_y: Y coordinate of new top-left pixel of visible area
3890 * Scrolls the tree view such that the top-left corner of the visible
3891 * area is @tree_x, @tree_y, where @tree_x and @tree_y are specified
3892 * in tree window coordinates. The @tree_view must be realized before
3893 * this function is called. If it isn't, you probably want ot be
3894 * using gtk_tree_view_scroll_to_cell.
3897 gtk_tree_view_scroll_to_point (GtkTreeView *tree_view,
3901 GtkAdjustment *hadj;
3902 GtkAdjustment *vadj;
3904 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3905 g_return_if_fail (GTK_WIDGET_REALIZED (tree_view));
3907 hadj = tree_view->priv->hadjustment;
3908 vadj = tree_view->priv->vadjustment;
3910 gtk_adjustment_set_value (hadj, CLAMP (tree_x, hadj->lower, hadj->upper));
3911 gtk_adjustment_set_value (vadj, CLAMP (tree_y, vadj->lower, vadj->upper));
3915 * gtk_tree_view_scroll_to_cell
3916 * @tree_view: A #GtkTreeView.
3917 * @path: The path of the row to move to.
3918 * @column: The #GtkTreeViewColumn to move horizontally to.
3919 * @row_align: The vertical alignment of the row specified by @path.
3920 * @col_align: The horizontal alignment of the column specified by @column.
3922 * Moves the alignments of @tree_view to the position specified by
3923 * @column and @path. If @column is NULL, then no horizontal
3924 * scrolling occurs. Likewise, if @path is NULL no vertical scrolling
3925 * occurs. @row_align determines where the row is placed, and
3926 * @col_align determines where @column is placed. Both are expected
3927 * to be between 0.0 and 1.0. 0.0 means left/top alignment, 1.0 means
3928 * right/bottom alignment, 0.5 means center.
3931 gtk_tree_view_scroll_to_cell (GtkTreeView *tree_view,
3933 GtkTreeViewColumn *column,
3937 GdkRectangle cell_rect;
3938 GdkRectangle vis_rect;
3939 gint dest_x, dest_y;
3941 /* FIXME work on unmapped/unrealized trees? maybe implement when
3942 * we do incremental reflow for trees
3945 g_return_if_fail (tree_view != NULL);
3946 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
3947 g_return_if_fail (row_align >= 0.0);
3948 g_return_if_fail (row_align <= 1.0);
3949 g_return_if_fail (col_align >= 0.0);
3950 g_return_if_fail (col_align <= 1.0);
3951 g_return_if_fail (path != NULL || column != NULL);
3953 row_align = CLAMP (row_align, 0.0, 1.0);
3954 col_align = CLAMP (col_align, 0.0, 1.0);
3956 if (! GTK_WIDGET_REALIZED (tree_view))
3959 tree_view->priv->scroll_to_path = gtk_tree_path_copy (path);
3961 tree_view->priv->scroll_to_column = column;
3962 tree_view->priv->scroll_to_row_align = row_align;
3963 tree_view->priv->scroll_to_col_align = col_align;
3968 gtk_tree_view_get_cell_rect (tree_view, path, column, &cell_rect);
3969 gtk_tree_view_get_visible_rect (tree_view, &vis_rect);
3971 dest_x = vis_rect.x;
3972 dest_y = vis_rect.y;
3976 dest_x = cell_rect.x +
3977 cell_rect.width * row_align -
3978 vis_rect.width * row_align;
3983 dest_y = cell_rect.y +
3984 cell_rect.height * col_align -
3985 vis_rect.height * col_align;
3988 gtk_tree_view_scroll_to_point (tree_view, dest_x, dest_y);
3992 * gtk_tree_view_get_path_at_pos:
3993 * @tree_view: A #GtkTreeView.
3994 * @window: The #GdkWindow to check against.
3995 * @x: The x position to be identified.
3996 * @y: The y position to be identified.
3997 * @path: A pointer to a #GtkTreePath pointer to be filled in, or %NULL
3998 * @column: A pointer to a #GtkTreeViewColumn pointer to be filled in, or %NULL
3999 * @cell_x: A pointer where the X coordinate relative to the cell can be placed, or %NULL
4000 * @cell_y: A pointer where the Y coordinate relative to the cell can be placed, or %NULL
4002 * Finds the path at the point (@x, @y) relative to @window. If @window is
4003 * NULL, then the point is found relative to the widget coordinates. This
4004 * function is expected to be called after an event, with event->window being
4005 * passed in as @window. It is primarily for things like popup menus. If @path
4006 * is non-NULL, then it will be filled with the #GtkTreePath at that point.
4007 * This path should be freed with #gtk_tree_path_free. If @column is non-NULL,
4008 * then it will be filled with the column at that point. @cell_x and @cell_y return
4009 * the coordinates relative to the cell.
4011 * Return value: TRUE if a row exists at that coordinate.
4014 gtk_tree_view_get_path_at_pos (GtkTreeView *tree_view,
4019 GtkTreeViewColumn **column,
4027 g_return_val_if_fail (tree_view != NULL, FALSE);
4028 g_return_val_if_fail (tree_view->priv->tree != NULL, FALSE);
4029 g_return_val_if_fail (tree_view->priv->bin_window != NULL, FALSE);
4032 g_return_val_if_fail (window == tree_view->priv->bin_window, FALSE);
4039 if (x > tree_view->priv->hadjustment->upper)
4045 if (column || cell_x)
4047 GtkTreeViewColumn *tmp_column;
4048 GtkTreeViewColumn *last_column = NULL;
4050 gint remaining_x = x;
4051 gboolean found = FALSE;
4053 for (list = tree_view->priv->columns; list; list = list->next)
4055 tmp_column = list->data;
4057 if (tmp_column->visible == FALSE)
4060 last_column = tmp_column;
4061 if (remaining_x <= tmp_column->width)
4066 *column = tmp_column;
4069 *cell_x = remaining_x;
4073 remaining_x -= tmp_column->width;
4079 *column = last_column;
4082 *cell_x = last_column->width + remaining_x;
4088 y_offset = _gtk_rbtree_find_offset (tree_view->priv->tree,
4089 y - TREE_VIEW_HEADER_HEIGHT (tree_view),
4094 if (y < TREE_VIEW_HEADER_HEIGHT (tree_view))
4097 y_offset = _gtk_rbtree_find_offset (tree_view->priv->tree, y - TREE_VIEW_HEADER_HEIGHT (tree_view) +
4098 tree_view->priv->vadjustment->value,
4109 *path = _gtk_tree_view_find_path (tree_view, tree, node);
4115 * gtk_tree_view_get_cell_rect:
4116 * @tree_view: a #GtkTreeView
4117 * @path: a #GtkTreePath for the row, or %NULL to get only horizontal coordinates
4118 * @column: a #GtkTreeViewColumn for the column, or %NULL to get only vertical coordiantes
4119 * @rect: rectangle to fill with cell rect
4121 * Fills the bounding rectangle in tree window coordinates for the cell
4122 * at the row specified by @path and the column specified by @column.
4123 * If @path is %NULL, the y and height fields of the rectangle will be filled
4124 * with 0. If @column is %NULL, the x and width fields will be filled with 0.
4128 gtk_tree_view_get_cell_rect (GtkTreeView *tree_view,
4130 GtkTreeViewColumn *column,
4133 GtkTreeViewColumn *tmp_column = NULL;
4136 GtkRBTree *tree = NULL;
4137 GtkRBNode *node = NULL;
4139 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4140 g_return_if_fail (column == NULL || GTK_IS_TREE_VIEW_COLUMN (column));
4141 g_return_if_fail (rect != NULL);
4150 /* Get vertical coords */
4152 _gtk_tree_view_find_node (tree_view, path, &tree, &node);
4156 g_warning (G_STRLOC": no row corresponding to path");
4160 rect->y = _gtk_rbtree_node_find_offset (tree, node) + TREE_VIEW_HEADER_HEIGHT (tree_view);
4162 rect->height = GTK_RBNODE_GET_HEIGHT (node);
4165 if (column && column->visible)
4168 for (list = tree_view->priv->columns; list; list = list->next)
4170 tmp_column = list->data;
4172 if (tmp_column == column)
4174 rect->x = total_width;
4178 if (tmp_column->visible)
4179 total_width += tmp_column->width;
4182 if (tmp_column != column)
4184 g_warning (G_STRLOC": passed-in column isn't in the tree");
4188 rect->width = column->width;
4193 gtk_tree_view_expand_all_helper (GtkRBTree *tree,
4197 GtkTreeView *tree_view = data;
4200 _gtk_rbtree_traverse (node->children,
4201 node->children->root,
4203 gtk_tree_view_expand_all_helper,
4205 else if ((node->flags & GTK_RBNODE_IS_PARENT) == GTK_RBNODE_IS_PARENT && node->children == NULL)
4211 node->children = _gtk_rbtree_new ();
4212 node->children->parent_tree = tree;
4213 node->children->parent_node = node;
4214 path = _gtk_tree_view_find_path (tree_view, tree, node);
4215 gtk_tree_model_get_iter (tree_view->priv->model, &iter, path);
4216 gtk_tree_model_iter_children (tree_view->priv->model, &child, &iter);
4217 gtk_tree_view_build_tree (tree_view,
4220 gtk_tree_path_get_depth (path) + 1,
4222 GTK_WIDGET_REALIZED (tree_view));
4223 gtk_tree_path_free (path);
4228 * gtk_tree_view_expand_all:
4229 * @tree_view: A #GtkTreeView.
4231 * Recursively expands all nodes in the @tree_view.
4234 gtk_tree_view_expand_all (GtkTreeView *tree_view)
4236 g_return_if_fail (tree_view != NULL);
4237 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4238 g_return_if_fail (tree_view->priv->tree != NULL);
4240 _gtk_rbtree_traverse (tree_view->priv->tree,
4241 tree_view->priv->tree->root,
4243 gtk_tree_view_expand_all_helper,
4246 _gtk_tree_view_set_size (tree_view, -1,-1);
4250 gtk_tree_view_collapse_all_helper (GtkRBTree *tree,
4259 path = _gtk_tree_view_find_path (GTK_TREE_VIEW (data),
4261 node->children->root);
4262 gtk_tree_model_get_iter (GTK_TREE_VIEW (data)->priv->model,
4265 gtk_tree_view_discover_dirty (GTK_TREE_VIEW (data),
4268 gtk_tree_path_get_depth (path));
4269 _gtk_rbtree_remove (node->children);
4270 gtk_tree_path_free (path);
4275 * gtk_tree_view_collapse_all:
4276 * @tree_view: A #GtkTreeView.
4278 * Recursively collapses all visible, expanded nodes in @tree_view.
4281 gtk_tree_view_collapse_all (GtkTreeView *tree_view)
4283 g_return_if_fail (tree_view != NULL);
4284 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4285 g_return_if_fail (tree_view->priv->tree != NULL);
4287 _gtk_rbtree_traverse (tree_view->priv->tree,
4288 tree_view->priv->tree->root,
4290 gtk_tree_view_collapse_all_helper,
4293 if (GTK_WIDGET_MAPPED (tree_view))
4294 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
4297 /* FIXME the bool return values for expand_row and collapse_row are
4298 * not analagous; they should be TRUE if the row had children and
4299 * was not already in the requested state.
4303 * gtk_tree_view_expand_row:
4304 * @tree_view: a #GtkTreeView
4305 * @path: path to a row
4306 * @open_all: whether to recursively expand, or just expand immediate children
4308 * Opens the row so its children are visible
4310 * Return value: %TRUE if the row existed and had children
4313 gtk_tree_view_expand_row (GtkTreeView *tree_view,
4322 g_return_val_if_fail (tree_view != NULL, FALSE);
4323 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), FALSE);
4324 g_return_val_if_fail (tree_view->priv->model != NULL, FALSE);
4325 g_return_val_if_fail (path != NULL, FALSE);
4327 if (_gtk_tree_view_find_node (tree_view,
4336 gtk_tree_model_get_iter (tree_view->priv->model, &iter, path);
4337 if (! gtk_tree_model_iter_has_child (tree_view->priv->model, &iter))
4340 node->children = _gtk_rbtree_new ();
4341 node->children->parent_tree = tree;
4342 node->children->parent_node = node;
4344 gtk_tree_model_iter_children (tree_view->priv->model, &child, &iter);
4345 gtk_tree_view_build_tree (tree_view,
4348 gtk_tree_path_get_depth (path) + 1,
4350 GTK_WIDGET_REALIZED (tree_view));
4352 if (GTK_WIDGET_MAPPED (tree_view))
4353 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
4359 * gtk_tree_view_collapse_row:
4360 * @tree_view: a #GtkTreeView
4361 * @path: path to a row in the @tree_view
4363 * Collapses a row (hides its child rows).
4365 * Return value: %TRUE if the row was expanded
4368 gtk_tree_view_collapse_row (GtkTreeView *tree_view,
4375 g_return_val_if_fail (tree_view != NULL, FALSE);
4376 g_return_val_if_fail (GTK_IS_TREE_VIEW (tree_view), FALSE);
4377 g_return_val_if_fail (tree_view->priv->tree != NULL, FALSE);
4378 g_return_val_if_fail (path != NULL, FALSE);
4380 if (_gtk_tree_view_find_node (tree_view,
4386 if (node->children == NULL)
4389 gtk_tree_model_get_iter (tree_view->priv->model, &iter, path);
4390 gtk_tree_view_discover_dirty (tree_view,
4393 gtk_tree_path_get_depth (path));
4394 _gtk_rbtree_remove (node->children);
4396 if (GTK_WIDGET_MAPPED (tree_view))
4397 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
4403 * gtk_tree_view_get_visible_rect:
4404 * @tree_view: a #GtkTreeView
4405 * @visible_rect: rectangle to fill
4407 * Fills @visible_rect with the currently-visible region of the
4408 * buffer, in tree coordinates. Convert to widget coordinates with
4409 * gtk_tree_view_tree_to_widget_coords(). Tree coordinates start at
4410 * 0,0 for row 0 of the tree, and cover the entire scrollable area of
4414 gtk_tree_view_get_visible_rect (GtkTreeView *tree_view,
4415 GdkRectangle *visible_rect)
4419 g_return_if_fail (tree_view != NULL);
4420 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4422 widget = GTK_WIDGET (tree_view);
4426 visible_rect->x = tree_view->priv->hadjustment->value;
4427 visible_rect->y = tree_view->priv->vadjustment->value;
4428 visible_rect->width = widget->allocation.width;
4429 visible_rect->height = widget->allocation.height - TREE_VIEW_HEADER_HEIGHT (tree_view);
4434 * gtk_tree_view_widget_to_tree_coords:
4435 * @tree_view: a #GtkTreeView
4436 * @wx: widget X coordinate
4437 * @wy: widget Y coordinate
4438 * @tx: return location for tree X coordinate
4439 * @ty: return location for tree Y coordinate
4441 * Converts widget coordinates to coordinates for the
4442 * tree window (the full scrollable area of the tree).
4446 gtk_tree_view_widget_to_tree_coords (GtkTreeView *tree_view,
4452 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4456 *tx = wx + tree_view->priv->hadjustment->value;
4461 *ty = wy - TREE_VIEW_HEADER_HEIGHT (tree_view) +
4462 tree_view->priv->vadjustment->value;
4467 * gtk_tree_view_tree_to_widget_coords:
4468 * @tree_view: a #GtkTreeView
4469 * @tx: tree X coordinate
4470 * @ty: tree Y coordinate
4471 * @wx: return location for widget X coordinate
4472 * @wy: return location for widget Y coordinate
4474 * Converts tree coordinates (coordinates in full scrollable
4475 * area of the tree) to widget coordinates.
4479 gtk_tree_view_tree_to_widget_coords (GtkTreeView *tree_view,
4485 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4489 *wx = tx - tree_view->priv->hadjustment->value;
4494 *wy = ty + TREE_VIEW_HEADER_HEIGHT (tree_view) -
4495 tree_view->priv->vadjustment->value;
4501 typedef struct _TreeViewDragInfo TreeViewDragInfo;
4503 struct _TreeViewDragInfo
4505 GdkModifierType start_button_mask;
4506 GtkTargetList *source_target_list;
4507 GdkDragAction source_actions;
4508 GClosure *row_draggable_closure;
4509 GtkTreePath *source_row;
4511 GtkTargetList *dest_target_list;
4512 GClosure *location_droppable_closure;
4514 guint source_set : 1;
4518 static TreeViewDragInfo*
4519 get_info (GtkTreeView *tree_view)
4521 return g_object_get_data (G_OBJECT (tree_view), "gtk-tree-view-drag-info");
4525 clear_source_info (TreeViewDragInfo *di)
4527 if (di->source_target_list)
4528 gtk_target_list_unref (di->source_target_list);
4530 if (di->row_draggable_closure)
4531 g_closure_unref (di->row_draggable_closure);
4534 gtk_tree_path_free (di->source_row);
4536 di->source_target_list = NULL;
4537 di->row_draggable_closure = NULL;
4538 di->source_row = NULL;
4542 clear_dest_info (TreeViewDragInfo *di)
4544 if (di->location_droppable_closure)
4545 g_closure_unref (di->location_droppable_closure);
4547 if (di->dest_target_list)
4548 gtk_target_list_unref (di->dest_target_list);
4550 di->location_droppable_closure = NULL;
4551 di->dest_target_list = NULL;
4555 destroy_info (TreeViewDragInfo *di)
4557 clear_source_info (di);
4558 clear_dest_info (di);
4562 static TreeViewDragInfo*
4563 ensure_info (GtkTreeView *tree_view)
4565 TreeViewDragInfo *di;
4567 di = get_info (tree_view);
4571 di = g_new0 (TreeViewDragInfo, 1);
4573 g_object_set_data_full (G_OBJECT (tree_view),
4574 "gtk-tree-view-drag-info",
4576 (GDestroyNotify) destroy_info);
4583 remove_info (GtkTreeView *tree_view)
4585 g_object_set_data (G_OBJECT (tree_view), "gtk-tree-view-drag-info", NULL);
4588 #define SCROLL_EDGE_SIZE 15
4591 drag_scan_timeout (gpointer data)
4593 GtkTreeView *tree_view;
4595 GdkModifierType state;
4596 GtkTreePath *path = NULL;
4597 GtkTreeViewColumn *column = NULL;
4598 GdkRectangle visible_rect;
4600 tree_view = GTK_TREE_VIEW (data);
4602 gdk_window_get_pointer (tree_view->priv->bin_window,
4605 gtk_tree_view_get_visible_rect (tree_view, &visible_rect);
4607 /* See if we are near the edge. */
4608 if ((x - visible_rect.x) < SCROLL_EDGE_SIZE ||
4609 (visible_rect.x + visible_rect.width - x) < SCROLL_EDGE_SIZE ||
4610 (y - visible_rect.y) < SCROLL_EDGE_SIZE ||
4611 (visible_rect.y + visible_rect.height - y) < SCROLL_EDGE_SIZE)
4613 gtk_tree_view_get_path_at_pos (tree_view,
4614 tree_view->priv->bin_window,
4623 gtk_tree_view_scroll_to_cell (tree_view,
4628 gtk_tree_path_free (path);
4637 ensure_scroll_timeout (GtkTreeView *tree_view)
4639 if (tree_view->priv->scroll_timeout == 0)
4640 tree_view->priv->scroll_timeout = gtk_timeout_add (50, drag_scan_timeout, tree_view);
4644 remove_scroll_timeout (GtkTreeView *tree_view)
4646 if (tree_view->priv->scroll_timeout != 0)
4648 gtk_timeout_remove (tree_view->priv->scroll_timeout);
4649 tree_view->priv->scroll_timeout = 0;
4654 #warning "implement g_closure_sink"
4656 #define g_closure_sink(c)
4659 gtk_tree_view_set_rows_drag_source (GtkTreeView *tree_view,
4660 GdkModifierType start_button_mask,
4661 const GtkTargetEntry *targets,
4663 GdkDragAction actions,
4664 GtkTreeViewDraggableFunc row_draggable_func,
4667 TreeViewDragInfo *di;
4669 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4671 di = ensure_info (tree_view);
4672 clear_source_info (di);
4674 di->start_button_mask = start_button_mask;
4675 di->source_target_list = gtk_target_list_new (targets, n_targets);
4676 di->source_actions = actions;
4678 if (row_draggable_func)
4680 di->row_draggable_closure = g_cclosure_new ((GCallback) row_draggable_func,
4682 g_closure_ref (di->row_draggable_closure);
4683 g_closure_sink (di->row_draggable_closure);
4686 di->source_set = TRUE;
4690 gtk_tree_view_set_rows_drag_dest (GtkTreeView *tree_view,
4691 const GtkTargetEntry *targets,
4693 GdkDragAction actions,
4694 GtkTreeViewDroppableFunc location_droppable_func,
4697 TreeViewDragInfo *di;
4699 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4701 gtk_drag_dest_set (GTK_WIDGET (tree_view),
4707 di = ensure_info (tree_view);
4708 clear_dest_info (di);
4711 di->dest_target_list = gtk_target_list_new (targets, n_targets);
4713 if (location_droppable_func)
4715 di->location_droppable_closure = g_cclosure_new ((GCallback) location_droppable_func,
4717 g_closure_ref (di->location_droppable_closure);
4718 g_closure_sink (di->location_droppable_closure);
4721 di->dest_set = TRUE;
4725 gtk_tree_view_unset_rows_drag_source (GtkTreeView *tree_view)
4727 TreeViewDragInfo *di;
4729 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4731 di = get_info (tree_view);
4737 clear_source_info (di);
4738 di->source_set = FALSE;
4741 if (!di->dest_set && !di->source_set)
4742 remove_info (tree_view);
4747 gtk_tree_view_unset_rows_drag_dest (GtkTreeView *tree_view)
4749 TreeViewDragInfo *di;
4751 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4753 di = get_info (tree_view);
4759 gtk_drag_dest_unset (GTK_WIDGET (tree_view));
4760 clear_dest_info (di);
4761 di->dest_set = FALSE;
4764 if (!di->dest_set && !di->source_set)
4765 remove_info (tree_view);
4771 gtk_tree_view_set_drag_dest_row (GtkTreeView *tree_view,
4773 GtkTreeViewDropPosition pos)
4775 /* Note; this function is exported to allow a custom DND
4776 * implementation, so it can't touch TreeViewDragInfo
4779 g_return_if_fail (GTK_IS_TREE_VIEW (tree_view));
4781 if (tree_view->priv->drag_dest_row)
4783 /* FIXME queue undraw on previous location */
4784 gtk_tree_path_free (tree_view->priv->drag_dest_row);
4787 tree_view->priv->drag_dest_pos = pos;
4790 tree_view->priv->drag_dest_row = gtk_tree_path_copy (path);
4792 tree_view->priv->drag_dest_row = NULL;
4794 /* FIXME this is crap, queue draw only on the newly-highlighted row */
4795 gtk_widget_queue_draw (GTK_WIDGET (tree_view));
4799 gtk_tree_view_get_dest_row_at_pos (GtkTreeView *tree_view,
4803 GtkTreeViewDropPosition *pos)
4806 gdouble offset_into_row;
4810 GtkTreeViewColumn *column = NULL;
4811 GtkTreePath *tmp_path = NULL;
4813 /* Note; this function is exported to allow a custom DND
4814 * implementation, so it can't touch TreeViewDragInfo
4817 g_return_val_if_fail (tree_view != NULL, FALSE);
4818 g_return_val_if_fail (tree_view->priv->tree != NULL, FALSE);
4819 g_return_val_if_fail (drag_x >= 0, FALSE);
4820 g_return_val_if_fail (drag_y >= 0, FALSE);
4821 g_return_val_if_fail (tree_view->priv->bin_window != NULL, FALSE);
4826 /* remember that drag_x and drag_y are in widget coords, convert to rbtree */
4828 gtk_tree_view_widget_to_tree_coords (tree_view, drag_x, drag_y,
4831 /* If in the top quarter of a row, we drop before that row; if
4832 * in the bottom quarter, drop after that row; if in the middle,
4833 * and the row has children, drop into the row.
4836 if (!gtk_tree_view_get_path_at_pos (tree_view,
4837 tree_view->priv->bin_window,
4845 gtk_tree_view_get_cell_rect (tree_view, tmp_path, column,
4848 offset_into_row = cell_y;
4853 gtk_tree_path_free (tmp_path);
4857 quarter = cell.height / 4.0;
4861 if (offset_into_row < quarter)
4863 *pos = GTK_TREE_VIEW_DROP_BEFORE;
4865 else if (offset_into_row < quarter * 2)
4867 *pos = GTK_TREE_VIEW_DROP_INTO_OR_BEFORE;
4869 else if (offset_into_row < quarter * 3)
4871 *pos = GTK_TREE_VIEW_DROP_INTO_OR_AFTER;
4875 *pos = GTK_TREE_VIEW_DROP_AFTER;
4883 gtk_selection_data_set_tree_row (GtkSelectionData *selection_data,
4884 GtkTreeView *tree_view,
4892 gtk_selection_data_get_tree_row (GtkSelectionData *selection_data,
4893 GtkTreeView **tree_view,
4901 gtk_tree_view_maybe_begin_dragging_row (GtkTreeView *tree_view,
4902 GdkEventMotion *event)
4904 GdkDragContext *context;
4905 TreeViewDragInfo *di;
4906 GtkTreePath *path = NULL;
4908 gint cell_x, cell_y;
4910 di = get_info (tree_view);
4915 if (tree_view->priv->pressed_button < 0)
4918 if (!gtk_drag_check_threshold (GTK_WIDGET (tree_view),
4919 tree_view->priv->press_start_x,
4920 tree_view->priv->press_start_y,
4921 event->x, event->y))
4924 button = tree_view->priv->pressed_button;
4925 tree_view->priv->pressed_button = -1;
4927 gtk_tree_view_get_path_at_pos (tree_view,
4928 tree_view->priv->bin_window,
4929 tree_view->priv->press_start_x,
4930 tree_view->priv->press_start_y,
4939 /* FIXME if the path doesn't match the row_draggable predicate,
4940 * return FALSE and free path
4943 /* FIXME Check whether we're a start button, if not return FALSE and
4947 context = gtk_drag_begin (GTK_WIDGET (tree_view),
4948 di->source_target_list,
4953 gtk_drag_set_icon_default (context);
4958 row_pix = gtk_tree_view_create_row_drag_icon (tree_view,
4961 gtk_drag_set_icon_pixmap (context,
4962 gdk_drawable_get_colormap (row_pix),
4965 /* the + 1 is for the black border in the icon */
4966 tree_view->priv->press_start_x + 1,
4969 gdk_pixmap_unref (row_pix);
4972 di->source_row = path;
4977 /* Default signal implementations for the drag signals */
4980 gtk_tree_view_drag_begin (GtkWidget *widget,
4981 GdkDragContext *context)
4987 gtk_tree_view_drag_end (GtkWidget *widget,
4988 GdkDragContext *context)
4994 gtk_tree_view_drag_data_get (GtkWidget *widget,
4995 GdkDragContext *context,
4996 GtkSelectionData *selection_data,
5000 GtkTreeView *tree_view;
5002 tree_view = GTK_TREE_VIEW (widget);
5004 if (selection_data->target == gdk_atom_intern ("GTK_TREE_VIEW_ROW", FALSE))
5006 TreeViewDragInfo *di;
5008 di = get_info (GTK_TREE_VIEW (widget));
5012 /* There's a race where someone could have unset
5013 * drag source before the data is requested
5018 gtk_selection_data_set_tree_row (selection_data,
5019 GTK_TREE_VIEW (widget),
5025 gtk_tree_view_drag_data_delete (GtkWidget *widget,
5026 GdkDragContext *context)
5028 /* FIXME we need to delete the source_row if we're doing automagical
5035 remove_open_timeout (GtkTreeView *tree_view)
5037 if (tree_view->priv->open_dest_timeout != 0)
5039 gtk_timeout_remove (tree_view->priv->open_dest_timeout);
5040 tree_view->priv->open_dest_timeout = 0;
5045 gtk_tree_view_drag_leave (GtkWidget *widget,
5046 GdkDragContext *context,
5049 TreeViewDragInfo *di;
5051 di = get_info (GTK_TREE_VIEW (widget));
5053 /* unset any highlight row */
5054 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget),
5056 GTK_TREE_VIEW_DROP_BEFORE);
5058 remove_scroll_timeout (GTK_TREE_VIEW (widget));
5059 remove_open_timeout (GTK_TREE_VIEW (widget));
5063 open_row_timeout (gpointer data)
5065 GtkTreeView *tree_view = data;
5067 if (tree_view->priv->drag_dest_row &&
5068 (tree_view->priv->drag_dest_pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER ||
5069 tree_view->priv->drag_dest_pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE))
5071 gtk_tree_view_expand_row (tree_view,
5072 tree_view->priv->drag_dest_row,
5074 tree_view->priv->open_dest_timeout = 0;
5082 gtk_tree_view_drag_motion (GtkWidget *widget,
5083 GdkDragContext *context,
5088 GtkTreePath *path = NULL;
5089 TreeViewDragInfo *di;
5090 GtkTreeViewDropPosition pos;
5091 GtkTreeView *tree_view;
5093 tree_view = GTK_TREE_VIEW (widget);
5095 di = get_info (GTK_TREE_VIEW (widget));
5099 /* someone unset us as a drag dest, note that if
5100 * we return FALSE drag_leave isn't called
5103 gtk_tree_view_set_drag_dest_row (tree_view,
5105 GTK_TREE_VIEW_DROP_BEFORE);
5107 remove_scroll_timeout (GTK_TREE_VIEW (widget));
5108 remove_open_timeout (GTK_TREE_VIEW (widget));
5110 return FALSE; /* no longer a drop site */
5113 ensure_scroll_timeout (tree_view);
5115 if (!gtk_tree_view_get_dest_row_at_pos (tree_view,
5120 /* can't drop here */
5121 remove_open_timeout (tree_view);
5123 gdk_drag_status (context, 0, time);
5125 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget),
5127 GTK_TREE_VIEW_DROP_BEFORE);
5129 /* don't propagate to parent though */
5135 /* If we left the current row's "open" zone, unset the timeout for
5138 if (tree_view->priv->drag_dest_row &&
5139 (gtk_tree_path_compare (path, tree_view->priv->drag_dest_row) != 0 ||
5140 !(pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER ||
5141 pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE)))
5142 remove_open_timeout (tree_view);
5144 if (TRUE /* FIXME if the location droppable predicate */ &&
5145 gtk_drag_dest_find_target (widget, context, di->dest_target_list) != GDK_NONE)
5147 GtkWidget *source_widget;
5148 GdkDragAction suggested_action;
5150 suggested_action = context->suggested_action;
5152 source_widget = gtk_drag_get_source_widget (context);
5154 if (source_widget == widget)
5156 /* Default to MOVE, unless the user has
5157 * pressed ctrl or alt to affect available actions
5159 if ((context->actions & GDK_ACTION_MOVE) != 0)
5160 suggested_action = GDK_ACTION_MOVE;
5163 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget),
5166 if (tree_view->priv->open_dest_timeout == 0)
5168 tree_view->priv->open_dest_timeout =
5169 gtk_timeout_add (250, open_row_timeout, tree_view);
5172 gdk_drag_status (context, suggested_action, time);
5176 /* can't drop here */
5177 remove_open_timeout (tree_view);
5179 gdk_drag_status (context, 0, time);
5181 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget),
5183 GTK_TREE_VIEW_DROP_BEFORE);
5186 gtk_tree_path_free (path);
5192 gtk_tree_view_drag_drop (GtkWidget *widget,
5193 GdkDragContext *context,
5198 GdkAtom target = GDK_NONE;
5199 TreeViewDragInfo *di;
5200 GtkTreeView *tree_view;
5202 tree_view = GTK_TREE_VIEW (widget);
5204 remove_scroll_timeout (GTK_TREE_VIEW (widget));
5205 remove_open_timeout (GTK_TREE_VIEW (widget));
5207 di = get_info (tree_view);
5209 if (di && tree_view->priv->drag_dest_row && di->dest_target_list)
5210 target = gtk_drag_dest_find_target (widget, context, di->dest_target_list);
5212 if (target != GDK_NONE)
5214 gtk_drag_get_data (widget, context, target, time);
5222 gtk_tree_view_drag_data_received (GtkWidget *widget,
5223 GdkDragContext *context,
5226 GtkSelectionData *selection_data,
5230 if (selection_data->length >= 0)
5232 /* FIXME respond to contents of selection_data */
5236 gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget),
5238 GTK_TREE_VIEW_DROP_BEFORE);
5240 gtk_drag_finish (context,
5241 (selection_data->length >= 0),
5242 (context->action == GDK_ACTION_MOVE),