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"
25 #include "wx/dcclient.h"
27 #include "wx/dialog.h"
28 #include "wx/settings.h"
29 #include "wx/msgdlg.h"
30 #include "wx/textctrl.h"
31 #include "wx/toolbar.h"
32 #include "wx/combobox.h"
33 #include "wx/layout.h"
34 #include "wx/statusbr.h"
36 #include "wx/module.h"
39 #if wxUSE_DRAG_AND_DROP
44 #include "wx/tooltip.h"
51 #include "wx/fontutil.h"
54 #include "wx/thread.h"
59 // FIXME: Due to a hack we use GtkCombo in here, which is deprecated since gtk2.3.0
60 #include <gtk/gtkversion.h>
61 #if defined(GTK_DISABLE_DEPRECATED) && GTK_CHECK_VERSION(2,3,0)
62 #undef GTK_DISABLE_DEPRECATED
65 #include "wx/gtk/private.h"
66 #include <gdk/gdkprivate.h>
67 #include <gdk/gdkkeysyms.h>
71 #include <gtk/gtkprivate.h>
73 #include "wx/gtk/win_gtk.h"
75 #include <pango/pangox.h>
81 //-----------------------------------------------------------------------------
82 // documentation on internals
83 //-----------------------------------------------------------------------------
86 I have been asked several times about writing some documentation about
87 the GTK port of wxWidgets, especially its internal structures. Obviously,
88 you cannot understand wxGTK without knowing a little about the GTK, but
89 some more information about what the wxWindow, which is the base class
90 for all other window classes, does seems required as well.
94 What does wxWindow do? It contains the common interface for the following
95 jobs of its descendants:
97 1) Define the rudimentary behaviour common to all window classes, such as
98 resizing, intercepting user input (so as to make it possible to use these
99 events for special purposes in a derived class), window names etc.
101 2) Provide the possibility to contain and manage children, if the derived
102 class is allowed to contain children, which holds true for those window
103 classes which do not display a native GTK widget. To name them, these
104 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
105 work classes are a special case and are handled a bit differently from
106 the rest. The same holds true for the wxNotebook class.
108 3) Provide the possibility to draw into a client area of a window. This,
109 too, only holds true for classes that do not display a native GTK widget
112 4) Provide the entire mechanism for scrolling widgets. This actual inter-
113 face for this is usually in wxScrolledWindow, but the GTK implementation
116 5) A multitude of helper or extra methods for special purposes, such as
117 Drag'n'Drop, managing validators etc.
119 6) Display a border (sunken, raised, simple or none).
121 Normally one might expect, that one wxWidgets window would always correspond
122 to one GTK widget. Under GTK, there is no such all-round widget that has all
123 the functionality. Moreover, the GTK defines a client area as a different
124 widget from the actual widget you are handling. Last but not least some
125 special classes (e.g. wxFrame) handle different categories of widgets and
126 still have the possibility to draw something in the client area.
127 It was therefore required to write a special purpose GTK widget, that would
128 represent a client area in the sense of wxWidgets capable to do the jobs
129 2), 3) and 4). I have written this class and it resides in win_gtk.c of
132 All windows must have a widget, with which they interact with other under-
133 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
134 the wxWindow class has a member variable called m_widget which holds a
135 pointer to this widget. When the window class represents a GTK native widget,
136 this is (in most cases) the only GTK widget the class manages. E.g. the
137 wxStaticText class handles only a GtkLabel widget a pointer to which you
138 can find in m_widget (defined in wxWindow)
140 When the class has a client area for drawing into and for containing children
141 it has to handle the client area widget (of the type GtkPizza, defined in
142 win_gtk.c), but there could be any number of widgets, handled by a class
143 The common rule for all windows is only, that the widget that interacts with
144 the rest of GTK must be referenced in m_widget and all other widgets must be
145 children of this widget on the GTK level. The top-most widget, which also
146 represents the client area, must be in the m_wxwindow field and must be of
149 As I said, the window classes that display a GTK native widget only have
150 one widget, so in the case of e.g. the wxButton class m_widget holds a
151 pointer to a GtkButton widget. But windows with client areas (for drawing
152 and children) have a m_widget field that is a pointer to a GtkScrolled-
153 Window and a m_wxwindow field that is pointer to a GtkPizza and this
154 one is (in the GTK sense) a child of the GtkScrolledWindow.
156 If the m_wxwindow field is set, then all input to this widget is inter-
157 cepted and sent to the wxWidgets class. If not, all input to the widget
158 that gets pointed to by m_widget gets intercepted and sent to the class.
162 The design of scrolling in wxWidgets is markedly different from that offered
163 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
164 clicking on a scrollbar belonging to scrolled window will inevitably move
165 the window. In wxWidgets, the scrollbar will only emit an event, send this
166 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
167 which actually moves the window and its sub-windows. Note that GtkPizza
168 memorizes how much it has been scrolled but that wxWidgets forgets this
169 so that the two coordinates systems have to be kept in synch. This is done
170 in various places using the pizza->xoffset and pizza->yoffset values.
174 Singularly the most broken code in GTK is the code that is supposed to
175 inform subwindows (child windows) about new positions. Very often, duplicate
176 events are sent without changes in size or position, equally often no
177 events are sent at all (All this is due to a bug in the GtkContainer code
178 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
179 GTK's own system and it simply waits for size events for toplevel windows
180 and then iterates down the respective size events to all window. This has
181 the disadvantage that windows might get size events before the GTK widget
182 actually has the reported size. This doesn't normally pose any problem, but
183 the OpenGL drawing routines rely on correct behaviour. Therefore, I have
184 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
185 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
186 window that is used for OpenGL output really has that size (as reported by
191 If someone at some point of time feels the immense desire to have a look at,
192 change or attempt to optimise the Refresh() logic, this person will need an
193 intimate understanding of what "draw" and "expose" events are and what
194 they are used for, in particular when used in connection with GTK's
195 own windowless widgets. Beware.
199 Cursors, too, have been a constant source of pleasure. The main difficulty
200 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
201 for the parent. To prevent this from doing too much harm, I use idle time
202 to set the cursor over and over again, starting from the toplevel windows
203 and ending with the youngest generation (speaking of parent and child windows).
204 Also don't forget that cursors (like much else) are connected to GdkWindows,
205 not GtkWidgets and that the "window" field of a GtkWidget might very well
206 point to the GdkWindow of the parent widget (-> "window-less widget") and
207 that the two obviously have very different meanings.
211 //-----------------------------------------------------------------------------
213 //-----------------------------------------------------------------------------
215 extern bool g_blockEventsOnDrag
;
216 extern bool g_blockEventsOnScroll
;
217 extern wxCursor g_globalCursor
;
219 // mouse capture state: the window which has it and if the mouse is currently
221 static wxWindowGTK
*g_captureWindow
= (wxWindowGTK
*) NULL
;
222 static bool g_captureWindowHasMouse
= false;
224 wxWindowGTK
*g_focusWindow
= (wxWindowGTK
*) NULL
;
226 // the last window which had the focus - this is normally never NULL (except
227 // if we never had focus at all) as even when g_focusWindow is NULL it still
228 // keeps its previous value
229 wxWindowGTK
*g_focusWindowLast
= (wxWindowGTK
*) NULL
;
231 // If a window get the focus set but has not been realized
232 // yet, defer setting the focus to idle time.
233 wxWindowGTK
*g_delayedFocus
= (wxWindowGTK
*) NULL
;
235 extern bool g_mainThreadLocked
;
237 //-----------------------------------------------------------------------------
239 //-----------------------------------------------------------------------------
244 # define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance");
246 # define DEBUG_MAIN_THREAD
249 #define DEBUG_MAIN_THREAD
252 // the trace mask used for the focus debugging messages
253 #define TRACE_FOCUS _T("focus")
255 //-----------------------------------------------------------------------------
256 // missing gdk functions
257 //-----------------------------------------------------------------------------
260 gdk_window_warp_pointer (GdkWindow
*window
,
265 window
= gdk_get_default_root_window();
267 if (!GDK_WINDOW_DESTROYED(window
))
269 XWarpPointer (GDK_WINDOW_XDISPLAY(window
),
270 None
, /* not source window -> move from anywhere */
271 GDK_WINDOW_XID(window
), /* dest window */
272 0, 0, 0, 0, /* not source window -> move from anywhere */
277 //-----------------------------------------------------------------------------
278 // local code (see below)
279 //-----------------------------------------------------------------------------
281 // returns the child of win which currently has focus or NULL if not found
283 // Note: can't be static, needed by textctrl.cpp.
284 wxWindow
*wxFindFocusedChild(wxWindowGTK
*win
)
286 wxWindow
*winFocus
= wxWindowGTK::FindFocus();
288 return (wxWindow
*)NULL
;
290 if ( winFocus
== win
)
291 return (wxWindow
*)win
;
293 for ( wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
295 node
= node
->GetNext() )
297 wxWindow
*child
= wxFindFocusedChild(node
->GetData());
302 return (wxWindow
*)NULL
;
305 static void GetScrollbarWidth(GtkWidget
* widget
, int& w
, int& h
)
307 GtkScrolledWindow
* scroll_window
= GTK_SCROLLED_WINDOW(widget
);
308 GtkScrolledWindowClass
* scroll_class
= GTK_SCROLLED_WINDOW_CLASS(GTK_OBJECT_GET_CLASS(scroll_window
));
309 GtkRequisition scroll_req
;
312 if (scroll_window
->vscrollbar_visible
)
314 scroll_req
.width
= 2;
315 scroll_req
.height
= 2;
316 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
317 (scroll_window
->vscrollbar
, &scroll_req
);
318 w
= scroll_req
.width
+
319 scroll_class
->scrollbar_spacing
;
323 if (scroll_window
->hscrollbar_visible
)
325 scroll_req
.width
= 2;
326 scroll_req
.height
= 2;
327 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
328 (scroll_window
->hscrollbar
, &scroll_req
);
329 h
= scroll_req
.height
+
330 scroll_class
->scrollbar_spacing
;
334 static void draw_frame( GtkWidget
*widget
, wxWindowGTK
*win
)
336 // wxUniversal widgets draw the borders and scrollbars themselves
337 #ifndef __WXUNIVERSAL__
343 if (GTK_WIDGET_NO_WINDOW (widget
))
345 dx
+= widget
->allocation
.x
;
346 dy
+= widget
->allocation
.y
;
354 if (win
->m_hasScrolling
)
356 GetScrollbarWidth(widget
, dw
, dh
);
358 if (win
->GetLayoutDirection() == wxLayout_RightToLeft
)
360 // This is actually wrong for old GTK+ version
361 // which do not display the scrollbar on the
367 int w
= widget
->allocation
.width
-dw
;
368 int h
= widget
->allocation
.height
-dh
;
370 if (win
->HasFlag(wxRAISED_BORDER
))
372 gtk_paint_shadow (widget
->style
,
376 NULL
, NULL
, NULL
, // FIXME: No clipping?
381 if (win
->HasFlag(wxSUNKEN_BORDER
))
383 gtk_paint_shadow (widget
->style
,
387 NULL
, NULL
, NULL
, // FIXME: No clipping?
392 if (win
->HasFlag(wxSIMPLE_BORDER
))
395 gc
= gdk_gc_new( widget
->window
);
396 gdk_gc_set_foreground( gc
, &widget
->style
->black
);
397 gdk_draw_rectangle( widget
->window
, gc
, FALSE
, x
, y
, w
-1, h
-1 );
401 #endif // __WXUNIVERSAL__
404 //-----------------------------------------------------------------------------
405 // "expose_event" of m_widget
406 //-----------------------------------------------------------------------------
410 gtk_window_own_expose_callback( GtkWidget
*widget
,
411 GdkEventExpose
*gdk_event
,
414 if (gdk_event
->count
== 0)
415 draw_frame(widget
, win
);
420 //-----------------------------------------------------------------------------
421 // "size_request" of m_widget
422 //-----------------------------------------------------------------------------
424 // make it extern because wxStaticText needs to disconnect this one
426 void wxgtk_window_size_request_callback(GtkWidget
*widget
,
427 GtkRequisition
*requisition
,
431 win
->GetSize( &w
, &h
);
437 requisition
->height
= h
;
438 requisition
->width
= w
;
444 void wxgtk_combo_size_request_callback(GtkWidget
*widget
,
445 GtkRequisition
*requisition
,
448 // This callback is actually hooked into the text entry
449 // of the combo box, not the GtkHBox.
452 win
->GetSize( &w
, &h
);
458 GtkCombo
*gcombo
= GTK_COMBO(win
->m_widget
);
460 GtkRequisition entry_req
;
462 entry_req
.height
= 2;
463 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(gcombo
->button
) )->size_request
)
464 (gcombo
->button
, &entry_req
);
466 requisition
->width
= w
- entry_req
.width
;
467 requisition
->height
= entry_req
.height
;
471 //-----------------------------------------------------------------------------
472 // "expose_event" of m_wxwindow
473 //-----------------------------------------------------------------------------
477 gtk_window_expose_callback( GtkWidget
*widget
,
478 GdkEventExpose
*gdk_event
,
483 // don't need to install idle handler, its done from "event" signal
485 // This callback gets called in drawing-idle time under
486 // GTK 2.0, so we don't need to defer anything to idle
489 GtkPizza
*pizza
= GTK_PIZZA( widget
);
490 if (gdk_event
->window
!= pizza
->bin_window
) return FALSE
;
496 wxPrintf( wxT("OnExpose from ") );
497 if (win
->GetClassInfo() && win
->GetClassInfo()->GetClassName())
498 wxPrintf( win
->GetClassInfo()->GetClassName() );
499 wxPrintf( wxT(" %d %d %d %d\n"), (int)gdk_event
->area
.x
,
500 (int)gdk_event
->area
.y
,
501 (int)gdk_event
->area
.width
,
502 (int)gdk_event
->area
.height
);
507 win
->m_wxwindow
->style
,
511 (GdkRectangle
*) NULL
,
513 (char *)"button", // const_cast
518 win
->GetUpdateRegion() = wxRegion( gdk_event
->region
);
520 win
->GtkSendPaintEvents();
522 // Let parent window draw window-less widgets
527 //-----------------------------------------------------------------------------
528 // "key_press_event" from any window
529 //-----------------------------------------------------------------------------
531 // These are used when transforming Ctrl-alpha to ascii values 1-26
532 inline bool wxIsLowerChar(int code
)
534 return (code
>= 'a' && code
<= 'z' );
537 inline bool wxIsUpperChar(int code
)
539 return (code
>= 'A' && code
<= 'Z' );
543 // set WXTRACE to this to see the key event codes on the console
544 #define TRACE_KEYS _T("keyevent")
546 // translates an X key symbol to WXK_XXX value
548 // if isChar is true it means that the value returned will be used for EVT_CHAR
549 // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
550 // for example, while if it is false it means that the value is going to be
551 // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
553 static long wxTranslateKeySymToWXKey(KeySym keysym
, bool isChar
)
559 // Shift, Control and Alt don't generate the CHAR events at all
562 key_code
= isChar
? 0 : WXK_SHIFT
;
566 key_code
= isChar
? 0 : WXK_CONTROL
;
574 key_code
= isChar
? 0 : WXK_ALT
;
577 // neither do the toggle modifies
578 case GDK_Scroll_Lock
:
579 key_code
= isChar
? 0 : WXK_SCROLL
;
583 key_code
= isChar
? 0 : WXK_CAPITAL
;
587 key_code
= isChar
? 0 : WXK_NUMLOCK
;
591 // various other special keys
604 case GDK_ISO_Left_Tab
:
611 key_code
= WXK_RETURN
;
615 key_code
= WXK_CLEAR
;
619 key_code
= WXK_PAUSE
;
623 key_code
= WXK_SELECT
;
627 key_code
= WXK_PRINT
;
631 key_code
= WXK_EXECUTE
;
635 key_code
= WXK_ESCAPE
;
638 // cursor and other extended keyboard keys
640 key_code
= WXK_DELETE
;
656 key_code
= WXK_RIGHT
;
663 case GDK_Prior
: // == GDK_Page_Up
664 key_code
= WXK_PAGEUP
;
667 case GDK_Next
: // == GDK_Page_Down
668 key_code
= WXK_PAGEDOWN
;
680 key_code
= WXK_INSERT
;
695 key_code
= (isChar
? '0' : WXK_NUMPAD0
) + keysym
- GDK_KP_0
;
699 key_code
= isChar
? ' ' : WXK_NUMPAD_SPACE
;
703 key_code
= isChar
? WXK_TAB
: WXK_NUMPAD_TAB
;
707 key_code
= isChar
? WXK_RETURN
: WXK_NUMPAD_ENTER
;
711 key_code
= isChar
? WXK_F1
: WXK_NUMPAD_F1
;
715 key_code
= isChar
? WXK_F2
: WXK_NUMPAD_F2
;
719 key_code
= isChar
? WXK_F3
: WXK_NUMPAD_F3
;
723 key_code
= isChar
? WXK_F4
: WXK_NUMPAD_F4
;
727 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_HOME
;
731 key_code
= isChar
? WXK_LEFT
: WXK_NUMPAD_LEFT
;
735 key_code
= isChar
? WXK_UP
: WXK_NUMPAD_UP
;
739 key_code
= isChar
? WXK_RIGHT
: WXK_NUMPAD_RIGHT
;
743 key_code
= isChar
? WXK_DOWN
: WXK_NUMPAD_DOWN
;
746 case GDK_KP_Prior
: // == GDK_KP_Page_Up
747 key_code
= isChar
? WXK_PAGEUP
: WXK_NUMPAD_PAGEUP
;
750 case GDK_KP_Next
: // == GDK_KP_Page_Down
751 key_code
= isChar
? WXK_PAGEDOWN
: WXK_NUMPAD_PAGEDOWN
;
755 key_code
= isChar
? WXK_END
: WXK_NUMPAD_END
;
759 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_BEGIN
;
763 key_code
= isChar
? WXK_INSERT
: WXK_NUMPAD_INSERT
;
767 key_code
= isChar
? WXK_DELETE
: WXK_NUMPAD_DELETE
;
771 key_code
= isChar
? '=' : WXK_NUMPAD_EQUAL
;
774 case GDK_KP_Multiply
:
775 key_code
= isChar
? '*' : WXK_NUMPAD_MULTIPLY
;
779 key_code
= isChar
? '+' : WXK_NUMPAD_ADD
;
782 case GDK_KP_Separator
:
783 // FIXME: what is this?
784 key_code
= isChar
? '.' : WXK_NUMPAD_SEPARATOR
;
787 case GDK_KP_Subtract
:
788 key_code
= isChar
? '-' : WXK_NUMPAD_SUBTRACT
;
792 key_code
= isChar
? '.' : WXK_NUMPAD_DECIMAL
;
796 key_code
= isChar
? '/' : WXK_NUMPAD_DIVIDE
;
813 key_code
= WXK_F1
+ keysym
- GDK_F1
;
823 static inline bool wxIsAsciiKeysym(KeySym ks
)
828 static void wxFillOtherKeyEventFields(wxKeyEvent
& event
,
830 GdkEventKey
*gdk_event
)
834 GdkModifierType state
;
835 if (gdk_event
->window
)
836 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
838 event
.SetTimestamp( gdk_event
->time
);
839 event
.SetId(win
->GetId());
840 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
) != 0;
841 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
) != 0;
842 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
) != 0;
843 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
) != 0;
844 event
.m_scanCode
= gdk_event
->keyval
;
845 event
.m_rawCode
= (wxUint32
) gdk_event
->keyval
;
846 event
.m_rawFlags
= 0;
848 event
.m_uniChar
= gdk_keyval_to_unicode(gdk_event
->keyval
);
850 wxGetMousePosition( &x
, &y
);
851 win
->ScreenToClient( &x
, &y
);
854 event
.SetEventObject( win
);
859 wxTranslateGTKKeyEventToWx(wxKeyEvent
& event
,
861 GdkEventKey
*gdk_event
)
863 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
864 // but only event->keyval which is quite useless to us, so remember
865 // the last character from GDK_KEY_PRESS and reuse it as last resort
867 // NB: should be MT-safe as we're always called from the main thread only
872 } s_lastKeyPress
= { 0, 0 };
874 KeySym keysym
= gdk_event
->keyval
;
876 wxLogTrace(TRACE_KEYS
, _T("Key %s event: keysym = %ld"),
877 event
.GetEventType() == wxEVT_KEY_UP
? _T("release")
881 long key_code
= wxTranslateKeySymToWXKey(keysym
, false /* !isChar */);
885 // do we have the translation or is it a plain ASCII character?
886 if ( (gdk_event
->length
== 1) || wxIsAsciiKeysym(keysym
) )
888 // we should use keysym if it is ASCII as X does some translations
889 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
890 // which we don't want here (but which we do use for OnChar())
891 if ( !wxIsAsciiKeysym(keysym
) )
893 keysym
= (KeySym
)gdk_event
->string
[0];
896 // we want to always get the same key code when the same key is
897 // pressed regardless of the state of the modifiers, i.e. on a
898 // standard US keyboard pressing '5' or '%' ('5' key with
899 // Shift) should result in the same key code in OnKeyDown():
900 // '5' (although OnChar() will get either '5' or '%').
902 // to do it we first translate keysym to keycode (== scan code)
903 // and then back but always using the lower register
904 Display
*dpy
= (Display
*)wxGetDisplay();
905 KeyCode keycode
= XKeysymToKeycode(dpy
, keysym
);
907 wxLogTrace(TRACE_KEYS
, _T("\t-> keycode %d"), keycode
);
909 KeySym keysymNormalized
= XKeycodeToKeysym(dpy
, keycode
, 0);
911 // use the normalized, i.e. lower register, keysym if we've
913 key_code
= keysymNormalized
? keysymNormalized
: keysym
;
915 // as explained above, we want to have lower register key codes
916 // normally but for the letter keys we want to have the upper ones
918 // NB: don't use XConvertCase() here, we want to do it for letters
920 key_code
= toupper(key_code
);
922 else // non ASCII key, what to do?
924 // by default, ignore it
927 // but if we have cached information from the last KEY_PRESS
928 if ( gdk_event
->type
== GDK_KEY_RELEASE
)
931 if ( keysym
== s_lastKeyPress
.keysym
)
933 key_code
= s_lastKeyPress
.keycode
;
938 if ( gdk_event
->type
== GDK_KEY_PRESS
)
940 // remember it to be reused for KEY_UP event later
941 s_lastKeyPress
.keysym
= keysym
;
942 s_lastKeyPress
.keycode
= key_code
;
946 wxLogTrace(TRACE_KEYS
, _T("\t-> wxKeyCode %ld"), key_code
);
948 // sending unknown key events doesn't really make sense
952 // now fill all the other fields
953 wxFillOtherKeyEventFields(event
, win
, gdk_event
);
955 event
.m_keyCode
= key_code
;
957 if ( gdk_event
->type
== GDK_KEY_PRESS
|| gdk_event
->type
== GDK_KEY_RELEASE
)
959 event
.m_uniChar
= key_code
;
969 GtkIMContext
*context
;
970 GdkEventKey
*lastKeyEvent
;
974 context
= gtk_im_multicontext_new();
979 g_object_unref (context
);
985 gtk_window_key_press_callback( GtkWidget
*widget
,
986 GdkEventKey
*gdk_event
,
991 // don't need to install idle handler, its done from "event" signal
995 if (g_blockEventsOnDrag
)
999 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1001 bool return_after_IM
= false;
1003 if( wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1005 // Emit KEY_DOWN event
1006 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1010 // Return after IM processing as we cannot do
1011 // anything with it anyhow.
1012 return_after_IM
= true;
1015 // 2005.01.26 modified by Hong Jen Yee (hzysoft@sina.com.tw):
1016 // When we get a key_press event here, it could be originate
1017 // from the current widget or its child widgets. However, only the widget
1018 // with the INPUT FOCUS can generate the INITIAL key_press event. That is,
1019 // if the CURRENT widget doesn't have the FOCUS at all, this event definitely
1020 // originated from its child widgets and shouldn't be passed to IM context.
1021 // In fact, what a GTK+ IM should do is filtering keyEvents and convert them
1022 // into text input ONLY WHEN THE WIDGET HAS INPUT FOCUS. Besides, when current
1023 // widgets has both IM context and input focus, the event should be filtered
1024 // by gtk_im_context_filter_keypress().
1025 // Then, we should, according to GTK+ 2.0 API doc, return whatever it returns.
1026 if ((!ret
) && (win
->m_imData
!= NULL
) && ( wxWindow::FindFocus() == win
))
1028 // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API
1029 // docs, if IM filter returns true, no further processing should be done.
1030 // we should send the key_down event anyway.
1031 bool intercepted_by_IM
= gtk_im_context_filter_keypress(win
->m_imData
->context
, gdk_event
);
1032 win
->m_imData
->lastKeyEvent
= NULL
;
1033 if (intercepted_by_IM
)
1035 wxLogTrace(TRACE_KEYS
, _T("Key event intercepted by IM"));
1040 if (return_after_IM
)
1046 wxWindowGTK
*ancestor
= win
;
1049 int command
= ancestor
->GetAcceleratorTable()->GetCommand( event
);
1052 wxCommandEvent
command_event( wxEVT_COMMAND_MENU_SELECTED
, command
);
1053 ret
= ancestor
->GetEventHandler()->ProcessEvent( command_event
);
1056 if (ancestor
->IsTopLevel())
1058 ancestor
= ancestor
->GetParent();
1061 #endif // wxUSE_ACCEL
1063 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
1064 // will only be sent if it is not in an accelerator table.
1068 KeySym keysym
= gdk_event
->keyval
;
1069 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
1070 key_code
= wxTranslateKeySymToWXKey(keysym
, true /* isChar */);
1073 if ( wxIsAsciiKeysym(keysym
) )
1076 key_code
= (unsigned char)keysym
;
1078 // gdk_event->string is actually deprecated
1079 else if ( gdk_event
->length
== 1 )
1081 key_code
= (unsigned char)gdk_event
->string
[0];
1087 wxLogTrace(TRACE_KEYS
, _T("Char event: %ld"), key_code
);
1089 event
.m_keyCode
= key_code
;
1091 // To conform to the docs we need to translate Ctrl-alpha
1092 // characters to values in the range 1-26.
1093 if ( event
.ControlDown() &&
1094 ( wxIsLowerChar(key_code
) || wxIsUpperChar(key_code
) ))
1096 if ( wxIsLowerChar(key_code
) )
1097 event
.m_keyCode
= key_code
- 'a' + 1;
1098 if ( wxIsUpperChar(key_code
) )
1099 event
.m_keyCode
= key_code
- 'A' + 1;
1101 event
.m_uniChar
= event
.m_keyCode
;
1105 // Implement OnCharHook by checking ancestor top level windows
1106 wxWindow
*parent
= win
;
1107 while (parent
&& !parent
->IsTopLevel())
1108 parent
= parent
->GetParent();
1111 event
.SetEventType( wxEVT_CHAR_HOOK
);
1112 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1117 event
.SetEventType(wxEVT_CHAR
);
1118 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1123 // win is a control: tab can be propagated up
1125 ((gdk_event
->keyval
== GDK_Tab
) || (gdk_event
->keyval
== GDK_ISO_Left_Tab
)) &&
1126 // VZ: testing for wxTE_PROCESS_TAB shouldn't be done here - the control may
1127 // have this style, yet choose not to process this particular TAB in which
1128 // case TAB must still work as a navigational character
1129 // JS: enabling again to make consistent with other platforms
1130 // (with wxTE_PROCESS_TAB you have to call Navigate to get default
1131 // navigation behaviour)
1133 (! (win
->HasFlag(wxTE_PROCESS_TAB
) && win
->IsKindOf(CLASSINFO(wxTextCtrl
)) )) &&
1135 win
->GetParent() && (win
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) )
1137 wxNavigationKeyEvent new_event
;
1138 new_event
.SetEventObject( win
->GetParent() );
1139 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1140 new_event
.SetDirection( (gdk_event
->keyval
== GDK_Tab
) );
1141 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1142 new_event
.SetWindowChange( (gdk_event
->state
& GDK_CONTROL_MASK
) );
1143 new_event
.SetCurrentFocus( win
);
1144 ret
= win
->GetParent()->GetEventHandler()->ProcessEvent( new_event
);
1153 gtk_wxwindow_commit_cb (GtkIMContext
*context
,
1157 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1159 // take modifiers, cursor position, timestamp etc. from the last
1160 // key_press_event that was fed into Input Method:
1161 if (window
->m_imData
->lastKeyEvent
)
1163 wxFillOtherKeyEventFields(event
,
1164 window
, window
->m_imData
->lastKeyEvent
);
1167 const wxWxCharBuffer
data(wxGTK_CONV_BACK(str
));
1173 // Implement OnCharHook by checking ancestor top level windows
1174 wxWindow
*parent
= window
;
1175 while (parent
&& !parent
->IsTopLevel())
1176 parent
= parent
->GetParent();
1178 for( const wxChar
* pstr
= data
; *pstr
; pstr
++ )
1181 event
.m_uniChar
= *pstr
;
1182 // Backward compatible for ISO-8859-1
1183 event
.m_keyCode
= *pstr
< 256 ? event
.m_uniChar
: 0;
1184 wxLogTrace(TRACE_KEYS
, _T("IM sent character '%c'"), event
.m_uniChar
);
1186 event
.m_keyCode
= *pstr
;
1187 #endif // wxUSE_UNICODE
1189 // To conform to the docs we need to translate Ctrl-alpha
1190 // characters to values in the range 1-26.
1191 if ( event
.ControlDown() &&
1192 ( wxIsLowerChar(*pstr
) || wxIsUpperChar(*pstr
) ))
1194 if ( wxIsLowerChar(*pstr
) )
1195 event
.m_keyCode
= *pstr
- 'a' + 1;
1196 if ( wxIsUpperChar(*pstr
) )
1197 event
.m_keyCode
= *pstr
- 'A' + 1;
1199 event
.m_keyCode
= *pstr
- 'a' + 1;
1201 event
.m_uniChar
= event
.m_keyCode
;
1207 event
.SetEventType( wxEVT_CHAR_HOOK
);
1208 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1213 event
.SetEventType(wxEVT_CHAR
);
1214 ret
= window
->GetEventHandler()->ProcessEvent( event
);
1221 //-----------------------------------------------------------------------------
1222 // "key_release_event" from any window
1223 //-----------------------------------------------------------------------------
1227 gtk_window_key_release_callback( GtkWidget
*widget
,
1228 GdkEventKey
*gdk_event
,
1233 // don't need to install idle handler, its done from "event" signal
1238 if (g_blockEventsOnDrag
)
1241 wxKeyEvent
event( wxEVT_KEY_UP
);
1242 if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1244 // unknown key pressed, ignore (the event would be useless anyhow)
1248 return win
->GTKProcessEvent(event
);
1252 // ============================================================================
1254 // ============================================================================
1256 // ----------------------------------------------------------------------------
1257 // mouse event processing helpers
1258 // ----------------------------------------------------------------------------
1260 // init wxMouseEvent with the info from GdkEventXXX struct
1261 template<typename T
> void InitMouseEvent(wxWindowGTK
*win
,
1262 wxMouseEvent
& event
,
1265 event
.SetTimestamp( gdk_event
->time
);
1266 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
);
1267 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
);
1268 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
);
1269 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
);
1270 event
.m_leftDown
= (gdk_event
->state
& GDK_BUTTON1_MASK
);
1271 event
.m_middleDown
= (gdk_event
->state
& GDK_BUTTON2_MASK
);
1272 event
.m_rightDown
= (gdk_event
->state
& GDK_BUTTON3_MASK
);
1273 if (event
.GetEventType() == wxEVT_MOUSEWHEEL
)
1275 event
.m_linesPerAction
= 3;
1276 event
.m_wheelDelta
= 120;
1277 if (((GdkEventButton
*)gdk_event
)->button
== 4)
1278 event
.m_wheelRotation
= 120;
1279 else if (((GdkEventButton
*)gdk_event
)->button
== 5)
1280 event
.m_wheelRotation
= -120;
1283 wxPoint pt
= win
->GetClientAreaOrigin();
1284 event
.m_x
= (wxCoord
)gdk_event
->x
- pt
.x
;
1285 event
.m_y
= (wxCoord
)gdk_event
->y
- pt
.y
;
1287 if ((win
->m_wxwindow
) && (win
->GetLayoutDirection() == wxLayout_RightToLeft
))
1289 // origin in the upper right corner
1290 int window_width
= gtk_pizza_get_rtl_offset( GTK_PIZZA(win
->m_wxwindow
) );
1291 event
.m_x
= window_width
- event
.m_x
;
1294 event
.SetEventObject( win
);
1295 event
.SetId( win
->GetId() );
1296 event
.SetTimestamp( gdk_event
->time
);
1299 static void AdjustEventButtonState(wxMouseEvent
& event
)
1301 // GDK reports the old state of the button for a button press event, but
1302 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1303 // for a LEFT_DOWN event, not FALSE, so we will invert
1304 // left/right/middleDown for the corresponding click events
1306 if ((event
.GetEventType() == wxEVT_LEFT_DOWN
) ||
1307 (event
.GetEventType() == wxEVT_LEFT_DCLICK
) ||
1308 (event
.GetEventType() == wxEVT_LEFT_UP
))
1310 event
.m_leftDown
= !event
.m_leftDown
;
1314 if ((event
.GetEventType() == wxEVT_MIDDLE_DOWN
) ||
1315 (event
.GetEventType() == wxEVT_MIDDLE_DCLICK
) ||
1316 (event
.GetEventType() == wxEVT_MIDDLE_UP
))
1318 event
.m_middleDown
= !event
.m_middleDown
;
1322 if ((event
.GetEventType() == wxEVT_RIGHT_DOWN
) ||
1323 (event
.GetEventType() == wxEVT_RIGHT_DCLICK
) ||
1324 (event
.GetEventType() == wxEVT_RIGHT_UP
))
1326 event
.m_rightDown
= !event
.m_rightDown
;
1331 // find the window to send the mouse event too
1333 wxWindowGTK
*FindWindowForMouseEvent(wxWindowGTK
*win
, wxCoord
& x
, wxCoord
& y
)
1338 if (win
->m_wxwindow
)
1340 GtkPizza
*pizza
= GTK_PIZZA(win
->m_wxwindow
);
1341 xx
+= gtk_pizza_get_xoffset( pizza
);
1342 yy
+= gtk_pizza_get_yoffset( pizza
);
1345 wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
1348 wxWindowGTK
*child
= node
->GetData();
1350 node
= node
->GetNext();
1351 if (!child
->IsShown())
1354 if (child
->IsTransparentForMouse())
1356 // wxStaticBox is transparent in the box itself
1357 int xx1
= child
->m_x
;
1358 int yy1
= child
->m_y
;
1359 int xx2
= child
->m_x
+ child
->m_width
;
1360 int yy2
= child
->m_y
+ child
->m_height
;
1363 if (((xx
>= xx1
) && (xx
<= xx1
+10) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1365 ((xx
>= xx2
-10) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1367 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy1
+10)) ||
1369 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy2
-1) && (yy
<= yy2
)))
1380 if ((child
->m_wxwindow
== (GtkWidget
*) NULL
) &&
1381 (child
->m_x
<= xx
) &&
1382 (child
->m_y
<= yy
) &&
1383 (child
->m_x
+child
->m_width
>= xx
) &&
1384 (child
->m_y
+child
->m_height
>= yy
))
1397 // ----------------------------------------------------------------------------
1398 // common event handlers helpers
1399 // ----------------------------------------------------------------------------
1401 bool wxWindowGTK::GTKProcessEvent(wxEvent
& event
) const
1403 // nothing special at this level
1404 return GetEventHandler()->ProcessEvent(event
);
1407 int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny
*event
) const
1411 // don't need to install idle handler, its done from "event" signal
1415 if (g_blockEventsOnDrag
)
1417 if (g_blockEventsOnScroll
)
1420 if (!GTKIsOwnWindow(event
->window
))
1426 // overloads for all GDK event types we use here: we need to have this as
1427 // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact,
1428 // derives from it in the sense that the structs have the same layout
1429 #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \
1430 static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \
1432 return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \
1435 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton
)
1436 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion
)
1437 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing
)
1439 #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD
1441 #define wxCOMMON_CALLBACK_PROLOGUE(event, win) \
1442 const int rc = wxGtkCallbackCommonPrologue(event, win); \
1446 // send the wxChildFocusEvent and wxFocusEvent, common code of
1447 // gtk_window_focus_in_callback() and SetFocus()
1448 static bool DoSendFocusEvents(wxWindow
*win
)
1450 // Notify the parent keeping track of focus for the kbd navigation
1451 // purposes that we got it.
1452 wxChildFocusEvent
eventChildFocus(win
);
1453 (void)win
->GetEventHandler()->ProcessEvent(eventChildFocus
);
1455 wxFocusEvent
eventFocus(wxEVT_SET_FOCUS
, win
->GetId());
1456 eventFocus
.SetEventObject(win
);
1458 return win
->GetEventHandler()->ProcessEvent(eventFocus
);
1461 // all event handlers must have C linkage as they're called from GTK+ C code
1465 //-----------------------------------------------------------------------------
1466 // "button_press_event"
1467 //-----------------------------------------------------------------------------
1470 gtk_window_button_press_callback( GtkWidget
*widget
,
1471 GdkEventButton
*gdk_event
,
1474 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1476 if (win
->m_wxwindow
&& (g_focusWindow
!= win
) && win
->AcceptsFocus())
1478 gtk_widget_grab_focus( win
->m_wxwindow
);
1481 // GDK sends surplus button down events
1482 // before a double click event. We
1483 // need to filter these out.
1484 if (gdk_event
->type
== GDK_BUTTON_PRESS
)
1486 GdkEvent
*peek_event
= gdk_event_peek();
1489 if ((peek_event
->type
== GDK_2BUTTON_PRESS
) ||
1490 (peek_event
->type
== GDK_3BUTTON_PRESS
))
1492 gdk_event_free( peek_event
);
1497 gdk_event_free( peek_event
);
1502 wxEventType event_type
= wxEVT_NULL
;
1504 // GdkDisplay is a GTK+ 2.2.0 thing
1505 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2, 2, 0)
1506 if ( gdk_event
->type
== GDK_2BUTTON_PRESS
&&
1507 !gtk_check_version(2,2,0) &&
1508 gdk_event
->button
>= 1 && gdk_event
->button
<= 3 )
1510 // Reset GDK internal timestamp variables in order to disable GDK
1511 // triple click events. GDK will then next time believe no button has
1512 // been clicked just before, and send a normal button click event.
1513 GdkDisplay
* display
= gtk_widget_get_display (widget
);
1514 display
->button_click_time
[1] = 0;
1515 display
->button_click_time
[0] = 0;
1519 if (gdk_event
->button
== 1)
1521 // note that GDK generates triple click events which are not supported
1522 // by wxWidgets but still have to be passed to the app as otherwise
1523 // clicks would simply go missing
1524 switch (gdk_event
->type
)
1526 // we shouldn't get triple clicks at all for GTK2 because we
1527 // suppress them artificially using the code above but we still
1528 // should map them to something for GTK1 and not just ignore them
1529 // as this would lose clicks
1530 case GDK_3BUTTON_PRESS
: // we could also map this to DCLICK...
1531 case GDK_BUTTON_PRESS
:
1532 event_type
= wxEVT_LEFT_DOWN
;
1535 case GDK_2BUTTON_PRESS
:
1536 event_type
= wxEVT_LEFT_DCLICK
;
1540 // just to silence gcc warnings
1544 else if (gdk_event
->button
== 2)
1546 switch (gdk_event
->type
)
1548 case GDK_3BUTTON_PRESS
:
1549 case GDK_BUTTON_PRESS
:
1550 event_type
= wxEVT_MIDDLE_DOWN
;
1553 case GDK_2BUTTON_PRESS
:
1554 event_type
= wxEVT_MIDDLE_DCLICK
;
1561 else if (gdk_event
->button
== 3)
1563 switch (gdk_event
->type
)
1565 case GDK_3BUTTON_PRESS
:
1566 case GDK_BUTTON_PRESS
:
1567 event_type
= wxEVT_RIGHT_DOWN
;
1570 case GDK_2BUTTON_PRESS
:
1571 event_type
= wxEVT_RIGHT_DCLICK
;
1578 else if (gdk_event
->button
== 4 || gdk_event
->button
== 5)
1580 if (gdk_event
->type
== GDK_BUTTON_PRESS
)
1582 event_type
= wxEVT_MOUSEWHEEL
;
1586 if ( event_type
== wxEVT_NULL
)
1588 // unknown mouse button or click type
1592 wxMouseEvent
event( event_type
);
1593 InitMouseEvent( win
, event
, gdk_event
);
1595 AdjustEventButtonState(event
);
1597 // wxListBox actually gets mouse events from the item, so we need to give it
1598 // a chance to correct this
1599 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1601 // find the correct window to send the event to: it may be a different one
1602 // from the one which got it at GTK+ level because some controls don't have
1603 // their own X window and thus cannot get any events.
1604 if ( !g_captureWindow
)
1605 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1607 // reset the event object and id in case win changed.
1608 event
.SetEventObject( win
);
1609 event
.SetId( win
->GetId() );
1611 if (win
->GTKProcessEvent( event
))
1616 if (event_type
== wxEVT_RIGHT_DOWN
)
1618 // generate a "context menu" event: this is similar to right mouse
1619 // click under many GUIs except that it is generated differently
1620 // (right up under MSW, ctrl-click under Mac, right down here) and
1622 // (a) it's a command event and so is propagated to the parent
1623 // (b) under some ports it can be generated from kbd too
1624 // (c) it uses screen coords (because of (a))
1625 wxContextMenuEvent
evtCtx(
1628 win
->ClientToScreen(event
.GetPosition()));
1629 evtCtx
.SetEventObject(win
);
1630 return win
->GTKProcessEvent(evtCtx
);
1636 //-----------------------------------------------------------------------------
1637 // "button_release_event"
1638 //-----------------------------------------------------------------------------
1641 gtk_window_button_release_callback( GtkWidget
*widget
,
1642 GdkEventButton
*gdk_event
,
1645 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1647 wxEventType event_type
= wxEVT_NULL
;
1649 switch (gdk_event
->button
)
1652 event_type
= wxEVT_LEFT_UP
;
1656 event_type
= wxEVT_MIDDLE_UP
;
1660 event_type
= wxEVT_RIGHT_UP
;
1664 // unknown button, don't process
1668 wxMouseEvent
event( event_type
);
1669 InitMouseEvent( win
, event
, gdk_event
);
1671 AdjustEventButtonState(event
);
1673 // same wxListBox hack as above
1674 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1676 if ( !g_captureWindow
)
1677 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1679 // reset the event object and id in case win changed.
1680 event
.SetEventObject( win
);
1681 event
.SetId( win
->GetId() );
1683 return win
->GTKProcessEvent(event
);
1686 //-----------------------------------------------------------------------------
1687 // "motion_notify_event"
1688 //-----------------------------------------------------------------------------
1691 gtk_window_motion_notify_callback( GtkWidget
*widget
,
1692 GdkEventMotion
*gdk_event
,
1695 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1697 if (gdk_event
->is_hint
)
1701 GdkModifierType state
;
1702 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1707 wxMouseEvent
event( wxEVT_MOTION
);
1708 InitMouseEvent(win
, event
, gdk_event
);
1710 if ( g_captureWindow
)
1712 // synthesise a mouse enter or leave event if needed
1713 GdkWindow
*winUnderMouse
= gdk_window_at_pointer(NULL
, NULL
);
1714 // This seems to be necessary and actually been added to
1715 // GDK itself in version 2.0.X
1718 bool hasMouse
= winUnderMouse
== gdk_event
->window
;
1719 if ( hasMouse
!= g_captureWindowHasMouse
)
1721 // the mouse changed window
1722 g_captureWindowHasMouse
= hasMouse
;
1724 wxMouseEvent
eventM(g_captureWindowHasMouse
? wxEVT_ENTER_WINDOW
1725 : wxEVT_LEAVE_WINDOW
);
1726 InitMouseEvent(win
, eventM
, gdk_event
);
1727 eventM
.SetEventObject(win
);
1728 win
->GTKProcessEvent(eventM
);
1733 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1735 // reset the event object and id in case win changed.
1736 event
.SetEventObject( win
);
1737 event
.SetId( win
->GetId() );
1740 if ( !g_captureWindow
)
1742 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1743 if (win
->GTKProcessEvent( cevent
))
1745 win
->SetCursor( cevent
.GetCursor() );
1749 return win
->GTKProcessEvent(event
);
1752 //-----------------------------------------------------------------------------
1753 // "scroll_event", (mouse wheel event)
1754 //-----------------------------------------------------------------------------
1757 window_scroll_event(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
)
1761 // don't need to install idle handler, its done from "event" signal
1763 if (gdk_event
->direction
!= GDK_SCROLL_UP
&&
1764 gdk_event
->direction
!= GDK_SCROLL_DOWN
)
1769 wxMouseEvent
event(wxEVT_MOUSEWHEEL
);
1770 // Can't use InitMouse macro because scroll events don't have button
1771 event
.SetTimestamp( gdk_event
->time
);
1772 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
);
1773 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
);
1774 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
);
1775 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
);
1776 event
.m_leftDown
= (gdk_event
->state
& GDK_BUTTON1_MASK
);
1777 event
.m_middleDown
= (gdk_event
->state
& GDK_BUTTON2_MASK
);
1778 event
.m_rightDown
= (gdk_event
->state
& GDK_BUTTON3_MASK
);
1779 event
.m_linesPerAction
= 3;
1780 event
.m_wheelDelta
= 120;
1781 if (gdk_event
->direction
== GDK_SCROLL_UP
)
1782 event
.m_wheelRotation
= 120;
1784 event
.m_wheelRotation
= -120;
1786 wxPoint pt
= win
->GetClientAreaOrigin();
1787 event
.m_x
= (wxCoord
)gdk_event
->x
- pt
.x
;
1788 event
.m_y
= (wxCoord
)gdk_event
->y
- pt
.y
;
1790 event
.SetEventObject( win
);
1791 event
.SetId( win
->GetId() );
1792 event
.SetTimestamp( gdk_event
->time
);
1794 return win
->GTKProcessEvent(event
);
1797 //-----------------------------------------------------------------------------
1799 //-----------------------------------------------------------------------------
1801 static gboolean
wxgtk_window_popup_menu_callback(GtkWidget
*, wxWindowGTK
* win
)
1803 wxContextMenuEvent
event(wxEVT_CONTEXT_MENU
, win
->GetId(), wxPoint(-1, -1));
1804 event
.SetEventObject(win
);
1805 return win
->GTKProcessEvent(event
);
1808 //-----------------------------------------------------------------------------
1810 //-----------------------------------------------------------------------------
1813 gtk_window_focus_in_callback( GtkWidget
*widget
,
1814 GdkEventFocus
*WXUNUSED(event
),
1819 // don't need to install idle handler, its done from "event" signal
1822 gtk_im_context_focus_in(win
->m_imData
->context
);
1825 g_focusWindow
= win
;
1827 wxLogTrace(TRACE_FOCUS
,
1828 _T("%s: focus in"), win
->GetName().c_str());
1832 gdk_im_begin(win
->m_ic
, win
->m_wxwindow
->window
);
1836 // caret needs to be informed about focus change
1837 wxCaret
*caret
= win
->GetCaret();
1840 caret
->OnSetFocus();
1842 #endif // wxUSE_CARET
1844 gboolean ret
= FALSE
;
1846 // does the window itself think that it has the focus?
1847 if ( !win
->m_hasFocus
)
1849 // not yet, notify it
1850 win
->m_hasFocus
= true;
1852 (void)DoSendFocusEvents(win
);
1857 // Disable default focus handling for custom windows
1858 // since the default GTK+ handler issues a repaint
1859 if (win
->m_wxwindow
)
1865 //-----------------------------------------------------------------------------
1866 // "focus_out_event"
1867 //-----------------------------------------------------------------------------
1870 gtk_window_focus_out_callback( GtkWidget
*widget
,
1871 GdkEventFocus
*gdk_event
,
1876 // don't need to install idle handler, its done from "event" signal
1879 gtk_im_context_focus_out(win
->m_imData
->context
);
1881 wxLogTrace( TRACE_FOCUS
,
1882 _T("%s: focus out"), win
->GetName().c_str() );
1885 wxWindowGTK
*winFocus
= wxFindFocusedChild(win
);
1889 g_focusWindow
= (wxWindowGTK
*)NULL
;
1897 // caret needs to be informed about focus change
1898 wxCaret
*caret
= win
->GetCaret();
1901 caret
->OnKillFocus();
1903 #endif // wxUSE_CARET
1905 gboolean ret
= FALSE
;
1907 // don't send the window a kill focus event if it thinks that it doesn't
1908 // have focus already
1909 if ( win
->m_hasFocus
)
1911 win
->m_hasFocus
= false;
1913 wxFocusEvent
event( wxEVT_KILL_FOCUS
, win
->GetId() );
1914 event
.SetEventObject( win
);
1916 (void)win
->GTKProcessEvent( event
);
1921 // Disable default focus handling for custom windows
1922 // since the default GTK+ handler issues a repaint
1923 if (win
->m_wxwindow
)
1929 //-----------------------------------------------------------------------------
1930 // "enter_notify_event"
1931 //-----------------------------------------------------------------------------
1934 gtk_window_enter_callback( GtkWidget
*widget
,
1935 GdkEventCrossing
*gdk_event
,
1938 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1940 // Event was emitted after a grab
1941 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1945 GdkModifierType state
= (GdkModifierType
)0;
1947 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1949 wxMouseEvent
event( wxEVT_ENTER_WINDOW
);
1950 InitMouseEvent(win
, event
, gdk_event
);
1951 wxPoint pt
= win
->GetClientAreaOrigin();
1952 event
.m_x
= x
+ pt
.x
;
1953 event
.m_y
= y
+ pt
.y
;
1955 if ( !g_captureWindow
)
1957 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1958 if (win
->GTKProcessEvent( cevent
))
1960 win
->SetCursor( cevent
.GetCursor() );
1964 return win
->GTKProcessEvent(event
);
1967 //-----------------------------------------------------------------------------
1968 // "leave_notify_event"
1969 //-----------------------------------------------------------------------------
1972 gtk_window_leave_callback( GtkWidget
*widget
,
1973 GdkEventCrossing
*gdk_event
,
1976 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1978 // Event was emitted after an ungrab
1979 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1981 wxMouseEvent
event( wxEVT_LEAVE_WINDOW
);
1982 event
.SetTimestamp( gdk_event
->time
);
1983 event
.SetEventObject( win
);
1987 GdkModifierType state
= (GdkModifierType
)0;
1989 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1991 event
.m_shiftDown
= (state
& GDK_SHIFT_MASK
) != 0;
1992 event
.m_controlDown
= (state
& GDK_CONTROL_MASK
) != 0;
1993 event
.m_altDown
= (state
& GDK_MOD1_MASK
) != 0;
1994 event
.m_metaDown
= (state
& GDK_MOD2_MASK
) != 0;
1995 event
.m_leftDown
= (state
& GDK_BUTTON1_MASK
) != 0;
1996 event
.m_middleDown
= (state
& GDK_BUTTON2_MASK
) != 0;
1997 event
.m_rightDown
= (state
& GDK_BUTTON3_MASK
) != 0;
1999 wxPoint pt
= win
->GetClientAreaOrigin();
2000 event
.m_x
= x
+ pt
.x
;
2001 event
.m_y
= y
+ pt
.y
;
2003 return win
->GTKProcessEvent(event
);
2006 //-----------------------------------------------------------------------------
2007 // "value_changed" from scrollbar
2008 //-----------------------------------------------------------------------------
2011 gtk_scrollbar_value_changed(GtkRange
* range
, wxWindow
* win
)
2013 wxEventType eventType
= win
->GetScrollEventType(range
);
2014 if (eventType
!= wxEVT_NULL
)
2016 // Convert scroll event type to scrollwin event type
2017 eventType
+= wxEVT_SCROLLWIN_TOP
- wxEVT_SCROLL_TOP
;
2019 // find the scrollbar which generated the event
2020 wxWindowGTK::ScrollDir dir
= win
->ScrollDirFromRange(range
);
2022 // generate the corresponding wx event
2023 const int orient
= win
->OrientFromScrollDir(dir
);
2024 wxScrollWinEvent
event(eventType
, win
->GetScrollPos(orient
), orient
);
2025 event
.SetEventObject(win
);
2027 win
->m_blockValueChanged
[dir
] = true;
2028 win
->GTKProcessEvent(event
);
2029 win
->m_blockValueChanged
[dir
] = false;
2033 //-----------------------------------------------------------------------------
2034 // "button_press_event" from scrollbar
2035 //-----------------------------------------------------------------------------
2038 gtk_scrollbar_button_press_event(GtkRange
*, GdkEventButton
*, wxWindow
* win
)
2042 // don't need to install idle handler, its done from "event" signal
2044 g_blockEventsOnScroll
= true;
2045 win
->m_mouseButtonDown
= true;
2050 //-----------------------------------------------------------------------------
2051 // "event_after" from scrollbar
2052 //-----------------------------------------------------------------------------
2055 gtk_scrollbar_event_after(GtkRange
* range
, GdkEvent
* event
, wxWindow
* win
)
2057 if (event
->type
== GDK_BUTTON_RELEASE
)
2059 g_signal_handlers_block_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
2061 const int orient
= win
->OrientFromScrollDir(
2062 win
->ScrollDirFromRange(range
));
2063 wxScrollWinEvent
event(wxEVT_SCROLLWIN_THUMBRELEASE
, win
->GetScrollPos(orient
), orient
);
2064 event
.SetEventObject(win
);
2065 win
->GTKProcessEvent(event
);
2069 //-----------------------------------------------------------------------------
2070 // "button_release_event" from scrollbar
2071 //-----------------------------------------------------------------------------
2074 gtk_scrollbar_button_release_event(GtkRange
* range
, GdkEventButton
*, wxWindow
* win
)
2078 g_blockEventsOnScroll
= false;
2079 win
->m_mouseButtonDown
= false;
2080 // If thumb tracking
2081 if (win
->m_isScrolling
)
2083 win
->m_isScrolling
= false;
2084 // Hook up handler to send thumb release event after this emission is finished.
2085 // To allow setting scroll position from event handler, sending event must
2086 // be deferred until after the GtkRange handler for this signal has run
2087 g_signal_handlers_unblock_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
2093 //-----------------------------------------------------------------------------
2094 // "realize" from m_widget
2095 //-----------------------------------------------------------------------------
2097 /* We cannot set colours and fonts before the widget has
2098 been realized, so we do this directly after realization. */
2101 gtk_window_realized_callback( GtkWidget
*m_widget
, wxWindow
*win
)
2106 wxapp_install_idle_handler();
2110 GtkPizza
*pizza
= GTK_PIZZA( m_widget
);
2111 gtk_im_context_set_client_window( win
->m_imData
->context
,
2112 pizza
->bin_window
);
2115 wxWindowCreateEvent
event( win
);
2116 event
.SetEventObject( win
);
2117 win
->GTKProcessEvent( event
);
2120 //-----------------------------------------------------------------------------
2122 //-----------------------------------------------------------------------------
2125 void gtk_window_size_callback( GtkWidget
*WXUNUSED(widget
),
2126 GtkAllocation
*alloc
,
2130 wxapp_install_idle_handler();
2132 int client_width
= 0;
2133 int client_height
= 0;
2134 win
->GetClientSize( &client_width
, &client_height
);
2135 if ((client_width
== win
->m_oldClientWidth
) && (client_height
== win
->m_oldClientHeight
))
2139 wxPrintf( wxT("size_allocate ") );
2140 if (win
->GetClassInfo() && win
->GetClassInfo()->GetClassName())
2141 wxPrintf( win
->GetClassInfo()->GetClassName() );
2142 wxPrintf( wxT(" %d %d %d %d\n"),
2149 GTK_PIZZA(win
->m_wxwindow
)->m_width
= win
->GetClientSize().x
;
2151 win
->m_oldClientWidth
= client_width
;
2152 win
->m_oldClientHeight
= client_height
;
2154 if (!win
->m_nativeSizeEvent
)
2156 wxSizeEvent
event( win
->GetSize(), win
->GetId() );
2157 event
.SetEventObject( win
);
2158 win
->GTKProcessEvent( event
);
2164 #define WXUNUSED_UNLESS_XIM(param) param
2166 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2169 /* Resize XIM window */
2171 void gtk_wxwindow_size_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2172 GtkAllocation
* WXUNUSED_UNLESS_XIM(alloc
),
2173 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2176 wxapp_install_idle_handler();
2182 if (gdk_ic_get_style (win
->m_ic
) & GDK_IM_PREEDIT_POSITION
)
2186 gdk_drawable_get_size (widget
->window
, &width
, &height
);
2187 win
->m_icattr
->preedit_area
.width
= width
;
2188 win
->m_icattr
->preedit_area
.height
= height
;
2189 gdk_ic_set_attr (win
->m_ic
, win
->m_icattr
, GDK_IC_PREEDIT_AREA
);
2194 //-----------------------------------------------------------------------------
2195 // "realize" from m_wxwindow
2196 //-----------------------------------------------------------------------------
2198 /* Initialize XIM support */
2201 gtk_wxwindow_realized_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2202 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2205 wxapp_install_idle_handler();
2208 if (win
->m_ic
) return;
2209 if (!widget
) return;
2210 if (!gdk_im_ready()) return;
2212 win
->m_icattr
= gdk_ic_attr_new();
2213 if (!win
->m_icattr
) return;
2217 GdkColormap
*colormap
;
2218 GdkICAttr
*attr
= win
->m_icattr
;
2219 unsigned attrmask
= GDK_IC_ALL_REQ
;
2221 GdkIMStyle supported_style
= (GdkIMStyle
)
2222 (GDK_IM_PREEDIT_NONE
|
2223 GDK_IM_PREEDIT_NOTHING
|
2224 GDK_IM_PREEDIT_POSITION
|
2225 GDK_IM_STATUS_NONE
|
2226 GDK_IM_STATUS_NOTHING
);
2228 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2229 supported_style
= (GdkIMStyle
)(supported_style
& ~GDK_IM_PREEDIT_POSITION
);
2231 attr
->style
= style
= gdk_im_decide_style (supported_style
);
2232 attr
->client_window
= widget
->window
;
2234 if ((colormap
= gtk_widget_get_colormap (widget
)) !=
2235 gtk_widget_get_default_colormap ())
2237 attrmask
|= GDK_IC_PREEDIT_COLORMAP
;
2238 attr
->preedit_colormap
= colormap
;
2241 attrmask
|= GDK_IC_PREEDIT_FOREGROUND
;
2242 attrmask
|= GDK_IC_PREEDIT_BACKGROUND
;
2243 attr
->preedit_foreground
= widget
->style
->fg
[GTK_STATE_NORMAL
];
2244 attr
->preedit_background
= widget
->style
->base
[GTK_STATE_NORMAL
];
2246 switch (style
& GDK_IM_PREEDIT_MASK
)
2248 case GDK_IM_PREEDIT_POSITION
:
2249 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2251 g_warning ("over-the-spot style requires fontset");
2255 gdk_drawable_get_size (widget
->window
, &width
, &height
);
2257 attrmask
|= GDK_IC_PREEDIT_POSITION_REQ
;
2258 attr
->spot_location
.x
= 0;
2259 attr
->spot_location
.y
= height
;
2260 attr
->preedit_area
.x
= 0;
2261 attr
->preedit_area
.y
= 0;
2262 attr
->preedit_area
.width
= width
;
2263 attr
->preedit_area
.height
= height
;
2264 attr
->preedit_fontset
= widget
->style
->font
;
2269 win
->m_ic
= gdk_ic_new (attr
, (GdkICAttributesType
)attrmask
);
2271 if (win
->m_ic
== NULL
)
2272 g_warning ("Can't create input context.");
2275 mask
= gdk_window_get_events (widget
->window
);
2276 mask
= (GdkEventMask
)(mask
| gdk_ic_get_events (win
->m_ic
));
2277 gdk_window_set_events (widget
->window
, mask
);
2279 if (GTK_WIDGET_HAS_FOCUS(widget
))
2280 gdk_im_begin (win
->m_ic
, widget
->window
);
2287 // ----------------------------------------------------------------------------
2288 // this wxWindowBase function is implemented here (in platform-specific file)
2289 // because it is static and so couldn't be made virtual
2290 // ----------------------------------------------------------------------------
2292 wxWindow
*wxWindowBase::DoFindFocus()
2294 // the cast is necessary when we compile in wxUniversal mode
2295 return (wxWindow
*)g_focusWindow
;
2298 //-----------------------------------------------------------------------------
2299 // InsertChild for wxWindowGTK.
2300 //-----------------------------------------------------------------------------
2302 /* Callback for wxWindowGTK. This very strange beast has to be used because
2303 * C++ has no virtual methods in a constructor. We have to emulate a
2304 * virtual function here as wxNotebook requires a different way to insert
2305 * a child in it. I had opted for creating a wxNotebookPage window class
2306 * which would have made this superfluous (such in the MDI window system),
2307 * but no-one was listening to me... */
2309 static void wxInsertChildInWindow( wxWindowGTK
* parent
, wxWindowGTK
* child
)
2311 /* the window might have been scrolled already, do we
2312 have to adapt the position */
2313 GtkPizza
*pizza
= GTK_PIZZA(parent
->m_wxwindow
);
2314 child
->m_x
+= gtk_pizza_get_xoffset( pizza
);
2315 child
->m_y
+= gtk_pizza_get_yoffset( pizza
);
2317 gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
),
2318 GTK_WIDGET(child
->m_widget
),
2325 //-----------------------------------------------------------------------------
2327 //-----------------------------------------------------------------------------
2329 wxWindow
*wxGetActiveWindow()
2331 return wxWindow::FindFocus();
2335 wxMouseState
wxGetMouseState()
2341 GdkModifierType mask
;
2343 gdk_window_get_pointer(NULL
, &x
, &y
, &mask
);
2347 ms
.SetLeftDown(mask
& GDK_BUTTON1_MASK
);
2348 ms
.SetMiddleDown(mask
& GDK_BUTTON2_MASK
);
2349 ms
.SetRightDown(mask
& GDK_BUTTON3_MASK
);
2351 ms
.SetControlDown(mask
& GDK_CONTROL_MASK
);
2352 ms
.SetShiftDown(mask
& GDK_SHIFT_MASK
);
2353 ms
.SetAltDown(mask
& GDK_MOD1_MASK
);
2354 ms
.SetMetaDown(mask
& GDK_MOD2_MASK
);
2359 //-----------------------------------------------------------------------------
2361 //-----------------------------------------------------------------------------
2363 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2365 #ifdef __WXUNIVERSAL__
2366 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
)
2368 IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
)
2369 #endif // __WXUNIVERSAL__/__WXGTK__
2371 void wxWindowGTK::Init()
2374 m_widget
= (GtkWidget
*) NULL
;
2375 m_wxwindow
= (GtkWidget
*) NULL
;
2376 m_focusWidget
= (GtkWidget
*) NULL
;
2386 m_needParent
= true;
2387 m_isBeingDeleted
= false;
2389 m_showOnIdle
= false;
2392 m_nativeSizeEvent
= false;
2394 m_hasScrolling
= false;
2395 m_isScrolling
= false;
2396 m_mouseButtonDown
= false;
2397 m_blockScrollEvent
= false;
2399 // initialize scrolling stuff
2400 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2402 m_scrollBar
[dir
] = NULL
;
2403 m_scrollPos
[dir
] = 0;
2404 m_blockValueChanged
[dir
] = false;
2408 m_oldClientHeight
= 0;
2412 m_insertCallback
= (wxInsertChildFunction
) NULL
;
2414 m_acceptsFocus
= false;
2417 m_clipPaintRegion
= false;
2419 m_needsStyleChange
= false;
2421 m_cursor
= *wxSTANDARD_CURSOR
;
2424 m_dirtyTabOrder
= false;
2427 wxWindowGTK::wxWindowGTK()
2432 wxWindowGTK::wxWindowGTK( wxWindow
*parent
,
2437 const wxString
&name
)
2441 Create( parent
, id
, pos
, size
, style
, name
);
2444 bool wxWindowGTK::Create( wxWindow
*parent
,
2449 const wxString
&name
)
2451 if (!PreCreation( parent
, pos
, size
) ||
2452 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
2454 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2458 m_insertCallback
= wxInsertChildInWindow
;
2460 m_widget
= gtk_scrolled_window_new( (GtkAdjustment
*) NULL
, (GtkAdjustment
*) NULL
);
2461 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS
);
2463 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(m_widget
);
2465 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2466 scroll_class
->scrollbar_spacing
= 0;
2468 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
2470 m_scrollBar
[ScrollDir_Horz
] = GTK_RANGE(scrolledWindow
->hscrollbar
);
2471 m_scrollBar
[ScrollDir_Vert
] = GTK_RANGE(scrolledWindow
->vscrollbar
);
2472 if (GetLayoutDirection() == wxLayout_RightToLeft
)
2473 gtk_range_set_inverted( m_scrollBar
[ScrollDir_Horz
], TRUE
);
2475 m_wxwindow
= gtk_pizza_new();
2477 #ifndef __WXUNIVERSAL__
2478 GtkPizza
*pizza
= GTK_PIZZA(m_wxwindow
);
2480 if (HasFlag(wxRAISED_BORDER
))
2482 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_OUT
);
2484 else if (HasFlag(wxSUNKEN_BORDER
))
2486 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_IN
);
2488 else if (HasFlag(wxSIMPLE_BORDER
))
2490 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_THIN
);
2494 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_NONE
);
2496 #endif // __WXUNIVERSAL__
2498 gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow
);
2500 GTK_WIDGET_SET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS
);
2501 m_acceptsFocus
= true;
2503 // connect various scroll-related events
2504 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2506 // these handlers block mouse events to any window during scrolling
2507 // such as motion events and prevent GTK and wxWidgets from fighting
2508 // over where the slider should be
2509 g_signal_connect(m_scrollBar
[dir
], "button_press_event",
2510 G_CALLBACK(gtk_scrollbar_button_press_event
), this);
2511 g_signal_connect(m_scrollBar
[dir
], "button_release_event",
2512 G_CALLBACK(gtk_scrollbar_button_release_event
), this);
2514 gulong handler_id
= g_signal_connect(m_scrollBar
[dir
], "event_after",
2515 G_CALLBACK(gtk_scrollbar_event_after
), this);
2516 g_signal_handler_block(m_scrollBar
[dir
], handler_id
);
2518 // these handlers get notified when scrollbar slider moves
2519 g_signal_connect(m_scrollBar
[dir
], "value_changed",
2520 G_CALLBACK(gtk_scrollbar_value_changed
), this);
2523 gtk_widget_show( m_wxwindow
);
2526 m_parent
->DoAddChild( this );
2528 m_focusWidget
= m_wxwindow
;
2535 wxWindowGTK::~wxWindowGTK()
2539 if (g_focusWindow
== this)
2540 g_focusWindow
= NULL
;
2542 if ( g_delayedFocus
== this )
2543 g_delayedFocus
= NULL
;
2545 m_isBeingDeleted
= true;
2548 // destroy children before destroying this window itself
2551 // unhook focus handlers to prevent stray events being
2552 // propagated to this (soon to be) dead object
2553 if (m_focusWidget
!= NULL
)
2555 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2556 (gpointer
) gtk_window_focus_in_callback
,
2558 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2559 (gpointer
) gtk_window_focus_out_callback
,
2568 gdk_ic_destroy (m_ic
);
2570 gdk_ic_attr_destroy (m_icattr
);
2573 // delete before the widgets to avoid a crash on solaris
2578 gtk_widget_destroy( m_wxwindow
);
2579 m_wxwindow
= (GtkWidget
*) NULL
;
2584 gtk_widget_destroy( m_widget
);
2585 m_widget
= (GtkWidget
*) NULL
;
2589 bool wxWindowGTK::PreCreation( wxWindowGTK
*parent
, const wxPoint
&pos
, const wxSize
&size
)
2591 wxCHECK_MSG( !m_needParent
|| parent
, false, wxT("Need complete parent.") );
2593 // Use either the given size, or the default if -1 is given.
2594 // See wxWindowBase for these functions.
2595 m_width
= WidthDefault(size
.x
) ;
2596 m_height
= HeightDefault(size
.y
);
2604 void wxWindowGTK::PostCreation()
2606 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2612 // these get reported to wxWidgets -> wxPaintEvent
2614 g_signal_connect (m_wxwindow
, "expose_event",
2615 G_CALLBACK (gtk_window_expose_callback
), this);
2617 if (GetLayoutDirection() == wxLayout_LeftToRight
)
2618 gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow
), HasFlag( wxFULL_REPAINT_ON_RESIZE
) );
2621 // Create input method handler
2622 m_imData
= new wxGtkIMData
;
2624 // Cannot handle drawing preedited text yet
2625 gtk_im_context_set_use_preedit( m_imData
->context
, FALSE
);
2627 g_signal_connect (m_imData
->context
, "commit",
2628 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2630 // these are called when the "sunken" or "raised" borders are drawn
2631 g_signal_connect (m_widget
, "expose_event",
2632 G_CALLBACK (gtk_window_own_expose_callback
), this);
2637 if (!GTK_IS_WINDOW(m_widget
))
2639 if (m_focusWidget
== NULL
)
2640 m_focusWidget
= m_widget
;
2644 g_signal_connect (m_focusWidget
, "focus_in_event",
2645 G_CALLBACK (gtk_window_focus_in_callback
), this);
2646 g_signal_connect (m_focusWidget
, "focus_out_event",
2647 G_CALLBACK (gtk_window_focus_out_callback
), this);
2651 g_signal_connect_after (m_focusWidget
, "focus_in_event",
2652 G_CALLBACK (gtk_window_focus_in_callback
), this);
2653 g_signal_connect_after (m_focusWidget
, "focus_out_event",
2654 G_CALLBACK (gtk_window_focus_out_callback
), this);
2658 // connect to the various key and mouse handlers
2660 GtkWidget
*connect_widget
= GetConnectWidget();
2662 ConnectWidget( connect_widget
);
2664 /* We cannot set colours, fonts and cursors before the widget has
2665 been realized, so we do this directly after realization */
2666 g_signal_connect (connect_widget
, "realize",
2667 G_CALLBACK (gtk_window_realized_callback
), this);
2671 // Catch native resize events
2672 g_signal_connect (m_wxwindow
, "size_allocate",
2673 G_CALLBACK (gtk_window_size_callback
), this);
2675 // Initialize XIM support
2676 g_signal_connect (m_wxwindow
, "realize",
2677 G_CALLBACK (gtk_wxwindow_realized_callback
), this);
2679 // And resize XIM window
2680 g_signal_connect (m_wxwindow
, "size_allocate",
2681 G_CALLBACK (gtk_wxwindow_size_callback
), this);
2684 if (GTK_IS_COMBO(m_widget
))
2686 GtkCombo
*gcombo
= GTK_COMBO(m_widget
);
2688 g_signal_connect (gcombo
->entry
, "size_request",
2689 G_CALLBACK (wxgtk_combo_size_request_callback
),
2692 #ifdef GTK_IS_FILE_CHOOSER_BUTTON
2693 else if (!gtk_check_version(2,6,0) && GTK_IS_FILE_CHOOSER_BUTTON(m_widget
))
2695 // If we connect to the "size_request" signal of a GtkFileChooserButton
2696 // then that control won't be sized properly when placed inside sizers
2697 // (this can be tested removing this elseif and running XRC or WIDGETS samples)
2698 // FIXME: what should be done here ?
2703 // This is needed if we want to add our windows into native
2704 // GTK controls, such as the toolbar. With this callback, the
2705 // toolbar gets to know the correct size (the one set by the
2706 // programmer). Sadly, it misbehaves for wxComboBox.
2707 g_signal_connect (m_widget
, "size_request",
2708 G_CALLBACK (wxgtk_window_size_request_callback
),
2712 InheritAttributes();
2716 SetLayoutDirection(wxLayout_Default
);
2718 // unless the window was created initially hidden (i.e. Hide() had been
2719 // called before Create()), we should show it at GTK+ level as well
2721 gtk_widget_show( m_widget
);
2724 void wxWindowGTK::ConnectWidget( GtkWidget
*widget
)
2726 g_signal_connect (widget
, "key_press_event",
2727 G_CALLBACK (gtk_window_key_press_callback
), this);
2728 g_signal_connect (widget
, "key_release_event",
2729 G_CALLBACK (gtk_window_key_release_callback
), this);
2730 g_signal_connect (widget
, "button_press_event",
2731 G_CALLBACK (gtk_window_button_press_callback
), this);
2732 g_signal_connect (widget
, "button_release_event",
2733 G_CALLBACK (gtk_window_button_release_callback
), this);
2734 g_signal_connect (widget
, "motion_notify_event",
2735 G_CALLBACK (gtk_window_motion_notify_callback
), this);
2736 g_signal_connect (widget
, "scroll_event",
2737 G_CALLBACK (window_scroll_event
), this);
2738 g_signal_connect (widget
, "popup_menu",
2739 G_CALLBACK (wxgtk_window_popup_menu_callback
), this);
2740 g_signal_connect (widget
, "enter_notify_event",
2741 G_CALLBACK (gtk_window_enter_callback
), this);
2742 g_signal_connect (widget
, "leave_notify_event",
2743 G_CALLBACK (gtk_window_leave_callback
), this);
2746 bool wxWindowGTK::Destroy()
2748 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2752 return wxWindowBase::Destroy();
2755 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
)
2757 // inform the parent to perform the move
2758 gtk_pizza_set_size( GTK_PIZZA(m_parent
->m_wxwindow
), m_widget
, x
, y
, width
, height
);
2762 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
2764 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2765 wxASSERT_MSG( (m_parent
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") );
2767 if (m_resizing
) return; /* I don't like recursions */
2770 int currentX
, currentY
;
2771 GetPosition(¤tX
, ¤tY
);
2772 if (x
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2774 if (y
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2776 AdjustForParentClientOrigin(x
, y
, sizeFlags
);
2778 // calculate the best size if we should auto size the window
2779 if ( ((sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1) ||
2780 ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1) )
2782 const wxSize sizeBest
= GetBestSize();
2783 if ( (sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1 )
2785 if ( (sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1 )
2786 height
= sizeBest
.y
;
2794 int minWidth
= GetMinWidth(),
2795 minHeight
= GetMinHeight(),
2796 maxWidth
= GetMaxWidth(),
2797 maxHeight
= GetMaxHeight();
2799 if ((minWidth
!= -1) && (m_width
< minWidth
)) m_width
= minWidth
;
2800 if ((minHeight
!= -1) && (m_height
< minHeight
)) m_height
= minHeight
;
2801 if ((maxWidth
!= -1) && (m_width
> maxWidth
)) m_width
= maxWidth
;
2802 if ((maxHeight
!= -1) && (m_height
> maxHeight
)) m_height
= maxHeight
;
2804 #if wxUSE_TOOLBAR_NATIVE
2805 if (wxDynamicCast(GetParent(), wxToolBar
))
2807 // don't take the x,y values, they're wrong because toolbar sets them
2808 GtkWidget
*widget
= GTK_WIDGET(m_widget
);
2809 gtk_widget_set_size_request (widget
, m_width
, m_height
);
2813 if (m_parent
->m_wxwindow
== NULL
) // i.e. wxNotebook
2815 // don't set the size for children of wxNotebook, just take the values.
2823 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
2824 if ((sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) == 0)
2826 if (x
!= -1) m_x
= x
+ gtk_pizza_get_xoffset( pizza
);
2827 if (y
!= -1) m_y
= y
+ gtk_pizza_get_yoffset( pizza
);
2831 m_x
= x
+ gtk_pizza_get_xoffset( pizza
);
2832 m_y
= y
+ gtk_pizza_get_yoffset( pizza
);
2835 int left_border
= 0;
2836 int right_border
= 0;
2838 int bottom_border
= 0;
2840 /* the default button has a border around it */
2841 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
2843 GtkBorder
*default_border
= NULL
;
2844 gtk_widget_style_get( m_widget
, "default_border", &default_border
, NULL
);
2847 left_border
+= default_border
->left
;
2848 right_border
+= default_border
->right
;
2849 top_border
+= default_border
->top
;
2850 bottom_border
+= default_border
->bottom
;
2851 g_free( default_border
);
2855 DoMoveWindow( m_x
-top_border
,
2857 m_width
+left_border
+right_border
,
2858 m_height
+top_border
+bottom_border
);
2863 /* Sometimes the client area changes size without the
2864 whole windows's size changing, but if the whole
2865 windows's size doesn't change, no wxSizeEvent will
2866 normally be sent. Here we add an extra test if
2867 the client test has been changed and this will
2869 GetClientSize( &m_oldClientWidth
, &m_oldClientHeight
);
2873 wxPrintf( "OnSize sent from " );
2874 if (GetClassInfo() && GetClassInfo()->GetClassName())
2875 wxPrintf( GetClassInfo()->GetClassName() );
2876 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
2879 if (!m_nativeSizeEvent
)
2881 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
2882 event
.SetEventObject( this );
2883 GetEventHandler()->ProcessEvent( event
);
2889 bool wxWindowGTK::GtkShowFromOnIdle()
2891 if (IsShown() && m_showOnIdle
&& !GTK_WIDGET_VISIBLE (m_widget
))
2893 GtkAllocation alloc
;
2896 alloc
.width
= m_width
;
2897 alloc
.height
= m_height
;
2898 gtk_widget_size_allocate( m_widget
, &alloc
);
2899 gtk_widget_show( m_widget
);
2900 wxShowEvent
eventShow(GetId(), true);
2901 eventShow
.SetEventObject(this);
2902 GetEventHandler()->ProcessEvent(eventShow
);
2903 m_showOnIdle
= false;
2910 void wxWindowGTK::OnInternalIdle()
2912 // Check if we have to show window now
2913 if (GtkShowFromOnIdle()) return;
2915 if ( m_dirtyTabOrder
)
2917 m_dirtyTabOrder
= false;
2921 // Update style if the window was not yet realized
2922 // and SetBackgroundStyle(wxBG_STYLE_CUSTOM) was called
2923 if (m_needsStyleChange
)
2925 SetBackgroundStyle(GetBackgroundStyle());
2926 m_needsStyleChange
= false;
2929 wxCursor cursor
= m_cursor
;
2930 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
2934 /* I now set the cursor anew in every OnInternalIdle call
2935 as setting the cursor in a parent window also effects the
2936 windows above so that checking for the current cursor is
2941 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
2943 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2945 if (!g_globalCursor
.Ok())
2946 cursor
= *wxSTANDARD_CURSOR
;
2948 window
= m_widget
->window
;
2949 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
2950 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2953 else if ( m_widget
)
2955 GdkWindow
*window
= m_widget
->window
;
2956 if ( window
&& !GTK_WIDGET_NO_WINDOW(m_widget
) )
2957 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2961 if (wxUpdateUIEvent::CanUpdate(this))
2962 UpdateWindowUI(wxUPDATE_UI_FROMIDLE
);
2965 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
2967 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2969 if (width
) (*width
) = m_width
;
2970 if (height
) (*height
) = m_height
;
2973 void wxWindowGTK::DoSetClientSize( int width
, int height
)
2975 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2984 GetScrollbarWidth(m_widget
, dw
, dh
);
2987 #ifndef __WXUNIVERSAL__
2988 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
2990 // shadow border size is 2
2994 if (HasFlag(wxSIMPLE_BORDER
))
2996 // simple border size is 1
3000 #endif // __WXUNIVERSAL__
3006 SetSize(width
, height
);
3009 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
3011 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3022 GetScrollbarWidth(m_widget
, dw
, dh
);
3024 #ifndef __WXUNIVERSAL__
3025 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3027 // shadow border size is 2
3031 if (HasFlag(wxSIMPLE_BORDER
))
3033 // simple border size is 1
3037 #endif // __WXUNIVERSAL__
3047 if (width
) *width
= w
;
3048 if (height
) *height
= h
;
3051 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
3053 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3057 if (m_parent
&& m_parent
->m_wxwindow
)
3059 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
3060 dx
= gtk_pizza_get_xoffset( pizza
);
3061 dy
= gtk_pizza_get_yoffset( pizza
);
3064 if (m_x
== -1 && m_y
== -1)
3066 GdkWindow
*source
= (GdkWindow
*) NULL
;
3068 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3070 source
= m_widget
->window
;
3076 gdk_window_get_origin( source
, &org_x
, &org_y
);
3079 GetParent()->ScreenToClient(&org_x
, &org_y
);
3081 ((wxWindowGTK
*) this)->m_x
= org_x
;
3082 ((wxWindowGTK
*) this)->m_y
= org_y
;
3086 if (x
) (*x
) = m_x
- dx
;
3087 if (y
) (*y
) = m_y
- dy
;
3090 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
3092 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3094 if (!m_widget
->window
) return;
3096 GdkWindow
*source
= (GdkWindow
*) NULL
;
3098 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3100 source
= m_widget
->window
;
3104 gdk_window_get_origin( source
, &org_x
, &org_y
);
3108 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3110 org_x
+= m_widget
->allocation
.x
;
3111 org_y
+= m_widget
->allocation
.y
;
3118 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3119 *x
= (GetClientSize().x
- *x
) + org_x
;
3127 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
3129 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3131 if (!m_widget
->window
) return;
3133 GdkWindow
*source
= (GdkWindow
*) NULL
;
3135 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3137 source
= m_widget
->window
;
3141 gdk_window_get_origin( source
, &org_x
, &org_y
);
3145 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3147 org_x
+= m_widget
->allocation
.x
;
3148 org_y
+= m_widget
->allocation
.y
;
3154 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3155 *x
= (GetClientSize().x
- *x
) - org_x
;
3162 bool wxWindowGTK::Show( bool show
)
3164 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3166 if (!wxWindowBase::Show(show
))
3176 gtk_widget_show( m_widget
);
3177 wxShowEvent
eventShow(GetId(), show
);
3178 eventShow
.SetEventObject(this);
3179 GetEventHandler()->ProcessEvent(eventShow
);
3184 gtk_widget_hide( m_widget
);
3185 wxShowEvent
eventShow(GetId(), show
);
3186 eventShow
.SetEventObject(this);
3187 GetEventHandler()->ProcessEvent(eventShow
);
3193 static void wxWindowNotifyEnable(wxWindowGTK
* win
, bool enable
)
3195 win
->OnParentEnable(enable
);
3197 // Recurse, so that children have the opportunity to Do The Right Thing
3198 // and reset colours that have been messed up by a parent's (really ancestor's)
3200 for ( wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
3202 node
= node
->GetNext() )
3204 wxWindow
*child
= node
->GetData();
3205 if (!child
->IsKindOf(CLASSINFO(wxDialog
)) && !child
->IsKindOf(CLASSINFO(wxFrame
)))
3206 wxWindowNotifyEnable(child
, enable
);
3210 bool wxWindowGTK::Enable( bool enable
)
3212 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3214 if (!wxWindowBase::Enable(enable
))
3220 gtk_widget_set_sensitive( m_widget
, enable
);
3222 gtk_widget_set_sensitive( m_wxwindow
, enable
);
3224 wxWindowNotifyEnable(this, enable
);
3229 int wxWindowGTK::GetCharHeight() const
3231 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
3233 wxFont font
= GetFont();
3234 wxCHECK_MSG( font
.Ok(), 12, wxT("invalid font") );
3236 PangoContext
*context
= NULL
;
3238 context
= gtk_widget_get_pango_context( m_widget
);
3243 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3244 PangoLayout
*layout
= pango_layout_new(context
);
3245 pango_layout_set_font_description(layout
, desc
);
3246 pango_layout_set_text(layout
, "H", 1);
3247 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3249 PangoRectangle rect
;
3250 pango_layout_line_get_extents(line
, NULL
, &rect
);
3252 g_object_unref (layout
);
3254 return (int) PANGO_PIXELS(rect
.height
);
3257 int wxWindowGTK::GetCharWidth() const
3259 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
3261 wxFont font
= GetFont();
3262 wxCHECK_MSG( font
.Ok(), 8, wxT("invalid font") );
3264 PangoContext
*context
= NULL
;
3266 context
= gtk_widget_get_pango_context( m_widget
);
3271 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3272 PangoLayout
*layout
= pango_layout_new(context
);
3273 pango_layout_set_font_description(layout
, desc
);
3274 pango_layout_set_text(layout
, "g", 1);
3275 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3277 PangoRectangle rect
;
3278 pango_layout_line_get_extents(line
, NULL
, &rect
);
3280 g_object_unref (layout
);
3282 return (int) PANGO_PIXELS(rect
.width
);
3285 void wxWindowGTK::GetTextExtent( const wxString
& string
,
3289 int *externalLeading
,
3290 const wxFont
*theFont
) const
3292 wxFont fontToUse
= theFont
? *theFont
: GetFont();
3294 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
3303 PangoContext
*context
= NULL
;
3305 context
= gtk_widget_get_pango_context( m_widget
);
3314 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
3315 PangoLayout
*layout
= pango_layout_new(context
);
3316 pango_layout_set_font_description(layout
, desc
);
3318 const wxCharBuffer data
= wxGTK_CONV( string
);
3320 pango_layout_set_text(layout
, data
, strlen(data
));
3323 PangoRectangle rect
;
3324 pango_layout_get_extents(layout
, NULL
, &rect
);
3326 if (x
) (*x
) = (wxCoord
) PANGO_PIXELS(rect
.width
);
3327 if (y
) (*y
) = (wxCoord
) PANGO_PIXELS(rect
.height
);
3330 PangoLayoutIter
*iter
= pango_layout_get_iter(layout
);
3331 int baseline
= pango_layout_iter_get_baseline(iter
);
3332 pango_layout_iter_free(iter
);
3333 *descent
= *y
- PANGO_PIXELS(baseline
);
3335 if (externalLeading
) (*externalLeading
) = 0; // ??
3337 g_object_unref (layout
);
3340 bool wxWindowGTK::GTKSetDelayedFocusIfNeeded()
3342 if ( g_delayedFocus
== this )
3344 if ( GTK_WIDGET_REALIZED(m_widget
) )
3346 gtk_widget_grab_focus(m_widget
);
3347 g_delayedFocus
= NULL
;
3356 void wxWindowGTK::SetFocus()
3358 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3361 // don't do anything if we already have focus
3367 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow
))
3369 gtk_widget_grab_focus (m_wxwindow
);
3374 if (GTK_IS_CONTAINER(m_widget
))
3376 gtk_widget_child_focus( m_widget
, GTK_DIR_TAB_FORWARD
);
3379 if (GTK_WIDGET_CAN_FOCUS(m_widget
) && !GTK_WIDGET_HAS_FOCUS (m_widget
) )
3382 if (!GTK_WIDGET_REALIZED(m_widget
))
3384 // we can't set the focus to the widget now so we remember that
3385 // it should be focused and will do it later, during the idle
3386 // time, as soon as we can
3387 wxLogTrace(TRACE_FOCUS
,
3388 _T("Delaying setting focus to %s(%s)"),
3389 GetClassInfo()->GetClassName(), GetLabel().c_str());
3391 g_delayedFocus
= this;
3395 wxLogTrace(TRACE_FOCUS
,
3396 _T("Setting focus to %s(%s)"),
3397 GetClassInfo()->GetClassName(), GetLabel().c_str());
3399 gtk_widget_grab_focus (m_widget
);
3404 wxLogTrace(TRACE_FOCUS
,
3405 _T("Can't set focus to %s(%s)"),
3406 GetClassInfo()->GetClassName(), GetLabel().c_str());
3411 bool wxWindowGTK::AcceptsFocus() const
3413 return m_acceptsFocus
&& wxWindowBase::AcceptsFocus();
3416 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3418 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3420 wxWindowGTK
*oldParent
= m_parent
,
3421 *newParent
= (wxWindowGTK
*)newParentBase
;
3423 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3425 if ( !wxWindowBase::Reparent(newParent
) )
3428 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3430 /* prevent GTK from deleting the widget arbitrarily */
3431 gtk_widget_ref( m_widget
);
3435 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3438 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3442 if (GTK_WIDGET_VISIBLE (newParent
->m_widget
))
3444 m_showOnIdle
= true;
3445 gtk_widget_hide( m_widget
);
3448 /* insert GTK representation */
3449 (*(newParent
->m_insertCallback
))(newParent
, this);
3452 /* reverse: prevent GTK from deleting the widget arbitrarily */
3453 gtk_widget_unref( m_widget
);
3455 SetLayoutDirection(wxLayout_Default
);
3460 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3462 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3464 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3466 wxASSERT_MSG( (m_insertCallback
!= NULL
), wxT("invalid child insertion function") );
3471 /* insert GTK representation */
3472 (*m_insertCallback
)(this, child
);
3475 void wxWindowGTK::AddChild(wxWindowBase
*child
)
3477 wxWindowBase::AddChild(child
);
3478 m_dirtyTabOrder
= true;
3480 wxapp_install_idle_handler();
3483 void wxWindowGTK::RemoveChild(wxWindowBase
*child
)
3485 wxWindowBase::RemoveChild(child
);
3486 m_dirtyTabOrder
= true;
3488 wxapp_install_idle_handler();
3492 wxLayoutDirection
wxWindowGTK::GTKGetLayout(GtkWidget
*widget
)
3494 return gtk_widget_get_direction(GTK_WIDGET(widget
)) == GTK_TEXT_DIR_RTL
3495 ? wxLayout_RightToLeft
3496 : wxLayout_LeftToRight
;
3500 void wxWindowGTK::GTKSetLayout(GtkWidget
*widget
, wxLayoutDirection dir
)
3502 wxASSERT_MSG( dir
!= wxLayout_Default
, _T("invalid layout direction") );
3504 gtk_widget_set_direction(GTK_WIDGET(widget
),
3505 dir
== wxLayout_RightToLeft
? GTK_TEXT_DIR_RTL
3506 : GTK_TEXT_DIR_LTR
);
3509 wxLayoutDirection
wxWindowGTK::GetLayoutDirection() const
3511 return GTKGetLayout(m_widget
);
3514 void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir
)
3516 if ( dir
== wxLayout_Default
)
3518 const wxWindow
*const parent
= GetParent();
3521 // inherit layout from parent.
3522 dir
= parent
->GetLayoutDirection();
3524 else // no parent, use global default layout
3526 dir
= wxTheApp
->GetLayoutDirection();
3530 if ( dir
== wxLayout_Default
)
3533 GTKSetLayout(m_widget
, dir
);
3536 GTKSetLayout(m_wxwindow
, dir
);
3540 wxWindowGTK::AdjustForLayoutDirection(wxCoord x
,
3541 wxCoord
WXUNUSED(width
),
3542 wxCoord
WXUNUSED(widthTotal
)) const
3544 // We now mirrors the coordinates of RTL windows in GtkPizza
3548 void wxWindowGTK::DoMoveInTabOrder(wxWindow
*win
, MoveKind move
)
3550 wxWindowBase::DoMoveInTabOrder(win
, move
);
3551 m_dirtyTabOrder
= true;
3553 wxapp_install_idle_handler();
3556 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3558 // none needed by default
3562 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget
* WXUNUSED(w
))
3564 // nothing to do by default since none is needed
3567 void wxWindowGTK::RealizeTabOrder()
3571 if ( !m_children
.empty() )
3573 // we don't only construct the correct focus chain but also use
3574 // this opportunity to update the mnemonic widgets for the widgets
3577 GList
*chain
= NULL
;
3578 wxWindowGTK
* mnemonicWindow
= NULL
;
3580 for ( wxWindowList::const_iterator i
= m_children
.begin();
3581 i
!= m_children
.end();
3584 wxWindowGTK
*win
= *i
;
3586 if ( mnemonicWindow
)
3588 if ( win
->AcceptsFocusFromKeyboard() )
3590 // wxComboBox et al. needs to focus on on a different
3591 // widget than m_widget, so if the main widget isn't
3592 // focusable try the connect widget
3593 GtkWidget
* w
= win
->m_widget
;
3594 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3596 w
= win
->GetConnectWidget();
3597 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3603 mnemonicWindow
->GTKWidgetDoSetMnemonic(w
);
3604 mnemonicWindow
= NULL
;
3608 else if ( win
->GTKWidgetNeedsMnemonic() )
3610 mnemonicWindow
= win
;
3613 chain
= g_list_prepend(chain
, win
->m_widget
);
3616 chain
= g_list_reverse(chain
);
3618 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow
), chain
);
3623 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow
));
3628 void wxWindowGTK::Raise()
3630 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3632 if (m_wxwindow
&& m_wxwindow
->window
)
3634 gdk_window_raise( m_wxwindow
->window
);
3636 else if (m_widget
->window
)
3638 gdk_window_raise( m_widget
->window
);
3642 void wxWindowGTK::Lower()
3644 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3646 if (m_wxwindow
&& m_wxwindow
->window
)
3648 gdk_window_lower( m_wxwindow
->window
);
3650 else if (m_widget
->window
)
3652 gdk_window_lower( m_widget
->window
);
3656 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3658 if ( !wxWindowBase::SetCursor(cursor
.Ok() ? cursor
: *wxSTANDARD_CURSOR
) )
3666 void wxWindowGTK::GTKUpdateCursor()
3668 wxCursor
cursor(g_globalCursor
.Ok() ? g_globalCursor
: GetCursor());
3671 wxArrayGdkWindows windowsThis
;
3672 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3675 gdk_window_set_cursor(winThis
, cursor
.GetCursor());
3679 const size_t count
= windowsThis
.size();
3680 for ( size_t n
= 0; n
< count
; n
++ )
3682 GdkWindow
*win
= windowsThis
[n
];
3685 wxFAIL_MSG(_T("NULL window returned by GTKGetWindow()?"));
3689 gdk_window_set_cursor(win
, cursor
.GetCursor());
3695 void wxWindowGTK::WarpPointer( int x
, int y
)
3697 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3699 // We provide this function ourselves as it is
3700 // missing in GDK (top of this file).
3702 GdkWindow
*window
= (GdkWindow
*) NULL
;
3704 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3706 window
= GetConnectWidget()->window
;
3709 gdk_window_warp_pointer( window
, x
, y
);
3712 wxWindowGTK::ScrollDir
wxWindowGTK::ScrollDirFromRange(GtkRange
*range
) const
3714 // find the scrollbar which generated the event
3715 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
3717 if ( range
== m_scrollBar
[dir
] )
3718 return (ScrollDir
)dir
;
3721 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3723 return ScrollDir_Max
;
3726 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir
, ScrollUnit unit
, int units
)
3728 bool changed
= false;
3729 GtkRange
* range
= m_scrollBar
[dir
];
3730 if ( range
&& units
)
3732 GtkAdjustment
* adj
= range
->adjustment
;
3733 gdouble inc
= unit
== ScrollUnit_Line
? adj
->step_increment
3734 : adj
->page_increment
;
3736 const int posOld
= int(adj
->value
+ 0.5);
3737 gtk_range_set_value(range
, posOld
+ units
*inc
);
3739 changed
= int(adj
->value
+ 0.5) != posOld
;
3745 bool wxWindowGTK::ScrollLines(int lines
)
3747 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Line
, lines
);
3750 bool wxWindowGTK::ScrollPages(int pages
)
3752 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Page
, pages
);
3755 void wxWindowGTK::Refresh( bool eraseBackground
, const wxRect
*rect
)
3759 if (!m_widget
->window
)
3764 if (!GTK_PIZZA(m_wxwindow
)->bin_window
) return;
3766 GdkRectangle gdk_rect
,
3770 gdk_rect
.x
= rect
->x
;
3771 gdk_rect
.y
= rect
->y
;
3772 gdk_rect
.width
= rect
->width
;
3773 gdk_rect
.height
= rect
->height
;
3774 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3775 gdk_rect
.x
= GetClientSize().x
- gdk_rect
.x
- gdk_rect
.width
;
3779 else // invalidate everything
3784 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, p
, TRUE
);
3788 void wxWindowGTK::Update()
3792 // when we call Update() we really want to update the window immediately on
3793 // screen, even if it means flushing the entire queue and hence slowing down
3794 // everything -- but it should still be done, it's just that Update() should
3795 // be called very rarely
3799 void wxWindowGTK::GtkUpdate()
3801 if (m_wxwindow
&& GTK_PIZZA(m_wxwindow
)->bin_window
)
3802 gdk_window_process_updates( GTK_PIZZA(m_wxwindow
)->bin_window
, FALSE
);
3803 if (m_widget
&& m_widget
->window
)
3804 gdk_window_process_updates( m_widget
->window
, FALSE
);
3806 // for consistency with other platforms (and also because it's convenient
3807 // to be able to update an entire TLW by calling Update() only once), we
3808 // should also update all our children here
3809 for ( wxWindowList::compatibility_iterator node
= GetChildren().GetFirst();
3811 node
= node
->GetNext() )
3813 node
->GetData()->GtkUpdate();
3817 bool wxWindowGTK::DoIsExposed( int x
, int y
) const
3819 return m_updateRegion
.Contains(x
, y
) != wxOutRegion
;
3823 bool wxWindowGTK::DoIsExposed( int x
, int y
, int w
, int h
) const
3826 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3827 return m_updateRegion
.Contains(x
-w
, y
, w
, h
) != wxOutRegion
;
3830 return m_updateRegion
.Contains(x
, y
, w
, h
) != wxOutRegion
;
3833 void wxWindowGTK::GtkSendPaintEvents()
3837 m_updateRegion
.Clear();
3841 // Clip to paint region in wxClientDC
3842 m_clipPaintRegion
= true;
3844 m_nativeUpdateRegion
= m_updateRegion
;
3846 if (GetLayoutDirection() == wxLayout_RightToLeft
)
3848 // Transform m_updateRegion under RTL
3849 m_updateRegion
.Clear();
3852 gdk_window_get_geometry( GTK_PIZZA(m_wxwindow
)->bin_window
,
3853 NULL
, NULL
, &width
, NULL
, NULL
);
3855 wxRegionIterator
upd( m_nativeUpdateRegion
);
3859 rect
.x
= upd
.GetX();
3860 rect
.y
= upd
.GetY();
3861 rect
.width
= upd
.GetWidth();
3862 rect
.height
= upd
.GetHeight();
3864 rect
.x
= width
- rect
.x
- rect
.width
;
3865 m_updateRegion
.Union( rect
);
3871 // widget to draw on
3872 GtkPizza
*pizza
= GTK_PIZZA (m_wxwindow
);
3874 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM
))
3876 // find ancestor from which to steal background
3877 wxWindow
*parent
= wxGetTopLevelParent((wxWindow
*)this);
3879 parent
= (wxWindow
*)this;
3881 if (GTK_WIDGET_MAPPED(parent
->m_widget
))
3883 wxRegionIterator
upd( m_nativeUpdateRegion
);
3887 rect
.x
= upd
.GetX();
3888 rect
.y
= upd
.GetY();
3889 rect
.width
= upd
.GetWidth();
3890 rect
.height
= upd
.GetHeight();
3892 gtk_paint_flat_box( parent
->m_widget
->style
,
3894 (GtkStateType
)GTK_WIDGET_STATE(m_wxwindow
),
3907 wxWindowDC
dc( (wxWindow
*)this );
3908 dc
.SetClippingRegion( m_updateRegion
);
3910 wxEraseEvent
erase_event( GetId(), &dc
);
3911 erase_event
.SetEventObject( this );
3913 GetEventHandler()->ProcessEvent(erase_event
);
3916 wxNcPaintEvent
nc_paint_event( GetId() );
3917 nc_paint_event
.SetEventObject( this );
3918 GetEventHandler()->ProcessEvent( nc_paint_event
);
3920 wxPaintEvent
paint_event( GetId() );
3921 paint_event
.SetEventObject( this );
3922 GetEventHandler()->ProcessEvent( paint_event
);
3924 m_clipPaintRegion
= false;
3926 m_updateRegion
.Clear();
3927 m_nativeUpdateRegion
.Clear();
3930 void wxWindowGTK::SetDoubleBuffered( bool on
)
3932 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3935 gtk_widget_set_double_buffered( m_wxwindow
, on
);
3938 void wxWindowGTK::ClearBackground()
3940 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3944 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3946 wxWindowBase::DoSetToolTip(tip
);
3949 m_tooltip
->Apply( (wxWindow
*)this );
3952 void wxWindowGTK::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
3954 wxString
tmp( tip
);
3955 gtk_tooltips_set_tip( tips
, GetConnectWidget(), wxGTK_CONV(tmp
), (gchar
*) NULL
);
3957 #endif // wxUSE_TOOLTIPS
3959 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
3961 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3963 if (!wxWindowBase::SetBackgroundColour(colour
))
3968 // We need the pixel value e.g. for background clearing.
3969 m_backgroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3972 // apply style change (forceStyle=true so that new style is applied
3973 // even if the bg colour changed from valid to wxNullColour)
3974 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM
)
3975 ApplyWidgetStyle(true);
3980 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
3982 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3984 if (!wxWindowBase::SetForegroundColour(colour
))
3991 // We need the pixel value e.g. for background clearing.
3992 m_foregroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3995 // apply style change (forceStyle=true so that new style is applied
3996 // even if the bg colour changed from valid to wxNullColour):
3997 ApplyWidgetStyle(true);
4002 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
4004 return gtk_widget_get_pango_context( m_widget
);
4007 GtkRcStyle
*wxWindowGTK::CreateWidgetStyle(bool forceStyle
)
4009 // do we need to apply any changes at all?
4012 !m_foregroundColour
.Ok() && !m_backgroundColour
.Ok() )
4017 GtkRcStyle
*style
= gtk_rc_style_new();
4022 pango_font_description_copy( m_font
.GetNativeFontInfo()->description
);
4025 if ( m_foregroundColour
.Ok() )
4027 const GdkColor
*fg
= m_foregroundColour
.GetColor();
4029 style
->fg
[GTK_STATE_NORMAL
] = *fg
;
4030 style
->color_flags
[GTK_STATE_NORMAL
] = GTK_RC_FG
;
4032 style
->fg
[GTK_STATE_PRELIGHT
] = *fg
;
4033 style
->color_flags
[GTK_STATE_PRELIGHT
] = GTK_RC_FG
;
4035 style
->fg
[GTK_STATE_ACTIVE
] = *fg
;
4036 style
->color_flags
[GTK_STATE_ACTIVE
] = GTK_RC_FG
;
4039 if ( m_backgroundColour
.Ok() )
4041 const GdkColor
*bg
= m_backgroundColour
.GetColor();
4043 style
->bg
[GTK_STATE_NORMAL
] = *bg
;
4044 style
->base
[GTK_STATE_NORMAL
] = *bg
;
4045 style
->color_flags
[GTK_STATE_NORMAL
] = (GtkRcFlags
)
4046 (style
->color_flags
[GTK_STATE_NORMAL
] | GTK_RC_BG
| GTK_RC_BASE
);
4048 style
->bg
[GTK_STATE_PRELIGHT
] = *bg
;
4049 style
->base
[GTK_STATE_PRELIGHT
] = *bg
;
4050 style
->color_flags
[GTK_STATE_PRELIGHT
] = (GtkRcFlags
)
4051 (style
->color_flags
[GTK_STATE_PRELIGHT
] | GTK_RC_BG
| GTK_RC_BASE
);
4053 style
->bg
[GTK_STATE_ACTIVE
] = *bg
;
4054 style
->base
[GTK_STATE_ACTIVE
] = *bg
;
4055 style
->color_flags
[GTK_STATE_ACTIVE
] = (GtkRcFlags
)
4056 (style
->color_flags
[GTK_STATE_ACTIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
4058 style
->bg
[GTK_STATE_INSENSITIVE
] = *bg
;
4059 style
->base
[GTK_STATE_INSENSITIVE
] = *bg
;
4060 style
->color_flags
[GTK_STATE_INSENSITIVE
] = (GtkRcFlags
)
4061 (style
->color_flags
[GTK_STATE_INSENSITIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
4067 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle
)
4069 GtkRcStyle
*style
= CreateWidgetStyle(forceStyle
);
4072 DoApplyWidgetStyle(style
);
4073 gtk_rc_style_unref(style
);
4076 // Style change may affect GTK+'s size calculation:
4077 InvalidateBestSize();
4080 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle
*style
)
4083 gtk_widget_modify_style(m_wxwindow
, style
);
4085 gtk_widget_modify_style(m_widget
, style
);
4088 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style
)
4090 wxWindowBase::SetBackgroundStyle(style
);
4092 if (style
== wxBG_STYLE_CUSTOM
)
4094 GdkWindow
*window
= (GdkWindow
*) NULL
;
4096 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4098 window
= GetConnectWidget()->window
;
4102 // Make sure GDK/X11 doesn't refresh the window
4104 gdk_window_set_back_pixmap( window
, None
, False
);
4106 Display
* display
= GDK_WINDOW_DISPLAY(window
);
4109 m_needsStyleChange
= false;
4112 // Do in OnIdle, because the window is not yet available
4113 m_needsStyleChange
= true;
4115 // Don't apply widget style, or we get a grey background
4119 // apply style change (forceStyle=true so that new style is applied
4120 // even if the bg colour changed from valid to wxNullColour):
4121 ApplyWidgetStyle(true);
4126 #if wxUSE_DRAG_AND_DROP
4128 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
4130 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4132 GtkWidget
*dnd_widget
= GetConnectWidget();
4134 if (m_dropTarget
) m_dropTarget
->UnregisterWidget( dnd_widget
);
4136 if (m_dropTarget
) delete m_dropTarget
;
4137 m_dropTarget
= dropTarget
;
4139 if (m_dropTarget
) m_dropTarget
->RegisterWidget( dnd_widget
);
4142 #endif // wxUSE_DRAG_AND_DROP
4144 GtkWidget
* wxWindowGTK::GetConnectWidget()
4146 GtkWidget
*connect_widget
= m_widget
;
4147 if (m_wxwindow
) connect_widget
= m_wxwindow
;
4149 return connect_widget
;
4152 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow
*window
) const
4154 wxArrayGdkWindows windowsThis
;
4155 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
4157 return winThis
? window
== winThis
4158 : windowsThis
.Index(window
) != wxNOT_FOUND
;
4161 GdkWindow
*wxWindowGTK::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const
4163 return m_wxwindow
? GTK_PIZZA(m_wxwindow
)->bin_window
: m_widget
->window
;
4166 bool wxWindowGTK::SetFont( const wxFont
&font
)
4168 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
4170 if (!wxWindowBase::SetFont(font
))
4173 // apply style change (forceStyle=true so that new style is applied
4174 // even if the font changed from valid to wxNullFont):
4175 ApplyWidgetStyle(true);
4180 void wxWindowGTK::DoCaptureMouse()
4182 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4184 GdkWindow
*window
= (GdkWindow
*) NULL
;
4186 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4188 window
= GetConnectWidget()->window
;
4190 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4192 const wxCursor
* cursor
= &m_cursor
;
4194 cursor
= wxSTANDARD_CURSOR
;
4196 gdk_pointer_grab( window
, FALSE
,
4198 (GDK_BUTTON_PRESS_MASK
|
4199 GDK_BUTTON_RELEASE_MASK
|
4200 GDK_POINTER_MOTION_HINT_MASK
|
4201 GDK_POINTER_MOTION_MASK
),
4203 cursor
->GetCursor(),
4204 (guint32
)GDK_CURRENT_TIME
);
4205 g_captureWindow
= this;
4206 g_captureWindowHasMouse
= true;
4209 void wxWindowGTK::DoReleaseMouse()
4211 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4213 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4215 g_captureWindow
= (wxWindowGTK
*) NULL
;
4217 GdkWindow
*window
= (GdkWindow
*) NULL
;
4219 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4221 window
= GetConnectWidget()->window
;
4226 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4230 wxWindow
*wxWindowBase::GetCapture()
4232 return (wxWindow
*)g_captureWindow
;
4235 bool wxWindowGTK::IsRetained() const
4240 void wxWindowGTK::BlockScrollEvent()
4242 wxASSERT(!m_blockScrollEvent
);
4243 m_blockScrollEvent
= true;
4246 void wxWindowGTK::UnblockScrollEvent()
4248 wxASSERT(m_blockScrollEvent
);
4249 m_blockScrollEvent
= false;
4252 void wxWindowGTK::SetScrollbar(int orient
,
4256 bool WXUNUSED(update
))
4258 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4259 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4263 m_hasScrolling
= true;
4267 // GtkRange requires upper > lower
4272 if (pos
> range
- thumbVisible
)
4273 pos
= range
- thumbVisible
;
4276 GtkAdjustment
* adj
= m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
;
4277 adj
->step_increment
= 1;
4278 adj
->page_increment
=
4279 adj
->page_size
= thumbVisible
;
4281 SetScrollPos(orient
, pos
);
4282 gtk_adjustment_changed(adj
);
4285 void wxWindowGTK::SetScrollPos(int orient
, int pos
, bool WXUNUSED(refresh
))
4287 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4288 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4290 // This check is more than an optimization. Without it, the slider
4291 // will not move smoothly while tracking when using wxScrollHelper.
4292 if (GetScrollPos(orient
) != pos
)
4294 const int dir
= ScrollDirFromOrient(orient
);
4295 GtkAdjustment
* adj
= m_scrollBar
[dir
]->adjustment
;
4296 const int max
= int(adj
->upper
- adj
->page_size
);
4301 m_scrollPos
[dir
] = adj
->value
= pos
;
4303 // If a "value_changed" signal emission is not already in progress
4304 if (!m_blockValueChanged
[dir
])
4306 gtk_adjustment_value_changed(adj
);
4311 int wxWindowGTK::GetScrollThumb(int orient
) const
4313 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4314 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4316 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->page_size
);
4319 int wxWindowGTK::GetScrollPos( int orient
) const
4321 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4322 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4324 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->value
+ 0.5);
4327 int wxWindowGTK::GetScrollRange( int orient
) const
4329 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4330 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4332 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->upper
);
4335 // Determine if increment is the same as +/-x, allowing for some small
4336 // difference due to possible inexactness in floating point arithmetic
4337 static inline bool IsScrollIncrement(double increment
, double x
)
4339 wxASSERT(increment
> 0);
4340 const double tolerance
= 1.0 / 1024;
4341 return fabs(increment
- fabs(x
)) < tolerance
;
4344 wxEventType
wxWindowGTK::GetScrollEventType(GtkRange
* range
)
4349 wxapp_install_idle_handler();
4351 wxASSERT(range
== m_scrollBar
[0] || range
== m_scrollBar
[1]);
4353 const int barIndex
= range
== m_scrollBar
[1];
4354 GtkAdjustment
* adj
= range
->adjustment
;
4356 const int value
= int(adj
->value
+ 0.5);
4358 // save previous position
4359 const double oldPos
= m_scrollPos
[barIndex
];
4360 // update current position
4361 m_scrollPos
[barIndex
] = adj
->value
;
4362 // If event should be ignored, or integral position has not changed
4363 if (!m_hasVMT
|| g_blockEventsOnDrag
|| value
== int(oldPos
+ 0.5))
4368 wxEventType eventType
= wxEVT_SCROLL_THUMBTRACK
;
4371 // Difference from last change event
4372 const double diff
= adj
->value
- oldPos
;
4373 const bool isDown
= diff
> 0;
4375 if (IsScrollIncrement(adj
->step_increment
, diff
))
4377 eventType
= isDown
? wxEVT_SCROLL_LINEDOWN
: wxEVT_SCROLL_LINEUP
;
4379 else if (IsScrollIncrement(adj
->page_increment
, diff
))
4381 eventType
= isDown
? wxEVT_SCROLL_PAGEDOWN
: wxEVT_SCROLL_PAGEUP
;
4383 else if (m_mouseButtonDown
)
4385 // Assume track event
4386 m_isScrolling
= true;
4392 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4394 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4396 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4398 // No scrolling requested.
4399 if ((dx
== 0) && (dy
== 0)) return;
4401 m_clipPaintRegion
= true;
4403 if (GetLayoutDirection() == wxLayout_RightToLeft
)
4404 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), dx
, -dy
);
4406 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), -dx
, -dy
);
4408 m_clipPaintRegion
= false;
4411 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget
* w
, int wxstyle
)
4413 //RN: Note that static controls usually have no border on gtk, so maybe
4414 //it makes sense to treat that as simply no border at the wx level
4416 if (!(wxstyle
& wxNO_BORDER
) && !(wxstyle
& wxBORDER_STATIC
))
4418 GtkShadowType gtkstyle
;
4420 if(wxstyle
& wxBORDER_RAISED
)
4421 gtkstyle
= GTK_SHADOW_OUT
;
4422 else if (wxstyle
& wxBORDER_SUNKEN
)
4423 gtkstyle
= GTK_SHADOW_IN
;
4424 else if (wxstyle
& wxBORDER_DOUBLE
)
4425 gtkstyle
= GTK_SHADOW_ETCHED_IN
;
4427 gtkstyle
= GTK_SHADOW_IN
;
4429 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w
),
4434 void wxWindowGTK::SetWindowStyleFlag( long style
)
4436 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4437 wxWindowBase::SetWindowStyleFlag(style
);
4440 // Find the wxWindow at the current mouse position, also returning the mouse
4442 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4444 pt
= wxGetMousePosition();
4445 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4449 // Get the current mouse position.
4450 wxPoint
wxGetMousePosition()
4452 /* This crashes when used within wxHelpContext,
4453 so we have to use the X-specific implementation below.
4455 GdkModifierType *mask;
4456 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4458 return wxPoint(x, y);
4462 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4464 Display
*display
= windowAtPtr
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY();
4465 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4466 Window rootReturn
, childReturn
;
4467 int rootX
, rootY
, winX
, winY
;
4468 unsigned int maskReturn
;
4470 XQueryPointer (display
,
4474 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4475 return wxPoint(rootX
, rootY
);
4479 // Needed for implementing e.g. combobox on wxGTK within a modal dialog.
4480 void wxAddGrab(wxWindow
* window
)
4482 gtk_grab_add( (GtkWidget
*) window
->GetHandle() );
4485 void wxRemoveGrab(wxWindow
* window
)
4487 gtk_grab_remove( (GtkWidget
*) window
->GetHandle() );