1 /* ///////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/win_gtk.c
3 // Purpose: Native GTK+ widget for wxWidgets, 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: wxWidgets licence
10 /////////////////////////////////////////////////////////////////////////// */
13 #define XCheckIfEvent XCHECKIFEVENT
16 #include "wx/platform.h"
17 #include "wx/gtk/win_gtk.h"
21 #endif /* __cplusplus */
23 typedef struct _GtkPizzaAdjData GtkPizzaAdjData
;
25 struct _GtkPizzaAdjData
31 static void gtk_pizza_class_init (GtkPizzaClass
*klass
);
32 static void gtk_pizza_init (GtkPizza
*pizza
);
34 static void gtk_pizza_realize (GtkWidget
*widget
);
35 static void gtk_pizza_unrealize (GtkWidget
*widget
);
37 static void gtk_pizza_map (GtkWidget
*widget
);
39 static void gtk_pizza_size_request (GtkWidget
*widget
,
40 GtkRequisition
*requisition
);
41 static void gtk_pizza_size_allocate (GtkWidget
*widget
,
42 GtkAllocation
*allocation
);
43 static gint
gtk_pizza_expose (GtkWidget
*widget
,
44 GdkEventExpose
*event
);
45 static void gtk_pizza_style_set (GtkWidget
*widget
,
46 GtkStyle
*previous_style
);
47 static void gtk_pizza_add (GtkContainer
*container
,
49 static void gtk_pizza_remove (GtkContainer
*container
,
51 static void gtk_pizza_forall (GtkContainer
*container
,
52 gboolean include_internals
,
54 gpointer callback_data
);
56 static void gtk_pizza_allocate_child (GtkPizza
*pizza
,
57 GtkPizzaChild
*child
);
58 static void gtk_pizza_adjust_allocations_recurse (GtkWidget
*widget
,
61 static GtkType
gtk_pizza_child_type (GtkContainer
*container
);
63 static void gtk_pizza_scroll_set_adjustments (GtkPizza
*pizza
,
67 static GtkWidgetClass
* pizza_parent_class
;
72 static GtkType pizza_type
= 0;
76 static const GTypeInfo pizza_info
=
78 sizeof (GtkPizzaClass
),
80 NULL
, /* base_finalize */
81 (GClassInitFunc
) gtk_pizza_class_init
,
82 NULL
, /* class_finalize */
83 NULL
, /* class_data */
86 (GInstanceInitFunc
) gtk_pizza_init
,
89 pizza_type
= g_type_register_static (GTK_TYPE_CONTAINER
, "GtkPizza", &pizza_info
, (GTypeFlags
)0);
95 /* Marshaller needed for set_scroll_adjustments signal,
96 generated with GLib-2.4.6 glib-genmarshal */
97 #define g_marshal_value_peek_object(v) g_value_get_object (v)
99 g_cclosure_user_marshal_VOID__OBJECT_OBJECT (GClosure
*closure
,
100 GValue
*return_value
,
101 guint n_param_values
,
102 const GValue
*param_values
,
103 gpointer invocation_hint
,
104 gpointer marshal_data
)
106 typedef void (*GMarshalFunc_VOID__OBJECT_OBJECT
) (gpointer data1
,
110 register GMarshalFunc_VOID__OBJECT_OBJECT callback
;
111 register GCClosure
*cc
= (GCClosure
*) closure
;
112 register gpointer data1
, data2
;
114 g_return_if_fail (n_param_values
== 3);
116 if (G_CCLOSURE_SWAP_DATA (closure
))
118 data1
= closure
->data
;
119 data2
= g_value_peek_pointer (param_values
+ 0);
123 data1
= g_value_peek_pointer (param_values
+ 0);
124 data2
= closure
->data
;
126 callback
= (GMarshalFunc_VOID__OBJECT_OBJECT
) (marshal_data
? marshal_data
: cc
->callback
);
129 g_marshal_value_peek_object (param_values
+ 1),
130 g_marshal_value_peek_object (param_values
+ 2),
135 gtk_pizza_class_init (GtkPizzaClass
*klass
)
137 GtkObjectClass
*object_class
;
138 GtkWidgetClass
*widget_class
;
139 GtkContainerClass
*container_class
;
141 object_class
= (GtkObjectClass
*) klass
;
142 widget_class
= (GtkWidgetClass
*) klass
;
143 container_class
= (GtkContainerClass
*) klass
;
144 pizza_parent_class
= gtk_type_class (GTK_TYPE_CONTAINER
);
146 widget_class
->map
= gtk_pizza_map
;
147 widget_class
->realize
= gtk_pizza_realize
;
148 widget_class
->unrealize
= gtk_pizza_unrealize
;
149 widget_class
->size_request
= gtk_pizza_size_request
;
150 widget_class
->size_allocate
= gtk_pizza_size_allocate
;
151 widget_class
->expose_event
= gtk_pizza_expose
;
152 widget_class
->style_set
= gtk_pizza_style_set
;
154 container_class
->add
= gtk_pizza_add
;
155 container_class
->remove
= gtk_pizza_remove
;
156 container_class
->forall
= gtk_pizza_forall
;
158 container_class
->child_type
= gtk_pizza_child_type
;
160 klass
->set_scroll_adjustments
= gtk_pizza_scroll_set_adjustments
;
162 widget_class
->set_scroll_adjustments_signal
=
164 "set_scroll_adjustments",
165 G_TYPE_FROM_CLASS(object_class
),
167 G_STRUCT_OFFSET(GtkPizzaClass
, set_scroll_adjustments
),
170 g_cclosure_user_marshal_VOID__OBJECT_OBJECT
,
174 GTK_TYPE_ADJUSTMENT
);
178 gtk_pizza_child_type (GtkContainer
*container
)
180 return GTK_TYPE_WIDGET
;
184 gtk_pizza_init (GtkPizza
*pizza
)
186 GTK_WIDGET_UNSET_FLAGS (pizza
, GTK_NO_WINDOW
);
188 pizza
->shadow_type
= GTK_MYSHADOW_NONE
;
190 pizza
->children
= NULL
;
192 pizza
->bin_window
= NULL
;
194 pizza
->m_xoffset
= 0;
195 pizza
->m_yoffset
= 0;
203 pizza
= g_object_new (gtk_pizza_get_type (), NULL
);
205 return GTK_WIDGET (pizza
);
208 gint
gtk_pizza_get_xoffset (GtkPizza
*pizza
)
210 g_return_val_if_fail ( (pizza
!= NULL
), -1 );
211 g_return_val_if_fail ( (GTK_IS_PIZZA (pizza
)), -1 );
213 return pizza
->m_xoffset
;
216 gint
gtk_pizza_get_yoffset (GtkPizza
*pizza
)
218 g_return_val_if_fail ( (pizza
!= NULL
), -1 );
219 g_return_val_if_fail ( (GTK_IS_PIZZA (pizza
)), -1 );
221 return pizza
->m_yoffset
;
224 void gtk_pizza_set_xoffset (GtkPizza
*pizza
, gint xoffset
)
226 g_return_if_fail (pizza
!= NULL
);
227 g_return_if_fail (GTK_IS_PIZZA (pizza
));
229 pizza
->m_xoffset
= xoffset
;
233 void gtk_pizza_set_yoffset (GtkPizza
*pizza
, gint yoffset
)
235 g_return_if_fail (pizza
!= NULL
);
236 g_return_if_fail (GTK_IS_PIZZA (pizza
));
238 pizza
->m_xoffset
= yoffset
;
242 gint
gtk_pizza_get_rtl_offset (GtkPizza
*pizza
)
246 g_return_val_if_fail ( (pizza
!= NULL
), 0 );
247 g_return_val_if_fail ( (GTK_IS_PIZZA (pizza
)), 0 );
249 if (!pizza
->bin_window
) return 0;
251 gdk_window_get_geometry( pizza
->bin_window
, NULL
, NULL
, &width
, NULL
, NULL
);
258 gtk_pizza_scroll_set_adjustments (GtkPizza
*pizza
,
262 /* We handle scrolling in the wxScrolledWindow, not here. */
266 gtk_pizza_set_shadow_type (GtkPizza
*pizza
,
267 GtkMyShadowType type
)
269 g_return_if_fail (pizza
!= NULL
);
270 g_return_if_fail (GTK_IS_PIZZA (pizza
));
272 if (pizza
->shadow_type
!= type
)
274 pizza
->shadow_type
= type
;
276 if (GTK_WIDGET_VISIBLE (pizza
))
278 gtk_widget_size_allocate (GTK_WIDGET (pizza
), &(GTK_WIDGET (pizza
)->allocation
));
279 gtk_widget_queue_draw (GTK_WIDGET (pizza
));
285 gtk_pizza_put (GtkPizza
*pizza
,
292 GtkPizzaChild
*child_info
;
294 g_return_if_fail (pizza
!= NULL
);
295 g_return_if_fail (GTK_IS_PIZZA (pizza
));
296 g_return_if_fail (widget
!= NULL
);
298 if (gtk_widget_get_direction( GTK_WIDGET(pizza
) ) == GTK_TEXT_DIR_RTL
)
300 /* reverse horizontal placement */
301 x
= GTK_WIDGET(pizza
)->allocation
.width
- x
- width
;
304 child_info
= g_new (GtkPizzaChild
, 1);
306 child_info
->widget
= widget
;
309 child_info
->width
= width
;
310 child_info
->height
= height
;
312 pizza
->children
= g_list_append (pizza
->children
, child_info
);
314 if (GTK_WIDGET_REALIZED (pizza
))
315 gtk_widget_set_parent_window (widget
, pizza
->bin_window
);
317 gtk_widget_set_parent (widget
, GTK_WIDGET (pizza
));
319 gtk_widget_set_size_request (widget
, width
, height
);
323 gtk_pizza_set_size (GtkPizza
*pizza
,
330 GtkPizzaChild
*child
;
333 g_return_if_fail (pizza
!= NULL
);
334 g_return_if_fail (GTK_IS_PIZZA (pizza
));
335 g_return_if_fail (widget
!= NULL
);
337 #ifndef WX_WARN_ILLEGAL_SETSIZE
338 /* this really shouldn't happen -- but it does, a lot, right now and we
339 can't pass negative values to gtk_widget_set_size_request() without getting
340 a warning printed out, so filter them out here */
347 children
= pizza
->children
;
350 child
= children
->data
;
351 children
= children
->next
;
353 if (child
->widget
== widget
)
355 if ((child
->x
== x
) &&
357 (child
->width
== width
) &&
358 (child
->height
== height
)) return;
362 child
->width
= width
;
363 child
->height
= height
;
365 gtk_widget_set_size_request (widget
, width
, height
);
373 gtk_pizza_map (GtkWidget
*widget
)
376 GtkPizzaChild
*child
;
379 g_return_if_fail (widget
!= NULL
);
380 g_return_if_fail (GTK_IS_PIZZA (widget
));
382 GTK_WIDGET_SET_FLAGS (widget
, GTK_MAPPED
);
383 pizza
= GTK_PIZZA (widget
);
385 children
= pizza
->children
;
388 child
= children
->data
;
389 children
= children
->next
;
391 if ( GTK_WIDGET_VISIBLE (child
->widget
) &&
392 !GTK_WIDGET_MAPPED (child
->widget
) )
394 gtk_widget_map (child
->widget
);
398 gdk_window_show (widget
->window
);
399 gdk_window_show (pizza
->bin_window
);
403 gtk_pizza_realize (GtkWidget
*widget
)
406 GdkWindowAttr attributes
;
407 gint attributes_mask
;
408 GtkPizzaChild
*child
;
411 g_return_if_fail (widget
!= NULL
);
412 g_return_if_fail (GTK_IS_PIZZA (widget
));
414 pizza
= GTK_PIZZA (widget
);
415 GTK_WIDGET_SET_FLAGS (widget
, GTK_REALIZED
);
417 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
;
424 #ifndef __WXUNIVERSAL__
425 if (pizza
->shadow_type
== GTK_MYSHADOW_NONE
)
427 /* no border, no changes to sizes */
429 else if (pizza
->shadow_type
== GTK_MYSHADOW_THIN
)
431 /* GTK_MYSHADOW_THIN == wxSIMPLE_BORDER */
434 attributes
.width
-= 2;
435 attributes
.height
-= 2;
439 /* GTK_MYSHADOW_IN == wxSUNKEN_BORDER */
440 /* GTK_MYSHADOW_OUT == wxRAISED_BORDER */
443 attributes
.width
-= 4;
444 attributes
.height
-= 4;
446 #endif /* __WXUNIVERSAL__ */
449 if (attributes
.width
< 2) attributes
.width
= 2;
450 if (attributes
.height
< 2) attributes
.height
= 2;
452 attributes
.wclass
= GDK_INPUT_OUTPUT
;
453 attributes
.visual
= gtk_widget_get_visual (widget
);
454 attributes
.colormap
= gtk_widget_get_colormap (widget
);
455 attributes
.event_mask
= GDK_VISIBILITY_NOTIFY_MASK
;
456 attributes_mask
= GDK_WA_X
| GDK_WA_Y
| GDK_WA_VISUAL
| GDK_WA_COLORMAP
;
458 widget
->window
= gdk_window_new(gtk_widget_get_parent_window (widget
),
459 &attributes
, attributes_mask
);
460 gdk_window_set_user_data (widget
->window
, widget
);
465 attributes
.event_mask
= gtk_widget_get_events (widget
);
466 attributes
.event_mask
|= GDK_EXPOSURE_MASK
|
468 GDK_POINTER_MOTION_MASK
|
469 GDK_POINTER_MOTION_HINT_MASK
|
470 GDK_BUTTON_MOTION_MASK
|
471 GDK_BUTTON1_MOTION_MASK
|
472 GDK_BUTTON2_MOTION_MASK
|
473 GDK_BUTTON3_MOTION_MASK
|
474 GDK_BUTTON_PRESS_MASK
|
475 GDK_BUTTON_RELEASE_MASK
|
477 GDK_KEY_RELEASE_MASK
|
478 GDK_ENTER_NOTIFY_MASK
|
479 GDK_LEAVE_NOTIFY_MASK
|
480 GDK_FOCUS_CHANGE_MASK
;
482 pizza
->bin_window
= gdk_window_new(widget
->window
,
483 &attributes
, attributes_mask
);
484 gdk_window_set_user_data (pizza
->bin_window
, widget
);
486 widget
->style
= gtk_style_attach (widget
->style
, widget
->window
);
487 gtk_style_set_background (widget
->style
, widget
->window
, GTK_STATE_NORMAL
);
488 gtk_style_set_background (widget
->style
, pizza
->bin_window
, GTK_STATE_NORMAL
);
491 gdk_window_set_back_pixmap( widget->window, NULL, FALSE );
492 gdk_window_set_back_pixmap( pizza->bin_window, NULL, FALSE );
495 /* cannot be done before realisation */
496 children
= pizza
->children
;
499 child
= children
->data
;
500 children
= children
->next
;
502 gtk_widget_set_parent_window (child
->widget
, pizza
->bin_window
);
507 gtk_pizza_unrealize (GtkWidget
*widget
)
511 g_return_if_fail (widget
!= NULL
);
512 g_return_if_fail (GTK_IS_PIZZA (widget
));
514 pizza
= GTK_PIZZA (widget
);
516 gdk_window_set_user_data (pizza
->bin_window
, NULL
);
517 gdk_window_destroy (pizza
->bin_window
);
518 pizza
->bin_window
= NULL
;
520 if (pizza_parent_class
->unrealize
)
521 pizza_parent_class
->unrealize(widget
);
525 gtk_pizza_size_request (GtkWidget
*widget
,
526 GtkRequisition
*requisition
)
529 GtkPizzaChild
*child
;
531 GtkRequisition child_requisition
;
533 g_return_if_fail (widget
!= NULL
);
534 g_return_if_fail (GTK_IS_PIZZA (widget
));
535 g_return_if_fail (requisition
!= NULL
);
537 pizza
= GTK_PIZZA (widget
);
539 children
= pizza
->children
;
542 child
= children
->data
;
543 children
= children
->next
;
545 if (GTK_WIDGET_VISIBLE (child
->widget
))
547 gtk_widget_size_request (child
->widget
, &child_requisition
);
551 /* request very little, I'm not sure if requesting nothing
552 will always have positive effects on stability... */
553 requisition
->width
= 2;
554 requisition
->height
= 2;
558 gtk_pizza_size_allocate (GtkWidget
*widget
,
559 GtkAllocation
*allocation
)
564 GtkPizzaChild
*child
;
566 gboolean only_resize
;
568 g_return_if_fail (widget
!= NULL
);
569 g_return_if_fail (GTK_IS_PIZZA(widget
));
570 g_return_if_fail (allocation
!= NULL
);
572 pizza
= GTK_PIZZA (widget
);
574 only_resize
= ((widget
->allocation
.x
== allocation
->x
) &&
575 (widget
->allocation
.y
== allocation
->y
));
576 widget
->allocation
= *allocation
;
578 if (pizza
->shadow_type
== GTK_MYSHADOW_NONE
)
581 if (pizza
->shadow_type
== GTK_MYSHADOW_THIN
)
586 x
= allocation
->x
+ border
;
587 y
= allocation
->y
+ border
;
588 w
= allocation
->width
- border
*2;
589 h
= allocation
->height
- border
*2;
595 if (GTK_WIDGET_REALIZED (widget
))
598 gdk_window_resize( widget
->window
, w
, h
);
600 gdk_window_move_resize( widget
->window
, x
, y
, w
, h
);
602 gdk_window_resize( pizza
->bin_window
, w
, h
);
605 children
= pizza
->children
;
608 child
= children
->data
;
609 children
= children
->next
;
611 gtk_pizza_allocate_child (pizza
, child
);
616 gtk_pizza_expose (GtkWidget
*widget
,
617 GdkEventExpose
*event
)
621 g_return_val_if_fail (widget
!= NULL
, FALSE
);
622 g_return_val_if_fail (GTK_IS_PIZZA (widget
), FALSE
);
623 g_return_val_if_fail (event
!= NULL
, FALSE
);
625 pizza
= (GtkPizza
*)widget
;
627 if (event
->window
!= pizza
->bin_window
)
630 pizza_parent_class
->expose_event(widget
, event
);
636 gtk_pizza_style_set(GtkWidget
*widget
, GtkStyle
*previous_style
)
638 if (GTK_WIDGET_REALIZED(widget
))
640 gtk_style_set_background(widget
->style
, widget
->window
, GTK_STATE_NORMAL
);
641 gtk_style_set_background(widget
->style
, GTK_PIZZA(widget
)->bin_window
, GTK_STATE_NORMAL
);
644 pizza_parent_class
->style_set(widget
, previous_style
);
648 gtk_pizza_add (GtkContainer
*container
,
651 g_return_if_fail (container
!= NULL
);
652 g_return_if_fail (GTK_IS_PIZZA (container
));
653 g_return_if_fail (widget
!= NULL
);
655 gtk_pizza_put (GTK_PIZZA (container
), widget
, 0, 0, 20, 20 );
659 gtk_pizza_remove (GtkContainer
*container
,
663 GtkPizzaChild
*child
;
666 g_return_if_fail (container
!= NULL
);
667 g_return_if_fail (GTK_IS_PIZZA (container
));
668 g_return_if_fail (widget
!= NULL
);
670 pizza
= GTK_PIZZA (container
);
672 children
= pizza
->children
;
675 child
= children
->data
;
677 if (child
->widget
== widget
)
679 gtk_widget_unparent (widget
);
681 /* security checks */
682 g_return_if_fail (GTK_IS_WIDGET (widget
));
684 pizza
->children
= g_list_remove_link (pizza
->children
, children
);
685 g_list_free (children
);
688 /* security checks */
689 g_return_if_fail (GTK_IS_WIDGET (widget
));
694 children
= children
->next
;
699 gtk_pizza_forall (GtkContainer
*container
,
700 gboolean include_internals
,
701 GtkCallback callback
,
702 gpointer callback_data
)
705 GtkPizzaChild
*child
;
708 g_return_if_fail (container
!= NULL
);
709 g_return_if_fail (GTK_IS_PIZZA (container
));
710 g_return_if_fail (callback
!= (GtkCallback
)NULL
);
712 pizza
= GTK_PIZZA (container
);
714 children
= pizza
->children
;
717 child
= children
->data
;
718 children
= children
->next
;
720 (* callback
) (child
->widget
, callback_data
);
725 gtk_pizza_allocate_child (GtkPizza
*pizza
,
726 GtkPizzaChild
*child
)
728 GtkAllocation allocation
;
729 GtkRequisition requisition
;
731 allocation
.x
= child
->x
- pizza
->m_xoffset
;
732 allocation
.y
= child
->y
- pizza
->m_yoffset
;
733 gtk_widget_get_child_requisition (child
->widget
, &requisition
);
734 allocation
.width
= requisition
.width
;
735 allocation
.height
= requisition
.height
;
737 if (gtk_widget_get_direction( GTK_WIDGET(pizza
) ) == GTK_TEXT_DIR_RTL
)
739 /* reverse horizontal placement */
740 /* printf( "alloc width %d\n", GTK_WIDGET(pizza)->allocation.width ); */
741 allocation
.x
= GTK_WIDGET(pizza
)->allocation
.width
- child
->x
- allocation
.width
- pizza
->m_xoffset
;
744 gtk_widget_size_allocate (child
->widget
, &allocation
);
748 gtk_pizza_adjust_allocations_recurse (GtkWidget
*widget
,
751 GtkPizzaAdjData
*data
= cb_data
;
753 widget
->allocation
.x
+= data
->dx
;
754 widget
->allocation
.y
+= data
->dy
;
756 if (GTK_WIDGET_NO_WINDOW (widget
) && GTK_IS_CONTAINER (widget
))
758 gtk_container_forall (GTK_CONTAINER (widget
),
759 gtk_pizza_adjust_allocations_recurse
,
765 gtk_pizza_adjust_allocations (GtkPizza
*pizza
,
770 GtkPizzaAdjData data
;
775 tmp_list
= pizza
->children
;
778 GtkPizzaChild
*child
= tmp_list
->data
;
779 tmp_list
= tmp_list
->next
;
781 child
->widget
->allocation
.x
+= dx
;
782 child
->widget
->allocation
.y
+= dy
;
784 if (GTK_WIDGET_NO_WINDOW (child
->widget
) &&
785 GTK_IS_CONTAINER (child
->widget
))
787 gtk_container_forall (GTK_CONTAINER (child
->widget
),
788 gtk_pizza_adjust_allocations_recurse
,
795 gtk_pizza_scroll (GtkPizza
*pizza
, gint dx
, gint dy
)
797 pizza
->m_xoffset
+= dx
;
798 pizza
->m_yoffset
+= dy
;
800 gtk_pizza_adjust_allocations (pizza
, -dx
, -dy
);
802 if (pizza
->bin_window
)
803 gdk_window_scroll( pizza
->bin_window
, -dx
, -dy
);
808 #endif /* __cplusplus */