1 /* ///////////////////////////////////////////////////////////////////////////
3 // Purpose: Native GTK+ widget for wxWindows, based on GtkLayout and
4 // GtkFixed. It makes use of the gravity window property and
5 // therefore does not work with GTK 1.0.
6 // Author: Robert Roebling
8 // Copyright: (c) 1998 Robert Roebling
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////// */
13 #define XCheckIfEvent XCHECKIFEVENT
17 #include "wx/gtk/win_gtk.h"
18 #include "gtk/gtksignal.h"
19 #include "gtk/gtkprivate.h"
24 #endif /* __cplusplus */
27 #include <X11/Xutil.h>
28 #include <X11/Xatom.h>
30 #define IS_ONSCREEN(x,y) ((x >= G_MINSHORT) && (x <= G_MAXSHORT) && \
31 (y >= G_MINSHORT) && (y <= G_MAXSHORT))
33 typedef struct _GtkPizzaAdjData GtkPizzaAdjData
;
35 struct _GtkPizzaAdjData
41 static void gtk_pizza_class_init (GtkPizzaClass
*klass
);
42 static void gtk_pizza_init (GtkPizza
*pizza
);
44 static void gtk_pizza_realize (GtkWidget
*widget
);
45 static void gtk_pizza_unrealize (GtkWidget
*widget
);
47 static void gtk_pizza_map (GtkWidget
*widget
);
49 static void gtk_pizza_size_request (GtkWidget
*widget
,
50 GtkRequisition
*requisition
);
51 static void gtk_pizza_size_allocate (GtkWidget
*widget
,
52 GtkAllocation
*allocation
);
53 static void gtk_pizza_draw (GtkWidget
*widget
,
55 static gint
gtk_pizza_expose (GtkWidget
*widget
,
56 GdkEventExpose
*event
);
57 static void gtk_pizza_add (GtkContainer
*container
,
59 static void gtk_pizza_remove (GtkContainer
*container
,
61 static void gtk_pizza_forall (GtkContainer
*container
,
62 gboolean include_internals
,
64 gpointer callback_data
);
66 static void gtk_pizza_position_child (GtkPizza
*pizza
,
67 GtkPizzaChild
*child
);
68 static void gtk_pizza_allocate_child (GtkPizza
*pizza
,
69 GtkPizzaChild
*child
);
70 static void gtk_pizza_position_children (GtkPizza
*pizza
);
72 static void gtk_pizza_adjust_allocations_recurse (GtkWidget
*widget
,
74 static void gtk_pizza_adjust_allocations (GtkPizza
*pizza
,
81 static void gtk_pizza_expose_area (GtkPizza
*pizza
,
86 static void gtk_pizza_adjustment_changed (GtkAdjustment
*adjustment
,
90 static GdkFilterReturn
gtk_pizza_filter (GdkXEvent
*gdk_xevent
,
93 static GdkFilterReturn
gtk_pizza_main_filter (GdkXEvent
*gdk_xevent
,
98 static GtkType
gtk_pizza_child_type (GtkContainer
*container
);
100 static void gtk_pizza_scroll_set_adjustments (GtkPizza
*pizza
,
102 GtkAdjustment
*vadj
);
105 static GtkContainerClass
*parent_class
= NULL
;
106 static gboolean gravity_works
;
109 gtk_pizza_get_type ()
111 static guint pizza_type
= 0;
115 GtkTypeInfo pizza_info
=
119 sizeof (GtkPizzaClass
),
120 (GtkClassInitFunc
) gtk_pizza_class_init
,
121 (GtkObjectInitFunc
) gtk_pizza_init
,
122 /* reserved_1 */ NULL
,
123 /* reserved_2 */ NULL
,
124 (GtkClassInitFunc
) NULL
,
126 pizza_type
= gtk_type_unique (gtk_container_get_type (), &pizza_info
);
133 gtk_pizza_class_init (GtkPizzaClass
*klass
)
135 GtkObjectClass
*object_class
;
136 GtkWidgetClass
*widget_class
;
137 GtkContainerClass
*container_class
;
139 object_class
= (GtkObjectClass
*) klass
;
140 widget_class
= (GtkWidgetClass
*) klass
;
141 container_class
= (GtkContainerClass
*) klass
;
142 parent_class
= gtk_type_class (GTK_TYPE_CONTAINER
);
144 widget_class
->map
= gtk_pizza_map
;
145 widget_class
->realize
= gtk_pizza_realize
;
146 widget_class
->unrealize
= gtk_pizza_unrealize
;
147 widget_class
->size_request
= gtk_pizza_size_request
;
148 widget_class
->size_allocate
= gtk_pizza_size_allocate
;
149 widget_class
->draw
= gtk_pizza_draw
;
150 widget_class
->expose_event
= gtk_pizza_expose
;
152 container_class
->add
= gtk_pizza_add
;
153 container_class
->remove
= gtk_pizza_remove
;
154 container_class
->forall
= gtk_pizza_forall
;
156 container_class
->child_type
= gtk_pizza_child_type
;
158 klass
->set_scroll_adjustments
= gtk_pizza_scroll_set_adjustments
;
160 widget_class
->set_scroll_adjustments_signal
=
161 gtk_signal_new ("set_scroll_adjustments",
165 GTK_CLASS_TYPE(object_class
),
169 GTK_SIGNAL_OFFSET (GtkPizzaClass
, set_scroll_adjustments
),
170 gtk_marshal_NONE__POINTER_POINTER
,
171 GTK_TYPE_NONE
, 2, GTK_TYPE_ADJUSTMENT
, GTK_TYPE_ADJUSTMENT
);
175 gtk_pizza_child_type (GtkContainer
*container
)
177 return GTK_TYPE_WIDGET
;
181 gtk_pizza_init (GtkPizza
*pizza
)
183 GTK_WIDGET_UNSET_FLAGS (pizza
, GTK_NO_WINDOW
);
185 pizza
->shadow_type
= GTK_MYSHADOW_NONE
;
187 pizza
->children
= NULL
;
192 pizza
->bin_window
= NULL
;
197 pizza
->configure_serial
= 0;
200 pizza
->visibility
= GDK_VISIBILITY_PARTIAL
;
202 pizza
->clear_on_draw
= TRUE
;
203 pizza
->use_filter
= TRUE
;
204 pizza
->external_expose
= FALSE
;
212 pizza
= gtk_type_new (gtk_pizza_get_type ());
214 return GTK_WIDGET (pizza
);
218 gtk_pizza_scroll_set_adjustments (GtkPizza
*pizza
,
222 /* We handle scrolling in the wxScrolledWindow, not here. */
226 gtk_pizza_set_shadow_type (GtkPizza
*pizza
,
227 GtkMyShadowType type
)
229 g_return_if_fail (pizza
!= NULL
);
230 g_return_if_fail (GTK_IS_PIZZA (pizza
));
232 if ((GtkMyShadowType
) pizza
->shadow_type
!= type
)
234 pizza
->shadow_type
= type
;
236 if (GTK_WIDGET_VISIBLE (pizza
))
238 gtk_widget_size_allocate (GTK_WIDGET (pizza
), &(GTK_WIDGET (pizza
)->allocation
));
239 gtk_widget_queue_draw (GTK_WIDGET (pizza
));
245 gtk_pizza_set_clear (GtkPizza
*pizza
,
248 g_return_if_fail (pizza
!= NULL
);
249 g_return_if_fail (GTK_IS_PIZZA (pizza
));
251 pizza
->clear_on_draw
= clear
;
255 gtk_pizza_set_filter (GtkPizza
*pizza
,
258 g_return_if_fail (pizza
!= NULL
);
259 g_return_if_fail (GTK_IS_PIZZA (pizza
));
261 pizza
->use_filter
= use
;
265 gtk_pizza_set_external (GtkPizza
*pizza
,
268 g_return_if_fail (pizza
!= NULL
);
269 g_return_if_fail (GTK_IS_PIZZA (pizza
));
271 pizza
->external_expose
= expose
;
275 gtk_pizza_put (GtkPizza
*pizza
,
282 GtkPizzaChild
*child_info
;
284 g_return_if_fail (pizza
!= NULL
);
285 g_return_if_fail (GTK_IS_PIZZA (pizza
));
286 g_return_if_fail (widget
!= NULL
);
288 child_info
= g_new (GtkPizzaChild
, 1);
290 child_info
->widget
= widget
;
293 child_info
->width
= width
;
294 child_info
->height
= height
;
296 pizza
->children
= g_list_append (pizza
->children
, child_info
);
298 gtk_widget_set_parent (widget
, GTK_WIDGET (pizza
));
300 if (GTK_WIDGET_REALIZED (pizza
))
301 gtk_widget_set_parent_window (widget
, pizza
->bin_window
);
303 #ifndef __WXGTK20__ /* FIXME? */
304 if (!IS_ONSCREEN (x
, y
))
305 GTK_PRIVATE_SET_FLAG (widget
, GTK_IS_OFFSCREEN
);
309 if (GTK_WIDGET_REALIZED (pizza))
310 gtk_widget_realize (widget);
313 gtk_widget_set_usize (widget
, width
, height
);
316 if (GTK_WIDGET_VISIBLE (pizza) && GTK_WIDGET_VISIBLE (widget))
318 if (GTK_WIDGET_MAPPED (pizza))
319 gtk_widget_map (widget);
321 gtk_widget_queue_resize (widget);
327 gtk_pizza_move (GtkPizza
*pizza
,
332 GtkPizzaChild
*child
;
335 g_return_if_fail (pizza
!= NULL
);
336 g_return_if_fail (GTK_IS_PIZZA (pizza
));
337 g_return_if_fail (widget
!= NULL
);
339 children
= pizza
->children
;
342 child
= children
->data
;
343 children
= children
->next
;
345 if (child
->widget
== widget
)
347 if ((child
->x
== x
) && (child
->y
== y
))
353 if (GTK_WIDGET_VISIBLE (widget
) && GTK_WIDGET_VISIBLE (pizza
))
354 gtk_widget_queue_resize (widget
);
361 gtk_pizza_resize (GtkPizza
*pizza
,
366 GtkPizzaChild
*child
;
369 g_return_if_fail (pizza
!= NULL
);
370 g_return_if_fail (GTK_IS_PIZZA (pizza
));
371 g_return_if_fail (widget
!= NULL
);
373 children
= pizza
->children
;
376 child
= children
->data
;
377 children
= children
->next
;
379 if (child
->widget
== widget
)
381 if ((child
->width
== width
) && (child
->height
== height
))
384 child
->width
= width
;
385 child
->height
= height
;
387 gtk_widget_set_usize (widget
, width
, height
);
389 if (GTK_WIDGET_VISIBLE (widget
) && GTK_WIDGET_VISIBLE (pizza
))
390 gtk_widget_queue_resize (widget
);
397 gtk_pizza_set_size (GtkPizza
*pizza
,
404 GtkPizzaChild
*child
;
407 g_return_if_fail (pizza
!= NULL
);
408 g_return_if_fail (GTK_IS_PIZZA (pizza
));
409 g_return_if_fail (widget
!= NULL
);
411 children
= pizza
->children
;
414 child
= children
->data
;
415 children
= children
->next
;
417 if (child
->widget
== widget
)
419 if ((child
->x
== x
) &&
421 (child
->width
== width
) &&
422 (child
->height
== height
)) return;
426 child
->width
= width
;
427 child
->height
= height
;
429 gtk_widget_set_usize (widget
, width
, height
);
431 if (GTK_WIDGET_VISIBLE (widget
) && GTK_WIDGET_VISIBLE (pizza
))
432 gtk_widget_queue_resize (widget
);
440 gtk_pizza_child_resized (GtkPizza
*pizza
,
443 GtkPizzaChild
*child
;
446 g_return_val_if_fail (pizza
!= NULL
, FALSE
);
447 g_return_val_if_fail (GTK_IS_PIZZA (pizza
), FALSE
);
448 g_return_val_if_fail (widget
!= NULL
, FALSE
);
450 children
= pizza
->children
;
453 child
= children
->data
;
454 children
= children
->next
;
456 if (child
->widget
== widget
)
458 return ((child
->width
== widget
->allocation
.width
) &&
459 (child
->height
== widget
->allocation
.height
));
467 gtk_pizza_map (GtkWidget
*widget
)
470 GtkPizzaChild
*child
;
473 g_return_if_fail (widget
!= NULL
);
474 g_return_if_fail (GTK_IS_PIZZA (widget
));
476 GTK_WIDGET_SET_FLAGS (widget
, GTK_MAPPED
);
477 pizza
= GTK_PIZZA (widget
);
479 children
= pizza
->children
;
482 child
= children
->data
;
483 children
= children
->next
;
485 if ( GTK_WIDGET_VISIBLE (child
->widget
) &&
486 !GTK_WIDGET_MAPPED (child
->widget
) &&
490 !GTK_WIDGET_IS_OFFSCREEN (child
->widget
))
493 gtk_widget_map (child
->widget
);
497 gdk_window_show (widget
->window
);
498 gdk_window_show (pizza
->bin_window
);
502 gtk_pizza_realize (GtkWidget
*widget
)
505 GdkWindowAttr attributes
;
506 gint attributes_mask
;
507 GtkPizzaChild
*child
;
510 g_return_if_fail (widget
!= NULL
);
511 g_return_if_fail (GTK_IS_PIZZA (widget
));
513 pizza
= GTK_PIZZA (widget
);
515 GTK_WIDGET_SET_FLAGS (widget
, GTK_REALIZED
);
517 attributes
.window_type
= GDK_WINDOW_CHILD
;
519 attributes
.x
= widget
->allocation
.x
;
520 attributes
.y
= widget
->allocation
.y
;
521 attributes
.width
= widget
->allocation
.width
;
522 attributes
.height
= widget
->allocation
.height
;
524 if (pizza
->shadow_type
== GTK_MYSHADOW_NONE
)
526 /* no border, no changes to sizes */
528 if (pizza
->shadow_type
== GTK_MYSHADOW_THIN
)
530 /* GTK_MYSHADOW_THIN == wxSIMPLE_BORDER */
533 attributes
.width
-= 2;
534 attributes
.height
-= 2;
537 /* GTK_MYSHADOW_IN == wxSUNKEN_BORDER */
538 /* GTK_MYSHADOW_OUT == wxRAISED_BORDER */
541 attributes
.width
-= 4;
542 attributes
.height
-= 4;
546 if (attributes
.width
< 2) attributes
.width
= 2;
547 if (attributes
.height
< 2) attributes
.height
= 2;
549 attributes
.wclass
= GDK_INPUT_OUTPUT
;
550 attributes
.visual
= gtk_widget_get_visual (widget
);
551 attributes
.colormap
= gtk_widget_get_colormap (widget
);
552 attributes
.event_mask
= GDK_VISIBILITY_NOTIFY_MASK
;
553 attributes_mask
= GDK_WA_X
| GDK_WA_Y
| GDK_WA_VISUAL
| GDK_WA_COLORMAP
;
555 widget
->window
= gdk_window_new(gtk_widget_get_parent_window (widget
),
556 &attributes
, attributes_mask
);
557 gdk_window_set_user_data (widget
->window
, widget
);
562 attributes
.event_mask
= gtk_widget_get_events (widget
);
563 attributes
.event_mask
|= GDK_EXPOSURE_MASK
|
564 GDK_POINTER_MOTION_MASK
|
565 GDK_POINTER_MOTION_HINT_MASK
|
566 GDK_BUTTON_MOTION_MASK
|
567 GDK_BUTTON1_MOTION_MASK
|
568 GDK_BUTTON2_MOTION_MASK
|
569 GDK_BUTTON3_MOTION_MASK
|
570 GDK_BUTTON_PRESS_MASK
|
571 GDK_BUTTON_RELEASE_MASK
|
573 GDK_KEY_RELEASE_MASK
|
574 GDK_ENTER_NOTIFY_MASK
|
575 GDK_LEAVE_NOTIFY_MASK
|
576 GDK_FOCUS_CHANGE_MASK
;
578 pizza
->bin_window
= gdk_window_new(widget
->window
,
579 &attributes
, attributes_mask
);
580 gdk_window_set_user_data (pizza
->bin_window
, widget
);
582 widget
->style
= gtk_style_attach (widget
->style
, widget
->window
);
583 gtk_style_set_background (widget
->style
, widget
->window
, GTK_STATE_NORMAL
);
584 gtk_style_set_background (widget
->style
, pizza
->bin_window
, GTK_STATE_NORMAL
);
587 gdk_window_set_back_pixmap( widget->window, NULL, FALSE );
588 gdk_window_set_back_pixmap( pizza->bin_window, NULL, FALSE );
591 /* add filters for intercepting visibility and expose events */
592 gdk_window_add_filter (widget
->window
, gtk_pizza_main_filter
, pizza
);
593 gdk_window_add_filter (pizza
->bin_window
, gtk_pizza_filter
, pizza
);
595 /* we NEED gravity or we'll give up */
596 gravity_works
= gdk_window_set_static_gravities (pizza
->bin_window
, TRUE
);
598 /* cannot be done before realisation */
599 children
= pizza
->children
;
602 child
= children
->data
;
603 children
= children
->next
;
605 gtk_widget_set_parent_window (child
->widget
, pizza
->bin_window
);
610 gtk_pizza_unrealize (GtkWidget
*widget
)
614 g_return_if_fail (widget
!= NULL
);
615 g_return_if_fail (GTK_IS_PIZZA (widget
));
617 pizza
= GTK_PIZZA (widget
);
619 gdk_window_set_user_data (pizza
->bin_window
, NULL
);
620 gdk_window_destroy (pizza
->bin_window
);
621 pizza
->bin_window
= NULL
;
623 if (GTK_WIDGET_CLASS (parent_class
)->unrealize
)
624 (* GTK_WIDGET_CLASS (parent_class
)->unrealize
) (widget
);
628 gtk_pizza_size_request (GtkWidget
*widget
,
629 GtkRequisition
*requisition
)
632 GtkPizzaChild
*child
;
634 GtkRequisition child_requisition
;
636 g_return_if_fail (widget
!= NULL
);
637 g_return_if_fail (GTK_IS_PIZZA (widget
));
638 g_return_if_fail (requisition
!= NULL
);
640 pizza
= GTK_PIZZA (widget
);
642 children
= pizza
->children
;
645 child
= children
->data
;
646 children
= children
->next
;
648 if (GTK_WIDGET_VISIBLE (child
->widget
))
650 gtk_widget_size_request (child
->widget
, &child_requisition
);
654 /* request very little, I'm not sure if requesting nothing
655 will always have positive effects on stability... */
656 requisition
->width
= 2;
657 requisition
->height
= 2;
661 gtk_pizza_size_allocate (GtkWidget
*widget
,
662 GtkAllocation
*allocation
)
667 GtkPizzaChild
*child
;
670 g_return_if_fail (widget
!= NULL
);
671 g_return_if_fail (GTK_IS_PIZZA(widget
));
672 g_return_if_fail (allocation
!= NULL
);
674 pizza
= GTK_PIZZA (widget
);
676 widget
->allocation
= *allocation
;
678 if (pizza
->shadow_type
== GTK_MYSHADOW_NONE
)
681 if (pizza
->shadow_type
== GTK_MYSHADOW_THIN
)
686 x
= allocation
->x
+ border
;
687 y
= allocation
->y
+ border
;
688 w
= allocation
->width
- border
*2;
689 h
= allocation
->height
- border
*2;
691 if (GTK_WIDGET_REALIZED (widget
))
693 gdk_window_move_resize( widget
->window
, x
, y
, w
, h
);
694 gdk_window_move_resize( pizza
->bin_window
, 0, 0, w
, h
);
697 children
= pizza
->children
;
700 child
= children
->data
;
701 children
= children
->next
;
703 gtk_pizza_position_child (pizza
, child
);
704 gtk_pizza_allocate_child (pizza
, child
);
709 gtk_pizza_draw (GtkWidget
*widget
,
713 GtkPizzaChild
*child
;
714 GdkRectangle child_area
;
717 g_return_if_fail (widget
!= NULL
);
718 g_return_if_fail (GTK_IS_PIZZA (widget
));
720 pizza
= GTK_PIZZA (widget
);
722 /* Sometimes, We handle all expose events in window.cpp now. */
723 if (pizza
->external_expose
)
726 children
= pizza
->children
;
727 if ( !(GTK_WIDGET_APP_PAINTABLE (widget
)) &&
728 (pizza
->clear_on_draw
))
730 gdk_window_clear_area( pizza
->bin_window
,
731 area
->x
, area
->y
, area
->width
, area
->height
);
736 child
= children
->data
;
737 children
= children
->next
;
739 if (gtk_widget_intersect (child
->widget
, area
, &child_area
))
740 gtk_widget_draw (child
->widget
, &child_area
);
745 gtk_pizza_expose (GtkWidget
*widget
,
746 GdkEventExpose
*event
)
749 GtkPizzaChild
*child
;
750 GdkEventExpose child_event
;
753 g_return_val_if_fail (widget
!= NULL
, FALSE
);
754 g_return_val_if_fail (GTK_IS_PIZZA (widget
), FALSE
);
755 g_return_val_if_fail (event
!= NULL
, FALSE
);
757 pizza
= GTK_PIZZA (widget
);
759 /* Sometimes, We handle all expose events in window.cpp now. */
760 if (pizza
->external_expose
)
763 if (event
->window
!= pizza
->bin_window
)
766 children
= pizza
->children
;
769 child
= children
->data
;
770 children
= children
->next
;
772 child_event
= *event
;
774 if (GTK_WIDGET_NO_WINDOW (child
->widget
) &&
775 GTK_WIDGET_DRAWABLE (child
->widget
) &&
776 gtk_widget_intersect (child
->widget
, &event
->area
, &child_event
.area
))
778 gtk_widget_event (child
->widget
, (GdkEvent
*) &child_event
);
786 gtk_pizza_add (GtkContainer
*container
,
789 g_return_if_fail (container
!= NULL
);
790 g_return_if_fail (GTK_IS_PIZZA (container
));
791 g_return_if_fail (widget
!= NULL
);
793 gtk_pizza_put (GTK_PIZZA (container
), widget
, 0, 0, 20, 20 );
797 gtk_pizza_remove (GtkContainer
*container
,
801 GtkPizzaChild
*child
;
804 g_return_if_fail (container
!= NULL
);
805 g_return_if_fail (GTK_IS_PIZZA (container
));
806 g_return_if_fail (widget
!= NULL
);
808 pizza
= GTK_PIZZA (container
);
810 children
= pizza
->children
;
813 child
= children
->data
;
815 if (child
->widget
== widget
)
817 gtk_widget_unparent (widget
);
819 /* security checks */
820 g_return_if_fail (GTK_IS_WIDGET (widget
));
822 pizza
->children
= g_list_remove_link (pizza
->children
, children
);
823 g_list_free (children
);
826 /* security checks */
827 g_return_if_fail (GTK_IS_WIDGET (widget
));
830 GTK_PRIVATE_UNSET_FLAG (widget
, GTK_IS_OFFSCREEN
);
836 children
= children
->next
;
841 gtk_pizza_forall (GtkContainer
*container
,
842 gboolean include_internals
,
843 GtkCallback callback
,
844 gpointer callback_data
)
847 GtkPizzaChild
*child
;
850 g_return_if_fail (container
!= NULL
);
851 g_return_if_fail (GTK_IS_PIZZA (container
));
852 g_return_if_fail (callback
!= NULL
);
854 pizza
= GTK_PIZZA (container
);
856 children
= pizza
->children
;
859 child
= children
->data
;
860 children
= children
->next
;
862 (* callback
) (child
->widget
, callback_data
);
867 /* Operations on children
871 gtk_pizza_position_child (GtkPizza
*pizza
,
872 GtkPizzaChild
*child
)
877 x
= child
->x
- pizza
->xoffset
;
878 y
= child
->y
- pizza
->yoffset
;
880 if (IS_ONSCREEN (x
,y
))
882 if (GTK_WIDGET_MAPPED (pizza
) &&
883 GTK_WIDGET_VISIBLE (child
->widget
))
885 if (!GTK_WIDGET_MAPPED (child
->widget
))
886 gtk_widget_map (child
->widget
);
890 if (GTK_WIDGET_IS_OFFSCREEN (child
->widget
))
891 GTK_PRIVATE_UNSET_FLAG (child
->widget
, GTK_IS_OFFSCREEN
);
897 if (!GTK_WIDGET_IS_OFFSCREEN (child
->widget
))
898 GTK_PRIVATE_SET_FLAG (child
->widget
, GTK_IS_OFFSCREEN
);
901 if (GTK_WIDGET_MAPPED (child
->widget
))
902 gtk_widget_unmap (child
->widget
);
907 gtk_pizza_allocate_child (GtkPizza
*pizza
,
908 GtkPizzaChild
*child
)
910 GtkAllocation allocation
;
911 GtkRequisition requisition
;
913 allocation
.x
= child
->x
- pizza
->xoffset
;
914 allocation
.y
= child
->y
- pizza
->yoffset
;
915 gtk_widget_get_child_requisition (child
->widget
, &requisition
);
916 allocation
.width
= requisition
.width
;
917 allocation
.height
= requisition
.height
;
919 gtk_widget_size_allocate (child
->widget
, &allocation
);
923 gtk_pizza_position_children (GtkPizza
*pizza
)
927 tmp_list
= pizza
->children
;
930 GtkPizzaChild
*child
= tmp_list
->data
;
931 tmp_list
= tmp_list
->next
;
933 gtk_pizza_position_child (pizza
, child
);
938 gtk_pizza_adjust_allocations_recurse (GtkWidget
*widget
,
941 GtkPizzaAdjData
*data
= cb_data
;
943 widget
->allocation
.x
+= data
->dx
;
944 widget
->allocation
.y
+= data
->dy
;
946 if (GTK_WIDGET_NO_WINDOW (widget
) && GTK_IS_CONTAINER (widget
))
948 gtk_container_forall (GTK_CONTAINER (widget
),
949 gtk_pizza_adjust_allocations_recurse
,
955 gtk_pizza_adjust_allocations (GtkPizza
*pizza
,
960 GtkPizzaAdjData data
;
965 tmp_list
= pizza
->children
;
968 GtkPizzaChild
*child
= tmp_list
->data
;
969 tmp_list
= tmp_list
->next
;
971 child
->widget
->allocation
.x
+= dx
;
972 child
->widget
->allocation
.y
+= dy
;
974 if (GTK_WIDGET_NO_WINDOW (child
->widget
) &&
975 GTK_IS_CONTAINER (child
->widget
))
977 gtk_container_forall (GTK_CONTAINER (child
->widget
),
978 gtk_pizza_adjust_allocations_recurse
,
988 /* Send a synthetic expose event to the widget
991 gtk_pizza_expose_area (GtkPizza
*pizza
,
992 gint x
, gint y
, gint width
, gint height
)
994 if (pizza
->visibility
== GDK_VISIBILITY_UNOBSCURED
)
996 GdkEventExpose event
;
998 event
.type
= GDK_EXPOSE
;
999 event
.send_event
= TRUE
;
1000 event
.window
= pizza
->bin_window
;
1005 event
.area
.width
= width
;
1006 event
.area
.height
= height
;
1008 gdk_window_ref (event
.window
);
1009 gtk_widget_event (GTK_WIDGET (pizza
), (GdkEvent
*)&event
);
1010 gdk_window_unref (event
.window
);
1015 /* This function is used to find events to process while scrolling
1019 gtk_pizza_expose_predicate (Display
*display
,
1023 if ((xevent
->type
== Expose
) ||
1024 ((xevent
->xany
.window
== *(Window
*)arg
) &&
1025 (xevent
->type
== ConfigureNotify
)))
1031 /* This is the main routine to do the scrolling. Scrolling is
1032 * done by "Guffaw" scrolling, as in the Mozilla XFE, with
1033 * a few modifications.
1035 * The main improvement is that we keep track of whether we
1036 * are obscured or not. If not, we ignore the generated expose
1037 * events and instead do the exposes ourself, without having
1038 * to wait for a roundtrip to the server. This also provides
1039 * a limited form of expose-event compression, since we do
1040 * the affected area as one big chunk.
1044 gtk_pizza_scroll (GtkPizza
*pizza
, gint dx
, gint dy
)
1049 gint x
,y
,w
,h
,border
;
1051 widget
= GTK_WIDGET (pizza
);
1053 pizza
->xoffset
+= dx
;
1054 pizza
->yoffset
+= dy
;
1056 if (!GTK_WIDGET_MAPPED (pizza
))
1058 gtk_pizza_position_children (pizza
);
1062 gtk_pizza_adjust_allocations (pizza
, -dx
, -dy
);
1064 if (pizza
->shadow_type
== GTK_MYSHADOW_NONE
)
1067 if (pizza
->shadow_type
== GTK_MYSHADOW_THIN
)
1074 w
= widget
->allocation
.width
- 2*border
;
1075 h
= widget
->allocation
.height
- 2*border
;
1081 gdk_window_resize (pizza
->bin_window
,
1084 gdk_window_move (pizza
->bin_window
, x
-dx
, y
);
1085 gdk_window_move_resize (pizza
->bin_window
, x
, y
, w
, h
);
1096 gdk_window_move_resize (pizza
->bin_window
,
1101 gdk_window_move (pizza
->bin_window
, x
, y
);
1102 gdk_window_resize (pizza
->bin_window
, w
, h
);
1114 gdk_window_resize (pizza
->bin_window
, w
, h
+ dy
);
1115 gdk_window_move (pizza
->bin_window
, x
, y
-dy
);
1116 gdk_window_move_resize (pizza
->bin_window
,
1128 gdk_window_move_resize (pizza
->bin_window
,
1129 x
, y
+dy
, w
, h
- dy
);
1130 gdk_window_move (pizza
->bin_window
, x
, y
);
1131 gdk_window_resize (pizza
->bin_window
, w
, h
);
1139 gtk_pizza_position_children (pizza
);
1142 while (XCheckIfEvent(GDK_WINDOW_XDISPLAY (pizza
->bin_window
),
1144 gtk_pizza_expose_predicate
,
1145 (XPointer
)&GDK_WINDOW_XWINDOW (pizza
->bin_window
)))
1148 GtkWidget
*event_widget
;
1150 if ((xevent
.xany
.window
== GDK_WINDOW_XWINDOW (pizza
->bin_window
)) )
1151 gtk_pizza_filter (&xevent
, &event
, pizza
);
1153 if (xevent
.type
== Expose
)
1155 event
.expose
.window
= gdk_window_lookup (xevent
.xany
.window
);
1156 gdk_window_get_user_data (event
.expose
.window
,
1157 (gpointer
*)&event_widget
);
1161 event
.expose
.type
= GDK_EXPOSE
;
1162 event
.expose
.area
.x
= xevent
.xexpose
.x
;
1163 event
.expose
.area
.y
= xevent
.xexpose
.y
;
1164 event
.expose
.area
.width
= xevent
.xexpose
.width
;
1165 event
.expose
.area
.height
= xevent
.xexpose
.height
;
1166 event
.expose
.count
= xevent
.xexpose
.count
;
1168 gdk_window_ref (event
.expose
.window
);
1169 gtk_widget_event (event_widget
, &event
);
1170 gdk_window_unref (event
.expose
.window
);
1176 /* The main event filter. Actually, we probably don't really need
1177 * to install this as a filter at all, since we are calling it
1178 * directly above in the expose-handling hack. But in case scrollbars
1179 * are fixed up in some manner...
1181 * This routine identifies expose events that are generated when
1182 * we've temporarily moved the bin_window_origin, and translates
1183 * them or discards them, depending on whether we are obscured
1186 static GdkFilterReturn
1187 gtk_pizza_filter (GdkXEvent
*gdk_xevent
,
1194 xevent
= (XEvent
*)gdk_xevent
;
1196 pizza
= GTK_PIZZA (data
);
1198 if (!pizza
->use_filter
)
1199 return GDK_FILTER_CONTINUE
;
1201 switch (xevent
->type
)
1204 if (xevent
->xexpose
.serial
== pizza
->configure_serial
)
1206 xevent
->xexpose
.x
+= pizza
->scroll_x
;
1207 xevent
->xexpose
.y
+= pizza
->scroll_y
;
1211 case ConfigureNotify
:
1213 pizza
->configure_serial
= xevent
->xconfigure
.serial
;
1214 pizza
->scroll_x
= xevent
->xconfigure
.x
;
1215 pizza
->scroll_y
= xevent
->xconfigure
.y
;
1220 return GDK_FILTER_CONTINUE
;
1223 /* Although GDK does have a GDK_VISIBILITY_NOTIFY event,
1224 * there is no corresponding event in GTK, so we have
1225 * to get the events from a filter
1227 static GdkFilterReturn
1228 gtk_pizza_main_filter (GdkXEvent
*gdk_xevent
,
1235 xevent
= (XEvent
*)gdk_xevent
;
1236 pizza
= GTK_PIZZA (data
);
1238 if (!pizza
->use_filter
)
1239 return GDK_FILTER_CONTINUE
;
1241 if (xevent
->type
== VisibilityNotify
)
1243 switch (xevent
->xvisibility
.state
)
1245 case VisibilityFullyObscured
:
1246 pizza
->visibility
= GDK_VISIBILITY_FULLY_OBSCURED
;
1249 case VisibilityPartiallyObscured
:
1250 pizza
->visibility
= GDK_VISIBILITY_PARTIAL
;
1253 case VisibilityUnobscured
:
1254 pizza
->visibility
= GDK_VISIBILITY_UNOBSCURED
;
1258 return GDK_FILTER_REMOVE
;
1261 return GDK_FILTER_CONTINUE
;
1267 #endif /* __cplusplus */