1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/window.cpp 
   3 // Purpose:     wxWindowGTK implementation 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling, Julian Smart 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  14 #define XWarpPointer XWARPPOINTER 
  17 #include "wx/window.h" 
  22     #include "wx/toplevel.h" 
  23     #include "wx/dcclient.h" 
  25     #include "wx/settings.h" 
  26     #include "wx/msgdlg.h" 
  31 #include "wx/tooltip.h" 
  33 #include "wx/fontutil.h" 
  34 #include "wx/sysopt.h" 
  38 #include "wx/gtk/private.h" 
  39 #include "wx/gtk/private/win_gtk.h" 
  40 #include <gdk/gdkkeysyms.h> 
  43 #if !GTK_CHECK_VERSION(2,10,0) 
  44     // GTK+ can reliably detect Meta key state only since 2.10 when 
  45     // GDK_META_MASK was introduced -- there wasn't any way to detect it 
  46     // in older versions. wxGTK used GDK_MOD2_MASK for this purpose, but 
  47     // GDK_MOD2_MASK is documented as: 
  49     //     the fifth modifier key (it depends on the modifier mapping of the X 
  50     //     server which key is interpreted as this modifier) 
  52     // In other words, it isn't guaranteed to map to Meta. This is a real 
  53     // problem: it is common to map NumLock to it (in fact, it's an exception 
  54     // if the X server _doesn't_ use it for NumLock).  So the old code caused 
  55     // wxKeyEvent::MetaDown() to always return true as long as NumLock was on 
  56     // on many systems, which broke all applications using 
  57     // wxKeyEvent::GetModifiers() to check modifiers state (see e.g.  here: 
  58     // http://tinyurl.com/56lsk2). 
  60     // Because of this, it's better to not detect Meta key state at all than 
  61     // to detect it incorrectly. Hence the following #define, which causes 
  62     // m_metaDown to be always set to false. 
  63     #define GDK_META_MASK 0 
  66 //----------------------------------------------------------------------------- 
  67 // documentation on internals 
  68 //----------------------------------------------------------------------------- 
  71    I have been asked several times about writing some documentation about 
  72    the GTK port of wxWidgets, especially its internal structures. Obviously, 
  73    you cannot understand wxGTK without knowing a little about the GTK, but 
  74    some more information about what the wxWindow, which is the base class 
  75    for all other window classes, does seems required as well. 
  79    What does wxWindow do? It contains the common interface for the following 
  80    jobs of its descendants: 
  82    1) Define the rudimentary behaviour common to all window classes, such as 
  83    resizing, intercepting user input (so as to make it possible to use these 
  84    events for special purposes in a derived class), window names etc. 
  86    2) Provide the possibility to contain and manage children, if the derived 
  87    class is allowed to contain children, which holds true for those window 
  88    classes which do not display a native GTK widget. To name them, these 
  89    classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame- 
  90    work classes are a special case and are handled a bit differently from 
  91    the rest. The same holds true for the wxNotebook class. 
  93    3) Provide the possibility to draw into a client area of a window. This, 
  94    too, only holds true for classes that do not display a native GTK widget 
  97    4) Provide the entire mechanism for scrolling widgets. This actual inter- 
  98    face for this is usually in wxScrolledWindow, but the GTK implementation 
 101    5) A multitude of helper or extra methods for special purposes, such as 
 102    Drag'n'Drop, managing validators etc. 
 104    6) Display a border (sunken, raised, simple or none). 
 106    Normally one might expect, that one wxWidgets window would always correspond 
 107    to one GTK widget. Under GTK, there is no such all-round widget that has all 
 108    the functionality. Moreover, the GTK defines a client area as a different 
 109    widget from the actual widget you are handling. Last but not least some 
 110    special classes (e.g. wxFrame) handle different categories of widgets and 
 111    still have the possibility to draw something in the client area. 
 112    It was therefore required to write a special purpose GTK widget, that would 
 113    represent a client area in the sense of wxWidgets capable to do the jobs 
 114    2), 3) and 4). I have written this class and it resides in win_gtk.c of 
 117    All windows must have a widget, with which they interact with other under- 
 118    lying GTK widgets. It is this widget, e.g. that has to be resized etc and 
 119    the wxWindow class has a member variable called m_widget which holds a 
 120    pointer to this widget. When the window class represents a GTK native widget, 
 121    this is (in most cases) the only GTK widget the class manages. E.g. the 
 122    wxStaticText class handles only a GtkLabel widget a pointer to which you 
 123    can find in m_widget (defined in wxWindow) 
 125    When the class has a client area for drawing into and for containing children 
 126    it has to handle the client area widget (of the type wxPizza, defined in 
 127    win_gtk.cpp), but there could be any number of widgets, handled by a class. 
 128    The common rule for all windows is only, that the widget that interacts with 
 129    the rest of GTK must be referenced in m_widget and all other widgets must be 
 130    children of this widget on the GTK level. The top-most widget, which also 
 131    represents the client area, must be in the m_wxwindow field and must be of 
 134    As I said, the window classes that display a GTK native widget only have 
 135    one widget, so in the case of e.g. the wxButton class m_widget holds a 
 136    pointer to a GtkButton widget. But windows with client areas (for drawing 
 137    and children) have a m_widget field that is a pointer to a GtkScrolled- 
 138    Window and a m_wxwindow field that is pointer to a wxPizza and this 
 139    one is (in the GTK sense) a child of the GtkScrolledWindow. 
 141    If the m_wxwindow field is set, then all input to this widget is inter- 
 142    cepted and sent to the wxWidgets class. If not, all input to the widget 
 143    that gets pointed to by m_widget gets intercepted and sent to the class. 
 147    The design of scrolling in wxWidgets is markedly different from that offered 
 148    by the GTK itself and therefore we cannot simply take it as it is. In GTK, 
 149    clicking on a scrollbar belonging to scrolled window will inevitably move 
 150    the window. In wxWidgets, the scrollbar will only emit an event, send this 
 151    to (normally) a wxScrolledWindow and that class will call ScrollWindow() 
 152    which actually moves the window and its sub-windows. Note that wxPizza 
 153    memorizes how much it has been scrolled but that wxWidgets forgets this 
 154    so that the two coordinates systems have to be kept in synch. This is done 
 155    in various places using the pizza->m_scroll_x and pizza->m_scroll_y values. 
 159    Singularly the most broken code in GTK is the code that is supposed to 
 160    inform subwindows (child windows) about new positions. Very often, duplicate 
 161    events are sent without changes in size or position, equally often no 
 162    events are sent at all (All this is due to a bug in the GtkContainer code 
 163    which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores 
 164    GTK's own system and it simply waits for size events for toplevel windows 
 165    and then iterates down the respective size events to all window. This has 
 166    the disadvantage that windows might get size events before the GTK widget 
 167    actually has the reported size. This doesn't normally pose any problem, but 
 168    the OpenGL drawing routines rely on correct behaviour. Therefore, I have 
 169    added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas, 
 170    i.e. the wxGLCanvas will emit a size event, when (and not before) the X11 
 171    window that is used for OpenGL output really has that size (as reported by 
 176    If someone at some point of time feels the immense desire to have a look at, 
 177    change or attempt to optimise the Refresh() logic, this person will need an 
 178    intimate understanding of what "draw" and "expose" events are and what 
 179    they are used for, in particular when used in connection with GTK's 
 180    own windowless widgets. Beware. 
 184    Cursors, too, have been a constant source of pleasure. The main difficulty 
 185    is that a GdkWindow inherits a cursor if the programmer sets a new cursor 
 186    for the parent. To prevent this from doing too much harm, I use idle time 
 187    to set the cursor over and over again, starting from the toplevel windows 
 188    and ending with the youngest generation (speaking of parent and child windows). 
 189    Also don't forget that cursors (like much else) are connected to GdkWindows, 
 190    not GtkWidgets and that the "window" field of a GtkWidget might very well 
 191    point to the GdkWindow of the parent widget (-> "window-less widget") and 
 192    that the two obviously have very different meanings. 
 196 //----------------------------------------------------------------------------- 
 198 //----------------------------------------------------------------------------- 
 200 // Don't allow event propagation during drag 
 201 bool g_blockEventsOnDrag
; 
 202 // Don't allow mouse event propagation during scroll 
 203 bool g_blockEventsOnScroll
; 
 204 extern wxCursor   g_globalCursor
; 
 206 // mouse capture state: the window which has it and if the mouse is currently 
 208 static wxWindowGTK  
*g_captureWindow 
= NULL
; 
 209 static bool g_captureWindowHasMouse 
= false; 
 211 // The window that currently has focus: 
 212 static wxWindowGTK 
*gs_currentFocus 
= NULL
; 
 213 // The window that is scheduled to get focus in the next event loop iteration 
 214 // or NULL if there's no pending focus change: 
 215 static wxWindowGTK 
*gs_pendingFocus 
= NULL
; 
 217 // the window that has deferred focus-out event pending, if any (see 
 218 // GTKAddDeferredFocusOut() for details) 
 219 static wxWindowGTK 
*gs_deferredFocusOut 
= NULL
; 
 221 // global variables because GTK+ DnD want to have the 
 222 // mouse event that caused it 
 223 GdkEvent    
*g_lastMouseEvent 
= NULL
; 
 224 int          g_lastButtonNumber 
= 0; 
 226 //----------------------------------------------------------------------------- 
 228 //----------------------------------------------------------------------------- 
 230 // the trace mask used for the focus debugging messages 
 231 #define TRACE_FOCUS wxT("focus") 
 233 //----------------------------------------------------------------------------- 
 234 // missing gdk functions 
 235 //----------------------------------------------------------------------------- 
 238 gdk_window_warp_pointer (GdkWindow      
*window
, 
 243     window 
= gdk_get_default_root_window(); 
 245   if (!GDK_WINDOW_DESTROYED(window
)) 
 247       XWarpPointer (GDK_WINDOW_XDISPLAY(window
), 
 248                     None
,              /* not source window -> move from anywhere */ 
 249                     GDK_WINDOW_XID(window
),  /* dest window */ 
 250                     0, 0, 0, 0,        /* not source window -> move from anywhere */ 
 256 //----------------------------------------------------------------------------- 
 257 // "size_request" of m_widget 
 258 //----------------------------------------------------------------------------- 
 262 wxgtk_window_size_request_callback(GtkWidget 
* WXUNUSED(widget
), 
 263                                    GtkRequisition 
*requisition
, 
 267     win
->GetSize( &w
, &h 
); 
 273     requisition
->height 
= h
; 
 274     requisition
->width 
= w
; 
 278 //----------------------------------------------------------------------------- 
 279 // "expose_event" of m_wxwindow 
 280 //----------------------------------------------------------------------------- 
 284 gtk_window_expose_callback( GtkWidget
* widget
, 
 285                             GdkEventExpose 
*gdk_event
, 
 288     if (gdk_event
->window 
== widget
->window
) 
 290         win
->GetUpdateRegion() = wxRegion( gdk_event
->region 
); 
 291         win
->GtkSendPaintEvents(); 
 293     // Let parent window draw window-less widgets 
 298 #ifndef __WXUNIVERSAL__ 
 299 //----------------------------------------------------------------------------- 
 300 // "expose_event" from m_wxwindow->parent, for drawing border 
 301 //----------------------------------------------------------------------------- 
 305 expose_event_border(GtkWidget
* widget
, GdkEventExpose
* gdk_event
, wxWindow
* win
) 
 307     if (gdk_event
->window 
!= widget
->window
) 
 310     const GtkAllocation
& alloc 
= win
->m_wxwindow
->allocation
; 
 311     const int x 
= alloc
.x
; 
 312     const int y 
= alloc
.y
; 
 313     const int w 
= alloc
.width
; 
 314     const int h 
= alloc
.height
; 
 316     if (w 
<= 0 || h 
<= 0) 
 319     if (win
->HasFlag(wxBORDER_SIMPLE
)) 
 321         gdk_draw_rectangle(gdk_event
->window
, 
 322             widget
->style
->black_gc
, false, x
, y
, w 
- 1, h 
- 1); 
 326         GtkShadowType shadow 
= GTK_SHADOW_IN
; 
 327         if (win
->HasFlag(wxBORDER_RAISED
)) 
 328             shadow 
= GTK_SHADOW_OUT
; 
 330         // Style detail to use 
 332         if (win
->m_widget 
== win
->m_wxwindow
) 
 333             // for non-scrollable wxWindows 
 336             // for scrollable ones 
 340            win
->m_wxwindow
->style
, gdk_event
->window
, GTK_STATE_NORMAL
, 
 341            shadow
, NULL
, wxGTKPrivate::GetEntryWidget(), detail
, x
, y
, w
, h
); 
 347 //----------------------------------------------------------------------------- 
 348 // "parent_set" from m_wxwindow 
 349 //----------------------------------------------------------------------------- 
 353 parent_set(GtkWidget
* widget
, GtkObject
* old_parent
, wxWindow
* win
) 
 357         g_signal_handlers_disconnect_by_func( 
 358             old_parent
, (void*)expose_event_border
, win
); 
 362         g_signal_connect_after(widget
->parent
, "expose_event", 
 363             G_CALLBACK(expose_event_border
), win
); 
 367 #endif // !__WXUNIVERSAL__ 
 369 //----------------------------------------------------------------------------- 
 370 // "key_press_event" from any window 
 371 //----------------------------------------------------------------------------- 
 373 // These are used when transforming Ctrl-alpha to ascii values 1-26 
 374 inline bool wxIsLowerChar(int code
) 
 376     return (code 
>= 'a' && code 
<= 'z' ); 
 379 inline bool wxIsUpperChar(int code
) 
 381     return (code 
>= 'A' && code 
<= 'Z' ); 
 385 // set WXTRACE to this to see the key event codes on the console 
 386 #define TRACE_KEYS  wxT("keyevent") 
 388 // translates an X key symbol to WXK_XXX value 
 390 // if isChar is true it means that the value returned will be used for EVT_CHAR 
 391 // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide, 
 392 // for example, while if it is false it means that the value is going to be 
 393 // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to 
 395 static long wxTranslateKeySymToWXKey(KeySym keysym
, bool isChar
) 
 401         // Shift, Control and Alt don't generate the CHAR events at all 
 404             key_code 
= isChar 
? 0 : WXK_SHIFT
; 
 408             key_code 
= isChar 
? 0 : WXK_CONTROL
; 
 416             key_code 
= isChar 
? 0 : WXK_ALT
; 
 419         // neither do the toggle modifies 
 420         case GDK_Scroll_Lock
: 
 421             key_code 
= isChar 
? 0 : WXK_SCROLL
; 
 425             key_code 
= isChar 
? 0 : WXK_CAPITAL
; 
 429             key_code 
= isChar 
? 0 : WXK_NUMLOCK
; 
 433         // various other special keys 
 446         case GDK_ISO_Left_Tab
: 
 453             key_code 
= WXK_RETURN
; 
 457             key_code 
= WXK_CLEAR
; 
 461             key_code 
= WXK_PAUSE
; 
 465             key_code 
= WXK_SELECT
; 
 469             key_code 
= WXK_PRINT
; 
 473             key_code 
= WXK_EXECUTE
; 
 477             key_code 
= WXK_ESCAPE
; 
 480         // cursor and other extended keyboard keys 
 482             key_code 
= WXK_DELETE
; 
 498             key_code 
= WXK_RIGHT
; 
 505         case GDK_Prior
:     // == GDK_Page_Up 
 506             key_code 
= WXK_PAGEUP
; 
 509         case GDK_Next
:      // == GDK_Page_Down 
 510             key_code 
= WXK_PAGEDOWN
; 
 522             key_code 
= WXK_INSERT
; 
 537             key_code 
= (isChar 
? '0' : int(WXK_NUMPAD0
)) + keysym 
- GDK_KP_0
; 
 541             key_code 
= isChar 
? ' ' : int(WXK_NUMPAD_SPACE
); 
 545             key_code 
= isChar 
? WXK_TAB 
: WXK_NUMPAD_TAB
; 
 549             key_code 
= isChar 
? WXK_RETURN 
: WXK_NUMPAD_ENTER
; 
 553             key_code 
= isChar 
? WXK_F1 
: WXK_NUMPAD_F1
; 
 557             key_code 
= isChar 
? WXK_F2 
: WXK_NUMPAD_F2
; 
 561             key_code 
= isChar 
? WXK_F3 
: WXK_NUMPAD_F3
; 
 565             key_code 
= isChar 
? WXK_F4 
: WXK_NUMPAD_F4
; 
 569             key_code 
= isChar 
? WXK_HOME 
: WXK_NUMPAD_HOME
; 
 573             key_code 
= isChar 
? WXK_LEFT 
: WXK_NUMPAD_LEFT
; 
 577             key_code 
= isChar 
? WXK_UP 
: WXK_NUMPAD_UP
; 
 581             key_code 
= isChar 
? WXK_RIGHT 
: WXK_NUMPAD_RIGHT
; 
 585             key_code 
= isChar 
? WXK_DOWN 
: WXK_NUMPAD_DOWN
; 
 588         case GDK_KP_Prior
: // == GDK_KP_Page_Up 
 589             key_code 
= isChar 
? WXK_PAGEUP 
: WXK_NUMPAD_PAGEUP
; 
 592         case GDK_KP_Next
: // == GDK_KP_Page_Down 
 593             key_code 
= isChar 
? WXK_PAGEDOWN 
: WXK_NUMPAD_PAGEDOWN
; 
 597             key_code 
= isChar 
? WXK_END 
: WXK_NUMPAD_END
; 
 601             key_code 
= isChar 
? WXK_HOME 
: WXK_NUMPAD_BEGIN
; 
 605             key_code 
= isChar 
? WXK_INSERT 
: WXK_NUMPAD_INSERT
; 
 609             key_code 
= isChar 
? WXK_DELETE 
: WXK_NUMPAD_DELETE
; 
 613             key_code 
= isChar 
? '=' : int(WXK_NUMPAD_EQUAL
); 
 616         case GDK_KP_Multiply
: 
 617             key_code 
= isChar 
? '*' : int(WXK_NUMPAD_MULTIPLY
); 
 621             key_code 
= isChar 
? '+' : int(WXK_NUMPAD_ADD
); 
 624         case GDK_KP_Separator
: 
 625             // FIXME: what is this? 
 626             key_code 
= isChar 
? '.' : int(WXK_NUMPAD_SEPARATOR
); 
 629         case GDK_KP_Subtract
: 
 630             key_code 
= isChar 
? '-' : int(WXK_NUMPAD_SUBTRACT
); 
 634             key_code 
= isChar 
? '.' : int(WXK_NUMPAD_DECIMAL
); 
 638             key_code 
= isChar 
? '/' : int(WXK_NUMPAD_DIVIDE
); 
 655             key_code 
= WXK_F1 
+ keysym 
- GDK_F1
; 
 665 static inline bool wxIsAsciiKeysym(KeySym ks
) 
 670 static void wxFillOtherKeyEventFields(wxKeyEvent
& event
, 
 672                                       GdkEventKey 
*gdk_event
) 
 676     GdkModifierType state
; 
 677     if (gdk_event
->window
) 
 678         gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
); 
 680     event
.SetTimestamp( gdk_event
->time 
); 
 681     event
.SetId(win
->GetId()); 
 682     event
.m_shiftDown 
= (gdk_event
->state 
& GDK_SHIFT_MASK
) != 0; 
 683     event
.m_controlDown 
= (gdk_event
->state 
& GDK_CONTROL_MASK
) != 0; 
 684     event
.m_altDown 
= (gdk_event
->state 
& GDK_MOD1_MASK
) != 0; 
 685     event
.m_metaDown 
= (gdk_event
->state 
& GDK_META_MASK
) != 0; 
 686     event
.m_rawCode 
= (wxUint32
) gdk_event
->keyval
; 
 687     event
.m_rawFlags 
= 0; 
 689     event
.m_uniChar 
= gdk_keyval_to_unicode(gdk_event
->keyval
); 
 691     wxGetMousePosition( &x
, &y 
); 
 692     win
->ScreenToClient( &x
, &y 
); 
 695     event
.SetEventObject( win 
); 
 700 wxTranslateGTKKeyEventToWx(wxKeyEvent
& event
, 
 702                            GdkEventKey 
*gdk_event
) 
 704     // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string 
 705     //     but only event->keyval which is quite useless to us, so remember 
 706     //     the last character from GDK_KEY_PRESS and reuse it as last resort 
 708     // NB: should be MT-safe as we're always called from the main thread only 
 713     } s_lastKeyPress 
= { 0, 0 }; 
 715     KeySym keysym 
= gdk_event
->keyval
; 
 717     wxLogTrace(TRACE_KEYS
, wxT("Key %s event: keysym = %ld"), 
 718                event
.GetEventType() == wxEVT_KEY_UP 
? wxT("release") 
 722     long key_code 
= wxTranslateKeySymToWXKey(keysym
, false /* !isChar */); 
 726         // do we have the translation or is it a plain ASCII character? 
 727         if ( (gdk_event
->length 
== 1) || wxIsAsciiKeysym(keysym
) ) 
 729             // we should use keysym if it is ASCII as X does some translations 
 730             // like "I pressed while Control is down" => "Ctrl-I" == "TAB" 
 731             // which we don't want here (but which we do use for OnChar()) 
 732             if ( !wxIsAsciiKeysym(keysym
) ) 
 734                 keysym 
= (KeySym
)gdk_event
->string
[0]; 
 737             // we want to always get the same key code when the same key is 
 738             // pressed regardless of the state of the modifiers, i.e. on a 
 739             // standard US keyboard pressing '5' or '%' ('5' key with 
 740             // Shift) should result in the same key code in OnKeyDown(): 
 741             // '5' (although OnChar() will get either '5' or '%'). 
 743             // to do it we first translate keysym to keycode (== scan code) 
 744             // and then back but always using the lower register 
 745             Display 
*dpy 
= (Display 
*)wxGetDisplay(); 
 746             KeyCode keycode 
= XKeysymToKeycode(dpy
, keysym
); 
 748             wxLogTrace(TRACE_KEYS
, wxT("\t-> keycode %d"), keycode
); 
 750             KeySym keysymNormalized 
= XKeycodeToKeysym(dpy
, keycode
, 0); 
 752             // use the normalized, i.e. lower register, keysym if we've 
 754             key_code 
= keysymNormalized 
? keysymNormalized 
: keysym
; 
 756             // as explained above, we want to have lower register key codes 
 757             // normally but for the letter keys we want to have the upper ones 
 759             // NB: don't use XConvertCase() here, we want to do it for letters 
 761             key_code 
= toupper(key_code
); 
 763         else // non ASCII key, what to do? 
 765             // by default, ignore it 
 768             // but if we have cached information from the last KEY_PRESS 
 769             if ( gdk_event
->type 
== GDK_KEY_RELEASE 
) 
 772                 if ( keysym 
== s_lastKeyPress
.keysym 
) 
 774                     key_code 
= s_lastKeyPress
.keycode
; 
 779         if ( gdk_event
->type 
== GDK_KEY_PRESS 
) 
 781             // remember it to be reused for KEY_UP event later 
 782             s_lastKeyPress
.keysym 
= keysym
; 
 783             s_lastKeyPress
.keycode 
= key_code
; 
 787     wxLogTrace(TRACE_KEYS
, wxT("\t-> wxKeyCode %ld"), key_code
); 
 789     // sending unknown key events doesn't really make sense 
 793     // now fill all the other fields 
 794     wxFillOtherKeyEventFields(event
, win
, gdk_event
); 
 796     event
.m_keyCode 
= key_code
; 
 798     if ( gdk_event
->type 
== GDK_KEY_PRESS 
||  gdk_event
->type 
== GDK_KEY_RELEASE 
) 
 800         event
.m_uniChar 
= key_code
; 
 810     GtkIMContext 
*context
; 
 811     GdkEventKey  
*lastKeyEvent
; 
 815         context 
= gtk_im_multicontext_new(); 
 820         g_object_unref (context
); 
 826 gtk_window_key_press_callback( GtkWidget 
*WXUNUSED(widget
), 
 827                                GdkEventKey 
*gdk_event
, 
 832     if (g_blockEventsOnDrag
) 
 835     wxKeyEvent 
event( wxEVT_KEY_DOWN 
); 
 837     bool return_after_IM 
= false; 
 839     if( wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) ) 
 841         // Emit KEY_DOWN event 
 842         ret 
= win
->HandleWindowEvent( event 
); 
 846         // Return after IM processing as we cannot do 
 847         // anything with it anyhow. 
 848         return_after_IM 
= true; 
 851     if ((!ret
) && (win
->m_imData 
!= NULL
)) 
 853         // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API 
 854         // docs, if IM filter returns true, no further processing should be done. 
 855         // we should send the key_down event anyway. 
 856         bool intercepted_by_IM 
= gtk_im_context_filter_keypress(win
->m_imData
->context
, gdk_event
); 
 857         win
->m_imData
->lastKeyEvent 
= NULL
; 
 858         if (intercepted_by_IM
) 
 860             wxLogTrace(TRACE_KEYS
, wxT("Key event intercepted by IM")); 
 871         wxWindowGTK 
*ancestor 
= win
; 
 874             int command 
= ancestor
->GetAcceleratorTable()->GetCommand( event 
); 
 877                 wxCommandEvent 
menu_event( wxEVT_COMMAND_MENU_SELECTED
, command 
); 
 878                 ret 
= ancestor
->HandleWindowEvent( menu_event 
); 
 882                     // if the accelerator wasn't handled as menu event, try 
 883                     // it as button click (for compatibility with other 
 885                     wxCommandEvent 
button_event( wxEVT_COMMAND_BUTTON_CLICKED
, command 
); 
 886                     ret 
= ancestor
->HandleWindowEvent( button_event 
); 
 891             if (ancestor
->IsTopLevel()) 
 893             ancestor 
= ancestor
->GetParent(); 
 896 #endif // wxUSE_ACCEL 
 898     // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x 
 899     // will only be sent if it is not in an accelerator table. 
 903         KeySym keysym 
= gdk_event
->keyval
; 
 904         // Find key code for EVT_CHAR and EVT_CHAR_HOOK events 
 905         key_code 
= wxTranslateKeySymToWXKey(keysym
, true /* isChar */); 
 908             if ( wxIsAsciiKeysym(keysym
) ) 
 911                 key_code 
= (unsigned char)keysym
; 
 913             // gdk_event->string is actually deprecated 
 914             else if ( gdk_event
->length 
== 1 ) 
 916                 key_code 
= (unsigned char)gdk_event
->string
[0]; 
 922             wxLogTrace(TRACE_KEYS
, wxT("Char event: %ld"), key_code
); 
 924             event
.m_keyCode 
= key_code
; 
 926             // To conform to the docs we need to translate Ctrl-alpha 
 927             // characters to values in the range 1-26. 
 928             if ( event
.ControlDown() && 
 929                  ( wxIsLowerChar(key_code
) || wxIsUpperChar(key_code
) )) 
 931                 if ( wxIsLowerChar(key_code
) ) 
 932                     event
.m_keyCode 
= key_code 
- 'a' + 1; 
 933                 if ( wxIsUpperChar(key_code
) ) 
 934                     event
.m_keyCode 
= key_code 
- 'A' + 1; 
 936                 event
.m_uniChar 
= event
.m_keyCode
; 
 940             // Implement OnCharHook by checking ancestor top level windows 
 941             wxWindow 
*parent 
= win
; 
 942             while (parent 
&& !parent
->IsTopLevel()) 
 943                 parent 
= parent
->GetParent(); 
 946                 event
.SetEventType( wxEVT_CHAR_HOOK 
); 
 947                 ret 
= parent
->HandleWindowEvent( event 
); 
 952                 event
.SetEventType(wxEVT_CHAR
); 
 953                 ret 
= win
->HandleWindowEvent( event 
); 
 964 gtk_wxwindow_commit_cb (GtkIMContext 
* WXUNUSED(context
), 
 968     wxKeyEvent 
event( wxEVT_KEY_DOWN 
); 
 970     // take modifiers, cursor position, timestamp etc. from the last 
 971     // key_press_event that was fed into Input Method: 
 972     if (window
->m_imData
->lastKeyEvent
) 
 974         wxFillOtherKeyEventFields(event
, 
 975                                   window
, window
->m_imData
->lastKeyEvent
); 
 979         event
.SetEventObject( window 
); 
 982     const wxString 
data(wxGTK_CONV_BACK_SYS(str
)); 
 988     // Implement OnCharHook by checking ancestor top level windows 
 989     wxWindow 
*parent 
= window
; 
 990     while (parent 
&& !parent
->IsTopLevel()) 
 991         parent 
= parent
->GetParent(); 
 993     for( wxString::const_iterator pstr 
= data
.begin(); pstr 
!= data
.end(); ++pstr 
) 
 996         event
.m_uniChar 
= *pstr
; 
 997         // Backward compatible for ISO-8859-1 
 998         event
.m_keyCode 
= *pstr 
< 256 ? event
.m_uniChar 
: 0; 
 999         wxLogTrace(TRACE_KEYS
, wxT("IM sent character '%c'"), event
.m_uniChar
); 
1001         event
.m_keyCode 
= (char)*pstr
; 
1002 #endif  // wxUSE_UNICODE 
1004         // To conform to the docs we need to translate Ctrl-alpha 
1005         // characters to values in the range 1-26. 
1006         if ( event
.ControlDown() && 
1007              ( wxIsLowerChar(*pstr
) || wxIsUpperChar(*pstr
) )) 
1009             if ( wxIsLowerChar(*pstr
) ) 
1010                 event
.m_keyCode 
= *pstr 
- 'a' + 1; 
1011             if ( wxIsUpperChar(*pstr
) ) 
1012                 event
.m_keyCode 
= *pstr 
- 'A' + 1; 
1014             event
.m_keyCode 
= *pstr 
- 'a' + 1; 
1016             event
.m_uniChar 
= event
.m_keyCode
; 
1022             event
.SetEventType( wxEVT_CHAR_HOOK 
); 
1023             ret 
= parent
->HandleWindowEvent( event 
); 
1028             event
.SetEventType(wxEVT_CHAR
); 
1029             ret 
= window
->HandleWindowEvent( event 
); 
1036 //----------------------------------------------------------------------------- 
1037 // "key_release_event" from any window 
1038 //----------------------------------------------------------------------------- 
1042 gtk_window_key_release_callback( GtkWidget 
* WXUNUSED(widget
), 
1043                                  GdkEventKey 
*gdk_event
, 
1049     if (g_blockEventsOnDrag
) 
1052     wxKeyEvent 
event( wxEVT_KEY_UP 
); 
1053     if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) ) 
1055         // unknown key pressed, ignore (the event would be useless anyhow) 
1059     return win
->GTKProcessEvent(event
); 
1063 // ============================================================================ 
1065 // ============================================================================ 
1067 // ---------------------------------------------------------------------------- 
1068 // mouse event processing helpers 
1069 // ---------------------------------------------------------------------------- 
1071 // init wxMouseEvent with the info from GdkEventXXX struct 
1072 template<typename T
> void InitMouseEvent(wxWindowGTK 
*win
, 
1073                                          wxMouseEvent
& event
, 
1076     event
.SetTimestamp( gdk_event
->time 
); 
1077     event
.m_shiftDown 
= (gdk_event
->state 
& GDK_SHIFT_MASK
) != 0; 
1078     event
.m_controlDown 
= (gdk_event
->state 
& GDK_CONTROL_MASK
) != 0; 
1079     event
.m_altDown 
= (gdk_event
->state 
& GDK_MOD1_MASK
) != 0; 
1080     event
.m_metaDown 
= (gdk_event
->state 
& GDK_META_MASK
) != 0; 
1081     event
.m_leftDown 
= (gdk_event
->state 
& GDK_BUTTON1_MASK
) != 0; 
1082     event
.m_middleDown 
= (gdk_event
->state 
& GDK_BUTTON2_MASK
) != 0; 
1083     event
.m_rightDown 
= (gdk_event
->state 
& GDK_BUTTON3_MASK
) != 0; 
1084     event
.m_aux1Down 
= (gdk_event
->state 
& GDK_BUTTON4_MASK
) != 0; 
1085     event
.m_aux2Down 
= (gdk_event
->state 
& GDK_BUTTON5_MASK
) != 0; 
1087     wxPoint pt 
= win
->GetClientAreaOrigin(); 
1088     event
.m_x 
= (wxCoord
)gdk_event
->x 
- pt
.x
; 
1089     event
.m_y 
= (wxCoord
)gdk_event
->y 
- pt
.y
; 
1091     if ((win
->m_wxwindow
) && (win
->GetLayoutDirection() == wxLayout_RightToLeft
)) 
1093         // origin in the upper right corner 
1094         int window_width 
= win
->m_wxwindow
->allocation
.width
; 
1095         event
.m_x 
= window_width 
- event
.m_x
; 
1098     event
.SetEventObject( win 
); 
1099     event
.SetId( win
->GetId() ); 
1100     event
.SetTimestamp( gdk_event
->time 
); 
1103 static void AdjustEventButtonState(wxMouseEvent
& event
) 
1105     // GDK reports the old state of the button for a button press event, but 
1106     // for compatibility with MSW and common sense we want m_leftDown be TRUE 
1107     // for a LEFT_DOWN event, not FALSE, so we will invert 
1108     // left/right/middleDown for the corresponding click events 
1110     if ((event
.GetEventType() == wxEVT_LEFT_DOWN
) || 
1111         (event
.GetEventType() == wxEVT_LEFT_DCLICK
) || 
1112         (event
.GetEventType() == wxEVT_LEFT_UP
)) 
1114         event
.m_leftDown 
= !event
.m_leftDown
; 
1118     if ((event
.GetEventType() == wxEVT_MIDDLE_DOWN
) || 
1119         (event
.GetEventType() == wxEVT_MIDDLE_DCLICK
) || 
1120         (event
.GetEventType() == wxEVT_MIDDLE_UP
)) 
1122         event
.m_middleDown 
= !event
.m_middleDown
; 
1126     if ((event
.GetEventType() == wxEVT_RIGHT_DOWN
) || 
1127         (event
.GetEventType() == wxEVT_RIGHT_DCLICK
) || 
1128         (event
.GetEventType() == wxEVT_RIGHT_UP
)) 
1130         event
.m_rightDown 
= !event
.m_rightDown
; 
1135 // find the window to send the mouse event too 
1137 wxWindowGTK 
*FindWindowForMouseEvent(wxWindowGTK 
*win
, wxCoord
& x
, wxCoord
& y
) 
1142     if (win
->m_wxwindow
) 
1144         wxPizza
* pizza 
= WX_PIZZA(win
->m_wxwindow
); 
1145         xx 
+= pizza
->m_scroll_x
; 
1146         yy 
+= pizza
->m_scroll_y
; 
1149     wxWindowList::compatibility_iterator node 
= win
->GetChildren().GetFirst(); 
1152         wxWindowGTK 
*child 
= node
->GetData(); 
1154         node 
= node
->GetNext(); 
1155         if (!child
->IsShown()) 
1158         if (child
->GTKIsTransparentForMouse()) 
1160             // wxStaticBox is transparent in the box itself 
1161             int xx1 
= child
->m_x
; 
1162             int yy1 
= child
->m_y
; 
1163             int xx2 
= child
->m_x 
+ child
->m_width
; 
1164             int yy2 
= child
->m_y 
+ child
->m_height
; 
1167             if (((xx 
>= xx1
) && (xx 
<= xx1
+10) && (yy 
>= yy1
) && (yy 
<= yy2
)) || 
1169                 ((xx 
>= xx2
-10) && (xx 
<= xx2
) && (yy 
>= yy1
) && (yy 
<= yy2
)) || 
1171                 ((xx 
>= xx1
) && (xx 
<= xx2
) && (yy 
>= yy1
) && (yy 
<= yy1
+10)) || 
1173                 ((xx 
>= xx1
) && (xx 
<= xx2
) && (yy 
>= yy2
-1) && (yy 
<= yy2
))) 
1184             if ((child
->m_wxwindow 
== NULL
) && 
1185                 (child
->m_x 
<= xx
) && 
1186                 (child
->m_y 
<= yy
) && 
1187                 (child
->m_x
+child
->m_width  
>= xx
) && 
1188                 (child
->m_y
+child
->m_height 
>= yy
)) 
1201 // ---------------------------------------------------------------------------- 
1202 // common event handlers helpers 
1203 // ---------------------------------------------------------------------------- 
1205 bool wxWindowGTK::GTKProcessEvent(wxEvent
& event
) const 
1207     // nothing special at this level 
1208     return HandleWindowEvent(event
); 
1211 bool wxWindowGTK::GTKShouldIgnoreEvent() const 
1213     return !m_hasVMT 
|| g_blockEventsOnDrag
; 
1216 int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny 
*event
) const 
1220     if (g_blockEventsOnDrag
) 
1222     if (g_blockEventsOnScroll
) 
1225     if (!GTKIsOwnWindow(event
->window
)) 
1231 // overloads for all GDK event types we use here: we need to have this as 
1232 // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact, 
1233 // derives from it in the sense that the structs have the same layout 
1234 #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T)                                  \ 
1235     static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win)        \ 
1237         return win->GTKCallbackCommonPrologue((GdkEventAny *)event);          \ 
1240 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton
) 
1241 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion
) 
1242 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing
) 
1244 #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD 
1246 #define wxCOMMON_CALLBACK_PROLOGUE(event, win)                                \ 
1247     const int rc = wxGtkCallbackCommonPrologue(event, win);                   \ 
1251 // all event handlers must have C linkage as they're called from GTK+ C code 
1255 //----------------------------------------------------------------------------- 
1256 // "button_press_event" 
1257 //----------------------------------------------------------------------------- 
1260 gtk_window_button_press_callback( GtkWidget 
*widget
, 
1261                                   GdkEventButton 
*gdk_event
, 
1264     wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
); 
1266     g_lastButtonNumber 
= gdk_event
->button
; 
1268     // GDK sends surplus button down events 
1269     // before a double click event. We 
1270     // need to filter these out. 
1271     if ((gdk_event
->type 
== GDK_BUTTON_PRESS
) && (win
->m_wxwindow
)) 
1273         GdkEvent 
*peek_event 
= gdk_event_peek(); 
1276             if ((peek_event
->type 
== GDK_2BUTTON_PRESS
) || 
1277                 (peek_event
->type 
== GDK_3BUTTON_PRESS
)) 
1279                 gdk_event_free( peek_event 
); 
1284                 gdk_event_free( peek_event 
); 
1289     wxEventType event_type 
= wxEVT_NULL
; 
1291     if ( gdk_event
->type 
== GDK_2BUTTON_PRESS 
&& 
1292             gdk_event
->button 
>= 1 && gdk_event
->button 
<= 3 ) 
1294         // Reset GDK internal timestamp variables in order to disable GDK 
1295         // triple click events. GDK will then next time believe no button has 
1296         // been clicked just before, and send a normal button click event. 
1297         GdkDisplay
* display 
= gtk_widget_get_display (widget
); 
1298         display
->button_click_time
[1] = 0; 
1299         display
->button_click_time
[0] = 0; 
1302     if (gdk_event
->button 
== 1) 
1304         // note that GDK generates triple click events which are not supported 
1305         // by wxWidgets but still have to be passed to the app as otherwise 
1306         // clicks would simply go missing 
1307         switch (gdk_event
->type
) 
1309             // we shouldn't get triple clicks at all for GTK2 because we 
1310             // suppress them artificially using the code above but we still 
1311             // should map them to something for GTK1 and not just ignore them 
1312             // as this would lose clicks 
1313             case GDK_3BUTTON_PRESS
:     // we could also map this to DCLICK... 
1314             case GDK_BUTTON_PRESS
: 
1315                 event_type 
= wxEVT_LEFT_DOWN
; 
1318             case GDK_2BUTTON_PRESS
: 
1319                 event_type 
= wxEVT_LEFT_DCLICK
; 
1323                 // just to silence gcc warnings 
1327     else if (gdk_event
->button 
== 2) 
1329         switch (gdk_event
->type
) 
1331             case GDK_3BUTTON_PRESS
: 
1332             case GDK_BUTTON_PRESS
: 
1333                 event_type 
= wxEVT_MIDDLE_DOWN
; 
1336             case GDK_2BUTTON_PRESS
: 
1337                 event_type 
= wxEVT_MIDDLE_DCLICK
; 
1344     else if (gdk_event
->button 
== 3) 
1346         switch (gdk_event
->type
) 
1348             case GDK_3BUTTON_PRESS
: 
1349             case GDK_BUTTON_PRESS
: 
1350                 event_type 
= wxEVT_RIGHT_DOWN
; 
1353             case GDK_2BUTTON_PRESS
: 
1354                 event_type 
= wxEVT_RIGHT_DCLICK
; 
1362     if ( event_type 
== wxEVT_NULL 
) 
1364         // unknown mouse button or click type 
1368     g_lastMouseEvent 
= (GdkEvent
*) gdk_event
; 
1370     wxMouseEvent 
event( event_type 
); 
1371     InitMouseEvent( win
, event
, gdk_event 
); 
1373     AdjustEventButtonState(event
); 
1375     // find the correct window to send the event to: it may be a different one 
1376     // from the one which got it at GTK+ level because some controls don't have 
1377     // their own X window and thus cannot get any events. 
1378     if ( !g_captureWindow 
) 
1379         win 
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
); 
1381     // reset the event object and id in case win changed. 
1382     event
.SetEventObject( win 
); 
1383     event
.SetId( win
->GetId() ); 
1385     bool ret 
= win
->GTKProcessEvent( event 
); 
1386     g_lastMouseEvent 
= NULL
; 
1390     if ((event_type 
== wxEVT_LEFT_DOWN
) && !win
->IsOfStandardClass() && 
1391         (gs_currentFocus 
!= win
) /* && win->IsFocusable() */) 
1396     if (event_type 
== wxEVT_RIGHT_DOWN
) 
1398         // generate a "context menu" event: this is similar to right mouse 
1399         // click under many GUIs except that it is generated differently 
1400         // (right up under MSW, ctrl-click under Mac, right down here) and 
1402         // (a) it's a command event and so is propagated to the parent 
1403         // (b) under some ports it can be generated from kbd too 
1404         // (c) it uses screen coords (because of (a)) 
1405         wxContextMenuEvent 
evtCtx( 
1408             win
->ClientToScreen(event
.GetPosition())); 
1409         evtCtx
.SetEventObject(win
); 
1410         return win
->GTKProcessEvent(evtCtx
); 
1416 //----------------------------------------------------------------------------- 
1417 // "button_release_event" 
1418 //----------------------------------------------------------------------------- 
1421 gtk_window_button_release_callback( GtkWidget 
*WXUNUSED(widget
), 
1422                                     GdkEventButton 
*gdk_event
, 
1425     wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
); 
1427     g_lastButtonNumber 
= 0; 
1429     wxEventType event_type 
= wxEVT_NULL
; 
1431     switch (gdk_event
->button
) 
1434             event_type 
= wxEVT_LEFT_UP
; 
1438             event_type 
= wxEVT_MIDDLE_UP
; 
1442             event_type 
= wxEVT_RIGHT_UP
; 
1446             // unknown button, don't process 
1450     g_lastMouseEvent 
= (GdkEvent
*) gdk_event
; 
1452     wxMouseEvent 
event( event_type 
); 
1453     InitMouseEvent( win
, event
, gdk_event 
); 
1455     AdjustEventButtonState(event
); 
1457     if ( !g_captureWindow 
) 
1458         win 
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
); 
1460     // reset the event object and id in case win changed. 
1461     event
.SetEventObject( win 
); 
1462     event
.SetId( win
->GetId() ); 
1464     bool ret 
= win
->GTKProcessEvent(event
); 
1466     g_lastMouseEvent 
= NULL
; 
1471 //----------------------------------------------------------------------------- 
1472 // "motion_notify_event" 
1473 //----------------------------------------------------------------------------- 
1476 gtk_window_motion_notify_callback( GtkWidget 
* WXUNUSED(widget
), 
1477                                    GdkEventMotion 
*gdk_event
, 
1480     wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
); 
1482     if (gdk_event
->is_hint
) 
1486         GdkModifierType state
; 
1487         gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
); 
1492     g_lastMouseEvent 
= (GdkEvent
*) gdk_event
; 
1494     wxMouseEvent 
event( wxEVT_MOTION 
); 
1495     InitMouseEvent(win
, event
, gdk_event
); 
1497     if ( g_captureWindow 
) 
1499         // synthesise a mouse enter or leave event if needed 
1500         GdkWindow 
*winUnderMouse 
= gdk_window_at_pointer(NULL
, NULL
); 
1501         // This seems to be necessary and actually been added to 
1502         // GDK itself in version 2.0.X 
1505         bool hasMouse 
= winUnderMouse 
== gdk_event
->window
; 
1506         if ( hasMouse 
!= g_captureWindowHasMouse 
) 
1508             // the mouse changed window 
1509             g_captureWindowHasMouse 
= hasMouse
; 
1511             wxMouseEvent 
eventM(g_captureWindowHasMouse 
? wxEVT_ENTER_WINDOW
 
1512                                                         : wxEVT_LEAVE_WINDOW
); 
1513             InitMouseEvent(win
, eventM
, gdk_event
); 
1514             eventM
.SetEventObject(win
); 
1515             win
->GTKProcessEvent(eventM
); 
1520         win 
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
); 
1522         // reset the event object and id in case win changed. 
1523         event
.SetEventObject( win 
); 
1524         event
.SetId( win
->GetId() ); 
1527     if ( !g_captureWindow 
) 
1529         wxSetCursorEvent 
cevent( event
.m_x
, event
.m_y 
); 
1530         if (win
->GTKProcessEvent( cevent 
)) 
1532             win
->SetCursor( cevent
.GetCursor() ); 
1536     bool ret 
= win
->GTKProcessEvent(event
); 
1538     g_lastMouseEvent 
= NULL
; 
1543 //----------------------------------------------------------------------------- 
1544 // "scroll_event" (mouse wheel event) 
1545 //----------------------------------------------------------------------------- 
1548 window_scroll_event_hscrollbar(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
) 
1550     if (gdk_event
->direction 
!= GDK_SCROLL_LEFT 
&& 
1551         gdk_event
->direction 
!= GDK_SCROLL_RIGHT
) 
1556     wxMouseEvent 
event(wxEVT_MOUSEWHEEL
); 
1557     InitMouseEvent(win
, event
, gdk_event
); 
1559     GtkRange 
*range 
= win
->m_scrollBar
[wxWindow::ScrollDir_Horz
]; 
1560     if (!range
) return FALSE
; 
1562     if (range 
&& GTK_WIDGET_VISIBLE (range
)) 
1564         GtkAdjustment 
*adj 
= range
->adjustment
; 
1565         gdouble delta 
= adj
->step_increment 
* 3; 
1566         if (gdk_event
->direction 
== GDK_SCROLL_LEFT
) 
1569         gdouble new_value 
= CLAMP (adj
->value 
+ delta
, adj
->lower
, adj
->upper 
- adj
->page_size
); 
1571         gtk_adjustment_set_value (adj
, new_value
); 
1580 window_scroll_event(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
) 
1582     if (gdk_event
->direction 
!= GDK_SCROLL_UP 
&& 
1583         gdk_event
->direction 
!= GDK_SCROLL_DOWN
) 
1588     wxMouseEvent 
event(wxEVT_MOUSEWHEEL
); 
1589     InitMouseEvent(win
, event
, gdk_event
); 
1591     // FIXME: Get these values from GTK or GDK 
1592     event
.m_linesPerAction 
= 3; 
1593     event
.m_wheelDelta 
= 120; 
1594     if (gdk_event
->direction 
== GDK_SCROLL_UP
) 
1595         event
.m_wheelRotation 
= 120; 
1597         event
.m_wheelRotation 
= -120; 
1599     if (win
->GTKProcessEvent(event
)) 
1602     GtkRange 
*range 
= win
->m_scrollBar
[wxWindow::ScrollDir_Vert
]; 
1603     if (!range
) return FALSE
; 
1605     if (range 
&& GTK_WIDGET_VISIBLE (range
)) 
1607         GtkAdjustment 
*adj 
= range
->adjustment
; 
1608         gdouble delta 
= adj
->step_increment 
* 3; 
1609         if (gdk_event
->direction 
== GDK_SCROLL_UP
) 
1612         gdouble new_value 
= CLAMP (adj
->value 
+ delta
, adj
->lower
, adj
->upper 
- adj
->page_size
); 
1614         gtk_adjustment_set_value (adj
, new_value
); 
1622 //----------------------------------------------------------------------------- 
1624 //----------------------------------------------------------------------------- 
1626 static gboolean 
wxgtk_window_popup_menu_callback(GtkWidget
*, wxWindowGTK
* win
) 
1628     wxContextMenuEvent 
event(wxEVT_CONTEXT_MENU
, win
->GetId(), wxPoint(-1, -1)); 
1629     event
.SetEventObject(win
); 
1630     return win
->GTKProcessEvent(event
); 
1633 //----------------------------------------------------------------------------- 
1635 //----------------------------------------------------------------------------- 
1638 gtk_window_focus_in_callback( GtkWidget 
* WXUNUSED(widget
), 
1639                               GdkEventFocus 
*WXUNUSED(event
), 
1642     return win
->GTKHandleFocusIn(); 
1645 //----------------------------------------------------------------------------- 
1646 // "focus_out_event" 
1647 //----------------------------------------------------------------------------- 
1650 gtk_window_focus_out_callback( GtkWidget 
* WXUNUSED(widget
), 
1651                                GdkEventFocus 
* WXUNUSED(gdk_event
), 
1654     return win
->GTKHandleFocusOut(); 
1657 //----------------------------------------------------------------------------- 
1659 //----------------------------------------------------------------------------- 
1662 wx_window_focus_callback(GtkWidget 
*widget
, 
1663                          GtkDirectionType 
WXUNUSED(direction
), 
1666     // the default handler for focus signal in GtkScrolledWindow sets 
1667     // focus to the window itself even if it doesn't accept focus, i.e. has no 
1668     // GTK_CAN_FOCUS in its style -- work around this by forcibly preventing 
1669     // the signal from reaching gtk_scrolled_window_focus() if we don't have 
1670     // any children which might accept focus (we know we don't accept the focus 
1671     // ourselves as this signal is only connected in this case) 
1672     if ( win
->GetChildren().empty() ) 
1673         g_signal_stop_emission_by_name(widget
, "focus"); 
1675     // we didn't change the focus 
1679 //----------------------------------------------------------------------------- 
1680 // "enter_notify_event" 
1681 //----------------------------------------------------------------------------- 
1684 gtk_window_enter_callback( GtkWidget 
*widget
, 
1685                            GdkEventCrossing 
*gdk_event
, 
1688     wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
); 
1690     // Event was emitted after a grab 
1691     if (gdk_event
->mode 
!= GDK_CROSSING_NORMAL
) return FALSE
; 
1695     GdkModifierType state 
= (GdkModifierType
)0; 
1697     gdk_window_get_pointer( widget
->window
, &x
, &y
, &state 
); 
1699     wxMouseEvent 
event( wxEVT_ENTER_WINDOW 
); 
1700     InitMouseEvent(win
, event
, gdk_event
); 
1701     wxPoint pt 
= win
->GetClientAreaOrigin(); 
1702     event
.m_x 
= x 
+ pt
.x
; 
1703     event
.m_y 
= y 
+ pt
.y
; 
1705     if ( !g_captureWindow 
) 
1707         wxSetCursorEvent 
cevent( event
.m_x
, event
.m_y 
); 
1708         if (win
->GTKProcessEvent( cevent 
)) 
1710             win
->SetCursor( cevent
.GetCursor() ); 
1714     return win
->GTKProcessEvent(event
); 
1717 //----------------------------------------------------------------------------- 
1718 // "leave_notify_event" 
1719 //----------------------------------------------------------------------------- 
1722 gtk_window_leave_callback( GtkWidget 
*widget
, 
1723                            GdkEventCrossing 
*gdk_event
, 
1726     wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
); 
1728     // Event was emitted after an ungrab 
1729     if (gdk_event
->mode 
!= GDK_CROSSING_NORMAL
) return FALSE
; 
1731     wxMouseEvent 
event( wxEVT_LEAVE_WINDOW 
); 
1735     GdkModifierType state 
= (GdkModifierType
)0; 
1737     gdk_window_get_pointer( widget
->window
, &x
, &y
, &state 
); 
1739     InitMouseEvent(win
, event
, gdk_event
); 
1741     return win
->GTKProcessEvent(event
); 
1744 //----------------------------------------------------------------------------- 
1745 // "value_changed" from scrollbar 
1746 //----------------------------------------------------------------------------- 
1749 gtk_scrollbar_value_changed(GtkRange
* range
, wxWindow
* win
) 
1751     wxEventType eventType 
= win
->GTKGetScrollEventType(range
); 
1752     if (eventType 
!= wxEVT_NULL
) 
1754         // Convert scroll event type to scrollwin event type 
1755         eventType 
+= wxEVT_SCROLLWIN_TOP 
- wxEVT_SCROLL_TOP
; 
1757         // find the scrollbar which generated the event 
1758         wxWindowGTK::ScrollDir dir 
= win
->ScrollDirFromRange(range
); 
1760         // generate the corresponding wx event 
1761         const int orient 
= wxWindow::OrientFromScrollDir(dir
); 
1762         wxScrollWinEvent 
event(eventType
, win
->GetScrollPos(orient
), orient
); 
1763         event
.SetEventObject(win
); 
1765         win
->GTKProcessEvent(event
); 
1769 //----------------------------------------------------------------------------- 
1770 // "button_press_event" from scrollbar 
1771 //----------------------------------------------------------------------------- 
1774 gtk_scrollbar_button_press_event(GtkRange
*, GdkEventButton
*, wxWindow
* win
) 
1776     g_blockEventsOnScroll 
= true; 
1777     win
->m_mouseButtonDown 
= true; 
1782 //----------------------------------------------------------------------------- 
1783 // "event_after" from scrollbar 
1784 //----------------------------------------------------------------------------- 
1787 gtk_scrollbar_event_after(GtkRange
* range
, GdkEvent
* event
, wxWindow
* win
) 
1789     if (event
->type 
== GDK_BUTTON_RELEASE
) 
1791         g_signal_handlers_block_by_func(range
, (void*)gtk_scrollbar_event_after
, win
); 
1793         const int orient 
= wxWindow::OrientFromScrollDir( 
1794                                         win
->ScrollDirFromRange(range
)); 
1795         wxScrollWinEvent 
evt(wxEVT_SCROLLWIN_THUMBRELEASE
, 
1796                                 win
->GetScrollPos(orient
), orient
); 
1797         evt
.SetEventObject(win
); 
1798         win
->GTKProcessEvent(evt
); 
1802 //----------------------------------------------------------------------------- 
1803 // "button_release_event" from scrollbar 
1804 //----------------------------------------------------------------------------- 
1807 gtk_scrollbar_button_release_event(GtkRange
* range
, GdkEventButton
*, wxWindow
* win
) 
1809     g_blockEventsOnScroll 
= false; 
1810     win
->m_mouseButtonDown 
= false; 
1811     // If thumb tracking 
1812     if (win
->m_isScrolling
) 
1814         win
->m_isScrolling 
= false; 
1815         // Hook up handler to send thumb release event after this emission is finished. 
1816         // To allow setting scroll position from event handler, sending event must 
1817         // be deferred until after the GtkRange handler for this signal has run 
1818         g_signal_handlers_unblock_by_func(range
, (void*)gtk_scrollbar_event_after
, win
); 
1824 //----------------------------------------------------------------------------- 
1825 // "realize" from m_widget 
1826 //----------------------------------------------------------------------------- 
1829 gtk_window_realized_callback(GtkWidget
* widget
, wxWindow
* win
) 
1833         gtk_im_context_set_client_window( win
->m_imData
->context
, 
1837     // We cannot set colours and fonts before the widget 
1838     // been realized, so we do this directly after realization 
1839     // or otherwise in idle time 
1841     if (win
->m_needsStyleChange
) 
1843         win
->SetBackgroundStyle(win
->GetBackgroundStyle()); 
1844         win
->m_needsStyleChange 
= false; 
1847     wxWindowCreateEvent 
event( win 
); 
1848     event
.SetEventObject( win 
); 
1849     win
->GTKProcessEvent( event 
); 
1852 //----------------------------------------------------------------------------- 
1853 // "size_allocate" from m_wxwindow or m_widget 
1854 //----------------------------------------------------------------------------- 
1857 size_allocate(GtkWidget
*, GtkAllocation
* alloc
, wxWindow
* win
) 
1859     int w 
= alloc
->width
; 
1860     int h 
= alloc
->height
; 
1861     if (win
->m_wxwindow
) 
1863         int border_x
, border_y
; 
1864         WX_PIZZA(win
->m_wxwindow
)->get_border_widths(border_x
, border_y
); 
1870     if (win
->m_oldClientWidth 
!= w 
|| win
->m_oldClientHeight 
!= h
) 
1872         win
->m_oldClientWidth  
= w
; 
1873         win
->m_oldClientHeight 
= h
; 
1874         // this callback can be connected to m_wxwindow, 
1875         // so always get size from m_widget->allocation 
1876         win
->m_width  
= win
->m_widget
->allocation
.width
; 
1877         win
->m_height 
= win
->m_widget
->allocation
.height
; 
1878         if (!win
->m_nativeSizeEvent
) 
1880             wxSizeEvent 
event(win
->GetSize(), win
->GetId()); 
1881             event
.SetEventObject(win
); 
1882             win
->GTKProcessEvent(event
); 
1887 //----------------------------------------------------------------------------- 
1889 //----------------------------------------------------------------------------- 
1891 #if GTK_CHECK_VERSION(2, 8, 0) 
1893 gtk_window_grab_broken( GtkWidget
*, 
1894                         GdkEventGrabBroken 
*event
, 
1897     // Mouse capture has been lost involuntarily, notify the application 
1898     if(!event
->keyboard 
&& wxWindow::GetCapture() == win
) 
1900         wxMouseCaptureLostEvent 
evt( win
->GetId() ); 
1901         evt
.SetEventObject( win 
); 
1902         win
->HandleWindowEvent( evt 
); 
1908 //----------------------------------------------------------------------------- 
1910 //----------------------------------------------------------------------------- 
1913 void gtk_window_style_set_callback( GtkWidget 
*WXUNUSED(widget
), 
1914                                GtkStyle 
*previous_style
, 
1917     if (win 
&& previous_style
) 
1919         wxSysColourChangedEvent event
; 
1920         event
.SetEventObject(win
); 
1922         win
->GTKProcessEvent( event 
); 
1928 // Helper to suspend colour change event event processing while we change a widget's style 
1929 class wxSuspendStyleEvents
 
1932     wxSuspendStyleEvents(wxWindow
* win
) 
1935         if (win
->m_wxwindow 
&& win
->IsTopLevel()) 
1938             g_signal_handlers_block_by_func( 
1939                 m_win
->m_wxwindow
, (void*)gtk_window_style_set_callback
, m_win
); 
1942     ~wxSuspendStyleEvents() 
1945             g_signal_handlers_unblock_by_func( 
1946                 m_win
->m_wxwindow
, (void*)gtk_window_style_set_callback
, m_win
); 
1952 // ---------------------------------------------------------------------------- 
1953 // this wxWindowBase function is implemented here (in platform-specific file) 
1954 // because it is static and so couldn't be made virtual 
1955 // ---------------------------------------------------------------------------- 
1957 wxWindow 
*wxWindowBase::DoFindFocus() 
1959     wxWindowGTK 
*focus 
= gs_pendingFocus 
? gs_pendingFocus 
: gs_currentFocus
; 
1960     // the cast is necessary when we compile in wxUniversal mode 
1961     return static_cast<wxWindow
*>(focus
); 
1964 void wxWindowGTK::AddChildGTK(wxWindowGTK
* child
) 
1966     wxASSERT_MSG(m_wxwindow
, "Cannot add a child to a window without a client area"); 
1968     // the window might have been scrolled already, we 
1969     // have to adapt the position 
1970     wxPizza
* pizza 
= WX_PIZZA(m_wxwindow
); 
1971     child
->m_x 
+= pizza
->m_scroll_x
; 
1972     child
->m_y 
+= pizza
->m_scroll_y
; 
1974     gtk_widget_set_size_request( 
1975         child
->m_widget
, child
->m_width
, child
->m_height
); 
1977         GTK_FIXED(m_wxwindow
), child
->m_widget
, child
->m_x
, child
->m_y
); 
1980 //----------------------------------------------------------------------------- 
1982 //----------------------------------------------------------------------------- 
1984 wxWindow 
*wxGetActiveWindow() 
1986     return wxWindow::FindFocus(); 
1990 wxMouseState 
wxGetMouseState() 
1996     GdkModifierType mask
; 
1998     gdk_window_get_pointer(NULL
, &x
, &y
, &mask
); 
2002     ms
.SetLeftDown((mask 
& GDK_BUTTON1_MASK
) != 0); 
2003     ms
.SetMiddleDown((mask 
& GDK_BUTTON2_MASK
) != 0); 
2004     ms
.SetRightDown((mask 
& GDK_BUTTON3_MASK
) != 0); 
2005     ms
.SetAux1Down((mask 
& GDK_BUTTON4_MASK
) != 0); 
2006     ms
.SetAux2Down((mask 
& GDK_BUTTON5_MASK
) != 0); 
2008     ms
.SetControlDown((mask 
& GDK_CONTROL_MASK
) != 0); 
2009     ms
.SetShiftDown((mask 
& GDK_SHIFT_MASK
) != 0); 
2010     ms
.SetAltDown((mask 
& GDK_MOD1_MASK
) != 0); 
2011     ms
.SetMetaDown((mask 
& GDK_META_MASK
) != 0); 
2016 //----------------------------------------------------------------------------- 
2018 //----------------------------------------------------------------------------- 
2020 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu() 
2022 #ifdef __WXUNIVERSAL__ 
2023     IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
) 
2025     IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
) 
2026 #endif // __WXUNIVERSAL__/__WXGTK__ 
2028 void wxWindowGTK::Init() 
2033     m_focusWidget 
= NULL
; 
2043     m_showOnIdle 
= false; 
2046     m_nativeSizeEvent 
= false; 
2048     m_isScrolling 
= false; 
2049     m_mouseButtonDown 
= false; 
2051     // initialize scrolling stuff 
2052     for ( int dir 
= 0; dir 
< ScrollDir_Max
; dir
++ ) 
2054         m_scrollBar
[dir
] = NULL
; 
2055         m_scrollPos
[dir
] = 0; 
2059     m_oldClientHeight 
= 0; 
2061     m_clipPaintRegion 
= false; 
2063     m_needsStyleChange 
= false; 
2065     m_cursor 
= *wxSTANDARD_CURSOR
; 
2068     m_dirtyTabOrder 
= false; 
2071 wxWindowGTK::wxWindowGTK() 
2076 wxWindowGTK::wxWindowGTK( wxWindow 
*parent
, 
2081                           const wxString 
&name  
) 
2085     Create( parent
, id
, pos
, size
, style
, name 
); 
2088 bool wxWindowGTK::Create( wxWindow 
*parent
, 
2093                           const wxString 
&name  
) 
2095     // Get default border 
2096     wxBorder border 
= GetBorder(style
); 
2098     style 
&= ~wxBORDER_MASK
; 
2101     if (!PreCreation( parent
, pos
, size 
) || 
2102         !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
2104         wxFAIL_MSG( wxT("wxWindowGTK creation failed") ); 
2108         // We should accept the native look 
2110         GtkScrolledWindowClass 
*scroll_class 
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) ); 
2111         scroll_class
->scrollbar_spacing 
= 0; 
2115     m_wxwindow 
= wxPizza::New(m_windowStyle
,this); 
2116 #ifndef __WXUNIVERSAL__ 
2117     if (HasFlag(wxPizza::BORDER_STYLES
)) 
2119         g_signal_connect(m_wxwindow
, "parent_set", 
2120             G_CALLBACK(parent_set
), this); 
2123     if (!HasFlag(wxHSCROLL
) && !HasFlag(wxVSCROLL
)) 
2124         m_widget 
= m_wxwindow
; 
2127         m_widget 
= gtk_scrolled_window_new( NULL
, NULL 
); 
2129         GtkScrolledWindow 
*scrolledWindow 
= GTK_SCROLLED_WINDOW(m_widget
); 
2131         // There is a conflict with default bindings at GTK+ 
2132         // level between scrolled windows and notebooks both of which want to use 
2133         // Ctrl-PageUp/Down: scrolled windows for scrolling in the horizontal 
2134         // direction and notebooks for changing pages -- we decide that if we don't 
2135         // have wxHSCROLL style we can safely sacrifice horizontal scrolling if it 
2136         // means we can get working keyboard navigation in notebooks 
2137         if ( !HasFlag(wxHSCROLL
) ) 
2140                 bindings 
= gtk_binding_set_by_class(G_OBJECT_GET_CLASS(m_widget
)); 
2143                 gtk_binding_entry_remove(bindings
, GDK_Page_Up
, GDK_CONTROL_MASK
); 
2144                 gtk_binding_entry_remove(bindings
, GDK_Page_Down
, GDK_CONTROL_MASK
); 
2148         if (HasFlag(wxALWAYS_SHOW_SB
)) 
2150             gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_ALWAYS
, GTK_POLICY_ALWAYS 
); 
2152             scrolledWindow
->hscrollbar_visible 
= TRUE
; 
2153             scrolledWindow
->vscrollbar_visible 
= TRUE
; 
2157             gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC 
); 
2160         m_scrollBar
[ScrollDir_Horz
] = GTK_RANGE(scrolledWindow
->hscrollbar
); 
2161         m_scrollBar
[ScrollDir_Vert
] = GTK_RANGE(scrolledWindow
->vscrollbar
); 
2162         if (GetLayoutDirection() == wxLayout_RightToLeft
) 
2163             gtk_range_set_inverted( m_scrollBar
[ScrollDir_Horz
], TRUE 
); 
2165         gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow 
); 
2167         // connect various scroll-related events 
2168         for ( int dir 
= 0; dir 
< ScrollDir_Max
; dir
++ ) 
2170             // these handlers block mouse events to any window during scrolling 
2171             // such as motion events and prevent GTK and wxWidgets from fighting 
2172             // over where the slider should be 
2173             g_signal_connect(m_scrollBar
[dir
], "button_press_event", 
2174                          G_CALLBACK(gtk_scrollbar_button_press_event
), this); 
2175             g_signal_connect(m_scrollBar
[dir
], "button_release_event", 
2176                          G_CALLBACK(gtk_scrollbar_button_release_event
), this); 
2178             gulong handler_id 
= g_signal_connect(m_scrollBar
[dir
], "event_after", 
2179                                 G_CALLBACK(gtk_scrollbar_event_after
), this); 
2180             g_signal_handler_block(m_scrollBar
[dir
], handler_id
); 
2182             // these handlers get notified when scrollbar slider moves 
2183             g_signal_connect_after(m_scrollBar
[dir
], "value_changed", 
2184                          G_CALLBACK(gtk_scrollbar_value_changed
), this); 
2187         gtk_widget_show( m_wxwindow 
); 
2189     g_object_ref(m_widget
); 
2192         m_parent
->DoAddChild( this ); 
2194     m_focusWidget 
= m_wxwindow
; 
2196     SetCanFocus(AcceptsFocus()); 
2203 wxWindowGTK::~wxWindowGTK() 
2207     if (gs_currentFocus 
== this) 
2208         gs_currentFocus 
= NULL
; 
2209     if (gs_pendingFocus 
== this) 
2210         gs_pendingFocus 
= NULL
; 
2212     if ( gs_deferredFocusOut 
== this ) 
2213         gs_deferredFocusOut 
= NULL
; 
2217     // destroy children before destroying this window itself 
2220     // unhook focus handlers to prevent stray events being 
2221     // propagated to this (soon to be) dead object 
2222     if (m_focusWidget 
!= NULL
) 
2224         g_signal_handlers_disconnect_by_func (m_focusWidget
, 
2225                                               (gpointer
) gtk_window_focus_in_callback
, 
2227         g_signal_handlers_disconnect_by_func (m_focusWidget
, 
2228                                               (gpointer
) gtk_window_focus_out_callback
, 
2235     // delete before the widgets to avoid a crash on solaris 
2240         // Note that gtk_widget_destroy() does not destroy the widget, it just 
2241         // emits the "destroy" signal. The widget is not actually destroyed 
2242         // until its reference count drops to zero. 
2243         gtk_widget_destroy(m_widget
); 
2244         // Release our reference, should be the last one 
2245         g_object_unref(m_widget
); 
2251 bool wxWindowGTK::PreCreation( wxWindowGTK 
*parent
, const wxPoint 
&pos
,  const wxSize 
&size 
) 
2253     if ( GTKNeedsParent() ) 
2255         wxCHECK_MSG( parent
, false, wxT("Must have non-NULL parent") ); 
2258     // Use either the given size, or the default if -1 is given. 
2259     // See wxWindowBase for these functions. 
2260     m_width 
= WidthDefault(size
.x
) ; 
2261     m_height 
= HeightDefault(size
.y
); 
2269 void wxWindowGTK::PostCreation() 
2271     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid window") ); 
2277             // these get reported to wxWidgets -> wxPaintEvent 
2279             g_signal_connect (m_wxwindow
, "expose_event", 
2280                               G_CALLBACK (gtk_window_expose_callback
), this); 
2282             if (GetLayoutDirection() == wxLayout_LeftToRight
) 
2283                 gtk_widget_set_redraw_on_allocate(m_wxwindow
, HasFlag(wxFULL_REPAINT_ON_RESIZE
)); 
2286         // Create input method handler 
2287         m_imData 
= new wxGtkIMData
; 
2289         // Cannot handle drawing preedited text yet 
2290         gtk_im_context_set_use_preedit( m_imData
->context
, FALSE 
); 
2292         g_signal_connect (m_imData
->context
, "commit", 
2293                           G_CALLBACK (gtk_wxwindow_commit_cb
), this); 
2298     if (!GTK_IS_WINDOW(m_widget
)) 
2300         if (m_focusWidget 
== NULL
) 
2301             m_focusWidget 
= m_widget
; 
2305             g_signal_connect (m_focusWidget
, "focus_in_event", 
2306                           G_CALLBACK (gtk_window_focus_in_callback
), this); 
2307             g_signal_connect (m_focusWidget
, "focus_out_event", 
2308                                 G_CALLBACK (gtk_window_focus_out_callback
), this); 
2312             g_signal_connect_after (m_focusWidget
, "focus_in_event", 
2313                           G_CALLBACK (gtk_window_focus_in_callback
), this); 
2314             g_signal_connect_after (m_focusWidget
, "focus_out_event", 
2315                                 G_CALLBACK (gtk_window_focus_out_callback
), this); 
2319     if ( !AcceptsFocusFromKeyboard() ) 
2323         g_signal_connect(m_widget
, "focus", 
2324                             G_CALLBACK(wx_window_focus_callback
), this); 
2327     // connect to the various key and mouse handlers 
2329     GtkWidget 
*connect_widget 
= GetConnectWidget(); 
2331     ConnectWidget( connect_widget 
); 
2333     /* We cannot set colours, fonts and cursors before the widget has 
2334        been realized, so we do this directly after realization */ 
2335     g_signal_connect (connect_widget
, "realize", 
2336                       G_CALLBACK (gtk_window_realized_callback
), this); 
2340         g_signal_connect(m_wxwindow 
? m_wxwindow 
: m_widget
, "size_allocate", 
2341             G_CALLBACK(size_allocate
), this); 
2344 #if GTK_CHECK_VERSION(2, 8, 0) 
2345     if ( gtk_check_version(2,8,0) == NULL 
) 
2347         // Make sure we can notify the app when mouse capture is lost 
2350             g_signal_connect (m_wxwindow
, "grab_broken_event", 
2351                           G_CALLBACK (gtk_window_grab_broken
), this); 
2354         if ( connect_widget 
!= m_wxwindow 
) 
2356             g_signal_connect (connect_widget
, "grab_broken_event", 
2357                         G_CALLBACK (gtk_window_grab_broken
), this); 
2360 #endif // GTK+ >= 2.8 
2362     if ( GTKShouldConnectSizeRequest() ) 
2364         // This is needed if we want to add our windows into native 
2365         // GTK controls, such as the toolbar. With this callback, the 
2366         // toolbar gets to know the correct size (the one set by the 
2367         // programmer). Sadly, it misbehaves for wxComboBox. 
2368         g_signal_connect (m_widget
, "size_request", 
2369                           G_CALLBACK (wxgtk_window_size_request_callback
), 
2373     InheritAttributes(); 
2377     SetLayoutDirection(wxLayout_Default
); 
2379     // unless the window was created initially hidden (i.e. Hide() had been 
2380     // called before Create()), we should show it at GTK+ level as well 
2382         gtk_widget_show( m_widget 
); 
2385 void wxWindowGTK::ConnectWidget( GtkWidget 
*widget 
) 
2387     g_signal_connect (widget
, "key_press_event", 
2388                       G_CALLBACK (gtk_window_key_press_callback
), this); 
2389     g_signal_connect (widget
, "key_release_event", 
2390                       G_CALLBACK (gtk_window_key_release_callback
), this); 
2391     g_signal_connect (widget
, "button_press_event", 
2392                       G_CALLBACK (gtk_window_button_press_callback
), this); 
2393     g_signal_connect (widget
, "button_release_event", 
2394                       G_CALLBACK (gtk_window_button_release_callback
), this); 
2395     g_signal_connect (widget
, "motion_notify_event", 
2396                       G_CALLBACK (gtk_window_motion_notify_callback
), this); 
2398     g_signal_connect (widget
, "scroll_event", 
2399                       G_CALLBACK (window_scroll_event
), this); 
2400     if (m_scrollBar
[ScrollDir_Horz
]) 
2401         g_signal_connect (m_scrollBar
[ScrollDir_Horz
], "scroll_event", 
2402                       G_CALLBACK (window_scroll_event_hscrollbar
), this); 
2403     if (m_scrollBar
[ScrollDir_Vert
]) 
2404         g_signal_connect (m_scrollBar
[ScrollDir_Vert
], "scroll_event", 
2405                       G_CALLBACK (window_scroll_event
), this); 
2407     g_signal_connect (widget
, "popup_menu", 
2408                      G_CALLBACK (wxgtk_window_popup_menu_callback
), this); 
2409     g_signal_connect (widget
, "enter_notify_event", 
2410                       G_CALLBACK (gtk_window_enter_callback
), this); 
2411     g_signal_connect (widget
, "leave_notify_event", 
2412                       G_CALLBACK (gtk_window_leave_callback
), this); 
2414     if (IsTopLevel() && m_wxwindow
) 
2415         g_signal_connect (m_wxwindow
, "style_set", 
2416                               G_CALLBACK (gtk_window_style_set_callback
), this); 
2419 bool wxWindowGTK::Destroy() 
2421     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid window") ); 
2425     return wxWindowBase::Destroy(); 
2428 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
) 
2430     gtk_widget_set_size_request(m_widget
, width
, height
); 
2432     // inform the parent to perform the move 
2433     wxASSERT_MSG(m_parent 
&& m_parent
->m_wxwindow
, 
2434                  "the parent window has no client area?"); 
2435     WX_PIZZA(m_parent
->m_wxwindow
)->move(m_widget
, x
, y
); 
2438 void wxWindowGTK::ConstrainSize() 
2441     // GPE's window manager doesn't like size hints at all, esp. when the user 
2442     // has to use the virtual keyboard, so don't constrain size there 
2446         const wxSize minSize 
= GetMinSize(); 
2447         const wxSize maxSize 
= GetMaxSize(); 
2448         if (minSize
.x 
> 0 && m_width  
< minSize
.x
) m_width  
= minSize
.x
; 
2449         if (minSize
.y 
> 0 && m_height 
< minSize
.y
) m_height 
= minSize
.y
; 
2450         if (maxSize
.x 
> 0 && m_width  
> maxSize
.x
) m_width  
= maxSize
.x
; 
2451         if (maxSize
.y 
> 0 && m_height 
> maxSize
.y
) m_height 
= maxSize
.y
; 
2455 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags 
) 
2457     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid window") ); 
2458     wxASSERT_MSG( (m_parent 
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") ); 
2460     int currentX
, currentY
; 
2461     GetPosition(¤tX
, ¤tY
); 
2462     if (x 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
2464     if (y 
== -1 && !(sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
)) 
2466     AdjustForParentClientOrigin(x
, y
, sizeFlags
); 
2468     // calculate the best size if we should auto size the window 
2469     if ( ((sizeFlags 
& wxSIZE_AUTO_WIDTH
) && width 
== -1) || 
2470          ((sizeFlags 
& wxSIZE_AUTO_HEIGHT
) && height 
== -1) ) 
2472         const wxSize sizeBest 
= GetBestSize(); 
2473         if ( (sizeFlags 
& wxSIZE_AUTO_WIDTH
) && width 
== -1 ) 
2475         if ( (sizeFlags 
& wxSIZE_AUTO_HEIGHT
) && height 
== -1 ) 
2476             height 
= sizeBest
.y
; 
2479     const wxSize 
oldSize(m_width
, m_height
); 
2485     if (m_parent
->m_wxwindow
) 
2487         wxPizza
* pizza 
= WX_PIZZA(m_parent
->m_wxwindow
); 
2488         m_x 
= x 
+ pizza
->m_scroll_x
; 
2489         m_y 
= y 
+ pizza
->m_scroll_y
; 
2491         int left_border 
= 0; 
2492         int right_border 
= 0; 
2494         int bottom_border 
= 0; 
2496         /* the default button has a border around it */ 
2497         if (GTK_WIDGET_CAN_DEFAULT(m_widget
)) 
2499             GtkBorder 
*default_border 
= NULL
; 
2500             gtk_widget_style_get( m_widget
, "default_border", &default_border
, NULL 
); 
2503                 left_border 
+= default_border
->left
; 
2504                 right_border 
+= default_border
->right
; 
2505                 top_border 
+= default_border
->top
; 
2506                 bottom_border 
+= default_border
->bottom
; 
2507                 gtk_border_free( default_border 
); 
2511         DoMoveWindow( m_x 
- left_border
, 
2513                       m_width
+left_border
+right_border
, 
2514                       m_height
+top_border
+bottom_border 
); 
2517     if (m_width 
!= oldSize
.x 
|| m_height 
!= oldSize
.y
) 
2519         // update these variables to keep size_allocate handler 
2520         // from sending another size event for this change 
2521         GetClientSize( &m_oldClientWidth
, &m_oldClientHeight 
); 
2523         gtk_widget_queue_resize(m_widget
); 
2524         if (!m_nativeSizeEvent
) 
2526             wxSizeEvent 
event( wxSize(m_width
,m_height
), GetId() ); 
2527             event
.SetEventObject( this ); 
2528             HandleWindowEvent( event 
); 
2531     if (sizeFlags 
& wxSIZE_FORCE_EVENT
) 
2533         wxSizeEvent 
event( wxSize(m_width
,m_height
), GetId() ); 
2534         event
.SetEventObject( this ); 
2535         HandleWindowEvent( event 
); 
2539 bool wxWindowGTK::GTKShowFromOnIdle() 
2541     if (IsShown() && m_showOnIdle 
&& !GTK_WIDGET_VISIBLE (m_widget
)) 
2543         GtkAllocation alloc
; 
2546         alloc
.width 
= m_width
; 
2547         alloc
.height 
= m_height
; 
2548         gtk_widget_size_allocate( m_widget
, &alloc 
); 
2549         gtk_widget_show( m_widget 
); 
2550         wxShowEvent 
eventShow(GetId(), true); 
2551         eventShow
.SetEventObject(this); 
2552         HandleWindowEvent(eventShow
); 
2553         m_showOnIdle 
= false; 
2560 void wxWindowGTK::OnInternalIdle() 
2562     if ( gs_deferredFocusOut 
) 
2563         GTKHandleDeferredFocusOut(); 
2565     // Check if we have to show window now 
2566     if (GTKShowFromOnIdle()) return; 
2568     if ( m_dirtyTabOrder 
) 
2570         m_dirtyTabOrder 
= false; 
2574     // Update style if the window was not yet realized when 
2575     // SetBackgroundStyle() was called 
2576     if (m_needsStyleChange
) 
2578         SetBackgroundStyle(GetBackgroundStyle()); 
2579         m_needsStyleChange 
= false; 
2582     wxCursor cursor 
= m_cursor
; 
2583     if (g_globalCursor
.Ok()) cursor 
= g_globalCursor
; 
2587         /* I now set the cursor anew in every OnInternalIdle call 
2588            as setting the cursor in a parent window also effects the 
2589            windows above so that checking for the current cursor is 
2592         if (m_wxwindow 
&& (m_wxwindow 
!= m_widget
)) 
2594             GdkWindow 
*window 
= m_wxwindow
->window
; 
2596                 gdk_window_set_cursor( window
, cursor
.GetCursor() ); 
2598             if (!g_globalCursor
.Ok()) 
2599                 cursor 
= *wxSTANDARD_CURSOR
; 
2601             window 
= m_widget
->window
; 
2602             if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
))) 
2603                 gdk_window_set_cursor( window
, cursor
.GetCursor() ); 
2606         else if ( m_widget 
) 
2608             GdkWindow 
*window 
= m_widget
->window
; 
2609             if ( window 
&& !GTK_WIDGET_NO_WINDOW(m_widget
) ) 
2610                gdk_window_set_cursor( window
, cursor
.GetCursor() ); 
2614     if (wxUpdateUIEvent::CanUpdate(this) && IsShownOnScreen()) 
2615         UpdateWindowUI(wxUPDATE_UI_FROMIDLE
); 
2618 void wxWindowGTK::DoGetSize( int *width
, int *height 
) const 
2620     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2622     if (width
) (*width
) = m_width
; 
2623     if (height
) (*height
) = m_height
; 
2626 void wxWindowGTK::DoSetClientSize( int width
, int height 
) 
2628     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2630     const wxSize size 
= GetSize(); 
2631     const wxSize clientSize 
= GetClientSize(); 
2632     SetSize(width 
+ (size
.x 
- clientSize
.x
), height 
+ (size
.y 
- clientSize
.y
)); 
2635 void wxWindowGTK::DoGetClientSize( int *width
, int *height 
) const 
2637     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2644         // if window is scrollable, account for scrollbars 
2645         if ( GTK_IS_SCROLLED_WINDOW(m_widget
) ) 
2647             GtkPolicyType policy
[ScrollDir_Max
]; 
2648             gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(m_widget
), 
2649                                            &policy
[ScrollDir_Horz
], 
2650                                            &policy
[ScrollDir_Vert
]); 
2652             for ( int i 
= 0; i 
< ScrollDir_Max
; i
++ ) 
2654                 // don't account for the scrollbars we don't have 
2655                 GtkRange 
* const range 
= m_scrollBar
[i
]; 
2659                 // nor for the ones we have but don't current show 
2660                 switch ( policy
[i
] ) 
2662                     case GTK_POLICY_NEVER
: 
2663                         // never shown so doesn't take any place 
2666                     case GTK_POLICY_ALWAYS
: 
2667                         // no checks necessary 
2670                     case GTK_POLICY_AUTOMATIC
: 
2671                         // may be shown or not, check 
2672                         GtkAdjustment 
*adj 
= gtk_range_get_adjustment(range
); 
2673                         if ( adj
->upper 
<= adj
->page_size 
) 
2677                 GtkScrolledWindowClass 
*scroll_class 
= 
2678                     GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) ); 
2681                 gtk_widget_size_request(GTK_WIDGET(range
), &req
); 
2682                 if (i 
== ScrollDir_Horz
) 
2683                     h 
-= req
.height 
+ scroll_class
->scrollbar_spacing
; 
2685                     w 
-= req
.width 
+ scroll_class
->scrollbar_spacing
; 
2689         int border_x
, border_y
; 
2690         WX_PIZZA(m_wxwindow
)->get_border_widths(border_x
, border_y
); 
2700     if (width
) *width 
= w
; 
2701     if (height
) *height 
= h
; 
2704 void wxWindowGTK::DoGetPosition( int *x
, int *y 
) const 
2706     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2710     if (!IsTopLevel() && m_parent 
&& m_parent
->m_wxwindow
) 
2712         wxPizza
* pizza 
= WX_PIZZA(m_parent
->m_wxwindow
); 
2713         dx 
= pizza
->m_scroll_x
; 
2714         dy 
= pizza
->m_scroll_y
; 
2717     if (m_x 
== -1 && m_y 
== -1) 
2719         GdkWindow 
*source 
= NULL
; 
2721             source 
= m_wxwindow
->window
; 
2723             source 
= m_widget
->window
; 
2729             gdk_window_get_origin( source
, &org_x
, &org_y 
); 
2732                 m_parent
->ScreenToClient(&org_x
, &org_y
); 
2734             const_cast<wxWindowGTK
*>(this)->m_x 
= org_x
; 
2735             const_cast<wxWindowGTK
*>(this)->m_y 
= org_y
; 
2739     if (x
) (*x
) = m_x 
- dx
; 
2740     if (y
) (*y
) = m_y 
- dy
; 
2743 void wxWindowGTK::DoClientToScreen( int *x
, int *y 
) const 
2745     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2747     if (!m_widget
->window
) return; 
2749     GdkWindow 
*source 
= NULL
; 
2751         source 
= m_wxwindow
->window
; 
2753         source 
= m_widget
->window
; 
2757     gdk_window_get_origin( source
, &org_x
, &org_y 
); 
2761         if (GTK_WIDGET_NO_WINDOW (m_widget
)) 
2763             org_x 
+= m_widget
->allocation
.x
; 
2764             org_y 
+= m_widget
->allocation
.y
; 
2771         if (GetLayoutDirection() == wxLayout_RightToLeft
) 
2772             *x 
= (GetClientSize().x 
- *x
) + org_x
; 
2780 void wxWindowGTK::DoScreenToClient( int *x
, int *y 
) const 
2782     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2784     if (!m_widget
->window
) return; 
2786     GdkWindow 
*source 
= NULL
; 
2788         source 
= m_wxwindow
->window
; 
2790         source 
= m_widget
->window
; 
2794     gdk_window_get_origin( source
, &org_x
, &org_y 
); 
2798         if (GTK_WIDGET_NO_WINDOW (m_widget
)) 
2800             org_x 
+= m_widget
->allocation
.x
; 
2801             org_y 
+= m_widget
->allocation
.y
; 
2807         if (GetLayoutDirection() == wxLayout_RightToLeft
) 
2808             *x 
= (GetClientSize().x 
- *x
) - org_x
; 
2815 bool wxWindowGTK::Show( bool show 
) 
2817     if ( !wxWindowBase::Show(show
) ) 
2823     // notice that we may call Hide() before the window is created and this is 
2824     // actually useful to create it hidden initially -- but we can't call 
2825     // Show() before it is created 
2828         wxASSERT_MSG( !show
, "can't show invalid window" ); 
2836             // defer until later 
2840         gtk_widget_show(m_widget
); 
2844         gtk_widget_hide(m_widget
); 
2847     wxShowEvent 
eventShow(GetId(), show
); 
2848     eventShow
.SetEventObject(this); 
2849     HandleWindowEvent(eventShow
); 
2854 void wxWindowGTK::DoEnable( bool enable 
) 
2856     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
2858     gtk_widget_set_sensitive( m_widget
, enable 
); 
2859     if (m_wxwindow 
&& (m_wxwindow 
!= m_widget
)) 
2860         gtk_widget_set_sensitive( m_wxwindow
, enable 
); 
2863 int wxWindowGTK::GetCharHeight() const 
2865     wxCHECK_MSG( (m_widget 
!= NULL
), 12, wxT("invalid window") ); 
2867     wxFont font 
= GetFont(); 
2868     wxCHECK_MSG( font
.Ok(), 12, wxT("invalid font") ); 
2870     PangoContext
* context 
= gtk_widget_get_pango_context(m_widget
); 
2875     PangoFontDescription 
*desc 
= font
.GetNativeFontInfo()->description
; 
2876     PangoLayout 
*layout 
= pango_layout_new(context
); 
2877     pango_layout_set_font_description(layout
, desc
); 
2878     pango_layout_set_text(layout
, "H", 1); 
2879     PangoLayoutLine 
*line 
= (PangoLayoutLine 
*)pango_layout_get_lines(layout
)->data
; 
2881     PangoRectangle rect
; 
2882     pango_layout_line_get_extents(line
, NULL
, &rect
); 
2884     g_object_unref (layout
); 
2886     return (int) PANGO_PIXELS(rect
.height
); 
2889 int wxWindowGTK::GetCharWidth() const 
2891     wxCHECK_MSG( (m_widget 
!= NULL
), 8, wxT("invalid window") ); 
2893     wxFont font 
= GetFont(); 
2894     wxCHECK_MSG( font
.Ok(), 8, wxT("invalid font") ); 
2896     PangoContext
* context 
= gtk_widget_get_pango_context(m_widget
); 
2901     PangoFontDescription 
*desc 
= font
.GetNativeFontInfo()->description
; 
2902     PangoLayout 
*layout 
= pango_layout_new(context
); 
2903     pango_layout_set_font_description(layout
, desc
); 
2904     pango_layout_set_text(layout
, "g", 1); 
2905     PangoLayoutLine 
*line 
= (PangoLayoutLine 
*)pango_layout_get_lines(layout
)->data
; 
2907     PangoRectangle rect
; 
2908     pango_layout_line_get_extents(line
, NULL
, &rect
); 
2910     g_object_unref (layout
); 
2912     return (int) PANGO_PIXELS(rect
.width
); 
2915 void wxWindowGTK::DoGetTextExtent( const wxString
& string
, 
2919                                    int *externalLeading
, 
2920                                    const wxFont 
*theFont 
) const 
2922     wxFont fontToUse 
= theFont 
? *theFont 
: GetFont(); 
2924     wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") ); 
2933     PangoContext 
*context 
= NULL
; 
2935         context 
= gtk_widget_get_pango_context( m_widget 
); 
2944     PangoFontDescription 
*desc 
= fontToUse
.GetNativeFontInfo()->description
; 
2945     PangoLayout 
*layout 
= pango_layout_new(context
); 
2946     pango_layout_set_font_description(layout
, desc
); 
2948         const wxCharBuffer data 
= wxGTK_CONV( string 
); 
2950             pango_layout_set_text(layout
, data
, strlen(data
)); 
2953     PangoRectangle rect
; 
2954     pango_layout_get_extents(layout
, NULL
, &rect
); 
2956     if (x
) (*x
) = (wxCoord
) PANGO_PIXELS(rect
.width
); 
2957     if (y
) (*y
) = (wxCoord
) PANGO_PIXELS(rect
.height
); 
2960         PangoLayoutIter 
*iter 
= pango_layout_get_iter(layout
); 
2961         int baseline 
= pango_layout_iter_get_baseline(iter
); 
2962         pango_layout_iter_free(iter
); 
2963         *descent 
= *y 
- PANGO_PIXELS(baseline
); 
2965     if (externalLeading
) (*externalLeading
) = 0;  // ?? 
2967     g_object_unref (layout
); 
2970 void wxWindowGTK::GTKDisableFocusOutEvent() 
2972     g_signal_handlers_block_by_func( m_focusWidget
, 
2973                                 (gpointer
) gtk_window_focus_out_callback
, this); 
2976 void wxWindowGTK::GTKEnableFocusOutEvent() 
2978     g_signal_handlers_unblock_by_func( m_focusWidget
, 
2979                                 (gpointer
) gtk_window_focus_out_callback
, this); 
2982 bool wxWindowGTK::GTKHandleFocusIn() 
2984     // Disable default focus handling for custom windows since the default GTK+ 
2985     // handler issues a repaint 
2986     const bool retval 
= m_wxwindow 
? true : false; 
2989     // NB: if there's still unprocessed deferred focus-out event (see 
2990     //     GTKHandleFocusOut() for explanation), we need to process it first so 
2991     //     that the order of focus events -- focus-out first, then focus-in 
2992     //     elsewhere -- is preserved 
2993     if ( gs_deferredFocusOut 
) 
2995         if ( GTKNeedsToFilterSameWindowFocus() && 
2996              gs_deferredFocusOut 
== this ) 
2998             // GTK+ focus changed from this wxWindow back to itself, so don't 
2999             // emit any events at all 
3000             wxLogTrace(TRACE_FOCUS
, 
3001                        "filtered out spurious focus change within %s(%p, %s)", 
3002                        GetClassInfo()->GetClassName(), this, GetLabel()); 
3003             gs_deferredFocusOut 
= NULL
; 
3007         // otherwise we need to send focus-out first 
3008         wxASSERT_MSG ( gs_deferredFocusOut 
!= this, 
3009                        "GTKHandleFocusIn(GTKFocus_Normal) called even though focus changed back to itself - derived class should handle this" ); 
3010         GTKHandleDeferredFocusOut(); 
3014     wxLogTrace(TRACE_FOCUS
, 
3015                "handling focus_in event for %s(%p, %s)", 
3016                GetClassInfo()->GetClassName(), this, GetLabel()); 
3019         gtk_im_context_focus_in(m_imData
->context
); 
3021     gs_currentFocus 
= this; 
3022     gs_pendingFocus 
= NULL
; 
3025     // caret needs to be informed about focus change 
3026     wxCaret 
*caret 
= GetCaret(); 
3029         caret
->OnSetFocus(); 
3031 #endif // wxUSE_CARET 
3033     // Notify the parent keeping track of focus for the kbd navigation 
3034     // purposes that we got it. 
3035     wxChildFocusEvent 
eventChildFocus(static_cast<wxWindow
*>(this)); 
3036     GTKProcessEvent(eventChildFocus
); 
3038     wxFocusEvent 
eventFocus(wxEVT_SET_FOCUS
, GetId()); 
3039     eventFocus
.SetEventObject(this); 
3040     GTKProcessEvent(eventFocus
); 
3045 bool wxWindowGTK::GTKHandleFocusOut() 
3047     // Disable default focus handling for custom windows since the default GTK+ 
3048     // handler issues a repaint 
3049     const bool retval 
= m_wxwindow 
? true : false; 
3052     // NB: If a control is composed of several GtkWidgets and when focus 
3053     //     changes from one of them to another within the same wxWindow, we get 
3054     //     a focus-out event followed by focus-in for another GtkWidget owned 
3055     //     by the same wx control. We don't want to generate two spurious 
3056     //     wxEVT_SET_FOCUS events in this case, so we defer sending wx events 
3057     //     from GTKHandleFocusOut() until we know for sure it's not coming back 
3058     //     (i.e. in GTKHandleFocusIn() or at idle time). 
3059     if ( GTKNeedsToFilterSameWindowFocus() ) 
3061         wxASSERT_MSG( gs_deferredFocusOut 
== NULL
, 
3062                       "deferred focus out event already pending" ); 
3063         wxLogTrace(TRACE_FOCUS
, 
3064                    "deferring focus_out event for %s(%p, %s)", 
3065                    GetClassInfo()->GetClassName(), this, GetLabel()); 
3066         gs_deferredFocusOut 
= this; 
3070     GTKHandleFocusOutNoDeferring(); 
3075 void wxWindowGTK::GTKHandleFocusOutNoDeferring() 
3077     wxLogTrace(TRACE_FOCUS
, 
3078                "handling focus_out event for %s(%p, %s)", 
3079                GetClassInfo()->GetClassName(), this, GetLabel()); 
3082         gtk_im_context_focus_out(m_imData
->context
); 
3084     if ( gs_currentFocus 
!= this ) 
3086         // Something is terribly wrong, gs_currentFocus is out of sync with the 
3087         // real focus. We will reset it to NULL anyway, because after this 
3088         // focus-out event is handled, one of the following with happen: 
3090         // * either focus will go out of the app altogether, in which case 
3091         //   gs_currentFocus _should_ be NULL 
3093         // * or it goes to another control, in which case focus-in event will 
3094         //   follow immediately and it will set gs_currentFocus to the right 
3096         wxLogDebug("window %s(%p, %s) lost focus even though it didn't have it", 
3097                    GetClassInfo()->GetClassName(), this, GetLabel()); 
3099     gs_currentFocus 
= NULL
; 
3102     // caret needs to be informed about focus change 
3103     wxCaret 
*caret 
= GetCaret(); 
3106         caret
->OnKillFocus(); 
3108 #endif // wxUSE_CARET 
3110     wxFocusEvent 
event( wxEVT_KILL_FOCUS
, GetId() ); 
3111     event
.SetEventObject( this ); 
3112     GTKProcessEvent( event 
); 
3116 void wxWindowGTK::GTKHandleDeferredFocusOut() 
3118     // NB: See GTKHandleFocusOut() for explanation. This function is called 
3119     //     from either GTKHandleFocusIn() or OnInternalIdle() to process 
3121     if ( gs_deferredFocusOut 
) 
3123         wxWindowGTK 
*win 
= gs_deferredFocusOut
; 
3124         gs_deferredFocusOut 
= NULL
; 
3126         wxLogTrace(TRACE_FOCUS
, 
3127                    "processing deferred focus_out event for %s(%p, %s)", 
3128                    win
->GetClassInfo()->GetClassName(), win
, win
->GetLabel()); 
3130         win
->GTKHandleFocusOutNoDeferring(); 
3134 void wxWindowGTK::SetFocus() 
3136     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
3138     // Setting "physical" focus is not immediate in GTK+ and while 
3139     // gtk_widget_is_focus ("determines if the widget is the focus widget 
3140     // within its toplevel", i.e. returns true for one widget per TLW, not 
3141     // globally) returns true immediately after grabbing focus, 
3142     // GTK_WIDGET_HAS_FOCUS (which returns true only for the one widget that 
3143     // has focus at the moment) takes affect only after the window is shown 
3144     // (if it was hidden at the moment of the call) or at the next event loop 
3147     // Because we want to FindFocus() call immediately following 
3148     // foo->SetFocus() to return foo, we have to keep track of "pending" focus 
3150     gs_pendingFocus 
= this; 
3152     GtkWidget 
*widget 
= m_wxwindow 
? m_wxwindow 
: m_focusWidget
; 
3154     if ( GTK_IS_CONTAINER(widget
) && 
3155          !GTK_WIDGET_CAN_FOCUS(widget
) ) 
3157         wxLogTrace(TRACE_FOCUS
, 
3158                    wxT("Setting focus to a child of %s(%p, %s)"), 
3159                    GetClassInfo()->GetClassName(), this, GetLabel().c_str()); 
3160         gtk_widget_child_focus(widget
, GTK_DIR_TAB_FORWARD
); 
3164         wxLogTrace(TRACE_FOCUS
, 
3165                    wxT("Setting focus to %s(%p, %s)"), 
3166                    GetClassInfo()->GetClassName(), this, GetLabel().c_str()); 
3167         gtk_widget_grab_focus(widget
); 
3171 void wxWindowGTK::SetCanFocus(bool canFocus
) 
3174         GTK_WIDGET_SET_FLAGS(m_widget
, GTK_CAN_FOCUS
); 
3176         GTK_WIDGET_UNSET_FLAGS(m_widget
, GTK_CAN_FOCUS
); 
3178     if ( m_wxwindow 
&& (m_widget 
!= m_wxwindow
) ) 
3181             GTK_WIDGET_SET_FLAGS(m_wxwindow
, GTK_CAN_FOCUS
); 
3183             GTK_WIDGET_UNSET_FLAGS(m_wxwindow
, GTK_CAN_FOCUS
); 
3187 bool wxWindowGTK::Reparent( wxWindowBase 
*newParentBase 
) 
3189     wxCHECK_MSG( (m_widget 
!= NULL
), false, wxT("invalid window") ); 
3191     wxWindowGTK 
*oldParent 
= m_parent
, 
3192              *newParent 
= (wxWindowGTK 
*)newParentBase
; 
3194     wxASSERT( GTK_IS_WIDGET(m_widget
) ); 
3196     if ( !wxWindowBase::Reparent(newParent
) ) 
3199     wxASSERT( GTK_IS_WIDGET(m_widget
) ); 
3202         gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget 
); 
3204     wxASSERT( GTK_IS_WIDGET(m_widget
) ); 
3208         if (GTK_WIDGET_VISIBLE (newParent
->m_widget
)) 
3210             m_showOnIdle 
= true; 
3211             gtk_widget_hide( m_widget 
); 
3213         /* insert GTK representation */ 
3214         newParent
->AddChildGTK(this); 
3217     SetLayoutDirection(wxLayout_Default
); 
3222 void wxWindowGTK::DoAddChild(wxWindowGTK 
*child
) 
3224     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid window") ); 
3225     wxASSERT_MSG( (child 
!= NULL
), wxT("invalid child window") ); 
3230     /* insert GTK representation */ 
3234 void wxWindowGTK::AddChild(wxWindowBase 
*child
) 
3236     wxWindowBase::AddChild(child
); 
3237     m_dirtyTabOrder 
= true; 
3238     wxTheApp
->WakeUpIdle(); 
3241 void wxWindowGTK::RemoveChild(wxWindowBase 
*child
) 
3243     wxWindowBase::RemoveChild(child
); 
3244     m_dirtyTabOrder 
= true; 
3245     wxTheApp
->WakeUpIdle(); 
3249 wxLayoutDirection 
wxWindowGTK::GTKGetLayout(GtkWidget 
*widget
) 
3251     return gtk_widget_get_direction(widget
) == GTK_TEXT_DIR_RTL
 
3252                 ? wxLayout_RightToLeft
 
3253                 : wxLayout_LeftToRight
; 
3257 void wxWindowGTK::GTKSetLayout(GtkWidget 
*widget
, wxLayoutDirection dir
) 
3259     wxASSERT_MSG( dir 
!= wxLayout_Default
, wxT("invalid layout direction") ); 
3261     gtk_widget_set_direction(widget
, 
3262                              dir 
== wxLayout_RightToLeft 
? GTK_TEXT_DIR_RTL
 
3263                                                          : GTK_TEXT_DIR_LTR
); 
3266 wxLayoutDirection 
wxWindowGTK::GetLayoutDirection() const 
3268     return GTKGetLayout(m_widget
); 
3271 void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir
) 
3273     if ( dir 
== wxLayout_Default 
) 
3275         const wxWindow 
*const parent 
= GetParent(); 
3278             // inherit layout from parent. 
3279             dir 
= parent
->GetLayoutDirection(); 
3281         else // no parent, use global default layout 
3283             dir 
= wxTheApp
->GetLayoutDirection(); 
3287     if ( dir 
== wxLayout_Default 
) 
3290     GTKSetLayout(m_widget
, dir
); 
3292     if (m_wxwindow 
&& (m_wxwindow 
!= m_widget
)) 
3293         GTKSetLayout(m_wxwindow
, dir
); 
3297 wxWindowGTK::AdjustForLayoutDirection(wxCoord x
, 
3298                                       wxCoord 
WXUNUSED(width
), 
3299                                       wxCoord 
WXUNUSED(widthTotal
)) const 
3301     // We now mirror the coordinates of RTL windows in wxPizza 
3305 void wxWindowGTK::DoMoveInTabOrder(wxWindow 
*win
, WindowOrder move
) 
3307     wxWindowBase::DoMoveInTabOrder(win
, move
); 
3308     m_dirtyTabOrder 
= true; 
3309     wxTheApp
->WakeUpIdle(); 
3312 bool wxWindowGTK::DoNavigateIn(int flags
) 
3314     if ( flags 
& wxNavigationKeyEvent::WinChange 
) 
3316         wxFAIL_MSG( wxT("not implemented") ); 
3320     else // navigate inside the container 
3322         wxWindow 
*parent 
= wxGetTopLevelParent((wxWindow 
*)this); 
3323         wxCHECK_MSG( parent
, false, wxT("every window must have a TLW parent") ); 
3325         GtkDirectionType dir
; 
3326         dir 
= flags 
& wxNavigationKeyEvent::IsForward 
? GTK_DIR_TAB_FORWARD
 
3327                                                       : GTK_DIR_TAB_BACKWARD
; 
3330         g_signal_emit_by_name(parent
->m_widget
, "focus", dir
, &rc
); 
3336 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const 
3338     // none needed by default 
3342 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget
* WXUNUSED(w
)) 
3344     // nothing to do by default since none is needed 
3347 void wxWindowGTK::RealizeTabOrder() 
3351         if ( !m_children
.empty() ) 
3353             // we don't only construct the correct focus chain but also use 
3354             // this opportunity to update the mnemonic widgets for the widgets 
3357             GList 
*chain 
= NULL
; 
3358             wxWindowGTK
* mnemonicWindow 
= NULL
; 
3360             for ( wxWindowList::const_iterator i 
= m_children
.begin(); 
3361                   i 
!= m_children
.end(); 
3364                 wxWindowGTK 
*win 
= *i
; 
3366                 if ( mnemonicWindow 
) 
3368                     if ( win
->AcceptsFocusFromKeyboard() ) 
3370                         // wxComboBox et al. needs to focus on on a different 
3371                         // widget than m_widget, so if the main widget isn't 
3372                         // focusable try the connect widget 
3373                         GtkWidget
* w 
= win
->m_widget
; 
3374                         if ( !GTK_WIDGET_CAN_FOCUS(w
) ) 
3376                             w 
= win
->GetConnectWidget(); 
3377                             if ( !GTK_WIDGET_CAN_FOCUS(w
) ) 
3383                             mnemonicWindow
->GTKWidgetDoSetMnemonic(w
); 
3384                             mnemonicWindow 
= NULL
; 
3388                 else if ( win
->GTKWidgetNeedsMnemonic() ) 
3390                     mnemonicWindow 
= win
; 
3393                 chain 
= g_list_prepend(chain
, win
->m_widget
); 
3396             chain 
= g_list_reverse(chain
); 
3398             gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow
), chain
); 
3403             gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow
)); 
3408 void wxWindowGTK::Raise() 
3410     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
3412     if (m_wxwindow 
&& m_wxwindow
->window
) 
3414         gdk_window_raise( m_wxwindow
->window 
); 
3416     else if (m_widget
->window
) 
3418         gdk_window_raise( m_widget
->window 
); 
3422 void wxWindowGTK::Lower() 
3424     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
3426     if (m_wxwindow 
&& m_wxwindow
->window
) 
3428         gdk_window_lower( m_wxwindow
->window 
); 
3430     else if (m_widget
->window
) 
3432         gdk_window_lower( m_widget
->window 
); 
3436 bool wxWindowGTK::SetCursor( const wxCursor 
&cursor 
) 
3438     if ( !wxWindowBase::SetCursor(cursor
.Ok() ? cursor 
: *wxSTANDARD_CURSOR
) ) 
3446 void wxWindowGTK::GTKUpdateCursor() 
3448     wxCursor 
cursor(g_globalCursor
.Ok() ? g_globalCursor 
: GetCursor()); 
3451         wxArrayGdkWindows windowsThis
; 
3452         GdkWindow 
* const winThis 
= GTKGetWindow(windowsThis
); 
3455             gdk_window_set_cursor(winThis
, cursor
.GetCursor()); 
3459             const size_t count 
= windowsThis
.size(); 
3460             for ( size_t n 
= 0; n 
< count
; n
++ ) 
3462                 GdkWindow 
*win 
= windowsThis
[n
]; 
3465                     wxFAIL_MSG(wxT("NULL window returned by GTKGetWindow()?")); 
3469                 gdk_window_set_cursor(win
, cursor
.GetCursor()); 
3475 void wxWindowGTK::WarpPointer( int x
, int y 
) 
3477     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
3479     // We provide this function ourselves as it is 
3480     // missing in GDK (top of this file). 
3482     GdkWindow 
*window 
= NULL
; 
3484         window 
= m_wxwindow
->window
; 
3486         window 
= GetConnectWidget()->window
; 
3489         gdk_window_warp_pointer( window
, x
, y 
); 
3492 wxWindowGTK::ScrollDir 
wxWindowGTK::ScrollDirFromRange(GtkRange 
*range
) const 
3494     // find the scrollbar which generated the event 
3495     for ( int dir 
= 0; dir 
< ScrollDir_Max
; dir
++ ) 
3497         if ( range 
== m_scrollBar
[dir
] ) 
3498             return (ScrollDir
)dir
; 
3501     wxFAIL_MSG( wxT("event from unknown scrollbar received") ); 
3503     return ScrollDir_Max
; 
3506 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir
, ScrollUnit unit
, int units
) 
3508     bool changed 
= false; 
3509     GtkRange
* range 
= m_scrollBar
[dir
]; 
3510     if ( range 
&& units 
) 
3512         GtkAdjustment
* adj 
= range
->adjustment
; 
3513         gdouble inc 
= unit 
== ScrollUnit_Line 
? adj
->step_increment
 
3514                                               : adj
->page_increment
; 
3516         const int posOld 
= int(adj
->value 
+ 0.5); 
3517         gtk_range_set_value(range
, posOld 
+ units
*inc
); 
3519         changed 
= int(adj
->value 
+ 0.5) != posOld
; 
3525 bool wxWindowGTK::ScrollLines(int lines
) 
3527     return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Line
, lines
); 
3530 bool wxWindowGTK::ScrollPages(int pages
) 
3532     return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Page
, pages
); 
3535 void wxWindowGTK::Refresh(bool WXUNUSED(eraseBackground
), 
3540         // it is valid to call Refresh() for a window which hasn't been created 
3541         // yet, it simply doesn't do anything in this case 
3548             gtk_widget_queue_draw_area( m_widget
, rect
->x
, rect
->y
, rect
->width
, rect
->height 
); 
3550             gtk_widget_queue_draw( m_widget 
); 
3554         // Just return if the widget or one of its ancestors isn't mapped 
3556         for (w 
= m_wxwindow
; w 
!= NULL
; w 
= w
->parent
) 
3557             if (!GTK_WIDGET_MAPPED (w
)) 
3563             if (GetLayoutDirection() == wxLayout_RightToLeft
) 
3564                 x 
= GetClientSize().x 
- x 
- rect
->width
; 
3568             r
.width 
= rect
->width
; 
3569             r
.height 
= rect
->height
; 
3570             gdk_window_invalidate_rect( m_wxwindow
->window
, &r
, TRUE 
); 
3573             gdk_window_invalidate_rect( m_wxwindow
->window
, NULL
, TRUE 
); 
3577 void wxWindowGTK::Update() 
3579     if (m_widget 
&& m_widget
->window
) 
3581         GdkDisplay
* display 
= gtk_widget_get_display(m_widget
); 
3582         // Flush everything out to the server, and wait for it to finish. 
3583         // This ensures nothing will overwrite the drawing we are about to do. 
3584         gdk_display_sync(display
); 
3586         gdk_window_process_updates(m_widget
->window
, TRUE
); 
3588         // Flush again, but no need to wait for it to finish 
3589         gdk_display_flush(display
); 
3593 bool wxWindowGTK::DoIsExposed( int x
, int y 
) const 
3595     return m_updateRegion
.Contains(x
, y
) != wxOutRegion
; 
3598 bool wxWindowGTK::DoIsExposed( int x
, int y
, int w
, int h 
) const 
3600     if (GetLayoutDirection() == wxLayout_RightToLeft
) 
3601         return m_updateRegion
.Contains(x
-w
, y
, w
, h
) != wxOutRegion
; 
3603         return m_updateRegion
.Contains(x
, y
, w
, h
) != wxOutRegion
; 
3606 void wxWindowGTK::GtkSendPaintEvents() 
3610         m_updateRegion
.Clear(); 
3614     // Clip to paint region in wxClientDC 
3615     m_clipPaintRegion 
= true; 
3617     m_nativeUpdateRegion 
= m_updateRegion
; 
3619     if (GetLayoutDirection() == wxLayout_RightToLeft
) 
3621         // Transform m_updateRegion under RTL 
3622         m_updateRegion
.Clear(); 
3625         gdk_drawable_get_size(m_wxwindow
->window
, &width
, NULL
); 
3627         wxRegionIterator 
upd( m_nativeUpdateRegion 
); 
3631             rect
.x 
= upd
.GetX(); 
3632             rect
.y 
= upd
.GetY(); 
3633             rect
.width 
= upd
.GetWidth(); 
3634             rect
.height 
= upd
.GetHeight(); 
3636             rect
.x 
= width 
- rect
.x 
- rect
.width
; 
3637             m_updateRegion
.Union( rect 
); 
3643     switch ( GetBackgroundStyle() ) 
3645         case wxBG_STYLE_ERASE
: 
3647                 wxWindowDC 
dc( (wxWindow
*)this ); 
3648                 dc
.SetDeviceClippingRegion( m_updateRegion 
); 
3650                 // Work around gtk-qt <= 0.60 bug whereby the window colour 
3654                             GetOptionInt("gtk.window.force-background-colour") ) 
3656                     dc
.SetBackground(GetBackgroundColour()); 
3660                 wxEraseEvent 
erase_event( GetId(), &dc 
); 
3661                 erase_event
.SetEventObject( this ); 
3663                 if ( HandleWindowEvent(erase_event
) ) 
3665                     // background erased, don't do it again 
3671         case wxBG_STYLE_SYSTEM
: 
3672             if ( GetThemeEnabled() ) 
3674                 // find ancestor from which to steal background 
3675                 wxWindow 
*parent 
= wxGetTopLevelParent((wxWindow 
*)this); 
3677                     parent 
= (wxWindow
*)this; 
3679                 if (GTK_WIDGET_MAPPED(parent
->m_widget
)) 
3681                     wxRegionIterator 
upd( m_nativeUpdateRegion 
); 
3685                         rect
.x 
= upd
.GetX(); 
3686                         rect
.y 
= upd
.GetY(); 
3687                         rect
.width 
= upd
.GetWidth(); 
3688                         rect
.height 
= upd
.GetHeight(); 
3690                         gtk_paint_flat_box( parent
->m_widget
->style
, 
3692                                     (GtkStateType
)GTK_WIDGET_STATE(m_wxwindow
), 
3705         case wxBG_STYLE_PAINT
: 
3706             // nothing to do: window will be painted over in EVT_PAINT 
3710             wxFAIL_MSG( "unsupported background style" ); 
3713     wxNcPaintEvent 
nc_paint_event( GetId() ); 
3714     nc_paint_event
.SetEventObject( this ); 
3715     HandleWindowEvent( nc_paint_event 
); 
3717     wxPaintEvent 
paint_event( GetId() ); 
3718     paint_event
.SetEventObject( this ); 
3719     HandleWindowEvent( paint_event 
); 
3721     m_clipPaintRegion 
= false; 
3723     m_updateRegion
.Clear(); 
3724     m_nativeUpdateRegion
.Clear(); 
3727 void wxWindowGTK::SetDoubleBuffered( bool on 
) 
3729     wxCHECK_RET( (m_widget 
!= NULL
), wxT("invalid window") ); 
3732         gtk_widget_set_double_buffered( m_wxwindow
, on 
); 
3735 bool wxWindowGTK::IsDoubleBuffered() const 
3737     return GTK_WIDGET_DOUBLE_BUFFERED( m_wxwindow 
); 
3740 void wxWindowGTK::ClearBackground() 
3742     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
3746 void wxWindowGTK::DoSetToolTip( wxToolTip 
*tip 
) 
3748     wxWindowBase::DoSetToolTip(tip
); 
3752         m_tooltip
->GTKApply( (wxWindow 
*)this ); 
3756         GtkWidget 
*w 
= GetConnectWidget(); 
3757         wxToolTip::GTKApply(w
, NULL
); 
3758 #if GTK_CHECK_VERSION(2, 12, 0) 
3759         // Just applying NULL doesn't work on 2.12.0, so also use 
3760         // gtk_widget_set_has_tooltip. It is part of the new GtkTooltip API 
3761         // but seems also to work with the old GtkTooltips. 
3762         if (gtk_check_version(2, 12, 0) == NULL
) 
3763             gtk_widget_set_has_tooltip(w
, FALSE
); 
3768 void wxWindowGTK::GTKApplyToolTip( GtkTooltips 
*tips
, const gchar 
*tip 
) 
3770     gtk_tooltips_set_tip(tips
, GetConnectWidget(), tip
, NULL
); 
3772 #endif // wxUSE_TOOLTIPS 
3774 bool wxWindowGTK::SetBackgroundColour( const wxColour 
&colour 
) 
3776     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid window") ); 
3778     if (!wxWindowBase::SetBackgroundColour(colour
)) 
3783         // We need the pixel value e.g. for background clearing. 
3784         m_backgroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
)); 
3787     // apply style change (forceStyle=true so that new style is applied 
3788     // even if the bg colour changed from valid to wxNullColour) 
3789     GTKApplyWidgetStyle(true); 
3794 bool wxWindowGTK::SetForegroundColour( const wxColour 
&colour 
) 
3796     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid window") ); 
3798     if (!wxWindowBase::SetForegroundColour(colour
)) 
3805         // We need the pixel value e.g. for background clearing. 
3806         m_foregroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
)); 
3809     // apply style change (forceStyle=true so that new style is applied 
3810     // even if the bg colour changed from valid to wxNullColour): 
3811     GTKApplyWidgetStyle(true); 
3816 PangoContext 
*wxWindowGTK::GTKGetPangoDefaultContext() 
3818     return gtk_widget_get_pango_context( m_widget 
); 
3821 GtkRcStyle 
*wxWindowGTK::GTKCreateWidgetStyle(bool forceStyle
) 
3823     // do we need to apply any changes at all? 
3826          !m_foregroundColour
.Ok() && !m_backgroundColour
.Ok() ) 
3831     GtkRcStyle 
*style 
= gtk_rc_style_new(); 
3836             pango_font_description_copy( m_font
.GetNativeFontInfo()->description 
); 
3839     int flagsNormal 
= 0, 
3842         flagsInsensitive 
= 0; 
3844     if ( m_foregroundColour
.Ok() ) 
3846         const GdkColor 
*fg 
= m_foregroundColour
.GetColor(); 
3848         style
->fg
[GTK_STATE_NORMAL
] = 
3849         style
->text
[GTK_STATE_NORMAL
] = *fg
; 
3850         flagsNormal 
|= GTK_RC_FG 
| GTK_RC_TEXT
; 
3852         style
->fg
[GTK_STATE_PRELIGHT
] = 
3853         style
->text
[GTK_STATE_PRELIGHT
] = *fg
; 
3854         flagsPrelight 
|= GTK_RC_FG 
| GTK_RC_TEXT
; 
3856         style
->fg
[GTK_STATE_ACTIVE
] = 
3857         style
->text
[GTK_STATE_ACTIVE
] = *fg
; 
3858         flagsActive 
|= GTK_RC_FG 
| GTK_RC_TEXT
; 
3861     if ( m_backgroundColour
.Ok() ) 
3863         const GdkColor 
*bg 
= m_backgroundColour
.GetColor(); 
3865         style
->bg
[GTK_STATE_NORMAL
] = 
3866         style
->base
[GTK_STATE_NORMAL
] = *bg
; 
3867         flagsNormal 
|= GTK_RC_BG 
| GTK_RC_BASE
; 
3869         style
->bg
[GTK_STATE_PRELIGHT
] = 
3870         style
->base
[GTK_STATE_PRELIGHT
] = *bg
; 
3871         flagsPrelight 
|= GTK_RC_BG 
| GTK_RC_BASE
; 
3873         style
->bg
[GTK_STATE_ACTIVE
] = 
3874         style
->base
[GTK_STATE_ACTIVE
] = *bg
; 
3875         flagsActive 
|= GTK_RC_BG 
| GTK_RC_BASE
; 
3877         style
->bg
[GTK_STATE_INSENSITIVE
] = 
3878         style
->base
[GTK_STATE_INSENSITIVE
] = *bg
; 
3879         flagsInsensitive 
|= GTK_RC_BG 
| GTK_RC_BASE
; 
3882     style
->color_flags
[GTK_STATE_NORMAL
] = (GtkRcFlags
)flagsNormal
; 
3883     style
->color_flags
[GTK_STATE_PRELIGHT
] = (GtkRcFlags
)flagsPrelight
; 
3884     style
->color_flags
[GTK_STATE_ACTIVE
] = (GtkRcFlags
)flagsActive
; 
3885     style
->color_flags
[GTK_STATE_INSENSITIVE
] = (GtkRcFlags
)flagsInsensitive
; 
3890 void wxWindowGTK::GTKApplyWidgetStyle(bool forceStyle
) 
3892     GtkRcStyle 
*style 
= GTKCreateWidgetStyle(forceStyle
); 
3895         DoApplyWidgetStyle(style
); 
3896         gtk_rc_style_unref(style
); 
3899     // Style change may affect GTK+'s size calculation: 
3900     InvalidateBestSize(); 
3903 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle 
*style
) 
3905     wxSuspendStyleEvents 
s(static_cast<wxWindow
*>(this)); 
3908         gtk_widget_modify_style(m_wxwindow
, style
); 
3910         gtk_widget_modify_style(m_widget
, style
); 
3913 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style
) 
3915     wxWindowBase::SetBackgroundStyle(style
); 
3917     if ( style 
== wxBG_STYLE_PAINT 
) 
3922             window 
= m_wxwindow
->window
; 
3926             GtkWidget 
* const w 
= GetConnectWidget(); 
3927             window 
= w 
? w
->window 
: NULL
; 
3932             // Make sure GDK/X11 doesn't refresh the window 
3934             gdk_window_set_back_pixmap( window
, None
, False 
); 
3936             Display
* display 
= GDK_WINDOW_DISPLAY(window
); 
3939             m_needsStyleChange 
= false; 
3941         else // window not realized yet 
3943             // Do in OnIdle, because the window is not yet available 
3944             m_needsStyleChange 
= true; 
3947         // Don't apply widget style, or we get a grey background 
3951         // apply style change (forceStyle=true so that new style is applied 
3952         // even if the bg colour changed from valid to wxNullColour): 
3953         GTKApplyWidgetStyle(true); 
3959 // ---------------------------------------------------------------------------- 
3960 // Pop-up menu stuff 
3961 // ---------------------------------------------------------------------------- 
3963 #if wxUSE_MENUS_NATIVE 
3965 static void SetInvokingWindow( wxMenu 
*menu
, wxWindow
* win 
) 
3967     menu
->SetInvokingWindow( win 
); 
3969     wxMenuItemList::compatibility_iterator node 
= menu
->GetMenuItems().GetFirst(); 
3972         wxMenuItem 
*menuitem 
= node
->GetData(); 
3973         if (menuitem
->IsSubMenu()) 
3975             SetInvokingWindow( menuitem
->GetSubMenu(), win 
); 
3978         node 
= node
->GetNext(); 
3984 void wxPopupMenuPositionCallback( GtkMenu 
*menu
, 
3986                                   gboolean 
* WXUNUSED(whatever
), 
3987                                   gpointer user_data 
) 
3989     // ensure that the menu appears entirely on screen 
3991     gtk_widget_get_child_requisition(GTK_WIDGET(menu
), &req
); 
3993     wxSize sizeScreen 
= wxGetDisplaySize(); 
3994     wxPoint 
*pos 
= (wxPoint
*)user_data
; 
3996     gint xmax 
= sizeScreen
.x 
- req
.width
, 
3997          ymax 
= sizeScreen
.y 
- req
.height
; 
3999     *x 
= pos
->x 
< xmax 
? pos
->x 
: xmax
; 
4000     *y 
= pos
->y 
< ymax 
? pos
->y 
: ymax
; 
4004 bool wxWindowGTK::DoPopupMenu( wxMenu 
*menu
, int x
, int y 
) 
4006     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid window") ); 
4008     wxCHECK_MSG( menu 
!= NULL
, false, wxT("invalid popup-menu") ); 
4010     SetInvokingWindow( menu
, this ); 
4016     GtkMenuPositionFunc posfunc
; 
4017     if ( x 
== -1 && y 
== -1 ) 
4019         // use GTK's default positioning algorithm 
4025         pos 
= ClientToScreen(wxPoint(x
, y
)); 
4027         posfunc 
= wxPopupMenuPositionCallback
; 
4030     menu
->m_popupShown 
= true; 
4032                   GTK_MENU(menu
->m_menu
), 
4033                   NULL
,           // parent menu shell 
4034                   NULL
,           // parent menu item 
4035                   posfunc
,                      // function to position it 
4036                   userdata
,                     // client data 
4037                   0,                            // button used to activate it 
4038                   gtk_get_current_event_time() 
4041     while (menu
->m_popupShown
) 
4043         gtk_main_iteration(); 
4049 #endif // wxUSE_MENUS_NATIVE 
4051 #if wxUSE_DRAG_AND_DROP 
4053 void wxWindowGTK::SetDropTarget( wxDropTarget 
*dropTarget 
) 
4055     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
4057     GtkWidget 
*dnd_widget 
= GetConnectWidget(); 
4059     if (m_dropTarget
) m_dropTarget
->GtkUnregisterWidget( dnd_widget 
); 
4061     if (m_dropTarget
) delete m_dropTarget
; 
4062     m_dropTarget 
= dropTarget
; 
4064     if (m_dropTarget
) m_dropTarget
->GtkRegisterWidget( dnd_widget 
); 
4067 #endif // wxUSE_DRAG_AND_DROP 
4069 GtkWidget
* wxWindowGTK::GetConnectWidget() 
4071     GtkWidget 
*connect_widget 
= m_widget
; 
4072     if (m_wxwindow
) connect_widget 
= m_wxwindow
; 
4074     return connect_widget
; 
4077 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow 
*window
) const 
4079     wxArrayGdkWindows windowsThis
; 
4080     GdkWindow 
* const winThis 
= GTKGetWindow(windowsThis
); 
4082     return winThis 
? window 
== winThis
 
4083                    : windowsThis
.Index(window
) != wxNOT_FOUND
; 
4086 GdkWindow 
*wxWindowGTK::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const 
4088     return m_wxwindow 
? m_wxwindow
->window 
: m_widget
->window
; 
4091 bool wxWindowGTK::SetFont( const wxFont 
&font 
) 
4093     wxCHECK_MSG( m_widget 
!= NULL
, false, wxT("invalid window") ); 
4095     if (!wxWindowBase::SetFont(font
)) 
4098     // apply style change (forceStyle=true so that new style is applied 
4099     // even if the font changed from valid to wxNullFont): 
4100     GTKApplyWidgetStyle(true); 
4105 void wxWindowGTK::DoCaptureMouse() 
4107     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
4109     GdkWindow 
*window 
= NULL
; 
4111         window 
= m_wxwindow
->window
; 
4113         window 
= GetConnectWidget()->window
; 
4115     wxCHECK_RET( window
, wxT("CaptureMouse() failed") ); 
4117     const wxCursor
* cursor 
= &m_cursor
; 
4119         cursor 
= wxSTANDARD_CURSOR
; 
4121     gdk_pointer_grab( window
, FALSE
, 
4123                          (GDK_BUTTON_PRESS_MASK 
| 
4124                           GDK_BUTTON_RELEASE_MASK 
| 
4125                           GDK_POINTER_MOTION_HINT_MASK 
| 
4126                           GDK_POINTER_MOTION_MASK
), 
4128                       cursor
->GetCursor(), 
4129                       (guint32
)GDK_CURRENT_TIME 
); 
4130     g_captureWindow 
= this; 
4131     g_captureWindowHasMouse 
= true; 
4134 void wxWindowGTK::DoReleaseMouse() 
4136     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
4138     wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") ); 
4140     g_captureWindow 
= NULL
; 
4142     GdkWindow 
*window 
= NULL
; 
4144         window 
= m_wxwindow
->window
; 
4146         window 
= GetConnectWidget()->window
; 
4151     gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME 
); 
4154 void wxWindowGTK::GTKReleaseMouseAndNotify() 
4157     wxMouseCaptureLostEvent 
evt(GetId()); 
4158     evt
.SetEventObject( this ); 
4159     HandleWindowEvent( evt 
); 
4163 wxWindow 
*wxWindowBase::GetCapture() 
4165     return (wxWindow 
*)g_captureWindow
; 
4168 bool wxWindowGTK::IsRetained() const 
4173 void wxWindowGTK::SetScrollbar(int orient
, 
4177                                bool WXUNUSED(update
)) 
4179     const int dir 
= ScrollDirFromOrient(orient
); 
4180     GtkRange
* const sb 
= m_scrollBar
[dir
]; 
4181     wxCHECK_RET( sb
, wxT("this window is not scrollable") ); 
4185         // GtkRange requires upper > lower 
4190     GtkAdjustment 
* const adj 
= sb
->adjustment
; 
4191     adj
->step_increment 
= 1; 
4192     adj
->page_increment 
= 
4193     adj
->page_size 
= thumbVisible
; 
4196     g_signal_handlers_block_by_func( 
4197         sb
, (void*)gtk_scrollbar_value_changed
, this); 
4199     gtk_range_set_range(sb
, 0, range
); 
4200     m_scrollPos
[dir
] = sb
->adjustment
->value
; 
4202     g_signal_handlers_unblock_by_func( 
4203         sb
, (void*)gtk_scrollbar_value_changed
, this); 
4206 void wxWindowGTK::SetScrollPos(int orient
, int pos
, bool WXUNUSED(refresh
)) 
4208     const int dir 
= ScrollDirFromOrient(orient
); 
4209     GtkRange 
* const sb 
= m_scrollBar
[dir
]; 
4210     wxCHECK_RET( sb
, wxT("this window is not scrollable") ); 
4212     // This check is more than an optimization. Without it, the slider 
4213     //   will not move smoothly while tracking when using wxScrollHelper. 
4214     if (GetScrollPos(orient
) != pos
) 
4216         g_signal_handlers_block_by_func( 
4217             sb
, (void*)gtk_scrollbar_value_changed
, this); 
4219         gtk_range_set_value(sb
, pos
); 
4220         m_scrollPos
[dir
] = sb
->adjustment
->value
; 
4222         g_signal_handlers_unblock_by_func( 
4223             sb
, (void*)gtk_scrollbar_value_changed
, this); 
4227 int wxWindowGTK::GetScrollThumb(int orient
) const 
4229     GtkRange 
* const sb 
= m_scrollBar
[ScrollDirFromOrient(orient
)]; 
4230     wxCHECK_MSG( sb
, 0, wxT("this window is not scrollable") ); 
4232     return wxRound(sb
->adjustment
->page_size
); 
4235 int wxWindowGTK::GetScrollPos( int orient 
) const 
4237     GtkRange 
* const sb 
= m_scrollBar
[ScrollDirFromOrient(orient
)]; 
4238     wxCHECK_MSG( sb
, 0, wxT("this window is not scrollable") ); 
4240     return wxRound(sb
->adjustment
->value
); 
4243 int wxWindowGTK::GetScrollRange( int orient 
) const 
4245     GtkRange 
* const sb 
= m_scrollBar
[ScrollDirFromOrient(orient
)]; 
4246     wxCHECK_MSG( sb
, 0, wxT("this window is not scrollable") ); 
4248     return wxRound(sb
->adjustment
->upper
); 
4251 // Determine if increment is the same as +/-x, allowing for some small 
4252 //   difference due to possible inexactness in floating point arithmetic 
4253 static inline bool IsScrollIncrement(double increment
, double x
) 
4255     wxASSERT(increment 
> 0); 
4256     const double tolerance 
= 1.0 / 1024; 
4257     return fabs(increment 
- fabs(x
)) < tolerance
; 
4260 wxEventType 
wxWindowGTK::GTKGetScrollEventType(GtkRange
* range
) 
4262     wxASSERT(range 
== m_scrollBar
[0] || range 
== m_scrollBar
[1]); 
4264     const int barIndex 
= range 
== m_scrollBar
[1]; 
4265     GtkAdjustment
* adj 
= range
->adjustment
; 
4267     const int value 
= wxRound(adj
->value
); 
4269     // save previous position 
4270     const double oldPos 
= m_scrollPos
[barIndex
]; 
4271     // update current position 
4272     m_scrollPos
[barIndex
] = adj
->value
; 
4273     // If event should be ignored, or integral position has not changed 
4274     if (!m_hasVMT 
|| g_blockEventsOnDrag 
|| value 
== wxRound(oldPos
)) 
4279     wxEventType eventType 
= wxEVT_SCROLL_THUMBTRACK
; 
4282         // Difference from last change event 
4283         const double diff 
= adj
->value 
- oldPos
; 
4284         const bool isDown 
= diff 
> 0; 
4286         if (IsScrollIncrement(adj
->step_increment
, diff
)) 
4288             eventType 
= isDown 
? wxEVT_SCROLL_LINEDOWN 
: wxEVT_SCROLL_LINEUP
; 
4290         else if (IsScrollIncrement(adj
->page_increment
, diff
)) 
4292             eventType 
= isDown 
? wxEVT_SCROLL_PAGEDOWN 
: wxEVT_SCROLL_PAGEUP
; 
4294         else if (m_mouseButtonDown
) 
4296             // Assume track event 
4297             m_isScrolling 
= true; 
4303 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) ) 
4305     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid window") ); 
4307     wxCHECK_RET( m_wxwindow 
!= NULL
, wxT("window needs client area for scrolling") ); 
4309     // No scrolling requested. 
4310     if ((dx 
== 0) && (dy 
== 0)) return; 
4312     m_clipPaintRegion 
= true; 
4314     WX_PIZZA(m_wxwindow
)->scroll(dx
, dy
); 
4316     m_clipPaintRegion 
= false; 
4319     bool restoreCaret 
= (GetCaret() != NULL 
&& GetCaret()->IsVisible()); 
4322         wxRect 
caretRect(GetCaret()->GetPosition(), GetCaret()->GetSize()); 
4324             caretRect
.width 
+= dx
; 
4327             caretRect
.x 
+= dx
; caretRect
.width 
-= dx
; 
4330             caretRect
.height 
+= dy
; 
4333             caretRect
.y 
+= dy
; caretRect
.height 
-= dy
; 
4336         RefreshRect(caretRect
); 
4338 #endif // wxUSE_CARET 
4341 void wxWindowGTK::GTKScrolledWindowSetBorder(GtkWidget
* w
, int wxstyle
) 
4343     //RN: Note that static controls usually have no border on gtk, so maybe 
4344     //it makes sense to treat that as simply no border at the wx level 
4346     if (!(wxstyle 
& wxNO_BORDER
) && !(wxstyle 
& wxBORDER_STATIC
)) 
4348         GtkShadowType gtkstyle
; 
4350         if(wxstyle 
& wxBORDER_RAISED
) 
4351             gtkstyle 
= GTK_SHADOW_OUT
; 
4352         else if ((wxstyle 
& wxBORDER_SUNKEN
) || (wxstyle 
& wxBORDER_THEME
)) 
4353             gtkstyle 
= GTK_SHADOW_IN
; 
4356         else if (wxstyle 
& wxBORDER_DOUBLE
) 
4357             gtkstyle 
= GTK_SHADOW_ETCHED_IN
; 
4360             gtkstyle 
= GTK_SHADOW_IN
; 
4362         gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w
), 
4367 void wxWindowGTK::SetWindowStyleFlag( long style 
) 
4369     // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already 
4370     wxWindowBase::SetWindowStyleFlag(style
); 
4373 // Find the wxWindow at the current mouse position, also returning the mouse 
4375 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
) 
4377     pt 
= wxGetMousePosition(); 
4378     wxWindow
* found 
= wxFindWindowAtPoint(pt
); 
4382 // Get the current mouse position. 
4383 wxPoint 
wxGetMousePosition() 
4385   /* This crashes when used within wxHelpContext, 
4386      so we have to use the X-specific implementation below. 
4388     GdkModifierType *mask; 
4389     (void) gdk_window_get_pointer(NULL, &x, &y, mask); 
4391     return wxPoint(x, y); 
4395     GdkWindow
* windowAtPtr 
= gdk_window_at_pointer(& x
, & y
); 
4397     Display 
*display 
= windowAtPtr 
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY(); 
4398     Window rootWindow 
= RootWindowOfScreen (DefaultScreenOfDisplay(display
)); 
4399     Window rootReturn
, childReturn
; 
4400     int rootX
, rootY
, winX
, winY
; 
4401     unsigned int maskReturn
; 
4403     XQueryPointer (display
, 
4407                    &rootX
, &rootY
, &winX
, &winY
, &maskReturn
); 
4408     return wxPoint(rootX
, rootY
); 
4412 GdkWindow
* wxWindowGTK::GTKGetDrawingWindow() const 
4414     GdkWindow
* window 
= NULL
; 
4416         window 
= m_wxwindow
->window
; 
4420 // ---------------------------------------------------------------------------- 
4422 // ---------------------------------------------------------------------------- 
4427 // this is called if we attempted to freeze unrealized widget when it finally 
4428 // is realized (and so can be frozen): 
4429 static void wx_frozen_widget_realize(GtkWidget
* w
, void* WXUNUSED(data
)) 
4431     wxASSERT( w 
&& !GTK_WIDGET_NO_WINDOW(w
) ); 
4432     wxASSERT( GTK_WIDGET_REALIZED(w
) ); 
4434     g_signal_handlers_disconnect_by_func
 
4437         (void*)wx_frozen_widget_realize
, 
4441     gdk_window_freeze_updates(w
->window
); 
4446 void wxWindowGTK::GTKFreezeWidget(GtkWidget 
*w
) 
4448     if ( !w 
|| GTK_WIDGET_NO_WINDOW(w
) ) 
4449         return; // window-less widget, cannot be frozen 
4451     if ( !GTK_WIDGET_REALIZED(w
) ) 
4453         // we can't thaw unrealized widgets because they don't have GdkWindow, 
4454         // so set it up to be done immediately after realization: 
4455         g_signal_connect_after
 
4459             G_CALLBACK(wx_frozen_widget_realize
), 
4465     gdk_window_freeze_updates(w
->window
); 
4468 void wxWindowGTK::GTKThawWidget(GtkWidget 
*w
) 
4470     if ( !w 
|| GTK_WIDGET_NO_WINDOW(w
) ) 
4471         return; // window-less widget, cannot be frozen 
4473     if ( !GTK_WIDGET_REALIZED(w
) ) 
4475         // the widget wasn't realized yet, no need to thaw 
4476         g_signal_handlers_disconnect_by_func
 
4479             (void*)wx_frozen_widget_realize
, 
4485     gdk_window_thaw_updates(w
->window
); 
4488 void wxWindowGTK::DoFreeze() 
4490     GTKFreezeWidget(m_widget
); 
4491     if ( m_wxwindow 
&& m_widget 
!= m_wxwindow 
) 
4492         GTKFreezeWidget(m_wxwindow
); 
4495 void wxWindowGTK::DoThaw() 
4497     GTKThawWidget(m_widget
); 
4498     if ( m_wxwindow 
&& m_widget 
!= m_wxwindow 
) 
4499         GTKThawWidget(m_wxwindow
);