]> git.saurik.com Git - wxWidgets.git/blob - src/gtk/window.cpp
a177129ae7c3fda28b63fc8082204406ae6a441e
[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 // Update invalidated regions.
2907 GtkUpdate();
2908
2909 wxCursor cursor = m_cursor;
2910 if (g_globalCursor.Ok()) cursor = g_globalCursor;
2911
2912 if (cursor.Ok())
2913 {
2914 /* I now set the cursor anew in every OnInternalIdle call
2915 as setting the cursor in a parent window also effects the
2916 windows above so that checking for the current cursor is
2917 not possible. */
2918
2919 if (m_wxwindow)
2920 {
2921 GdkWindow *window = GTK_PIZZA(m_wxwindow)->bin_window;
2922 if (window)
2923 gdk_window_set_cursor( window, cursor.GetCursor() );
2924
2925 if (!g_globalCursor.Ok())
2926 cursor = *wxSTANDARD_CURSOR;
2927
2928 window = m_widget->window;
2929 if ((window) && !(GTK_WIDGET_NO_WINDOW(m_widget)))
2930 gdk_window_set_cursor( window, cursor.GetCursor() );
2931
2932 }
2933 else if ( m_widget )
2934 {
2935 GdkWindow *window = m_widget->window;
2936 if ( window && !GTK_WIDGET_NO_WINDOW(m_widget) )
2937 gdk_window_set_cursor( window, cursor.GetCursor() );
2938 }
2939 }
2940
2941 if (wxUpdateUIEvent::CanUpdate(this))
2942 UpdateWindowUI(wxUPDATE_UI_FROMIDLE);
2943 }
2944
2945 void wxWindowGTK::DoGetSize( int *width, int *height ) const
2946 {
2947 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2948
2949 if (width) (*width) = m_width;
2950 if (height) (*height) = m_height;
2951 }
2952
2953 void wxWindowGTK::DoSetClientSize( int width, int height )
2954 {
2955 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2956
2957 if (m_wxwindow)
2958 {
2959 int dw = 0;
2960 int dh = 0;
2961
2962 if (m_hasScrolling)
2963 {
2964 GetScrollbarWidth(m_widget, dw, dh);
2965 }
2966
2967 #ifndef __WXUNIVERSAL__
2968 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
2969 {
2970 // shadow border size is 2
2971 dw += 2 * 2;
2972 dh += 2 * 2;
2973 }
2974 if (HasFlag(wxSIMPLE_BORDER))
2975 {
2976 // simple border size is 1
2977 dw += 1 * 2;
2978 dh += 1 * 2;
2979 }
2980 #endif // __WXUNIVERSAL__
2981
2982 width += dw;
2983 height += dh;
2984 }
2985
2986 SetSize(width, height);
2987 }
2988
2989 void wxWindowGTK::DoGetClientSize( int *width, int *height ) const
2990 {
2991 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2992
2993 int w = m_width;
2994 int h = m_height;
2995
2996 if (m_wxwindow)
2997 {
2998 int dw = 0;
2999 int dh = 0;
3000
3001 if (m_hasScrolling)
3002 {
3003 GetScrollbarWidth(m_widget, dw, dh);
3004 }
3005
3006 #ifndef __WXUNIVERSAL__
3007 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
3008 {
3009 // shadow border size is 2
3010 dw += 2 * 2;
3011 dh += 2 * 2;
3012 }
3013 if (HasFlag(wxSIMPLE_BORDER))
3014 {
3015 // simple border size is 1
3016 dw += 1 * 2;
3017 dh += 1 * 2;
3018 }
3019 #endif // __WXUNIVERSAL__
3020
3021 w -= dw;
3022 h -= dh;
3023 }
3024
3025 if (width) *width = w;
3026 if (height) *height = h;
3027 }
3028
3029 void wxWindowGTK::DoGetPosition( int *x, int *y ) const
3030 {
3031 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3032
3033 int dx = 0;
3034 int dy = 0;
3035 if (m_parent && m_parent->m_wxwindow)
3036 {
3037 GtkPizza *pizza = GTK_PIZZA(m_parent->m_wxwindow);
3038 dx = pizza->xoffset;
3039 dy = pizza->yoffset;
3040 }
3041
3042 if (x) (*x) = m_x - dx;
3043 if (y) (*y) = m_y - dy;
3044 }
3045
3046 void wxWindowGTK::DoClientToScreen( int *x, int *y ) const
3047 {
3048 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3049
3050 if (!m_widget->window) return;
3051
3052 GdkWindow *source = (GdkWindow *) NULL;
3053 if (m_wxwindow)
3054 source = GTK_PIZZA(m_wxwindow)->bin_window;
3055 else
3056 source = m_widget->window;
3057
3058 int org_x = 0;
3059 int org_y = 0;
3060 gdk_window_get_origin( source, &org_x, &org_y );
3061
3062 if (!m_wxwindow)
3063 {
3064 if (GTK_WIDGET_NO_WINDOW (m_widget))
3065 {
3066 org_x += m_widget->allocation.x;
3067 org_y += m_widget->allocation.y;
3068 }
3069 }
3070
3071 if (x) *x += org_x;
3072 if (y) *y += org_y;
3073 }
3074
3075 void wxWindowGTK::DoScreenToClient( int *x, int *y ) const
3076 {
3077 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3078
3079 if (!m_widget->window) return;
3080
3081 GdkWindow *source = (GdkWindow *) NULL;
3082 if (m_wxwindow)
3083 source = GTK_PIZZA(m_wxwindow)->bin_window;
3084 else
3085 source = m_widget->window;
3086
3087 int org_x = 0;
3088 int org_y = 0;
3089 gdk_window_get_origin( source, &org_x, &org_y );
3090
3091 if (!m_wxwindow)
3092 {
3093 if (GTK_WIDGET_NO_WINDOW (m_widget))
3094 {
3095 org_x += m_widget->allocation.x;
3096 org_y += m_widget->allocation.y;
3097 }
3098 }
3099
3100 if (x) *x -= org_x;
3101 if (y) *y -= org_y;
3102 }
3103
3104 bool wxWindowGTK::Show( bool show )
3105 {
3106 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3107
3108 if (!wxWindowBase::Show(show))
3109 {
3110 // nothing to do
3111 return false;
3112 }
3113
3114 if (show)
3115 {
3116 if (!m_showOnIdle)
3117 {
3118 gtk_widget_show( m_widget );
3119 wxShowEvent eventShow(GetId(), show);
3120 eventShow.SetEventObject(this);
3121 GetEventHandler()->ProcessEvent(eventShow);
3122 }
3123 }
3124 else
3125 {
3126 gtk_widget_hide( m_widget );
3127 wxShowEvent eventShow(GetId(), show);
3128 eventShow.SetEventObject(this);
3129 GetEventHandler()->ProcessEvent(eventShow);
3130 }
3131
3132 return true;
3133 }
3134
3135 static void wxWindowNotifyEnable(wxWindowGTK* win, bool enable)
3136 {
3137 win->OnParentEnable(enable);
3138
3139 // Recurse, so that children have the opportunity to Do The Right Thing
3140 // and reset colours that have been messed up by a parent's (really ancestor's)
3141 // Enable call
3142 for ( wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst();
3143 node;
3144 node = node->GetNext() )
3145 {
3146 wxWindow *child = node->GetData();
3147 if (!child->IsKindOf(CLASSINFO(wxDialog)) && !child->IsKindOf(CLASSINFO(wxFrame)))
3148 wxWindowNotifyEnable(child, enable);
3149 }
3150 }
3151
3152 bool wxWindowGTK::Enable( bool enable )
3153 {
3154 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3155
3156 if (!wxWindowBase::Enable(enable))
3157 {
3158 // nothing to do
3159 return false;
3160 }
3161
3162 gtk_widget_set_sensitive( m_widget, enable );
3163 if ( m_wxwindow )
3164 gtk_widget_set_sensitive( m_wxwindow, enable );
3165
3166 wxWindowNotifyEnable(this, enable);
3167
3168 return true;
3169 }
3170
3171 int wxWindowGTK::GetCharHeight() const
3172 {
3173 wxCHECK_MSG( (m_widget != NULL), 12, wxT("invalid window") );
3174
3175 wxFont font = GetFont();
3176 wxCHECK_MSG( font.Ok(), 12, wxT("invalid font") );
3177
3178 PangoContext *context = NULL;
3179 if (m_widget)
3180 context = gtk_widget_get_pango_context( m_widget );
3181
3182 if (!context)
3183 return 0;
3184
3185 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
3186 PangoLayout *layout = pango_layout_new(context);
3187 pango_layout_set_font_description(layout, desc);
3188 pango_layout_set_text(layout, "H", 1);
3189 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3190
3191 PangoRectangle rect;
3192 pango_layout_line_get_extents(line, NULL, &rect);
3193
3194 g_object_unref (layout);
3195
3196 return (int) PANGO_PIXELS(rect.height);
3197 }
3198
3199 int wxWindowGTK::GetCharWidth() const
3200 {
3201 wxCHECK_MSG( (m_widget != NULL), 8, wxT("invalid window") );
3202
3203 wxFont font = GetFont();
3204 wxCHECK_MSG( font.Ok(), 8, wxT("invalid font") );
3205
3206 PangoContext *context = NULL;
3207 if (m_widget)
3208 context = gtk_widget_get_pango_context( m_widget );
3209
3210 if (!context)
3211 return 0;
3212
3213 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
3214 PangoLayout *layout = pango_layout_new(context);
3215 pango_layout_set_font_description(layout, desc);
3216 pango_layout_set_text(layout, "g", 1);
3217 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3218
3219 PangoRectangle rect;
3220 pango_layout_line_get_extents(line, NULL, &rect);
3221
3222 g_object_unref (layout);
3223
3224 return (int) PANGO_PIXELS(rect.width);
3225 }
3226
3227 void wxWindowGTK::GetTextExtent( const wxString& string,
3228 int *x,
3229 int *y,
3230 int *descent,
3231 int *externalLeading,
3232 const wxFont *theFont ) const
3233 {
3234 wxFont fontToUse = theFont ? *theFont : GetFont();
3235
3236 wxCHECK_RET( fontToUse.Ok(), wxT("invalid font") );
3237
3238 if (string.empty())
3239 {
3240 if (x) (*x) = 0;
3241 if (y) (*y) = 0;
3242 return;
3243 }
3244
3245 PangoContext *context = NULL;
3246 if (m_widget)
3247 context = gtk_widget_get_pango_context( m_widget );
3248
3249 if (!context)
3250 {
3251 if (x) (*x) = 0;
3252 if (y) (*y) = 0;
3253 return;
3254 }
3255
3256 PangoFontDescription *desc = fontToUse.GetNativeFontInfo()->description;
3257 PangoLayout *layout = pango_layout_new(context);
3258 pango_layout_set_font_description(layout, desc);
3259 {
3260 const wxCharBuffer data = wxGTK_CONV( string );
3261 if ( data )
3262 pango_layout_set_text(layout, data, strlen(data));
3263 }
3264
3265 PangoRectangle rect;
3266 pango_layout_get_extents(layout, NULL, &rect);
3267
3268 if (x) (*x) = (wxCoord) PANGO_PIXELS(rect.width);
3269 if (y) (*y) = (wxCoord) PANGO_PIXELS(rect.height);
3270 if (descent)
3271 {
3272 PangoLayoutIter *iter = pango_layout_get_iter(layout);
3273 int baseline = pango_layout_iter_get_baseline(iter);
3274 pango_layout_iter_free(iter);
3275 *descent = *y - PANGO_PIXELS(baseline);
3276 }
3277 if (externalLeading) (*externalLeading) = 0; // ??
3278
3279 g_object_unref (layout);
3280 }
3281
3282 bool wxWindowGTK::GTKSetDelayedFocusIfNeeded()
3283 {
3284 if ( g_delayedFocus == this )
3285 {
3286 if ( GTK_WIDGET_REALIZED(m_widget) )
3287 {
3288 gtk_widget_grab_focus(m_widget);
3289 g_delayedFocus = NULL;
3290
3291 return true;
3292 }
3293 }
3294
3295 return false;
3296 }
3297
3298 void wxWindowGTK::SetFocus()
3299 {
3300 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3301 if ( m_hasFocus )
3302 {
3303 // don't do anything if we already have focus
3304 return;
3305 }
3306
3307 if (m_wxwindow)
3308 {
3309 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow))
3310 {
3311 gtk_widget_grab_focus (m_wxwindow);
3312 }
3313 }
3314 else if (m_widget)
3315 {
3316 if (GTK_IS_CONTAINER(m_widget))
3317 {
3318 gtk_widget_child_focus( m_widget, GTK_DIR_TAB_FORWARD );
3319 }
3320 else
3321 if (GTK_WIDGET_CAN_FOCUS(m_widget) && !GTK_WIDGET_HAS_FOCUS (m_widget) )
3322 {
3323
3324 if (!GTK_WIDGET_REALIZED(m_widget))
3325 {
3326 // we can't set the focus to the widget now so we remember that
3327 // it should be focused and will do it later, during the idle
3328 // time, as soon as we can
3329 wxLogTrace(TRACE_FOCUS,
3330 _T("Delaying setting focus to %s(%s)"),
3331 GetClassInfo()->GetClassName(), GetLabel().c_str());
3332
3333 g_delayedFocus = this;
3334 }
3335 else
3336 {
3337 wxLogTrace(TRACE_FOCUS,
3338 _T("Setting focus to %s(%s)"),
3339 GetClassInfo()->GetClassName(), GetLabel().c_str());
3340
3341 gtk_widget_grab_focus (m_widget);
3342 }
3343 }
3344 else
3345 {
3346 wxLogTrace(TRACE_FOCUS,
3347 _T("Can't set focus to %s(%s)"),
3348 GetClassInfo()->GetClassName(), GetLabel().c_str());
3349 }
3350 }
3351 }
3352
3353 bool wxWindowGTK::AcceptsFocus() const
3354 {
3355 return m_acceptsFocus && wxWindowBase::AcceptsFocus();
3356 }
3357
3358 bool wxWindowGTK::Reparent( wxWindowBase *newParentBase )
3359 {
3360 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3361
3362 wxWindowGTK *oldParent = m_parent,
3363 *newParent = (wxWindowGTK *)newParentBase;
3364
3365 wxASSERT( GTK_IS_WIDGET(m_widget) );
3366
3367 if ( !wxWindowBase::Reparent(newParent) )
3368 return false;
3369
3370 wxASSERT( GTK_IS_WIDGET(m_widget) );
3371
3372 /* prevent GTK from deleting the widget arbitrarily */
3373 gtk_widget_ref( m_widget );
3374
3375 if (oldParent)
3376 {
3377 gtk_container_remove( GTK_CONTAINER(m_widget->parent), m_widget );
3378 }
3379
3380 wxASSERT( GTK_IS_WIDGET(m_widget) );
3381
3382 if (newParent)
3383 {
3384 if (GTK_WIDGET_VISIBLE (newParent->m_widget))
3385 {
3386 m_showOnIdle = true;
3387 gtk_widget_hide( m_widget );
3388 }
3389
3390 /* insert GTK representation */
3391 (*(newParent->m_insertCallback))(newParent, this);
3392 }
3393
3394 /* reverse: prevent GTK from deleting the widget arbitrarily */
3395 gtk_widget_unref( m_widget );
3396
3397 return true;
3398 }
3399
3400 void wxWindowGTK::DoAddChild(wxWindowGTK *child)
3401 {
3402 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
3403
3404 wxASSERT_MSG( (child != NULL), wxT("invalid child window") );
3405
3406 wxASSERT_MSG( (m_insertCallback != NULL), wxT("invalid child insertion function") );
3407
3408 /* add to list */
3409 AddChild( child );
3410
3411 /* insert GTK representation */
3412 (*m_insertCallback)(this, child);
3413 }
3414
3415 void wxWindowGTK::AddChild(wxWindowBase *child)
3416 {
3417 wxWindowBase::AddChild(child);
3418 m_dirtyTabOrder = true;
3419 if (g_isIdle)
3420 wxapp_install_idle_handler();
3421 }
3422
3423 void wxWindowGTK::RemoveChild(wxWindowBase *child)
3424 {
3425 wxWindowBase::RemoveChild(child);
3426 m_dirtyTabOrder = true;
3427 if (g_isIdle)
3428 wxapp_install_idle_handler();
3429 }
3430
3431 void wxWindowGTK::DoMoveInTabOrder(wxWindow *win, MoveKind move)
3432 {
3433 wxWindowBase::DoMoveInTabOrder(win, move);
3434 m_dirtyTabOrder = true;
3435 if (g_isIdle)
3436 wxapp_install_idle_handler();
3437 }
3438
3439 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3440 {
3441 // none needed by default
3442 return false;
3443 }
3444
3445 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget* WXUNUSED(w))
3446 {
3447 // nothing to do by default since none is needed
3448 }
3449
3450 void wxWindowGTK::RealizeTabOrder()
3451 {
3452 if (m_wxwindow)
3453 {
3454 if ( !m_children.empty() )
3455 {
3456 // we don't only construct the correct focus chain but also use
3457 // this opportunity to update the mnemonic widgets for the widgets
3458 // that need them
3459
3460 GList *chain = NULL;
3461 wxWindowGTK* mnemonicWindow = NULL;
3462
3463 for ( wxWindowList::const_iterator i = m_children.begin();
3464 i != m_children.end();
3465 ++i )
3466 {
3467 wxWindowGTK *win = *i;
3468
3469 if ( mnemonicWindow )
3470 {
3471 if ( win->AcceptsFocusFromKeyboard() )
3472 {
3473 // wxComboBox et al. needs to focus on on a different
3474 // widget than m_widget, so if the main widget isn't
3475 // focusable try the connect widget
3476 GtkWidget* w = win->m_widget;
3477 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3478 {
3479 w = win->GetConnectWidget();
3480 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3481 w = NULL;
3482 }
3483
3484 if ( w )
3485 {
3486 mnemonicWindow->GTKWidgetDoSetMnemonic(w);
3487 mnemonicWindow = NULL;
3488 }
3489 }
3490 }
3491 else if ( win->GTKWidgetNeedsMnemonic() )
3492 {
3493 mnemonicWindow = win;
3494 }
3495
3496 chain = g_list_prepend(chain, win->m_widget);
3497 }
3498
3499 chain = g_list_reverse(chain);
3500
3501 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow), chain);
3502 g_list_free(chain);
3503 }
3504 else // no children
3505 {
3506 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow));
3507 }
3508 }
3509 }
3510
3511 void wxWindowGTK::Raise()
3512 {
3513 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3514
3515 if (m_wxwindow && m_wxwindow->window)
3516 {
3517 gdk_window_raise( m_wxwindow->window );
3518 }
3519 else if (m_widget->window)
3520 {
3521 gdk_window_raise( m_widget->window );
3522 }
3523 }
3524
3525 void wxWindowGTK::Lower()
3526 {
3527 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3528
3529 if (m_wxwindow && m_wxwindow->window)
3530 {
3531 gdk_window_lower( m_wxwindow->window );
3532 }
3533 else if (m_widget->window)
3534 {
3535 gdk_window_lower( m_widget->window );
3536 }
3537 }
3538
3539 bool wxWindowGTK::SetCursor( const wxCursor &cursor )
3540 {
3541 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
3542
3543 if (cursor == m_cursor)
3544 return false;
3545
3546 if (g_isIdle)
3547 wxapp_install_idle_handler();
3548
3549 return wxWindowBase::SetCursor( cursor.Ok() ? cursor : *wxSTANDARD_CURSOR );
3550 }
3551
3552 void wxWindowGTK::GTKUpdateCursor()
3553 {
3554 wxCursor cursor(g_globalCursor.Ok() ? g_globalCursor : GetCursor());
3555 if ( cursor.Ok() )
3556 {
3557 wxArrayGdkWindows windowsThis;
3558 GdkWindow * const winThis = GTKGetWindow(windowsThis);
3559 if ( winThis )
3560 {
3561 gdk_window_set_cursor(winThis, cursor.GetCursor());
3562 }
3563 else
3564 {
3565 const size_t count = windowsThis.size();
3566 for ( size_t n = 0; n < count; n++ )
3567 {
3568 gdk_window_set_cursor(windowsThis[n], cursor.GetCursor());
3569 }
3570 }
3571 }
3572 }
3573
3574 void wxWindowGTK::WarpPointer( int x, int y )
3575 {
3576 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3577
3578 // We provide this function ourselves as it is
3579 // missing in GDK (top of this file).
3580
3581 GdkWindow *window = (GdkWindow*) NULL;
3582 if (m_wxwindow)
3583 window = GTK_PIZZA(m_wxwindow)->bin_window;
3584 else
3585 window = GetConnectWidget()->window;
3586
3587 if (window)
3588 gdk_window_warp_pointer( window, x, y );
3589 }
3590
3591 wxWindowGTK::ScrollDir wxWindowGTK::ScrollDirFromRange(GtkRange *range) const
3592 {
3593 // find the scrollbar which generated the event
3594 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
3595 {
3596 if ( range == m_scrollBar[dir] )
3597 return (ScrollDir)dir;
3598 }
3599
3600 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3601
3602 return ScrollDir_Max;
3603 }
3604
3605 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir, ScrollUnit unit, int units)
3606 {
3607 bool changed = false;
3608 GtkRange* range = m_scrollBar[dir];
3609 if ( range && units )
3610 {
3611 GtkAdjustment* adj = range->adjustment;
3612 gdouble inc = unit == ScrollUnit_Line ? adj->step_increment
3613 : adj->page_increment;
3614
3615 const int posOld = int(adj->value + 0.5);
3616 gtk_range_set_value(range, posOld + units*inc);
3617
3618 changed = int(adj->value + 0.5) != posOld;
3619 }
3620
3621 return changed;
3622 }
3623
3624 bool wxWindowGTK::ScrollLines(int lines)
3625 {
3626 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Line, lines);
3627 }
3628
3629 bool wxWindowGTK::ScrollPages(int pages)
3630 {
3631 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Page, pages);
3632 }
3633
3634 void wxWindowGTK::Refresh( bool eraseBackground, const wxRect *rect )
3635 {
3636 if (!m_widget)
3637 return;
3638 if (!m_widget->window)
3639 return;
3640
3641 if (m_wxwindow)
3642 {
3643 if (!GTK_PIZZA(m_wxwindow)->bin_window) return;
3644
3645 GdkRectangle gdk_rect,
3646 *p;
3647 if (rect)
3648 {
3649 gdk_rect.x = rect->x;
3650 gdk_rect.y = rect->y;
3651 gdk_rect.width = rect->width;
3652 gdk_rect.height = rect->height;
3653 p = &gdk_rect;
3654 }
3655 else // invalidate everything
3656 {
3657 p = NULL;
3658 }
3659
3660 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow)->bin_window, p, TRUE );
3661 }
3662 }
3663
3664 void wxWindowGTK::Update()
3665 {
3666 GtkUpdate();
3667
3668 // when we call Update() we really want to update the window immediately on
3669 // screen, even if it means flushing the entire queue and hence slowing down
3670 // everything -- but it should still be done, it's just that Update() should
3671 // be called very rarely
3672 gdk_flush();
3673 }
3674
3675 void wxWindowGTK::GtkUpdate()
3676 {
3677 if (m_wxwindow && GTK_PIZZA(m_wxwindow)->bin_window)
3678 gdk_window_process_updates( GTK_PIZZA(m_wxwindow)->bin_window, FALSE );
3679 if (m_widget && m_widget->window)
3680 gdk_window_process_updates( m_widget->window, FALSE );
3681
3682 // for consistency with other platforms (and also because it's convenient
3683 // to be able to update an entire TLW by calling Update() only once), we
3684 // should also update all our children here
3685 for ( wxWindowList::compatibility_iterator node = GetChildren().GetFirst();
3686 node;
3687 node = node->GetNext() )
3688 {
3689 node->GetData()->GtkUpdate();
3690 }
3691 }
3692
3693 void wxWindowGTK::GtkSendPaintEvents()
3694 {
3695 if (!m_wxwindow)
3696 {
3697 m_updateRegion.Clear();
3698 return;
3699 }
3700
3701 // Clip to paint region in wxClientDC
3702 m_clipPaintRegion = true;
3703
3704 // widget to draw on
3705 GtkPizza *pizza = GTK_PIZZA (m_wxwindow);
3706
3707 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM))
3708 {
3709 // find ancestor from which to steal background
3710 wxWindow *parent = wxGetTopLevelParent((wxWindow *)this);
3711 if (!parent)
3712 parent = (wxWindow*)this;
3713
3714 if (GTK_WIDGET_MAPPED(parent->m_widget))
3715 {
3716 wxRegionIterator upd( m_updateRegion );
3717 while (upd)
3718 {
3719 GdkRectangle rect;
3720 rect.x = upd.GetX();
3721 rect.y = upd.GetY();
3722 rect.width = upd.GetWidth();
3723 rect.height = upd.GetHeight();
3724
3725 gtk_paint_flat_box( parent->m_widget->style,
3726 pizza->bin_window,
3727 (GtkStateType)GTK_WIDGET_STATE(m_wxwindow),
3728 GTK_SHADOW_NONE,
3729 &rect,
3730 parent->m_widget,
3731 (char *)"base",
3732 0, 0, -1, -1 );
3733
3734 ++upd;
3735 }
3736 }
3737 }
3738 else
3739
3740 {
3741 wxWindowDC dc( (wxWindow*)this );
3742 dc.SetClippingRegion( m_updateRegion );
3743
3744 wxEraseEvent erase_event( GetId(), &dc );
3745 erase_event.SetEventObject( this );
3746
3747 GetEventHandler()->ProcessEvent(erase_event);
3748 }
3749
3750 wxNcPaintEvent nc_paint_event( GetId() );
3751 nc_paint_event.SetEventObject( this );
3752 GetEventHandler()->ProcessEvent( nc_paint_event );
3753
3754 wxPaintEvent paint_event( GetId() );
3755 paint_event.SetEventObject( this );
3756 GetEventHandler()->ProcessEvent( paint_event );
3757
3758 m_clipPaintRegion = false;
3759
3760 m_updateRegion.Clear();
3761 }
3762
3763 void wxWindowGTK::SetDoubleBuffered( bool on )
3764 {
3765 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3766
3767 if ( m_wxwindow )
3768 gtk_widget_set_double_buffered( m_wxwindow, on );
3769 }
3770
3771 void wxWindowGTK::ClearBackground()
3772 {
3773 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3774 }
3775
3776 #if wxUSE_TOOLTIPS
3777 void wxWindowGTK::DoSetToolTip( wxToolTip *tip )
3778 {
3779 wxWindowBase::DoSetToolTip(tip);
3780
3781 if (m_tooltip)
3782 m_tooltip->Apply( (wxWindow *)this );
3783 }
3784
3785 void wxWindowGTK::ApplyToolTip( GtkTooltips *tips, const wxChar *tip )
3786 {
3787 wxString tmp( tip );
3788 gtk_tooltips_set_tip( tips, GetConnectWidget(), wxGTK_CONV(tmp), (gchar*) NULL );
3789 }
3790 #endif // wxUSE_TOOLTIPS
3791
3792 bool wxWindowGTK::SetBackgroundColour( const wxColour &colour )
3793 {
3794 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
3795
3796 if (!wxWindowBase::SetBackgroundColour(colour))
3797 return false;
3798
3799 if (colour.Ok())
3800 {
3801 // We need the pixel value e.g. for background clearing.
3802 m_backgroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
3803 }
3804
3805 // apply style change (forceStyle=true so that new style is applied
3806 // even if the bg colour changed from valid to wxNullColour)
3807 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM)
3808 ApplyWidgetStyle(true);
3809
3810 return true;
3811 }
3812
3813 bool wxWindowGTK::SetForegroundColour( const wxColour &colour )
3814 {
3815 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
3816
3817 if (!wxWindowBase::SetForegroundColour(colour))
3818 {
3819 return false;
3820 }
3821
3822 if (colour.Ok())
3823 {
3824 // We need the pixel value e.g. for background clearing.
3825 m_foregroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
3826 }
3827
3828 // apply style change (forceStyle=true so that new style is applied
3829 // even if the bg colour changed from valid to wxNullColour):
3830 ApplyWidgetStyle(true);
3831
3832 return true;
3833 }
3834
3835 PangoContext *wxWindowGTK::GtkGetPangoDefaultContext()
3836 {
3837 return gtk_widget_get_pango_context( m_widget );
3838 }
3839
3840 GtkRcStyle *wxWindowGTK::CreateWidgetStyle(bool forceStyle)
3841 {
3842 // do we need to apply any changes at all?
3843 if ( !forceStyle &&
3844 !m_font.Ok() &&
3845 !m_foregroundColour.Ok() && !m_backgroundColour.Ok() )
3846 {
3847 return NULL;
3848 }
3849
3850 GtkRcStyle *style = gtk_rc_style_new();
3851
3852 if ( m_font.Ok() )
3853 {
3854 style->font_desc =
3855 pango_font_description_copy( m_font.GetNativeFontInfo()->description );
3856 }
3857
3858 if ( m_foregroundColour.Ok() )
3859 {
3860 const GdkColor *fg = m_foregroundColour.GetColor();
3861
3862 style->fg[GTK_STATE_NORMAL] = *fg;
3863 style->color_flags[GTK_STATE_NORMAL] = GTK_RC_FG;
3864
3865 style->fg[GTK_STATE_PRELIGHT] = *fg;
3866 style->color_flags[GTK_STATE_PRELIGHT] = GTK_RC_FG;
3867
3868 style->fg[GTK_STATE_ACTIVE] = *fg;
3869 style->color_flags[GTK_STATE_ACTIVE] = GTK_RC_FG;
3870 }
3871
3872 if ( m_backgroundColour.Ok() )
3873 {
3874 const GdkColor *bg = m_backgroundColour.GetColor();
3875
3876 style->bg[GTK_STATE_NORMAL] = *bg;
3877 style->base[GTK_STATE_NORMAL] = *bg;
3878 style->color_flags[GTK_STATE_NORMAL] = (GtkRcFlags)
3879 (style->color_flags[GTK_STATE_NORMAL] | GTK_RC_BG | GTK_RC_BASE);
3880
3881 style->bg[GTK_STATE_PRELIGHT] = *bg;
3882 style->base[GTK_STATE_PRELIGHT] = *bg;
3883 style->color_flags[GTK_STATE_PRELIGHT] = (GtkRcFlags)
3884 (style->color_flags[GTK_STATE_PRELIGHT] | GTK_RC_BG | GTK_RC_BASE);
3885
3886 style->bg[GTK_STATE_ACTIVE] = *bg;
3887 style->base[GTK_STATE_ACTIVE] = *bg;
3888 style->color_flags[GTK_STATE_ACTIVE] = (GtkRcFlags)
3889 (style->color_flags[GTK_STATE_ACTIVE] | GTK_RC_BG | GTK_RC_BASE);
3890
3891 style->bg[GTK_STATE_INSENSITIVE] = *bg;
3892 style->base[GTK_STATE_INSENSITIVE] = *bg;
3893 style->color_flags[GTK_STATE_INSENSITIVE] = (GtkRcFlags)
3894 (style->color_flags[GTK_STATE_INSENSITIVE] | GTK_RC_BG | GTK_RC_BASE);
3895 }
3896
3897 return style;
3898 }
3899
3900 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle)
3901 {
3902 GtkRcStyle *style = CreateWidgetStyle(forceStyle);
3903 if ( style )
3904 {
3905 DoApplyWidgetStyle(style);
3906 gtk_rc_style_unref(style);
3907 }
3908
3909 // Style change may affect GTK+'s size calculation:
3910 InvalidateBestSize();
3911 }
3912
3913 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle *style)
3914 {
3915 if (m_wxwindow)
3916 gtk_widget_modify_style(m_wxwindow, style);
3917 else
3918 gtk_widget_modify_style(m_widget, style);
3919 }
3920
3921 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style)
3922 {
3923 wxWindowBase::SetBackgroundStyle(style);
3924
3925 if (style == wxBG_STYLE_CUSTOM)
3926 {
3927 GdkWindow *window = (GdkWindow*) NULL;
3928 if (m_wxwindow)
3929 window = GTK_PIZZA(m_wxwindow)->bin_window;
3930 else
3931 window = GetConnectWidget()->window;
3932
3933 if (window)
3934 {
3935 // Make sure GDK/X11 doesn't refresh the window
3936 // automatically.
3937 gdk_window_set_back_pixmap( window, None, False );
3938 #ifdef __X__
3939 Display* display = GDK_WINDOW_DISPLAY(window);
3940 XFlush(display);
3941 #endif
3942 m_needsStyleChange = false;
3943 }
3944 else
3945 // Do in OnIdle, because the window is not yet available
3946 m_needsStyleChange = true;
3947
3948 // Don't apply widget style, or we get a grey background
3949 }
3950 else
3951 {
3952 // apply style change (forceStyle=true so that new style is applied
3953 // even if the bg colour changed from valid to wxNullColour):
3954 ApplyWidgetStyle(true);
3955 }
3956 return true;
3957 }
3958
3959 #if wxUSE_DRAG_AND_DROP
3960
3961 void wxWindowGTK::SetDropTarget( wxDropTarget *dropTarget )
3962 {
3963 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3964
3965 GtkWidget *dnd_widget = GetConnectWidget();
3966
3967 if (m_dropTarget) m_dropTarget->UnregisterWidget( dnd_widget );
3968
3969 if (m_dropTarget) delete m_dropTarget;
3970 m_dropTarget = dropTarget;
3971
3972 if (m_dropTarget) m_dropTarget->RegisterWidget( dnd_widget );
3973 }
3974
3975 #endif // wxUSE_DRAG_AND_DROP
3976
3977 GtkWidget* wxWindowGTK::GetConnectWidget()
3978 {
3979 GtkWidget *connect_widget = m_widget;
3980 if (m_wxwindow) connect_widget = m_wxwindow;
3981
3982 return connect_widget;
3983 }
3984
3985 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow *window) const
3986 {
3987 wxArrayGdkWindows windowsThis;
3988 GdkWindow * const winThis = GTKGetWindow(windowsThis);
3989
3990 return winThis ? window == winThis
3991 : windowsThis.Index(window) != wxNOT_FOUND;
3992 }
3993
3994 GdkWindow *wxWindowGTK::GTKGetWindow(wxArrayGdkWindows& WXUNUSED(windows)) const
3995 {
3996 return m_wxwindow ? GTK_PIZZA(m_wxwindow)->bin_window : m_widget->window;
3997 }
3998
3999 bool wxWindowGTK::SetFont( const wxFont &font )
4000 {
4001 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
4002
4003 if (!wxWindowBase::SetFont(font))
4004 return false;
4005
4006 // apply style change (forceStyle=true so that new style is applied
4007 // even if the font changed from valid to wxNullFont):
4008 ApplyWidgetStyle(true);
4009
4010 return true;
4011 }
4012
4013 void wxWindowGTK::DoCaptureMouse()
4014 {
4015 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4016
4017 GdkWindow *window = (GdkWindow*) NULL;
4018 if (m_wxwindow)
4019 window = GTK_PIZZA(m_wxwindow)->bin_window;
4020 else
4021 window = GetConnectWidget()->window;
4022
4023 wxCHECK_RET( window, _T("CaptureMouse() failed") );
4024
4025 const wxCursor* cursor = &m_cursor;
4026 if (!cursor->Ok())
4027 cursor = wxSTANDARD_CURSOR;
4028
4029 gdk_pointer_grab( window, FALSE,
4030 (GdkEventMask)
4031 (GDK_BUTTON_PRESS_MASK |
4032 GDK_BUTTON_RELEASE_MASK |
4033 GDK_POINTER_MOTION_HINT_MASK |
4034 GDK_POINTER_MOTION_MASK),
4035 (GdkWindow *) NULL,
4036 cursor->GetCursor(),
4037 (guint32)GDK_CURRENT_TIME );
4038 g_captureWindow = this;
4039 g_captureWindowHasMouse = true;
4040 }
4041
4042 void wxWindowGTK::DoReleaseMouse()
4043 {
4044 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4045
4046 wxCHECK_RET( g_captureWindow, wxT("can't release mouse - not captured") );
4047
4048 g_captureWindow = (wxWindowGTK*) NULL;
4049
4050 GdkWindow *window = (GdkWindow*) NULL;
4051 if (m_wxwindow)
4052 window = GTK_PIZZA(m_wxwindow)->bin_window;
4053 else
4054 window = GetConnectWidget()->window;
4055
4056 if (!window)
4057 return;
4058
4059 gdk_pointer_ungrab ( (guint32)GDK_CURRENT_TIME );
4060 }
4061
4062 /* static */
4063 wxWindow *wxWindowBase::GetCapture()
4064 {
4065 return (wxWindow *)g_captureWindow;
4066 }
4067
4068 bool wxWindowGTK::IsRetained() const
4069 {
4070 return false;
4071 }
4072
4073 void wxWindowGTK::BlockScrollEvent()
4074 {
4075 wxASSERT(!m_blockScrollEvent);
4076 m_blockScrollEvent = true;
4077 }
4078
4079 void wxWindowGTK::UnblockScrollEvent()
4080 {
4081 wxASSERT(m_blockScrollEvent);
4082 m_blockScrollEvent = false;
4083 }
4084
4085 void wxWindowGTK::SetScrollbar(int orient,
4086 int pos,
4087 int thumbVisible,
4088 int range,
4089 bool WXUNUSED(update))
4090 {
4091 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4092 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4093
4094 if (range > 0)
4095 {
4096 m_hasScrolling = true;
4097 }
4098 else
4099 {
4100 // GtkRange requires upper > lower
4101 range =
4102 thumbVisible = 1;
4103 }
4104
4105 if (pos > range - thumbVisible)
4106 pos = range - thumbVisible;
4107 if (pos < 0)
4108 pos = 0;
4109 GtkAdjustment* adj = m_scrollBar[ScrollDirFromOrient(orient)]->adjustment;
4110 adj->step_increment = 1;
4111 adj->page_increment =
4112 adj->page_size = thumbVisible;
4113 adj->upper = range;
4114 SetScrollPos(orient, pos);
4115 gtk_adjustment_changed(adj);
4116 }
4117
4118 void wxWindowGTK::SetScrollPos(int orient, int pos, bool WXUNUSED(refresh))
4119 {
4120 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4121 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4122
4123 // This check is more than an optimization. Without it, the slider
4124 // will not move smoothly while tracking when using wxScrollHelper.
4125 if (GetScrollPos(orient) != pos)
4126 {
4127 const int dir = ScrollDirFromOrient(orient);
4128 GtkAdjustment* adj = m_scrollBar[dir]->adjustment;
4129 const int max = int(adj->upper - adj->page_size);
4130 if (pos > max)
4131 pos = max;
4132 if (pos < 0)
4133 pos = 0;
4134 m_scrollPos[dir] =
4135 adj->value = pos;
4136 // If a "value_changed" signal emission is not already in progress
4137 if (!m_blockValueChanged[dir])
4138 {
4139 gtk_adjustment_value_changed(adj);
4140 }
4141 }
4142 }
4143
4144 int wxWindowGTK::GetScrollThumb(int orient) const
4145 {
4146 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4147 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4148
4149 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->page_size);
4150 }
4151
4152 int wxWindowGTK::GetScrollPos( int orient ) const
4153 {
4154 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4155 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4156
4157 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->value + 0.5);
4158 }
4159
4160 int wxWindowGTK::GetScrollRange( int orient ) const
4161 {
4162 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
4163 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
4164
4165 return int(m_scrollBar[ScrollDirFromOrient(orient)]->adjustment->upper);
4166 }
4167
4168 // Determine if increment is the same as +/-x, allowing for some small
4169 // difference due to possible inexactness in floating point arithmetic
4170 static inline bool IsScrollIncrement(double increment, double x)
4171 {
4172 wxASSERT(increment > 0);
4173 const double tolerance = 1.0 / 1024;
4174 return fabs(increment - fabs(x)) < tolerance;
4175 }
4176
4177 wxEventType wxWindowGTK::GetScrollEventType(GtkRange* range)
4178 {
4179 DEBUG_MAIN_THREAD
4180
4181 if (g_isIdle)
4182 wxapp_install_idle_handler();
4183
4184 wxASSERT(range == m_scrollBar[0] || range == m_scrollBar[1]);
4185
4186 const int barIndex = range == m_scrollBar[1];
4187 GtkAdjustment* adj = range->adjustment;
4188 const int value = int(adj->value + 0.5);
4189 // save previous position
4190 const double oldPos = m_scrollPos[barIndex];
4191 // update current position
4192 m_scrollPos[barIndex] = adj->value;
4193 // If event should be ignored, or integral position has not changed
4194 if (!m_hasVMT || g_blockEventsOnDrag || value == int(oldPos + 0.5))
4195 {
4196 return wxEVT_NULL;
4197 }
4198
4199 wxEventType eventType = wxEVT_SCROLL_THUMBTRACK;
4200 if (!m_isScrolling)
4201 {
4202 // Difference from last change event
4203 const double diff = adj->value - oldPos;
4204 const bool isDown = diff > 0;
4205
4206 if (IsScrollIncrement(adj->step_increment, diff))
4207 {
4208 eventType = isDown ? wxEVT_SCROLL_LINEDOWN : wxEVT_SCROLL_LINEUP;
4209 }
4210 else if (IsScrollIncrement(adj->page_increment, diff))
4211 {
4212 eventType = isDown ? wxEVT_SCROLL_PAGEDOWN : wxEVT_SCROLL_PAGEUP;
4213 }
4214 else if (m_mouseButtonDown)
4215 {
4216 // Assume track event
4217 m_isScrolling = true;
4218 }
4219 }
4220 return eventType;
4221 }
4222
4223 void wxWindowGTK::ScrollWindow( int dx, int dy, const wxRect* WXUNUSED(rect) )
4224 {
4225 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4226
4227 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
4228
4229 // No scrolling requested.
4230 if ((dx == 0) && (dy == 0)) return;
4231
4232 m_clipPaintRegion = true;
4233
4234 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow), -dx, -dy );
4235
4236 m_clipPaintRegion = false;
4237 }
4238
4239 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget* w, int wxstyle)
4240 {
4241 //RN: Note that static controls usually have no border on gtk, so maybe
4242 //it makes sense to treat that as simply no border at the wx level
4243 //as well...
4244 if (!(wxstyle & wxNO_BORDER) && !(wxstyle & wxBORDER_STATIC))
4245 {
4246 GtkShadowType gtkstyle;
4247
4248 if(wxstyle & wxBORDER_RAISED)
4249 gtkstyle = GTK_SHADOW_OUT;
4250 else if (wxstyle & wxBORDER_SUNKEN)
4251 gtkstyle = GTK_SHADOW_IN;
4252 else if (wxstyle & wxBORDER_DOUBLE)
4253 gtkstyle = GTK_SHADOW_ETCHED_IN;
4254 else //default
4255 gtkstyle = GTK_SHADOW_IN;
4256
4257 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w),
4258 gtkstyle );
4259 }
4260 }
4261
4262 void wxWindowGTK::SetWindowStyleFlag( long style )
4263 {
4264 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4265 wxWindowBase::SetWindowStyleFlag(style);
4266 }
4267
4268 // Find the wxWindow at the current mouse position, also returning the mouse
4269 // position.
4270 wxWindow* wxFindWindowAtPointer(wxPoint& pt)
4271 {
4272 pt = wxGetMousePosition();
4273 wxWindow* found = wxFindWindowAtPoint(pt);
4274 return found;
4275 }
4276
4277 // Get the current mouse position.
4278 wxPoint wxGetMousePosition()
4279 {
4280 /* This crashes when used within wxHelpContext,
4281 so we have to use the X-specific implementation below.
4282 gint x, y;
4283 GdkModifierType *mask;
4284 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4285
4286 return wxPoint(x, y);
4287 */
4288
4289 int x, y;
4290 GdkWindow* windowAtPtr = gdk_window_at_pointer(& x, & y);
4291
4292 Display *display = windowAtPtr ? GDK_WINDOW_XDISPLAY(windowAtPtr) : GDK_DISPLAY();
4293 Window rootWindow = RootWindowOfScreen (DefaultScreenOfDisplay(display));
4294 Window rootReturn, childReturn;
4295 int rootX, rootY, winX, winY;
4296 unsigned int maskReturn;
4297
4298 XQueryPointer (display,
4299 rootWindow,
4300 &rootReturn,
4301 &childReturn,
4302 &rootX, &rootY, &winX, &winY, &maskReturn);
4303 return wxPoint(rootX, rootY);
4304
4305 }
4306
4307 // Needed for implementing e.g. combobox on wxGTK within a modal dialog.
4308 void wxAddGrab(wxWindow* window)
4309 {
4310 gtk_grab_add( (GtkWidget*) window->GetHandle() );
4311 }
4312
4313 void wxRemoveGrab(wxWindow* window)
4314 {
4315 gtk_grab_remove( (GtkWidget*) window->GetHandle() );
4316 }
4317
4318 // ----------------------------------------------------------------------------
4319 // wxWinModule
4320 // ----------------------------------------------------------------------------
4321
4322 class wxWinModule : public wxModule
4323 {
4324 public:
4325 bool OnInit();
4326 void OnExit();
4327
4328 private:
4329 DECLARE_DYNAMIC_CLASS(wxWinModule)
4330 };
4331
4332 IMPLEMENT_DYNAMIC_CLASS(wxWinModule, wxModule)
4333
4334 bool wxWinModule::OnInit()
4335 {
4336 // g_eraseGC = gdk_gc_new( gdk_get_default_root_window() );
4337 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
4338
4339 return true;
4340 }
4341
4342 void wxWinModule::OnExit()
4343 {
4344 if (g_eraseGC)
4345 g_object_unref (g_eraseGC);
4346 }