1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/win_gtk.cpp
3 // Purpose: native GTK+ widget for wxWindow
4 // Author: Paul Cornett
6 // Copyright: (c) 2007 Paul Cornett
7 // Licence: wxWindows licence
8 ///////////////////////////////////////////////////////////////////////////////
11 #include "wx/gtk/win_gtk.h"
14 wxPizza is a custom GTK+ widget derived from GtkFixed. A custom widget
15 is needed to adapt GTK+ to wxWidgets needs in 3 areas: scrolling, window
18 For scrolling, the "set_scroll_adjustments" signal is implemented
19 to make wxPizza appear scrollable to GTK+, allowing it to be put in a
20 GtkScrolledWindow. Child widget positions are adjusted for the scrolling
21 position in size_allocate. A second same-size GdkWindow is placed behind
22 widget->window, to allow GDK to use a more efficient scrolling technique.
24 For borders, space is reserved in realize and size_allocate.
25 For non-scrolling wxWindows, a second GdkWindow is used for the
26 border; scrolling wxWindows draw their border on wxPizza's parent
27 (a GtkScrolledWindow) GdkWindow. Border widths for sunken and raised
28 styles are taken from widget->style, so they will match the current theme.
30 For RTL, child widget positions are mirrored in size_allocate.
33 static GtkWidgetClass
* parent_class
;
40 void (*set_scroll_adjustments
)(GtkWidget
*, GtkAdjustment
*, GtkAdjustment
*);
43 static void size_allocate(GtkWidget
* widget
, GtkAllocation
* alloc
)
45 const bool is_resize
=
46 widget
->allocation
.width
!= alloc
->width
||
47 widget
->allocation
.height
!= alloc
->height
;
49 widget
->allocation
.x
!= alloc
->x
||
50 widget
->allocation
.y
!= alloc
->y
;
52 widget
->allocation
= *alloc
;
54 wxPizza
* pizza
= WX_PIZZA(widget
);
55 int border_x
, border_y
;
56 pizza
->get_border_widths(border_x
, border_y
);
57 int w
= alloc
->width
- 2 * border_x
;
60 if (GTK_WIDGET_REALIZED(widget
) && (is_move
|| is_resize
))
62 int h
= alloc
->height
- 2 * border_y
;
65 if (pizza
->m_is_scrollable
)
67 // backing window is inside border
68 gdk_window_move_resize(pizza
->m_backing_window
,
69 alloc
->x
+ border_x
, alloc
->y
+ border_y
, w
, h
);
73 // border window (or only window if
74 // no-scroll no-border) is full size
75 GdkWindow
* window
= pizza
->m_backing_window
;
77 window
= widget
->window
;
78 gdk_window_move_resize(
79 window
, alloc
->x
, alloc
->y
, alloc
->width
, alloc
->height
);
81 if (is_resize
&& pizza
->m_backing_window
)
83 // main window is inside border
84 if (pizza
->m_is_scrollable
)
85 gdk_window_resize(widget
->window
, w
, h
);
87 // need move as well as resize because border can change
88 gdk_window_move_resize(widget
->window
, border_x
, border_y
, w
, h
);
90 // wxWidgets turns off redraw-on-allocate by default,
91 // so border area needs to be invalidated
92 if (border_x
> 1 || border_y
> 1)
94 if (pizza
->m_is_scrollable
)
95 ; // invalidate does not seem to be needed in this case
97 gdk_window_invalidate_rect(pizza
->m_backing_window
, NULL
, false);
101 // adjust child positions
102 for (const GList
* list
= pizza
->m_fixed
.children
; list
; list
= list
->next
)
104 const GtkFixedChild
* child
= static_cast<GtkFixedChild
*>(list
->data
);
105 if (GTK_WIDGET_VISIBLE(child
->widget
))
107 GtkAllocation child_alloc
;
108 // note that child positions do not take border into
109 // account, they need to be relative to widget->window,
110 // which has already been adjusted
111 child_alloc
.x
= child
->x
- pizza
->m_scroll_x
;
112 child_alloc
.y
= child
->y
- pizza
->m_scroll_y
;
114 gtk_widget_get_child_requisition(child
->widget
, &req
);
115 child_alloc
.width
= req
.width
;
116 child_alloc
.height
= req
.height
;
117 if (gtk_widget_get_direction(widget
) == GTK_TEXT_DIR_RTL
)
119 child_alloc
.x
= w
- child_alloc
.x
- child_alloc
.width
;
121 gtk_widget_size_allocate(child
->widget
, &child_alloc
);
126 static void realize(GtkWidget
* widget
)
128 parent_class
->realize(widget
);
130 wxPizza
* pizza
= WX_PIZZA(widget
);
131 // second window is created if there is a border, or wxWindow is scrollable
132 if (pizza
->m_border_style
|| pizza
->m_is_scrollable
)
135 attr
.event_mask
= pizza
->m_is_scrollable
? 0 : GDK_EXPOSURE_MASK
;
136 attr
.x
= widget
->allocation
.x
;
137 attr
.y
= widget
->allocation
.y
;
138 attr
.width
= widget
->allocation
.width
;
139 attr
.height
= widget
->allocation
.height
;
140 int border_x
, border_y
;
141 pizza
->get_border_widths(border_x
, border_y
);
142 int w
= widget
->allocation
.width
- 2 * border_x
;
143 int h
= widget
->allocation
.height
- 2 * border_y
;
146 if (pizza
->m_is_scrollable
)
153 attr
.wclass
= GDK_INPUT_OUTPUT
;
154 attr
.visual
= gtk_widget_get_visual(widget
);
155 attr
.colormap
= gtk_widget_get_colormap(widget
);
156 attr
.window_type
= GDK_WINDOW_CHILD
;
158 pizza
->m_backing_window
= gdk_window_new(
159 gdk_window_get_parent(widget
->window
),
161 GDK_WA_X
| GDK_WA_Y
| GDK_WA_VISUAL
| GDK_WA_COLORMAP
);
163 gdk_window_set_user_data(pizza
->m_backing_window
, widget
);
164 if (pizza
->m_is_scrollable
)
165 gdk_window_reparent(widget
->window
, pizza
->m_backing_window
, 0, 0);
167 gdk_window_reparent(widget
->window
, pizza
->m_backing_window
, border_x
, border_y
);
168 gdk_window_resize(widget
->window
, w
, h
);
172 static void unrealize(GtkWidget
* widget
)
174 parent_class
->unrealize(widget
);
176 wxPizza
* pizza
= WX_PIZZA(widget
);
177 if (pizza
->m_backing_window
)
179 gdk_window_set_user_data(pizza
->m_backing_window
, NULL
);
180 gdk_window_destroy(pizza
->m_backing_window
);
181 pizza
->m_backing_window
= NULL
;
185 static void map(GtkWidget
* widget
)
187 parent_class
->map(widget
);
189 wxPizza
* pizza
= WX_PIZZA(widget
);
190 if (pizza
->m_backing_window
)
191 gdk_window_show(pizza
->m_backing_window
);
194 static void unmap(GtkWidget
* widget
)
196 parent_class
->unmap(widget
);
198 wxPizza
* pizza
= WX_PIZZA(widget
);
199 if (pizza
->m_backing_window
)
200 gdk_window_hide(pizza
->m_backing_window
);
203 // not used, but needs to exist so gtk_widget_set_scroll_adjustments will work
204 static void set_scroll_adjustments(GtkWidget
*, GtkAdjustment
*, GtkAdjustment
*)
208 // Marshaller needed for set_scroll_adjustments signal,
209 // generated with GLib-2.4.6 glib-genmarshal
210 #define g_marshal_value_peek_object(v) g_value_get_object (v)
212 g_cclosure_user_marshal_VOID__OBJECT_OBJECT (GClosure
*closure
,
213 GValue
* /*return_value*/,
214 guint n_param_values
,
215 const GValue
*param_values
,
216 gpointer
/*invocation_hint*/,
217 gpointer marshal_data
)
219 typedef void (*GMarshalFunc_VOID__OBJECT_OBJECT
) (gpointer data1
,
223 register GMarshalFunc_VOID__OBJECT_OBJECT callback
;
224 register GCClosure
*cc
= (GCClosure
*) closure
;
225 register gpointer data1
, data2
;
227 g_return_if_fail (n_param_values
== 3);
229 if (G_CCLOSURE_SWAP_DATA (closure
))
231 data1
= closure
->data
;
232 data2
= g_value_peek_pointer (param_values
+ 0);
236 data1
= g_value_peek_pointer (param_values
+ 0);
237 data2
= closure
->data
;
239 callback
= (GMarshalFunc_VOID__OBJECT_OBJECT
) (marshal_data
? marshal_data
: cc
->callback
);
242 g_marshal_value_peek_object (param_values
+ 1),
243 g_marshal_value_peek_object (param_values
+ 2),
247 static void class_init(void* g_class
, void*)
249 GtkWidgetClass
* widget_class
= (GtkWidgetClass
*)g_class
;
250 widget_class
->size_allocate
= size_allocate
;
251 widget_class
->realize
= realize
;
252 widget_class
->unrealize
= unrealize
;
253 widget_class
->map
= map
;
254 widget_class
->unmap
= unmap
;
255 wxPizzaClass
* klass
= (wxPizzaClass
*)g_class
;
257 // needed to make widget appear scrollable to GTK+
258 klass
->set_scroll_adjustments
= set_scroll_adjustments
;
259 widget_class
->set_scroll_adjustments_signal
=
261 "set_scroll_adjustments",
262 G_TYPE_FROM_CLASS(g_class
),
264 G_STRUCT_OFFSET(wxPizzaClass
, set_scroll_adjustments
),
266 g_cclosure_user_marshal_VOID__OBJECT_OBJECT
,
267 G_TYPE_NONE
, 2, GTK_TYPE_ADJUSTMENT
, GTK_TYPE_ADJUSTMENT
);
269 parent_class
= GTK_WIDGET_CLASS(g_type_class_peek_parent(g_class
));
274 GType
wxPizza::type()
279 const GTypeInfo info
= {
280 sizeof(wxPizzaClass
),
287 type
= g_type_register_static(
288 GTK_TYPE_FIXED
, "wxPizza", &info
, GTypeFlags(0));
293 GtkWidget
* wxPizza::New(long windowStyle
)
295 GtkWidget
* widget
= GTK_WIDGET(g_object_new(type(), NULL
));
296 wxPizza
* pizza
= WX_PIZZA(widget
);
297 pizza
->m_backing_window
= NULL
;
298 pizza
->m_scroll_x
= 0;
299 pizza
->m_scroll_y
= 0;
300 pizza
->m_is_scrollable
= (windowStyle
& (wxHSCROLL
| wxVSCROLL
)) != 0;
301 // mask off border styles not useable with wxPizza
302 pizza
->m_border_style
= int(windowStyle
& BORDER_STYLES
);
303 gtk_fixed_set_has_window(GTK_FIXED(widget
), true);
304 gtk_widget_add_events(widget
,
307 GDK_POINTER_MOTION_MASK
|
308 GDK_POINTER_MOTION_HINT_MASK
|
309 GDK_BUTTON_MOTION_MASK
|
310 GDK_BUTTON1_MOTION_MASK
|
311 GDK_BUTTON2_MOTION_MASK
|
312 GDK_BUTTON3_MOTION_MASK
|
313 GDK_BUTTON_PRESS_MASK
|
314 GDK_BUTTON_RELEASE_MASK
|
316 GDK_KEY_RELEASE_MASK
|
317 GDK_ENTER_NOTIFY_MASK
|
318 GDK_LEAVE_NOTIFY_MASK
|
319 GDK_FOCUS_CHANGE_MASK
);
323 // gtk_fixed_move does not check for a change before issuing a queue_resize,
324 // we need to avoid that to prevent endless sizing loops, so check first
325 void wxPizza::move(GtkWidget
* widget
, int x
, int y
)
327 GtkFixed
* fixed
= &m_fixed
;
328 for (const GList
* list
= fixed
->children
; list
; list
= list
->next
)
330 const GtkFixedChild
* child
= static_cast<GtkFixedChild
*>(list
->data
);
331 if (child
->widget
== widget
)
333 if (child
->x
!= x
|| child
->y
!= y
)
334 gtk_fixed_move(fixed
, widget
, x
, y
);
340 void wxPizza::scroll(int dx
, int dy
)
342 GtkWidget
* widget
= GTK_WIDGET(this);
343 if (gtk_widget_get_direction(widget
) == GTK_TEXT_DIR_RTL
)
348 gdk_window_scroll(widget
->window
, dx
, dy
);
349 const GList
* list
= m_fixed
.children
;
352 const GtkFixedChild
* child
= static_cast<GtkFixedChild
*>(list
->data
);
353 // queueing a resize on any child will update them all
354 gtk_widget_queue_resize(child
->widget
);
358 extern GtkWidget
*GetEntryWidget();
360 void wxPizza::get_border_widths(int& x
, int& y
)
363 if (m_border_style
& wxBORDER_SIMPLE
)
365 else if (m_border_style
)
367 GtkWidget
*entry_widget
= GetEntryWidget();
368 if (entry_widget
->style
)
370 x
= entry_widget
->style
->xthickness
;
371 y
= entry_widget
->style
->ythickness
;