1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/window.cpp
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 _T("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 _T("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_scanCode
= gdk_event
->keyval
;
687 event
.m_rawCode
= (wxUint32
) gdk_event
->keyval
;
688 event
.m_rawFlags
= 0;
690 event
.m_uniChar
= gdk_keyval_to_unicode(gdk_event
->keyval
);
692 wxGetMousePosition( &x
, &y
);
693 win
->ScreenToClient( &x
, &y
);
696 event
.SetEventObject( win
);
701 wxTranslateGTKKeyEventToWx(wxKeyEvent
& event
,
703 GdkEventKey
*gdk_event
)
705 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
706 // but only event->keyval which is quite useless to us, so remember
707 // the last character from GDK_KEY_PRESS and reuse it as last resort
709 // NB: should be MT-safe as we're always called from the main thread only
714 } s_lastKeyPress
= { 0, 0 };
716 KeySym keysym
= gdk_event
->keyval
;
718 wxLogTrace(TRACE_KEYS
, _T("Key %s event: keysym = %ld"),
719 event
.GetEventType() == wxEVT_KEY_UP
? _T("release")
723 long key_code
= wxTranslateKeySymToWXKey(keysym
, false /* !isChar */);
727 // do we have the translation or is it a plain ASCII character?
728 if ( (gdk_event
->length
== 1) || wxIsAsciiKeysym(keysym
) )
730 // we should use keysym if it is ASCII as X does some translations
731 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
732 // which we don't want here (but which we do use for OnChar())
733 if ( !wxIsAsciiKeysym(keysym
) )
735 keysym
= (KeySym
)gdk_event
->string
[0];
738 // we want to always get the same key code when the same key is
739 // pressed regardless of the state of the modifiers, i.e. on a
740 // standard US keyboard pressing '5' or '%' ('5' key with
741 // Shift) should result in the same key code in OnKeyDown():
742 // '5' (although OnChar() will get either '5' or '%').
744 // to do it we first translate keysym to keycode (== scan code)
745 // and then back but always using the lower register
746 Display
*dpy
= (Display
*)wxGetDisplay();
747 KeyCode keycode
= XKeysymToKeycode(dpy
, keysym
);
749 wxLogTrace(TRACE_KEYS
, _T("\t-> keycode %d"), keycode
);
751 KeySym keysymNormalized
= XKeycodeToKeysym(dpy
, keycode
, 0);
753 // use the normalized, i.e. lower register, keysym if we've
755 key_code
= keysymNormalized
? keysymNormalized
: keysym
;
757 // as explained above, we want to have lower register key codes
758 // normally but for the letter keys we want to have the upper ones
760 // NB: don't use XConvertCase() here, we want to do it for letters
762 key_code
= toupper(key_code
);
764 else // non ASCII key, what to do?
766 // by default, ignore it
769 // but if we have cached information from the last KEY_PRESS
770 if ( gdk_event
->type
== GDK_KEY_RELEASE
)
773 if ( keysym
== s_lastKeyPress
.keysym
)
775 key_code
= s_lastKeyPress
.keycode
;
780 if ( gdk_event
->type
== GDK_KEY_PRESS
)
782 // remember it to be reused for KEY_UP event later
783 s_lastKeyPress
.keysym
= keysym
;
784 s_lastKeyPress
.keycode
= key_code
;
788 wxLogTrace(TRACE_KEYS
, _T("\t-> wxKeyCode %ld"), key_code
);
790 // sending unknown key events doesn't really make sense
794 // now fill all the other fields
795 wxFillOtherKeyEventFields(event
, win
, gdk_event
);
797 event
.m_keyCode
= key_code
;
799 if ( gdk_event
->type
== GDK_KEY_PRESS
|| gdk_event
->type
== GDK_KEY_RELEASE
)
801 event
.m_uniChar
= key_code
;
811 GtkIMContext
*context
;
812 GdkEventKey
*lastKeyEvent
;
816 context
= gtk_im_multicontext_new();
821 g_object_unref (context
);
827 gtk_window_key_press_callback( GtkWidget
*WXUNUSED(widget
),
828 GdkEventKey
*gdk_event
,
833 if (g_blockEventsOnDrag
)
836 wxKeyEvent
event( wxEVT_KEY_DOWN
);
838 bool return_after_IM
= false;
840 if( wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
842 // Emit KEY_DOWN event
843 ret
= win
->HandleWindowEvent( event
);
847 // Return after IM processing as we cannot do
848 // anything with it anyhow.
849 return_after_IM
= true;
852 if ((!ret
) && (win
->m_imData
!= NULL
))
854 // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API
855 // docs, if IM filter returns true, no further processing should be done.
856 // we should send the key_down event anyway.
857 bool intercepted_by_IM
= gtk_im_context_filter_keypress(win
->m_imData
->context
, gdk_event
);
858 win
->m_imData
->lastKeyEvent
= NULL
;
859 if (intercepted_by_IM
)
861 wxLogTrace(TRACE_KEYS
, _T("Key event intercepted by IM"));
872 wxWindowGTK
*ancestor
= win
;
875 int command
= ancestor
->GetAcceleratorTable()->GetCommand( event
);
878 wxCommandEvent
menu_event( wxEVT_COMMAND_MENU_SELECTED
, command
);
879 ret
= ancestor
->HandleWindowEvent( menu_event
);
883 // if the accelerator wasn't handled as menu event, try
884 // it as button click (for compatibility with other
886 wxCommandEvent
button_event( wxEVT_COMMAND_BUTTON_CLICKED
, command
);
887 ret
= ancestor
->HandleWindowEvent( button_event
);
892 if (ancestor
->IsTopLevel())
894 ancestor
= ancestor
->GetParent();
897 #endif // wxUSE_ACCEL
899 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
900 // will only be sent if it is not in an accelerator table.
904 KeySym keysym
= gdk_event
->keyval
;
905 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
906 key_code
= wxTranslateKeySymToWXKey(keysym
, true /* isChar */);
909 if ( wxIsAsciiKeysym(keysym
) )
912 key_code
= (unsigned char)keysym
;
914 // gdk_event->string is actually deprecated
915 else if ( gdk_event
->length
== 1 )
917 key_code
= (unsigned char)gdk_event
->string
[0];
923 wxLogTrace(TRACE_KEYS
, _T("Char event: %ld"), key_code
);
925 event
.m_keyCode
= key_code
;
927 // To conform to the docs we need to translate Ctrl-alpha
928 // characters to values in the range 1-26.
929 if ( event
.ControlDown() &&
930 ( wxIsLowerChar(key_code
) || wxIsUpperChar(key_code
) ))
932 if ( wxIsLowerChar(key_code
) )
933 event
.m_keyCode
= key_code
- 'a' + 1;
934 if ( wxIsUpperChar(key_code
) )
935 event
.m_keyCode
= key_code
- 'A' + 1;
937 event
.m_uniChar
= event
.m_keyCode
;
941 // Implement OnCharHook by checking ancestor top level windows
942 wxWindow
*parent
= win
;
943 while (parent
&& !parent
->IsTopLevel())
944 parent
= parent
->GetParent();
947 event
.SetEventType( wxEVT_CHAR_HOOK
);
948 ret
= parent
->HandleWindowEvent( event
);
953 event
.SetEventType(wxEVT_CHAR
);
954 ret
= win
->HandleWindowEvent( event
);
965 gtk_wxwindow_commit_cb (GtkIMContext
* WXUNUSED(context
),
969 wxKeyEvent
event( wxEVT_KEY_DOWN
);
971 // take modifiers, cursor position, timestamp etc. from the last
972 // key_press_event that was fed into Input Method:
973 if (window
->m_imData
->lastKeyEvent
)
975 wxFillOtherKeyEventFields(event
,
976 window
, window
->m_imData
->lastKeyEvent
);
980 event
.SetEventObject( window
);
983 const wxString
data(wxGTK_CONV_BACK_SYS(str
));
989 // Implement OnCharHook by checking ancestor top level windows
990 wxWindow
*parent
= window
;
991 while (parent
&& !parent
->IsTopLevel())
992 parent
= parent
->GetParent();
994 for( wxString::const_iterator pstr
= data
.begin(); pstr
!= data
.end(); ++pstr
)
997 event
.m_uniChar
= *pstr
;
998 // Backward compatible for ISO-8859-1
999 event
.m_keyCode
= *pstr
< 256 ? event
.m_uniChar
: 0;
1000 wxLogTrace(TRACE_KEYS
, _T("IM sent character '%c'"), event
.m_uniChar
);
1002 event
.m_keyCode
= (char)*pstr
;
1003 #endif // wxUSE_UNICODE
1005 // To conform to the docs we need to translate Ctrl-alpha
1006 // characters to values in the range 1-26.
1007 if ( event
.ControlDown() &&
1008 ( wxIsLowerChar(*pstr
) || wxIsUpperChar(*pstr
) ))
1010 if ( wxIsLowerChar(*pstr
) )
1011 event
.m_keyCode
= *pstr
- 'a' + 1;
1012 if ( wxIsUpperChar(*pstr
) )
1013 event
.m_keyCode
= *pstr
- 'A' + 1;
1015 event
.m_keyCode
= *pstr
- 'a' + 1;
1017 event
.m_uniChar
= event
.m_keyCode
;
1023 event
.SetEventType( wxEVT_CHAR_HOOK
);
1024 ret
= parent
->HandleWindowEvent( event
);
1029 event
.SetEventType(wxEVT_CHAR
);
1030 ret
= window
->HandleWindowEvent( event
);
1037 //-----------------------------------------------------------------------------
1038 // "key_release_event" from any window
1039 //-----------------------------------------------------------------------------
1043 gtk_window_key_release_callback( GtkWidget
* WXUNUSED(widget
),
1044 GdkEventKey
*gdk_event
,
1050 if (g_blockEventsOnDrag
)
1053 wxKeyEvent
event( wxEVT_KEY_UP
);
1054 if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1056 // unknown key pressed, ignore (the event would be useless anyhow)
1060 return win
->GTKProcessEvent(event
);
1064 // ============================================================================
1066 // ============================================================================
1068 // ----------------------------------------------------------------------------
1069 // mouse event processing helpers
1070 // ----------------------------------------------------------------------------
1072 // init wxMouseEvent with the info from GdkEventXXX struct
1073 template<typename T
> void InitMouseEvent(wxWindowGTK
*win
,
1074 wxMouseEvent
& event
,
1077 event
.SetTimestamp( gdk_event
->time
);
1078 event
.m_shiftDown
= gdk_event
->state
& GDK_SHIFT_MASK
;
1079 event
.m_controlDown
= gdk_event
->state
& GDK_CONTROL_MASK
;
1080 event
.m_altDown
= gdk_event
->state
& GDK_MOD1_MASK
;
1081 event
.m_metaDown
= gdk_event
->state
& GDK_META_MASK
;
1082 event
.m_leftDown
= gdk_event
->state
& GDK_BUTTON1_MASK
;
1083 event
.m_middleDown
= gdk_event
->state
& GDK_BUTTON2_MASK
;
1084 event
.m_rightDown
= gdk_event
->state
& GDK_BUTTON3_MASK
;
1085 event
.m_aux1Down
= gdk_event
->state
& GDK_BUTTON4_MASK
;
1086 event
.m_aux2Down
= gdk_event
->state
& GDK_BUTTON5_MASK
;
1088 wxPoint pt
= win
->GetClientAreaOrigin();
1089 event
.m_x
= (wxCoord
)gdk_event
->x
- pt
.x
;
1090 event
.m_y
= (wxCoord
)gdk_event
->y
- pt
.y
;
1092 if ((win
->m_wxwindow
) && (win
->GetLayoutDirection() == wxLayout_RightToLeft
))
1094 // origin in the upper right corner
1095 int window_width
= win
->m_wxwindow
->allocation
.width
;
1096 event
.m_x
= window_width
- event
.m_x
;
1099 event
.SetEventObject( win
);
1100 event
.SetId( win
->GetId() );
1101 event
.SetTimestamp( gdk_event
->time
);
1104 static void AdjustEventButtonState(wxMouseEvent
& event
)
1106 // GDK reports the old state of the button for a button press event, but
1107 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1108 // for a LEFT_DOWN event, not FALSE, so we will invert
1109 // left/right/middleDown for the corresponding click events
1111 if ((event
.GetEventType() == wxEVT_LEFT_DOWN
) ||
1112 (event
.GetEventType() == wxEVT_LEFT_DCLICK
) ||
1113 (event
.GetEventType() == wxEVT_LEFT_UP
))
1115 event
.m_leftDown
= !event
.m_leftDown
;
1119 if ((event
.GetEventType() == wxEVT_MIDDLE_DOWN
) ||
1120 (event
.GetEventType() == wxEVT_MIDDLE_DCLICK
) ||
1121 (event
.GetEventType() == wxEVT_MIDDLE_UP
))
1123 event
.m_middleDown
= !event
.m_middleDown
;
1127 if ((event
.GetEventType() == wxEVT_RIGHT_DOWN
) ||
1128 (event
.GetEventType() == wxEVT_RIGHT_DCLICK
) ||
1129 (event
.GetEventType() == wxEVT_RIGHT_UP
))
1131 event
.m_rightDown
= !event
.m_rightDown
;
1136 // find the window to send the mouse event too
1138 wxWindowGTK
*FindWindowForMouseEvent(wxWindowGTK
*win
, wxCoord
& x
, wxCoord
& y
)
1143 if (win
->m_wxwindow
)
1145 wxPizza
* pizza
= WX_PIZZA(win
->m_wxwindow
);
1146 xx
+= pizza
->m_scroll_x
;
1147 yy
+= pizza
->m_scroll_y
;
1150 wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
1153 wxWindowGTK
*child
= node
->GetData();
1155 node
= node
->GetNext();
1156 if (!child
->IsShown())
1159 if (child
->GTKIsTransparentForMouse())
1161 // wxStaticBox is transparent in the box itself
1162 int xx1
= child
->m_x
;
1163 int yy1
= child
->m_y
;
1164 int xx2
= child
->m_x
+ child
->m_width
;
1165 int yy2
= child
->m_y
+ child
->m_height
;
1168 if (((xx
>= xx1
) && (xx
<= xx1
+10) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1170 ((xx
>= xx2
-10) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1172 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy1
+10)) ||
1174 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy2
-1) && (yy
<= yy2
)))
1185 if ((child
->m_wxwindow
== NULL
) &&
1186 (child
->m_x
<= xx
) &&
1187 (child
->m_y
<= yy
) &&
1188 (child
->m_x
+child
->m_width
>= xx
) &&
1189 (child
->m_y
+child
->m_height
>= yy
))
1202 // ----------------------------------------------------------------------------
1203 // common event handlers helpers
1204 // ----------------------------------------------------------------------------
1206 bool wxWindowGTK::GTKProcessEvent(wxEvent
& event
) const
1208 // nothing special at this level
1209 return HandleWindowEvent(event
);
1212 int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny
*event
) const
1216 if (g_blockEventsOnDrag
)
1218 if (g_blockEventsOnScroll
)
1221 if (!GTKIsOwnWindow(event
->window
))
1227 // overloads for all GDK event types we use here: we need to have this as
1228 // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact,
1229 // derives from it in the sense that the structs have the same layout
1230 #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \
1231 static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \
1233 return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \
1236 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton
)
1237 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion
)
1238 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing
)
1240 #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD
1242 #define wxCOMMON_CALLBACK_PROLOGUE(event, win) \
1243 const int rc = wxGtkCallbackCommonPrologue(event, win); \
1247 // all event handlers must have C linkage as they're called from GTK+ C code
1251 //-----------------------------------------------------------------------------
1252 // "button_press_event"
1253 //-----------------------------------------------------------------------------
1256 gtk_window_button_press_callback( GtkWidget
*widget
,
1257 GdkEventButton
*gdk_event
,
1260 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1262 g_lastButtonNumber
= gdk_event
->button
;
1264 // GDK sends surplus button down events
1265 // before a double click event. We
1266 // need to filter these out.
1267 if ((gdk_event
->type
== GDK_BUTTON_PRESS
) && (win
->m_wxwindow
))
1269 GdkEvent
*peek_event
= gdk_event_peek();
1272 if ((peek_event
->type
== GDK_2BUTTON_PRESS
) ||
1273 (peek_event
->type
== GDK_3BUTTON_PRESS
))
1275 gdk_event_free( peek_event
);
1280 gdk_event_free( peek_event
);
1285 wxEventType event_type
= wxEVT_NULL
;
1287 if ( gdk_event
->type
== GDK_2BUTTON_PRESS
&&
1288 gdk_event
->button
>= 1 && gdk_event
->button
<= 3 )
1290 // Reset GDK internal timestamp variables in order to disable GDK
1291 // triple click events. GDK will then next time believe no button has
1292 // been clicked just before, and send a normal button click event.
1293 GdkDisplay
* display
= gtk_widget_get_display (widget
);
1294 display
->button_click_time
[1] = 0;
1295 display
->button_click_time
[0] = 0;
1298 if (gdk_event
->button
== 1)
1300 // note that GDK generates triple click events which are not supported
1301 // by wxWidgets but still have to be passed to the app as otherwise
1302 // clicks would simply go missing
1303 switch (gdk_event
->type
)
1305 // we shouldn't get triple clicks at all for GTK2 because we
1306 // suppress them artificially using the code above but we still
1307 // should map them to something for GTK1 and not just ignore them
1308 // as this would lose clicks
1309 case GDK_3BUTTON_PRESS
: // we could also map this to DCLICK...
1310 case GDK_BUTTON_PRESS
:
1311 event_type
= wxEVT_LEFT_DOWN
;
1314 case GDK_2BUTTON_PRESS
:
1315 event_type
= wxEVT_LEFT_DCLICK
;
1319 // just to silence gcc warnings
1323 else if (gdk_event
->button
== 2)
1325 switch (gdk_event
->type
)
1327 case GDK_3BUTTON_PRESS
:
1328 case GDK_BUTTON_PRESS
:
1329 event_type
= wxEVT_MIDDLE_DOWN
;
1332 case GDK_2BUTTON_PRESS
:
1333 event_type
= wxEVT_MIDDLE_DCLICK
;
1340 else if (gdk_event
->button
== 3)
1342 switch (gdk_event
->type
)
1344 case GDK_3BUTTON_PRESS
:
1345 case GDK_BUTTON_PRESS
:
1346 event_type
= wxEVT_RIGHT_DOWN
;
1349 case GDK_2BUTTON_PRESS
:
1350 event_type
= wxEVT_RIGHT_DCLICK
;
1358 if ( event_type
== wxEVT_NULL
)
1360 // unknown mouse button or click type
1364 g_lastMouseEvent
= (GdkEvent
*) gdk_event
;
1366 wxMouseEvent
event( event_type
);
1367 InitMouseEvent( win
, event
, gdk_event
);
1369 AdjustEventButtonState(event
);
1371 // find the correct window to send the event to: it may be a different one
1372 // from the one which got it at GTK+ level because some controls don't have
1373 // their own X window and thus cannot get any events.
1374 if ( !g_captureWindow
)
1375 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1377 // reset the event object and id in case win changed.
1378 event
.SetEventObject( win
);
1379 event
.SetId( win
->GetId() );
1381 bool ret
= win
->GTKProcessEvent( event
);
1382 g_lastMouseEvent
= NULL
;
1386 if ((event_type
== wxEVT_LEFT_DOWN
) && !win
->IsOfStandardClass() &&
1387 (gs_currentFocus
!= win
) /* && win->IsFocusable() */)
1392 if (event_type
== wxEVT_RIGHT_DOWN
)
1394 // generate a "context menu" event: this is similar to right mouse
1395 // click under many GUIs except that it is generated differently
1396 // (right up under MSW, ctrl-click under Mac, right down here) and
1398 // (a) it's a command event and so is propagated to the parent
1399 // (b) under some ports it can be generated from kbd too
1400 // (c) it uses screen coords (because of (a))
1401 wxContextMenuEvent
evtCtx(
1404 win
->ClientToScreen(event
.GetPosition()));
1405 evtCtx
.SetEventObject(win
);
1406 return win
->GTKProcessEvent(evtCtx
);
1412 //-----------------------------------------------------------------------------
1413 // "button_release_event"
1414 //-----------------------------------------------------------------------------
1417 gtk_window_button_release_callback( GtkWidget
*WXUNUSED(widget
),
1418 GdkEventButton
*gdk_event
,
1421 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1423 g_lastButtonNumber
= 0;
1425 wxEventType event_type
= wxEVT_NULL
;
1427 switch (gdk_event
->button
)
1430 event_type
= wxEVT_LEFT_UP
;
1434 event_type
= wxEVT_MIDDLE_UP
;
1438 event_type
= wxEVT_RIGHT_UP
;
1442 // unknown button, don't process
1446 g_lastMouseEvent
= (GdkEvent
*) gdk_event
;
1448 wxMouseEvent
event( event_type
);
1449 InitMouseEvent( win
, event
, gdk_event
);
1451 AdjustEventButtonState(event
);
1453 if ( !g_captureWindow
)
1454 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1456 // reset the event object and id in case win changed.
1457 event
.SetEventObject( win
);
1458 event
.SetId( win
->GetId() );
1460 bool ret
= win
->GTKProcessEvent(event
);
1462 g_lastMouseEvent
= NULL
;
1467 //-----------------------------------------------------------------------------
1468 // "motion_notify_event"
1469 //-----------------------------------------------------------------------------
1472 gtk_window_motion_notify_callback( GtkWidget
* WXUNUSED(widget
),
1473 GdkEventMotion
*gdk_event
,
1476 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1478 if (gdk_event
->is_hint
)
1482 GdkModifierType state
;
1483 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1488 g_lastMouseEvent
= (GdkEvent
*) gdk_event
;
1490 wxMouseEvent
event( wxEVT_MOTION
);
1491 InitMouseEvent(win
, event
, gdk_event
);
1493 if ( g_captureWindow
)
1495 // synthesise a mouse enter or leave event if needed
1496 GdkWindow
*winUnderMouse
= gdk_window_at_pointer(NULL
, NULL
);
1497 // This seems to be necessary and actually been added to
1498 // GDK itself in version 2.0.X
1501 bool hasMouse
= winUnderMouse
== gdk_event
->window
;
1502 if ( hasMouse
!= g_captureWindowHasMouse
)
1504 // the mouse changed window
1505 g_captureWindowHasMouse
= hasMouse
;
1507 wxMouseEvent
eventM(g_captureWindowHasMouse
? wxEVT_ENTER_WINDOW
1508 : wxEVT_LEAVE_WINDOW
);
1509 InitMouseEvent(win
, eventM
, gdk_event
);
1510 eventM
.SetEventObject(win
);
1511 win
->GTKProcessEvent(eventM
);
1516 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1518 // reset the event object and id in case win changed.
1519 event
.SetEventObject( win
);
1520 event
.SetId( win
->GetId() );
1523 if ( !g_captureWindow
)
1525 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1526 if (win
->GTKProcessEvent( cevent
))
1528 win
->SetCursor( cevent
.GetCursor() );
1532 bool ret
= win
->GTKProcessEvent(event
);
1534 g_lastMouseEvent
= NULL
;
1539 //-----------------------------------------------------------------------------
1540 // "scroll_event" (mouse wheel event)
1541 //-----------------------------------------------------------------------------
1544 window_scroll_event_hscrollbar(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
)
1546 if (gdk_event
->direction
!= GDK_SCROLL_LEFT
&&
1547 gdk_event
->direction
!= GDK_SCROLL_RIGHT
)
1552 wxMouseEvent
event(wxEVT_MOUSEWHEEL
);
1553 InitMouseEvent(win
, event
, gdk_event
);
1555 GtkRange
*range
= win
->m_scrollBar
[wxWindow::ScrollDir_Horz
];
1556 if (!range
) return FALSE
;
1558 if (range
&& GTK_WIDGET_VISIBLE (range
))
1560 GtkAdjustment
*adj
= range
->adjustment
;
1561 gdouble delta
= adj
->step_increment
* 3;
1562 if (gdk_event
->direction
== GDK_SCROLL_LEFT
)
1565 gdouble new_value
= CLAMP (adj
->value
+ delta
, adj
->lower
, adj
->upper
- adj
->page_size
);
1567 gtk_adjustment_set_value (adj
, new_value
);
1576 window_scroll_event(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
)
1578 if (gdk_event
->direction
!= GDK_SCROLL_UP
&&
1579 gdk_event
->direction
!= GDK_SCROLL_DOWN
)
1584 wxMouseEvent
event(wxEVT_MOUSEWHEEL
);
1585 InitMouseEvent(win
, event
, gdk_event
);
1587 // FIXME: Get these values from GTK or GDK
1588 event
.m_linesPerAction
= 3;
1589 event
.m_wheelDelta
= 120;
1590 if (gdk_event
->direction
== GDK_SCROLL_UP
)
1591 event
.m_wheelRotation
= 120;
1593 event
.m_wheelRotation
= -120;
1595 if (win
->GTKProcessEvent(event
))
1598 GtkRange
*range
= win
->m_scrollBar
[wxWindow::ScrollDir_Vert
];
1599 if (!range
) return FALSE
;
1601 if (range
&& GTK_WIDGET_VISIBLE (range
))
1603 GtkAdjustment
*adj
= range
->adjustment
;
1604 gdouble delta
= adj
->step_increment
* 3;
1605 if (gdk_event
->direction
== GDK_SCROLL_UP
)
1608 gdouble new_value
= CLAMP (adj
->value
+ delta
, adj
->lower
, adj
->upper
- adj
->page_size
);
1610 gtk_adjustment_set_value (adj
, new_value
);
1618 //-----------------------------------------------------------------------------
1620 //-----------------------------------------------------------------------------
1622 static gboolean
wxgtk_window_popup_menu_callback(GtkWidget
*, wxWindowGTK
* win
)
1624 wxContextMenuEvent
event(wxEVT_CONTEXT_MENU
, win
->GetId(), wxPoint(-1, -1));
1625 event
.SetEventObject(win
);
1626 return win
->GTKProcessEvent(event
);
1629 //-----------------------------------------------------------------------------
1631 //-----------------------------------------------------------------------------
1634 gtk_window_focus_in_callback( GtkWidget
* WXUNUSED(widget
),
1635 GdkEventFocus
*WXUNUSED(event
),
1638 return win
->GTKHandleFocusIn();
1641 //-----------------------------------------------------------------------------
1642 // "focus_out_event"
1643 //-----------------------------------------------------------------------------
1646 gtk_window_focus_out_callback( GtkWidget
* WXUNUSED(widget
),
1647 GdkEventFocus
* WXUNUSED(gdk_event
),
1650 return win
->GTKHandleFocusOut();
1653 //-----------------------------------------------------------------------------
1655 //-----------------------------------------------------------------------------
1658 wx_window_focus_callback(GtkWidget
*widget
,
1659 GtkDirectionType
WXUNUSED(direction
),
1662 // the default handler for focus signal in GtkScrolledWindow sets
1663 // focus to the window itself even if it doesn't accept focus, i.e. has no
1664 // GTK_CAN_FOCUS in its style -- work around this by forcibly preventing
1665 // the signal from reaching gtk_scrolled_window_focus() if we don't have
1666 // any children which might accept focus (we know we don't accept the focus
1667 // ourselves as this signal is only connected in this case)
1668 if ( win
->GetChildren().empty() )
1669 g_signal_stop_emission_by_name(widget
, "focus");
1671 // we didn't change the focus
1675 //-----------------------------------------------------------------------------
1676 // "enter_notify_event"
1677 //-----------------------------------------------------------------------------
1680 gtk_window_enter_callback( GtkWidget
*widget
,
1681 GdkEventCrossing
*gdk_event
,
1684 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1686 // Event was emitted after a grab
1687 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1691 GdkModifierType state
= (GdkModifierType
)0;
1693 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1695 wxMouseEvent
event( wxEVT_ENTER_WINDOW
);
1696 InitMouseEvent(win
, event
, gdk_event
);
1697 wxPoint pt
= win
->GetClientAreaOrigin();
1698 event
.m_x
= x
+ pt
.x
;
1699 event
.m_y
= y
+ pt
.y
;
1701 if ( !g_captureWindow
)
1703 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1704 if (win
->GTKProcessEvent( cevent
))
1706 win
->SetCursor( cevent
.GetCursor() );
1710 return win
->GTKProcessEvent(event
);
1713 //-----------------------------------------------------------------------------
1714 // "leave_notify_event"
1715 //-----------------------------------------------------------------------------
1718 gtk_window_leave_callback( GtkWidget
*widget
,
1719 GdkEventCrossing
*gdk_event
,
1722 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1724 // Event was emitted after an ungrab
1725 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1727 wxMouseEvent
event( wxEVT_LEAVE_WINDOW
);
1731 GdkModifierType state
= (GdkModifierType
)0;
1733 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1735 InitMouseEvent(win
, event
, gdk_event
);
1737 return win
->GTKProcessEvent(event
);
1740 //-----------------------------------------------------------------------------
1741 // "value_changed" from scrollbar
1742 //-----------------------------------------------------------------------------
1745 gtk_scrollbar_value_changed(GtkRange
* range
, wxWindow
* win
)
1747 wxEventType eventType
= win
->GTKGetScrollEventType(range
);
1748 if (eventType
!= wxEVT_NULL
)
1750 // Convert scroll event type to scrollwin event type
1751 eventType
+= wxEVT_SCROLLWIN_TOP
- wxEVT_SCROLL_TOP
;
1753 // find the scrollbar which generated the event
1754 wxWindowGTK::ScrollDir dir
= win
->ScrollDirFromRange(range
);
1756 // generate the corresponding wx event
1757 const int orient
= wxWindow::OrientFromScrollDir(dir
);
1758 wxScrollWinEvent
event(eventType
, win
->GetScrollPos(orient
), orient
);
1759 event
.SetEventObject(win
);
1761 win
->GTKProcessEvent(event
);
1765 //-----------------------------------------------------------------------------
1766 // "button_press_event" from scrollbar
1767 //-----------------------------------------------------------------------------
1770 gtk_scrollbar_button_press_event(GtkRange
*, GdkEventButton
*, wxWindow
* win
)
1772 g_blockEventsOnScroll
= true;
1773 win
->m_mouseButtonDown
= true;
1778 //-----------------------------------------------------------------------------
1779 // "event_after" from scrollbar
1780 //-----------------------------------------------------------------------------
1783 gtk_scrollbar_event_after(GtkRange
* range
, GdkEvent
* event
, wxWindow
* win
)
1785 if (event
->type
== GDK_BUTTON_RELEASE
)
1787 g_signal_handlers_block_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
1789 const int orient
= wxWindow::OrientFromScrollDir(
1790 win
->ScrollDirFromRange(range
));
1791 wxScrollWinEvent
evt(wxEVT_SCROLLWIN_THUMBRELEASE
,
1792 win
->GetScrollPos(orient
), orient
);
1793 evt
.SetEventObject(win
);
1794 win
->GTKProcessEvent(evt
);
1798 //-----------------------------------------------------------------------------
1799 // "button_release_event" from scrollbar
1800 //-----------------------------------------------------------------------------
1803 gtk_scrollbar_button_release_event(GtkRange
* range
, GdkEventButton
*, wxWindow
* win
)
1805 g_blockEventsOnScroll
= false;
1806 win
->m_mouseButtonDown
= false;
1807 // If thumb tracking
1808 if (win
->m_isScrolling
)
1810 win
->m_isScrolling
= false;
1811 // Hook up handler to send thumb release event after this emission is finished.
1812 // To allow setting scroll position from event handler, sending event must
1813 // be deferred until after the GtkRange handler for this signal has run
1814 g_signal_handlers_unblock_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
1820 //-----------------------------------------------------------------------------
1821 // "realize" from m_widget
1822 //-----------------------------------------------------------------------------
1825 gtk_window_realized_callback(GtkWidget
* widget
, wxWindow
* win
)
1829 gtk_im_context_set_client_window( win
->m_imData
->context
,
1833 // We cannot set colours and fonts before the widget
1834 // been realized, so we do this directly after realization
1835 // or otherwise in idle time
1837 if (win
->m_needsStyleChange
)
1839 win
->SetBackgroundStyle(win
->GetBackgroundStyle());
1840 win
->m_needsStyleChange
= false;
1843 wxWindowCreateEvent
event( win
);
1844 event
.SetEventObject( win
);
1845 win
->GTKProcessEvent( event
);
1848 //-----------------------------------------------------------------------------
1849 // "size_allocate" from m_wxwindow or m_widget
1850 //-----------------------------------------------------------------------------
1853 size_allocate(GtkWidget
*, GtkAllocation
* alloc
, wxWindow
* win
)
1855 int w
= alloc
->width
;
1856 int h
= alloc
->height
;
1857 if (win
->m_wxwindow
)
1859 int border_x
, border_y
;
1860 WX_PIZZA(win
->m_wxwindow
)->get_border_widths(border_x
, border_y
);
1866 if (win
->m_oldClientWidth
!= w
|| win
->m_oldClientHeight
!= h
)
1868 win
->m_oldClientWidth
= w
;
1869 win
->m_oldClientHeight
= h
;
1870 // this callback can be connected to m_wxwindow,
1871 // so always get size from m_widget->allocation
1872 win
->m_width
= win
->m_widget
->allocation
.width
;
1873 win
->m_height
= win
->m_widget
->allocation
.height
;
1874 if (!win
->m_nativeSizeEvent
)
1876 wxSizeEvent
event(win
->GetSize(), win
->GetId());
1877 event
.SetEventObject(win
);
1878 win
->GTKProcessEvent(event
);
1883 //-----------------------------------------------------------------------------
1885 //-----------------------------------------------------------------------------
1887 #if GTK_CHECK_VERSION(2, 8, 0)
1889 gtk_window_grab_broken( GtkWidget
*,
1890 GdkEventGrabBroken
*event
,
1893 // Mouse capture has been lost involuntarily, notify the application
1894 if(!event
->keyboard
&& wxWindow::GetCapture() == win
)
1896 wxMouseCaptureLostEvent
evt( win
->GetId() );
1897 evt
.SetEventObject( win
);
1898 win
->HandleWindowEvent( evt
);
1904 //-----------------------------------------------------------------------------
1906 //-----------------------------------------------------------------------------
1909 void gtk_window_style_set_callback( GtkWidget
*WXUNUSED(widget
),
1910 GtkStyle
*previous_style
,
1913 if (win
&& previous_style
)
1915 wxSysColourChangedEvent event
;
1916 event
.SetEventObject(win
);
1918 win
->GTKProcessEvent( event
);
1924 // Helper to suspend colour change event event processing while we change a widget's style
1925 class wxSuspendStyleEvents
1928 wxSuspendStyleEvents(wxWindow
* win
)
1931 if (win
->m_wxwindow
&& win
->IsTopLevel())
1934 g_signal_handlers_block_by_func(
1935 m_win
->m_wxwindow
, (void*)gtk_window_style_set_callback
, m_win
);
1938 ~wxSuspendStyleEvents()
1941 g_signal_handlers_unblock_by_func(
1942 m_win
->m_wxwindow
, (void*)gtk_window_style_set_callback
, m_win
);
1948 // ----------------------------------------------------------------------------
1949 // this wxWindowBase function is implemented here (in platform-specific file)
1950 // because it is static and so couldn't be made virtual
1951 // ----------------------------------------------------------------------------
1953 wxWindow
*wxWindowBase::DoFindFocus()
1955 wxWindowGTK
*focus
= gs_pendingFocus
? gs_pendingFocus
: gs_currentFocus
;
1956 // the cast is necessary when we compile in wxUniversal mode
1957 return static_cast<wxWindow
*>(focus
);
1960 void wxWindowGTK::AddChildGTK(wxWindowGTK
* child
)
1962 /* the window might have been scrolled already, do we
1963 have to adapt the position */
1964 wxPizza
* pizza
= WX_PIZZA(m_wxwindow
);
1965 child
->m_x
+= pizza
->m_scroll_x
;
1966 child
->m_y
+= pizza
->m_scroll_y
;
1968 gtk_widget_set_size_request(
1969 child
->m_widget
, child
->m_width
, child
->m_height
);
1971 GTK_FIXED(m_wxwindow
), child
->m_widget
, child
->m_x
, child
->m_y
);
1974 //-----------------------------------------------------------------------------
1976 //-----------------------------------------------------------------------------
1978 wxWindow
*wxGetActiveWindow()
1980 return wxWindow::FindFocus();
1984 wxMouseState
wxGetMouseState()
1990 GdkModifierType mask
;
1992 gdk_window_get_pointer(NULL
, &x
, &y
, &mask
);
1996 ms
.SetLeftDown(mask
& GDK_BUTTON1_MASK
);
1997 ms
.SetMiddleDown(mask
& GDK_BUTTON2_MASK
);
1998 ms
.SetRightDown(mask
& GDK_BUTTON3_MASK
);
1999 ms
.SetAux1Down(mask
& GDK_BUTTON4_MASK
);
2000 ms
.SetAux2Down(mask
& GDK_BUTTON5_MASK
);
2002 ms
.SetControlDown(mask
& GDK_CONTROL_MASK
);
2003 ms
.SetShiftDown(mask
& GDK_SHIFT_MASK
);
2004 ms
.SetAltDown(mask
& GDK_MOD1_MASK
);
2005 ms
.SetMetaDown(mask
& GDK_META_MASK
);
2010 //-----------------------------------------------------------------------------
2012 //-----------------------------------------------------------------------------
2014 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2016 #ifdef __WXUNIVERSAL__
2017 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
)
2019 IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
)
2020 #endif // __WXUNIVERSAL__/__WXGTK__
2022 void wxWindowGTK::Init()
2027 m_focusWidget
= NULL
;
2037 m_showOnIdle
= false;
2040 m_nativeSizeEvent
= false;
2042 m_isScrolling
= false;
2043 m_mouseButtonDown
= false;
2045 // initialize scrolling stuff
2046 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2048 m_scrollBar
[dir
] = NULL
;
2049 m_scrollPos
[dir
] = 0;
2053 m_oldClientHeight
= 0;
2055 m_clipPaintRegion
= false;
2057 m_needsStyleChange
= false;
2059 m_cursor
= *wxSTANDARD_CURSOR
;
2062 m_dirtyTabOrder
= false;
2065 wxWindowGTK::wxWindowGTK()
2070 wxWindowGTK::wxWindowGTK( wxWindow
*parent
,
2075 const wxString
&name
)
2079 Create( parent
, id
, pos
, size
, style
, name
);
2082 bool wxWindowGTK::Create( wxWindow
*parent
,
2087 const wxString
&name
)
2089 // Get default border
2090 wxBorder border
= GetBorder(style
);
2092 style
&= ~wxBORDER_MASK
;
2095 if (!PreCreation( parent
, pos
, size
) ||
2096 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
2098 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2102 // We should accept the native look
2104 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2105 scroll_class
->scrollbar_spacing
= 0;
2109 m_wxwindow
= wxPizza::New(m_windowStyle
,this);
2110 #ifndef __WXUNIVERSAL__
2111 if (HasFlag(wxPizza::BORDER_STYLES
))
2113 g_signal_connect(m_wxwindow
, "parent_set",
2114 G_CALLBACK(parent_set
), this);
2117 if (!HasFlag(wxHSCROLL
) && !HasFlag(wxVSCROLL
))
2118 m_widget
= m_wxwindow
;
2121 m_widget
= gtk_scrolled_window_new( NULL
, NULL
);
2123 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(m_widget
);
2125 // There is a conflict with default bindings at GTK+
2126 // level between scrolled windows and notebooks both of which want to use
2127 // Ctrl-PageUp/Down: scrolled windows for scrolling in the horizontal
2128 // direction and notebooks for changing pages -- we decide that if we don't
2129 // have wxHSCROLL style we can safely sacrifice horizontal scrolling if it
2130 // means we can get working keyboard navigation in notebooks
2131 if ( !HasFlag(wxHSCROLL
) )
2134 bindings
= gtk_binding_set_by_class(G_OBJECT_GET_CLASS(m_widget
));
2137 gtk_binding_entry_remove(bindings
, GDK_Page_Up
, GDK_CONTROL_MASK
);
2138 gtk_binding_entry_remove(bindings
, GDK_Page_Down
, GDK_CONTROL_MASK
);
2142 if (HasFlag(wxALWAYS_SHOW_SB
))
2144 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_ALWAYS
, GTK_POLICY_ALWAYS
);
2146 scrolledWindow
->hscrollbar_visible
= TRUE
;
2147 scrolledWindow
->vscrollbar_visible
= TRUE
;
2151 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
2154 m_scrollBar
[ScrollDir_Horz
] = GTK_RANGE(scrolledWindow
->hscrollbar
);
2155 m_scrollBar
[ScrollDir_Vert
] = GTK_RANGE(scrolledWindow
->vscrollbar
);
2156 if (GetLayoutDirection() == wxLayout_RightToLeft
)
2157 gtk_range_set_inverted( m_scrollBar
[ScrollDir_Horz
], TRUE
);
2159 gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow
);
2161 // connect various scroll-related events
2162 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2164 // these handlers block mouse events to any window during scrolling
2165 // such as motion events and prevent GTK and wxWidgets from fighting
2166 // over where the slider should be
2167 g_signal_connect(m_scrollBar
[dir
], "button_press_event",
2168 G_CALLBACK(gtk_scrollbar_button_press_event
), this);
2169 g_signal_connect(m_scrollBar
[dir
], "button_release_event",
2170 G_CALLBACK(gtk_scrollbar_button_release_event
), this);
2172 gulong handler_id
= g_signal_connect(m_scrollBar
[dir
], "event_after",
2173 G_CALLBACK(gtk_scrollbar_event_after
), this);
2174 g_signal_handler_block(m_scrollBar
[dir
], handler_id
);
2176 // these handlers get notified when scrollbar slider moves
2177 g_signal_connect_after(m_scrollBar
[dir
], "value_changed",
2178 G_CALLBACK(gtk_scrollbar_value_changed
), this);
2181 gtk_widget_show( m_wxwindow
);
2183 g_object_ref(m_widget
);
2186 m_parent
->DoAddChild( this );
2188 m_focusWidget
= m_wxwindow
;
2190 SetCanFocus(AcceptsFocus());
2197 wxWindowGTK::~wxWindowGTK()
2201 if (gs_currentFocus
== this)
2202 gs_currentFocus
= NULL
;
2203 if (gs_pendingFocus
== this)
2204 gs_pendingFocus
= NULL
;
2206 if ( gs_deferredFocusOut
== this )
2207 gs_deferredFocusOut
= NULL
;
2211 // destroy children before destroying this window itself
2214 // unhook focus handlers to prevent stray events being
2215 // propagated to this (soon to be) dead object
2216 if (m_focusWidget
!= NULL
)
2218 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2219 (gpointer
) gtk_window_focus_in_callback
,
2221 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2222 (gpointer
) gtk_window_focus_out_callback
,
2229 // delete before the widgets to avoid a crash on solaris
2234 // Note that gtk_widget_destroy() does not destroy the widget, it just
2235 // emits the "destroy" signal. The widget is not actually destroyed
2236 // until its reference count drops to zero.
2237 gtk_widget_destroy(m_widget
);
2238 // Release our reference, should be the last one
2239 g_object_unref(m_widget
);
2245 bool wxWindowGTK::PreCreation( wxWindowGTK
*parent
, const wxPoint
&pos
, const wxSize
&size
)
2247 if ( GTKNeedsParent() )
2249 wxCHECK_MSG( parent
, false, wxT("Must have non-NULL parent") );
2252 // Use either the given size, or the default if -1 is given.
2253 // See wxWindowBase for these functions.
2254 m_width
= WidthDefault(size
.x
) ;
2255 m_height
= HeightDefault(size
.y
);
2263 void wxWindowGTK::PostCreation()
2265 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2271 // these get reported to wxWidgets -> wxPaintEvent
2273 g_signal_connect (m_wxwindow
, "expose_event",
2274 G_CALLBACK (gtk_window_expose_callback
), this);
2276 if (GetLayoutDirection() == wxLayout_LeftToRight
)
2277 gtk_widget_set_redraw_on_allocate(m_wxwindow
, HasFlag(wxFULL_REPAINT_ON_RESIZE
));
2280 // Create input method handler
2281 m_imData
= new wxGtkIMData
;
2283 // Cannot handle drawing preedited text yet
2284 gtk_im_context_set_use_preedit( m_imData
->context
, FALSE
);
2286 g_signal_connect (m_imData
->context
, "commit",
2287 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2292 if (!GTK_IS_WINDOW(m_widget
))
2294 if (m_focusWidget
== NULL
)
2295 m_focusWidget
= m_widget
;
2299 g_signal_connect (m_focusWidget
, "focus_in_event",
2300 G_CALLBACK (gtk_window_focus_in_callback
), this);
2301 g_signal_connect (m_focusWidget
, "focus_out_event",
2302 G_CALLBACK (gtk_window_focus_out_callback
), this);
2306 g_signal_connect_after (m_focusWidget
, "focus_in_event",
2307 G_CALLBACK (gtk_window_focus_in_callback
), this);
2308 g_signal_connect_after (m_focusWidget
, "focus_out_event",
2309 G_CALLBACK (gtk_window_focus_out_callback
), this);
2313 if ( !AcceptsFocusFromKeyboard() )
2317 g_signal_connect(m_widget
, "focus",
2318 G_CALLBACK(wx_window_focus_callback
), this);
2321 // connect to the various key and mouse handlers
2323 GtkWidget
*connect_widget
= GetConnectWidget();
2325 ConnectWidget( connect_widget
);
2327 /* We cannot set colours, fonts and cursors before the widget has
2328 been realized, so we do this directly after realization */
2329 g_signal_connect (connect_widget
, "realize",
2330 G_CALLBACK (gtk_window_realized_callback
), this);
2334 g_signal_connect(m_wxwindow
? m_wxwindow
: m_widget
, "size_allocate",
2335 G_CALLBACK(size_allocate
), this);
2340 #if GTK_CHECK_VERSION(2, 8, 0)
2341 if (!gtk_check_version(2,8,0))
2343 // Make sure we can notify the app when mouse capture is lost
2344 g_signal_connect (m_wxwindow
, "grab_broken_event",
2345 G_CALLBACK (gtk_window_grab_broken
), this);
2350 if ( connect_widget
!= m_wxwindow
)
2352 #if GTK_CHECK_VERSION(2, 8, 0)
2353 if (!gtk_check_version(2,8,0))
2355 // Make sure we can notify app code when mouse capture is lost
2356 g_signal_connect (connect_widget
, "grab_broken_event",
2357 G_CALLBACK (gtk_window_grab_broken
), this);
2362 #ifdef GTK_IS_FILE_CHOOSER_BUTTON
2363 if (!gtk_check_version(2,6,0) && GTK_IS_FILE_CHOOSER_BUTTON(m_widget
))
2365 // If we connect to the "size_request" signal of a GtkFileChooserButton
2366 // then that control won't be sized properly when placed inside sizers
2367 // (this can be tested removing this elseif and running XRC or WIDGETS samples)
2368 // FIXME: what should be done here ?
2371 if ( !IsTopLevel() ) // top level windows use their own callback
2373 // This is needed if we want to add our windows into native
2374 // GTK controls, such as the toolbar. With this callback, the
2375 // toolbar gets to know the correct size (the one set by the
2376 // programmer). Sadly, it misbehaves for wxComboBox.
2377 g_signal_connect (m_widget
, "size_request",
2378 G_CALLBACK (wxgtk_window_size_request_callback
),
2382 InheritAttributes();
2386 SetLayoutDirection(wxLayout_Default
);
2388 // unless the window was created initially hidden (i.e. Hide() had been
2389 // called before Create()), we should show it at GTK+ level as well
2391 gtk_widget_show( m_widget
);
2394 void wxWindowGTK::ConnectWidget( GtkWidget
*widget
)
2396 g_signal_connect (widget
, "key_press_event",
2397 G_CALLBACK (gtk_window_key_press_callback
), this);
2398 g_signal_connect (widget
, "key_release_event",
2399 G_CALLBACK (gtk_window_key_release_callback
), this);
2400 g_signal_connect (widget
, "button_press_event",
2401 G_CALLBACK (gtk_window_button_press_callback
), this);
2402 g_signal_connect (widget
, "button_release_event",
2403 G_CALLBACK (gtk_window_button_release_callback
), this);
2404 g_signal_connect (widget
, "motion_notify_event",
2405 G_CALLBACK (gtk_window_motion_notify_callback
), this);
2407 g_signal_connect (widget
, "scroll_event",
2408 G_CALLBACK (window_scroll_event
), this);
2409 if (m_scrollBar
[ScrollDir_Horz
])
2410 g_signal_connect (m_scrollBar
[ScrollDir_Horz
], "scroll_event",
2411 G_CALLBACK (window_scroll_event_hscrollbar
), this);
2412 if (m_scrollBar
[ScrollDir_Vert
])
2413 g_signal_connect (m_scrollBar
[ScrollDir_Vert
], "scroll_event",
2414 G_CALLBACK (window_scroll_event
), this);
2416 g_signal_connect (widget
, "popup_menu",
2417 G_CALLBACK (wxgtk_window_popup_menu_callback
), this);
2418 g_signal_connect (widget
, "enter_notify_event",
2419 G_CALLBACK (gtk_window_enter_callback
), this);
2420 g_signal_connect (widget
, "leave_notify_event",
2421 G_CALLBACK (gtk_window_leave_callback
), this);
2423 if (IsTopLevel() && m_wxwindow
)
2424 g_signal_connect (m_wxwindow
, "style_set",
2425 G_CALLBACK (gtk_window_style_set_callback
), this);
2428 bool wxWindowGTK::Destroy()
2430 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2434 return wxWindowBase::Destroy();
2437 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
)
2439 gtk_widget_set_size_request(m_widget
, width
, height
);
2440 // inform the parent to perform the move
2441 WX_PIZZA(m_parent
->m_wxwindow
)->move(m_widget
, x
, y
);
2444 void wxWindowGTK::ConstrainSize()
2447 // GPE's window manager doesn't like size hints at all, esp. when the user
2448 // has to use the virtual keyboard, so don't constrain size there
2452 const wxSize minSize
= GetMinSize();
2453 const wxSize maxSize
= GetMaxSize();
2454 if (minSize
.x
> 0 && m_width
< minSize
.x
) m_width
= minSize
.x
;
2455 if (minSize
.y
> 0 && m_height
< minSize
.y
) m_height
= minSize
.y
;
2456 if (maxSize
.x
> 0 && m_width
> maxSize
.x
) m_width
= maxSize
.x
;
2457 if (maxSize
.y
> 0 && m_height
> maxSize
.y
) m_height
= maxSize
.y
;
2461 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
2463 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2464 wxASSERT_MSG( (m_parent
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") );
2466 int currentX
, currentY
;
2467 GetPosition(¤tX
, ¤tY
);
2468 if (x
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2470 if (y
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2472 AdjustForParentClientOrigin(x
, y
, sizeFlags
);
2474 // calculate the best size if we should auto size the window
2475 if ( ((sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1) ||
2476 ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1) )
2478 const wxSize sizeBest
= GetBestSize();
2479 if ( (sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1 )
2481 if ( (sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1 )
2482 height
= sizeBest
.y
;
2485 const wxSize
oldSize(m_width
, m_height
);
2491 if (m_parent
->m_wxwindow
)
2493 wxPizza
* pizza
= WX_PIZZA(m_parent
->m_wxwindow
);
2494 m_x
= x
+ pizza
->m_scroll_x
;
2495 m_y
= y
+ pizza
->m_scroll_y
;
2497 int left_border
= 0;
2498 int right_border
= 0;
2500 int bottom_border
= 0;
2502 /* the default button has a border around it */
2503 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
2505 GtkBorder
*default_border
= NULL
;
2506 gtk_widget_style_get( m_widget
, "default_border", &default_border
, NULL
);
2509 left_border
+= default_border
->left
;
2510 right_border
+= default_border
->right
;
2511 top_border
+= default_border
->top
;
2512 bottom_border
+= default_border
->bottom
;
2513 gtk_border_free( default_border
);
2517 DoMoveWindow( m_x
- left_border
,
2519 m_width
+left_border
+right_border
,
2520 m_height
+top_border
+bottom_border
);
2523 if (m_width
!= oldSize
.x
|| m_height
!= oldSize
.y
)
2525 // update these variables to keep size_allocate handler
2526 // from sending another size event for this change
2527 GetClientSize( &m_oldClientWidth
, &m_oldClientHeight
);
2529 gtk_widget_queue_resize(m_widget
);
2530 if (!m_nativeSizeEvent
)
2532 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
2533 event
.SetEventObject( this );
2534 HandleWindowEvent( event
);
2537 if (sizeFlags
& wxSIZE_FORCE_EVENT
)
2539 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
2540 event
.SetEventObject( this );
2541 HandleWindowEvent( event
);
2545 bool wxWindowGTK::GTKShowFromOnIdle()
2547 if (IsShown() && m_showOnIdle
&& !GTK_WIDGET_VISIBLE (m_widget
))
2549 GtkAllocation alloc
;
2552 alloc
.width
= m_width
;
2553 alloc
.height
= m_height
;
2554 gtk_widget_size_allocate( m_widget
, &alloc
);
2555 gtk_widget_show( m_widget
);
2556 wxShowEvent
eventShow(GetId(), true);
2557 eventShow
.SetEventObject(this);
2558 HandleWindowEvent(eventShow
);
2559 m_showOnIdle
= false;
2566 void wxWindowGTK::OnInternalIdle()
2568 if ( gs_deferredFocusOut
)
2569 GTKHandleDeferredFocusOut();
2571 // Check if we have to show window now
2572 if (GTKShowFromOnIdle()) return;
2574 if ( m_dirtyTabOrder
)
2576 m_dirtyTabOrder
= false;
2580 // Update style if the window was not yet realized
2581 // and SetBackgroundStyle(wxBG_STYLE_CUSTOM) was called
2582 if (m_needsStyleChange
)
2584 SetBackgroundStyle(GetBackgroundStyle());
2585 m_needsStyleChange
= false;
2588 wxCursor cursor
= m_cursor
;
2589 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
2593 /* I now set the cursor anew in every OnInternalIdle call
2594 as setting the cursor in a parent window also effects the
2595 windows above so that checking for the current cursor is
2598 if (m_wxwindow
&& (m_wxwindow
!= m_widget
))
2600 GdkWindow
*window
= m_wxwindow
->window
;
2602 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2604 if (!g_globalCursor
.Ok())
2605 cursor
= *wxSTANDARD_CURSOR
;
2607 window
= m_widget
->window
;
2608 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
2609 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2612 else if ( m_widget
)
2614 GdkWindow
*window
= m_widget
->window
;
2615 if ( window
&& !GTK_WIDGET_NO_WINDOW(m_widget
) )
2616 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2620 if (wxUpdateUIEvent::CanUpdate(this) && IsShownOnScreen())
2621 UpdateWindowUI(wxUPDATE_UI_FROMIDLE
);
2624 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
2626 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2628 if (width
) (*width
) = m_width
;
2629 if (height
) (*height
) = m_height
;
2632 void wxWindowGTK::DoSetClientSize( int width
, int height
)
2634 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2636 const wxSize size
= GetSize();
2637 const wxSize clientSize
= GetClientSize();
2638 SetSize(width
+ (size
.x
- clientSize
.x
), height
+ (size
.y
- clientSize
.y
));
2641 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
2643 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2650 // if window is scrollable, account for scrollbars
2651 if ( GTK_IS_SCROLLED_WINDOW(m_widget
) )
2653 GtkPolicyType policy
[ScrollDir_Max
];
2654 gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(m_widget
),
2655 &policy
[ScrollDir_Horz
],
2656 &policy
[ScrollDir_Vert
]);
2658 for ( int i
= 0; i
< ScrollDir_Max
; i
++ )
2660 // don't account for the scrollbars we don't have
2661 GtkRange
* const range
= m_scrollBar
[i
];
2665 // nor for the ones we have but don't current show
2666 switch ( policy
[i
] )
2668 case GTK_POLICY_NEVER
:
2669 // never shown so doesn't take any place
2672 case GTK_POLICY_ALWAYS
:
2673 // no checks necessary
2676 case GTK_POLICY_AUTOMATIC
:
2677 // may be shown or not, check
2678 GtkAdjustment
*adj
= gtk_range_get_adjustment(range
);
2679 if ( adj
->upper
<= adj
->page_size
)
2683 GtkScrolledWindowClass
*scroll_class
=
2684 GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2687 gtk_widget_size_request(GTK_WIDGET(range
), &req
);
2688 if (i
== ScrollDir_Horz
)
2689 h
-= req
.height
+ scroll_class
->scrollbar_spacing
;
2691 w
-= req
.width
+ scroll_class
->scrollbar_spacing
;
2695 int border_x
, border_y
;
2696 WX_PIZZA(m_wxwindow
)->get_border_widths(border_x
, border_y
);
2706 if (width
) *width
= w
;
2707 if (height
) *height
= h
;
2710 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
2712 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2716 if (!IsTopLevel() && m_parent
&& m_parent
->m_wxwindow
)
2718 wxPizza
* pizza
= WX_PIZZA(m_parent
->m_wxwindow
);
2719 dx
= pizza
->m_scroll_x
;
2720 dy
= pizza
->m_scroll_y
;
2723 if (m_x
== -1 && m_y
== -1)
2725 GdkWindow
*source
= NULL
;
2727 source
= m_wxwindow
->window
;
2729 source
= m_widget
->window
;
2735 gdk_window_get_origin( source
, &org_x
, &org_y
);
2738 m_parent
->ScreenToClient(&org_x
, &org_y
);
2740 const_cast<wxWindowGTK
*>(this)->m_x
= org_x
;
2741 const_cast<wxWindowGTK
*>(this)->m_y
= org_y
;
2745 if (x
) (*x
) = m_x
- dx
;
2746 if (y
) (*y
) = m_y
- dy
;
2749 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
2751 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2753 if (!m_widget
->window
) return;
2755 GdkWindow
*source
= NULL
;
2757 source
= m_wxwindow
->window
;
2759 source
= m_widget
->window
;
2763 gdk_window_get_origin( source
, &org_x
, &org_y
);
2767 if (GTK_WIDGET_NO_WINDOW (m_widget
))
2769 org_x
+= m_widget
->allocation
.x
;
2770 org_y
+= m_widget
->allocation
.y
;
2777 if (GetLayoutDirection() == wxLayout_RightToLeft
)
2778 *x
= (GetClientSize().x
- *x
) + org_x
;
2786 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
2788 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2790 if (!m_widget
->window
) return;
2792 GdkWindow
*source
= NULL
;
2794 source
= m_wxwindow
->window
;
2796 source
= m_widget
->window
;
2800 gdk_window_get_origin( source
, &org_x
, &org_y
);
2804 if (GTK_WIDGET_NO_WINDOW (m_widget
))
2806 org_x
+= m_widget
->allocation
.x
;
2807 org_y
+= m_widget
->allocation
.y
;
2813 if (GetLayoutDirection() == wxLayout_RightToLeft
)
2814 *x
= (GetClientSize().x
- *x
) - org_x
;
2821 bool wxWindowGTK::Show( bool show
)
2823 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
2825 if (!wxWindowBase::Show(show
))
2831 if (show
&& m_showOnIdle
)
2838 gtk_widget_show(m_widget
);
2840 gtk_widget_hide(m_widget
);
2841 wxShowEvent
eventShow(GetId(), show
);
2842 eventShow
.SetEventObject(this);
2843 HandleWindowEvent(eventShow
);
2849 void wxWindowGTK::DoEnable( bool enable
)
2851 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2853 gtk_widget_set_sensitive( m_widget
, enable
);
2854 if (m_wxwindow
&& (m_wxwindow
!= m_widget
))
2855 gtk_widget_set_sensitive( m_wxwindow
, enable
);
2858 int wxWindowGTK::GetCharHeight() const
2860 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
2862 wxFont font
= GetFont();
2863 wxCHECK_MSG( font
.Ok(), 12, wxT("invalid font") );
2865 PangoContext
* context
= gtk_widget_get_pango_context(m_widget
);
2870 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
2871 PangoLayout
*layout
= pango_layout_new(context
);
2872 pango_layout_set_font_description(layout
, desc
);
2873 pango_layout_set_text(layout
, "H", 1);
2874 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
2876 PangoRectangle rect
;
2877 pango_layout_line_get_extents(line
, NULL
, &rect
);
2879 g_object_unref (layout
);
2881 return (int) PANGO_PIXELS(rect
.height
);
2884 int wxWindowGTK::GetCharWidth() const
2886 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
2888 wxFont font
= GetFont();
2889 wxCHECK_MSG( font
.Ok(), 8, wxT("invalid font") );
2891 PangoContext
* context
= gtk_widget_get_pango_context(m_widget
);
2896 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
2897 PangoLayout
*layout
= pango_layout_new(context
);
2898 pango_layout_set_font_description(layout
, desc
);
2899 pango_layout_set_text(layout
, "g", 1);
2900 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
2902 PangoRectangle rect
;
2903 pango_layout_line_get_extents(line
, NULL
, &rect
);
2905 g_object_unref (layout
);
2907 return (int) PANGO_PIXELS(rect
.width
);
2910 void wxWindowGTK::GetTextExtent( const wxString
& string
,
2914 int *externalLeading
,
2915 const wxFont
*theFont
) const
2917 wxFont fontToUse
= theFont
? *theFont
: GetFont();
2919 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
2928 PangoContext
*context
= NULL
;
2930 context
= gtk_widget_get_pango_context( m_widget
);
2939 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
2940 PangoLayout
*layout
= pango_layout_new(context
);
2941 pango_layout_set_font_description(layout
, desc
);
2943 const wxCharBuffer data
= wxGTK_CONV( string
);
2945 pango_layout_set_text(layout
, data
, strlen(data
));
2948 PangoRectangle rect
;
2949 pango_layout_get_extents(layout
, NULL
, &rect
);
2951 if (x
) (*x
) = (wxCoord
) PANGO_PIXELS(rect
.width
);
2952 if (y
) (*y
) = (wxCoord
) PANGO_PIXELS(rect
.height
);
2955 PangoLayoutIter
*iter
= pango_layout_get_iter(layout
);
2956 int baseline
= pango_layout_iter_get_baseline(iter
);
2957 pango_layout_iter_free(iter
);
2958 *descent
= *y
- PANGO_PIXELS(baseline
);
2960 if (externalLeading
) (*externalLeading
) = 0; // ??
2962 g_object_unref (layout
);
2965 void wxWindowGTK::GTKDisableFocusOutEvent()
2967 g_signal_handlers_block_by_func( m_focusWidget
,
2968 (gpointer
) gtk_window_focus_out_callback
, this);
2971 void wxWindowGTK::GTKEnableFocusOutEvent()
2973 g_signal_handlers_unblock_by_func( m_focusWidget
,
2974 (gpointer
) gtk_window_focus_out_callback
, this);
2977 bool wxWindowGTK::GTKHandleFocusIn()
2979 // Disable default focus handling for custom windows since the default GTK+
2980 // handler issues a repaint
2981 const bool retval
= m_wxwindow
? true : false;
2984 // NB: if there's still unprocessed deferred focus-out event (see
2985 // GTKHandleFocusOut() for explanation), we need to process it first so
2986 // that the order of focus events -- focus-out first, then focus-in
2987 // elsewhere -- is preserved
2988 if ( gs_deferredFocusOut
)
2990 if ( GTKNeedsToFilterSameWindowFocus() &&
2991 gs_deferredFocusOut
== this )
2993 // GTK+ focus changed from this wxWindow back to itself, so don't
2994 // emit any events at all
2995 wxLogTrace(TRACE_FOCUS
,
2996 "filtered out spurious focus change within %s(%p, %s)",
2997 GetClassInfo()->GetClassName(), this, GetLabel());
2998 gs_deferredFocusOut
= NULL
;
3002 // otherwise we need to send focus-out first
3003 wxASSERT_MSG ( gs_deferredFocusOut
!= this,
3004 "GTKHandleFocusIn(GTKFocus_Normal) called even though focus changed back to itself - derived class should handle this" );
3005 GTKHandleDeferredFocusOut();
3009 wxLogTrace(TRACE_FOCUS
,
3010 "handling focus_in event for %s(%p, %s)",
3011 GetClassInfo()->GetClassName(), this, GetLabel());
3014 gtk_im_context_focus_in(m_imData
->context
);
3016 gs_currentFocus
= this;
3017 gs_pendingFocus
= NULL
;
3020 // caret needs to be informed about focus change
3021 wxCaret
*caret
= GetCaret();
3024 caret
->OnSetFocus();
3026 #endif // wxUSE_CARET
3028 // Notify the parent keeping track of focus for the kbd navigation
3029 // purposes that we got it.
3030 wxChildFocusEvent
eventChildFocus(static_cast<wxWindow
*>(this));
3031 GTKProcessEvent(eventChildFocus
);
3033 wxFocusEvent
eventFocus(wxEVT_SET_FOCUS
, GetId());
3034 eventFocus
.SetEventObject(this);
3035 GTKProcessEvent(eventFocus
);
3040 bool wxWindowGTK::GTKHandleFocusOut()
3042 // Disable default focus handling for custom windows since the default GTK+
3043 // handler issues a repaint
3044 const bool retval
= m_wxwindow
? true : false;
3047 // NB: If a control is composed of several GtkWidgets and when focus
3048 // changes from one of them to another within the same wxWindow, we get
3049 // a focus-out event followed by focus-in for another GtkWidget owned
3050 // by the same wx control. We don't want to generate two spurious
3051 // wxEVT_SET_FOCUS events in this case, so we defer sending wx events
3052 // from GTKHandleFocusOut() until we know for sure it's not coming back
3053 // (i.e. in GTKHandleFocusIn() or at idle time).
3054 if ( GTKNeedsToFilterSameWindowFocus() )
3056 wxASSERT_MSG( gs_deferredFocusOut
== NULL
,
3057 "deferred focus out event already pending" );
3058 wxLogTrace(TRACE_FOCUS
,
3059 "deferring focus_out event for %s(%p, %s)",
3060 GetClassInfo()->GetClassName(), this, GetLabel());
3061 gs_deferredFocusOut
= this;
3065 GTKHandleFocusOutNoDeferring();
3070 void wxWindowGTK::GTKHandleFocusOutNoDeferring()
3072 wxLogTrace(TRACE_FOCUS
,
3073 "handling focus_out event for %s(%p, %s)",
3074 GetClassInfo()->GetClassName(), this, GetLabel());
3077 gtk_im_context_focus_out(m_imData
->context
);
3079 if ( gs_currentFocus
!= this )
3081 // Something is terribly wrong, gs_currentFocus is out of sync with the
3082 // real focus. We will reset it to NULL anyway, because after this
3083 // focus-out event is handled, one of the following with happen:
3085 // * either focus will go out of the app altogether, in which case
3086 // gs_currentFocus _should_ be NULL
3088 // * or it goes to another control, in which case focus-in event will
3089 // follow immediately and it will set gs_currentFocus to the right
3091 wxLogDebug("window %s(%p, %s) lost focus even though it didn't have it",
3092 GetClassInfo()->GetClassName(), this, GetLabel());
3094 gs_currentFocus
= NULL
;
3097 // caret needs to be informed about focus change
3098 wxCaret
*caret
= GetCaret();
3101 caret
->OnKillFocus();
3103 #endif // wxUSE_CARET
3105 wxFocusEvent
event( wxEVT_KILL_FOCUS
, GetId() );
3106 event
.SetEventObject( this );
3107 GTKProcessEvent( event
);
3111 void wxWindowGTK::GTKHandleDeferredFocusOut()
3113 // NB: See GTKHandleFocusOut() for explanation. This function is called
3114 // from either GTKHandleFocusIn() or OnInternalIdle() to process
3116 if ( gs_deferredFocusOut
)
3118 wxWindowGTK
*win
= gs_deferredFocusOut
;
3119 gs_deferredFocusOut
= NULL
;
3121 wxLogTrace(TRACE_FOCUS
,
3122 "processing deferred focus_out event for %s(%p, %s)",
3123 win
->GetClassInfo()->GetClassName(), win
, win
->GetLabel());
3125 win
->GTKHandleFocusOutNoDeferring();
3129 void wxWindowGTK::SetFocus()
3131 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3133 // Setting "physical" focus is not immediate in GTK+ and while
3134 // gtk_widget_is_focus ("determines if the widget is the focus widget
3135 // within its toplevel", i.e. returns true for one widget per TLW, not
3136 // globally) returns true immediately after grabbing focus,
3137 // GTK_WIDGET_HAS_FOCUS (which returns true only for the one widget that
3138 // has focus at the moment) takes affect only after the window is shown
3139 // (if it was hidden at the moment of the call) or at the next event loop
3142 // Because we want to FindFocus() call immediately following
3143 // foo->SetFocus() to return foo, we have to keep track of "pending" focus
3145 gs_pendingFocus
= this;
3147 GtkWidget
*widget
= m_wxwindow
? m_wxwindow
: m_focusWidget
;
3149 if ( GTK_IS_CONTAINER(widget
) &&
3150 !GTK_WIDGET_CAN_FOCUS(widget
) )
3152 wxLogTrace(TRACE_FOCUS
,
3153 _T("Setting focus to a child of %s(%p, %s)"),
3154 GetClassInfo()->GetClassName(), this, GetLabel().c_str());
3155 gtk_widget_child_focus(widget
, GTK_DIR_TAB_FORWARD
);
3159 wxLogTrace(TRACE_FOCUS
,
3160 _T("Setting focus to %s(%p, %s)"),
3161 GetClassInfo()->GetClassName(), this, GetLabel().c_str());
3162 gtk_widget_grab_focus(widget
);
3166 void wxWindowGTK::SetCanFocus(bool canFocus
)
3169 GTK_WIDGET_SET_FLAGS(m_widget
, GTK_CAN_FOCUS
);
3171 GTK_WIDGET_UNSET_FLAGS(m_widget
, GTK_CAN_FOCUS
);
3173 if ( m_wxwindow
&& (m_widget
!= m_wxwindow
) )
3176 GTK_WIDGET_SET_FLAGS(m_wxwindow
, GTK_CAN_FOCUS
);
3178 GTK_WIDGET_UNSET_FLAGS(m_wxwindow
, GTK_CAN_FOCUS
);
3182 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3184 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3186 wxWindowGTK
*oldParent
= m_parent
,
3187 *newParent
= (wxWindowGTK
*)newParentBase
;
3189 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3191 if ( !wxWindowBase::Reparent(newParent
) )
3194 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3197 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3199 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3203 if (GTK_WIDGET_VISIBLE (newParent
->m_widget
))
3205 m_showOnIdle
= true;
3206 gtk_widget_hide( m_widget
);
3208 /* insert GTK representation */
3209 newParent
->AddChildGTK(this);
3212 SetLayoutDirection(wxLayout_Default
);
3217 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3219 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3220 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3225 /* insert GTK representation */
3229 void wxWindowGTK::AddChild(wxWindowBase
*child
)
3231 wxWindowBase::AddChild(child
);
3232 m_dirtyTabOrder
= true;
3233 wxTheApp
->WakeUpIdle();
3236 void wxWindowGTK::RemoveChild(wxWindowBase
*child
)
3238 wxWindowBase::RemoveChild(child
);
3239 m_dirtyTabOrder
= true;
3240 wxTheApp
->WakeUpIdle();
3244 wxLayoutDirection
wxWindowGTK::GTKGetLayout(GtkWidget
*widget
)
3246 return gtk_widget_get_direction(widget
) == GTK_TEXT_DIR_RTL
3247 ? wxLayout_RightToLeft
3248 : wxLayout_LeftToRight
;
3252 void wxWindowGTK::GTKSetLayout(GtkWidget
*widget
, wxLayoutDirection dir
)
3254 wxASSERT_MSG( dir
!= wxLayout_Default
, _T("invalid layout direction") );
3256 gtk_widget_set_direction(widget
,
3257 dir
== wxLayout_RightToLeft
? GTK_TEXT_DIR_RTL
3258 : GTK_TEXT_DIR_LTR
);
3261 wxLayoutDirection
wxWindowGTK::GetLayoutDirection() const
3263 return GTKGetLayout(m_widget
);
3266 void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir
)
3268 if ( dir
== wxLayout_Default
)
3270 const wxWindow
*const parent
= GetParent();
3273 // inherit layout from parent.
3274 dir
= parent
->GetLayoutDirection();
3276 else // no parent, use global default layout
3278 dir
= wxTheApp
->GetLayoutDirection();
3282 if ( dir
== wxLayout_Default
)
3285 GTKSetLayout(m_widget
, dir
);
3287 if (m_wxwindow
&& (m_wxwindow
!= m_widget
))
3288 GTKSetLayout(m_wxwindow
, dir
);
3292 wxWindowGTK::AdjustForLayoutDirection(wxCoord x
,
3293 wxCoord
WXUNUSED(width
),
3294 wxCoord
WXUNUSED(widthTotal
)) const
3296 // We now mirror the coordinates of RTL windows in wxPizza
3300 void wxWindowGTK::DoMoveInTabOrder(wxWindow
*win
, WindowOrder move
)
3302 wxWindowBase::DoMoveInTabOrder(win
, move
);
3303 m_dirtyTabOrder
= true;
3304 wxTheApp
->WakeUpIdle();
3307 bool wxWindowGTK::DoNavigateIn(int flags
)
3309 if ( flags
& wxNavigationKeyEvent::WinChange
)
3311 wxFAIL_MSG( _T("not implemented") );
3315 else // navigate inside the container
3317 wxWindow
*parent
= wxGetTopLevelParent((wxWindow
*)this);
3318 wxCHECK_MSG( parent
, false, _T("every window must have a TLW parent") );
3320 GtkDirectionType dir
;
3321 dir
= flags
& wxNavigationKeyEvent::IsForward
? GTK_DIR_TAB_FORWARD
3322 : GTK_DIR_TAB_BACKWARD
;
3325 g_signal_emit_by_name(parent
->m_widget
, "focus", dir
, &rc
);
3331 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3333 // none needed by default
3337 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget
* WXUNUSED(w
))
3339 // nothing to do by default since none is needed
3342 void wxWindowGTK::RealizeTabOrder()
3346 if ( !m_children
.empty() )
3348 // we don't only construct the correct focus chain but also use
3349 // this opportunity to update the mnemonic widgets for the widgets
3352 GList
*chain
= NULL
;
3353 wxWindowGTK
* mnemonicWindow
= NULL
;
3355 for ( wxWindowList::const_iterator i
= m_children
.begin();
3356 i
!= m_children
.end();
3359 wxWindowGTK
*win
= *i
;
3361 if ( mnemonicWindow
)
3363 if ( win
->AcceptsFocusFromKeyboard() )
3365 // wxComboBox et al. needs to focus on on a different
3366 // widget than m_widget, so if the main widget isn't
3367 // focusable try the connect widget
3368 GtkWidget
* w
= win
->m_widget
;
3369 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3371 w
= win
->GetConnectWidget();
3372 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3378 mnemonicWindow
->GTKWidgetDoSetMnemonic(w
);
3379 mnemonicWindow
= NULL
;
3383 else if ( win
->GTKWidgetNeedsMnemonic() )
3385 mnemonicWindow
= win
;
3388 chain
= g_list_prepend(chain
, win
->m_widget
);
3391 chain
= g_list_reverse(chain
);
3393 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow
), chain
);
3398 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow
));
3403 void wxWindowGTK::Raise()
3405 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3407 if (m_wxwindow
&& m_wxwindow
->window
)
3409 gdk_window_raise( m_wxwindow
->window
);
3411 else if (m_widget
->window
)
3413 gdk_window_raise( m_widget
->window
);
3417 void wxWindowGTK::Lower()
3419 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3421 if (m_wxwindow
&& m_wxwindow
->window
)
3423 gdk_window_lower( m_wxwindow
->window
);
3425 else if (m_widget
->window
)
3427 gdk_window_lower( m_widget
->window
);
3431 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3433 if ( !wxWindowBase::SetCursor(cursor
.Ok() ? cursor
: *wxSTANDARD_CURSOR
) )
3441 void wxWindowGTK::GTKUpdateCursor()
3443 wxCursor
cursor(g_globalCursor
.Ok() ? g_globalCursor
: GetCursor());
3446 wxArrayGdkWindows windowsThis
;
3447 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3450 gdk_window_set_cursor(winThis
, cursor
.GetCursor());
3454 const size_t count
= windowsThis
.size();
3455 for ( size_t n
= 0; n
< count
; n
++ )
3457 GdkWindow
*win
= windowsThis
[n
];
3460 wxFAIL_MSG(_T("NULL window returned by GTKGetWindow()?"));
3464 gdk_window_set_cursor(win
, cursor
.GetCursor());
3470 void wxWindowGTK::WarpPointer( int x
, int y
)
3472 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3474 // We provide this function ourselves as it is
3475 // missing in GDK (top of this file).
3477 GdkWindow
*window
= NULL
;
3479 window
= m_wxwindow
->window
;
3481 window
= GetConnectWidget()->window
;
3484 gdk_window_warp_pointer( window
, x
, y
);
3487 wxWindowGTK::ScrollDir
wxWindowGTK::ScrollDirFromRange(GtkRange
*range
) const
3489 // find the scrollbar which generated the event
3490 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
3492 if ( range
== m_scrollBar
[dir
] )
3493 return (ScrollDir
)dir
;
3496 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3498 return ScrollDir_Max
;
3501 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir
, ScrollUnit unit
, int units
)
3503 bool changed
= false;
3504 GtkRange
* range
= m_scrollBar
[dir
];
3505 if ( range
&& units
)
3507 GtkAdjustment
* adj
= range
->adjustment
;
3508 gdouble inc
= unit
== ScrollUnit_Line
? adj
->step_increment
3509 : adj
->page_increment
;
3511 const int posOld
= int(adj
->value
+ 0.5);
3512 gtk_range_set_value(range
, posOld
+ units
*inc
);
3514 changed
= int(adj
->value
+ 0.5) != posOld
;
3520 bool wxWindowGTK::ScrollLines(int lines
)
3522 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Line
, lines
);
3525 bool wxWindowGTK::ScrollPages(int pages
)
3527 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Page
, pages
);
3530 void wxWindowGTK::Refresh(bool WXUNUSED(eraseBackground
),
3535 widget
= m_wxwindow
;
3541 if (!widget
->window
)
3545 gtk_widget_queue_draw(widget
);
3549 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3550 x
= GetClientSize().x
- x
- rect
->width
;
3553 gtk_widget_queue_draw_area(widget
, x
, rect
->y
, rect
->width
, rect
->height
);
3558 r
.width
= rect
->width
;
3559 r
.height
= rect
->height
;
3560 gdk_window_invalidate_rect( m_wxwindow
->window
, &r
, TRUE
);
3565 void wxWindowGTK::Update()
3567 if (m_widget
&& m_widget
->window
)
3569 GdkDisplay
* display
= gtk_widget_get_display(m_widget
);
3570 // Flush everything out to the server, and wait for it to finish.
3571 // This ensures nothing will overwrite the drawing we are about to do.
3572 gdk_display_sync(display
);
3574 gdk_window_process_updates(m_widget
->window
, true);
3576 // Flush again, but no need to wait for it to finish
3577 gdk_display_flush(display
);
3581 bool wxWindowGTK::DoIsExposed( int x
, int y
) const
3583 return m_updateRegion
.Contains(x
, y
) != wxOutRegion
;
3586 bool wxWindowGTK::DoIsExposed( int x
, int y
, int w
, int h
) const
3588 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3589 return m_updateRegion
.Contains(x
-w
, y
, w
, h
) != wxOutRegion
;
3591 return m_updateRegion
.Contains(x
, y
, w
, h
) != wxOutRegion
;
3594 void wxWindowGTK::GtkSendPaintEvents()
3598 m_updateRegion
.Clear();
3602 // Clip to paint region in wxClientDC
3603 m_clipPaintRegion
= true;
3605 m_nativeUpdateRegion
= m_updateRegion
;
3607 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3609 // Transform m_updateRegion under RTL
3610 m_updateRegion
.Clear();
3613 gdk_drawable_get_size(m_wxwindow
->window
, &width
, NULL
);
3615 wxRegionIterator
upd( m_nativeUpdateRegion
);
3619 rect
.x
= upd
.GetX();
3620 rect
.y
= upd
.GetY();
3621 rect
.width
= upd
.GetWidth();
3622 rect
.height
= upd
.GetHeight();
3624 rect
.x
= width
- rect
.x
- rect
.width
;
3625 m_updateRegion
.Union( rect
);
3631 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM
))
3633 // find ancestor from which to steal background
3634 wxWindow
*parent
= wxGetTopLevelParent((wxWindow
*)this);
3636 parent
= (wxWindow
*)this;
3638 if (GTK_WIDGET_MAPPED(parent
->m_widget
))
3640 wxRegionIterator
upd( m_nativeUpdateRegion
);
3644 rect
.x
= upd
.GetX();
3645 rect
.y
= upd
.GetY();
3646 rect
.width
= upd
.GetWidth();
3647 rect
.height
= upd
.GetHeight();
3649 gtk_paint_flat_box( parent
->m_widget
->style
,
3651 (GtkStateType
)GTK_WIDGET_STATE(m_wxwindow
),
3664 wxWindowDC
dc( (wxWindow
*)this );
3665 dc
.SetDeviceClippingRegion( m_updateRegion
);
3667 // Work around gtk-qt <= 0.60 bug whereby the window colour
3669 if (GetBackgroundStyle() == wxBG_STYLE_COLOUR
&& GetBackgroundColour().Ok() && wxSystemOptions::GetOptionInt(wxT("gtk.window.force-background-colour")) == 1)
3671 dc
.SetBackground(wxBrush(GetBackgroundColour()));
3675 wxEraseEvent
erase_event( GetId(), &dc
);
3676 erase_event
.SetEventObject( this );
3678 HandleWindowEvent(erase_event
);
3681 wxNcPaintEvent
nc_paint_event( GetId() );
3682 nc_paint_event
.SetEventObject( this );
3683 HandleWindowEvent( nc_paint_event
);
3685 wxPaintEvent
paint_event( GetId() );
3686 paint_event
.SetEventObject( this );
3687 HandleWindowEvent( paint_event
);
3689 m_clipPaintRegion
= false;
3691 m_updateRegion
.Clear();
3692 m_nativeUpdateRegion
.Clear();
3695 void wxWindowGTK::SetDoubleBuffered( bool on
)
3697 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3700 gtk_widget_set_double_buffered( m_wxwindow
, on
);
3703 bool wxWindowGTK::IsDoubleBuffered() const
3705 return GTK_WIDGET_DOUBLE_BUFFERED( m_wxwindow
);
3708 void wxWindowGTK::ClearBackground()
3710 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3714 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3716 wxWindowBase::DoSetToolTip(tip
);
3719 m_tooltip
->GTKApply( (wxWindow
*)this );
3722 void wxWindowGTK::GTKApplyToolTip( GtkTooltips
*tips
, const gchar
*tip
)
3724 gtk_tooltips_set_tip(tips
, GetConnectWidget(), tip
, NULL
);
3726 #endif // wxUSE_TOOLTIPS
3728 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
3730 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3732 if (!wxWindowBase::SetBackgroundColour(colour
))
3737 // We need the pixel value e.g. for background clearing.
3738 m_backgroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3741 // apply style change (forceStyle=true so that new style is applied
3742 // even if the bg colour changed from valid to wxNullColour)
3743 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM
)
3744 ApplyWidgetStyle(true);
3749 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
3751 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3753 if (!wxWindowBase::SetForegroundColour(colour
))
3760 // We need the pixel value e.g. for background clearing.
3761 m_foregroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3764 // apply style change (forceStyle=true so that new style is applied
3765 // even if the bg colour changed from valid to wxNullColour):
3766 ApplyWidgetStyle(true);
3771 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
3773 return gtk_widget_get_pango_context( m_widget
);
3776 GtkRcStyle
*wxWindowGTK::CreateWidgetStyle(bool forceStyle
)
3778 // do we need to apply any changes at all?
3781 !m_foregroundColour
.Ok() && !m_backgroundColour
.Ok() )
3786 GtkRcStyle
*style
= gtk_rc_style_new();
3791 pango_font_description_copy( m_font
.GetNativeFontInfo()->description
);
3794 int flagsNormal
= 0,
3797 flagsInsensitive
= 0;
3799 if ( m_foregroundColour
.Ok() )
3801 const GdkColor
*fg
= m_foregroundColour
.GetColor();
3803 style
->fg
[GTK_STATE_NORMAL
] =
3804 style
->text
[GTK_STATE_NORMAL
] = *fg
;
3805 flagsNormal
|= GTK_RC_FG
| GTK_RC_TEXT
;
3807 style
->fg
[GTK_STATE_PRELIGHT
] =
3808 style
->text
[GTK_STATE_PRELIGHT
] = *fg
;
3809 flagsPrelight
|= GTK_RC_FG
| GTK_RC_TEXT
;
3811 style
->fg
[GTK_STATE_ACTIVE
] =
3812 style
->text
[GTK_STATE_ACTIVE
] = *fg
;
3813 flagsActive
|= GTK_RC_FG
| GTK_RC_TEXT
;
3816 if ( m_backgroundColour
.Ok() )
3818 const GdkColor
*bg
= m_backgroundColour
.GetColor();
3820 style
->bg
[GTK_STATE_NORMAL
] =
3821 style
->base
[GTK_STATE_NORMAL
] = *bg
;
3822 flagsNormal
|= GTK_RC_BG
| GTK_RC_BASE
;
3824 style
->bg
[GTK_STATE_PRELIGHT
] =
3825 style
->base
[GTK_STATE_PRELIGHT
] = *bg
;
3826 flagsPrelight
|= GTK_RC_BG
| GTK_RC_BASE
;
3828 style
->bg
[GTK_STATE_ACTIVE
] =
3829 style
->base
[GTK_STATE_ACTIVE
] = *bg
;
3830 flagsActive
|= GTK_RC_BG
| GTK_RC_BASE
;
3832 style
->bg
[GTK_STATE_INSENSITIVE
] =
3833 style
->base
[GTK_STATE_INSENSITIVE
] = *bg
;
3834 flagsInsensitive
|= GTK_RC_BG
| GTK_RC_BASE
;
3837 style
->color_flags
[GTK_STATE_NORMAL
] = (GtkRcFlags
)flagsNormal
;
3838 style
->color_flags
[GTK_STATE_PRELIGHT
] = (GtkRcFlags
)flagsPrelight
;
3839 style
->color_flags
[GTK_STATE_ACTIVE
] = (GtkRcFlags
)flagsActive
;
3840 style
->color_flags
[GTK_STATE_INSENSITIVE
] = (GtkRcFlags
)flagsInsensitive
;
3845 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle
)
3847 GtkRcStyle
*style
= CreateWidgetStyle(forceStyle
);
3850 DoApplyWidgetStyle(style
);
3851 gtk_rc_style_unref(style
);
3854 // Style change may affect GTK+'s size calculation:
3855 InvalidateBestSize();
3858 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle
*style
)
3860 wxSuspendStyleEvents
s(static_cast<wxWindow
*>(this));
3863 gtk_widget_modify_style(m_wxwindow
, style
);
3865 gtk_widget_modify_style(m_widget
, style
);
3868 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style
)
3870 wxWindowBase::SetBackgroundStyle(style
);
3872 if (style
== wxBG_STYLE_CUSTOM
)
3877 window
= m_wxwindow
->window
;
3881 GtkWidget
* const w
= GetConnectWidget();
3882 window
= w
? w
->window
: NULL
;
3887 // Make sure GDK/X11 doesn't refresh the window
3889 gdk_window_set_back_pixmap( window
, None
, False
);
3891 Display
* display
= GDK_WINDOW_DISPLAY(window
);
3894 m_needsStyleChange
= false;
3896 else // window not realized yet
3898 // Do in OnIdle, because the window is not yet available
3899 m_needsStyleChange
= true;
3902 // Don't apply widget style, or we get a grey background
3906 // apply style change (forceStyle=true so that new style is applied
3907 // even if the bg colour changed from valid to wxNullColour):
3908 ApplyWidgetStyle(true);
3913 // ----------------------------------------------------------------------------
3914 // Pop-up menu stuff
3915 // ----------------------------------------------------------------------------
3917 #if wxUSE_MENUS_NATIVE
3919 static void SetInvokingWindow( wxMenu
*menu
, wxWindow
* win
)
3921 menu
->SetInvokingWindow( win
);
3923 wxMenuItemList::compatibility_iterator node
= menu
->GetMenuItems().GetFirst();
3926 wxMenuItem
*menuitem
= node
->GetData();
3927 if (menuitem
->IsSubMenu())
3929 SetInvokingWindow( menuitem
->GetSubMenu(), win
);
3932 node
= node
->GetNext();
3938 void wxPopupMenuPositionCallback( GtkMenu
*menu
,
3940 gboolean
* WXUNUSED(whatever
),
3941 gpointer user_data
)
3943 // ensure that the menu appears entirely on screen
3945 gtk_widget_get_child_requisition(GTK_WIDGET(menu
), &req
);
3947 wxSize sizeScreen
= wxGetDisplaySize();
3948 wxPoint
*pos
= (wxPoint
*)user_data
;
3950 gint xmax
= sizeScreen
.x
- req
.width
,
3951 ymax
= sizeScreen
.y
- req
.height
;
3953 *x
= pos
->x
< xmax
? pos
->x
: xmax
;
3954 *y
= pos
->y
< ymax
? pos
->y
: ymax
;
3958 bool wxWindowGTK::DoPopupMenu( wxMenu
*menu
, int x
, int y
)
3960 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3962 wxCHECK_MSG( menu
!= NULL
, false, wxT("invalid popup-menu") );
3964 SetInvokingWindow( menu
, this );
3970 GtkMenuPositionFunc posfunc
;
3971 if ( x
== -1 && y
== -1 )
3973 // use GTK's default positioning algorithm
3979 pos
= ClientToScreen(wxPoint(x
, y
));
3981 posfunc
= wxPopupMenuPositionCallback
;
3984 menu
->m_popupShown
= true;
3986 GTK_MENU(menu
->m_menu
),
3987 NULL
, // parent menu shell
3988 NULL
, // parent menu item
3989 posfunc
, // function to position it
3990 userdata
, // client data
3991 0, // button used to activate it
3992 gtk_get_current_event_time()
3995 while (menu
->m_popupShown
)
3997 gtk_main_iteration();
4003 #endif // wxUSE_MENUS_NATIVE
4005 #if wxUSE_DRAG_AND_DROP
4007 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
4009 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4011 GtkWidget
*dnd_widget
= GetConnectWidget();
4013 if (m_dropTarget
) m_dropTarget
->GtkUnregisterWidget( dnd_widget
);
4015 if (m_dropTarget
) delete m_dropTarget
;
4016 m_dropTarget
= dropTarget
;
4018 if (m_dropTarget
) m_dropTarget
->GtkRegisterWidget( dnd_widget
);
4021 #endif // wxUSE_DRAG_AND_DROP
4023 GtkWidget
* wxWindowGTK::GetConnectWidget()
4025 GtkWidget
*connect_widget
= m_widget
;
4026 if (m_wxwindow
) connect_widget
= m_wxwindow
;
4028 return connect_widget
;
4031 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow
*window
) const
4033 wxArrayGdkWindows windowsThis
;
4034 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
4036 return winThis
? window
== winThis
4037 : windowsThis
.Index(window
) != wxNOT_FOUND
;
4040 GdkWindow
*wxWindowGTK::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const
4042 return m_wxwindow
? m_wxwindow
->window
: m_widget
->window
;
4045 bool wxWindowGTK::SetFont( const wxFont
&font
)
4047 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
4049 if (!wxWindowBase::SetFont(font
))
4052 // apply style change (forceStyle=true so that new style is applied
4053 // even if the font changed from valid to wxNullFont):
4054 ApplyWidgetStyle(true);
4059 void wxWindowGTK::DoCaptureMouse()
4061 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4063 GdkWindow
*window
= NULL
;
4065 window
= m_wxwindow
->window
;
4067 window
= GetConnectWidget()->window
;
4069 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4071 const wxCursor
* cursor
= &m_cursor
;
4073 cursor
= wxSTANDARD_CURSOR
;
4075 gdk_pointer_grab( window
, FALSE
,
4077 (GDK_BUTTON_PRESS_MASK
|
4078 GDK_BUTTON_RELEASE_MASK
|
4079 GDK_POINTER_MOTION_HINT_MASK
|
4080 GDK_POINTER_MOTION_MASK
),
4082 cursor
->GetCursor(),
4083 (guint32
)GDK_CURRENT_TIME
);
4084 g_captureWindow
= this;
4085 g_captureWindowHasMouse
= true;
4088 void wxWindowGTK::DoReleaseMouse()
4090 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4092 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4094 g_captureWindow
= NULL
;
4096 GdkWindow
*window
= NULL
;
4098 window
= m_wxwindow
->window
;
4100 window
= GetConnectWidget()->window
;
4105 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4108 void wxWindowGTK::GTKReleaseMouseAndNotify()
4111 wxMouseCaptureLostEvent
evt(GetId());
4112 evt
.SetEventObject( this );
4113 HandleWindowEvent( evt
);
4117 wxWindow
*wxWindowBase::GetCapture()
4119 return (wxWindow
*)g_captureWindow
;
4122 bool wxWindowGTK::IsRetained() const
4127 void wxWindowGTK::SetScrollbar(int orient
,
4131 bool WXUNUSED(update
))
4133 const int dir
= ScrollDirFromOrient(orient
);
4134 GtkRange
* const sb
= m_scrollBar
[dir
];
4135 wxCHECK_RET( sb
, _T("this window is not scrollable") );
4139 // GtkRange requires upper > lower
4144 GtkAdjustment
* const adj
= sb
->adjustment
;
4145 adj
->step_increment
= 1;
4146 adj
->page_increment
=
4147 adj
->page_size
= thumbVisible
;
4150 g_signal_handlers_block_by_func(
4151 sb
, (void*)gtk_scrollbar_value_changed
, this);
4153 gtk_range_set_range(sb
, 0, range
);
4154 m_scrollPos
[dir
] = sb
->adjustment
->value
;
4156 g_signal_handlers_unblock_by_func(
4157 sb
, (void*)gtk_scrollbar_value_changed
, this);
4160 void wxWindowGTK::SetScrollPos(int orient
, int pos
, bool WXUNUSED(refresh
))
4162 const int dir
= ScrollDirFromOrient(orient
);
4163 GtkRange
* const sb
= m_scrollBar
[dir
];
4164 wxCHECK_RET( sb
, _T("this window is not scrollable") );
4166 // This check is more than an optimization. Without it, the slider
4167 // will not move smoothly while tracking when using wxScrollHelper.
4168 if (GetScrollPos(orient
) != pos
)
4170 g_signal_handlers_block_by_func(
4171 sb
, (void*)gtk_scrollbar_value_changed
, this);
4173 gtk_range_set_value(sb
, pos
);
4174 m_scrollPos
[dir
] = sb
->adjustment
->value
;
4176 g_signal_handlers_unblock_by_func(
4177 sb
, (void*)gtk_scrollbar_value_changed
, this);
4181 int wxWindowGTK::GetScrollThumb(int orient
) const
4183 GtkRange
* const sb
= m_scrollBar
[ScrollDirFromOrient(orient
)];
4184 wxCHECK_MSG( sb
, 0, _T("this window is not scrollable") );
4186 return wxRound(sb
->adjustment
->page_size
);
4189 int wxWindowGTK::GetScrollPos( int orient
) const
4191 GtkRange
* const sb
= m_scrollBar
[ScrollDirFromOrient(orient
)];
4192 wxCHECK_MSG( sb
, 0, _T("this window is not scrollable") );
4194 return wxRound(sb
->adjustment
->value
);
4197 int wxWindowGTK::GetScrollRange( int orient
) const
4199 GtkRange
* const sb
= m_scrollBar
[ScrollDirFromOrient(orient
)];
4200 wxCHECK_MSG( sb
, 0, _T("this window is not scrollable") );
4202 return wxRound(sb
->adjustment
->upper
);
4205 // Determine if increment is the same as +/-x, allowing for some small
4206 // difference due to possible inexactness in floating point arithmetic
4207 static inline bool IsScrollIncrement(double increment
, double x
)
4209 wxASSERT(increment
> 0);
4210 const double tolerance
= 1.0 / 1024;
4211 return fabs(increment
- fabs(x
)) < tolerance
;
4214 wxEventType
wxWindowGTK::GTKGetScrollEventType(GtkRange
* range
)
4216 wxASSERT(range
== m_scrollBar
[0] || range
== m_scrollBar
[1]);
4218 const int barIndex
= range
== m_scrollBar
[1];
4219 GtkAdjustment
* adj
= range
->adjustment
;
4221 const int value
= wxRound(adj
->value
);
4223 // save previous position
4224 const double oldPos
= m_scrollPos
[barIndex
];
4225 // update current position
4226 m_scrollPos
[barIndex
] = adj
->value
;
4227 // If event should be ignored, or integral position has not changed
4228 if (!m_hasVMT
|| g_blockEventsOnDrag
|| value
== wxRound(oldPos
))
4233 wxEventType eventType
= wxEVT_SCROLL_THUMBTRACK
;
4236 // Difference from last change event
4237 const double diff
= adj
->value
- oldPos
;
4238 const bool isDown
= diff
> 0;
4240 if (IsScrollIncrement(adj
->step_increment
, diff
))
4242 eventType
= isDown
? wxEVT_SCROLL_LINEDOWN
: wxEVT_SCROLL_LINEUP
;
4244 else if (IsScrollIncrement(adj
->page_increment
, diff
))
4246 eventType
= isDown
? wxEVT_SCROLL_PAGEDOWN
: wxEVT_SCROLL_PAGEUP
;
4248 else if (m_mouseButtonDown
)
4250 // Assume track event
4251 m_isScrolling
= true;
4257 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4259 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4261 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4263 // No scrolling requested.
4264 if ((dx
== 0) && (dy
== 0)) return;
4266 m_clipPaintRegion
= true;
4268 WX_PIZZA(m_wxwindow
)->scroll(dx
, dy
);
4270 m_clipPaintRegion
= false;
4273 bool restoreCaret
= (GetCaret() != NULL
&& GetCaret()->IsVisible());
4276 wxRect
caretRect(GetCaret()->GetPosition(), GetCaret()->GetSize());
4278 caretRect
.width
+= dx
;
4281 caretRect
.x
+= dx
; caretRect
.width
-= dx
;
4284 caretRect
.height
+= dy
;
4287 caretRect
.y
+= dy
; caretRect
.height
-= dy
;
4290 RefreshRect(caretRect
);
4292 #endif // wxUSE_CARET
4295 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget
* w
, int wxstyle
)
4297 //RN: Note that static controls usually have no border on gtk, so maybe
4298 //it makes sense to treat that as simply no border at the wx level
4300 if (!(wxstyle
& wxNO_BORDER
) && !(wxstyle
& wxBORDER_STATIC
))
4302 GtkShadowType gtkstyle
;
4304 if(wxstyle
& wxBORDER_RAISED
)
4305 gtkstyle
= GTK_SHADOW_OUT
;
4306 else if (wxstyle
& wxBORDER_SUNKEN
)
4307 gtkstyle
= GTK_SHADOW_IN
;
4310 else if (wxstyle
& wxBORDER_DOUBLE
)
4311 gtkstyle
= GTK_SHADOW_ETCHED_IN
;
4314 gtkstyle
= GTK_SHADOW_IN
;
4316 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w
),
4321 void wxWindowGTK::SetWindowStyleFlag( long style
)
4323 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4324 wxWindowBase::SetWindowStyleFlag(style
);
4327 // Find the wxWindow at the current mouse position, also returning the mouse
4329 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4331 pt
= wxGetMousePosition();
4332 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4336 // Get the current mouse position.
4337 wxPoint
wxGetMousePosition()
4339 /* This crashes when used within wxHelpContext,
4340 so we have to use the X-specific implementation below.
4342 GdkModifierType *mask;
4343 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4345 return wxPoint(x, y);
4349 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4351 Display
*display
= windowAtPtr
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY();
4352 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4353 Window rootReturn
, childReturn
;
4354 int rootX
, rootY
, winX
, winY
;
4355 unsigned int maskReturn
;
4357 XQueryPointer (display
,
4361 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4362 return wxPoint(rootX
, rootY
);
4366 GdkWindow
* wxWindowGTK::GTKGetDrawingWindow() const
4368 GdkWindow
* window
= NULL
;
4370 window
= m_wxwindow
->window
;
4374 // ----------------------------------------------------------------------------
4376 // ----------------------------------------------------------------------------
4381 // this is called if we attempted to freeze unrealized widget when it finally
4382 // is realized (and so can be frozen):
4383 static void wx_frozen_widget_realize(GtkWidget
* w
, void* WXUNUSED(data
))
4385 wxASSERT( w
&& !GTK_WIDGET_NO_WINDOW(w
) );
4386 wxASSERT( GTK_WIDGET_REALIZED(w
) );
4388 g_signal_handlers_disconnect_by_func
4391 (void*)wx_frozen_widget_realize
,
4395 gdk_window_freeze_updates(w
->window
);
4400 void wxWindowGTK::GTKFreezeWidget(GtkWidget
*w
)
4402 if ( !w
|| GTK_WIDGET_NO_WINDOW(w
) )
4403 return; // window-less widget, cannot be frozen
4405 if ( !GTK_WIDGET_REALIZED(w
) )
4407 // we can't thaw unrealized widgets because they don't have GdkWindow,
4408 // so set it up to be done immediately after realization:
4409 g_signal_connect_after
4413 G_CALLBACK(wx_frozen_widget_realize
),
4419 gdk_window_freeze_updates(w
->window
);
4422 void wxWindowGTK::GTKThawWidget(GtkWidget
*w
)
4424 if ( !w
|| GTK_WIDGET_NO_WINDOW(w
) )
4425 return; // window-less widget, cannot be frozen
4427 if ( !GTK_WIDGET_REALIZED(w
) )
4429 // the widget wasn't realized yet, no need to thaw
4430 g_signal_handlers_disconnect_by_func
4433 (void*)wx_frozen_widget_realize
,
4439 gdk_window_thaw_updates(w
->window
);
4442 void wxWindowGTK::DoFreeze()
4444 GTKFreezeWidget(m_widget
);
4445 if ( m_wxwindow
&& m_widget
!= m_wxwindow
)
4446 GTKFreezeWidget(m_wxwindow
);
4449 void wxWindowGTK::DoThaw()
4451 GTKThawWidget(m_widget
);
4452 if ( m_wxwindow
&& m_widget
!= m_wxwindow
)
4453 GTKThawWidget(m_wxwindow
);