More RTL work. Now transform m_updateRegion and
[wxWidgets.git] / src / gtk / window.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/window.cpp
3 // Purpose:
4 // Author: Robert Roebling
5 // Id: $Id$
6 // Copyright: (c) 1998 Robert Roebling, Julian Smart
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
9
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
12
13 #ifdef __VMS
14 #define XWarpPointer XWARPPOINTER
15 #endif
16
17 #include "wx/window.h"
18
19 #ifndef WX_PRECOMP
20 #include "wx/intl.h"
21 #include "wx/log.h"
22 #include "wx/app.h"
23 #include "wx/utils.h"
24 #include "wx/frame.h"
25 #include "wx/dcclient.h"
26 #include "wx/menu.h"
27 #include "wx/dialog.h"
28 #include "wx/settings.h"
29 #include "wx/msgdlg.h"
30 #include "wx/textctrl.h"
31 #include "wx/toolbar.h"
32 #include "wx/combobox.h"
33 #include "wx/layout.h"
34 #include "wx/statusbr.h"
35 #include "wx/math.h"
36 #include "wx/module.h"
37 #endif
38
39 #if wxUSE_DRAG_AND_DROP
40 #include "wx/dnd.h"
41 #endif
42
43 #if wxUSE_TOOLTIPS
44 #include "wx/tooltip.h"
45 #endif
46
47 #if wxUSE_CARET
48 #include "wx/caret.h"
49 #endif // wxUSE_CARET
50
51 #include "wx/fontutil.h"
52
53 #ifdef __WXDEBUG__
54 #include "wx/thread.h"
55 #endif
56
57 #include <ctype.h>
58
59 // FIXME: Due to a hack we use GtkCombo in here, which is deprecated since gtk2.3.0
60 #include <gtk/gtkversion.h>
61 #if defined(GTK_DISABLE_DEPRECATED) && GTK_CHECK_VERSION(2,3,0)
62 #undef GTK_DISABLE_DEPRECATED
63 #endif
64
65 #include "wx/gtk/private.h"
66 #include <gdk/gdkprivate.h>
67 #include <gdk/gdkkeysyms.h>
68 #include <gdk/gdkx.h>
69
70 #include <gtk/gtk.h>
71 #include <gtk/gtkprivate.h>
72
73 #include "wx/gtk/win_gtk.h"
74
75 #include <pango/pangox.h>
76
77 #ifdef HAVE_XIM
78 #undef HAVE_XIM
79 #endif
80
81 //-----------------------------------------------------------------------------
82 // documentation on internals
83 //-----------------------------------------------------------------------------
84
85 /*
86 I have been asked several times about writing some documentation about
87 the GTK port of wxWidgets, especially its internal structures. Obviously,
88 you cannot understand wxGTK without knowing a little about the GTK, but
89 some more information about what the wxWindow, which is the base class
90 for all other window classes, does seems required as well.
91
92 I)
93
94 What does wxWindow do? It contains the common interface for the following
95 jobs of its descendants:
96
97 1) Define the rudimentary behaviour common to all window classes, such as
98 resizing, intercepting user input (so as to make it possible to use these
99 events for special purposes in a derived class), window names etc.
100
101 2) Provide the possibility to contain and manage children, if the derived
102 class is allowed to contain children, which holds true for those window
103 classes which do not display a native GTK widget. To name them, these
104 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
105 work classes are a special case and are handled a bit differently from
106 the rest. The same holds true for the wxNotebook class.
107
108 3) Provide the possibility to draw into a client area of a window. This,
109 too, only holds true for classes that do not display a native GTK widget
110 as above.
111
112 4) Provide the entire mechanism for scrolling widgets. This actual inter-
113 face for this is usually in wxScrolledWindow, but the GTK implementation
114 is in this class.
115
116 5) A multitude of helper or extra methods for special purposes, such as
117 Drag'n'Drop, managing validators etc.
118
119 6) Display a border (sunken, raised, simple or none).
120
121 Normally one might expect, that one wxWidgets window would always correspond
122 to one GTK widget. Under GTK, there is no such all-round widget that has all
123 the functionality. Moreover, the GTK defines a client area as a different
124 widget from the actual widget you are handling. Last but not least some
125 special classes (e.g. wxFrame) handle different categories of widgets and
126 still have the possibility to draw something in the client area.
127 It was therefore required to write a special purpose GTK widget, that would
128 represent a client area in the sense of wxWidgets capable to do the jobs
129 2), 3) and 4). I have written this class and it resides in win_gtk.c of
130 this directory.
131
132 All windows must have a widget, with which they interact with other under-
133 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
134 the wxWindow class has a member variable called m_widget which holds a
135 pointer to this widget. When the window class represents a GTK native widget,
136 this is (in most cases) the only GTK widget the class manages. E.g. the
137 wxStaticText class handles only a GtkLabel widget a pointer to which you
138 can find in m_widget (defined in wxWindow)
139
140 When the class has a client area for drawing into and for containing children
141 it has to handle the client area widget (of the type GtkPizza, defined in
142 win_gtk.c), but there could be any number of widgets, handled by a class
143 The common rule for all windows is only, that the widget that interacts with
144 the rest of GTK must be referenced in m_widget and all other widgets must be
145 children of this widget on the GTK level. The top-most widget, which also
146 represents the client area, must be in the m_wxwindow field and must be of
147 the type GtkPizza.
148
149 As I said, the window classes that display a GTK native widget only have
150 one widget, so in the case of e.g. the wxButton class m_widget holds a
151 pointer to a GtkButton widget. But windows with client areas (for drawing
152 and children) have a m_widget field that is a pointer to a GtkScrolled-
153 Window and a m_wxwindow field that is pointer to a GtkPizza and this
154 one is (in the GTK sense) a child of the GtkScrolledWindow.
155
156 If the m_wxwindow field is set, then all input to this widget is inter-
157 cepted and sent to the wxWidgets class. If not, all input to the widget
158 that gets pointed to by m_widget gets intercepted and sent to the class.
159
160 II)
161
162 The design of scrolling in wxWidgets is markedly different from that offered
163 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
164 clicking on a scrollbar belonging to scrolled window will inevitably move
165 the window. In wxWidgets, the scrollbar will only emit an event, send this
166 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
167 which actually moves the window and its sub-windows. Note that GtkPizza
168 memorizes how much it has been scrolled but that wxWidgets forgets this
169 so that the two coordinates systems have to be kept in synch. This is done
170 in various places using the pizza->xoffset and pizza->yoffset values.
171
172 III)
173
174 Singularly the most broken code in GTK is the code that is supposed to
175 inform subwindows (child windows) about new positions. Very often, duplicate
176 events are sent without changes in size or position, equally often no
177 events are sent at all (All this is due to a bug in the GtkContainer code
178 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
179 GTK's own system and it simply waits for size events for toplevel windows
180 and then iterates down the respective size events to all window. This has
181 the disadvantage that windows might get size events before the GTK widget
182 actually has the reported size. This doesn't normally pose any problem, but
183 the OpenGL drawing routines rely on correct behaviour. Therefore, I have
184 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
185 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
186 window that is used for OpenGL output really has that size (as reported by
187 GTK).
188
189 IV)
190
191 If someone at some point of time feels the immense desire to have a look at,
192 change or attempt to optimise the Refresh() logic, this person will need an
193 intimate understanding of what "draw" and "expose" events are and what
194 they are used for, in particular when used in connection with GTK's
195 own windowless widgets. Beware.
196
197 V)
198
199 Cursors, too, have been a constant source of pleasure. The main difficulty
200 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
201 for the parent. To prevent this from doing too much harm, I use idle time
202 to set the cursor over and over again, starting from the toplevel windows
203 and ending with the youngest generation (speaking of parent and child windows).
204 Also don't forget that cursors (like much else) are connected to GdkWindows,
205 not GtkWidgets and that the "window" field of a GtkWidget might very well
206 point to the GdkWindow of the parent widget (-> "window-less widget") and
207 that the two obviously have very different meanings.
208
209 */
210
211 //-----------------------------------------------------------------------------
212 // data
213 //-----------------------------------------------------------------------------
214
215 extern bool g_blockEventsOnDrag;
216 extern bool g_blockEventsOnScroll;
217 extern wxCursor g_globalCursor;
218
219 // mouse capture state: the window which has it and if the mouse is currently
220 // inside it
221 static wxWindowGTK *g_captureWindow = (wxWindowGTK*) NULL;
222 static bool g_captureWindowHasMouse = false;
223
224 wxWindowGTK *g_focusWindow = (wxWindowGTK*) NULL;
225
226 // the last window which had the focus - this is normally never NULL (except
227 // if we never had focus at all) as even when g_focusWindow is NULL it still
228 // keeps its previous value
229 wxWindowGTK *g_focusWindowLast = (wxWindowGTK*) NULL;
230
231 // If a window get the focus set but has not been realized
232 // yet, defer setting the focus to idle time.
233 wxWindowGTK *g_delayedFocus = (wxWindowGTK*) NULL;
234
235 extern bool g_mainThreadLocked;
236
237 //-----------------------------------------------------------------------------
238 // debug
239 //-----------------------------------------------------------------------------
240
241 #ifdef __WXDEBUG__
242
243 #if wxUSE_THREADS
244 # define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance");
245 #else
246 # define DEBUG_MAIN_THREAD
247 #endif
248 #else
249 #define DEBUG_MAIN_THREAD
250 #endif // Debug
251
252 // the trace mask used for the focus debugging messages
253 #define TRACE_FOCUS _T("focus")
254
255 //-----------------------------------------------------------------------------
256 // missing gdk functions
257 //-----------------------------------------------------------------------------
258
259 void
260 gdk_window_warp_pointer (GdkWindow *window,
261 gint x,
262 gint y)
263 {
264 if (!window)
265 window = gdk_get_default_root_window();
266
267 if (!GDK_WINDOW_DESTROYED(window))
268 {
269 XWarpPointer (GDK_WINDOW_XDISPLAY(window),
270 None, /* not source window -> move from anywhere */
271 GDK_WINDOW_XID(window), /* dest window */
272 0, 0, 0, 0, /* not source window -> move from anywhere */
273 x, y );
274 }
275 }
276
277 //-----------------------------------------------------------------------------
278 // local code (see below)
279 //-----------------------------------------------------------------------------
280
281 // returns the child of win which currently has focus or NULL if not found
282 //
283 // Note: can't be static, needed by textctrl.cpp.
284 wxWindow *wxFindFocusedChild(wxWindowGTK *win)
285 {
286 wxWindow *winFocus = wxWindowGTK::FindFocus();
287 if ( !winFocus )
288 return (wxWindow *)NULL;
289
290 if ( winFocus == win )
291 return (wxWindow *)win;
292
293 for ( wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst();
294 node;
295 node = node->GetNext() )
296 {
297 wxWindow *child = wxFindFocusedChild(node->GetData());
298 if ( child )
299 return child;
300 }
301
302 return (wxWindow *)NULL;
303 }
304
305 static void GetScrollbarWidth(GtkWidget* widget, int& w, int& h)
306 {
307 GtkScrolledWindow* scroll_window = GTK_SCROLLED_WINDOW(widget);
308 GtkScrolledWindowClass* scroll_class = GTK_SCROLLED_WINDOW_CLASS(GTK_OBJECT_GET_CLASS(scroll_window));
309 GtkRequisition scroll_req;
310
311 w = 0;
312 if (scroll_window->vscrollbar_visible)
313 {
314 scroll_req.width = 2;
315 scroll_req.height = 2;
316 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->vscrollbar) )->size_request )
317 (scroll_window->vscrollbar, &scroll_req );
318 w = scroll_req.width +
319 scroll_class->scrollbar_spacing;
320 }
321
322 h = 0;
323 if (scroll_window->hscrollbar_visible)
324 {
325 scroll_req.width = 2;
326 scroll_req.height = 2;
327 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->hscrollbar) )->size_request )
328 (scroll_window->hscrollbar, &scroll_req );
329 h = scroll_req.height +
330 scroll_class->scrollbar_spacing;
331 }
332 }
333
334 static void draw_frame( GtkWidget *widget, wxWindowGTK *win )
335 {
336 // wxUniversal widgets draw the borders and scrollbars themselves
337 #ifndef __WXUNIVERSAL__
338 if (!win->m_hasVMT)
339 return;
340
341 int dx = 0;
342 int dy = 0;
343 if (GTK_WIDGET_NO_WINDOW (widget))
344 {
345 dx += widget->allocation.x;
346 dy += widget->allocation.y;
347 }
348
349 int x = dx;
350 int y = dy;
351
352 int dw = 0;
353 int dh = 0;
354 if (win->m_hasScrolling)
355 {
356 GetScrollbarWidth(widget, dw, dh);
357
358 if (win->GetLayoutDirection() == wxLayout_RightToLeft)
359 {
360 // This is actually wrong for old GTK+ version
361 // which do not display the scrollbar on the
362 // left side in RTL
363 x += dw;
364 }
365 }
366
367 int w = widget->allocation.width-dw;
368 int h = widget->allocation.height-dh;
369
370 if (win->HasFlag(wxRAISED_BORDER))
371 {
372 gtk_paint_shadow (widget->style,
373 widget->window,
374 GTK_STATE_NORMAL,
375 GTK_SHADOW_OUT,
376 NULL, NULL, NULL, // FIXME: No clipping?
377 x, y, w, h );
378 return;
379 }
380
381 if (win->HasFlag(wxSUNKEN_BORDER))
382 {
383 gtk_paint_shadow (widget->style,
384 widget->window,
385 GTK_STATE_NORMAL,
386 GTK_SHADOW_IN,
387 NULL, NULL, NULL, // FIXME: No clipping?
388 x, y, w, h );
389 return;
390 }
391
392 if (win->HasFlag(wxSIMPLE_BORDER))
393 {
394 GdkGC *gc;
395 gc = gdk_gc_new( widget->window );
396 gdk_gc_set_foreground( gc, &widget->style->black );
397 gdk_draw_rectangle( widget->window, gc, FALSE, x, y, w-1, h-1 );
398 g_object_unref (gc);
399 return;
400 }
401 #endif // __WXUNIVERSAL__
402 }
403
404 //-----------------------------------------------------------------------------
405 // "expose_event" of m_widget
406 //-----------------------------------------------------------------------------
407
408 extern "C" {
409 static gboolean
410 gtk_window_own_expose_callback( GtkWidget *widget,
411 GdkEventExpose *gdk_event,
412 wxWindowGTK *win )
413 {
414 if (gdk_event->count == 0)
415 draw_frame(widget, win);
416 return false;
417 }
418 }
419
420 //-----------------------------------------------------------------------------
421 // "size_request" of m_widget
422 //-----------------------------------------------------------------------------
423
424 // make it extern because wxStaticText needs to disconnect this one
425 extern "C" {
426 void wxgtk_window_size_request_callback(GtkWidget *widget,
427 GtkRequisition *requisition,
428 wxWindow *win)
429 {
430 int w, h;
431 win->GetSize( &w, &h );
432 if (w < 2)
433 w = 2;
434 if (h < 2)
435 h = 2;
436
437 requisition->height = h;
438 requisition->width = w;
439 }
440 }
441
442 extern "C" {
443 static
444 void wxgtk_combo_size_request_callback(GtkWidget *widget,
445 GtkRequisition *requisition,
446 wxComboBox *win)
447 {
448 // This callback is actually hooked into the text entry
449 // of the combo box, not the GtkHBox.
450
451 int w, h;
452 win->GetSize( &w, &h );
453 if (w < 2)
454 w = 2;
455 if (h < 2)
456 h = 2;
457
458 GtkCombo *gcombo = GTK_COMBO(win->m_widget);
459
460 GtkRequisition entry_req;
461 entry_req.width = 2;
462 entry_req.height = 2;
463 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(gcombo->button) )->size_request )
464 (gcombo->button, &entry_req );
465
466 requisition->width = w - entry_req.width;
467 requisition->height = entry_req.height;
468 }
469 }
470
471 //-----------------------------------------------------------------------------
472 // "expose_event" of m_wxwindow
473 //-----------------------------------------------------------------------------
474
475 extern "C" {
476 static gboolean
477 gtk_window_expose_callback( GtkWidget *widget,
478 GdkEventExpose *gdk_event,
479 wxWindow *win )
480 {
481 DEBUG_MAIN_THREAD
482
483 // don't need to install idle handler, its done from "event" signal
484
485 // This callback gets called in drawing-idle time under
486 // GTK 2.0, so we don't need to defer anything to idle
487 // time anymore.
488
489 GtkPizza *pizza = GTK_PIZZA( widget );
490 if (gdk_event->window != pizza->bin_window) return FALSE;
491
492
493 #if 0
494 if (win->GetName())
495 {
496 wxPrintf( wxT("OnExpose from ") );
497 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
498 wxPrintf( win->GetClassInfo()->GetClassName() );
499 wxPrintf( wxT(" %d %d %d %d\n"), (int)gdk_event->area.x,
500 (int)gdk_event->area.y,
501 (int)gdk_event->area.width,
502 (int)gdk_event->area.height );
503 }
504
505 gtk_paint_box
506 (
507 win->m_wxwindow->style,
508 pizza->bin_window,
509 GTK_STATE_NORMAL,
510 GTK_SHADOW_OUT,
511 (GdkRectangle*) NULL,
512 win->m_wxwindow,
513 (char *)"button", // const_cast
514 20,20,24,24
515 );
516 #endif
517
518 win->GetUpdateRegion() = wxRegion( gdk_event->region );
519
520 win->GtkSendPaintEvents();
521
522 // Let parent window draw window-less widgets
523 return FALSE;
524 }
525 }
526
527 //-----------------------------------------------------------------------------
528 // "key_press_event" from any window
529 //-----------------------------------------------------------------------------
530
531 // These are used when transforming Ctrl-alpha to ascii values 1-26
532 inline bool wxIsLowerChar(int code)
533 {
534 return (code >= 'a' && code <= 'z' );
535 }
536
537 inline bool wxIsUpperChar(int code)
538 {
539 return (code >= 'A' && code <= 'Z' );
540 }
541
542
543 // set WXTRACE to this to see the key event codes on the console
544 #define TRACE_KEYS _T("keyevent")
545
546 // translates an X key symbol to WXK_XXX value
547 //
548 // if isChar is true it means that the value returned will be used for EVT_CHAR
549 // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
550 // for example, while if it is false it means that the value is going to be
551 // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
552 // WXK_NUMPAD_DIVIDE
553 static long wxTranslateKeySymToWXKey(KeySym keysym, bool isChar)
554 {
555 long key_code;
556
557 switch ( keysym )
558 {
559 // Shift, Control and Alt don't generate the CHAR events at all
560 case GDK_Shift_L:
561 case GDK_Shift_R:
562 key_code = isChar ? 0 : WXK_SHIFT;
563 break;
564 case GDK_Control_L:
565 case GDK_Control_R:
566 key_code = isChar ? 0 : WXK_CONTROL;
567 break;
568 case GDK_Meta_L:
569 case GDK_Meta_R:
570 case GDK_Alt_L:
571 case GDK_Alt_R:
572 case GDK_Super_L:
573 case GDK_Super_R:
574 key_code = isChar ? 0 : WXK_ALT;
575 break;
576
577 // neither do the toggle modifies
578 case GDK_Scroll_Lock:
579 key_code = isChar ? 0 : WXK_SCROLL;
580 break;
581
582 case GDK_Caps_Lock:
583 key_code = isChar ? 0 : WXK_CAPITAL;
584 break;
585
586 case GDK_Num_Lock:
587 key_code = isChar ? 0 : WXK_NUMLOCK;
588 break;
589
590
591 // various other special keys
592 case GDK_Menu:
593 key_code = WXK_MENU;
594 break;
595
596 case GDK_Help:
597 key_code = WXK_HELP;
598 break;
599
600 case GDK_BackSpace:
601 key_code = WXK_BACK;
602 break;
603
604 case GDK_ISO_Left_Tab:
605 case GDK_Tab:
606 key_code = WXK_TAB;
607 break;
608
609 case GDK_Linefeed:
610 case GDK_Return:
611 key_code = WXK_RETURN;
612 break;
613
614 case GDK_Clear:
615 key_code = WXK_CLEAR;
616 break;
617
618 case GDK_Pause:
619 key_code = WXK_PAUSE;
620 break;
621
622 case GDK_Select:
623 key_code = WXK_SELECT;
624 break;
625
626 case GDK_Print:
627 key_code = WXK_PRINT;
628 break;
629
630 case GDK_Execute:
631 key_code = WXK_EXECUTE;
632 break;
633
634 case GDK_Escape:
635 key_code = WXK_ESCAPE;
636 break;
637
638 // cursor and other extended keyboard keys
639 case GDK_Delete:
640 key_code = WXK_DELETE;
641 break;
642
643 case GDK_Home:
644 key_code = WXK_HOME;
645 break;
646
647 case GDK_Left:
648 key_code = WXK_LEFT;
649 break;
650
651 case GDK_Up:
652 key_code = WXK_UP;
653 break;
654
655 case GDK_Right:
656 key_code = WXK_RIGHT;
657 break;
658
659 case GDK_Down:
660 key_code = WXK_DOWN;
661 break;
662
663 case GDK_Prior: // == GDK_Page_Up
664 key_code = WXK_PAGEUP;
665 break;
666
667 case GDK_Next: // == GDK_Page_Down
668 key_code = WXK_PAGEDOWN;
669 break;
670
671 case GDK_End:
672 key_code = WXK_END;
673 break;
674
675 case GDK_Begin:
676 key_code = WXK_HOME;
677 break;
678
679 case GDK_Insert:
680 key_code = WXK_INSERT;
681 break;
682
683
684 // numpad keys
685 case GDK_KP_0:
686 case GDK_KP_1:
687 case GDK_KP_2:
688 case GDK_KP_3:
689 case GDK_KP_4:
690 case GDK_KP_5:
691 case GDK_KP_6:
692 case GDK_KP_7:
693 case GDK_KP_8:
694 case GDK_KP_9:
695 key_code = (isChar ? '0' : WXK_NUMPAD0) + keysym - GDK_KP_0;
696 break;
697
698 case GDK_KP_Space:
699 key_code = isChar ? ' ' : WXK_NUMPAD_SPACE;
700 break;
701
702 case GDK_KP_Tab:
703 key_code = isChar ? WXK_TAB : WXK_NUMPAD_TAB;
704 break;
705
706 case GDK_KP_Enter:
707 key_code = isChar ? WXK_RETURN : WXK_NUMPAD_ENTER;
708 break;
709
710 case GDK_KP_F1:
711 key_code = isChar ? WXK_F1 : WXK_NUMPAD_F1;
712 break;
713
714 case GDK_KP_F2:
715 key_code = isChar ? WXK_F2 : WXK_NUMPAD_F2;
716 break;
717
718 case GDK_KP_F3:
719 key_code = isChar ? WXK_F3 : WXK_NUMPAD_F3;
720 break;
721
722 case GDK_KP_F4:
723 key_code = isChar ? WXK_F4 : WXK_NUMPAD_F4;
724 break;
725
726 case GDK_KP_Home:
727 key_code = isChar ? WXK_HOME : WXK_NUMPAD_HOME;
728 break;
729
730 case GDK_KP_Left:
731 key_code = isChar ? WXK_LEFT : WXK_NUMPAD_LEFT;
732 break;
733
734 case GDK_KP_Up:
735 key_code = isChar ? WXK_UP : WXK_NUMPAD_UP;
736 break;
737
738 case GDK_KP_Right:
739 key_code = isChar ? WXK_RIGHT : WXK_NUMPAD_RIGHT;
740 break;
741
742 case GDK_KP_Down:
743 key_code = isChar ? WXK_DOWN : WXK_NUMPAD_DOWN;
744 break;
745
746 case GDK_KP_Prior: // == GDK_KP_Page_Up
747 key_code = isChar ? WXK_PAGEUP : WXK_NUMPAD_PAGEUP;
748 break;
749
750 case GDK_KP_Next: // == GDK_KP_Page_Down
751 key_code = isChar ? WXK_PAGEDOWN : WXK_NUMPAD_PAGEDOWN;
752 break;
753
754 case GDK_KP_End:
755 key_code = isChar ? WXK_END : WXK_NUMPAD_END;
756 break;
757
758 case GDK_KP_Begin:
759 key_code = isChar ? WXK_HOME : WXK_NUMPAD_BEGIN;
760 break;
761
762 case GDK_KP_Insert:
763 key_code = isChar ? WXK_INSERT : WXK_NUMPAD_INSERT;
764 break;
765
766 case GDK_KP_Delete:
767 key_code = isChar ? WXK_DELETE : WXK_NUMPAD_DELETE;
768 break;
769
770 case GDK_KP_Equal:
771 key_code = isChar ? '=' : WXK_NUMPAD_EQUAL;
772 break;
773
774 case GDK_KP_Multiply:
775 key_code = isChar ? '*' : WXK_NUMPAD_MULTIPLY;
776 break;
777
778 case GDK_KP_Add:
779 key_code = isChar ? '+' : WXK_NUMPAD_ADD;
780 break;
781
782 case GDK_KP_Separator:
783 // FIXME: what is this?
784 key_code = isChar ? '.' : WXK_NUMPAD_SEPARATOR;
785 break;
786
787 case GDK_KP_Subtract:
788 key_code = isChar ? '-' : WXK_NUMPAD_SUBTRACT;
789 break;
790
791 case GDK_KP_Decimal:
792 key_code = isChar ? '.' : WXK_NUMPAD_DECIMAL;
793 break;
794
795 case GDK_KP_Divide:
796 key_code = isChar ? '/' : WXK_NUMPAD_DIVIDE;
797 break;
798
799
800 // function keys
801 case GDK_F1:
802 case GDK_F2:
803 case GDK_F3:
804 case GDK_F4:
805 case GDK_F5:
806 case GDK_F6:
807 case GDK_F7:
808 case GDK_F8:
809 case GDK_F9:
810 case GDK_F10:
811 case GDK_F11:
812 case GDK_F12:
813 key_code = WXK_F1 + keysym - GDK_F1;
814 break;
815
816 default:
817 key_code = 0;
818 }
819
820 return key_code;
821 }
822
823 static inline bool wxIsAsciiKeysym(KeySym ks)
824 {
825 return ks < 256;
826 }
827
828 static void wxFillOtherKeyEventFields(wxKeyEvent& event,
829 wxWindowGTK *win,
830 GdkEventKey *gdk_event)
831 {
832 int x = 0;
833 int y = 0;
834 GdkModifierType state;
835 if (gdk_event->window)
836 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
837
838 event.SetTimestamp( gdk_event->time );
839 event.SetId(win->GetId());
840 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0;
841 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0;
842 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0;
843 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK) != 0;
844 event.m_scanCode = gdk_event->keyval;
845 event.m_rawCode = (wxUint32) gdk_event->keyval;
846 event.m_rawFlags = 0;
847 #if wxUSE_UNICODE
848 event.m_uniChar = gdk_keyval_to_unicode(gdk_event->keyval);
849 #endif
850 wxGetMousePosition( &x, &y );
851 win->ScreenToClient( &x, &y );
852 event.m_x = x;
853 event.m_y = y;
854 event.SetEventObject( win );
855 }
856
857
858 static bool
859 wxTranslateGTKKeyEventToWx(wxKeyEvent& event,
860 wxWindowGTK *win,
861 GdkEventKey *gdk_event)
862 {
863 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
864 // but only event->keyval which is quite useless to us, so remember
865 // the last character from GDK_KEY_PRESS and reuse it as last resort
866 //
867 // NB: should be MT-safe as we're always called from the main thread only
868 static struct
869 {
870 KeySym keysym;
871 long keycode;
872 } s_lastKeyPress = { 0, 0 };
873
874 KeySym keysym = gdk_event->keyval;
875
876 wxLogTrace(TRACE_KEYS, _T("Key %s event: keysym = %ld"),
877 event.GetEventType() == wxEVT_KEY_UP ? _T("release")
878 : _T("press"),
879 keysym);
880
881 long key_code = wxTranslateKeySymToWXKey(keysym, false /* !isChar */);
882
883 if ( !key_code )
884 {
885 // do we have the translation or is it a plain ASCII character?
886 if ( (gdk_event->length == 1) || wxIsAsciiKeysym(keysym) )
887 {
888 // we should use keysym if it is ASCII as X does some translations
889 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
890 // which we don't want here (but which we do use for OnChar())
891 if ( !wxIsAsciiKeysym(keysym) )
892 {
893 keysym = (KeySym)gdk_event->string[0];
894 }
895
896 // we want to always get the same key code when the same key is
897 // pressed regardless of the state of the modifiers, i.e. on a
898 // standard US keyboard pressing '5' or '%' ('5' key with
899 // Shift) should result in the same key code in OnKeyDown():
900 // '5' (although OnChar() will get either '5' or '%').
901 //
902 // to do it we first translate keysym to keycode (== scan code)
903 // and then back but always using the lower register
904 Display *dpy = (Display *)wxGetDisplay();
905 KeyCode keycode = XKeysymToKeycode(dpy, keysym);
906
907 wxLogTrace(TRACE_KEYS, _T("\t-> keycode %d"), keycode);
908
909 KeySym keysymNormalized = XKeycodeToKeysym(dpy, keycode, 0);
910
911 // use the normalized, i.e. lower register, keysym if we've
912 // got one
913 key_code = keysymNormalized ? keysymNormalized : keysym;
914
915 // as explained above, we want to have lower register key codes
916 // normally but for the letter keys we want to have the upper ones
917 //
918 // NB: don't use XConvertCase() here, we want to do it for letters
919 // only
920 key_code = toupper(key_code);
921 }
922 else // non ASCII key, what to do?
923 {
924 // by default, ignore it
925 key_code = 0;
926
927 // but if we have cached information from the last KEY_PRESS
928 if ( gdk_event->type == GDK_KEY_RELEASE )
929 {
930 // then reuse it
931 if ( keysym == s_lastKeyPress.keysym )
932 {
933 key_code = s_lastKeyPress.keycode;
934 }
935 }
936 }
937
938 if ( gdk_event->type == GDK_KEY_PRESS )
939 {
940 // remember it to be reused for KEY_UP event later
941 s_lastKeyPress.keysym = keysym;
942 s_lastKeyPress.keycode = key_code;
943 }
944 }
945
946 wxLogTrace(TRACE_KEYS, _T("\t-> wxKeyCode %ld"), key_code);
947
948 // sending unknown key events doesn't really make sense
949 if ( !key_code )
950 return false;
951
952 // now fill all the other fields
953 wxFillOtherKeyEventFields(event, win, gdk_event);
954
955 event.m_keyCode = key_code;
956 #if wxUSE_UNICODE
957 if ( gdk_event->type == GDK_KEY_PRESS || gdk_event->type == GDK_KEY_RELEASE )
958 {
959 event.m_uniChar = key_code;
960 }
961 #endif
962
963 return true;
964 }
965
966
967 struct wxGtkIMData
968 {
969 GtkIMContext *context;
970 GdkEventKey *lastKeyEvent;
971
972 wxGtkIMData()
973 {
974 context = gtk_im_multicontext_new();
975 lastKeyEvent = NULL;
976 }
977 ~wxGtkIMData()
978 {
979 g_object_unref (context);
980 }
981 };
982
983 extern "C" {
984 static gboolean
985 gtk_window_key_press_callback( GtkWidget *widget,
986 GdkEventKey *gdk_event,
987 wxWindow *win )
988 {
989 DEBUG_MAIN_THREAD
990
991 // don't need to install idle handler, its done from "event" signal
992
993 if (!win->m_hasVMT)
994 return FALSE;
995 if (g_blockEventsOnDrag)
996 return FALSE;
997
998
999 wxKeyEvent event( wxEVT_KEY_DOWN );
1000 bool ret = false;
1001 bool return_after_IM = false;
1002
1003 if( wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
1004 {
1005 // Emit KEY_DOWN event
1006 ret = win->GetEventHandler()->ProcessEvent( event );
1007 }
1008 else
1009 {
1010 // Return after IM processing as we cannot do
1011 // anything with it anyhow.
1012 return_after_IM = true;
1013 }
1014
1015 // 2005.01.26 modified by Hong Jen Yee (hzysoft@sina.com.tw):
1016 // When we get a key_press event here, it could be originate
1017 // from the current widget or its child widgets. However, only the widget
1018 // with the INPUT FOCUS can generate the INITIAL key_press event. That is,
1019 // if the CURRENT widget doesn't have the FOCUS at all, this event definitely
1020 // originated from its child widgets and shouldn't be passed to IM context.
1021 // In fact, what a GTK+ IM should do is filtering keyEvents and convert them
1022 // into text input ONLY WHEN THE WIDGET HAS INPUT FOCUS. Besides, when current
1023 // widgets has both IM context and input focus, the event should be filtered
1024 // by gtk_im_context_filter_keypress().
1025 // Then, we should, according to GTK+ 2.0 API doc, return whatever it returns.
1026 if ((!ret) && (win->m_imData != NULL) && ( wxWindow::FindFocus() == win ))
1027 {
1028 // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API
1029 // docs, if IM filter returns true, no further processing should be done.
1030 // we should send the key_down event anyway.
1031 bool intercepted_by_IM = gtk_im_context_filter_keypress(win->m_imData->context, gdk_event);
1032 win->m_imData->lastKeyEvent = NULL;
1033 if (intercepted_by_IM)
1034 {
1035 wxLogTrace(TRACE_KEYS, _T("Key event intercepted by IM"));
1036 return TRUE;
1037 }
1038 }
1039
1040 if (return_after_IM)
1041 return FALSE;
1042
1043 #if wxUSE_ACCEL
1044 if (!ret)
1045 {
1046 wxWindowGTK *ancestor = win;
1047 while (ancestor)
1048 {
1049 int command = ancestor->GetAcceleratorTable()->GetCommand( event );
1050 if (command != -1)
1051 {
1052 wxCommandEvent command_event( wxEVT_COMMAND_MENU_SELECTED, command );
1053 ret = ancestor->GetEventHandler()->ProcessEvent( command_event );
1054 break;
1055 }
1056 if (ancestor->IsTopLevel())
1057 break;
1058 ancestor = ancestor->GetParent();
1059 }
1060 }
1061 #endif // wxUSE_ACCEL
1062
1063 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
1064 // will only be sent if it is not in an accelerator table.
1065 if (!ret)
1066 {
1067 long key_code;
1068 KeySym keysym = gdk_event->keyval;
1069 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
1070 key_code = wxTranslateKeySymToWXKey(keysym, true /* isChar */);
1071 if ( !key_code )
1072 {
1073 if ( wxIsAsciiKeysym(keysym) )
1074 {
1075 // ASCII key
1076 key_code = (unsigned char)keysym;
1077 }
1078 // gdk_event->string is actually deprecated
1079 else if ( gdk_event->length == 1 )
1080 {
1081 key_code = (unsigned char)gdk_event->string[0];
1082 }
1083 }
1084
1085 if ( key_code )
1086 {
1087 wxLogTrace(TRACE_KEYS, _T("Char event: %ld"), key_code);
1088
1089 event.m_keyCode = key_code;
1090
1091 // To conform to the docs we need to translate Ctrl-alpha
1092 // characters to values in the range 1-26.
1093 if ( event.ControlDown() &&
1094 ( wxIsLowerChar(key_code) || wxIsUpperChar(key_code) ))
1095 {
1096 if ( wxIsLowerChar(key_code) )
1097 event.m_keyCode = key_code - 'a' + 1;
1098 if ( wxIsUpperChar(key_code) )
1099 event.m_keyCode = key_code - 'A' + 1;
1100 #if wxUSE_UNICODE
1101 event.m_uniChar = event.m_keyCode;
1102 #endif
1103 }
1104
1105 // Implement OnCharHook by checking ancestor top level windows
1106 wxWindow *parent = win;
1107 while (parent && !parent->IsTopLevel())
1108 parent = parent->GetParent();
1109 if (parent)
1110 {
1111 event.SetEventType( wxEVT_CHAR_HOOK );
1112 ret = parent->GetEventHandler()->ProcessEvent( event );
1113 }
1114
1115 if (!ret)
1116 {
1117 event.SetEventType(wxEVT_CHAR);
1118 ret = win->GetEventHandler()->ProcessEvent( event );
1119 }
1120 }
1121 }
1122
1123 // win is a control: tab can be propagated up
1124 if ( !ret &&
1125 ((gdk_event->keyval == GDK_Tab) || (gdk_event->keyval == GDK_ISO_Left_Tab)) &&
1126 // VZ: testing for wxTE_PROCESS_TAB shouldn't be done here - the control may
1127 // have this style, yet choose not to process this particular TAB in which
1128 // case TAB must still work as a navigational character
1129 // JS: enabling again to make consistent with other platforms
1130 // (with wxTE_PROCESS_TAB you have to call Navigate to get default
1131 // navigation behaviour)
1132 #if wxUSE_TEXTCTRL
1133 (! (win->HasFlag(wxTE_PROCESS_TAB) && win->IsKindOf(CLASSINFO(wxTextCtrl)) )) &&
1134 #endif
1135 win->GetParent() && (win->GetParent()->HasFlag( wxTAB_TRAVERSAL)) )
1136 {
1137 wxNavigationKeyEvent new_event;
1138 new_event.SetEventObject( win->GetParent() );
1139 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1140 new_event.SetDirection( (gdk_event->keyval == GDK_Tab) );
1141 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1142 new_event.SetWindowChange( (gdk_event->state & GDK_CONTROL_MASK) );
1143 new_event.SetCurrentFocus( win );
1144 ret = win->GetParent()->GetEventHandler()->ProcessEvent( new_event );
1145 }
1146
1147 return ret;
1148 }
1149 }
1150
1151 extern "C" {
1152 static void
1153 gtk_wxwindow_commit_cb (GtkIMContext *context,
1154 const gchar *str,
1155 wxWindow *window)
1156 {
1157 wxKeyEvent event( wxEVT_KEY_DOWN );
1158
1159 // take modifiers, cursor position, timestamp etc. from the last
1160 // key_press_event that was fed into Input Method:
1161 if (window->m_imData->lastKeyEvent)
1162 {
1163 wxFillOtherKeyEventFields(event,
1164 window, window->m_imData->lastKeyEvent);
1165 }
1166
1167 const wxWxCharBuffer data(wxGTK_CONV_BACK(str));
1168 if( !data )
1169 return;
1170
1171 bool ret = false;
1172
1173 // Implement OnCharHook by checking ancestor top level windows
1174 wxWindow *parent = window;
1175 while (parent && !parent->IsTopLevel())
1176 parent = parent->GetParent();
1177
1178 for( const wxChar* pstr = data; *pstr; pstr++ )
1179 {
1180 #if wxUSE_UNICODE
1181 event.m_uniChar = *pstr;
1182 // Backward compatible for ISO-8859-1
1183 event.m_keyCode = *pstr < 256 ? event.m_uniChar : 0;
1184 wxLogTrace(TRACE_KEYS, _T("IM sent character '%c'"), event.m_uniChar);
1185 #else
1186 event.m_keyCode = *pstr;
1187 #endif // wxUSE_UNICODE
1188
1189 // To conform to the docs we need to translate Ctrl-alpha
1190 // characters to values in the range 1-26.
1191 if ( event.ControlDown() &&
1192 ( wxIsLowerChar(*pstr) || wxIsUpperChar(*pstr) ))
1193 {
1194 if ( wxIsLowerChar(*pstr) )
1195 event.m_keyCode = *pstr - 'a' + 1;
1196 if ( wxIsUpperChar(*pstr) )
1197 event.m_keyCode = *pstr - 'A' + 1;
1198
1199 event.m_keyCode = *pstr - 'a' + 1;
1200 #if wxUSE_UNICODE
1201 event.m_uniChar = event.m_keyCode;
1202 #endif
1203 }
1204
1205 if (parent)
1206 {
1207 event.SetEventType( wxEVT_CHAR_HOOK );
1208 ret = parent->GetEventHandler()->ProcessEvent( event );
1209 }
1210
1211 if (!ret)
1212 {
1213 event.SetEventType(wxEVT_CHAR);
1214 ret = window->GetEventHandler()->ProcessEvent( event );
1215 }
1216 }
1217 }
1218 }
1219
1220
1221 //-----------------------------------------------------------------------------
1222 // "key_release_event" from any window
1223 //-----------------------------------------------------------------------------
1224
1225 extern "C" {
1226 static gboolean
1227 gtk_window_key_release_callback( GtkWidget *widget,
1228 GdkEventKey *gdk_event,
1229 wxWindowGTK *win )
1230 {
1231 DEBUG_MAIN_THREAD
1232
1233 // don't need to install idle handler, its done from "event" signal
1234
1235 if (!win->m_hasVMT)
1236 return FALSE;
1237
1238 if (g_blockEventsOnDrag)
1239 return FALSE;
1240
1241 wxKeyEvent event( wxEVT_KEY_UP );
1242 if ( !wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
1243 {
1244 // unknown key pressed, ignore (the event would be useless anyhow)
1245 return FALSE;
1246 }
1247
1248 return win->GTKProcessEvent(event);
1249 }
1250 }
1251
1252 // ============================================================================
1253 // the mouse events
1254 // ============================================================================
1255
1256 // ----------------------------------------------------------------------------
1257 // mouse event processing helpers
1258 // ----------------------------------------------------------------------------
1259
1260 // init wxMouseEvent with the info from GdkEventXXX struct
1261 template<typename T> void InitMouseEvent(wxWindowGTK *win,
1262 wxMouseEvent& event,
1263 T *gdk_event)
1264 {
1265 event.SetTimestamp( gdk_event->time );
1266 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK);
1267 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK);
1268 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK);
1269 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK);
1270 event.m_leftDown = (gdk_event->state & GDK_BUTTON1_MASK);
1271 event.m_middleDown = (gdk_event->state & GDK_BUTTON2_MASK);
1272 event.m_rightDown = (gdk_event->state & GDK_BUTTON3_MASK);
1273 if (event.GetEventType() == wxEVT_MOUSEWHEEL)
1274 {
1275 event.m_linesPerAction = 3;
1276 event.m_wheelDelta = 120;
1277 if (((GdkEventButton*)gdk_event)->button == 4)
1278 event.m_wheelRotation = 120;
1279 else if (((GdkEventButton*)gdk_event)->button == 5)
1280 event.m_wheelRotation = -120;
1281 }
1282
1283 wxPoint pt = win->GetClientAreaOrigin();
1284 event.m_x = (wxCoord)gdk_event->x - pt.x;
1285 event.m_y = (wxCoord)gdk_event->y - pt.y;
1286
1287 if ((win->m_wxwindow) && (win->GetLayoutDirection() == wxLayout_RightToLeft))
1288 {
1289 // origin in the upper right corner
1290 int window_width = gtk_pizza_get_rtl_offset( GTK_PIZZA(win->m_wxwindow) );
1291 event.m_x = window_width - event.m_x;
1292 }
1293
1294 event.SetEventObject( win );
1295 event.SetId( win->GetId() );
1296 event.SetTimestamp( gdk_event->time );
1297 }
1298
1299 static void AdjustEventButtonState(wxMouseEvent& event)
1300 {
1301 // GDK reports the old state of the button for a button press event, but
1302 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1303 // for a LEFT_DOWN event, not FALSE, so we will invert
1304 // left/right/middleDown for the corresponding click events
1305
1306 if ((event.GetEventType() == wxEVT_LEFT_DOWN) ||
1307 (event.GetEventType() == wxEVT_LEFT_DCLICK) ||
1308 (event.GetEventType() == wxEVT_LEFT_UP))
1309 {
1310 event.m_leftDown = !event.m_leftDown;
1311 return;
1312 }
1313
1314 if ((event.GetEventType() == wxEVT_MIDDLE_DOWN) ||
1315 (event.GetEventType() == wxEVT_MIDDLE_DCLICK) ||
1316 (event.GetEventType() == wxEVT_MIDDLE_UP))
1317 {
1318 event.m_middleDown = !event.m_middleDown;
1319 return;
1320 }
1321
1322 if ((event.GetEventType() == wxEVT_RIGHT_DOWN) ||
1323 (event.GetEventType() == wxEVT_RIGHT_DCLICK) ||
1324 (event.GetEventType() == wxEVT_RIGHT_UP))
1325 {
1326 event.m_rightDown = !event.m_rightDown;
1327 return;
1328 }
1329 }
1330
1331 // find the window to send the mouse event too
1332 static
1333 wxWindowGTK *FindWindowForMouseEvent(wxWindowGTK *win, wxCoord& x, wxCoord& y)
1334 {
1335 wxCoord xx = x;
1336 wxCoord yy = y;
1337
1338 if (win->m_wxwindow)
1339 {
1340 GtkPizza *pizza = GTK_PIZZA(win->m_wxwindow);
1341 xx += gtk_pizza_get_xoffset( pizza );
1342 yy += gtk_pizza_get_yoffset( pizza );
1343 }
1344
1345 wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst();
1346 while (node)
1347 {
1348 wxWindowGTK *child = node->GetData();
1349
1350 node = node->GetNext();
1351 if (!child->IsShown())
1352 continue;
1353
1354 if (child->IsTransparentForMouse())
1355 {
1356 // wxStaticBox is transparent in the box itself
1357 int xx1 = child->m_x;
1358 int yy1 = child->m_y;
1359 int xx2 = child->m_x + child->m_width;
1360 int yy2 = child->m_y + child->m_height;
1361
1362 // left
1363 if (((xx >= xx1) && (xx <= xx1+10) && (yy >= yy1) && (yy <= yy2)) ||
1364 // right
1365 ((xx >= xx2-10) && (xx <= xx2) && (yy >= yy1) && (yy <= yy2)) ||
1366 // top
1367 ((xx >= xx1) && (xx <= xx2) && (yy >= yy1) && (yy <= yy1+10)) ||
1368 // bottom
1369 ((xx >= xx1) && (xx <= xx2) && (yy >= yy2-1) && (yy <= yy2)))
1370 {
1371 win = child;
1372 x -= child->m_x;
1373 y -= child->m_y;
1374 break;
1375 }
1376
1377 }
1378 else
1379 {
1380 if ((child->m_wxwindow == (GtkWidget*) NULL) &&
1381 (child->m_x <= xx) &&
1382 (child->m_y <= yy) &&
1383 (child->m_x+child->m_width >= xx) &&
1384 (child->m_y+child->m_height >= yy))
1385 {
1386 win = child;
1387 x -= child->m_x;
1388 y -= child->m_y;
1389 break;
1390 }
1391 }
1392 }
1393
1394 return win;
1395 }
1396
1397 // ----------------------------------------------------------------------------
1398 // common event handlers helpers
1399 // ----------------------------------------------------------------------------
1400
1401 bool wxWindowGTK::GTKProcessEvent(wxEvent& event) const
1402 {
1403 // nothing special at this level
1404 return GetEventHandler()->ProcessEvent(event);
1405 }
1406
1407 int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny *event) const
1408 {
1409 DEBUG_MAIN_THREAD
1410
1411 // don't need to install idle handler, its done from "event" signal
1412
1413 if (!m_hasVMT)
1414 return FALSE;
1415 if (g_blockEventsOnDrag)
1416 return TRUE;
1417 if (g_blockEventsOnScroll)
1418 return TRUE;
1419
1420 if (!GTKIsOwnWindow(event->window))
1421 return FALSE;
1422
1423 return -1;
1424 }
1425
1426 // overloads for all GDK event types we use here: we need to have this as
1427 // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact,
1428 // derives from it in the sense that the structs have the same layout
1429 #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \
1430 static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \
1431 { \
1432 return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \
1433 }
1434
1435 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton)
1436 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion)
1437 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing)
1438
1439 #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD
1440
1441 #define wxCOMMON_CALLBACK_PROLOGUE(event, win) \
1442 const int rc = wxGtkCallbackCommonPrologue(event, win); \
1443 if ( rc != -1 ) \
1444 return rc
1445
1446 // send the wxChildFocusEvent and wxFocusEvent, common code of
1447 // gtk_window_focus_in_callback() and SetFocus()
1448 static bool DoSendFocusEvents(wxWindow *win)
1449 {
1450 // Notify the parent keeping track of focus for the kbd navigation
1451 // purposes that we got it.
1452 wxChildFocusEvent eventChildFocus(win);
1453 (void)win->GetEventHandler()->ProcessEvent(eventChildFocus);
1454
1455 wxFocusEvent eventFocus(wxEVT_SET_FOCUS, win->GetId());
1456 eventFocus.SetEventObject(win);
1457
1458 return win->GetEventHandler()->ProcessEvent(eventFocus);
1459 }
1460
1461 // all event handlers must have C linkage as they're called from GTK+ C code
1462 extern "C"
1463 {
1464
1465 //-----------------------------------------------------------------------------
1466 // "button_press_event"
1467 //-----------------------------------------------------------------------------
1468
1469 static gboolean
1470 gtk_window_button_press_callback( GtkWidget *widget,
1471 GdkEventButton *gdk_event,
1472 wxWindowGTK *win )
1473 {
1474 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
1475
1476 if (win->m_wxwindow && (g_focusWindow != win) && win->AcceptsFocus())
1477 {
1478 gtk_widget_grab_focus( win->m_wxwindow );
1479 }
1480
1481 // GDK sends surplus button down events
1482 // before a double click event. We
1483 // need to filter these out.
1484 if (gdk_event->type == GDK_BUTTON_PRESS)
1485 {
1486 GdkEvent *peek_event = gdk_event_peek();
1487 if (peek_event)
1488 {
1489 if ((peek_event->type == GDK_2BUTTON_PRESS) ||
1490 (peek_event->type == GDK_3BUTTON_PRESS))
1491 {
1492 gdk_event_free( peek_event );
1493 return TRUE;
1494 }
1495 else
1496 {
1497 gdk_event_free( peek_event );
1498 }
1499 }
1500 }
1501
1502 wxEventType event_type = wxEVT_NULL;
1503
1504 // GdkDisplay is a GTK+ 2.2.0 thing
1505 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2, 2, 0)
1506 if ( gdk_event->type == GDK_2BUTTON_PRESS &&
1507 !gtk_check_version(2,2,0) &&
1508 gdk_event->button >= 1 && gdk_event->button <= 3 )
1509 {
1510 // Reset GDK internal timestamp variables in order to disable GDK
1511 // triple click events. GDK will then next time believe no button has
1512 // been clicked just before, and send a normal button click event.
1513 GdkDisplay* display = gtk_widget_get_display (widget);
1514 display->button_click_time[1] = 0;
1515 display->button_click_time[0] = 0;
1516 }
1517 #endif // GTK 2+
1518
1519 if (gdk_event->button == 1)
1520 {
1521 // note that GDK generates triple click events which are not supported
1522 // by wxWidgets but still have to be passed to the app as otherwise
1523 // clicks would simply go missing
1524 switch (gdk_event->type)
1525 {
1526 // we shouldn't get triple clicks at all for GTK2 because we
1527 // suppress them artificially using the code above but we still
1528 // should map them to something for GTK1 and not just ignore them
1529 // as this would lose clicks
1530 case GDK_3BUTTON_PRESS: // we could also map this to DCLICK...
1531 case GDK_BUTTON_PRESS:
1532 event_type = wxEVT_LEFT_DOWN;
1533 break;
1534
1535 case GDK_2BUTTON_PRESS:
1536 event_type = wxEVT_LEFT_DCLICK;
1537 break;
1538
1539 default:
1540 // just to silence gcc warnings
1541 ;
1542 }
1543 }
1544 else if (gdk_event->button == 2)
1545 {
1546 switch (gdk_event->type)
1547 {
1548 case GDK_3BUTTON_PRESS:
1549 case GDK_BUTTON_PRESS:
1550 event_type = wxEVT_MIDDLE_DOWN;
1551 break;
1552
1553 case GDK_2BUTTON_PRESS:
1554 event_type = wxEVT_MIDDLE_DCLICK;
1555 break;
1556
1557 default:
1558 ;
1559 }
1560 }
1561 else if (gdk_event->button == 3)
1562 {
1563 switch (gdk_event->type)
1564 {
1565 case GDK_3BUTTON_PRESS:
1566 case GDK_BUTTON_PRESS:
1567 event_type = wxEVT_RIGHT_DOWN;
1568 break;
1569
1570 case GDK_2BUTTON_PRESS:
1571 event_type = wxEVT_RIGHT_DCLICK;
1572 break;
1573
1574 default:
1575 ;
1576 }
1577 }
1578 else if (gdk_event->button == 4 || gdk_event->button == 5)
1579 {
1580 if (gdk_event->type == GDK_BUTTON_PRESS )
1581 {
1582 event_type = wxEVT_MOUSEWHEEL;
1583 }
1584 }
1585
1586 if ( event_type == wxEVT_NULL )
1587 {
1588 // unknown mouse button or click type
1589 return FALSE;
1590 }
1591
1592 wxMouseEvent event( event_type );
1593 InitMouseEvent( win, event, gdk_event );
1594
1595 AdjustEventButtonState(event);
1596
1597 // wxListBox actually gets mouse events from the item, so we need to give it
1598 // a chance to correct this
1599 win->FixUpMouseEvent(widget, event.m_x, event.m_y);
1600
1601 // find the correct window to send the event to: it may be a different one
1602 // from the one which got it at GTK+ level because some controls don't have
1603 // their own X window and thus cannot get any events.
1604 if ( !g_captureWindow )
1605 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
1606
1607 // reset the event object and id in case win changed.
1608 event.SetEventObject( win );
1609 event.SetId( win->GetId() );
1610
1611 if (win->GTKProcessEvent( event ))
1612 {
1613 return TRUE;
1614 }
1615
1616 if (event_type == wxEVT_RIGHT_DOWN)
1617 {
1618 // generate a "context menu" event: this is similar to right mouse
1619 // click under many GUIs except that it is generated differently
1620 // (right up under MSW, ctrl-click under Mac, right down here) and
1621 //
1622 // (a) it's a command event and so is propagated to the parent
1623 // (b) under some ports it can be generated from kbd too
1624 // (c) it uses screen coords (because of (a))
1625 wxContextMenuEvent evtCtx(
1626 wxEVT_CONTEXT_MENU,
1627 win->GetId(),
1628 win->ClientToScreen(event.GetPosition()));
1629 evtCtx.SetEventObject(win);
1630 return win->GTKProcessEvent(evtCtx);
1631 }
1632
1633 return FALSE;
1634 }
1635
1636 //-----------------------------------------------------------------------------
1637 // "button_release_event"
1638 //-----------------------------------------------------------------------------
1639
1640 static gboolean
1641 gtk_window_button_release_callback( GtkWidget *widget,
1642 GdkEventButton *gdk_event,
1643 wxWindowGTK *win )
1644 {
1645 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
1646
1647 wxEventType event_type = wxEVT_NULL;
1648
1649 switch (gdk_event->button)
1650 {
1651 case 1:
1652 event_type = wxEVT_LEFT_UP;
1653 break;
1654
1655 case 2:
1656 event_type = wxEVT_MIDDLE_UP;
1657 break;
1658
1659 case 3:
1660 event_type = wxEVT_RIGHT_UP;
1661 break;
1662
1663 default:
1664 // unknown button, don't process
1665 return FALSE;
1666 }
1667
1668 wxMouseEvent event( event_type );
1669 InitMouseEvent( win, event, gdk_event );
1670
1671 AdjustEventButtonState(event);
1672
1673 // same wxListBox hack as above
1674 win->FixUpMouseEvent(widget, event.m_x, event.m_y);
1675
1676 if ( !g_captureWindow )
1677 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
1678
1679 // reset the event object and id in case win changed.
1680 event.SetEventObject( win );
1681 event.SetId( win->GetId() );
1682
1683 return win->GTKProcessEvent(event);
1684 }
1685
1686 //-----------------------------------------------------------------------------
1687 // "motion_notify_event"
1688 //-----------------------------------------------------------------------------
1689
1690 static gboolean
1691 gtk_window_motion_notify_callback( GtkWidget *widget,
1692 GdkEventMotion *gdk_event,
1693 wxWindowGTK *win )
1694 {
1695 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
1696
1697 if (gdk_event->is_hint)
1698 {
1699 int x = 0;
1700 int y = 0;
1701 GdkModifierType state;
1702 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
1703 gdk_event->x = x;
1704 gdk_event->y = y;
1705 }
1706
1707 wxMouseEvent event( wxEVT_MOTION );
1708 InitMouseEvent(win, event, gdk_event);
1709
1710 if ( g_captureWindow )
1711 {
1712 // synthesise a mouse enter or leave event if needed
1713 GdkWindow *winUnderMouse = gdk_window_at_pointer(NULL, NULL);
1714 // This seems to be necessary and actually been added to
1715 // GDK itself in version 2.0.X
1716 gdk_flush();
1717
1718 bool hasMouse = winUnderMouse == gdk_event->window;
1719 if ( hasMouse != g_captureWindowHasMouse )
1720 {
1721 // the mouse changed window
1722 g_captureWindowHasMouse = hasMouse;
1723
1724 wxMouseEvent eventM(g_captureWindowHasMouse ? wxEVT_ENTER_WINDOW
1725 : wxEVT_LEAVE_WINDOW);
1726 InitMouseEvent(win, eventM, gdk_event);
1727 eventM.SetEventObject(win);
1728 win->GTKProcessEvent(eventM);
1729 }
1730 }
1731 else // no capture
1732 {
1733 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
1734
1735 // reset the event object and id in case win changed.
1736 event.SetEventObject( win );
1737 event.SetId( win->GetId() );
1738 }
1739
1740 if ( !g_captureWindow )
1741 {
1742 wxSetCursorEvent cevent( event.m_x, event.m_y );
1743 if (win->GTKProcessEvent( cevent ))
1744 {
1745 win->SetCursor( cevent.GetCursor() );
1746 }
1747 }
1748
1749 return win->GTKProcessEvent(event);
1750 }
1751
1752 //-----------------------------------------------------------------------------
1753 // "scroll_event", (mouse wheel event)
1754 //-----------------------------------------------------------------------------
1755
1756 static gboolean
1757 window_scroll_event(GtkWidget*, GdkEventScroll* gdk_event, wxWindow* win)
1758 {
1759 DEBUG_MAIN_THREAD
1760
1761 // don't need to install idle handler, its done from "event" signal
1762
1763 if (gdk_event->direction != GDK_SCROLL_UP &&
1764 gdk_event->direction != GDK_SCROLL_DOWN)
1765 {
1766 return false;
1767 }
1768
1769 wxMouseEvent event(wxEVT_MOUSEWHEEL);
1770 // Can't use InitMouse macro because scroll events don't have button
1771 event.SetTimestamp( gdk_event->time );
1772 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK);
1773 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK);
1774 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK);
1775 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK);
1776 event.m_leftDown = (gdk_event->state & GDK_BUTTON1_MASK);
1777 event.m_middleDown = (gdk_event->state & GDK_BUTTON2_MASK);
1778 event.m_rightDown = (gdk_event->state & GDK_BUTTON3_MASK);
1779 event.m_linesPerAction = 3;
1780 event.m_wheelDelta = 120;
1781 if (gdk_event->direction == GDK_SCROLL_UP)
1782 event.m_wheelRotation = 120;
1783 else
1784 event.m_wheelRotation = -120;
1785
1786 wxPoint pt = win->GetClientAreaOrigin();
1787 event.m_x = (wxCoord)gdk_event->x - pt.x;
1788 event.m_y = (wxCoord)gdk_event->y - pt.y;
1789
1790 event.SetEventObject( win );
1791 event.SetId( win->GetId() );
1792 event.SetTimestamp( gdk_event->time );
1793
1794 return win->GTKProcessEvent(event);
1795 }
1796
1797 //-----------------------------------------------------------------------------
1798 // "popup-menu"
1799 //-----------------------------------------------------------------------------
1800
1801 static gboolean wxgtk_window_popup_menu_callback(GtkWidget*, wxWindowGTK* win)
1802 {
1803 wxContextMenuEvent event(wxEVT_CONTEXT_MENU, win->GetId(), wxPoint(-1, -1));
1804 event.SetEventObject(win);
1805 return win->GTKProcessEvent(event);
1806 }
1807
1808 //-----------------------------------------------------------------------------
1809 // "focus_in_event"
1810 //-----------------------------------------------------------------------------
1811
1812 static gboolean
1813 gtk_window_focus_in_callback( GtkWidget *widget,
1814 GdkEventFocus *WXUNUSED(event),
1815 wxWindow *win )
1816 {
1817 DEBUG_MAIN_THREAD
1818
1819 // don't need to install idle handler, its done from "event" signal
1820
1821 if (win->m_imData)
1822 gtk_im_context_focus_in(win->m_imData->context);
1823
1824 g_focusWindowLast =
1825 g_focusWindow = win;
1826
1827 wxLogTrace(TRACE_FOCUS,
1828 _T("%s: focus in"), win->GetName().c_str());
1829
1830 #ifdef HAVE_XIM
1831 if (win->m_ic)
1832 gdk_im_begin(win->m_ic, win->m_wxwindow->window);
1833 #endif
1834
1835 #if wxUSE_CARET
1836 // caret needs to be informed about focus change
1837 wxCaret *caret = win->GetCaret();
1838 if ( caret )
1839 {
1840 caret->OnSetFocus();
1841 }
1842 #endif // wxUSE_CARET
1843
1844 gboolean ret = FALSE;
1845
1846 // does the window itself think that it has the focus?
1847 if ( !win->m_hasFocus )
1848 {
1849 // not yet, notify it
1850 win->m_hasFocus = true;
1851
1852 (void)DoSendFocusEvents(win);
1853
1854 ret = TRUE;
1855 }
1856
1857 // Disable default focus handling for custom windows
1858 // since the default GTK+ handler issues a repaint
1859 if (win->m_wxwindow)
1860 return ret;
1861
1862 return FALSE;
1863 }
1864
1865 //-----------------------------------------------------------------------------
1866 // "focus_out_event"
1867 //-----------------------------------------------------------------------------
1868
1869 static gboolean
1870 gtk_window_focus_out_callback( GtkWidget *widget,
1871 GdkEventFocus *gdk_event,
1872 wxWindowGTK *win )
1873 {
1874 DEBUG_MAIN_THREAD
1875
1876 // don't need to install idle handler, its done from "event" signal
1877
1878 if (win->m_imData)
1879 gtk_im_context_focus_out(win->m_imData->context);
1880
1881 wxLogTrace( TRACE_FOCUS,
1882 _T("%s: focus out"), win->GetName().c_str() );
1883
1884
1885 wxWindowGTK *winFocus = wxFindFocusedChild(win);
1886 if ( winFocus )
1887 win = winFocus;
1888
1889 g_focusWindow = (wxWindowGTK *)NULL;
1890
1891 #ifdef HAVE_XIM
1892 if (win->m_ic)
1893 gdk_im_end();
1894 #endif
1895
1896 #if wxUSE_CARET
1897 // caret needs to be informed about focus change
1898 wxCaret *caret = win->GetCaret();
1899 if ( caret )
1900 {
1901 caret->OnKillFocus();
1902 }
1903 #endif // wxUSE_CARET
1904
1905 gboolean ret = FALSE;
1906
1907 // don't send the window a kill focus event if it thinks that it doesn't
1908 // have focus already
1909 if ( win->m_hasFocus )
1910 {
1911 win->m_hasFocus = false;
1912
1913 wxFocusEvent event( wxEVT_KILL_FOCUS, win->GetId() );
1914 event.SetEventObject( win );
1915
1916 (void)win->GTKProcessEvent( event );
1917
1918 ret = TRUE;
1919 }
1920
1921 // Disable default focus handling for custom windows
1922 // since the default GTK+ handler issues a repaint
1923 if (win->m_wxwindow)
1924 return ret;
1925
1926 return FALSE;
1927 }
1928
1929 //-----------------------------------------------------------------------------
1930 // "enter_notify_event"
1931 //-----------------------------------------------------------------------------
1932
1933 static gboolean
1934 gtk_window_enter_callback( GtkWidget *widget,
1935 GdkEventCrossing *gdk_event,
1936 wxWindowGTK *win )
1937 {
1938 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
1939
1940 // Event was emitted after a grab
1941 if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE;
1942
1943 int x = 0;
1944 int y = 0;
1945 GdkModifierType state = (GdkModifierType)0;
1946
1947 gdk_window_get_pointer( widget->window, &x, &y, &state );
1948
1949 wxMouseEvent event( wxEVT_ENTER_WINDOW );
1950 InitMouseEvent(win, event, gdk_event);
1951 wxPoint pt = win->GetClientAreaOrigin();
1952 event.m_x = x + pt.x;
1953 event.m_y = y + pt.y;
1954
1955 if ( !g_captureWindow )
1956 {
1957 wxSetCursorEvent cevent( event.m_x, event.m_y );
1958 if (win->GTKProcessEvent( cevent ))
1959 {
1960 win->SetCursor( cevent.GetCursor() );
1961 }
1962 }
1963
1964 return win->GTKProcessEvent(event);
1965 }
1966
1967 //-----------------------------------------------------------------------------
1968 // "leave_notify_event"
1969 //-----------------------------------------------------------------------------
1970
1971 static gboolean
1972 gtk_window_leave_callback( GtkWidget *widget,
1973 GdkEventCrossing *gdk_event,
1974 wxWindowGTK *win )
1975 {
1976 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
1977
1978 // Event was emitted after an ungrab
1979 if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE;
1980
1981 wxMouseEvent event( wxEVT_LEAVE_WINDOW );
1982 event.SetTimestamp( gdk_event->time );
1983 event.SetEventObject( win );
1984
1985 int x = 0;
1986 int y = 0;
1987 GdkModifierType state = (GdkModifierType)0;
1988
1989 gdk_window_get_pointer( widget->window, &x, &y, &state );
1990
1991 event.m_shiftDown = (state & GDK_SHIFT_MASK) != 0;
1992 event.m_controlDown = (state & GDK_CONTROL_MASK) != 0;
1993 event.m_altDown = (state & GDK_MOD1_MASK) != 0;
1994 event.m_metaDown = (state & GDK_MOD2_MASK) != 0;
1995 event.m_leftDown = (state & GDK_BUTTON1_MASK) != 0;
1996 event.m_middleDown = (state & GDK_BUTTON2_MASK) != 0;
1997 event.m_rightDown = (state & GDK_BUTTON3_MASK) != 0;
1998
1999 wxPoint pt = win->GetClientAreaOrigin();
2000 event.m_x = x + pt.x;
2001 event.m_y = y + pt.y;
2002
2003 return win->GTKProcessEvent(event);
2004 }
2005
2006 //-----------------------------------------------------------------------------
2007 // "value_changed" from scrollbar
2008 //-----------------------------------------------------------------------------
2009
2010 static void
2011 gtk_scrollbar_value_changed(GtkRange* range, wxWindow* win)
2012 {
2013 wxEventType eventType = win->GetScrollEventType(range);
2014 if (eventType != wxEVT_NULL)
2015 {
2016 // Convert scroll event type to scrollwin event type
2017 eventType += wxEVT_SCROLLWIN_TOP - wxEVT_SCROLL_TOP;
2018
2019 // find the scrollbar which generated the event
2020 wxWindowGTK::ScrollDir dir = win->ScrollDirFromRange(range);
2021
2022 // generate the corresponding wx event
2023 const int orient = win->OrientFromScrollDir(dir);
2024 wxScrollWinEvent event(eventType, win->GetScrollPos(orient), orient);
2025 event.SetEventObject(win);
2026
2027 win->m_blockValueChanged[dir] = true;
2028 win->GTKProcessEvent(event);
2029 win->m_blockValueChanged[dir] = false;
2030 }
2031 }
2032
2033 //-----------------------------------------------------------------------------
2034 // "button_press_event" from scrollbar
2035 //-----------------------------------------------------------------------------
2036
2037 static gboolean
2038 gtk_scrollbar_button_press_event(GtkRange*, GdkEventButton*, wxWindow* win)
2039 {
2040 DEBUG_MAIN_THREAD
2041
2042 // don't need to install idle handler, its done from "event" signal
2043
2044 g_blockEventsOnScroll = true;
2045 win->m_mouseButtonDown = true;
2046
2047 return false;
2048 }
2049
2050 //-----------------------------------------------------------------------------
2051 // "event_after" from scrollbar
2052 //-----------------------------------------------------------------------------
2053
2054 static void
2055 gtk_scrollbar_event_after(GtkRange* range, GdkEvent* event, wxWindow* win)
2056 {
2057 if (event->type == GDK_BUTTON_RELEASE)
2058 {
2059 g_signal_handlers_block_by_func(range, (void*)gtk_scrollbar_event_after, win);
2060
2061 const int orient = win->OrientFromScrollDir(
2062 win->ScrollDirFromRange(range));
2063 wxScrollWinEvent event(wxEVT_SCROLLWIN_THUMBRELEASE, win->GetScrollPos(orient), orient);
2064 event.SetEventObject(win);
2065 win->GTKProcessEvent(event);
2066 }
2067 }
2068
2069 //-----------------------------------------------------------------------------
2070 // "button_release_event" from scrollbar
2071 //-----------------------------------------------------------------------------
2072
2073 static gboolean
2074 gtk_scrollbar_button_release_event(GtkRange* range, GdkEventButton*, wxWindow* win)
2075 {
2076 DEBUG_MAIN_THREAD
2077
2078 g_blockEventsOnScroll = false;
2079 win->m_mouseButtonDown = false;
2080 // If thumb tracking
2081 if (win->m_isScrolling)
2082 {
2083 win->m_isScrolling = false;
2084 // Hook up handler to send thumb release event after this emission is finished.
2085 // To allow setting scroll position from event handler, sending event must
2086 // be deferred until after the GtkRange handler for this signal has run
2087 g_signal_handlers_unblock_by_func(range, (void*)gtk_scrollbar_event_after, win);
2088 }
2089
2090 return false;
2091 }
2092
2093 //-----------------------------------------------------------------------------
2094 // "realize" from m_widget
2095 //-----------------------------------------------------------------------------
2096
2097 /* We cannot set colours and fonts before the widget has
2098 been realized, so we do this directly after realization. */
2099
2100 static void
2101 gtk_window_realized_callback( GtkWidget *m_widget, wxWindow *win )
2102 {
2103 DEBUG_MAIN_THREAD
2104
2105 if (g_isIdle)
2106 wxapp_install_idle_handler();
2107
2108 if (win->m_imData)
2109 {
2110 GtkPizza *pizza = GTK_PIZZA( m_widget );
2111 gtk_im_context_set_client_window( win->m_imData->context,
2112 pizza->bin_window );
2113 }
2114
2115 wxWindowCreateEvent event( win );
2116 event.SetEventObject( win );
2117 win->GTKProcessEvent( event );
2118 }
2119
2120 //-----------------------------------------------------------------------------
2121 // "size_allocate"
2122 //-----------------------------------------------------------------------------
2123
2124 static
2125 void gtk_window_size_callback( GtkWidget *WXUNUSED(widget),
2126 GtkAllocation *alloc,
2127 wxWindow *win )
2128 {
2129 if (g_isIdle)
2130 wxapp_install_idle_handler();
2131
2132 int client_width = 0;
2133 int client_height = 0;
2134 win->GetClientSize( &client_width, &client_height );
2135 if ((client_width == win->m_oldClientWidth) && (client_height == win->m_oldClientHeight))
2136 return;
2137
2138 #if 0
2139 wxPrintf( wxT("size_allocate ") );
2140 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
2141 wxPrintf( win->GetClassInfo()->GetClassName() );
2142 wxPrintf( wxT(" %d %d %d %d\n"),
2143 alloc->x,
2144 alloc->y,
2145 alloc->width,
2146 alloc->height );
2147 #endif
2148
2149 GTK_PIZZA(win->m_wxwindow)->m_width = win->GetClientSize().x;
2150
2151 win->m_oldClientWidth = client_width;
2152 win->m_oldClientHeight = client_height;
2153
2154 if (!win->m_nativeSizeEvent)
2155 {
2156 wxSizeEvent event( win->GetSize(), win->GetId() );
2157 event.SetEventObject( win );
2158 win->GTKProcessEvent( event );
2159 }
2160 }
2161
2162
2163 #ifdef HAVE_XIM
2164 #define WXUNUSED_UNLESS_XIM(param) param
2165 #else
2166 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2167 #endif
2168
2169 /* Resize XIM window */
2170 static
2171 void gtk_wxwindow_size_callback( GtkWidget* WXUNUSED_UNLESS_XIM(widget),
2172 GtkAllocation* WXUNUSED_UNLESS_XIM(alloc),
2173 wxWindowGTK* WXUNUSED_UNLESS_XIM(win) )
2174 {
2175 if (g_isIdle)
2176 wxapp_install_idle_handler();
2177
2178 #ifdef HAVE_XIM
2179 if (!win->m_ic)
2180 return;
2181
2182 if (gdk_ic_get_style (win->m_ic) & GDK_IM_PREEDIT_POSITION)
2183 {
2184 gint width, height;
2185
2186 gdk_drawable_get_size (widget->window, &width, &height);
2187 win->m_icattr->preedit_area.width = width;
2188 win->m_icattr->preedit_area.height = height;
2189 gdk_ic_set_attr (win->m_ic, win->m_icattr, GDK_IC_PREEDIT_AREA);
2190 }
2191 #endif // HAVE_XIM
2192 }
2193
2194 //-----------------------------------------------------------------------------
2195 // "realize" from m_wxwindow
2196 //-----------------------------------------------------------------------------
2197
2198 /* Initialize XIM support */
2199
2200 static void
2201 gtk_wxwindow_realized_callback( GtkWidget * WXUNUSED_UNLESS_XIM(widget),
2202 wxWindowGTK * WXUNUSED_UNLESS_XIM(win) )
2203 {
2204 if (g_isIdle)
2205 wxapp_install_idle_handler();
2206
2207 #ifdef HAVE_XIM
2208 if (win->m_ic) return;
2209 if (!widget) return;
2210 if (!gdk_im_ready()) return;
2211
2212 win->m_icattr = gdk_ic_attr_new();
2213 if (!win->m_icattr) return;
2214
2215 gint width, height;
2216 GdkEventMask mask;
2217 GdkColormap *colormap;
2218 GdkICAttr *attr = win->m_icattr;
2219 unsigned attrmask = GDK_IC_ALL_REQ;
2220 GdkIMStyle style;
2221 GdkIMStyle supported_style = (GdkIMStyle)
2222 (GDK_IM_PREEDIT_NONE |
2223 GDK_IM_PREEDIT_NOTHING |
2224 GDK_IM_PREEDIT_POSITION |
2225 GDK_IM_STATUS_NONE |
2226 GDK_IM_STATUS_NOTHING);
2227
2228 if (widget->style && widget->style->font->type != GDK_FONT_FONTSET)
2229 supported_style = (GdkIMStyle)(supported_style & ~GDK_IM_PREEDIT_POSITION);
2230
2231 attr->style = style = gdk_im_decide_style (supported_style);
2232 attr->client_window = widget->window;
2233
2234 if ((colormap = gtk_widget_get_colormap (widget)) !=
2235 gtk_widget_get_default_colormap ())
2236 {
2237 attrmask |= GDK_IC_PREEDIT_COLORMAP;
2238 attr->preedit_colormap = colormap;
2239 }
2240
2241 attrmask |= GDK_IC_PREEDIT_FOREGROUND;
2242 attrmask |= GDK_IC_PREEDIT_BACKGROUND;
2243 attr->preedit_foreground = widget->style->fg[GTK_STATE_NORMAL];
2244 attr->preedit_background = widget->style->base[GTK_STATE_NORMAL];
2245
2246 switch (style & GDK_IM_PREEDIT_MASK)
2247 {
2248 case GDK_IM_PREEDIT_POSITION:
2249 if (widget->style && widget->style->font->type != GDK_FONT_FONTSET)
2250 {
2251 g_warning ("over-the-spot style requires fontset");
2252 break;
2253 }
2254
2255 gdk_drawable_get_size (widget->window, &width, &height);
2256
2257 attrmask |= GDK_IC_PREEDIT_POSITION_REQ;
2258 attr->spot_location.x = 0;
2259 attr->spot_location.y = height;
2260 attr->preedit_area.x = 0;
2261 attr->preedit_area.y = 0;
2262 attr->preedit_area.width = width;
2263 attr->preedit_area.height = height;
2264 attr->preedit_fontset = widget->style->font;
2265
2266 break;
2267 }
2268
2269 win->m_ic = gdk_ic_new (attr, (GdkICAttributesType)attrmask);
2270
2271 if (win->m_ic == NULL)
2272 g_warning ("Can't create input context.");
2273 else
2274 {
2275 mask = gdk_window_get_events (widget->window);
2276 mask = (GdkEventMask)(mask | gdk_ic_get_events (win->m_ic));
2277 gdk_window_set_events (widget->window, mask);
2278
2279 if (GTK_WIDGET_HAS_FOCUS(widget))
2280 gdk_im_begin (win->m_ic, widget->window);
2281 }
2282 #endif // HAVE_XIM
2283 }
2284
2285 } // extern "C"
2286
2287 // ----------------------------------------------------------------------------
2288 // this wxWindowBase function is implemented here (in platform-specific file)
2289 // because it is static and so couldn't be made virtual
2290 // ----------------------------------------------------------------------------
2291
2292 wxWindow *wxWindowBase::DoFindFocus()
2293 {
2294 // the cast is necessary when we compile in wxUniversal mode
2295 return (wxWindow *)g_focusWindow;
2296 }
2297
2298 //-----------------------------------------------------------------------------
2299 // InsertChild for wxWindowGTK.
2300 //-----------------------------------------------------------------------------
2301
2302 /* Callback for wxWindowGTK. This very strange beast has to be used because
2303 * C++ has no virtual methods in a constructor. We have to emulate a
2304 * virtual function here as wxNotebook requires a different way to insert
2305 * a child in it. I had opted for creating a wxNotebookPage window class
2306 * which would have made this superfluous (such in the MDI window system),
2307 * but no-one was listening to me... */
2308
2309 static void wxInsertChildInWindow( wxWindowGTK* parent, wxWindowGTK* child )
2310 {
2311 /* the window might have been scrolled already, do we
2312 have to adapt the position */
2313 GtkPizza *pizza = GTK_PIZZA(parent->m_wxwindow);
2314 child->m_x += gtk_pizza_get_xoffset( pizza );
2315 child->m_y += gtk_pizza_get_yoffset( pizza );
2316
2317 gtk_pizza_put( GTK_PIZZA(parent->m_wxwindow),
2318 GTK_WIDGET(child->m_widget),
2319 child->m_x,
2320 child->m_y,
2321 child->m_width,
2322 child->m_height );
2323 }
2324
2325 //-----------------------------------------------------------------------------
2326 // global functions
2327 //-----------------------------------------------------------------------------
2328
2329 wxWindow *wxGetActiveWindow()
2330 {
2331 return wxWindow::FindFocus();
2332 }
2333
2334
2335 wxMouseState wxGetMouseState()
2336 {
2337 wxMouseState ms;
2338
2339 gint x;
2340 gint y;
2341 GdkModifierType mask;
2342
2343 gdk_window_get_pointer(NULL, &x, &y, &mask);
2344
2345 ms.SetX(x);
2346 ms.SetY(y);
2347 ms.SetLeftDown(mask & GDK_BUTTON1_MASK);
2348 ms.SetMiddleDown(mask & GDK_BUTTON2_MASK);
2349 ms.SetRightDown(mask & GDK_BUTTON3_MASK);
2350
2351 ms.SetControlDown(mask & GDK_CONTROL_MASK);
2352 ms.SetShiftDown(mask & GDK_SHIFT_MASK);
2353 ms.SetAltDown(mask & GDK_MOD1_MASK);
2354 ms.SetMetaDown(mask & GDK_MOD2_MASK);
2355
2356 return ms;
2357 }
2358
2359 //-----------------------------------------------------------------------------
2360 // wxWindowGTK
2361 //-----------------------------------------------------------------------------
2362
2363 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2364 // method
2365 #ifdef __WXUNIVERSAL__
2366 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK, wxWindowBase)
2367 #else // __WXGTK__
2368 IMPLEMENT_DYNAMIC_CLASS(wxWindow, wxWindowBase)
2369 #endif // __WXUNIVERSAL__/__WXGTK__
2370
2371 void wxWindowGTK::Init()
2372 {
2373 // GTK specific
2374 m_widget = (GtkWidget *) NULL;
2375 m_wxwindow = (GtkWidget *) NULL;
2376 m_focusWidget = (GtkWidget *) NULL;
2377
2378 // position/size
2379 m_x = 0;
2380 m_y = 0;
2381 m_width = 0;
2382 m_height = 0;
2383
2384 m_sizeSet = false;
2385 m_hasVMT = false;
2386 m_needParent = true;
2387 m_isBeingDeleted = false;
2388
2389 m_showOnIdle= false;
2390
2391 m_noExpose = false;
2392 m_nativeSizeEvent = false;
2393
2394 m_hasScrolling = false;
2395 m_isScrolling = false;
2396 m_mouseButtonDown = false;
2397 m_blockScrollEvent = false;
2398
2399 // initialize scrolling stuff
2400 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
2401 {
2402 m_scrollBar[dir] = NULL;
2403 m_scrollPos[dir] = 0;
2404 m_blockValueChanged[dir] = false;
2405 }
2406
2407 m_oldClientWidth =
2408 m_oldClientHeight = 0;
2409
2410 m_resizing = false;
2411
2412 m_insertCallback = (wxInsertChildFunction) NULL;
2413
2414 m_acceptsFocus = false;
2415 m_hasFocus = false;
2416
2417 m_clipPaintRegion = false;
2418
2419 m_needsStyleChange = false;
2420
2421 m_cursor = *wxSTANDARD_CURSOR;
2422
2423 m_imData = NULL;
2424 m_dirtyTabOrder = false;
2425 }
2426
2427 wxWindowGTK::wxWindowGTK()
2428 {
2429 Init();
2430 }
2431
2432 wxWindowGTK::wxWindowGTK( wxWindow *parent,
2433 wxWindowID id,
2434 const wxPoint &pos,
2435 const wxSize &size,
2436 long style,
2437 const wxString &name )
2438 {
2439 Init();
2440
2441 Create( parent, id, pos, size, style, name );
2442 }
2443
2444 bool wxWindowGTK::Create( wxWindow *parent,
2445 wxWindowID id,
2446 const wxPoint &pos,
2447 const wxSize &size,
2448 long style,
2449 const wxString &name )
2450 {
2451 if (!PreCreation( parent, pos, size ) ||
2452 !CreateBase( parent, id, pos, size, style, wxDefaultValidator, name ))
2453 {
2454 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2455 return false;
2456 }
2457
2458 m_insertCallback = wxInsertChildInWindow;
2459
2460 m_widget = gtk_scrolled_window_new( (GtkAdjustment *) NULL, (GtkAdjustment *) NULL );
2461 GTK_WIDGET_UNSET_FLAGS( m_widget, GTK_CAN_FOCUS );
2462
2463 GtkScrolledWindow *scrolledWindow = GTK_SCROLLED_WINDOW(m_widget);
2464
2465 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
2466 scroll_class->scrollbar_spacing = 0;
2467
2468 gtk_scrolled_window_set_policy( scrolledWindow, GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC );
2469
2470 m_scrollBar[ScrollDir_Horz] = GTK_RANGE(scrolledWindow->hscrollbar);
2471 m_scrollBar[ScrollDir_Vert] = GTK_RANGE(scrolledWindow->vscrollbar);
2472 if (GetLayoutDirection() == wxLayout_RightToLeft)
2473 gtk_range_set_inverted( m_scrollBar[ScrollDir_Horz], TRUE );
2474
2475 m_wxwindow = gtk_pizza_new();
2476
2477 #ifndef __WXUNIVERSAL__
2478 GtkPizza *pizza = GTK_PIZZA(m_wxwindow);
2479
2480 if (HasFlag(wxRAISED_BORDER))
2481 {
2482 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_OUT );
2483 }
2484 else if (HasFlag(wxSUNKEN_BORDER))
2485 {
2486 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_IN );
2487 }
2488 else if (HasFlag(wxSIMPLE_BORDER))
2489 {
2490 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_THIN );
2491 }
2492 else
2493 {
2494 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_NONE );
2495 }
2496 #endif // __WXUNIVERSAL__
2497
2498 gtk_container_add( GTK_CONTAINER(m_widget), m_wxwindow );
2499
2500 GTK_WIDGET_SET_FLAGS( m_wxwindow, GTK_CAN_FOCUS );
2501 m_acceptsFocus = true;
2502
2503 // connect various scroll-related events
2504 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
2505 {
2506 // these handlers block mouse events to any window during scrolling
2507 // such as motion events and prevent GTK and wxWidgets from fighting
2508 // over where the slider should be
2509 g_signal_connect(m_scrollBar[dir], "button_press_event",
2510 G_CALLBACK(gtk_scrollbar_button_press_event), this);
2511 g_signal_connect(m_scrollBar[dir], "button_release_event",
2512 G_CALLBACK(gtk_scrollbar_button_release_event), this);
2513
2514 gulong handler_id = g_signal_connect(m_scrollBar[dir], "event_after",
2515 G_CALLBACK(gtk_scrollbar_event_after), this);
2516 g_signal_handler_block(m_scrollBar[dir], handler_id);
2517
2518 // these handlers get notified when scrollbar slider moves
2519 g_signal_connect(m_scrollBar[dir], "value_changed",
2520 G_CALLBACK(gtk_scrollbar_value_changed), this);
2521 }
2522
2523 gtk_widget_show( m_wxwindow );
2524
2525 if (m_parent)
2526 m_parent->DoAddChild( this );
2527
2528 m_focusWidget = m_wxwindow;
2529
2530 PostCreation();
2531
2532 return true;
2533 }
2534
2535 wxWindowGTK::~wxWindowGTK()
2536 {
2537 SendDestroyEvent();
2538
2539 if (g_focusWindow == this)
2540 g_focusWindow = NULL;
2541
2542 if ( g_delayedFocus == this )
2543 g_delayedFocus = NULL;
2544
2545 m_isBeingDeleted = true;
2546 m_hasVMT = false;
2547
2548 // destroy children before destroying this window itself
2549 DestroyChildren();
2550
2551 // unhook focus handlers to prevent stray events being
2552 // propagated to this (soon to be) dead object
2553 if (m_focusWidget != NULL)
2554 {
2555 g_signal_handlers_disconnect_by_func (m_focusWidget,
2556 (gpointer) gtk_window_focus_in_callback,
2557 this);
2558 g_signal_handlers_disconnect_by_func (m_focusWidget,
2559 (gpointer) gtk_window_focus_out_callback,
2560 this);
2561 }
2562
2563 if (m_widget)
2564 Show( false );
2565
2566 #ifdef HAVE_XIM
2567 if (m_ic)
2568 gdk_ic_destroy (m_ic);
2569 if (m_icattr)
2570 gdk_ic_attr_destroy (m_icattr);
2571 #endif
2572
2573 // delete before the widgets to avoid a crash on solaris
2574 delete m_imData;
2575
2576 if (m_wxwindow)
2577 {
2578 gtk_widget_destroy( m_wxwindow );
2579 m_wxwindow = (GtkWidget*) NULL;
2580 }
2581
2582 if (m_widget)
2583 {
2584 gtk_widget_destroy( m_widget );
2585 m_widget = (GtkWidget*) NULL;
2586 }
2587 }
2588
2589 bool wxWindowGTK::PreCreation( wxWindowGTK *parent, const wxPoint &pos, const wxSize &size )
2590 {
2591 wxCHECK_MSG( !m_needParent || parent, false, wxT("Need complete parent.") );
2592
2593 // Use either the given size, or the default if -1 is given.
2594 // See wxWindowBase for these functions.
2595 m_width = WidthDefault(size.x) ;
2596 m_height = HeightDefault(size.y);
2597
2598 m_x = (int)pos.x;
2599 m_y = (int)pos.y;
2600
2601 return true;
2602 }
2603
2604 void wxWindowGTK::PostCreation()
2605 {
2606 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
2607
2608 if (m_wxwindow)
2609 {
2610 if (!m_noExpose)
2611 {
2612 // these get reported to wxWidgets -> wxPaintEvent
2613
2614 g_signal_connect (m_wxwindow, "expose_event",
2615 G_CALLBACK (gtk_window_expose_callback), this);
2616
2617 if (GetLayoutDirection() == wxLayout_LeftToRight)
2618 gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow), HasFlag( wxFULL_REPAINT_ON_RESIZE ) );
2619 }
2620
2621 // Create input method handler
2622 m_imData = new wxGtkIMData;
2623
2624 // Cannot handle drawing preedited text yet
2625 gtk_im_context_set_use_preedit( m_imData->context, FALSE );
2626
2627 g_signal_connect (m_imData->context, "commit",
2628 G_CALLBACK (gtk_wxwindow_commit_cb), this);
2629
2630 // these are called when the "sunken" or "raised" borders are drawn
2631 g_signal_connect (m_widget, "expose_event",
2632 G_CALLBACK (gtk_window_own_expose_callback), this);
2633 }
2634
2635 // focus handling
2636
2637 if (!GTK_IS_WINDOW(m_widget))
2638 {
2639 if (m_focusWidget == NULL)
2640 m_focusWidget = m_widget;
2641
2642 if (m_wxwindow)
2643 {
2644 g_signal_connect (m_focusWidget, "focus_in_event",
2645 G_CALLBACK (gtk_window_focus_in_callback), this);
2646 g_signal_connect (m_focusWidget, "focus_out_event",
2647 G_CALLBACK (gtk_window_focus_out_callback), this);
2648 }
2649 else
2650 {
2651 g_signal_connect_after (m_focusWidget, "focus_in_event",
2652 G_CALLBACK (gtk_window_focus_in_callback), this);
2653 g_signal_connect_after (m_focusWidget, "focus_out_event",
2654 G_CALLBACK (gtk_window_focus_out_callback), this);
2655 }
2656 }
2657
2658 // connect to the various key and mouse handlers
2659
2660 GtkWidget *connect_widget = GetConnectWidget();
2661
2662 ConnectWidget( connect_widget );
2663
2664 /* We cannot set colours, fonts and cursors before the widget has
2665 been realized, so we do this directly after realization */
2666 g_signal_connect (connect_widget, "realize",
2667 G_CALLBACK (gtk_window_realized_callback), this);
2668
2669 if (m_wxwindow)
2670 {
2671 // Catch native resize events
2672 g_signal_connect (m_wxwindow, "size_allocate",
2673 G_CALLBACK (gtk_window_size_callback), this);
2674
2675 // Initialize XIM support
2676 g_signal_connect (m_wxwindow, "realize",
2677 G_CALLBACK (gtk_wxwindow_realized_callback), this);
2678
2679 // And resize XIM window
2680 g_signal_connect (m_wxwindow, "size_allocate",
2681 G_CALLBACK (gtk_wxwindow_size_callback), this);
2682 }
2683
2684 if (GTK_IS_COMBO(m_widget))
2685 {
2686 GtkCombo *gcombo = GTK_COMBO(m_widget);
2687
2688 g_signal_connect (gcombo->entry, "size_request",
2689 G_CALLBACK (wxgtk_combo_size_request_callback),
2690 this);
2691 }
2692 #ifdef GTK_IS_FILE_CHOOSER_BUTTON
2693 else if (!gtk_check_version(2,6,0) && GTK_IS_FILE_CHOOSER_BUTTON(m_widget))
2694 {
2695 // If we connect to the "size_request" signal of a GtkFileChooserButton
2696 // then that control won't be sized properly when placed inside sizers
2697 // (this can be tested removing this elseif and running XRC or WIDGETS samples)
2698 // FIXME: what should be done here ?
2699 }
2700 #endif
2701 else
2702 {
2703 // This is needed if we want to add our windows into native
2704 // GTK controls, such as the toolbar. With this callback, the
2705 // toolbar gets to know the correct size (the one set by the
2706 // programmer). Sadly, it misbehaves for wxComboBox.
2707 g_signal_connect (m_widget, "size_request",
2708 G_CALLBACK (wxgtk_window_size_request_callback),
2709 this);
2710 }
2711
2712 InheritAttributes();
2713
2714 m_hasVMT = true;
2715
2716 SetLayoutDirection(wxLayout_Default);
2717
2718 // unless the window was created initially hidden (i.e. Hide() had been
2719 // called before Create()), we should show it at GTK+ level as well
2720 if ( IsShown() )
2721 gtk_widget_show( m_widget );
2722 }
2723
2724 void wxWindowGTK::ConnectWidget( GtkWidget *widget )
2725 {
2726 g_signal_connect (widget, "key_press_event",
2727 G_CALLBACK (gtk_window_key_press_callback), this);
2728 g_signal_connect (widget, "key_release_event",
2729 G_CALLBACK (gtk_window_key_release_callback), this);
2730 g_signal_connect (widget, "button_press_event",
2731 G_CALLBACK (gtk_window_button_press_callback), this);
2732 g_signal_connect (widget, "button_release_event",
2733 G_CALLBACK (gtk_window_button_release_callback), this);
2734 g_signal_connect (widget, "motion_notify_event",
2735 G_CALLBACK (gtk_window_motion_notify_callback), this);
2736 g_signal_connect (widget, "scroll_event",
2737 G_CALLBACK (window_scroll_event), this);
2738 g_signal_connect (widget, "popup_menu",
2739 G_CALLBACK (wxgtk_window_popup_menu_callback), this);
2740 g_signal_connect (widget, "enter_notify_event",
2741 G_CALLBACK (gtk_window_enter_callback), this);
2742 g_signal_connect (widget, "leave_notify_event",
2743 G_CALLBACK (gtk_window_leave_callback), this);
2744 }
2745
2746 bool wxWindowGTK::Destroy()
2747 {
2748 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
2749
2750 m_hasVMT = false;
2751
2752 return wxWindowBase::Destroy();
2753 }
2754
2755 void wxWindowGTK::DoMoveWindow(int x, int y, int width, int height)
2756 {
2757 // inform the parent to perform the move
2758 gtk_pizza_set_size( GTK_PIZZA(m_parent->m_wxwindow), m_widget, x, y, width, height );
2759
2760 }
2761
2762 void wxWindowGTK::DoSetSize( int x, int y, int width, int height, int sizeFlags )
2763 {
2764 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
2765 wxASSERT_MSG( (m_parent != NULL), wxT("wxWindowGTK::SetSize requires parent.\n") );
2766
2767 if (m_resizing) return; /* I don't like recursions */
2768 m_resizing = true;
2769
2770 int currentX, currentY;
2771 GetPosition(&currentX, &currentY);
2772 if (x == -1 && !(sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
2773 x = currentX;
2774 if (y == -1 && !(sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
2775 y = currentY;
2776 AdjustForParentClientOrigin(x, y, sizeFlags);
2777
2778 // calculate the best size if we should auto size the window
2779 if ( ((sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1) ||
2780 ((sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1) )
2781 {
2782 const wxSize sizeBest = GetBestSize();
2783 if ( (sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1 )
2784 width = sizeBest.x;
2785 if ( (sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1 )
2786 height = sizeBest.y;
2787 }
2788
2789 if (width != -1)
2790 m_width = width;
2791 if (height != -1)
2792 m_height = height;
2793
2794 int minWidth = GetMinWidth(),
2795 minHeight = GetMinHeight(),
2796 maxWidth = GetMaxWidth(),
2797 maxHeight = GetMaxHeight();
2798
2799 if ((minWidth != -1) && (m_width < minWidth )) m_width = minWidth;
2800 if ((minHeight != -1) && (m_height < minHeight)) m_height = minHeight;
2801 if ((maxWidth != -1) && (m_width > maxWidth )) m_width = maxWidth;
2802 if ((maxHeight != -1) && (m_height > maxHeight)) m_height = maxHeight;
2803
2804 #if wxUSE_TOOLBAR_NATIVE
2805 if (wxDynamicCast(GetParent(), wxToolBar))
2806 {
2807 // don't take the x,y values, they're wrong because toolbar sets them
2808 GtkWidget *widget = GTK_WIDGET(m_widget);
2809 gtk_widget_set_size_request (widget, m_width, m_height);
2810 }
2811 else
2812 #endif
2813 if (m_parent->m_wxwindow == NULL) // i.e. wxNotebook
2814 {
2815 // don't set the size for children of wxNotebook, just take the values.
2816 m_x = x;
2817 m_y = y;
2818 m_width = width;
2819 m_height = height;
2820 }
2821 else
2822 {
2823 GtkPizza *pizza = GTK_PIZZA(m_parent->m_wxwindow);
2824 if ((sizeFlags & wxSIZE_ALLOW_MINUS_ONE) == 0)
2825 {
2826 if (x != -1) m_x = x + gtk_pizza_get_xoffset( pizza );
2827 if (y != -1) m_y = y + gtk_pizza_get_yoffset( pizza );
2828 }
2829 else
2830 {
2831 m_x = x + gtk_pizza_get_xoffset( pizza );
2832 m_y = y + gtk_pizza_get_yoffset( pizza );
2833 }
2834
2835 int left_border = 0;
2836 int right_border = 0;
2837 int top_border = 0;
2838 int bottom_border = 0;
2839
2840 /* the default button has a border around it */
2841 if (GTK_WIDGET_CAN_DEFAULT(m_widget))
2842 {
2843 GtkBorder *default_border = NULL;
2844 gtk_widget_style_get( m_widget, "default_border", &default_border, NULL );
2845 if (default_border)
2846 {
2847 left_border += default_border->left;
2848 right_border += default_border->right;
2849 top_border += default_border->top;
2850 bottom_border += default_border->bottom;
2851 g_free( default_border );
2852 }
2853 }
2854
2855 DoMoveWindow( m_x-top_border,
2856 m_y-left_border,
2857 m_width+left_border+right_border,
2858 m_height+top_border+bottom_border );
2859 }
2860
2861 if (m_hasScrolling)
2862 {
2863 /* Sometimes the client area changes size without the
2864 whole windows's size changing, but if the whole
2865 windows's size doesn't change, no wxSizeEvent will
2866 normally be sent. Here we add an extra test if
2867 the client test has been changed and this will
2868 be used then. */
2869 GetClientSize( &m_oldClientWidth, &m_oldClientHeight );
2870 }
2871
2872 /*
2873 wxPrintf( "OnSize sent from " );
2874 if (GetClassInfo() && GetClassInfo()->GetClassName())
2875 wxPrintf( GetClassInfo()->GetClassName() );
2876 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
2877 */
2878
2879 if (!m_nativeSizeEvent)
2880 {
2881 wxSizeEvent event( wxSize(m_width,m_height), GetId() );
2882 event.SetEventObject( this );
2883 GetEventHandler()->ProcessEvent( event );
2884 }
2885
2886 m_resizing = false;
2887 }
2888
2889 bool wxWindowGTK::GtkShowFromOnIdle()
2890 {
2891 if (IsShown() && m_showOnIdle && !GTK_WIDGET_VISIBLE (m_widget))
2892 {
2893 GtkAllocation alloc;
2894 alloc.x = m_x;
2895 alloc.y = m_y;
2896 alloc.width = m_width;
2897 alloc.height = m_height;
2898 gtk_widget_size_allocate( m_widget, &alloc );
2899 gtk_widget_show( m_widget );
2900 wxShowEvent eventShow(GetId(), true);
2901 eventShow.SetEventObject(this);
2902 GetEventHandler()->ProcessEvent(eventShow);
2903 m_showOnIdle = false;
2904 return true;
2905 }
2906
2907 return false;
2908 }
2909
2910 void wxWindowGTK::OnInternalIdle()
2911 {
2912 // Check if we have to show window now
2913 if (GtkShowFromOnIdle()) return;
2914
2915 if ( m_dirtyTabOrder )
2916 {
2917 m_dirtyTabOrder = false;
2918 RealizeTabOrder();
2919 }
2920
2921 // Update style if the window was not yet realized
2922 // and SetBackgroundStyle(wxBG_STYLE_CUSTOM) was called
2923 if (m_needsStyleChange)
2924 {
2925 SetBackgroundStyle(GetBackgroundStyle());
2926 m_needsStyleChange = false;
2927 }
2928
2929 wxCursor cursor = m_cursor;
2930 if (g_globalCursor.Ok()) cursor = g_globalCursor;
2931
2932 if (cursor.Ok())
2933 {
2934 /* I now set the cursor anew in every OnInternalIdle call
2935 as setting the cursor in a parent window also effects the
2936 windows above so that checking for the current cursor is
2937 not possible. */
2938
2939 if (m_wxwindow)
2940 {
2941 GdkWindow *window = GTK_PIZZA(m_wxwindow)->bin_window;
2942 if (window)
2943 gdk_window_set_cursor( window, cursor.GetCursor() );
2944
2945 if (!g_globalCursor.Ok())
2946 cursor = *wxSTANDARD_CURSOR;
2947
2948 window = m_widget->window;
2949 if ((window) && !(GTK_WIDGET_NO_WINDOW(m_widget)))
2950 gdk_window_set_cursor( window, cursor.GetCursor() );
2951
2952 }
2953 else if ( m_widget )
2954 {
2955 GdkWindow *window = m_widget->window;
2956 if ( window && !GTK_WIDGET_NO_WINDOW(m_widget) )
2957 gdk_window_set_cursor( window, cursor.GetCursor() );
2958 }
2959 }
2960
2961 if (wxUpdateUIEvent::CanUpdate(this))
2962 UpdateWindowUI(wxUPDATE_UI_FROMIDLE);
2963 }
2964
2965 void wxWindowGTK::DoGetSize( int *width, int *height ) const
2966 {
2967 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2968
2969 if (width) (*width) = m_width;
2970 if (height) (*height) = m_height;
2971 }
2972
2973 void wxWindowGTK::DoSetClientSize( int width, int height )
2974 {
2975 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2976
2977 if (m_wxwindow)
2978 {
2979 int dw = 0;
2980 int dh = 0;
2981
2982 if (m_hasScrolling)
2983 {
2984 GetScrollbarWidth(m_widget, dw, dh);
2985 }
2986
2987 #ifndef __WXUNIVERSAL__
2988 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
2989 {
2990 // shadow border size is 2
2991 dw += 2 * 2;
2992 dh += 2 * 2;
2993 }
2994 if (HasFlag(wxSIMPLE_BORDER))
2995 {
2996 // simple border size is 1
2997 dw += 1 * 2;
2998 dh += 1 * 2;
2999 }
3000 #endif // __WXUNIVERSAL__
3001
3002 width += dw;
3003 height += dh;
3004 }
3005
3006 SetSize(width, height);
3007 }
3008
3009 void wxWindowGTK::DoGetClientSize( int *width, int *height ) const
3010 {
3011 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3012
3013 int w = m_width;
3014 int h = m_height;
3015
3016 if (m_wxwindow)
3017 {
3018 int dw = 0;
3019 int dh = 0;
3020
3021 if (m_hasScrolling)
3022 {
3023 GetScrollbarWidth(m_widget, dw, dh);
3024 }
3025
3026 #ifndef __WXUNIVERSAL__
3027 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
3028 {
3029 // shadow border size is 2
3030 dw += 2 * 2;
3031 dh += 2 * 2;
3032 }
3033 if (HasFlag(wxSIMPLE_BORDER))
3034 {
3035 // simple border size is 1
3036 dw += 1 * 2;
3037 dh += 1 * 2;
3038 }
3039 #endif // __WXUNIVERSAL__
3040
3041 w -= dw;
3042 h -= dh;
3043 if (w < 0)
3044 w = 0;
3045 if (h < 0)
3046 h = 0;
3047 }
3048
3049 if (width) *width = w;
3050 if (height) *height = h;
3051 }
3052
3053 void wxWindowGTK::DoGetPosition( int *x, int *y ) const
3054 {
3055 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3056
3057 int dx = 0;
3058 int dy = 0;
3059 if (m_parent && m_parent->m_wxwindow)
3060 {
3061 GtkPizza *pizza = GTK_PIZZA(m_parent->m_wxwindow);
3062 dx = gtk_pizza_get_xoffset( pizza );
3063 dy = gtk_pizza_get_yoffset( pizza );
3064 }
3065
3066 if (m_x == -1 && m_y == -1)
3067 {
3068 GdkWindow *source = (GdkWindow *) NULL;
3069 if (m_wxwindow)
3070 source = GTK_PIZZA(m_wxwindow)->bin_window;
3071 else
3072 source = m_widget->window;
3073
3074 if (source)
3075 {
3076 int org_x = 0;
3077 int org_y = 0;
3078 gdk_window_get_origin( source, &org_x, &org_y );
3079
3080 if (GetParent())
3081 GetParent()->ScreenToClient(&org_x, &org_y);
3082
3083 ((wxWindowGTK*) this)->m_x = org_x;
3084 ((wxWindowGTK*) this)->m_y = org_y;
3085 }
3086 }
3087
3088 if (x) (*x) = m_x - dx;
3089 if (y) (*y) = m_y - dy;
3090 }
3091
3092 void wxWindowGTK::DoClientToScreen( int *x, int *y ) const
3093 {
3094 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3095
3096 if (!m_widget->window) return;
3097
3098 GdkWindow *source = (GdkWindow *) NULL;
3099 if (m_wxwindow)
3100 source = GTK_PIZZA(m_wxwindow)->bin_window;
3101 else
3102 source = m_widget->window;
3103
3104 int org_x = 0;
3105 int org_y = 0;
3106 gdk_window_get_origin( source, &org_x, &org_y );
3107
3108 if (!m_wxwindow)
3109 {
3110 if (GTK_WIDGET_NO_WINDOW (m_widget))
3111 {
3112 org_x += m_widget->allocation.x;
3113 org_y += m_widget->allocation.y;
3114 }
3115 }
3116
3117
3118 if (x)
3119 {
3120 if (GetLayoutDirection() == wxLayout_RightToLeft)
3121 *x = (GetClientSize().x - *x) + org_x;
3122 else
3123 *x += org_x;
3124 }
3125
3126 if (y) *y += org_y;
3127 }
3128
3129 void wxWindowGTK::DoScreenToClient( int *x, int *y ) const
3130 {
3131 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3132
3133 if (!m_widget->window) return;
3134
3135 GdkWindow *source = (GdkWindow *) NULL;
3136 if (m_wxwindow)
3137 source = GTK_PIZZA(m_wxwindow)->bin_window;
3138 else
3139 source = m_widget->window;
3140
3141 int org_x = 0;
3142 int org_y = 0;
3143 gdk_window_get_origin( source, &org_x, &org_y );
3144
3145 if (!m_wxwindow)
3146 {
3147 if (GTK_WIDGET_NO_WINDOW (m_widget))
3148 {
3149 org_x += m_widget->allocation.x;
3150 org_y += m_widget->allocation.y;
3151 }
3152 }
3153
3154 if (x)
3155 {
3156 if (GetLayoutDirection() == wxLayout_RightToLeft)
3157 *x = (GetClientSize().x - *x) - org_x;
3158 else
3159 *x -= org_x;
3160 }
3161 if (y) *y -= org_y;
3162 }
3163
3164 bool wxWindowGTK::Show( bool show )
3165 {
3166 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3167
3168 if (!wxWindowBase::Show(show))
3169 {
3170 // nothing to do
3171 return false;
3172 }
3173
3174 if (show)
3175 {
3176 if (!m_showOnIdle)
3177 {
3178 gtk_widget_show( m_widget );
3179 wxShowEvent eventShow(GetId(), show);
3180 eventShow.SetEventObject(this);
3181 GetEventHandler()->ProcessEvent(eventShow);
3182 }
3183 }
3184 else
3185 {
3186 gtk_widget_hide( m_widget );
3187 wxShowEvent eventShow(GetId(), show);
3188 eventShow.SetEventObject(this);
3189 GetEventHandler()->ProcessEvent(eventShow);
3190 }
3191
3192 return true;
3193 }
3194
3195 static void wxWindowNotifyEnable(wxWindowGTK* win, bool enable)
3196 {
3197 win->OnParentEnable(enable);
3198
3199 // Recurse, so that children have the opportunity to Do The Right Thing
3200 // and reset colours that have been messed up by a parent's (really ancestor's)
3201 // Enable call
3202 for ( wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst();
3203 node;
3204 node = node->GetNext() )
3205 {
3206 wxWindow *child = node->GetData();
3207 if (!child->IsKindOf(CLASSINFO(wxDialog)) && !child->IsKindOf(CLASSINFO(wxFrame)))
3208 wxWindowNotifyEnable(child, enable);
3209 }
3210 }
3211
3212 bool wxWindowGTK::Enable( bool enable )
3213 {
3214 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3215
3216 if (!wxWindowBase::Enable(enable))
3217 {
3218 // nothing to do
3219 return false;
3220 }
3221
3222 gtk_widget_set_sensitive( m_widget, enable );
3223 if ( m_wxwindow )
3224 gtk_widget_set_sensitive( m_wxwindow, enable );
3225
3226 wxWindowNotifyEnable(this, enable);
3227
3228 return true;
3229 }
3230
3231 int wxWindowGTK::GetCharHeight() const
3232 {
3233 wxCHECK_MSG( (m_widget != NULL), 12, wxT("invalid window") );
3234
3235 wxFont font = GetFont();
3236 wxCHECK_MSG( font.Ok(), 12, wxT("invalid font") );
3237
3238 PangoContext *context = NULL;
3239 if (m_widget)
3240 context = gtk_widget_get_pango_context( m_widget );
3241
3242 if (!context)
3243 return 0;
3244
3245 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
3246 PangoLayout *layout = pango_layout_new(context);
3247 pango_layout_set_font_description(layout, desc);
3248 pango_layout_set_text(layout, "H", 1);
3249 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3250
3251 PangoRectangle rect;
3252 pango_layout_line_get_extents(line, NULL, &rect);
3253
3254 g_object_unref (layout);
3255
3256 return (int) PANGO_PIXELS(rect.height);
3257 }
3258
3259 int wxWindowGTK::GetCharWidth() const
3260 {
3261 wxCHECK_MSG( (m_widget != NULL), 8, wxT("invalid window") );
3262
3263 wxFont font = GetFont();
3264 wxCHECK_MSG( font.Ok(), 8, wxT("invalid font") );
3265
3266 PangoContext *context = NULL;
3267 if (m_widget)
3268 context = gtk_widget_get_pango_context( m_widget );
3269
3270 if (!context)
3271 return 0;
3272
3273 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
3274 PangoLayout *layout = pango_layout_new(context);
3275 pango_layout_set_font_description(layout, desc);
3276 pango_layout_set_text(layout, "g", 1);
3277 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3278
3279 PangoRectangle rect;
3280 pango_layout_line_get_extents(line, NULL, &rect);
3281
3282 g_object_unref (layout);
3283
3284 return (int) PANGO_PIXELS(rect.width);
3285 }
3286
3287 void wxWindowGTK::GetTextExtent( const wxString& string,
3288 int *x,
3289 int *y,
3290 int *descent,
3291 int *externalLeading,
3292 const wxFont *theFont ) const
3293 {
3294 wxFont fontToUse = theFont ? *theFont : GetFont();
3295
3296 wxCHECK_RET( fontToUse.Ok(), wxT("invalid font") );
3297
3298 if (string.empty())
3299 {
3300 if (x) (*x) = 0;
3301 if (y) (*y) = 0;
3302 return;
3303 }
3304
3305 PangoContext *context = NULL;
3306 if (m_widget)
3307 context = gtk_widget_get_pango_context( m_widget );
3308
3309 if (!context)
3310 {
3311 if (x) (*x) = 0;
3312 if (y) (*y) = 0;
3313 return;
3314 }
3315
3316 PangoFontDescription *desc = fontToUse.GetNativeFontInfo()->description;
3317 PangoLayout *layout = pango_layout_new(context);
3318 pango_layout_set_font_description(layout, desc);
3319 {
3320 const wxCharBuffer data = wxGTK_CONV( string );
3321 if ( data )
3322 pango_layout_set_text(layout, data, strlen(data));
3323 }
3324
3325 PangoRectangle rect;
3326 pango_layout_get_extents(layout, NULL, &rect);
3327
3328 if (x) (*x) = (wxCoord) PANGO_PIXELS(rect.width);
3329 if (y) (*y) = (wxCoord) PANGO_PIXELS(rect.height);
3330 if (descent)
3331 {
3332 PangoLayoutIter *iter = pango_layout_get_iter(layout);
3333 int baseline = pango_layout_iter_get_baseline(iter);
3334 pango_layout_iter_free(iter);
3335 *descent = *y - PANGO_PIXELS(baseline);
3336 }
3337 if (externalLeading) (*externalLeading) = 0; // ??
3338
3339 g_object_unref (layout);
3340 }
3341
3342 bool wxWindowGTK::GTKSetDelayedFocusIfNeeded()
3343 {
3344 if ( g_delayedFocus == this )
3345 {
3346 if ( GTK_WIDGET_REALIZED(m_widget) )
3347 {
3348 gtk_widget_grab_focus(m_widget);
3349 g_delayedFocus = NULL;
3350
3351 return true;
3352 }
3353 }
3354
3355 return false;
3356 }
3357
3358 void wxWindowGTK::SetFocus()
3359 {
3360 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3361 if ( m_hasFocus )
3362 {
3363 // don't do anything if we already have focus
3364 return;
3365 }
3366
3367 if (m_wxwindow)
3368 {
3369 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow))
3370 {
3371 gtk_widget_grab_focus (m_wxwindow);
3372 }
3373 }
3374 else if (m_widget)
3375 {
3376 if (GTK_IS_CONTAINER(m_widget))
3377 {
3378 gtk_widget_child_focus( m_widget, GTK_DIR_TAB_FORWARD );
3379 }
3380 else
3381 if (GTK_WIDGET_CAN_FOCUS(m_widget) && !GTK_WIDGET_HAS_FOCUS (m_widget) )
3382 {
3383
3384 if (!GTK_WIDGET_REALIZED(m_widget))
3385 {
3386 // we can't set the focus to the widget now so we remember that
3387 // it should be focused and will do it later, during the idle
3388 // time, as soon as we can
3389 wxLogTrace(TRACE_FOCUS,
3390 _T("Delaying setting focus to %s(%s)"),
3391 GetClassInfo()->GetClassName(), GetLabel().c_str());
3392
3393 g_delayedFocus = this;
3394 }
3395 else
3396 {
3397 wxLogTrace(TRACE_FOCUS,
3398 _T("Setting focus to %s(%s)"),
3399 GetClassInfo()->GetClassName(), GetLabel().c_str());
3400
3401 gtk_widget_grab_focus (m_widget);
3402 }
3403 }
3404 else
3405 {
3406 wxLogTrace(TRACE_FOCUS,
3407 _T("Can't set focus to %s(%s)"),
3408 GetClassInfo()->GetClassName(), GetLabel().c_str());
3409 }
3410 }
3411 }
3412
3413 bool wxWindowGTK::AcceptsFocus() const
3414 {
3415 return m_acceptsFocus && wxWindowBase::AcceptsFocus();
3416 }
3417
3418 bool wxWindowGTK::Reparent( wxWindowBase *newParentBase )
3419 {
3420 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3421
3422 wxWindowGTK *oldParent = m_parent,
3423 *newParent = (wxWindowGTK *)newParentBase;
3424
3425 wxASSERT( GTK_IS_WIDGET(m_widget) );
3426
3427 if ( !wxWindowBase::Reparent(newParent) )
3428 return false;
3429
3430 wxASSERT( GTK_IS_WIDGET(m_widget) );
3431
3432 /* prevent GTK from deleting the widget arbitrarily */
3433 gtk_widget_ref( m_widget );
3434
3435 if (oldParent)
3436 {
3437 gtk_container_remove( GTK_CONTAINER(m_widget->parent), m_widget );
3438 }
3439
3440 wxASSERT( GTK_IS_WIDGET(m_widget) );
3441
3442 if (newParent)
3443 {
3444 if (GTK_WIDGET_VISIBLE (newParent->m_widget))
3445 {
3446 m_showOnIdle = true;
3447 gtk_widget_hide( m_widget );
3448 }
3449
3450 /* insert GTK representation */
3451 (*(newParent->m_insertCallback))(newParent, this);
3452 }
3453
3454 /* reverse: prevent GTK from deleting the widget arbitrarily */
3455 gtk_widget_unref( m_widget );
3456
3457 SetLayoutDirection(wxLayout_Default);
3458
3459 return true;
3460 }
3461
3462 void wxWindowGTK::DoAddChild(wxWindowGTK *child)
3463 {
3464 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
3465
3466 wxASSERT_MSG( (child != NULL), wxT("invalid child window") );
3467
3468 wxASSERT_MSG( (m_insertCallback != NULL), wxT("invalid child insertion function") );
3469
3470 /* add to list */
3471 AddChild( child );
3472
3473 /* insert GTK representation */
3474 (*m_insertCallback)(this, child);
3475 }
3476
3477 void wxWindowGTK::AddChild(wxWindowBase *child)
3478 {
3479 wxWindowBase::AddChild(child);
3480 m_dirtyTabOrder = true;
3481 if (g_isIdle)
3482 wxapp_install_idle_handler();
3483 }
3484
3485 void wxWindowGTK::RemoveChild(wxWindowBase *child)
3486 {
3487 wxWindowBase::RemoveChild(child);
3488 m_dirtyTabOrder = true;
3489 if (g_isIdle)
3490 wxapp_install_idle_handler();
3491 }
3492
3493 /* static */
3494 wxLayoutDirection wxWindowGTK::GTKGetLayout(GtkWidget *widget)
3495 {
3496 return gtk_widget_get_direction(GTK_WIDGET(widget)) == GTK_TEXT_DIR_RTL
3497 ? wxLayout_RightToLeft
3498 : wxLayout_LeftToRight;
3499 }
3500
3501 /* static */
3502 void wxWindowGTK::GTKSetLayout(GtkWidget *widget, wxLayoutDirection dir)
3503 {
3504 wxASSERT_MSG( dir != wxLayout_Default, _T("invalid layout direction") );
3505
3506 gtk_widget_set_direction(GTK_WIDGET(widget),
3507 dir == wxLayout_RightToLeft ? GTK_TEXT_DIR_RTL
3508 : GTK_TEXT_DIR_LTR);
3509 }
3510
3511 wxLayoutDirection wxWindowGTK::GetLayoutDirection() const
3512 {
3513 return GTKGetLayout(m_widget);
3514 }
3515
3516 void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir)
3517 {
3518 if ( dir == wxLayout_Default )
3519 {
3520 const wxWindow *const parent = GetParent();
3521 if ( parent )
3522 {
3523 // inherit layout from parent.
3524 dir = parent->GetLayoutDirection();
3525 }
3526 else // no parent, use global default layout
3527 {
3528 dir = wxTheApp->GetLayoutDirection();
3529 }
3530 }
3531
3532 if ( dir == wxLayout_Default )
3533 return;
3534
3535 GTKSetLayout(m_widget, dir);
3536
3537 if (m_wxwindow)
3538 GTKSetLayout(m_wxwindow, dir);
3539 }
3540
3541 wxCoord
3542 wxWindowGTK::AdjustForLayoutDirection(wxCoord x,
3543 wxCoord WXUNUSED(width),
3544 wxCoord WXUNUSED(widthTotal)) const
3545 {
3546 // We now mirrors the coordinates of RTL windows in GtkPizza
3547 return x;
3548 }
3549
3550 void wxWindowGTK::DoMoveInTabOrder(wxWindow *win, MoveKind move)
3551 {
3552 wxWindowBase::DoMoveInTabOrder(win, move);
3553 m_dirtyTabOrder = true;
3554 if (g_isIdle)
3555 wxapp_install_idle_handler();
3556 }
3557
3558 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3559 {
3560 // none needed by default
3561 return false;
3562 }
3563
3564 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget* WXUNUSED(w))
3565 {
3566 // nothing to do by default since none is needed
3567 }
3568
3569 void wxWindowGTK::RealizeTabOrder()
3570 {
3571 if (m_wxwindow)
3572 {
3573 if ( !m_children.empty() )
3574 {
3575 // we don't only construct the correct focus chain but also use
3576 // this opportunity to update the mnemonic widgets for the widgets
3577 // that need them
3578
3579 GList *chain = NULL;
3580 wxWindowGTK* mnemonicWindow = NULL;
3581
3582 for ( wxWindowList::const_iterator i = m_children.begin();
3583 i != m_children.end();
3584 ++i )
3585 {
3586 wxWindowGTK *win = *i;
3587
3588 if ( mnemonicWindow )
3589 {
3590 if ( win->AcceptsFocusFromKeyboard() )
3591 {
3592 // wxComboBox et al. needs to focus on on a different
3593 // widget than m_widget, so if the main widget isn't
3594 // focusable try the connect widget
3595 GtkWidget* w = win->m_widget;
3596 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3597 {
3598 w = win->GetConnectWidget();
3599 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3600 w = NULL;
3601 }
3602
3603 if ( w )
3604 {
3605 mnemonicWindow->GTKWidgetDoSetMnemonic(w);
3606 mnemonicWindow = NULL;
3607 }
3608 }
3609 }
3610 else if ( win->GTKWidgetNeedsMnemonic() )
3611 {
3612 mnemonicWindow = win;
3613 }
3614
3615 chain = g_list_prepend(chain, win->m_widget);
3616 }
3617
3618 chain = g_list_reverse(chain);
3619
3620 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow), chain);
3621 g_list_free(chain);
3622 }
3623 else // no children
3624 {
3625 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow));
3626 }
3627 }
3628 }
3629
3630 void wxWindowGTK::Raise()
3631 {
3632 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3633
3634 if (m_wxwindow && m_wxwindow->window)
3635 {
3636 gdk_window_raise( m_wxwindow->window );
3637 }
3638 else if (m_widget->window)
3639 {
3640 gdk_window_raise( m_widget->window );
3641 }
3642 }
3643
3644 void wxWindowGTK::Lower()
3645 {
3646 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3647
3648 if (m_wxwindow && m_wxwindow->window)
3649 {
3650 gdk_window_lower( m_wxwindow->window );
3651 }
3652 else if (m_widget->window)
3653 {
3654 gdk_window_lower( m_widget->window );
3655 }
3656 }
3657
3658 bool wxWindowGTK::SetCursor( const wxCursor &cursor )
3659 {
3660 if ( !wxWindowBase::SetCursor(cursor.Ok() ? cursor : *wxSTANDARD_CURSOR) )
3661 return false;
3662
3663 GTKUpdateCursor();
3664
3665 return true;
3666 }
3667
3668 void wxWindowGTK::GTKUpdateCursor()
3669 {
3670 wxCursor cursor(g_globalCursor.Ok() ? g_globalCursor : GetCursor());
3671 if ( cursor.Ok() )
3672 {
3673 wxArrayGdkWindows windowsThis;
3674 GdkWindow * const winThis = GTKGetWindow(windowsThis);
3675 if ( winThis )
3676 {
3677 gdk_window_set_cursor(winThis, cursor.GetCursor());
3678 }
3679 else
3680 {
3681 const size_t count = windowsThis.size();
3682 for ( size_t n = 0; n < count; n++ )
3683 {
3684 GdkWindow *win = windowsThis[n];
3685 if ( !win )
3686 {
3687 wxFAIL_MSG(_T("NULL window returned by GTKGetWindow()?"));
3688 continue;
3689 }
3690
3691 gdk_window_set_cursor(win, cursor.GetCursor());
3692 }
3693 }
3694 }
3695 }
3696
3697 void wxWindowGTK::WarpPointer( int x, int y )
3698 {
3699 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3700
3701 // We provide this function ourselves as it is
3702 // missing in GDK (top of this file).
3703
3704 GdkWindow *window = (GdkWindow*) NULL;
3705 if (m_wxwindow)
3706 window = GTK_PIZZA(m_wxwindow)->bin_window;
3707 else
3708 window = GetConnectWidget()->window;
3709
3710 if (window)
3711 gdk_window_warp_pointer( window, x, y );
3712 }
3713
3714 wxWindowGTK::ScrollDir wxWindowGTK::ScrollDirFromRange(GtkRange *range) const
3715 {
3716 // find the scrollbar which generated the event
3717 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
3718 {
3719 if ( range == m_scrollBar[dir] )
3720 return (ScrollDir)dir;
3721 }
3722
3723 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3724
3725 return ScrollDir_Max;
3726 }
3727
3728 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir, ScrollUnit unit, int units)
3729 {
3730 bool changed = false;
3731 GtkRange* range = m_scrollBar[dir];
3732 if ( range && units )
3733 {
3734 GtkAdjustment* adj = range->adjustment;
3735 gdouble inc = unit == ScrollUnit_Line ? adj->step_increment
3736 : adj->page_increment;
3737
3738 const int posOld = int(adj->value + 0.5);
3739 gtk_range_set_value(range, posOld + units*inc);
3740
3741 changed = int(adj->value + 0.5) != posOld;
3742 }
3743
3744 return changed;
3745 }
3746
3747 bool wxWindowGTK::ScrollLines(int lines)
3748 {
3749 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Line, lines);
3750 }
3751
3752 bool wxWindowGTK::ScrollPages(int pages)
3753 {
3754 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Page, pages);
3755 }
3756
3757 void wxWindowGTK::Refresh( bool eraseBackground, const wxRect *rect )
3758 {
3759 if (!m_widget)
3760 return;
3761 if (!m_widget->window)
3762 return;
3763
3764 if (m_wxwindow)
3765 {
3766 if (!GTK_PIZZA(m_wxwindow)->bin_window) return;
3767
3768 GdkRectangle gdk_rect,
3769 *p;
3770 if (rect)
3771 {
3772 gdk_rect.x = rect->x;
3773 gdk_rect.y = rect->y;
3774 gdk_rect.width = rect->width;
3775 gdk_rect.height = rect->height;
3776 if (GetLayoutDirection() == wxLayout_RightToLeft)
3777 gdk_rect.x = GetClientSize().x - gdk_rect.x - gdk_rect.width;
3778
3779 p = &gdk_rect;
3780 }
3781 else // invalidate everything
3782 {
3783 p = NULL;
3784 }
3785
3786 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow)->bin_window, p, TRUE );
3787 }
3788 }
3789
3790 void wxWindowGTK::Update()
3791 {
3792 GtkUpdate();
3793
3794 // when we call Update() we really want to update the window immediately on
3795 // screen, even if it means flushing the entire queue and hence slowing down
3796 // everything -- but it should still be done, it's just that Update() should
3797 // be called very rarely
3798 gdk_flush();
3799 }
3800
3801 void wxWindowGTK::GtkUpdate()
3802 {
3803 if (m_wxwindow && GTK_PIZZA(m_wxwindow)->bin_window)
3804 gdk_window_process_updates( GTK_PIZZA(m_wxwindow)->bin_window, FALSE );
3805 if (m_widget && m_widget->window)
3806 gdk_window_process_updates( m_widget->window, FALSE );
3807
3808 // for consistency with other platforms (and also because it's convenient
3809 // to be able to update an entire TLW by calling Update() only once), we
3810 // should also update all our children here
3811 for ( wxWindowList::compatibility_iterator node = GetChildren().GetFirst();
3812 node;
3813 node = node->GetNext() )
3814 {
3815 node->GetData()->GtkUpdate();
3816 }
3817 }
3818
3819 bool wxWindowGTK::DoIsExposed( int x, int y ) const
3820 {
3821 return m_updateRegion.Contains(x, y) != wxOutRegion;
3822 }
3823
3824
3825 bool wxWindowGTK::DoIsExposed( int x, int y, int w, int h ) const
3826 {
3827 #if 0
3828 if (GetLayoutDirection() == wxLayout_RightToLeft)
3829 return m_updateRegion.Contains(x-w, y, w, h) != wxOutRegion;
3830 else
3831 #endif
3832 return m_updateRegion.Contains(x, y, w, h) != wxOutRegion;
3833 }
3834
3835 void wxWindowGTK::GtkSendPaintEvents()
3836 {
3837 if (!m_wxwindow)
3838 {
3839 m_updateRegion.Clear();
3840 return;
3841 }
3842
3843 // Clip to paint region in wxClientDC
3844 m_clipPaintRegion = true;
3845
3846 m_nativeUpdateRegion = m_updateRegion;
3847
3848 if (GetLayoutDirection() == wxLayout_RightToLeft)
3849 {
3850 // Transform m_updateRegion under RTL
3851 m_updateRegion.Clear();
3852
3853 gint width;
3854 gdk_window_get_geometry( GTK_PIZZA(m_wxwindow)->bin_window,
3855 NULL, NULL, &width, NULL, NULL );
3856
3857 wxRegionIterator upd( m_nativeUpdateRegion );
3858 while (upd)
3859 {
3860 wxRect rect;
3861 rect.x = upd.GetX();
3862 rect.y = upd.GetY();
3863 rect.width = upd.GetWidth();
3864 rect.height = upd.GetHeight();
3865
3866 rect.x = width - rect.x - rect.width;
3867 m_updateRegion.Union( rect );
3868
3869 ++upd;
3870 }
3871 }
3872
3873 // widget to draw on
3874 GtkPizza *pizza = GTK_PIZZA (m_wxwindow);
3875
3876 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM))
3877 {
3878 // find ancestor from which to steal background
3879 wxWindow *parent = wxGetTopLevelParent((wxWindow *)this);
3880 if (!parent)
3881 parent = (wxWindow*)this;
3882
3883 if (GTK_WIDGET_MAPPED(parent->m_widget))
3884 {
3885 wxRegionIterator upd( m_nativeUpdateRegion );
3886 while (upd)
3887 {
3888 GdkRectangle rect;
3889 rect.x = upd.GetX();
3890 rect.y = upd.GetY();
3891 rect.width = upd.GetWidth();
3892 rect.height = upd.GetHeight();
3893
3894 gtk_paint_flat_box( parent->m_widget->style,
3895 pizza->bin_window,
3896 (GtkStateType)GTK_WIDGET_STATE(m_wxwindow),
3897 GTK_SHADOW_NONE,
3898 &rect,
3899 parent->m_widget,
3900 (char *)"base",
3901 0, 0, -1, -1 );
3902
3903 ++upd;
3904 }
3905 }
3906 }
3907 else
3908 {
3909 wxWindowDC dc( (wxWindow*)this );
3910 dc.SetClippingRegion( m_updateRegion );
3911
3912 wxEraseEvent erase_event( GetId(), &dc );
3913 erase_event.SetEventObject( this );
3914
3915 GetEventHandler()->ProcessEvent(erase_event);
3916 }
3917
3918 wxNcPaintEvent nc_paint_event( GetId() );
3919 nc_paint_event.SetEventObject( this );
3920 GetEventHandler()->ProcessEvent( nc_paint_event );
3921
3922 wxPaintEvent paint_event( GetId() );
3923 paint_event.SetEventObject( this );
3924 GetEventHandler()->ProcessEvent( paint_event );
3925
3926 m_clipPaintRegion = false;
3927
3928 m_updateRegion.Clear();
3929 m_nativeUpdateRegion.Clear();
3930 }
3931
3932 void wxWindowGTK::SetDoubleBuffered( bool on )
3933 {
3934 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3935
3936 if ( m_wxwindow )
3937 gtk_widget_set_double_buffered( m_wxwindow, on );
3938 }
3939
3940 void wxWindowGTK::ClearBackground()
3941 {
3942 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3943 }
3944
3945 #if wxUSE_TOOLTIPS
3946 void wxWindowGTK::DoSetToolTip( wxToolTip *tip )
3947 {
3948 wxWindowBase::DoSetToolTip(tip);
3949
3950 if (m_tooltip)
3951 m_tooltip->Apply( (wxWindow *)this );
3952 }
3953
3954 void wxWindowGTK::ApplyToolTip( GtkTooltips *tips, const wxChar *tip )
3955 {
3956 wxString tmp( tip );
3957 gtk_tooltips_set_tip( tips, GetConnectWidget(), wxGTK_CONV(tmp), (gchar*) NULL );
3958 }
3959 #endif // wxUSE_TOOLTIPS
3960
3961 bool wxWindowGTK::SetBackgroundColour( const wxColour &colour )
3962 {
3963 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
3964
3965 if (!wxWindowBase::SetBackgroundColour(colour))
3966 return false;
3967
3968 if (colour.Ok())
3969 {
3970 // We need the pixel value e.g. for background clearing.
3971 m_backgroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
3972 }
3973
3974 // apply style change (forceStyle=true so that new style is applied
3975 // even if the bg colour changed from valid to wxNullColour)
3976 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM)
3977 ApplyWidgetStyle(true);
3978
3979 return true;
3980 }
3981
3982 bool wxWindowGTK::SetForegroundColour( const wxColour &colour )
3983 {
3984 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
3985
3986 if (!wxWindowBase::SetForegroundColour(colour))
3987 {
3988 return false;
3989 }
3990
3991 if (colour.Ok())
3992 {
3993 // We need the pixel value e.g. for background clearing.
3994 m_foregroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
3995 }
3996
3997 // apply style change (forceStyle=true so that new style is applied
3998 // even if the bg colour changed from valid to wxNullColour):
3999 ApplyWidgetStyle(true);
4000
4001 return true;
4002 }
4003
4004 PangoContext *wxWindowGTK::GtkGetPangoDefaultContext()
4005 {
4006 return gtk_widget_get_pango_context( m_widget );
4007 }
4008
4009 GtkRcStyle *wxWindowGTK::CreateWidgetStyle(bool forceStyle)
4010 {
4011 // do we need to apply any changes at all?
4012 if ( !forceStyle &&
4013 !m_font.Ok() &&
4014 !m_foregroundColour.Ok() && !m_backgroundColour.Ok() )
4015 {
4016 return NULL;
4017 }
4018
4019 GtkRcStyle *style = gtk_rc_style_new();
4020
4021 if ( m_font.Ok() )
4022 {
4023 style->font_desc =
4024 pango_font_description_copy( m_font.GetNativeFontInfo()->description );
4025 }
4026
4027 if ( m_foregroundColour.Ok() )
4028 {
4029 const GdkColor *fg = m_foregroundColour.GetColor();
4030
4031 style->fg[GTK_STATE_NORMAL] = *fg;
4032 style->color_flags[GTK_STATE_NORMAL] = GTK_RC_FG;
4033
4034 style->fg[GTK_STATE_PRELIGHT] = *fg;
4035 style->color_flags[GTK_STATE_PRELIGHT] = GTK_RC_FG;
4036
4037 style->fg[GTK_STATE_ACTIVE] = *fg;
4038 style->color_flags[GTK_STATE_ACTIVE] = GTK_RC_FG;
4039 }
4040
4041 if ( m_backgroundColour.Ok() )
4042 {
4043 const GdkColor *bg = m_backgroundColour.GetColor();
4044
4045 style->bg[GTK_STATE_NORMAL] = *bg;
4046 style->base[GTK_STATE_NORMAL] = *bg;
4047 style->color_flags[GTK_STATE_NORMAL] = (GtkRcFlags)
4048 (style->color_flags[GTK_STATE_NORMAL] | GTK_RC_BG | GTK_RC_BASE);
4049
4050 style->bg[GTK_STATE_PRELIGHT] = *bg;
4051 style->base[GTK_STATE_PRELIGHT] = *bg;
4052 style->color_flags[GTK_STATE_PRELIGHT] = (GtkRcFlags)
4053 (style->color_flags[GTK_STATE_PRELIGHT] | GTK_RC_BG | GTK_RC_BASE);
4054
4055 style->bg[GTK_STATE_ACTIVE] = *bg;
4056 style->base[GTK_STATE_ACTIVE] = *bg;
4057 style->color_flags[GTK_STATE_ACTIVE] = (GtkRcFlags)
4058 (style->color_flags[GTK_STATE_ACTIVE] | GTK_RC_BG | GTK_RC_BASE);
4059
4060 style->bg[GTK_STATE_INSENSITIVE] = *bg;
4061 style->base[GTK_STATE_INSENSITIVE] = *bg;
4062 style->color_flags[GTK_STATE_INSENSITIVE] = (GtkRcFlags)
4063 (style->color_flags[GTK_STATE_INSENSITIVE] | GTK_RC_BG | GTK_RC_BASE);
4064 }
4065
4066 return style;
4067 }
4068
4069 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle)
4070 {
4071 GtkRcStyle *style = CreateWidgetStyle(forceStyle);
4072 if ( style )
4073 {
4074 DoApplyWidgetStyle(style);
4075 gtk_rc_style_unref(style);
4076 }
4077
4078 // Style change may affect GTK+'s size calculation:
4079 InvalidateBestSize();
4080 }
4081
4082 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle *style)
4083 {
4084 if (m_wxwindow)
4085 gtk_widget_modify_style(m_wxwindow, style);
4086 else
4087 gtk_widget_modify_style(m_widget, style);
4088 }
4089
4090 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style)
4091 {
4092 wxWindowBase::SetBackgroundStyle(style);
4093
4094 if (style == wxBG_STYLE_CUSTOM)
4095 {
4096 GdkWindow *window = (GdkWindow*) NULL;
4097 if (m_wxwindow)
4098 window = GTK_PIZZA(m_wxwindow)->bin_window;
4099 else
4100 window = GetConnectWidget()->window;
4101
4102 if (window)
4103 {
4104 // Make sure GDK/X11 doesn't refresh the window
4105 // automatically.
4106 gdk_window_set_back_pixmap( window, None, False );
4107 #ifdef __X__
4108 Display* display = GDK_WINDOW_DISPLAY(window);
4109 XFlush(display);
4110 #endif
4111 m_needsStyleChange = false;
4112 }
4113 else
4114 // Do in OnIdle, because the window is not yet available
4115 m_needsStyleChange = true;
4116
4117 // Don't apply widget style, or we get a grey background
4118 }
4119 else
4120 {
4121 // apply style change (forceStyle=true so that new style is applied
4122 // even if the bg colour changed from valid to wxNullColour):
4123 ApplyWidgetStyle(true);
4124 }
4125 return true;
4126 }
4127
4128 #if wxUSE_DRAG_AND_DROP
4129
4130 void wxWindowGTK::SetDropTarget( wxDropTarget *dropTarget )
4131 {
4132 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4133
4134 GtkWidget *dnd_widget = GetConnectWidget();
4135
4136 if (m_dropTarget) m_dropTarget->UnregisterWidget( dnd_widget );
4137
4138 if (m_dropTarget) delete m_dropTarget;
4139 m_dropTarget = dropTarget;
4140
4141 if (m_dropTarget) m_dropTarget->RegisterWidget( dnd_widget );
4142 }
4143
4144 #endif // wxUSE_DRAG_AND_DROP
4145
4146 GtkWidget* wxWindowGTK::GetConnectWidget()
4147 {
4148 GtkWidget *connect_widget = m_widget;
4149 if (m_wxwindow) connect_widget = m_wxwindow;
4150
4151 return connect_widget;
4152 }
4153
4154 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow *window) const
4155 {
4156 wxArrayGdkWindows windowsThis;
4157 GdkWindow * const winThis = GTKGetWindow(windowsThis);
4158
4159 return winThis ? window == winThis
4160 : windowsThis.Index(window) != wxNOT_FOUND;
4161 }
4162
4163 GdkWindow *wxWindowGTK::GTKGetWindow(wxArrayGdkWindows& WXUNUSED(windows)) const
4164 {
4165 return m_wxwindow ? GTK_PIZZA(m_wxwindow)->bin_window : m_widget->window;
4166 }
4167
4168 bool wxWindowGTK::SetFont( const wxFont &font )
4169 {
4170 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
4171
4172 if (!wxWindowBase::SetFont(font))
4173 return false;
4174
4175 // apply style change (forceStyle=true so that new style is applied
4176 // even if the font changed from valid to wxNullFont):
4177 ApplyWidgetStyle(true);
4178
4179 return true;
4180 }
4181
4182 void wxWindowGTK::DoCaptureMouse()
4183 {
4184 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4185
4186 GdkWindow *window = (GdkWindow*) NULL;
4187 if (m_wxwindow)
4188 window = GTK_PIZZA(m_wxwindow)->bin_window;
4189 else
4190 window = GetConnectWidget()->window;
4191
4192 wxCHECK_RET( window, _T("CaptureMouse() failed") );
4193
4194 const wxCursor* cursor = &m_cursor;
4195 if (!cursor->Ok())
4196 cursor = wxSTANDARD_CURSOR;
4197
4198 gdk_pointer_grab( window, FALSE,
4199 (GdkEventMask)
4200 (GDK_BUTTON_PRESS_MASK |
4201 GDK_BUTTON_RELEASE_MASK |
4202 GDK_POINTER_MOTION_HINT_MASK |
4203 GDK_POINTER_MOTION_MASK),
4204 (GdkWindow *) NULL,
4205 cursor->GetCursor(),
4206 (guint32)GDK_CURRENT_TIME );
4207 g_captureWindow = this;
4208 g_captureWindowHasMouse = true;
4209 }
4210
4211 void wxWindowGTK::DoReleaseMouse()
4212 {
4213 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4214
4215 wxCHECK_RET( g_captureWindow, wxT("can't release mouse - not captured") );
4216
4217 g_captureWindow = (wxWindowGTK*) NULL;
4218
4219 GdkWindow *window = (GdkWindow*) NULL;
4220 if (m_wxwindow)
4221 window = GTK_PIZZA(m_wxwindow)->bin_window;
4222 else
4223 window = GetConnectWidget()->window;
4224
4225 if (!window)
4226 return;
4227
4228 gdk_pointer_ungrab ( (guint32)GDK_CURRENT_TIME );
4229 }
4230
4231 /* static */
4232 wxWindow *wxWindowBase::GetCapture()
4233 {
4234 return (wxWindow *)g_captureWindow;
4235 }
4236
4237 bool wxWindowGTK::IsRetained() const
4238 {
4239 return false;
4240 }
4241
4242 void wxWindowGTK::BlockScrollEvent()
4243 {
4244 wxASSERT(!m_blockScrollEvent);
4245 m_blockScrollEvent = true;
4246 }
4247
4248 void wxWindowGTK::UnblockScrollEvent()
4249 {
4250 wxASSERT(m_blockScrollEvent);
4251 m_blockScrollEvent = false;
4252 }
4253
4254 void wxWindowGTK::SetScrollbar(int orient,
4255 int pos,
4256 int thumbVisible,
4257 int range,
4258 bool WXUNUSED(update))
4259 {
4260 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4261 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4262
4263 if (range > 0)
4264 {
4265 m_hasScrolling = true;
4266 }
4267 else
4268 {
4269 // GtkRange requires upper > lower
4270 range =
4271 thumbVisible = 1;
4272 }
4273
4274 if (pos > range - thumbVisible)
4275 pos = range - thumbVisible;
4276 if (pos < 0)
4277 pos = 0;
4278 GtkAdjustment* adj = m_scrollBar[ScrollDirFromOrient(orient)]->adjustment;
4279 adj->step_increment = 1;
4280 adj->page_increment =
4281 adj->page_size = thumbVisible;
4282 adj->upper = range;
4283 SetScrollPos(orient, pos);
4284 gtk_adjustment_changed(adj);
4285 }
4286
4287 void wxWindowGTK::SetScrollPos(int orient, int pos, bool WXUNUSED(refresh))
4288 {
4289 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4290 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4291
4292 // This check is more than an optimization. Without it, the slider
4293 // will not move smoothly while tracking when using wxScrollHelper.
4294 if (GetScrollPos(orient) != pos)
4295 {
4296 const int dir = ScrollDirFromOrient(orient);
4297 GtkAdjustment* adj = m_scrollBar[dir]->adjustment;
4298 const int max = int(adj->upper - adj->page_size);
4299 if (pos > max)
4300 pos = max;
4301 if (pos < 0)
4302 pos = 0;
4303 m_scrollPos[dir] = adj->value = pos;
4304
4305 // If a "value_changed" signal emission is not already in progress
4306 if (!m_blockValueChanged[dir])
4307 {
4308 gtk_adjustment_value_changed(adj);
4309 }
4310 }
4311 }
4312
4313 int wxWindowGTK::GetScrollThumb(int orient) const
4314 {
4315 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4316 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4317
4318 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->page_size);
4319 }
4320
4321 int wxWindowGTK::GetScrollPos( int orient ) const
4322 {
4323 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4324 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4325
4326 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->value + 0.5);
4327 }
4328
4329 int wxWindowGTK::GetScrollRange( int orient ) const
4330 {
4331 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4332 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4333
4334 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->upper);
4335 }
4336
4337 // Determine if increment is the same as +/-x, allowing for some small
4338 // difference due to possible inexactness in floating point arithmetic
4339 static inline bool IsScrollIncrement(double increment, double x)
4340 {
4341 wxASSERT(increment > 0);
4342 const double tolerance = 1.0 / 1024;
4343 return fabs(increment - fabs(x)) < tolerance;
4344 }
4345
4346 wxEventType wxWindowGTK::GetScrollEventType(GtkRange* range)
4347 {
4348 DEBUG_MAIN_THREAD
4349
4350 if (g_isIdle)
4351 wxapp_install_idle_handler();
4352
4353 wxASSERT(range == m_scrollBar[0] || range == m_scrollBar[1]);
4354
4355 const int barIndex = range == m_scrollBar[1];
4356 GtkAdjustment* adj = range->adjustment;
4357
4358 const int value = int(adj->value + 0.5);
4359
4360 // save previous position
4361 const double oldPos = m_scrollPos[barIndex];
4362 // update current position
4363 m_scrollPos[barIndex] = adj->value;
4364 // If event should be ignored, or integral position has not changed
4365 if (!m_hasVMT || g_blockEventsOnDrag || value == int(oldPos + 0.5))
4366 {
4367 return wxEVT_NULL;
4368 }
4369
4370 wxEventType eventType = wxEVT_SCROLL_THUMBTRACK;
4371 if (!m_isScrolling)
4372 {
4373 // Difference from last change event
4374 const double diff = adj->value - oldPos;
4375 const bool isDown = diff > 0;
4376
4377 if (IsScrollIncrement(adj->step_increment, diff))
4378 {
4379 eventType = isDown ? wxEVT_SCROLL_LINEDOWN : wxEVT_SCROLL_LINEUP;
4380 }
4381 else if (IsScrollIncrement(adj->page_increment, diff))
4382 {
4383 eventType = isDown ? wxEVT_SCROLL_PAGEDOWN : wxEVT_SCROLL_PAGEUP;
4384 }
4385 else if (m_mouseButtonDown)
4386 {
4387 // Assume track event
4388 m_isScrolling = true;
4389 }
4390 }
4391 return eventType;
4392 }
4393
4394 void wxWindowGTK::ScrollWindow( int dx, int dy, const wxRect* WXUNUSED(rect) )
4395 {
4396 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4397
4398 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4399
4400 // No scrolling requested.
4401 if ((dx == 0) && (dy == 0)) return;
4402
4403 m_clipPaintRegion = true;
4404
4405 if (GetLayoutDirection() == wxLayout_RightToLeft)
4406 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow), dx, -dy );
4407 else
4408 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow), -dx, -dy );
4409
4410 m_clipPaintRegion = false;
4411 }
4412
4413 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget* w, int wxstyle)
4414 {
4415 //RN: Note that static controls usually have no border on gtk, so maybe
4416 //it makes sense to treat that as simply no border at the wx level
4417 //as well...
4418 if (!(wxstyle & wxNO_BORDER) && !(wxstyle & wxBORDER_STATIC))
4419 {
4420 GtkShadowType gtkstyle;
4421
4422 if(wxstyle & wxBORDER_RAISED)
4423 gtkstyle = GTK_SHADOW_OUT;
4424 else if (wxstyle & wxBORDER_SUNKEN)
4425 gtkstyle = GTK_SHADOW_IN;
4426 else if (wxstyle & wxBORDER_DOUBLE)
4427 gtkstyle = GTK_SHADOW_ETCHED_IN;
4428 else //default
4429 gtkstyle = GTK_SHADOW_IN;
4430
4431 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w),
4432 gtkstyle );
4433 }
4434 }
4435
4436 void wxWindowGTK::SetWindowStyleFlag( long style )
4437 {
4438 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4439 wxWindowBase::SetWindowStyleFlag(style);
4440 }
4441
4442 // Find the wxWindow at the current mouse position, also returning the mouse
4443 // position.
4444 wxWindow* wxFindWindowAtPointer(wxPoint& pt)
4445 {
4446 pt = wxGetMousePosition();
4447 wxWindow* found = wxFindWindowAtPoint(pt);
4448 return found;
4449 }
4450
4451 // Get the current mouse position.
4452 wxPoint wxGetMousePosition()
4453 {
4454 /* This crashes when used within wxHelpContext,
4455 so we have to use the X-specific implementation below.
4456 gint x, y;
4457 GdkModifierType *mask;
4458 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4459
4460 return wxPoint(x, y);
4461 */
4462
4463 int x, y;
4464 GdkWindow* windowAtPtr = gdk_window_at_pointer(& x, & y);
4465
4466 Display *display = windowAtPtr ? GDK_WINDOW_XDISPLAY(windowAtPtr) : GDK_DISPLAY();
4467 Window rootWindow = RootWindowOfScreen (DefaultScreenOfDisplay(display));
4468 Window rootReturn, childReturn;
4469 int rootX, rootY, winX, winY;
4470 unsigned int maskReturn;
4471
4472 XQueryPointer (display,
4473 rootWindow,
4474 &rootReturn,
4475 &childReturn,
4476 &rootX, &rootY, &winX, &winY, &maskReturn);
4477 return wxPoint(rootX, rootY);
4478
4479 }
4480
4481 // Needed for implementing e.g. combobox on wxGTK within a modal dialog.
4482 void wxAddGrab(wxWindow* window)
4483 {
4484 gtk_grab_add( (GtkWidget*) window->GetHandle() );
4485 }
4486
4487 void wxRemoveGrab(wxWindow* window)
4488 {
4489 gtk_grab_remove( (GtkWidget*) window->GetHandle() );
4490 }