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 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 * Modified by the GTK+ Team and others 1997-1999. 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/.
30 static void gtk_paned_class_init (GtkPanedClass *klass);
31 static void gtk_paned_init (GtkPaned *paned);
32 static void gtk_paned_realize (GtkWidget *widget);
33 static void gtk_paned_map (GtkWidget *widget);
34 static void gtk_paned_unmap (GtkWidget *widget);
35 static void gtk_paned_unrealize (GtkWidget *widget);
36 static gint gtk_paned_expose (GtkWidget *widget,
37 GdkEventExpose *event);
38 static void gtk_paned_add (GtkContainer *container,
40 static void gtk_paned_remove (GtkContainer *container,
42 static void gtk_paned_forall (GtkContainer *container,
43 gboolean include_internals,
45 gpointer callback_data);
46 static GtkType gtk_paned_child_type (GtkContainer *container);
49 static GtkContainerClass *parent_class = NULL;
53 gtk_paned_get_type (void)
55 static GtkType paned_type = 0;
59 static const GtkTypeInfo paned_info =
63 sizeof (GtkPanedClass),
64 (GtkClassInitFunc) gtk_paned_class_init,
65 (GtkObjectInitFunc) gtk_paned_init,
66 /* reserved_1 */ NULL,
67 /* reserved_2 */ NULL,
68 (GtkClassInitFunc) NULL,
71 paned_type = gtk_type_unique (GTK_TYPE_CONTAINER, &paned_info);
78 gtk_paned_class_init (GtkPanedClass *class)
80 GtkObjectClass *object_class;
81 GtkWidgetClass *widget_class;
82 GtkContainerClass *container_class;
84 object_class = (GtkObjectClass*) class;
85 widget_class = (GtkWidgetClass*) class;
86 container_class = (GtkContainerClass*) class;
88 parent_class = gtk_type_class (GTK_TYPE_CONTAINER);
90 widget_class->realize = gtk_paned_realize;
91 widget_class->map = gtk_paned_map;
92 widget_class->unmap = gtk_paned_unmap;
93 widget_class->unrealize = gtk_paned_unrealize;
94 widget_class->expose_event = gtk_paned_expose;
96 container_class->add = gtk_paned_add;
97 container_class->remove = gtk_paned_remove;
98 container_class->forall = gtk_paned_forall;
99 container_class->child_type = gtk_paned_child_type;
103 gtk_paned_child_type (GtkContainer *container)
105 if (!GTK_PANED (container)->child1 || !GTK_PANED (container)->child2)
106 return GTK_TYPE_WIDGET;
108 return GTK_TYPE_NONE;
112 gtk_paned_init (GtkPaned *paned)
114 GTK_WIDGET_UNSET_FLAGS (paned, GTK_NO_WINDOW);
116 paned->child1 = NULL;
117 paned->child2 = NULL;
118 paned->handle = NULL;
119 paned->xor_gc = NULL;
121 paned->handle_size = 10;
122 paned->gutter_size = 6;
123 paned->position_set = FALSE;
124 paned->last_allocation = -1;
125 paned->in_drag = FALSE;
127 paned->handle_xpos = -1;
128 paned->handle_ypos = -1;
133 gtk_paned_realize (GtkWidget *widget)
136 GdkWindowAttr attributes;
137 gint attributes_mask;
139 g_return_if_fail (widget != NULL);
140 g_return_if_fail (GTK_IS_PANED (widget));
142 GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED);
143 paned = GTK_PANED (widget);
145 attributes.x = widget->allocation.x;
146 attributes.y = widget->allocation.y;
147 attributes.width = widget->allocation.width;
148 attributes.height = widget->allocation.height;
149 attributes.window_type = GDK_WINDOW_CHILD;
150 attributes.wclass = GDK_INPUT_OUTPUT;
151 attributes.visual = gtk_widget_get_visual (widget);
152 attributes.colormap = gtk_widget_get_colormap (widget);
153 attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
154 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
156 widget->window = gdk_window_new (gtk_widget_get_parent_window (widget),
157 &attributes, attributes_mask);
158 gdk_window_set_user_data (widget->window, paned);
160 attributes.x = paned->handle_xpos;
161 attributes.y = paned->handle_ypos;
162 attributes.width = paned->handle_size;
163 attributes.height = paned->handle_size;
164 attributes.cursor = gdk_cursor_new (GDK_CROSS);
165 attributes.event_mask |= (GDK_BUTTON_PRESS_MASK |
166 GDK_BUTTON_RELEASE_MASK |
167 GDK_POINTER_MOTION_MASK |
168 GDK_POINTER_MOTION_HINT_MASK);
169 attributes_mask |= GDK_WA_CURSOR;
171 paned->handle = gdk_window_new (widget->window,
172 &attributes, attributes_mask);
173 gdk_window_set_user_data (paned->handle, paned);
174 gdk_cursor_destroy (attributes.cursor);
176 widget->style = gtk_style_attach (widget->style, widget->window);
178 gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL);
179 gtk_style_set_background (widget->style, paned->handle, GTK_STATE_NORMAL);
181 gdk_window_set_back_pixmap (widget->window, NULL, TRUE);
183 gdk_window_show (paned->handle);
187 gtk_paned_map (GtkWidget *widget)
191 g_return_if_fail (widget != NULL);
192 g_return_if_fail (GTK_IS_PANED (widget));
194 GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED);
195 paned = GTK_PANED (widget);
198 GTK_WIDGET_VISIBLE (paned->child1) &&
199 !GTK_WIDGET_MAPPED (paned->child1))
200 gtk_widget_map (paned->child1);
202 GTK_WIDGET_VISIBLE (paned->child2) &&
203 !GTK_WIDGET_MAPPED (paned->child2))
204 gtk_widget_map (paned->child2);
206 gdk_window_show (widget->window);
210 gtk_paned_unmap (GtkWidget *widget)
212 g_return_if_fail (widget != NULL);
213 g_return_if_fail (GTK_IS_PANED (widget));
215 GTK_WIDGET_UNSET_FLAGS (widget, GTK_MAPPED);
217 gdk_window_hide (widget->window);
221 gtk_paned_unrealize (GtkWidget *widget)
225 g_return_if_fail (widget != NULL);
226 g_return_if_fail (GTK_IS_PANED (widget));
228 paned = GTK_PANED (widget);
232 gdk_gc_destroy (paned->xor_gc);
233 paned->xor_gc = NULL;
238 gdk_window_set_user_data (paned->handle, NULL);
239 gdk_window_destroy (paned->handle);
240 paned->handle = NULL;
243 if (GTK_WIDGET_CLASS (parent_class)->unrealize)
244 (* GTK_WIDGET_CLASS (parent_class)->unrealize) (widget);
248 gtk_paned_expose (GtkWidget *widget,
249 GdkEventExpose *event)
252 GdkEventExpose child_event;
254 g_return_val_if_fail (widget != NULL, FALSE);
255 g_return_val_if_fail (GTK_IS_PANED (widget), FALSE);
256 g_return_val_if_fail (event != NULL, FALSE);
258 if (GTK_WIDGET_DRAWABLE (widget))
260 paned = GTK_PANED (widget);
262 /* An expose event for the handle */
263 if (event->window == paned->handle)
265 gtk_paint_box (widget->style, paned->handle,
266 GTK_WIDGET_STATE(widget),
268 &event->area, widget, "paned",
270 paned->handle_size, paned->handle_size);
274 child_event = *event;
276 GTK_WIDGET_NO_WINDOW (paned->child1) &&
277 gtk_widget_intersect (paned->child1, &event->area, &child_event.area))
278 gtk_widget_event (paned->child1, (GdkEvent*) &child_event);
281 GTK_WIDGET_NO_WINDOW (paned->child2) &&
282 gtk_widget_intersect (paned->child2, &event->area, &child_event.area))
283 gtk_widget_event (paned->child2, (GdkEvent*) &child_event);
285 /* redraw the groove if necessary */
286 if (gdk_rectangle_intersect (&paned->groove_rectangle,
289 gtk_widget_draw (widget, &child_event.area);
296 gtk_paned_add1 (GtkPaned *paned,
299 gtk_paned_pack1 (paned, widget, FALSE, TRUE);
303 gtk_paned_add2 (GtkPaned *paned,
306 gtk_paned_pack2 (paned, widget, TRUE, TRUE);
310 gtk_paned_pack1 (GtkPaned *paned,
315 g_return_if_fail (paned != NULL);
316 g_return_if_fail (GTK_IS_PANED (paned));
317 g_return_if_fail (widget != NULL);
321 gtk_widget_set_parent (widget, GTK_WIDGET (paned));
323 if (GTK_WIDGET_VISIBLE (widget->parent))
325 if (GTK_WIDGET_REALIZED (widget->parent) &&
326 !GTK_WIDGET_REALIZED (widget))
327 gtk_widget_realize (widget);
329 if (GTK_WIDGET_MAPPED (widget->parent) &&
330 !GTK_WIDGET_MAPPED (widget))
331 gtk_widget_map (widget);
334 paned->child1 = widget;
335 paned->child1_resize = resize;
336 paned->child1_shrink = shrink;
338 if (GTK_WIDGET_VISIBLE (widget) && GTK_WIDGET_VISIBLE (paned))
339 gtk_widget_queue_resize (widget);
344 gtk_paned_pack2 (GtkPaned *paned,
349 g_return_if_fail (paned != NULL);
350 g_return_if_fail (GTK_IS_PANED (paned));
351 g_return_if_fail (widget != NULL);
355 gtk_widget_set_parent (widget, GTK_WIDGET (paned));
357 if (GTK_WIDGET_VISIBLE (widget->parent))
359 if (GTK_WIDGET_REALIZED (widget->parent) &&
360 !GTK_WIDGET_REALIZED (widget))
361 gtk_widget_realize (widget);
363 if (GTK_WIDGET_MAPPED (widget->parent) &&
364 !GTK_WIDGET_MAPPED (widget))
365 gtk_widget_map (widget);
368 paned->child2 = widget;
369 paned->child2_resize = resize;
370 paned->child2_shrink = shrink;
372 if (GTK_WIDGET_VISIBLE (widget) && GTK_WIDGET_VISIBLE (paned))
373 gtk_widget_queue_resize (widget);
378 gtk_paned_add (GtkContainer *container,
383 g_return_if_fail (container != NULL);
384 g_return_if_fail (GTK_IS_PANED (container));
385 g_return_if_fail (widget != NULL);
387 paned = GTK_PANED (container);
390 gtk_paned_add1 (GTK_PANED (container),widget);
391 else if (!paned->child2)
392 gtk_paned_add2 (GTK_PANED (container),widget);
396 gtk_paned_remove (GtkContainer *container,
400 gboolean was_visible;
402 g_return_if_fail (container != NULL);
403 g_return_if_fail (GTK_IS_PANED (container));
404 g_return_if_fail (widget != NULL);
406 paned = GTK_PANED (container);
407 was_visible = GTK_WIDGET_VISIBLE (widget);
409 if (paned->child1 == widget)
411 gtk_widget_unparent (widget);
413 paned->child1 = NULL;
415 if (was_visible && GTK_WIDGET_VISIBLE (container))
416 gtk_widget_queue_resize (GTK_WIDGET (container));
418 else if (paned->child2 == widget)
420 gtk_widget_unparent (widget);
422 paned->child2 = NULL;
424 if (was_visible && GTK_WIDGET_VISIBLE (container))
425 gtk_widget_queue_resize (GTK_WIDGET (container));
430 gtk_paned_forall (GtkContainer *container,
431 gboolean include_internals,
432 GtkCallback callback,
433 gpointer callback_data)
437 g_return_if_fail (container != NULL);
438 g_return_if_fail (GTK_IS_PANED (container));
439 g_return_if_fail (callback != NULL);
441 paned = GTK_PANED (container);
444 (* callback) (paned->child1, callback_data);
446 (* callback) (paned->child2, callback_data);
450 gtk_paned_set_position (GtkPaned *paned,
453 g_return_if_fail (paned != NULL);
454 g_return_if_fail (GTK_IS_PANED (paned));
458 paned->child1_size = CLAMP (position,
460 paned->max_position);
461 paned->position_set = TRUE;
464 paned->position_set = FALSE;
466 gtk_widget_queue_resize (GTK_WIDGET (paned));
470 gtk_paned_set_handle_size (GtkPaned *paned,
475 g_return_if_fail (paned != NULL);
476 g_return_if_fail (GTK_IS_PANED (paned));
480 gdk_window_get_geometry (paned->handle, &x, &y, NULL, NULL, NULL);
481 gdk_window_move_resize (paned->handle,
482 x + paned->handle_size / 2 - size / 2,
483 y + paned->handle_size / 2 - size / 2,
486 paned->handle_size = size;
490 gtk_paned_set_gutter_size (GtkPaned *paned,
493 g_return_if_fail (paned != NULL);
494 g_return_if_fail (GTK_IS_PANED (paned));
496 paned->gutter_size = size;
498 if (GTK_WIDGET_VISIBLE (GTK_WIDGET (paned)))
499 gtk_widget_queue_resize (GTK_WIDGET (paned));
503 gtk_paned_compute_position (GtkPaned *paned,
508 g_return_if_fail (paned != NULL);
509 g_return_if_fail (GTK_IS_PANED (paned));
511 paned->min_position = paned->child1_shrink ? 0 : child1_req;
513 paned->max_position = allocation;
514 if (!paned->child2_shrink)
515 paned->max_position -= child2_req;
517 if (!paned->position_set)
519 if (paned->child1_resize && !paned->child2_resize)
520 paned->child1_size = allocation - child2_req;
521 else if (!paned->child1_resize && paned->child2_resize)
522 paned->child1_size = child1_req;
524 paned->child1_size = allocation * ((gdouble)child1_req / (child1_req + child2_req));
528 if (paned->last_allocation < 0)
529 paned->last_allocation = allocation;
531 if (paned->child1_resize && !paned->child2_resize)
532 paned->child1_size += (allocation - paned->last_allocation);
533 else if (!(!paned->child1_resize && paned->child2_resize))
534 paned->child1_size = allocation * ((gdouble)paned->child1_size / (paned->last_allocation));
537 paned->child1_size = CLAMP (paned->child1_size,
539 paned->max_position);
541 paned->last_allocation = allocation;