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