1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser 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 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser 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 * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
22 * file for a list of people on the GTK+ Team. See the ChangeLog
23 * files for a list of changes. These files are distributed with
24 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
28 #include "gtkeventbox.h"
29 #include "gtkpixmap.h"
31 #include "gtksignal.h"
32 #define GTK_ENABLE_BROKEN
34 #include "gtktreeitem.h"
36 #include "tree_plus.xpm"
37 #include "tree_minus.xpm"
39 #define DEFAULT_DELTA 9
47 typedef struct _GtkTreePixmaps GtkTreePixmaps;
49 struct _GtkTreePixmaps {
51 GdkColormap *colormap;
53 GdkPixmap *pixmap_plus;
54 GdkPixmap *pixmap_minus;
56 GdkBitmap *mask_minus;
59 static GList *pixmaps = NULL;
61 static void gtk_tree_item_class_init (GtkTreeItemClass *klass);
62 static void gtk_tree_item_init (GtkTreeItem *tree_item);
63 static void gtk_tree_item_realize (GtkWidget *widget);
64 static void gtk_tree_item_size_request (GtkWidget *widget,
65 GtkRequisition *requisition);
66 static void gtk_tree_item_size_allocate (GtkWidget *widget,
67 GtkAllocation *allocation);
68 static void gtk_tree_item_draw_focus (GtkWidget *widget);
69 static void gtk_tree_item_paint (GtkWidget *widget,
71 static gint gtk_tree_item_button_press (GtkWidget *widget,
72 GdkEventButton *event);
73 static gint gtk_tree_item_expose (GtkWidget *widget,
74 GdkEventExpose *event);
75 static gint gtk_tree_item_focus_in (GtkWidget *widget,
76 GdkEventFocus *event);
77 static gint gtk_tree_item_focus_out (GtkWidget *widget,
78 GdkEventFocus *event);
79 static void gtk_tree_item_forall (GtkContainer *container,
80 gboolean include_internals,
82 gpointer callback_data);
84 static void gtk_real_tree_item_select (GtkItem *item);
85 static void gtk_real_tree_item_deselect (GtkItem *item);
86 static void gtk_real_tree_item_toggle (GtkItem *item);
87 static void gtk_real_tree_item_expand (GtkTreeItem *item);
88 static void gtk_real_tree_item_collapse (GtkTreeItem *item);
89 static void gtk_real_tree_item_expand (GtkTreeItem *item);
90 static void gtk_real_tree_item_collapse (GtkTreeItem *item);
91 static void gtk_tree_item_destroy (GtkObject *object);
92 static void gtk_tree_item_subtree_button_click (GtkWidget *widget);
93 static void gtk_tree_item_subtree_button_changed_state (GtkWidget *widget);
95 static void gtk_tree_item_map(GtkWidget*);
96 static void gtk_tree_item_unmap(GtkWidget*);
98 static void gtk_tree_item_add_pixmaps (GtkTreeItem *tree_item);
99 static void gtk_tree_item_remove_pixmaps (GtkTreeItem *tree_item);
101 static GtkItemClass *parent_class = NULL;
102 static guint tree_item_signals[LAST_SIGNAL] = { 0 };
105 gtk_tree_item_get_type (void)
107 static GtkType tree_item_type = 0;
111 static const GtkTypeInfo tree_item_info =
114 sizeof (GtkTreeItem),
115 sizeof (GtkTreeItemClass),
116 (GtkClassInitFunc) gtk_tree_item_class_init,
117 (GtkObjectInitFunc) gtk_tree_item_init,
118 /* reserved_1 */ NULL,
119 /* reserved_2 */ NULL,
120 (GtkClassInitFunc) NULL,
123 tree_item_type = gtk_type_unique (gtk_item_get_type (), &tree_item_info);
126 return tree_item_type;
130 gtk_tree_item_class_init (GtkTreeItemClass *class)
132 GtkObjectClass *object_class;
133 GtkWidgetClass *widget_class;
134 GtkContainerClass *container_class;
135 GtkItemClass *item_class;
137 parent_class = gtk_type_class (GTK_TYPE_ITEM);
139 object_class = (GtkObjectClass*) class;
140 widget_class = (GtkWidgetClass*) class;
141 item_class = (GtkItemClass*) class;
142 container_class = (GtkContainerClass*) class;
144 object_class->destroy = gtk_tree_item_destroy;
146 widget_class->realize = gtk_tree_item_realize;
147 widget_class->size_request = gtk_tree_item_size_request;
148 widget_class->size_allocate = gtk_tree_item_size_allocate;
149 widget_class->draw_focus = gtk_tree_item_draw_focus;
150 widget_class->button_press_event = gtk_tree_item_button_press;
151 widget_class->expose_event = gtk_tree_item_expose;
152 widget_class->focus_in_event = gtk_tree_item_focus_in;
153 widget_class->focus_out_event = gtk_tree_item_focus_out;
154 widget_class->map = gtk_tree_item_map;
155 widget_class->unmap = gtk_tree_item_unmap;
157 container_class->forall = gtk_tree_item_forall;
159 item_class->select = gtk_real_tree_item_select;
160 item_class->deselect = gtk_real_tree_item_deselect;
161 item_class->toggle = gtk_real_tree_item_toggle;
163 class->expand = gtk_real_tree_item_expand;
164 class->collapse = gtk_real_tree_item_collapse;
166 tree_item_signals[EXPAND_TREE] =
167 gtk_signal_new ("expand",
169 GTK_CLASS_TYPE (object_class),
170 GTK_SIGNAL_OFFSET (GtkTreeItemClass, expand),
171 gtk_marshal_VOID__VOID,
173 tree_item_signals[COLLAPSE_TREE] =
174 gtk_signal_new ("collapse",
176 GTK_CLASS_TYPE (object_class),
177 GTK_SIGNAL_OFFSET (GtkTreeItemClass, collapse),
178 gtk_marshal_VOID__VOID,
182 /* callback for event box mouse event */
184 gtk_tree_item_subtree_button_click (GtkWidget *widget)
188 g_return_if_fail (widget != NULL);
189 g_return_if_fail (GTK_IS_EVENT_BOX (widget));
191 item = (GtkTreeItem*) gtk_object_get_user_data (GTK_OBJECT (widget));
192 if (!GTK_WIDGET_IS_SENSITIVE (item))
196 gtk_tree_item_collapse (item);
198 gtk_tree_item_expand (item);
201 /* callback for event box state changed */
203 gtk_tree_item_subtree_button_changed_state (GtkWidget *widget)
205 g_return_if_fail (widget != NULL);
206 g_return_if_fail (GTK_IS_EVENT_BOX (widget));
208 if (GTK_WIDGET_VISIBLE (widget))
211 if (widget->state == GTK_STATE_NORMAL)
212 gdk_window_set_background (widget->window, &widget->style->base[widget->state]);
214 gdk_window_set_background (widget->window, &widget->style->bg[widget->state]);
216 if (GTK_WIDGET_DRAWABLE (widget))
217 gdk_window_clear_area (widget->window, 0, 0,
218 widget->allocation.width, widget->allocation.height);
223 gtk_tree_item_init (GtkTreeItem *tree_item)
225 GtkWidget *eventbox, *pixmapwid;
227 g_return_if_fail (tree_item != NULL);
228 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
230 tree_item->expanded = FALSE;
231 tree_item->subtree = NULL;
232 GTK_WIDGET_SET_FLAGS (tree_item, GTK_CAN_FOCUS);
234 /* create an event box containing one pixmaps */
235 eventbox = gtk_event_box_new();
236 gtk_widget_set_events (eventbox, GDK_BUTTON_PRESS_MASK);
237 gtk_signal_connect(GTK_OBJECT(eventbox), "state_changed",
238 (GtkSignalFunc)gtk_tree_item_subtree_button_changed_state,
240 gtk_signal_connect(GTK_OBJECT(eventbox), "realize",
241 (GtkSignalFunc)gtk_tree_item_subtree_button_changed_state,
243 gtk_signal_connect(GTK_OBJECT(eventbox), "button_press_event",
244 (GtkSignalFunc)gtk_tree_item_subtree_button_click,
246 gtk_object_set_user_data(GTK_OBJECT(eventbox), tree_item);
247 tree_item->pixmaps_box = eventbox;
249 /* create pixmap for button '+' */
250 pixmapwid = gtk_type_new (gtk_pixmap_get_type ());
251 if (!tree_item->expanded)
252 gtk_container_add (GTK_CONTAINER (eventbox), pixmapwid);
253 gtk_widget_show (pixmapwid);
254 tree_item->plus_pix_widget = pixmapwid;
255 gtk_widget_ref (tree_item->plus_pix_widget);
256 gtk_object_sink (GTK_OBJECT (tree_item->plus_pix_widget));
258 /* create pixmap for button '-' */
259 pixmapwid = gtk_type_new (gtk_pixmap_get_type ());
260 if (tree_item->expanded)
261 gtk_container_add (GTK_CONTAINER (eventbox), pixmapwid);
262 gtk_widget_show (pixmapwid);
263 tree_item->minus_pix_widget = pixmapwid;
264 gtk_widget_ref (tree_item->minus_pix_widget);
265 gtk_object_sink (GTK_OBJECT (tree_item->minus_pix_widget));
267 gtk_widget_set_parent (eventbox, GTK_WIDGET (tree_item));
272 gtk_tree_item_new (void)
274 GtkWidget *tree_item;
276 tree_item = GTK_WIDGET (gtk_type_new (gtk_tree_item_get_type ()));
282 gtk_tree_item_new_with_label (const gchar *label)
284 GtkWidget *tree_item;
285 GtkWidget *label_widget;
287 tree_item = gtk_tree_item_new ();
288 label_widget = gtk_label_new (label);
289 gtk_misc_set_alignment (GTK_MISC (label_widget), 0.0, 0.5);
291 gtk_container_add (GTK_CONTAINER (tree_item), label_widget);
292 gtk_widget_show (label_widget);
299 gtk_tree_item_set_subtree (GtkTreeItem *tree_item,
302 g_return_if_fail (tree_item != NULL);
303 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
304 g_return_if_fail (subtree != NULL);
305 g_return_if_fail (GTK_IS_TREE (subtree));
307 if (tree_item->subtree)
309 g_warning("there is already a subtree for this tree item\n");
313 tree_item->subtree = subtree;
314 GTK_TREE (subtree)->tree_owner = GTK_WIDGET (tree_item);
316 /* show subtree button */
317 if (tree_item->pixmaps_box)
318 gtk_widget_show (tree_item->pixmaps_box);
320 if (tree_item->expanded)
321 gtk_widget_show (subtree);
323 gtk_widget_hide (subtree);
325 gtk_widget_set_parent (subtree, GTK_WIDGET (tree_item)->parent);
327 if (GTK_WIDGET_REALIZED (subtree->parent))
328 gtk_widget_realize (subtree);
330 if (GTK_WIDGET_VISIBLE (subtree->parent) && GTK_WIDGET_VISIBLE (subtree))
332 if (GTK_WIDGET_MAPPED (subtree->parent))
333 gtk_widget_map (subtree);
335 gtk_widget_queue_resize (subtree);
340 gtk_tree_item_select (GtkTreeItem *tree_item)
342 g_return_if_fail (tree_item != NULL);
343 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
345 gtk_item_select (GTK_ITEM (tree_item));
349 gtk_tree_item_deselect (GtkTreeItem *tree_item)
351 g_return_if_fail (tree_item != NULL);
352 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
354 gtk_item_deselect (GTK_ITEM (tree_item));
358 gtk_tree_item_expand (GtkTreeItem *tree_item)
360 g_return_if_fail (tree_item != NULL);
361 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
363 gtk_signal_emit (GTK_OBJECT (tree_item), tree_item_signals[EXPAND_TREE], NULL);
367 gtk_tree_item_collapse (GtkTreeItem *tree_item)
369 g_return_if_fail (tree_item != NULL);
370 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
372 gtk_signal_emit (GTK_OBJECT (tree_item), tree_item_signals[COLLAPSE_TREE], NULL);
376 gtk_tree_item_add_pixmaps (GtkTreeItem *tree_item)
379 GdkColormap *colormap;
380 GtkTreePixmaps *pixmap_node = NULL;
382 g_return_if_fail (tree_item != NULL);
383 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
385 if (tree_item->pixmaps)
388 colormap = gtk_widget_get_colormap (GTK_WIDGET (tree_item));
393 pixmap_node = (GtkTreePixmaps *)tmp_list->data;
395 if (pixmap_node->colormap == colormap)
398 tmp_list = tmp_list->next;
403 pixmap_node->refcount++;
404 tree_item->pixmaps = tmp_list;
408 pixmap_node = g_new (GtkTreePixmaps, 1);
410 pixmap_node->colormap = colormap;
411 gdk_colormap_ref (colormap);
413 pixmap_node->refcount = 1;
415 /* create pixmaps for plus icon */
416 pixmap_node->pixmap_plus =
417 gdk_pixmap_create_from_xpm_d (GTK_WIDGET (tree_item)->window,
418 &pixmap_node->mask_plus,
422 /* create pixmaps for minus icon */
423 pixmap_node->pixmap_minus =
424 gdk_pixmap_create_from_xpm_d (GTK_WIDGET (tree_item)->window,
425 &pixmap_node->mask_minus,
429 tree_item->pixmaps = pixmaps = g_list_prepend (pixmaps, pixmap_node);
432 gtk_pixmap_set (GTK_PIXMAP (tree_item->plus_pix_widget),
433 pixmap_node->pixmap_plus, pixmap_node->mask_plus);
434 gtk_pixmap_set (GTK_PIXMAP (tree_item->minus_pix_widget),
435 pixmap_node->pixmap_minus, pixmap_node->mask_minus);
439 gtk_tree_item_remove_pixmaps (GtkTreeItem *tree_item)
441 g_return_if_fail (tree_item != NULL);
442 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
444 if (tree_item->pixmaps)
446 GtkTreePixmaps *pixmap_node = (GtkTreePixmaps *)tree_item->pixmaps->data;
448 g_assert (pixmap_node->refcount > 0);
450 if (--pixmap_node->refcount == 0)
452 gdk_colormap_unref (pixmap_node->colormap);
453 gdk_pixmap_unref (pixmap_node->pixmap_plus);
454 gdk_bitmap_unref (pixmap_node->mask_plus);
455 gdk_pixmap_unref (pixmap_node->pixmap_minus);
456 gdk_bitmap_unref (pixmap_node->mask_minus);
458 pixmaps = g_list_remove_link (pixmaps, tree_item->pixmaps);
459 g_list_free_1 (tree_item->pixmaps);
460 g_free (pixmap_node);
463 tree_item->pixmaps = NULL;
468 gtk_tree_item_realize (GtkWidget *widget)
470 g_return_if_fail (widget != NULL);
471 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
473 if (GTK_WIDGET_CLASS (parent_class)->realize)
474 (* GTK_WIDGET_CLASS (parent_class)->realize) (widget);
476 gdk_window_set_background (widget->window,
477 &widget->style->base[GTK_STATE_NORMAL]);
479 gtk_tree_item_add_pixmaps (GTK_TREE_ITEM (widget));
483 gtk_tree_item_size_request (GtkWidget *widget,
484 GtkRequisition *requisition)
488 GtkRequisition child_requisition;
490 g_return_if_fail (widget != NULL);
491 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
492 g_return_if_fail (requisition != NULL);
494 bin = GTK_BIN (widget);
495 item = GTK_TREE_ITEM(widget);
497 requisition->width = (GTK_CONTAINER (widget)->border_width +
498 widget->style->xthickness) * 2;
499 requisition->height = GTK_CONTAINER (widget)->border_width * 2;
501 if (bin->child && GTK_WIDGET_VISIBLE (bin->child))
503 GtkRequisition pix_requisition;
505 gtk_widget_size_request (bin->child, &child_requisition);
507 requisition->width += child_requisition.width;
509 gtk_widget_size_request (item->pixmaps_box,
511 requisition->width += pix_requisition.width + DEFAULT_DELTA +
512 GTK_TREE (widget->parent)->current_indent;
514 requisition->height += MAX (child_requisition.height,
515 pix_requisition.height);
520 gtk_tree_item_size_allocate (GtkWidget *widget,
521 GtkAllocation *allocation)
525 GtkAllocation child_allocation;
529 g_return_if_fail (widget != NULL);
530 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
531 g_return_if_fail (allocation != NULL);
533 widget->allocation = *allocation;
534 if (GTK_WIDGET_REALIZED (widget))
535 gdk_window_move_resize (widget->window,
536 allocation->x, allocation->y,
537 allocation->width, allocation->height);
539 bin = GTK_BIN (widget);
540 item = GTK_TREE_ITEM(widget);
544 border_width = (GTK_CONTAINER (widget)->border_width +
545 widget->style->xthickness);
547 child_allocation.x = border_width + GTK_TREE(widget->parent)->current_indent;
548 child_allocation.y = GTK_CONTAINER (widget)->border_width;
550 child_allocation.width = item->pixmaps_box->requisition.width;
551 child_allocation.height = item->pixmaps_box->requisition.height;
553 temp = allocation->height - child_allocation.height;
554 child_allocation.y += ( temp / 2 ) + ( temp % 2 );
556 gtk_widget_size_allocate (item->pixmaps_box, &child_allocation);
558 child_allocation.y = GTK_CONTAINER (widget)->border_width;
559 child_allocation.height = MAX (1, (gint)allocation->height - child_allocation.y * 2);
560 child_allocation.x += item->pixmaps_box->requisition.width+DEFAULT_DELTA;
562 child_allocation.width =
563 MAX (1, (gint)allocation->width - ((gint)child_allocation.x + border_width));
565 gtk_widget_size_allocate (bin->child, &child_allocation);
570 gtk_tree_item_draw_lines (GtkWidget *widget)
574 guint lx1, ly1, lx2, ly2;
576 g_return_if_fail (widget != NULL);
577 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
579 item = GTK_TREE_ITEM(widget);
580 tree = GTK_TREE(widget->parent);
582 if (!tree->view_line)
585 /* draw vertical line */
586 lx1 = item->pixmaps_box->allocation.width;
587 lx1 = lx2 = ((lx1 / 2) + (lx1 % 2) +
588 GTK_CONTAINER (widget)->border_width + 1 + tree->current_indent);
590 ly2 = widget->allocation.height;
592 if (g_list_last (tree->children)->data == widget)
593 ly2 = (ly2 / 2) + (ly2 % 2);
595 if (tree != tree->root_tree)
596 gdk_draw_line (widget->window, widget->style->black_gc, lx1, ly1, lx2, ly2);
598 /* draw vertical line for subtree connecting */
599 if(g_list_last(tree->children)->data != (gpointer)widget)
600 ly2 = (ly2 / 2) + (ly2 % 2);
602 lx2 += DEFAULT_DELTA;
604 if (item->subtree && item->expanded)
605 gdk_draw_line (widget->window, widget->style->black_gc,
606 lx2, ly2, lx2, widget->allocation.height);
608 /* draw horizontal line */
612 gdk_draw_line (widget->window, widget->style->black_gc,
615 lx2 -= DEFAULT_DELTA+2;
617 ly2 = widget->allocation.height;
619 if (tree != tree->root_tree)
621 item = GTK_TREE_ITEM (tree->tree_owner);
622 tree = GTK_TREE (GTK_WIDGET (tree)->parent);
623 while (tree != tree->root_tree)
625 lx1 = lx2 -= tree->indent_value;
627 if (g_list_last (tree->children)->data != item)
628 gdk_draw_line (widget->window, widget->style->black_gc, lx1, ly1, lx2, ly2);
629 item = GTK_TREE_ITEM (tree->tree_owner);
630 tree = GTK_TREE (GTK_WIDGET (tree)->parent);
636 gtk_tree_item_paint (GtkWidget *widget,
640 GdkRectangle child_area, item_area;
641 GtkTreeItem* tree_item;
643 g_return_if_fail (widget != NULL);
644 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
645 g_return_if_fail (area != NULL);
647 /* FIXME: We should honor tree->view_mode, here - I think
648 * the desired effect is that when the mode is VIEW_ITEM,
649 * only the subitem is drawn as selected, not the entire
650 * line. (Like the way that the tree in Windows Explorer
653 if (GTK_WIDGET_DRAWABLE (widget))
655 bin = GTK_BIN (widget);
656 tree_item = GTK_TREE_ITEM(widget);
658 if (widget->state == GTK_STATE_NORMAL)
660 gdk_window_set_back_pixmap (widget->window, NULL, TRUE);
661 gdk_window_clear_area (widget->window, area->x, area->y, area->width, area->height);
665 if (!GTK_WIDGET_IS_SENSITIVE (widget))
666 gtk_paint_flat_box(widget->style, widget->window,
667 widget->state, GTK_STATE_INSENSITIVE,
668 area, widget, "treeitem",
671 gtk_paint_flat_box(widget->style, widget->window,
672 widget->state, GTK_SHADOW_ETCHED_OUT,
673 area, widget, "treeitem",
677 /* draw left size of tree item */
680 item_area.width = (tree_item->pixmaps_box->allocation.width + DEFAULT_DELTA +
681 GTK_TREE (widget->parent)->current_indent + 2);
682 item_area.height = widget->allocation.height;
685 if (gdk_rectangle_intersect(&item_area, area, &child_area))
688 gtk_tree_item_draw_lines(widget);
690 if (tree_item->pixmaps_box &&
691 GTK_WIDGET_VISIBLE(tree_item->pixmaps_box) &&
692 gtk_widget_intersect (tree_item->pixmaps_box, area, &child_area))
693 gtk_widget_draw (tree_item->pixmaps_box, &child_area);
696 if (GTK_WIDGET_HAS_FOCUS (widget))
697 gtk_paint_focus (widget->style, widget->window,
698 NULL, widget, "treeitem",
700 widget->allocation.width - 1,
701 widget->allocation.height - 1);
707 gtk_tree_item_draw_focus (GtkWidget *widget)
709 g_return_if_fail (widget != NULL);
710 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
712 gtk_widget_draw(widget, NULL);
716 gtk_tree_item_button_press (GtkWidget *widget,
717 GdkEventButton *event)
720 g_return_val_if_fail (widget != NULL, FALSE);
721 g_return_val_if_fail (GTK_IS_TREE_ITEM (widget), FALSE);
722 g_return_val_if_fail (event != NULL, FALSE);
724 if (event->type == GDK_BUTTON_PRESS
725 && GTK_WIDGET_IS_SENSITIVE(widget)
726 && !GTK_WIDGET_HAS_FOCUS (widget))
727 gtk_widget_grab_focus (widget);
733 gtk_tree_item_expose (GtkWidget *widget,
734 GdkEventExpose *event)
736 GdkEventExpose child_event;
739 g_return_val_if_fail (widget != NULL, FALSE);
740 g_return_val_if_fail (GTK_IS_TREE_ITEM (widget), FALSE);
741 g_return_val_if_fail (event != NULL, FALSE);
743 if (GTK_WIDGET_DRAWABLE (widget))
745 bin = GTK_BIN (widget);
747 gtk_tree_item_paint (widget, &event->area);
749 child_event = *event;
750 if (bin->child && GTK_WIDGET_NO_WINDOW (bin->child) &&
751 gtk_widget_intersect (bin->child, &event->area, &child_event.area))
752 gtk_widget_event (bin->child, (GdkEvent*) &child_event);
759 gtk_tree_item_focus_in (GtkWidget *widget,
760 GdkEventFocus *event)
762 g_return_val_if_fail (widget != NULL, FALSE);
763 g_return_val_if_fail (GTK_IS_TREE_ITEM (widget), FALSE);
764 g_return_val_if_fail (event != NULL, FALSE);
766 GTK_WIDGET_SET_FLAGS (widget, GTK_HAS_FOCUS);
767 gtk_widget_draw_focus (widget);
774 gtk_tree_item_focus_out (GtkWidget *widget,
775 GdkEventFocus *event)
777 g_return_val_if_fail (widget != NULL, FALSE);
778 g_return_val_if_fail (GTK_IS_TREE_ITEM (widget), FALSE);
779 g_return_val_if_fail (event != NULL, FALSE);
781 GTK_WIDGET_UNSET_FLAGS (widget, GTK_HAS_FOCUS);
782 gtk_widget_draw_focus (widget);
789 gtk_real_tree_item_select (GtkItem *item)
791 GtkTreeItem *tree_item;
794 g_return_if_fail (item != NULL);
795 g_return_if_fail (GTK_IS_TREE_ITEM (item));
797 tree_item = GTK_TREE_ITEM (item);
798 widget = GTK_WIDGET (item);
800 gtk_widget_set_state (GTK_WIDGET (item), GTK_STATE_SELECTED);
802 if (!widget->parent || GTK_TREE (widget->parent)->view_mode == GTK_TREE_VIEW_LINE)
803 gtk_widget_set_state (GTK_TREE_ITEM (item)->pixmaps_box, GTK_STATE_SELECTED);
807 gtk_real_tree_item_deselect (GtkItem *item)
809 GtkTreeItem *tree_item;
812 g_return_if_fail (item != NULL);
813 g_return_if_fail (GTK_IS_TREE_ITEM (item));
815 tree_item = GTK_TREE_ITEM (item);
816 widget = GTK_WIDGET (item);
818 gtk_widget_set_state (widget, GTK_STATE_NORMAL);
820 if (!widget->parent || GTK_TREE (widget->parent)->view_mode == GTK_TREE_VIEW_LINE)
821 gtk_widget_set_state (tree_item->pixmaps_box, GTK_STATE_NORMAL);
825 gtk_real_tree_item_toggle (GtkItem *item)
827 g_return_if_fail (item != NULL);
828 g_return_if_fail (GTK_IS_TREE_ITEM (item));
830 if(!GTK_WIDGET_IS_SENSITIVE(item))
833 if (GTK_WIDGET (item)->parent && GTK_IS_TREE (GTK_WIDGET (item)->parent))
834 gtk_tree_select_child (GTK_TREE (GTK_WIDGET (item)->parent),
838 /* Should we really bother with this bit? A listitem not in a list?
839 * -Johannes Keukelaar
840 * yes, always be on the safe side!
843 if (GTK_WIDGET (item)->state == GTK_STATE_SELECTED)
844 gtk_widget_set_state (GTK_WIDGET (item), GTK_STATE_NORMAL);
846 gtk_widget_set_state (GTK_WIDGET (item), GTK_STATE_SELECTED);
851 gtk_real_tree_item_expand (GtkTreeItem *tree_item)
855 g_return_if_fail (tree_item != NULL);
856 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
858 if (tree_item->subtree && !tree_item->expanded)
860 tree = GTK_TREE (GTK_WIDGET (tree_item)->parent);
862 /* hide subtree widget */
863 gtk_widget_show (tree_item->subtree);
865 /* hide button '+' and show button '-' */
866 if (tree_item->pixmaps_box)
868 gtk_container_remove (GTK_CONTAINER (tree_item->pixmaps_box),
869 tree_item->plus_pix_widget);
870 gtk_container_add (GTK_CONTAINER (tree_item->pixmaps_box),
871 tree_item->minus_pix_widget);
874 gtk_widget_queue_resize (GTK_WIDGET (tree->root_tree));
875 tree_item->expanded = TRUE;
880 gtk_real_tree_item_collapse (GtkTreeItem *tree_item)
884 g_return_if_fail (tree_item != NULL);
885 g_return_if_fail (GTK_IS_TREE_ITEM (tree_item));
887 if (tree_item->subtree && tree_item->expanded)
889 tree = GTK_TREE (GTK_WIDGET (tree_item)->parent);
891 /* hide subtree widget */
892 gtk_widget_hide (tree_item->subtree);
894 /* hide button '-' and show button '+' */
895 if (tree_item->pixmaps_box)
897 gtk_container_remove (GTK_CONTAINER (tree_item->pixmaps_box),
898 tree_item->minus_pix_widget);
899 gtk_container_add (GTK_CONTAINER (tree_item->pixmaps_box),
900 tree_item->plus_pix_widget);
903 gtk_widget_queue_resize (GTK_WIDGET (tree->root_tree));
904 tree_item->expanded = FALSE;
909 gtk_tree_item_destroy (GtkObject *object)
914 g_return_if_fail (object != NULL);
915 g_return_if_fail (GTK_IS_TREE_ITEM (object));
918 g_message("+ gtk_tree_item_destroy [object %#x]\n", (int)object);
919 #endif /* TREE_DEBUG */
921 item = GTK_TREE_ITEM(object);
923 /* free sub tree if it exist */
924 child = item->subtree;
927 gtk_widget_ref (child);
928 gtk_widget_unparent (child);
929 gtk_widget_destroy (child);
930 gtk_widget_unref (child);
931 item->subtree = NULL;
934 /* free pixmaps box */
935 child = item->pixmaps_box;
938 gtk_widget_ref (child);
939 gtk_widget_unparent (child);
940 gtk_widget_destroy (child);
941 gtk_widget_unref (child);
942 item->pixmaps_box = NULL;
946 /* destroy plus pixmap */
947 if (item->plus_pix_widget)
949 gtk_widget_destroy (item->plus_pix_widget);
950 gtk_widget_unref (item->plus_pix_widget);
951 item->plus_pix_widget = NULL;
954 /* destroy minus pixmap */
955 if (item->minus_pix_widget)
957 gtk_widget_destroy (item->minus_pix_widget);
958 gtk_widget_unref (item->minus_pix_widget);
959 item->minus_pix_widget = NULL;
962 /* By removing the pixmaps here, and not in unrealize, we depend on
963 * the fact that a widget can never change colormap or visual.
965 gtk_tree_item_remove_pixmaps (item);
967 GTK_OBJECT_CLASS (parent_class)->destroy (object);
970 g_message("- gtk_tree_item_destroy\n");
971 #endif /* TREE_DEBUG */
975 gtk_tree_item_remove_subtree (GtkTreeItem* item)
977 g_return_if_fail (item != NULL);
978 g_return_if_fail (GTK_IS_TREE_ITEM(item));
979 g_return_if_fail (item->subtree != NULL);
981 if (GTK_TREE (item->subtree)->children)
983 /* The following call will remove the children and call
984 * gtk_tree_item_remove_subtree() again. So we are done.
986 gtk_tree_remove_items (GTK_TREE (item->subtree),
987 GTK_TREE (item->subtree)->children);
991 if (GTK_WIDGET_MAPPED (item->subtree))
992 gtk_widget_unmap (item->subtree);
994 gtk_widget_unparent (item->subtree);
996 if (item->pixmaps_box)
997 gtk_widget_hide (item->pixmaps_box);
999 item->subtree = NULL;
1003 item->expanded = FALSE;
1004 if (item->pixmaps_box)
1006 gtk_container_remove (GTK_CONTAINER (item->pixmaps_box),
1007 item->minus_pix_widget);
1008 gtk_container_add (GTK_CONTAINER (item->pixmaps_box),
1009 item->plus_pix_widget);
1015 gtk_tree_item_map (GtkWidget *widget)
1020 g_return_if_fail (widget != NULL);
1021 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
1023 bin = GTK_BIN (widget);
1024 item = GTK_TREE_ITEM(widget);
1026 GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED);
1028 if(item->pixmaps_box &&
1029 GTK_WIDGET_VISIBLE (item->pixmaps_box) &&
1030 !GTK_WIDGET_MAPPED (item->pixmaps_box))
1031 gtk_widget_map (item->pixmaps_box);
1034 GTK_WIDGET_VISIBLE (bin->child) &&
1035 !GTK_WIDGET_MAPPED (bin->child))
1036 gtk_widget_map (bin->child);
1038 gdk_window_show (widget->window);
1042 gtk_tree_item_unmap (GtkWidget *widget)
1047 g_return_if_fail (widget != NULL);
1048 g_return_if_fail (GTK_IS_TREE_ITEM (widget));
1050 GTK_WIDGET_UNSET_FLAGS (widget, GTK_MAPPED);
1051 bin = GTK_BIN (widget);
1052 item = GTK_TREE_ITEM(widget);
1054 gdk_window_hide (widget->window);
1056 if(item->pixmaps_box &&
1057 GTK_WIDGET_VISIBLE (item->pixmaps_box) &&
1058 GTK_WIDGET_MAPPED (item->pixmaps_box))
1059 gtk_widget_unmap (bin->child);
1062 GTK_WIDGET_VISIBLE (bin->child) &&
1063 GTK_WIDGET_MAPPED (bin->child))
1064 gtk_widget_unmap (bin->child);
1068 gtk_tree_item_forall (GtkContainer *container,
1069 gboolean include_internals,
1070 GtkCallback callback,
1071 gpointer callback_data)
1074 GtkTreeItem *tree_item;
1076 g_return_if_fail (container != NULL);
1077 g_return_if_fail (GTK_IS_TREE_ITEM (container));
1078 g_return_if_fail (callback != NULL);
1080 bin = GTK_BIN (container);
1081 tree_item = GTK_TREE_ITEM (container);
1084 (* callback) (bin->child, callback_data);
1085 if (include_internals && tree_item->subtree)
1086 (* callback) (tree_item->subtree, callback_data);