Commit | Line | Data |
---|---|---|
c67daf87 | 1 | /* /////////////////////////////////////////////////////////////////////////// |
c67d8618 | 2 | // Name: win_gtk.c |
01111366 | 3 | // Purpose: native GTK+ widget for wxWindows |
c801d85f | 4 | // Author: Robert Roebling |
c67d8618 | 5 | // Id: $Id$ |
01111366 | 6 | // Copyright: (c) 1998 Robert Roebling |
c801d85f | 7 | // Licence: wxWindows licence |
c67daf87 | 8 | /////////////////////////////////////////////////////////////////////////// */ |
c801d85f KB |
9 | |
10 | #include "wx/gtk/win_gtk.h" | |
75ed1d15 | 11 | #include <gtk/gtkfeatures.h> |
c801d85f | 12 | |
38c7b3d3 RR |
13 | /*------------------------------------------------------------------------- |
14 | // conditional compilation | |
15 | //------------------------------------------------------------------------- */ | |
16 | ||
17 | #if (GTK_MINOR_VERSION == 1) | |
18 | #if (GTK_MICRO_VERSION >= 5) | |
19 | #define NEW_GTK_CONSTRUCT_CODE | |
20 | #endif | |
21 | #endif | |
22 | ||
c801d85f KB |
23 | #ifdef __cplusplus |
24 | extern "C" { | |
25 | #endif /* __cplusplus */ | |
26 | ||
27 | static void gtk_myfixed_class_init (GtkMyFixedClass *klass); | |
28 | static void gtk_myfixed_init (GtkMyFixed *myfixed); | |
29 | static void gtk_myfixed_map (GtkWidget *widget); | |
38c7b3d3 | 30 | #ifndef NEW_GTK_CONSTRUCT_CODE |
c801d85f | 31 | static void gtk_myfixed_unmap (GtkWidget *widget); |
38c7b3d3 | 32 | #endif |
c801d85f KB |
33 | static void gtk_myfixed_realize (GtkWidget *widget); |
34 | static void gtk_myfixed_size_request (GtkWidget *widget, | |
35 | GtkRequisition *requisition); | |
36 | static void gtk_myfixed_size_allocate (GtkWidget *widget, | |
37 | GtkAllocation *allocation); | |
38 | static void gtk_myfixed_paint (GtkWidget *widget, | |
39 | GdkRectangle *area); | |
40 | static void gtk_myfixed_draw (GtkWidget *widget, | |
41 | GdkRectangle *area); | |
42 | static gint gtk_myfixed_expose (GtkWidget *widget, | |
43 | GdkEventExpose *event); | |
44 | static void gtk_myfixed_add (GtkContainer *container, | |
45 | GtkWidget *widget); | |
46 | static void gtk_myfixed_remove (GtkContainer *container, | |
47 | GtkWidget *widget); | |
48 | static void gtk_myfixed_foreach (GtkContainer *container, | |
75ed1d15 GL |
49 | #if (GTK_MINOR_VERSION == 1) |
50 | gboolean include_internals, | |
51 | #endif | |
c801d85f KB |
52 | GtkCallback callback, |
53 | gpointer callback_data); | |
38c7b3d3 RR |
54 | #ifdef NEW_GTK_CONSTRUCT_CODE |
55 | static GtkType gtk_myfixed_child_type (GtkContainer *container); | |
56 | #endif | |
c801d85f KB |
57 | |
58 | ||
59 | static GtkContainerClass *parent_class = NULL; | |
60 | ||
61 | ||
62 | guint | |
63 | gtk_myfixed_get_type () | |
64 | { | |
65 | static guint myfixed_type = 0; | |
66 | ||
67 | if (!myfixed_type) | |
68 | { | |
69 | GtkTypeInfo myfixed_info = | |
70 | { | |
71 | "GtkMyFixed", | |
72 | sizeof (GtkMyFixed), | |
73 | sizeof (GtkMyFixedClass), | |
74 | (GtkClassInitFunc) gtk_myfixed_class_init, | |
75 | (GtkObjectInitFunc) gtk_myfixed_init, | |
38c7b3d3 RR |
76 | #ifndef NEW_GTK_CONSTRUCT_CODE |
77 | (GtkArgSetFunc) NULL, | |
c801d85f | 78 | (GtkArgGetFunc) NULL, |
38c7b3d3 RR |
79 | #else |
80 | /* reserved_1 */ NULL, | |
81 | /* reserved_2 */ NULL, | |
82 | (GtkClassInitFunc) NULL, | |
83 | #endif | |
c801d85f KB |
84 | }; |
85 | ||
86 | myfixed_type = gtk_type_unique (gtk_container_get_type (), &myfixed_info); | |
87 | } | |
88 | ||
89 | return myfixed_type; | |
90 | } | |
91 | ||
92 | static void | |
93 | gtk_myfixed_class_init (GtkMyFixedClass *klass) | |
94 | { | |
95 | GtkObjectClass *object_class; | |
96 | GtkWidgetClass *widget_class; | |
97 | GtkContainerClass *container_class; | |
98 | ||
99 | object_class = (GtkObjectClass*) klass; | |
100 | widget_class = (GtkWidgetClass*) klass; | |
101 | container_class = (GtkContainerClass*) klass; | |
f5368809 | 102 | |
38c7b3d3 | 103 | #ifndef NEW_GTK_CONSTRUCT_CODE |
c801d85f | 104 | parent_class = gtk_type_class (gtk_container_get_type ()); |
38c7b3d3 RR |
105 | #else |
106 | parent_class = gtk_type_class (GTK_TYPE_CONTAINER); | |
107 | #endif | |
c801d85f KB |
108 | |
109 | widget_class->map = gtk_myfixed_map; | |
38c7b3d3 | 110 | #ifndef NEW_GTK_CONSTRUCT_CODE |
c801d85f | 111 | widget_class->unmap = gtk_myfixed_unmap; |
38c7b3d3 | 112 | #endif |
c801d85f KB |
113 | widget_class->realize = gtk_myfixed_realize; |
114 | widget_class->size_request = gtk_myfixed_size_request; | |
115 | widget_class->size_allocate = gtk_myfixed_size_allocate; | |
116 | widget_class->draw = gtk_myfixed_draw; | |
117 | widget_class->expose_event = gtk_myfixed_expose; | |
118 | ||
119 | container_class->add = gtk_myfixed_add; | |
120 | container_class->remove = gtk_myfixed_remove; | |
75ed1d15 GL |
121 | #if (GTK_MINOR_VERSION == 1) |
122 | container_class->forall = gtk_myfixed_foreach; | |
123 | #else | |
c801d85f | 124 | container_class->foreach = gtk_myfixed_foreach; |
75ed1d15 | 125 | #endif |
38c7b3d3 RR |
126 | |
127 | #ifdef NEW_GTK_CONSTRUCT_CODE | |
128 | container_class->child_type = gtk_myfixed_child_type; | |
129 | #endif | |
130 | } | |
131 | ||
132 | #ifdef NEW_GTK_CONSTRUCT_CODE | |
133 | static GtkType | |
134 | gtk_myfixed_child_type (GtkContainer *container) | |
135 | { | |
136 | return GTK_TYPE_WIDGET; | |
c801d85f | 137 | } |
38c7b3d3 | 138 | #endif |
c801d85f KB |
139 | |
140 | static void | |
141 | gtk_myfixed_init (GtkMyFixed *myfixed) | |
142 | { | |
143 | GTK_WIDGET_UNSET_FLAGS (myfixed, GTK_NO_WINDOW); | |
38c7b3d3 RR |
144 | |
145 | #ifndef NEW_GTK_CONSTRUCT_CODE | |
c801d85f | 146 | GTK_WIDGET_SET_FLAGS (myfixed, GTK_BASIC); |
38c7b3d3 | 147 | #endif |
f5368809 | 148 | |
c801d85f KB |
149 | myfixed->children = NULL; |
150 | } | |
151 | ||
152 | GtkWidget* | |
153 | gtk_myfixed_new () | |
154 | { | |
155 | GtkMyFixed *myfixed; | |
156 | ||
157 | myfixed = gtk_type_new (gtk_myfixed_get_type ()); | |
158 | ||
c801d85f KB |
159 | return GTK_WIDGET (myfixed); |
160 | } | |
161 | ||
c801d85f KB |
162 | void |
163 | gtk_myfixed_put (GtkMyFixed *myfixed, | |
164 | GtkWidget *widget, | |
165 | gint16 x, | |
166 | gint16 y) | |
167 | { | |
168 | GtkMyFixedChild *child_info; | |
169 | ||
170 | g_return_if_fail (myfixed != NULL); | |
171 | g_return_if_fail (GTK_IS_MYFIXED (myfixed)); | |
172 | g_return_if_fail (widget != NULL); | |
173 | ||
174 | child_info = g_new (GtkMyFixedChild, 1); | |
175 | child_info->widget = widget; | |
11026f7b RR |
176 | child_info->x = x; |
177 | child_info->y = y; | |
d4c99d6f | 178 | |
c801d85f KB |
179 | gtk_widget_set_parent (widget, GTK_WIDGET (myfixed)); |
180 | ||
181 | myfixed->children = g_list_append (myfixed->children, child_info); | |
182 | ||
183 | if (GTK_WIDGET_REALIZED (myfixed) && !GTK_WIDGET_REALIZED (widget)) | |
184 | gtk_widget_realize (widget); | |
185 | ||
186 | if (GTK_WIDGET_MAPPED (myfixed) && !GTK_WIDGET_MAPPED (widget)) | |
187 | gtk_widget_map (widget); | |
188 | ||
189 | if (GTK_WIDGET_VISIBLE (widget) && GTK_WIDGET_VISIBLE (myfixed)) | |
190 | gtk_widget_queue_resize (GTK_WIDGET (myfixed)); | |
191 | } | |
192 | ||
193 | void | |
194 | gtk_myfixed_move (GtkMyFixed *myfixed, | |
195 | GtkWidget *widget, | |
196 | gint16 x, | |
197 | gint16 y) | |
198 | { | |
199 | GtkMyFixedChild *child; | |
200 | GList *children; | |
201 | ||
202 | g_return_if_fail (myfixed != NULL); | |
203 | g_return_if_fail (GTK_IS_MYFIXED (myfixed)); | |
204 | g_return_if_fail (widget != NULL); | |
205 | ||
206 | children = myfixed->children; | |
207 | while (children) | |
208 | { | |
209 | child = children->data; | |
210 | children = children->next; | |
211 | ||
212 | if (child->widget == widget) | |
213 | { | |
463c1fa1 | 214 | if ((child->x == x) && (child->y == y)) return; |
fb1585ae | 215 | |
11026f7b RR |
216 | child->x = x; |
217 | child->y = y; | |
c801d85f KB |
218 | |
219 | if (GTK_WIDGET_VISIBLE (widget) && GTK_WIDGET_VISIBLE (myfixed)) | |
220 | gtk_widget_queue_resize (GTK_WIDGET (myfixed)); | |
221 | ||
222 | break; | |
223 | } | |
224 | } | |
225 | } | |
226 | ||
227 | static void | |
228 | gtk_myfixed_map (GtkWidget *widget) | |
229 | { | |
230 | GtkMyFixed *myfixed; | |
231 | GtkMyFixedChild *child; | |
232 | GList *children; | |
233 | ||
234 | g_return_if_fail (widget != NULL); | |
235 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
236 | ||
237 | GTK_WIDGET_SET_FLAGS (widget, GTK_MAPPED); | |
238 | myfixed = GTK_MYFIXED (widget); | |
239 | ||
240 | gdk_window_show (widget->window); | |
241 | ||
242 | children = myfixed->children; | |
243 | while (children) | |
244 | { | |
245 | child = children->data; | |
246 | children = children->next; | |
247 | ||
248 | if (GTK_WIDGET_VISIBLE (child->widget) && | |
249 | !GTK_WIDGET_MAPPED (child->widget)) | |
250 | gtk_widget_map (child->widget); | |
251 | } | |
252 | } | |
253 | ||
38c7b3d3 | 254 | #ifndef NEW_GTK_CONSTRUCT_CODE |
c801d85f KB |
255 | static void |
256 | gtk_myfixed_unmap (GtkWidget *widget) | |
257 | { | |
258 | g_return_if_fail (widget != NULL); | |
259 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
260 | ||
261 | GTK_WIDGET_UNSET_FLAGS (widget, GTK_MAPPED); | |
262 | } | |
38c7b3d3 | 263 | #endif |
c801d85f KB |
264 | |
265 | static void | |
266 | gtk_myfixed_realize (GtkWidget *widget) | |
267 | { | |
d4c99d6f | 268 | GtkMyFixed *myfixed; |
c801d85f KB |
269 | GdkWindowAttr attributes; |
270 | gint attributes_mask; | |
271 | ||
272 | g_return_if_fail (widget != NULL); | |
273 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
274 | ||
d4c99d6f RR |
275 | myfixed = GTK_MYFIXED (widget); |
276 | ||
c801d85f KB |
277 | GTK_WIDGET_SET_FLAGS (widget, GTK_REALIZED); |
278 | ||
279 | attributes.window_type = GDK_WINDOW_CHILD; | |
11026f7b RR |
280 | attributes.x = widget->allocation.x; |
281 | attributes.y = widget->allocation.y; | |
c801d85f KB |
282 | attributes.width = 32000; |
283 | attributes.height = 32000; | |
284 | attributes.wclass = GDK_INPUT_OUTPUT; | |
285 | attributes.visual = gtk_widget_get_visual (widget); | |
286 | attributes.colormap = gtk_widget_get_colormap (widget); | |
287 | attributes.event_mask = gtk_widget_get_events (widget); | |
288 | attributes.event_mask |= | |
289 | GDK_EXPOSURE_MASK | | |
290 | GDK_POINTER_MOTION_MASK | | |
291 | GDK_BUTTON_MOTION_MASK | | |
292 | GDK_BUTTON1_MOTION_MASK | | |
293 | GDK_BUTTON2_MOTION_MASK | | |
294 | GDK_BUTTON3_MOTION_MASK | | |
295 | GDK_BUTTON_PRESS_MASK | | |
296 | GDK_BUTTON_RELEASE_MASK | | |
297 | GDK_KEY_PRESS_MASK | | |
298 | GDK_KEY_RELEASE_MASK | | |
299 | GDK_ENTER_NOTIFY_MASK | | |
300 | GDK_LEAVE_NOTIFY_MASK | | |
301 | GDK_FOCUS_CHANGE_MASK; | |
302 | ||
303 | attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP; | |
304 | ||
305 | widget->window = gdk_window_new (gtk_widget_get_parent_window (widget), &attributes, | |
306 | attributes_mask); | |
307 | gdk_window_set_user_data (widget->window, widget); | |
308 | ||
309 | widget->style = gtk_style_attach (widget->style, widget->window); | |
310 | gtk_style_set_background (widget->style, widget->window, GTK_STATE_NORMAL); | |
311 | } | |
312 | ||
313 | static void | |
314 | gtk_myfixed_size_request (GtkWidget *widget, | |
315 | GtkRequisition *requisition) | |
316 | { | |
317 | GtkMyFixed *myfixed; | |
318 | GtkMyFixedChild *child; | |
319 | GList *children; | |
320 | ||
321 | g_return_if_fail (widget != NULL); | |
322 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
323 | g_return_if_fail (requisition != NULL); | |
324 | ||
325 | myfixed = GTK_MYFIXED (widget); | |
326 | ||
327 | requisition->width = 0; | |
328 | requisition->height = 0; | |
329 | ||
330 | children = myfixed->children; | |
331 | while (children) | |
332 | { | |
333 | child = children->data; | |
334 | children = children->next; | |
335 | ||
336 | if (GTK_WIDGET_VISIBLE (child->widget)) | |
337 | { | |
338 | gtk_widget_size_request (child->widget, &child->widget->requisition); | |
339 | } | |
340 | } | |
341 | } | |
342 | ||
343 | static void | |
344 | gtk_myfixed_size_allocate (GtkWidget *widget, | |
345 | GtkAllocation *allocation) | |
346 | { | |
347 | GtkMyFixed *myfixed; | |
348 | GtkMyFixedChild *child; | |
349 | GtkAllocation child_allocation; | |
350 | GList *children; | |
351 | guint16 border_width; | |
352 | ||
353 | g_return_if_fail (widget != NULL); | |
354 | g_return_if_fail (GTK_IS_MYFIXED(widget)); | |
355 | g_return_if_fail (allocation != NULL); | |
356 | ||
357 | myfixed = GTK_MYFIXED (widget); | |
358 | ||
359 | widget->allocation = *allocation; | |
360 | if (GTK_WIDGET_REALIZED (widget)) | |
11026f7b | 361 | gdk_window_move_resize (widget->window, allocation->x, allocation->y, 32000, 32000 ); |
c801d85f KB |
362 | |
363 | border_width = GTK_CONTAINER (myfixed)->border_width; | |
364 | ||
365 | children = myfixed->children; | |
366 | while (children) | |
367 | { | |
368 | child = children->data; | |
369 | children = children->next; | |
370 | ||
371 | if (GTK_WIDGET_VISIBLE (child->widget)) | |
372 | { | |
373 | child_allocation.x = child->x + border_width; | |
374 | child_allocation.y = child->y + border_width; | |
375 | child_allocation.width = child->widget->requisition.width; | |
376 | child_allocation.height = child->widget->requisition.height; | |
377 | gtk_widget_size_allocate (child->widget, &child_allocation); | |
378 | } | |
379 | } | |
380 | } | |
381 | ||
382 | static void | |
383 | gtk_myfixed_paint (GtkWidget *widget, | |
384 | GdkRectangle *area) | |
385 | { | |
386 | g_return_if_fail (widget != NULL); | |
387 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
388 | g_return_if_fail (area != NULL); | |
389 | ||
390 | if (GTK_WIDGET_DRAWABLE (widget)) | |
391 | gdk_window_clear_area (widget->window, | |
392 | area->x, area->y, | |
393 | area->width, area->height); | |
394 | } | |
395 | ||
396 | static void | |
397 | gtk_myfixed_draw (GtkWidget *widget, | |
398 | GdkRectangle *area) | |
399 | { | |
400 | GtkMyFixed *myfixed; | |
401 | GtkMyFixedChild *child; | |
402 | GdkRectangle child_area; | |
403 | GList *children; | |
404 | ||
405 | g_return_if_fail (widget != NULL); | |
406 | g_return_if_fail (GTK_IS_MYFIXED (widget)); | |
407 | ||
408 | if (GTK_WIDGET_DRAWABLE (widget)) | |
409 | { | |
410 | myfixed = GTK_MYFIXED (widget); | |
411 | gtk_myfixed_paint (widget, area); | |
412 | ||
413 | children = myfixed->children; | |
414 | while (children) | |
415 | { | |
416 | child = children->data; | |
417 | children = children->next; | |
418 | ||
419 | if (gtk_widget_intersect (child->widget, area, &child_area)) | |
420 | gtk_widget_draw (child->widget, &child_area); | |
421 | } | |
422 | } | |
423 | } | |
424 | ||
425 | static gint | |
426 | gtk_myfixed_expose (GtkWidget *widget, | |
427 | GdkEventExpose *event) | |
428 | { | |
429 | GtkMyFixed *myfixed; | |
430 | GtkMyFixedChild *child; | |
431 | GdkEventExpose child_event; | |
432 | GList *children; | |
433 | ||
434 | g_return_val_if_fail (widget != NULL, FALSE); | |
435 | g_return_val_if_fail (GTK_IS_MYFIXED (widget), FALSE); | |
436 | g_return_val_if_fail (event != NULL, FALSE); | |
437 | ||
438 | if (GTK_WIDGET_DRAWABLE (widget)) | |
439 | { | |
440 | myfixed = GTK_MYFIXED (widget); | |
441 | ||
442 | child_event = *event; | |
443 | ||
444 | children = myfixed->children; | |
445 | while (children) | |
446 | { | |
447 | child = children->data; | |
448 | children = children->next; | |
449 | ||
450 | if (GTK_WIDGET_NO_WINDOW (child->widget) && | |
451 | gtk_widget_intersect (child->widget, &event->area, | |
452 | &child_event.area)) | |
453 | gtk_widget_event (child->widget, (GdkEvent*) &child_event); | |
454 | } | |
455 | } | |
456 | ||
457 | return FALSE; | |
458 | } | |
459 | ||
460 | static void | |
461 | gtk_myfixed_add (GtkContainer *container, | |
462 | GtkWidget *widget) | |
463 | { | |
464 | g_return_if_fail (container != NULL); | |
465 | g_return_if_fail (GTK_IS_MYFIXED (container)); | |
466 | g_return_if_fail (widget != NULL); | |
467 | ||
468 | gtk_myfixed_put (GTK_MYFIXED (container), widget, 0, 0); | |
469 | } | |
470 | ||
471 | static void | |
472 | gtk_myfixed_remove (GtkContainer *container, | |
473 | GtkWidget *widget) | |
474 | { | |
475 | GtkMyFixed *myfixed; | |
476 | GtkMyFixedChild *child; | |
477 | GList *children; | |
478 | ||
479 | g_return_if_fail (container != NULL); | |
480 | g_return_if_fail (GTK_IS_MYFIXED (container)); | |
481 | g_return_if_fail (widget != NULL); | |
482 | ||
483 | myfixed = GTK_MYFIXED (container); | |
484 | ||
485 | children = myfixed->children; | |
486 | while (children) | |
487 | { | |
488 | child = children->data; | |
489 | ||
490 | if (child->widget == widget) | |
491 | { | |
492 | gtk_widget_unparent (widget); | |
493 | ||
494 | myfixed->children = g_list_remove_link (myfixed->children, children); | |
495 | g_list_free (children); | |
496 | g_free (child); | |
497 | ||
498 | if (GTK_WIDGET_VISIBLE (widget) && GTK_WIDGET_VISIBLE (container)) | |
499 | gtk_widget_queue_resize (GTK_WIDGET (container)); | |
500 | ||
501 | break; | |
502 | } | |
503 | ||
504 | children = children->next; | |
505 | } | |
506 | } | |
507 | ||
508 | static void | |
509 | gtk_myfixed_foreach (GtkContainer *container, | |
75ed1d15 GL |
510 | #if (GTK_MINOR_VERSION == 1) |
511 | gboolean include_internals, | |
512 | #endif | |
c801d85f KB |
513 | GtkCallback callback, |
514 | gpointer callback_data) | |
515 | { | |
516 | GtkMyFixed *myfixed; | |
517 | GtkMyFixedChild *child; | |
518 | GList *children; | |
519 | ||
520 | g_return_if_fail (container != NULL); | |
521 | g_return_if_fail (GTK_IS_MYFIXED (container)); | |
522 | g_return_if_fail (callback != NULL); | |
523 | ||
524 | myfixed = GTK_MYFIXED (container); | |
525 | ||
526 | children = myfixed->children; | |
527 | while (children) | |
528 | { | |
529 | child = children->data; | |
530 | children = children->next; | |
531 | ||
532 | (* callback) (child->widget, callback_data); | |
533 | } | |
534 | } | |
535 | ||
536 | ||
537 | #ifdef __cplusplus | |
538 | } | |
539 | #endif /* __cplusplus */ | |
540 |