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"
38 #include "wx/module.h"
40 #if wxUSE_DRAG_AND_DROP
45 #include "wx/tooltip.h"
52 #include "wx/fontutil.h"
55 #include "wx/thread.h"
60 // FIXME: Due to a hack we use GtkCombo in here, which is deprecated since gtk2.3.0
61 #include <gtk/gtkversion.h>
62 #if defined(GTK_DISABLE_DEPRECATED) && GTK_CHECK_VERSION(2,3,0)
63 #undef GTK_DISABLE_DEPRECATED
66 #include "wx/gtk/private.h"
67 #include <gdk/gdkprivate.h>
68 #include <gdk/gdkkeysyms.h>
72 #include <gtk/gtkprivate.h>
74 #include "wx/gtk/win_gtk.h"
76 #include <pango/pangox.h>
82 extern GtkContainerClass
*pizza_parent_class
;
84 //-----------------------------------------------------------------------------
85 // documentation on internals
86 //-----------------------------------------------------------------------------
89 I have been asked several times about writing some documentation about
90 the GTK port of wxWidgets, especially its internal structures. Obviously,
91 you cannot understand wxGTK without knowing a little about the GTK, but
92 some more information about what the wxWindow, which is the base class
93 for all other window classes, does seems required as well.
97 What does wxWindow do? It contains the common interface for the following
98 jobs of its descendants:
100 1) Define the rudimentary behaviour common to all window classes, such as
101 resizing, intercepting user input (so as to make it possible to use these
102 events for special purposes in a derived class), window names etc.
104 2) Provide the possibility to contain and manage children, if the derived
105 class is allowed to contain children, which holds true for those window
106 classes which do not display a native GTK widget. To name them, these
107 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
108 work classes are a special case and are handled a bit differently from
109 the rest. The same holds true for the wxNotebook class.
111 3) Provide the possibility to draw into a client area of a window. This,
112 too, only holds true for classes that do not display a native GTK widget
115 4) Provide the entire mechanism for scrolling widgets. This actual inter-
116 face for this is usually in wxScrolledWindow, but the GTK implementation
119 5) A multitude of helper or extra methods for special purposes, such as
120 Drag'n'Drop, managing validators etc.
122 6) Display a border (sunken, raised, simple or none).
124 Normally one might expect, that one wxWidgets window would always correspond
125 to one GTK widget. Under GTK, there is no such all-round widget that has all
126 the functionality. Moreover, the GTK defines a client area as a different
127 widget from the actual widget you are handling. Last but not least some
128 special classes (e.g. wxFrame) handle different categories of widgets and
129 still have the possibility to draw something in the client area.
130 It was therefore required to write a special purpose GTK widget, that would
131 represent a client area in the sense of wxWidgets capable to do the jobs
132 2), 3) and 4). I have written this class and it resides in win_gtk.c of
135 All windows must have a widget, with which they interact with other under-
136 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
137 the wxWindow class has a member variable called m_widget which holds a
138 pointer to this widget. When the window class represents a GTK native widget,
139 this is (in most cases) the only GTK widget the class manages. E.g. the
140 wxStaticText class handles only a GtkLabel widget a pointer to which you
141 can find in m_widget (defined in wxWindow)
143 When the class has a client area for drawing into and for containing children
144 it has to handle the client area widget (of the type GtkPizza, defined in
145 win_gtk.c), but there could be any number of widgets, handled by a class
146 The common rule for all windows is only, that the widget that interacts with
147 the rest of GTK must be referenced in m_widget and all other widgets must be
148 children of this widget on the GTK level. The top-most widget, which also
149 represents the client area, must be in the m_wxwindow field and must be of
152 As I said, the window classes that display a GTK native widget only have
153 one widget, so in the case of e.g. the wxButton class m_widget holds a
154 pointer to a GtkButton widget. But windows with client areas (for drawing
155 and children) have a m_widget field that is a pointer to a GtkScrolled-
156 Window and a m_wxwindow field that is pointer to a GtkPizza and this
157 one is (in the GTK sense) a child of the GtkScrolledWindow.
159 If the m_wxwindow field is set, then all input to this widget is inter-
160 cepted and sent to the wxWidgets class. If not, all input to the widget
161 that gets pointed to by m_widget gets intercepted and sent to the class.
165 The design of scrolling in wxWidgets is markedly different from that offered
166 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
167 clicking on a scrollbar belonging to scrolled window will inevitably move
168 the window. In wxWidgets, the scrollbar will only emit an event, send this
169 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
170 which actually moves the window and its sub-windows. Note that GtkPizza
171 memorizes how much it has been scrolled but that wxWidgets forgets this
172 so that the two coordinates systems have to be kept in synch. This is done
173 in various places using the pizza->xoffset and pizza->yoffset values.
177 Singularly the most broken code in GTK is the code that is supposed to
178 inform subwindows (child windows) about new positions. Very often, duplicate
179 events are sent without changes in size or position, equally often no
180 events are sent at all (All this is due to a bug in the GtkContainer code
181 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
182 GTK's own system and it simply waits for size events for toplevel windows
183 and then iterates down the respective size events to all window. This has
184 the disadvantage that windows might get size events before the GTK widget
185 actually has the reported size. This doesn't normally pose any problem, but
186 the OpenGL drawing routines rely on correct behaviour. Therefore, I have
187 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
188 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
189 window that is used for OpenGL output really has that size (as reported by
194 If someone at some point of time feels the immense desire to have a look at,
195 change or attempt to optimise the Refresh() logic, this person will need an
196 intimate understanding of what "draw" and "expose" events are and what
197 they are used for, in particular when used in connection with GTK's
198 own windowless widgets. Beware.
202 Cursors, too, have been a constant source of pleasure. The main difficulty
203 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
204 for the parent. To prevent this from doing too much harm, I use idle time
205 to set the cursor over and over again, starting from the toplevel windows
206 and ending with the youngest generation (speaking of parent and child windows).
207 Also don't forget that cursors (like much else) are connected to GdkWindows,
208 not GtkWidgets and that the "window" field of a GtkWidget might very well
209 point to the GdkWindow of the parent widget (-> "window-less widget") and
210 that the two obviously have very different meanings.
214 //-----------------------------------------------------------------------------
216 //-----------------------------------------------------------------------------
218 extern bool g_blockEventsOnDrag
;
219 extern bool g_blockEventsOnScroll
;
220 extern wxCursor g_globalCursor
;
222 static GdkGC
*g_eraseGC
= NULL
;
224 // mouse capture state: the window which has it and if the mouse is currently
226 static wxWindowGTK
*g_captureWindow
= (wxWindowGTK
*) NULL
;
227 static bool g_captureWindowHasMouse
= false;
229 wxWindowGTK
*g_focusWindow
= (wxWindowGTK
*) NULL
;
231 // the last window which had the focus - this is normally never NULL (except
232 // if we never had focus at all) as even when g_focusWindow is NULL it still
233 // keeps its previous value
234 wxWindowGTK
*g_focusWindowLast
= (wxWindowGTK
*) NULL
;
236 // If a window get the focus set but has not been realized
237 // yet, defer setting the focus to idle time.
238 wxWindowGTK
*g_delayedFocus
= (wxWindowGTK
*) NULL
;
240 extern bool g_mainThreadLocked
;
242 //-----------------------------------------------------------------------------
244 //-----------------------------------------------------------------------------
249 # define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance");
251 # define DEBUG_MAIN_THREAD
254 #define DEBUG_MAIN_THREAD
257 // the trace mask used for the focus debugging messages
258 #define TRACE_FOCUS _T("focus")
260 //-----------------------------------------------------------------------------
261 // missing gdk functions
262 //-----------------------------------------------------------------------------
265 gdk_window_warp_pointer (GdkWindow
*window
,
270 window
= gdk_get_default_root_window();
272 if (!GDK_WINDOW_DESTROYED(window
))
274 XWarpPointer (GDK_WINDOW_XDISPLAY(window
),
275 None
, /* not source window -> move from anywhere */
276 GDK_WINDOW_XID(window
), /* dest window */
277 0, 0, 0, 0, /* not source window -> move from anywhere */
282 //-----------------------------------------------------------------------------
283 // local code (see below)
284 //-----------------------------------------------------------------------------
286 // returns the child of win which currently has focus or NULL if not found
288 // Note: can't be static, needed by textctrl.cpp.
289 wxWindow
*wxFindFocusedChild(wxWindowGTK
*win
)
291 wxWindow
*winFocus
= wxWindowGTK::FindFocus();
293 return (wxWindow
*)NULL
;
295 if ( winFocus
== win
)
296 return (wxWindow
*)win
;
298 for ( wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
300 node
= node
->GetNext() )
302 wxWindow
*child
= wxFindFocusedChild(node
->GetData());
307 return (wxWindow
*)NULL
;
310 static void GetScrollbarWidth(GtkWidget
* widget
, int& w
, int& h
)
312 GtkScrolledWindow
* scroll_window
= GTK_SCROLLED_WINDOW(widget
);
313 GtkScrolledWindowClass
* scroll_class
= GTK_SCROLLED_WINDOW_CLASS(GTK_OBJECT_GET_CLASS(scroll_window
));
314 GtkRequisition scroll_req
;
317 if (scroll_window
->vscrollbar_visible
)
319 scroll_req
.width
= 2;
320 scroll_req
.height
= 2;
321 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
322 (scroll_window
->vscrollbar
, &scroll_req
);
323 w
= scroll_req
.width
+
324 scroll_class
->scrollbar_spacing
;
328 if (scroll_window
->hscrollbar_visible
)
330 scroll_req
.width
= 2;
331 scroll_req
.height
= 2;
332 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
333 (scroll_window
->hscrollbar
, &scroll_req
);
334 h
= scroll_req
.height
+
335 scroll_class
->scrollbar_spacing
;
339 static void draw_frame( GtkWidget
*widget
, wxWindowGTK
*win
)
341 // wxUniversal widgets draw the borders and scrollbars themselves
342 #ifndef __WXUNIVERSAL__
349 if (win
->m_hasScrolling
)
351 GetScrollbarWidth(widget
, dw
, dh
);
356 if (GTK_WIDGET_NO_WINDOW (widget
))
358 dx
+= widget
->allocation
.x
;
359 dy
+= widget
->allocation
.y
;
362 if (win
->HasFlag(wxRAISED_BORDER
))
364 gtk_paint_shadow (widget
->style
,
368 NULL
, NULL
, NULL
, // FIXME: No clipping?
370 widget
->allocation
.width
-dw
, widget
->allocation
.height
-dh
);
374 if (win
->HasFlag(wxSUNKEN_BORDER
))
376 gtk_paint_shadow (widget
->style
,
380 NULL
, NULL
, NULL
, // FIXME: No clipping?
382 widget
->allocation
.width
-dw
, widget
->allocation
.height
-dh
);
386 if (win
->HasFlag(wxSIMPLE_BORDER
))
389 gc
= gdk_gc_new( widget
->window
);
390 gdk_gc_set_foreground( gc
, &widget
->style
->black
);
391 gdk_draw_rectangle( widget
->window
, gc
, FALSE
,
393 widget
->allocation
.width
-dw
-1, widget
->allocation
.height
-dh
-1 );
397 #endif // __WXUNIVERSAL__
400 //-----------------------------------------------------------------------------
401 // "expose_event" of m_widget
402 //-----------------------------------------------------------------------------
406 gtk_window_own_expose_callback( GtkWidget
*widget
,
407 GdkEventExpose
*gdk_event
,
410 if (gdk_event
->count
> 0) return FALSE
;
412 draw_frame( widget
, win
);
414 (* GTK_WIDGET_CLASS (pizza_parent_class
)->expose_event
) (widget
, gdk_event
);
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
;
495 wxPrintf( wxT("OnExpose from ") );
496 if (win
->GetClassInfo() && win
->GetClassInfo()->GetClassName())
497 wxPrintf( win
->GetClassInfo()->GetClassName() );
498 wxPrintf( wxT(" %d %d %d %d\n"), (int)gdk_event
->area
.x
,
499 (int)gdk_event
->area
.y
,
500 (int)gdk_event
->area
.width
,
501 (int)gdk_event
->area
.height
);
506 win
->m_wxwindow
->style
,
510 (GdkRectangle
*) NULL
,
512 (char *)"button", // const_cast
517 win
->GetUpdateRegion() = wxRegion( gdk_event
->region
);
519 win
->GtkSendPaintEvents();
522 // Let parent window draw window-less widgets
523 (* GTK_WIDGET_CLASS (pizza_parent_class
)->expose_event
) (widget
, gdk_event
);
529 //-----------------------------------------------------------------------------
530 // "key_press_event" from any window
531 //-----------------------------------------------------------------------------
533 // These are used when transforming Ctrl-alpha to ascii values 1-26
534 inline bool wxIsLowerChar(int code
)
536 return (code
>= 'a' && code
<= 'z' );
539 inline bool wxIsUpperChar(int code
)
541 return (code
>= 'A' && code
<= 'Z' );
545 // set WXTRACE to this to see the key event codes on the console
546 #define TRACE_KEYS _T("keyevent")
548 // translates an X key symbol to WXK_XXX value
550 // if isChar is true it means that the value returned will be used for EVT_CHAR
551 // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
552 // for example, while if it is false it means that the value is going to be
553 // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
555 static long wxTranslateKeySymToWXKey(KeySym keysym
, bool isChar
)
561 // Shift, Control and Alt don't generate the CHAR events at all
564 key_code
= isChar
? 0 : WXK_SHIFT
;
568 key_code
= isChar
? 0 : WXK_CONTROL
;
576 key_code
= isChar
? 0 : WXK_ALT
;
579 // neither do the toggle modifies
580 case GDK_Scroll_Lock
:
581 key_code
= isChar
? 0 : WXK_SCROLL
;
585 key_code
= isChar
? 0 : WXK_CAPITAL
;
589 key_code
= isChar
? 0 : WXK_NUMLOCK
;
593 // various other special keys
606 case GDK_ISO_Left_Tab
:
613 key_code
= WXK_RETURN
;
617 key_code
= WXK_CLEAR
;
621 key_code
= WXK_PAUSE
;
625 key_code
= WXK_SELECT
;
629 key_code
= WXK_PRINT
;
633 key_code
= WXK_EXECUTE
;
637 key_code
= WXK_ESCAPE
;
640 // cursor and other extended keyboard keys
642 key_code
= WXK_DELETE
;
658 key_code
= WXK_RIGHT
;
665 case GDK_Prior
: // == GDK_Page_Up
666 key_code
= WXK_PAGEUP
;
669 case GDK_Next
: // == GDK_Page_Down
670 key_code
= WXK_PAGEDOWN
;
682 key_code
= WXK_INSERT
;
697 key_code
= (isChar
? '0' : WXK_NUMPAD0
) + keysym
- GDK_KP_0
;
701 key_code
= isChar
? ' ' : WXK_NUMPAD_SPACE
;
705 key_code
= isChar
? WXK_TAB
: WXK_NUMPAD_TAB
;
709 key_code
= isChar
? WXK_RETURN
: WXK_NUMPAD_ENTER
;
713 key_code
= isChar
? WXK_F1
: WXK_NUMPAD_F1
;
717 key_code
= isChar
? WXK_F2
: WXK_NUMPAD_F2
;
721 key_code
= isChar
? WXK_F3
: WXK_NUMPAD_F3
;
725 key_code
= isChar
? WXK_F4
: WXK_NUMPAD_F4
;
729 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_HOME
;
733 key_code
= isChar
? WXK_LEFT
: WXK_NUMPAD_LEFT
;
737 key_code
= isChar
? WXK_UP
: WXK_NUMPAD_UP
;
741 key_code
= isChar
? WXK_RIGHT
: WXK_NUMPAD_RIGHT
;
745 key_code
= isChar
? WXK_DOWN
: WXK_NUMPAD_DOWN
;
748 case GDK_KP_Prior
: // == GDK_KP_Page_Up
749 key_code
= isChar
? WXK_PAGEUP
: WXK_NUMPAD_PAGEUP
;
752 case GDK_KP_Next
: // == GDK_KP_Page_Down
753 key_code
= isChar
? WXK_PAGEDOWN
: WXK_NUMPAD_PAGEDOWN
;
757 key_code
= isChar
? WXK_END
: WXK_NUMPAD_END
;
761 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_BEGIN
;
765 key_code
= isChar
? WXK_INSERT
: WXK_NUMPAD_INSERT
;
769 key_code
= isChar
? WXK_DELETE
: WXK_NUMPAD_DELETE
;
773 key_code
= isChar
? '=' : WXK_NUMPAD_EQUAL
;
776 case GDK_KP_Multiply
:
777 key_code
= isChar
? '*' : WXK_NUMPAD_MULTIPLY
;
781 key_code
= isChar
? '+' : WXK_NUMPAD_ADD
;
784 case GDK_KP_Separator
:
785 // FIXME: what is this?
786 key_code
= isChar
? '.' : WXK_NUMPAD_SEPARATOR
;
789 case GDK_KP_Subtract
:
790 key_code
= isChar
? '-' : WXK_NUMPAD_SUBTRACT
;
794 key_code
= isChar
? '.' : WXK_NUMPAD_DECIMAL
;
798 key_code
= isChar
? '/' : WXK_NUMPAD_DIVIDE
;
815 key_code
= WXK_F1
+ keysym
- GDK_F1
;
825 static inline bool wxIsAsciiKeysym(KeySym ks
)
830 static void wxFillOtherKeyEventFields(wxKeyEvent
& event
,
832 GdkEventKey
*gdk_event
)
836 GdkModifierType state
;
837 if (gdk_event
->window
)
838 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
840 event
.SetTimestamp( gdk_event
->time
);
841 event
.SetId(win
->GetId());
842 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
) != 0;
843 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
) != 0;
844 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
) != 0;
845 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
) != 0;
846 event
.m_scanCode
= gdk_event
->keyval
;
847 event
.m_rawCode
= (wxUint32
) gdk_event
->keyval
;
848 event
.m_rawFlags
= 0;
850 event
.m_uniChar
= gdk_keyval_to_unicode(gdk_event
->keyval
);
852 wxGetMousePosition( &x
, &y
);
853 win
->ScreenToClient( &x
, &y
);
856 event
.SetEventObject( win
);
861 wxTranslateGTKKeyEventToWx(wxKeyEvent
& event
,
863 GdkEventKey
*gdk_event
)
865 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
866 // but only event->keyval which is quite useless to us, so remember
867 // the last character from GDK_KEY_PRESS and reuse it as last resort
869 // NB: should be MT-safe as we're always called from the main thread only
874 } s_lastKeyPress
= { 0, 0 };
876 KeySym keysym
= gdk_event
->keyval
;
878 wxLogTrace(TRACE_KEYS
, _T("Key %s event: keysym = %ld"),
879 event
.GetEventType() == wxEVT_KEY_UP
? _T("release")
883 long key_code
= wxTranslateKeySymToWXKey(keysym
, false /* !isChar */);
887 // do we have the translation or is it a plain ASCII character?
888 if ( (gdk_event
->length
== 1) || wxIsAsciiKeysym(keysym
) )
890 // we should use keysym if it is ASCII as X does some translations
891 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
892 // which we don't want here (but which we do use for OnChar())
893 if ( !wxIsAsciiKeysym(keysym
) )
895 keysym
= (KeySym
)gdk_event
->string
[0];
898 // we want to always get the same key code when the same key is
899 // pressed regardless of the state of the modifiers, i.e. on a
900 // standard US keyboard pressing '5' or '%' ('5' key with
901 // Shift) should result in the same key code in OnKeyDown():
902 // '5' (although OnChar() will get either '5' or '%').
904 // to do it we first translate keysym to keycode (== scan code)
905 // and then back but always using the lower register
906 Display
*dpy
= (Display
*)wxGetDisplay();
907 KeyCode keycode
= XKeysymToKeycode(dpy
, keysym
);
909 wxLogTrace(TRACE_KEYS
, _T("\t-> keycode %d"), keycode
);
911 KeySym keysymNormalized
= XKeycodeToKeysym(dpy
, keycode
, 0);
913 // use the normalized, i.e. lower register, keysym if we've
915 key_code
= keysymNormalized
? keysymNormalized
: keysym
;
917 // as explained above, we want to have lower register key codes
918 // normally but for the letter keys we want to have the upper ones
920 // NB: don't use XConvertCase() here, we want to do it for letters
922 key_code
= toupper(key_code
);
924 else // non ASCII key, what to do?
926 // by default, ignore it
929 // but if we have cached information from the last KEY_PRESS
930 if ( gdk_event
->type
== GDK_KEY_RELEASE
)
933 if ( keysym
== s_lastKeyPress
.keysym
)
935 key_code
= s_lastKeyPress
.keycode
;
940 if ( gdk_event
->type
== GDK_KEY_PRESS
)
942 // remember it to be reused for KEY_UP event later
943 s_lastKeyPress
.keysym
= keysym
;
944 s_lastKeyPress
.keycode
= key_code
;
948 wxLogTrace(TRACE_KEYS
, _T("\t-> wxKeyCode %ld"), key_code
);
950 // sending unknown key events doesn't really make sense
954 // now fill all the other fields
955 wxFillOtherKeyEventFields(event
, win
, gdk_event
);
957 event
.m_keyCode
= key_code
;
959 if ( gdk_event
->type
== GDK_KEY_PRESS
|| gdk_event
->type
== GDK_KEY_RELEASE
)
961 event
.m_uniChar
= key_code
;
971 GtkIMContext
*context
;
972 GdkEventKey
*lastKeyEvent
;
976 context
= gtk_im_multicontext_new();
981 g_object_unref (context
);
987 gtk_window_key_press_callback( GtkWidget
*widget
,
988 GdkEventKey
*gdk_event
,
993 // don't need to install idle handler, its done from "event" signal
997 if (g_blockEventsOnDrag
)
1001 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1003 bool return_after_IM
= false;
1005 if( wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1007 // Emit KEY_DOWN event
1008 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1012 // Return after IM processing as we cannot do
1013 // anything with it anyhow.
1014 return_after_IM
= true;
1017 // 2005.01.26 modified by Hong Jen Yee (hzysoft@sina.com.tw):
1018 // When we get a key_press event here, it could be originate
1019 // from the current widget or its child widgets. However, only the widget
1020 // with the INPUT FOCUS can generate the INITIAL key_press event. That is,
1021 // if the CURRENT widget doesn't have the FOCUS at all, this event definitely
1022 // originated from its child widgets and shouldn't be passed to IM context.
1023 // In fact, what a GTK+ IM should do is filtering keyEvents and convert them
1024 // into text input ONLY WHEN THE WIDGET HAS INPUT FOCUS. Besides, when current
1025 // widgets has both IM context and input focus, the event should be filtered
1026 // by gtk_im_context_filter_keypress().
1027 // Then, we should, according to GTK+ 2.0 API doc, return whatever it returns.
1028 if ((!ret
) && (win
->m_imData
!= NULL
) && ( wxWindow::FindFocus() == win
))
1030 // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API
1031 // docs, if IM filter returns true, no further processing should be done.
1032 // we should send the key_down event anyway.
1033 bool intercepted_by_IM
= gtk_im_context_filter_keypress(win
->m_imData
->context
, gdk_event
);
1034 win
->m_imData
->lastKeyEvent
= NULL
;
1035 if (intercepted_by_IM
)
1037 wxLogTrace(TRACE_KEYS
, _T("Key event intercepted by IM"));
1042 if (return_after_IM
)
1048 wxWindowGTK
*ancestor
= win
;
1051 int command
= ancestor
->GetAcceleratorTable()->GetCommand( event
);
1054 wxCommandEvent
command_event( wxEVT_COMMAND_MENU_SELECTED
, command
);
1055 ret
= ancestor
->GetEventHandler()->ProcessEvent( command_event
);
1058 if (ancestor
->IsTopLevel())
1060 ancestor
= ancestor
->GetParent();
1063 #endif // wxUSE_ACCEL
1065 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
1066 // will only be sent if it is not in an accelerator table.
1070 KeySym keysym
= gdk_event
->keyval
;
1071 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
1072 key_code
= wxTranslateKeySymToWXKey(keysym
, true /* isChar */);
1075 if ( wxIsAsciiKeysym(keysym
) )
1078 key_code
= (unsigned char)keysym
;
1080 // gdk_event->string is actually deprecated
1081 else if ( gdk_event
->length
== 1 )
1083 key_code
= (unsigned char)gdk_event
->string
[0];
1089 wxLogTrace(TRACE_KEYS
, _T("Char event: %ld"), key_code
);
1091 event
.m_keyCode
= key_code
;
1093 // To conform to the docs we need to translate Ctrl-alpha
1094 // characters to values in the range 1-26.
1095 if ( event
.ControlDown() &&
1096 ( wxIsLowerChar(key_code
) || wxIsUpperChar(key_code
) ))
1098 if ( wxIsLowerChar(key_code
) )
1099 event
.m_keyCode
= key_code
- 'a' + 1;
1100 if ( wxIsUpperChar(key_code
) )
1101 event
.m_keyCode
= key_code
- 'A' + 1;
1103 event
.m_uniChar
= event
.m_keyCode
;
1107 // Implement OnCharHook by checking ancestor top level windows
1108 wxWindow
*parent
= win
;
1109 while (parent
&& !parent
->IsTopLevel())
1110 parent
= parent
->GetParent();
1113 event
.SetEventType( wxEVT_CHAR_HOOK
);
1114 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1119 event
.SetEventType(wxEVT_CHAR
);
1120 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1125 // win is a control: tab can be propagated up
1127 ((gdk_event
->keyval
== GDK_Tab
) || (gdk_event
->keyval
== GDK_ISO_Left_Tab
)) &&
1128 // VZ: testing for wxTE_PROCESS_TAB shouldn't be done here - the control may
1129 // have this style, yet choose not to process this particular TAB in which
1130 // case TAB must still work as a navigational character
1131 // JS: enabling again to make consistent with other platforms
1132 // (with wxTE_PROCESS_TAB you have to call Navigate to get default
1133 // navigation behaviour)
1135 (! (win
->HasFlag(wxTE_PROCESS_TAB
) && win
->IsKindOf(CLASSINFO(wxTextCtrl
)) )) &&
1137 win
->GetParent() && (win
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) )
1139 wxNavigationKeyEvent new_event
;
1140 new_event
.SetEventObject( win
->GetParent() );
1141 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1142 new_event
.SetDirection( (gdk_event
->keyval
== GDK_Tab
) );
1143 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1144 new_event
.SetWindowChange( (gdk_event
->state
& GDK_CONTROL_MASK
) );
1145 new_event
.SetCurrentFocus( win
);
1146 ret
= win
->GetParent()->GetEventHandler()->ProcessEvent( new_event
);
1155 gtk_wxwindow_commit_cb (GtkIMContext
*context
,
1159 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1161 // take modifiers, cursor position, timestamp etc. from the last
1162 // key_press_event that was fed into Input Method:
1163 if (window
->m_imData
->lastKeyEvent
)
1165 wxFillOtherKeyEventFields(event
,
1166 window
, window
->m_imData
->lastKeyEvent
);
1169 const wxWxCharBuffer
data(wxGTK_CONV_BACK(str
));
1175 // Implement OnCharHook by checking ancestor top level windows
1176 wxWindow
*parent
= window
;
1177 while (parent
&& !parent
->IsTopLevel())
1178 parent
= parent
->GetParent();
1180 for( const wxChar
* pstr
= data
; *pstr
; pstr
++ )
1183 event
.m_uniChar
= *pstr
;
1184 // Backward compatible for ISO-8859-1
1185 event
.m_keyCode
= *pstr
< 256 ? event
.m_uniChar
: 0;
1186 wxLogTrace(TRACE_KEYS
, _T("IM sent character '%c'"), event
.m_uniChar
);
1188 event
.m_keyCode
= *pstr
;
1189 #endif // wxUSE_UNICODE
1191 // To conform to the docs we need to translate Ctrl-alpha
1192 // characters to values in the range 1-26.
1193 if ( event
.ControlDown() &&
1194 ( wxIsLowerChar(*pstr
) || wxIsUpperChar(*pstr
) ))
1196 if ( wxIsLowerChar(*pstr
) )
1197 event
.m_keyCode
= *pstr
- 'a' + 1;
1198 if ( wxIsUpperChar(*pstr
) )
1199 event
.m_keyCode
= *pstr
- 'A' + 1;
1201 event
.m_keyCode
= *pstr
- 'a' + 1;
1203 event
.m_uniChar
= event
.m_keyCode
;
1209 event
.SetEventType( wxEVT_CHAR_HOOK
);
1210 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1215 event
.SetEventType(wxEVT_CHAR
);
1216 ret
= window
->GetEventHandler()->ProcessEvent( event
);
1223 //-----------------------------------------------------------------------------
1224 // "key_release_event" from any window
1225 //-----------------------------------------------------------------------------
1229 gtk_window_key_release_callback( GtkWidget
*widget
,
1230 GdkEventKey
*gdk_event
,
1235 // don't need to install idle handler, its done from "event" signal
1240 if (g_blockEventsOnDrag
)
1243 wxKeyEvent
event( wxEVT_KEY_UP
);
1244 if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1246 // unknown key pressed, ignore (the event would be useless anyhow)
1250 return win
->GetEventHandler()->ProcessEvent(event
);
1254 // ============================================================================
1256 // ============================================================================
1258 // ----------------------------------------------------------------------------
1259 // mouse event processing helpers
1260 // ----------------------------------------------------------------------------
1262 // init wxMouseEvent with the info from GdkEventXXX struct
1263 template<typename T
> void InitMouseEvent(wxWindowGTK
*win
,
1264 wxMouseEvent
& event
,
1267 event
.SetTimestamp( gdk_event
->time
);
1268 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
);
1269 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
);
1270 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
);
1271 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
);
1272 event
.m_leftDown
= (gdk_event
->state
& GDK_BUTTON1_MASK
);
1273 event
.m_middleDown
= (gdk_event
->state
& GDK_BUTTON2_MASK
);
1274 event
.m_rightDown
= (gdk_event
->state
& GDK_BUTTON3_MASK
);
1275 if (event
.GetEventType() == wxEVT_MOUSEWHEEL
)
1277 event
.m_linesPerAction
= 3;
1278 event
.m_wheelDelta
= 120;
1279 if (((GdkEventButton
*)gdk_event
)->button
== 4)
1280 event
.m_wheelRotation
= 120;
1281 else if (((GdkEventButton
*)gdk_event
)->button
== 5)
1282 event
.m_wheelRotation
= -120;
1285 wxPoint pt
= win
->GetClientAreaOrigin();
1286 event
.m_x
= (wxCoord
)gdk_event
->x
- pt
.x
;
1287 event
.m_y
= (wxCoord
)gdk_event
->y
- pt
.y
;
1289 event
.SetEventObject( win
);
1290 event
.SetId( win
->GetId() );
1291 event
.SetTimestamp( gdk_event
->time
);
1294 static void AdjustEventButtonState(wxMouseEvent
& event
)
1296 // GDK reports the old state of the button for a button press event, but
1297 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1298 // for a LEFT_DOWN event, not FALSE, so we will invert
1299 // left/right/middleDown for the corresponding click events
1301 if ((event
.GetEventType() == wxEVT_LEFT_DOWN
) ||
1302 (event
.GetEventType() == wxEVT_LEFT_DCLICK
) ||
1303 (event
.GetEventType() == wxEVT_LEFT_UP
))
1305 event
.m_leftDown
= !event
.m_leftDown
;
1309 if ((event
.GetEventType() == wxEVT_MIDDLE_DOWN
) ||
1310 (event
.GetEventType() == wxEVT_MIDDLE_DCLICK
) ||
1311 (event
.GetEventType() == wxEVT_MIDDLE_UP
))
1313 event
.m_middleDown
= !event
.m_middleDown
;
1317 if ((event
.GetEventType() == wxEVT_RIGHT_DOWN
) ||
1318 (event
.GetEventType() == wxEVT_RIGHT_DCLICK
) ||
1319 (event
.GetEventType() == wxEVT_RIGHT_UP
))
1321 event
.m_rightDown
= !event
.m_rightDown
;
1326 // find the window to send the mouse event too
1328 wxWindowGTK
*FindWindowForMouseEvent(wxWindowGTK
*win
, wxCoord
& x
, wxCoord
& y
)
1333 if (win
->m_wxwindow
)
1335 GtkPizza
*pizza
= GTK_PIZZA(win
->m_wxwindow
);
1336 xx
+= pizza
->xoffset
;
1337 yy
+= pizza
->yoffset
;
1340 wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
1343 wxWindowGTK
*child
= node
->GetData();
1345 node
= node
->GetNext();
1346 if (!child
->IsShown())
1349 if (child
->IsTransparentForMouse())
1351 // wxStaticBox is transparent in the box itself
1352 int xx1
= child
->m_x
;
1353 int yy1
= child
->m_y
;
1354 int xx2
= child
->m_x
+ child
->m_width
;
1355 int yy2
= child
->m_y
+ child
->m_height
;
1358 if (((xx
>= xx1
) && (xx
<= xx1
+10) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1360 ((xx
>= xx2
-10) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1362 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy1
+10)) ||
1364 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy2
-1) && (yy
<= yy2
)))
1375 if ((child
->m_wxwindow
== (GtkWidget
*) NULL
) &&
1376 (child
->m_x
<= xx
) &&
1377 (child
->m_y
<= yy
) &&
1378 (child
->m_x
+child
->m_width
>= xx
) &&
1379 (child
->m_y
+child
->m_height
>= yy
))
1392 // ----------------------------------------------------------------------------
1393 // common event handlers helpers
1394 // ----------------------------------------------------------------------------
1396 int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny
*event
) const
1400 // don't need to install idle handler, its done from "event" signal
1404 if (g_blockEventsOnDrag
)
1406 if (g_blockEventsOnScroll
)
1409 if (!GTKIsOwnWindow(event
->window
))
1415 // overloads for all GDK event types we use here: we need to have this as
1416 // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact,
1417 // derives from it in the sense that the structs have the same layout
1418 #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \
1419 static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \
1421 return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \
1424 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton
)
1425 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion
)
1426 wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing
)
1428 #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD
1430 #define wxCOMMON_CALLBACK_PROLOGUE(event, win) \
1431 const int rc = wxGtkCallbackCommonPrologue(event, win); \
1435 // send the wxChildFocusEvent and wxFocusEvent, common code of
1436 // gtk_window_focus_in_callback() and SetFocus()
1437 static bool DoSendFocusEvents(wxWindow
*win
)
1439 // Notify the parent keeping track of focus for the kbd navigation
1440 // purposes that we got it.
1441 wxChildFocusEvent
eventChildFocus(win
);
1442 (void)win
->GetEventHandler()->ProcessEvent(eventChildFocus
);
1444 wxFocusEvent
eventFocus(wxEVT_SET_FOCUS
, win
->GetId());
1445 eventFocus
.SetEventObject(win
);
1447 return win
->GetEventHandler()->ProcessEvent(eventFocus
);
1450 // all event handlers must have C linkage as they're called from GTK+ C code
1454 //-----------------------------------------------------------------------------
1455 // "button_press_event"
1456 //-----------------------------------------------------------------------------
1459 gtk_window_button_press_callback( GtkWidget
*widget
,
1460 GdkEventButton
*gdk_event
,
1463 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1465 if (win
->m_wxwindow
&& (g_focusWindow
!= win
) && win
->AcceptsFocus())
1467 gtk_widget_grab_focus( win
->m_wxwindow
);
1470 // GDK sends surplus button down events
1471 // before a double click event. We
1472 // need to filter these out.
1473 if (gdk_event
->type
== GDK_BUTTON_PRESS
)
1475 GdkEvent
*peek_event
= gdk_event_peek();
1478 if ((peek_event
->type
== GDK_2BUTTON_PRESS
) ||
1479 (peek_event
->type
== GDK_3BUTTON_PRESS
))
1481 gdk_event_free( peek_event
);
1486 gdk_event_free( peek_event
);
1491 wxEventType event_type
= wxEVT_NULL
;
1493 // GdkDisplay is a GTK+ 2.2.0 thing
1494 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2, 2, 0)
1495 if ( gdk_event
->type
== GDK_2BUTTON_PRESS
&&
1496 !gtk_check_version(2,2,0) &&
1497 gdk_event
->button
>= 1 && gdk_event
->button
<= 3 )
1499 // Reset GDK internal timestamp variables in order to disable GDK
1500 // triple click events. GDK will then next time believe no button has
1501 // been clicked just before, and send a normal button click event.
1502 GdkDisplay
* display
= gtk_widget_get_display (widget
);
1503 display
->button_click_time
[1] = 0;
1504 display
->button_click_time
[0] = 0;
1508 if (gdk_event
->button
== 1)
1510 // note that GDK generates triple click events which are not supported
1511 // by wxWidgets but still have to be passed to the app as otherwise
1512 // clicks would simply go missing
1513 switch (gdk_event
->type
)
1515 // we shouldn't get triple clicks at all for GTK2 because we
1516 // suppress them artificially using the code above but we still
1517 // should map them to something for GTK1 and not just ignore them
1518 // as this would lose clicks
1519 case GDK_3BUTTON_PRESS
: // we could also map this to DCLICK...
1520 case GDK_BUTTON_PRESS
:
1521 event_type
= wxEVT_LEFT_DOWN
;
1524 case GDK_2BUTTON_PRESS
:
1525 event_type
= wxEVT_LEFT_DCLICK
;
1529 // just to silence gcc warnings
1533 else if (gdk_event
->button
== 2)
1535 switch (gdk_event
->type
)
1537 case GDK_3BUTTON_PRESS
:
1538 case GDK_BUTTON_PRESS
:
1539 event_type
= wxEVT_MIDDLE_DOWN
;
1542 case GDK_2BUTTON_PRESS
:
1543 event_type
= wxEVT_MIDDLE_DCLICK
;
1550 else if (gdk_event
->button
== 3)
1552 switch (gdk_event
->type
)
1554 case GDK_3BUTTON_PRESS
:
1555 case GDK_BUTTON_PRESS
:
1556 event_type
= wxEVT_RIGHT_DOWN
;
1559 case GDK_2BUTTON_PRESS
:
1560 event_type
= wxEVT_RIGHT_DCLICK
;
1567 else if (gdk_event
->button
== 4 || gdk_event
->button
== 5)
1569 if (gdk_event
->type
== GDK_BUTTON_PRESS
)
1571 event_type
= wxEVT_MOUSEWHEEL
;
1575 if ( event_type
== wxEVT_NULL
)
1577 // unknown mouse button or click type
1581 wxMouseEvent
event( event_type
);
1582 InitMouseEvent( win
, event
, gdk_event
);
1584 AdjustEventButtonState(event
);
1586 // wxListBox actually gets mouse events from the item, so we need to give it
1587 // a chance to correct this
1588 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1590 // find the correct window to send the event to: it may be a different one
1591 // from the one which got it at GTK+ level because some controls don't have
1592 // their own X window and thus cannot get any events.
1593 if ( !g_captureWindow
)
1594 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1596 // reset the event object and id in case win changed.
1597 event
.SetEventObject( win
);
1598 event
.SetId( win
->GetId() );
1600 if (win
->GetEventHandler()->ProcessEvent( event
))
1605 if (event_type
== wxEVT_RIGHT_DOWN
)
1607 // generate a "context menu" event: this is similar to right mouse
1608 // click under many GUIs except that it is generated differently
1609 // (right up under MSW, ctrl-click under Mac, right down here) and
1611 // (a) it's a command event and so is propagated to the parent
1612 // (b) under some ports it can be generated from kbd too
1613 // (c) it uses screen coords (because of (a))
1614 wxContextMenuEvent
evtCtx(
1617 win
->ClientToScreen(event
.GetPosition()));
1618 evtCtx
.SetEventObject(win
);
1619 return win
->GetEventHandler()->ProcessEvent(evtCtx
);
1625 //-----------------------------------------------------------------------------
1626 // "button_release_event"
1627 //-----------------------------------------------------------------------------
1630 gtk_window_button_release_callback( GtkWidget
*widget
,
1631 GdkEventButton
*gdk_event
,
1634 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1636 wxEventType event_type
= wxEVT_NULL
;
1638 switch (gdk_event
->button
)
1641 event_type
= wxEVT_LEFT_UP
;
1645 event_type
= wxEVT_MIDDLE_UP
;
1649 event_type
= wxEVT_RIGHT_UP
;
1653 // unknown button, don't process
1657 wxMouseEvent
event( event_type
);
1658 InitMouseEvent( win
, event
, gdk_event
);
1660 AdjustEventButtonState(event
);
1662 // same wxListBox hack as above
1663 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1665 if ( !g_captureWindow
)
1666 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1668 // reset the event object and id in case win changed.
1669 event
.SetEventObject( win
);
1670 event
.SetId( win
->GetId() );
1672 return win
->GetEventHandler()->ProcessEvent(event
);
1675 //-----------------------------------------------------------------------------
1676 // "motion_notify_event"
1677 //-----------------------------------------------------------------------------
1680 gtk_window_motion_notify_callback( GtkWidget
*widget
,
1681 GdkEventMotion
*gdk_event
,
1684 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1686 if (gdk_event
->is_hint
)
1690 GdkModifierType state
;
1691 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1696 wxMouseEvent
event( wxEVT_MOTION
);
1697 InitMouseEvent(win
, event
, gdk_event
);
1699 if ( g_captureWindow
)
1701 // synthesise a mouse enter or leave event if needed
1702 GdkWindow
*winUnderMouse
= gdk_window_at_pointer(NULL
, NULL
);
1703 // This seems to be necessary and actually been added to
1704 // GDK itself in version 2.0.X
1707 bool hasMouse
= winUnderMouse
== gdk_event
->window
;
1708 if ( hasMouse
!= g_captureWindowHasMouse
)
1710 // the mouse changed window
1711 g_captureWindowHasMouse
= hasMouse
;
1713 wxMouseEvent
eventM(g_captureWindowHasMouse
? wxEVT_ENTER_WINDOW
1714 : wxEVT_LEAVE_WINDOW
);
1715 InitMouseEvent(win
, eventM
, gdk_event
);
1716 eventM
.SetEventObject(win
);
1717 win
->GetEventHandler()->ProcessEvent(eventM
);
1722 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1724 // reset the event object and id in case win changed.
1725 event
.SetEventObject( win
);
1726 event
.SetId( win
->GetId() );
1729 if ( !g_captureWindow
)
1731 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1732 if (win
->GetEventHandler()->ProcessEvent( cevent
))
1734 // Rewrite cursor handling here (away from idle).
1738 return win
->GetEventHandler()->ProcessEvent(event
);
1741 //-----------------------------------------------------------------------------
1742 // "scroll_event", (mouse wheel event)
1743 //-----------------------------------------------------------------------------
1746 window_scroll_event(GtkWidget
*, GdkEventScroll
* gdk_event
, wxWindow
* win
)
1750 // don't need to install idle handler, its done from "event" signal
1752 if (gdk_event
->direction
!= GDK_SCROLL_UP
&&
1753 gdk_event
->direction
!= GDK_SCROLL_DOWN
)
1758 wxMouseEvent
event(wxEVT_MOUSEWHEEL
);
1759 // Can't use InitMouse macro because scroll events don't have button
1760 event
.SetTimestamp( gdk_event
->time
);
1761 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
);
1762 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
);
1763 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
);
1764 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
);
1765 event
.m_leftDown
= (gdk_event
->state
& GDK_BUTTON1_MASK
);
1766 event
.m_middleDown
= (gdk_event
->state
& GDK_BUTTON2_MASK
);
1767 event
.m_rightDown
= (gdk_event
->state
& GDK_BUTTON3_MASK
);
1768 event
.m_linesPerAction
= 3;
1769 event
.m_wheelDelta
= 120;
1770 if (gdk_event
->direction
== GDK_SCROLL_UP
)
1771 event
.m_wheelRotation
= 120;
1773 event
.m_wheelRotation
= -120;
1775 wxPoint pt
= win
->GetClientAreaOrigin();
1776 event
.m_x
= (wxCoord
)gdk_event
->x
- pt
.x
;
1777 event
.m_y
= (wxCoord
)gdk_event
->y
- pt
.y
;
1779 event
.SetEventObject( win
);
1780 event
.SetId( win
->GetId() );
1781 event
.SetTimestamp( gdk_event
->time
);
1783 return win
->GetEventHandler()->ProcessEvent(event
);
1786 //-----------------------------------------------------------------------------
1788 //-----------------------------------------------------------------------------
1790 static gboolean
wxgtk_window_popup_menu_callback(GtkWidget
*, wxWindowGTK
* win
)
1792 wxContextMenuEvent
event(wxEVT_CONTEXT_MENU
, win
->GetId(), wxPoint(-1, -1));
1793 event
.SetEventObject(win
);
1794 return win
->GetEventHandler()->ProcessEvent(event
);
1797 //-----------------------------------------------------------------------------
1799 //-----------------------------------------------------------------------------
1802 gtk_window_focus_in_callback( GtkWidget
*widget
,
1803 GdkEventFocus
*WXUNUSED(event
),
1808 // don't need to install idle handler, its done from "event" signal
1811 gtk_im_context_focus_in(win
->m_imData
->context
);
1814 g_focusWindow
= win
;
1816 wxLogTrace(TRACE_FOCUS
,
1817 _T("%s: focus in"), win
->GetName().c_str());
1821 gdk_im_begin(win
->m_ic
, win
->m_wxwindow
->window
);
1825 // caret needs to be informed about focus change
1826 wxCaret
*caret
= win
->GetCaret();
1829 caret
->OnSetFocus();
1831 #endif // wxUSE_CARET
1833 gboolean ret
= FALSE
;
1835 // does the window itself think that it has the focus?
1836 if ( !win
->m_hasFocus
)
1838 // not yet, notify it
1839 win
->m_hasFocus
= true;
1841 (void)DoSendFocusEvents(win
);
1846 // Disable default focus handling for custom windows
1847 // since the default GTK+ handler issues a repaint
1848 if (win
->m_wxwindow
)
1854 //-----------------------------------------------------------------------------
1855 // "focus_out_event"
1856 //-----------------------------------------------------------------------------
1859 gtk_window_focus_out_callback( GtkWidget
*widget
,
1860 GdkEventFocus
*gdk_event
,
1865 // don't need to install idle handler, its done from "event" signal
1868 gtk_im_context_focus_out(win
->m_imData
->context
);
1870 wxLogTrace( TRACE_FOCUS
,
1871 _T("%s: focus out"), win
->GetName().c_str() );
1874 wxWindowGTK
*winFocus
= wxFindFocusedChild(win
);
1878 g_focusWindow
= (wxWindowGTK
*)NULL
;
1886 // caret needs to be informed about focus change
1887 wxCaret
*caret
= win
->GetCaret();
1890 caret
->OnKillFocus();
1892 #endif // wxUSE_CARET
1894 gboolean ret
= FALSE
;
1896 // don't send the window a kill focus event if it thinks that it doesn't
1897 // have focus already
1898 if ( win
->m_hasFocus
)
1900 win
->m_hasFocus
= false;
1902 wxFocusEvent
event( wxEVT_KILL_FOCUS
, win
->GetId() );
1903 event
.SetEventObject( win
);
1905 (void)win
->GetEventHandler()->ProcessEvent( event
);
1910 // Disable default focus handling for custom windows
1911 // since the default GTK+ handler issues a repaint
1912 if (win
->m_wxwindow
)
1918 //-----------------------------------------------------------------------------
1919 // "enter_notify_event"
1920 //-----------------------------------------------------------------------------
1923 gtk_window_enter_callback( GtkWidget
*widget
,
1924 GdkEventCrossing
*gdk_event
,
1927 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1929 // Event was emitted after a grab
1930 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1934 GdkModifierType state
= (GdkModifierType
)0;
1936 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1938 wxMouseEvent
event( wxEVT_ENTER_WINDOW
);
1939 InitMouseEvent(win
, event
, gdk_event
);
1940 wxPoint pt
= win
->GetClientAreaOrigin();
1941 event
.m_x
= x
+ pt
.x
;
1942 event
.m_y
= y
+ pt
.y
;
1944 if ( !g_captureWindow
)
1946 wxSetCursorEvent
cevent( event
.m_x
, event
.m_y
);
1947 if (win
->GetEventHandler()->ProcessEvent( cevent
))
1949 // Rewrite cursor handling here (away from idle).
1953 return win
->GetEventHandler()->ProcessEvent(event
);
1956 //-----------------------------------------------------------------------------
1957 // "leave_notify_event"
1958 //-----------------------------------------------------------------------------
1961 gtk_window_leave_callback( GtkWidget
*widget
,
1962 GdkEventCrossing
*gdk_event
,
1965 wxCOMMON_CALLBACK_PROLOGUE(gdk_event
, win
);
1967 // Event was emitted after an ungrab
1968 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
1970 wxMouseEvent
event( wxEVT_LEAVE_WINDOW
);
1971 event
.SetTimestamp( gdk_event
->time
);
1972 event
.SetEventObject( win
);
1976 GdkModifierType state
= (GdkModifierType
)0;
1978 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
1980 event
.m_shiftDown
= (state
& GDK_SHIFT_MASK
) != 0;
1981 event
.m_controlDown
= (state
& GDK_CONTROL_MASK
) != 0;
1982 event
.m_altDown
= (state
& GDK_MOD1_MASK
) != 0;
1983 event
.m_metaDown
= (state
& GDK_MOD2_MASK
) != 0;
1984 event
.m_leftDown
= (state
& GDK_BUTTON1_MASK
) != 0;
1985 event
.m_middleDown
= (state
& GDK_BUTTON2_MASK
) != 0;
1986 event
.m_rightDown
= (state
& GDK_BUTTON3_MASK
) != 0;
1988 wxPoint pt
= win
->GetClientAreaOrigin();
1989 event
.m_x
= x
+ pt
.x
;
1990 event
.m_y
= y
+ pt
.y
;
1992 return win
->GetEventHandler()->ProcessEvent(event
);
1995 //-----------------------------------------------------------------------------
1996 // "value_changed" from scrollbar
1997 //-----------------------------------------------------------------------------
2000 gtk_scrollbar_value_changed(GtkRange
* range
, wxWindow
* win
)
2002 wxEventType eventType
= win
->GetScrollEventType(range
);
2003 if (eventType
!= wxEVT_NULL
)
2005 // Convert scroll event type to scrollwin event type
2006 eventType
+= wxEVT_SCROLLWIN_TOP
- wxEVT_SCROLL_TOP
;
2008 // find the scrollbar which generated the event
2009 wxWindowGTK::ScrollDir dir
= win
->ScrollDirFromRange(range
);
2011 // generate the corresponding wx event
2012 const int orient
= win
->OrientFromScrollDir(dir
);
2013 wxScrollWinEvent
event(eventType
, win
->GetScrollPos(orient
), orient
);
2014 event
.SetEventObject(win
);
2016 win
->m_blockValueChanged
[dir
] = true;
2017 win
->GetEventHandler()->ProcessEvent(event
);
2018 win
->m_blockValueChanged
[dir
] = false;
2022 //-----------------------------------------------------------------------------
2023 // "button_press_event" from scrollbar
2024 //-----------------------------------------------------------------------------
2027 gtk_scrollbar_button_press_event(GtkRange
*, GdkEventButton
*, wxWindow
* win
)
2031 // don't need to install idle handler, its done from "event" signal
2033 g_blockEventsOnScroll
= true;
2034 win
->m_mouseButtonDown
= true;
2039 //-----------------------------------------------------------------------------
2040 // "event_after" from scrollbar
2041 //-----------------------------------------------------------------------------
2044 gtk_scrollbar_event_after(GtkRange
* range
, GdkEvent
* event
, wxWindow
* win
)
2046 if (event
->type
== GDK_BUTTON_RELEASE
)
2048 g_signal_handlers_block_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
2050 const int orient
= win
->OrientFromScrollDir(
2051 win
->ScrollDirFromRange(range
));
2052 wxScrollWinEvent
event(wxEVT_SCROLLWIN_THUMBRELEASE
, win
->GetScrollPos(orient
), orient
);
2053 event
.SetEventObject(win
);
2054 win
->GetEventHandler()->ProcessEvent(event
);
2058 //-----------------------------------------------------------------------------
2059 // "button_release_event" from scrollbar
2060 //-----------------------------------------------------------------------------
2063 gtk_scrollbar_button_release_event(GtkRange
* range
, GdkEventButton
*, wxWindow
* win
)
2067 g_blockEventsOnScroll
= false;
2068 win
->m_mouseButtonDown
= false;
2069 // If thumb tracking
2070 if (win
->m_isScrolling
)
2072 win
->m_isScrolling
= false;
2073 // Hook up handler to send thumb release event after this emission is finished.
2074 // To allow setting scroll position from event handler, sending event must
2075 // be deferred until after the GtkRange handler for this signal has run
2076 g_signal_handlers_unblock_by_func(range
, (void*)gtk_scrollbar_event_after
, win
);
2082 //-----------------------------------------------------------------------------
2083 // "realize" from m_widget
2084 //-----------------------------------------------------------------------------
2086 /* We cannot set colours and fonts before the widget has
2087 been realized, so we do this directly after realization. */
2090 gtk_window_realized_callback( GtkWidget
*m_widget
, wxWindow
*win
)
2095 wxapp_install_idle_handler();
2099 GtkPizza
*pizza
= GTK_PIZZA( m_widget
);
2100 gtk_im_context_set_client_window( win
->m_imData
->context
,
2101 pizza
->bin_window
);
2104 wxWindowCreateEvent
event( win
);
2105 event
.SetEventObject( win
);
2106 win
->GetEventHandler()->ProcessEvent( event
);
2109 //-----------------------------------------------------------------------------
2111 //-----------------------------------------------------------------------------
2114 void gtk_window_size_callback( GtkWidget
*WXUNUSED(widget
),
2115 GtkAllocation
*WXUNUSED(alloc
),
2119 wxapp_install_idle_handler();
2121 int client_width
= 0;
2122 int client_height
= 0;
2123 win
->GetClientSize( &client_width
, &client_height
);
2124 if ((client_width
== win
->m_oldClientWidth
) && (client_height
== win
->m_oldClientHeight
))
2127 win
->m_oldClientWidth
= client_width
;
2128 win
->m_oldClientHeight
= client_height
;
2130 if (!win
->m_nativeSizeEvent
)
2132 wxSizeEvent
event( win
->GetSize(), win
->GetId() );
2133 event
.SetEventObject( win
);
2134 win
->GetEventHandler()->ProcessEvent( event
);
2140 #define WXUNUSED_UNLESS_XIM(param) param
2142 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2145 /* Resize XIM window */
2147 void gtk_wxwindow_size_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2148 GtkAllocation
* WXUNUSED_UNLESS_XIM(alloc
),
2149 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2152 wxapp_install_idle_handler();
2158 if (gdk_ic_get_style (win
->m_ic
) & GDK_IM_PREEDIT_POSITION
)
2162 gdk_drawable_get_size (widget
->window
, &width
, &height
);
2163 win
->m_icattr
->preedit_area
.width
= width
;
2164 win
->m_icattr
->preedit_area
.height
= height
;
2165 gdk_ic_set_attr (win
->m_ic
, win
->m_icattr
, GDK_IC_PREEDIT_AREA
);
2170 //-----------------------------------------------------------------------------
2171 // "realize" from m_wxwindow
2172 //-----------------------------------------------------------------------------
2174 /* Initialize XIM support */
2177 gtk_wxwindow_realized_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2178 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2181 wxapp_install_idle_handler();
2184 if (win
->m_ic
) return;
2185 if (!widget
) return;
2186 if (!gdk_im_ready()) return;
2188 win
->m_icattr
= gdk_ic_attr_new();
2189 if (!win
->m_icattr
) return;
2193 GdkColormap
*colormap
;
2194 GdkICAttr
*attr
= win
->m_icattr
;
2195 unsigned attrmask
= GDK_IC_ALL_REQ
;
2197 GdkIMStyle supported_style
= (GdkIMStyle
)
2198 (GDK_IM_PREEDIT_NONE
|
2199 GDK_IM_PREEDIT_NOTHING
|
2200 GDK_IM_PREEDIT_POSITION
|
2201 GDK_IM_STATUS_NONE
|
2202 GDK_IM_STATUS_NOTHING
);
2204 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2205 supported_style
= (GdkIMStyle
)(supported_style
& ~GDK_IM_PREEDIT_POSITION
);
2207 attr
->style
= style
= gdk_im_decide_style (supported_style
);
2208 attr
->client_window
= widget
->window
;
2210 if ((colormap
= gtk_widget_get_colormap (widget
)) !=
2211 gtk_widget_get_default_colormap ())
2213 attrmask
|= GDK_IC_PREEDIT_COLORMAP
;
2214 attr
->preedit_colormap
= colormap
;
2217 attrmask
|= GDK_IC_PREEDIT_FOREGROUND
;
2218 attrmask
|= GDK_IC_PREEDIT_BACKGROUND
;
2219 attr
->preedit_foreground
= widget
->style
->fg
[GTK_STATE_NORMAL
];
2220 attr
->preedit_background
= widget
->style
->base
[GTK_STATE_NORMAL
];
2222 switch (style
& GDK_IM_PREEDIT_MASK
)
2224 case GDK_IM_PREEDIT_POSITION
:
2225 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2227 g_warning ("over-the-spot style requires fontset");
2231 gdk_drawable_get_size (widget
->window
, &width
, &height
);
2233 attrmask
|= GDK_IC_PREEDIT_POSITION_REQ
;
2234 attr
->spot_location
.x
= 0;
2235 attr
->spot_location
.y
= height
;
2236 attr
->preedit_area
.x
= 0;
2237 attr
->preedit_area
.y
= 0;
2238 attr
->preedit_area
.width
= width
;
2239 attr
->preedit_area
.height
= height
;
2240 attr
->preedit_fontset
= widget
->style
->font
;
2245 win
->m_ic
= gdk_ic_new (attr
, (GdkICAttributesType
)attrmask
);
2247 if (win
->m_ic
== NULL
)
2248 g_warning ("Can't create input context.");
2251 mask
= gdk_window_get_events (widget
->window
);
2252 mask
= (GdkEventMask
)(mask
| gdk_ic_get_events (win
->m_ic
));
2253 gdk_window_set_events (widget
->window
, mask
);
2255 if (GTK_WIDGET_HAS_FOCUS(widget
))
2256 gdk_im_begin (win
->m_ic
, widget
->window
);
2263 // ----------------------------------------------------------------------------
2264 // this wxWindowBase function is implemented here (in platform-specific file)
2265 // because it is static and so couldn't be made virtual
2266 // ----------------------------------------------------------------------------
2268 wxWindow
*wxWindowBase::DoFindFocus()
2270 // the cast is necessary when we compile in wxUniversal mode
2271 return (wxWindow
*)g_focusWindow
;
2274 //-----------------------------------------------------------------------------
2275 // InsertChild for wxWindowGTK.
2276 //-----------------------------------------------------------------------------
2278 /* Callback for wxWindowGTK. This very strange beast has to be used because
2279 * C++ has no virtual methods in a constructor. We have to emulate a
2280 * virtual function here as wxNotebook requires a different way to insert
2281 * a child in it. I had opted for creating a wxNotebookPage window class
2282 * which would have made this superfluous (such in the MDI window system),
2283 * but no-one was listening to me... */
2285 static void wxInsertChildInWindow( wxWindowGTK
* parent
, wxWindowGTK
* child
)
2287 /* the window might have been scrolled already, do we
2288 have to adapt the position */
2289 GtkPizza
*pizza
= GTK_PIZZA(parent
->m_wxwindow
);
2290 child
->m_x
+= pizza
->xoffset
;
2291 child
->m_y
+= pizza
->yoffset
;
2293 gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
),
2294 GTK_WIDGET(child
->m_widget
),
2301 //-----------------------------------------------------------------------------
2303 //-----------------------------------------------------------------------------
2305 wxWindow
*wxGetActiveWindow()
2307 return wxWindow::FindFocus();
2311 wxMouseState
wxGetMouseState()
2317 GdkModifierType mask
;
2319 gdk_window_get_pointer(NULL
, &x
, &y
, &mask
);
2323 ms
.SetLeftDown(mask
& GDK_BUTTON1_MASK
);
2324 ms
.SetMiddleDown(mask
& GDK_BUTTON2_MASK
);
2325 ms
.SetRightDown(mask
& GDK_BUTTON3_MASK
);
2327 ms
.SetControlDown(mask
& GDK_CONTROL_MASK
);
2328 ms
.SetShiftDown(mask
& GDK_SHIFT_MASK
);
2329 ms
.SetAltDown(mask
& GDK_MOD1_MASK
);
2330 ms
.SetMetaDown(mask
& GDK_MOD2_MASK
);
2335 //-----------------------------------------------------------------------------
2337 //-----------------------------------------------------------------------------
2339 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2341 #ifdef __WXUNIVERSAL__
2342 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
)
2344 IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
)
2345 #endif // __WXUNIVERSAL__/__WXGTK__
2347 void wxWindowGTK::Init()
2350 m_widget
= (GtkWidget
*) NULL
;
2351 m_wxwindow
= (GtkWidget
*) NULL
;
2352 m_focusWidget
= (GtkWidget
*) NULL
;
2362 m_needParent
= true;
2363 m_isBeingDeleted
= false;
2365 m_showOnIdle
= false;
2368 m_nativeSizeEvent
= false;
2370 m_hasScrolling
= false;
2371 m_isScrolling
= false;
2372 m_mouseButtonDown
= false;
2373 m_blockScrollEvent
= false;
2375 // initialize scrolling stuff
2376 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2378 m_scrollBar
[dir
] = NULL
;
2379 m_scrollPos
[dir
] = 0;
2380 m_blockValueChanged
[dir
] = false;
2384 m_oldClientHeight
= 0;
2388 m_insertCallback
= (wxInsertChildFunction
) NULL
;
2390 m_acceptsFocus
= false;
2393 m_clipPaintRegion
= false;
2395 m_needsStyleChange
= false;
2397 m_cursor
= *wxSTANDARD_CURSOR
;
2400 m_dirtyTabOrder
= false;
2403 wxWindowGTK::wxWindowGTK()
2408 wxWindowGTK::wxWindowGTK( wxWindow
*parent
,
2413 const wxString
&name
)
2417 Create( parent
, id
, pos
, size
, style
, name
);
2420 bool wxWindowGTK::Create( wxWindow
*parent
,
2425 const wxString
&name
)
2427 if (!PreCreation( parent
, pos
, size
) ||
2428 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
2430 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2434 m_insertCallback
= wxInsertChildInWindow
;
2436 m_widget
= gtk_scrolled_window_new( (GtkAdjustment
*) NULL
, (GtkAdjustment
*) NULL
);
2437 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS
);
2439 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(m_widget
);
2441 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2442 scroll_class
->scrollbar_spacing
= 0;
2444 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
2446 m_scrollBar
[ScrollDir_Horz
] = GTK_RANGE(scrolledWindow
->hscrollbar
);
2447 m_scrollBar
[ScrollDir_Vert
] = GTK_RANGE(scrolledWindow
->vscrollbar
);
2449 m_wxwindow
= gtk_pizza_new();
2451 #ifndef __WXUNIVERSAL__
2452 GtkPizza
*pizza
= GTK_PIZZA(m_wxwindow
);
2454 if (HasFlag(wxRAISED_BORDER
))
2456 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_OUT
);
2458 else if (HasFlag(wxSUNKEN_BORDER
))
2460 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_IN
);
2462 else if (HasFlag(wxSIMPLE_BORDER
))
2464 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_THIN
);
2468 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_NONE
);
2470 #endif // __WXUNIVERSAL__
2472 gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow
);
2474 GTK_WIDGET_SET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS
);
2475 m_acceptsFocus
= true;
2477 // connect various scroll-related events
2478 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
2480 // these handlers block mouse events to any window during scrolling
2481 // such as motion events and prevent GTK and wxWidgets from fighting
2482 // over where the slider should be
2483 g_signal_connect(m_scrollBar
[dir
], "button_press_event",
2484 G_CALLBACK(gtk_scrollbar_button_press_event
), this);
2485 g_signal_connect(m_scrollBar
[dir
], "button_release_event",
2486 G_CALLBACK(gtk_scrollbar_button_release_event
), this);
2488 gulong handler_id
= g_signal_connect(m_scrollBar
[dir
], "event_after",
2489 G_CALLBACK(gtk_scrollbar_event_after
), this);
2490 g_signal_handler_block(m_scrollBar
[dir
], handler_id
);
2492 // these handlers get notified when scrollbar slider moves
2493 g_signal_connect(m_scrollBar
[dir
], "value_changed",
2494 G_CALLBACK(gtk_scrollbar_value_changed
), this);
2497 gtk_widget_show( m_wxwindow
);
2500 m_parent
->DoAddChild( this );
2502 m_focusWidget
= m_wxwindow
;
2509 wxWindowGTK::~wxWindowGTK()
2513 if (g_focusWindow
== this)
2514 g_focusWindow
= NULL
;
2516 if ( g_delayedFocus
== this )
2517 g_delayedFocus
= NULL
;
2519 m_isBeingDeleted
= true;
2522 // destroy children before destroying this window itself
2525 // unhook focus handlers to prevent stray events being
2526 // propagated to this (soon to be) dead object
2527 if (m_focusWidget
!= NULL
)
2529 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2530 (gpointer
) gtk_window_focus_in_callback
,
2532 g_signal_handlers_disconnect_by_func (m_focusWidget
,
2533 (gpointer
) gtk_window_focus_out_callback
,
2542 gdk_ic_destroy (m_ic
);
2544 gdk_ic_attr_destroy (m_icattr
);
2547 // delete before the widgets to avoid a crash on solaris
2552 gtk_widget_destroy( m_wxwindow
);
2553 m_wxwindow
= (GtkWidget
*) NULL
;
2558 gtk_widget_destroy( m_widget
);
2559 m_widget
= (GtkWidget
*) NULL
;
2563 bool wxWindowGTK::PreCreation( wxWindowGTK
*parent
, const wxPoint
&pos
, const wxSize
&size
)
2565 wxCHECK_MSG( !m_needParent
|| parent
, false, wxT("Need complete parent.") );
2567 // Use either the given size, or the default if -1 is given.
2568 // See wxWindowBase for these functions.
2569 m_width
= WidthDefault(size
.x
) ;
2570 m_height
= HeightDefault(size
.y
);
2578 void wxWindowGTK::PostCreation()
2580 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2586 // these get reported to wxWidgets -> wxPaintEvent
2588 gtk_pizza_set_external( GTK_PIZZA(m_wxwindow
), TRUE
);
2590 g_signal_connect (m_wxwindow
, "expose_event",
2591 G_CALLBACK (gtk_window_expose_callback
), this);
2593 gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow
), HasFlag( wxFULL_REPAINT_ON_RESIZE
) );
2596 // Create input method handler
2597 m_imData
= new wxGtkIMData
;
2599 // Cannot handle drawing preedited text yet
2600 gtk_im_context_set_use_preedit( m_imData
->context
, FALSE
);
2602 g_signal_connect (m_imData
->context
, "commit",
2603 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2605 // these are called when the "sunken" or "raised" borders are drawn
2606 g_signal_connect (m_widget
, "expose_event",
2607 G_CALLBACK (gtk_window_own_expose_callback
), this);
2612 if (!GTK_IS_WINDOW(m_widget
))
2614 if (m_focusWidget
== NULL
)
2615 m_focusWidget
= m_widget
;
2619 g_signal_connect (m_focusWidget
, "focus_in_event",
2620 G_CALLBACK (gtk_window_focus_in_callback
), this);
2621 g_signal_connect (m_focusWidget
, "focus_out_event",
2622 G_CALLBACK (gtk_window_focus_out_callback
), this);
2626 g_signal_connect_after (m_focusWidget
, "focus_in_event",
2627 G_CALLBACK (gtk_window_focus_in_callback
), this);
2628 g_signal_connect_after (m_focusWidget
, "focus_out_event",
2629 G_CALLBACK (gtk_window_focus_out_callback
), this);
2633 // connect to the various key and mouse handlers
2635 GtkWidget
*connect_widget
= GetConnectWidget();
2637 ConnectWidget( connect_widget
);
2639 /* We cannot set colours, fonts and cursors before the widget has
2640 been realized, so we do this directly after realization */
2641 g_signal_connect (connect_widget
, "realize",
2642 G_CALLBACK (gtk_window_realized_callback
), this);
2646 // Catch native resize events
2647 g_signal_connect (m_wxwindow
, "size_allocate",
2648 G_CALLBACK (gtk_window_size_callback
), this);
2650 // Initialize XIM support
2651 g_signal_connect (m_wxwindow
, "realize",
2652 G_CALLBACK (gtk_wxwindow_realized_callback
), this);
2654 // And resize XIM window
2655 g_signal_connect (m_wxwindow
, "size_allocate",
2656 G_CALLBACK (gtk_wxwindow_size_callback
), this);
2659 if (GTK_IS_COMBO(m_widget
))
2661 GtkCombo
*gcombo
= GTK_COMBO(m_widget
);
2663 g_signal_connect (gcombo
->entry
, "size_request",
2664 G_CALLBACK (wxgtk_combo_size_request_callback
),
2667 #ifdef GTK_IS_FILE_CHOOSER_BUTTON
2668 else if (GTK_IS_FILE_CHOOSER_BUTTON(m_widget
))
2670 // If we connect to the "size_request" signal of a GtkFileChooserButton
2671 // then that control won't be sized properly when placed inside sizers
2672 // (this can be tested removing this elseif and running XRC or WIDGETS samples)
2673 // FIXME: what should be done here ?
2678 // This is needed if we want to add our windows into native
2679 // GTK controls, such as the toolbar. With this callback, the
2680 // toolbar gets to know the correct size (the one set by the
2681 // programmer). Sadly, it misbehaves for wxComboBox.
2682 g_signal_connect (m_widget
, "size_request",
2683 G_CALLBACK (wxgtk_window_size_request_callback
),
2687 InheritAttributes();
2691 // unless the window was created initially hidden (i.e. Hide() had been
2692 // called before Create()), we should show it at GTK+ level as well
2694 gtk_widget_show( m_widget
);
2697 void wxWindowGTK::ConnectWidget( GtkWidget
*widget
)
2699 g_signal_connect (widget
, "key_press_event",
2700 G_CALLBACK (gtk_window_key_press_callback
), this);
2701 g_signal_connect (widget
, "key_release_event",
2702 G_CALLBACK (gtk_window_key_release_callback
), this);
2703 g_signal_connect (widget
, "button_press_event",
2704 G_CALLBACK (gtk_window_button_press_callback
), this);
2705 g_signal_connect (widget
, "button_release_event",
2706 G_CALLBACK (gtk_window_button_release_callback
), this);
2707 g_signal_connect (widget
, "motion_notify_event",
2708 G_CALLBACK (gtk_window_motion_notify_callback
), this);
2709 g_signal_connect (widget
, "scroll_event",
2710 G_CALLBACK (window_scroll_event
), this);
2711 g_signal_connect (widget
, "popup_menu",
2712 G_CALLBACK (wxgtk_window_popup_menu_callback
), this);
2713 g_signal_connect (widget
, "enter_notify_event",
2714 G_CALLBACK (gtk_window_enter_callback
), this);
2715 g_signal_connect (widget
, "leave_notify_event",
2716 G_CALLBACK (gtk_window_leave_callback
), this);
2719 bool wxWindowGTK::Destroy()
2721 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2725 return wxWindowBase::Destroy();
2728 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
)
2730 gtk_pizza_set_size( GTK_PIZZA(m_parent
->m_wxwindow
), m_widget
, x
, y
, width
, height
);
2733 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
2735 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2736 wxASSERT_MSG( (m_parent
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") );
2739 printf( "DoSetSize: name %s, x,y,w,h: %d,%d,%d,%d \n", GetName().c_str(), x,y,width,height );
2742 if (m_resizing
) return; /* I don't like recursions */
2745 int currentX
, currentY
;
2746 GetPosition(¤tX
, ¤tY
);
2747 if (x
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2749 if (y
== -1 && !(sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
))
2751 AdjustForParentClientOrigin(x
, y
, sizeFlags
);
2753 // calculate the best size if we should auto size the window
2754 if ( ((sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1) ||
2755 ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1) )
2757 const wxSize sizeBest
= GetBestSize();
2758 if ( (sizeFlags
& wxSIZE_AUTO_WIDTH
) && width
== -1 )
2760 if ( (sizeFlags
& wxSIZE_AUTO_HEIGHT
) && height
== -1 )
2761 height
= sizeBest
.y
;
2769 int minWidth
= GetMinWidth(),
2770 minHeight
= GetMinHeight(),
2771 maxWidth
= GetMaxWidth(),
2772 maxHeight
= GetMaxHeight();
2774 if ((minWidth
!= -1) && (m_width
< minWidth
)) m_width
= minWidth
;
2775 if ((minHeight
!= -1) && (m_height
< minHeight
)) m_height
= minHeight
;
2776 if ((maxWidth
!= -1) && (m_width
> maxWidth
)) m_width
= maxWidth
;
2777 if ((maxHeight
!= -1) && (m_height
> maxHeight
)) m_height
= maxHeight
;
2779 #if wxUSE_TOOLBAR_NATIVE
2780 if (wxDynamicCast(GetParent(), wxToolBar
))
2782 // don't take the x,y values, they're wrong because toolbar sets them
2783 GtkWidget
*widget
= GTK_WIDGET(m_widget
);
2784 gtk_widget_set_size_request (widget
, m_width
, m_height
);
2785 if (GTK_WIDGET_VISIBLE (widget
))
2786 gtk_widget_queue_resize (widget
);
2790 if (m_parent
->m_wxwindow
== NULL
) // i.e. wxNotebook
2792 // don't set the size for children of wxNotebook, just take the values.
2800 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
2801 if ((sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) == 0)
2803 if (x
!= -1) m_x
= x
+ pizza
->xoffset
;
2804 if (y
!= -1) m_y
= y
+ pizza
->yoffset
;
2808 m_x
= x
+ pizza
->xoffset
;
2809 m_y
= y
+ pizza
->yoffset
;
2812 int left_border
= 0;
2813 int right_border
= 0;
2815 int bottom_border
= 0;
2817 /* the default button has a border around it */
2818 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
2820 GtkBorder
*default_border
= NULL
;
2821 gtk_widget_style_get( m_widget
, "default_border", &default_border
, NULL
);
2824 left_border
+= default_border
->left
;
2825 right_border
+= default_border
->right
;
2826 top_border
+= default_border
->top
;
2827 bottom_border
+= default_border
->bottom
;
2828 g_free( default_border
);
2832 DoMoveWindow( m_x
-top_border
,
2834 m_width
+left_border
+right_border
,
2835 m_height
+top_border
+bottom_border
);
2840 /* Sometimes the client area changes size without the
2841 whole windows's size changing, but if the whole
2842 windows's size doesn't change, no wxSizeEvent will
2843 normally be sent. Here we add an extra test if
2844 the client test has been changed and this will
2846 GetClientSize( &m_oldClientWidth
, &m_oldClientHeight
);
2850 wxPrintf( "OnSize sent from " );
2851 if (GetClassInfo() && GetClassInfo()->GetClassName())
2852 wxPrintf( GetClassInfo()->GetClassName() );
2853 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
2856 if (!m_nativeSizeEvent
)
2858 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
2859 event
.SetEventObject( this );
2860 GetEventHandler()->ProcessEvent( event
);
2866 bool wxWindowGTK::GtkShowFromOnIdle()
2868 if (IsShown() && m_showOnIdle
&& !GTK_WIDGET_VISIBLE (m_widget
))
2870 GtkAllocation alloc
;
2873 alloc
.width
= m_width
;
2874 alloc
.height
= m_height
;
2875 gtk_widget_size_allocate( m_widget
, &alloc
);
2876 gtk_widget_show( m_widget
);
2877 wxShowEvent
eventShow(GetId(), true);
2878 eventShow
.SetEventObject(this);
2879 GetEventHandler()->ProcessEvent(eventShow
);
2880 m_showOnIdle
= false;
2887 void wxWindowGTK::OnInternalIdle()
2889 // Check if we have to show window now
2890 if (GtkShowFromOnIdle()) return;
2892 if ( m_dirtyTabOrder
)
2894 m_dirtyTabOrder
= false;
2898 // Update style if the window was not yet realized
2899 // and SetBackgroundStyle(wxBG_STYLE_CUSTOM) was called
2900 if (m_needsStyleChange
)
2902 SetBackgroundStyle(GetBackgroundStyle());
2903 m_needsStyleChange
= false;
2906 wxCursor cursor
= m_cursor
;
2907 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
2911 /* I now set the cursor anew in every OnInternalIdle call
2912 as setting the cursor in a parent window also effects the
2913 windows above so that checking for the current cursor is
2918 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
2920 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2922 if (!g_globalCursor
.Ok())
2923 cursor
= *wxSTANDARD_CURSOR
;
2925 window
= m_widget
->window
;
2926 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
2927 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2930 else if ( m_widget
)
2932 GdkWindow
*window
= m_widget
->window
;
2933 if ( window
&& !GTK_WIDGET_NO_WINDOW(m_widget
) )
2934 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2938 if (wxUpdateUIEvent::CanUpdate(this))
2939 UpdateWindowUI(wxUPDATE_UI_FROMIDLE
);
2942 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
2944 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2946 if (width
) (*width
) = m_width
;
2947 if (height
) (*height
) = m_height
;
2950 void wxWindowGTK::DoSetClientSize( int width
, int height
)
2952 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2961 GetScrollbarWidth(m_widget
, dw
, dh
);
2964 #ifndef __WXUNIVERSAL__
2965 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
2967 // shadow border size is 2
2971 if (HasFlag(wxSIMPLE_BORDER
))
2973 // simple border size is 1
2977 #endif // __WXUNIVERSAL__
2983 SetSize(width
, height
);
2986 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
2988 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3000 GetScrollbarWidth(m_widget
, dw
, dh
);
3003 #ifndef __WXUNIVERSAL__
3004 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3006 // shadow border size is 2
3010 if (HasFlag(wxSIMPLE_BORDER
))
3012 // simple border size is 1
3016 #endif // __WXUNIVERSAL__
3022 if (width
) *width
= w
;
3023 if (height
) *height
= h
;
3026 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
3028 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3032 if (m_parent
&& m_parent
->m_wxwindow
)
3034 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
3035 dx
= pizza
->xoffset
;
3036 dy
= pizza
->yoffset
;
3039 if (x
) (*x
) = m_x
- dx
;
3040 if (y
) (*y
) = m_y
- dy
;
3043 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
3045 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3047 if (!m_widget
->window
) return;
3049 GdkWindow
*source
= (GdkWindow
*) NULL
;
3051 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3053 source
= m_widget
->window
;
3057 gdk_window_get_origin( source
, &org_x
, &org_y
);
3061 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3063 org_x
+= m_widget
->allocation
.x
;
3064 org_y
+= m_widget
->allocation
.y
;
3072 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
3074 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3076 if (!m_widget
->window
) return;
3078 GdkWindow
*source
= (GdkWindow
*) NULL
;
3080 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3082 source
= m_widget
->window
;
3086 gdk_window_get_origin( source
, &org_x
, &org_y
);
3090 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3092 org_x
+= m_widget
->allocation
.x
;
3093 org_y
+= m_widget
->allocation
.y
;
3101 bool wxWindowGTK::Show( bool show
)
3103 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3105 if (!wxWindowBase::Show(show
))
3115 gtk_widget_show( m_widget
);
3116 wxShowEvent
eventShow(GetId(), show
);
3117 eventShow
.SetEventObject(this);
3118 GetEventHandler()->ProcessEvent(eventShow
);
3123 gtk_widget_hide( m_widget
);
3124 wxShowEvent
eventShow(GetId(), show
);
3125 eventShow
.SetEventObject(this);
3126 GetEventHandler()->ProcessEvent(eventShow
);
3132 static void wxWindowNotifyEnable(wxWindowGTK
* win
, bool enable
)
3134 win
->OnParentEnable(enable
);
3136 // Recurse, so that children have the opportunity to Do The Right Thing
3137 // and reset colours that have been messed up by a parent's (really ancestor's)
3139 for ( wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
3141 node
= node
->GetNext() )
3143 wxWindow
*child
= node
->GetData();
3144 if (!child
->IsKindOf(CLASSINFO(wxDialog
)) && !child
->IsKindOf(CLASSINFO(wxFrame
)))
3145 wxWindowNotifyEnable(child
, enable
);
3149 bool wxWindowGTK::Enable( bool enable
)
3151 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3153 if (!wxWindowBase::Enable(enable
))
3159 gtk_widget_set_sensitive( m_widget
, enable
);
3161 gtk_widget_set_sensitive( m_wxwindow
, enable
);
3163 wxWindowNotifyEnable(this, enable
);
3168 int wxWindowGTK::GetCharHeight() const
3170 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
3172 wxFont font
= GetFont();
3173 wxCHECK_MSG( font
.Ok(), 12, wxT("invalid font") );
3175 PangoContext
*context
= NULL
;
3177 context
= gtk_widget_get_pango_context( m_widget
);
3182 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3183 PangoLayout
*layout
= pango_layout_new(context
);
3184 pango_layout_set_font_description(layout
, desc
);
3185 pango_layout_set_text(layout
, "H", 1);
3186 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3188 PangoRectangle rect
;
3189 pango_layout_line_get_extents(line
, NULL
, &rect
);
3191 g_object_unref (layout
);
3193 return (int) PANGO_PIXELS(rect
.height
);
3196 int wxWindowGTK::GetCharWidth() const
3198 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
3200 wxFont font
= GetFont();
3201 wxCHECK_MSG( font
.Ok(), 8, wxT("invalid font") );
3203 PangoContext
*context
= NULL
;
3205 context
= gtk_widget_get_pango_context( m_widget
);
3210 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3211 PangoLayout
*layout
= pango_layout_new(context
);
3212 pango_layout_set_font_description(layout
, desc
);
3213 pango_layout_set_text(layout
, "g", 1);
3214 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3216 PangoRectangle rect
;
3217 pango_layout_line_get_extents(line
, NULL
, &rect
);
3219 g_object_unref (layout
);
3221 return (int) PANGO_PIXELS(rect
.width
);
3224 void wxWindowGTK::GetTextExtent( const wxString
& string
,
3228 int *externalLeading
,
3229 const wxFont
*theFont
) const
3231 wxFont fontToUse
= theFont
? *theFont
: GetFont();
3233 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
3242 PangoContext
*context
= NULL
;
3244 context
= gtk_widget_get_pango_context( m_widget
);
3253 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
3254 PangoLayout
*layout
= pango_layout_new(context
);
3255 pango_layout_set_font_description(layout
, desc
);
3257 const wxCharBuffer data
= wxGTK_CONV( string
);
3259 pango_layout_set_text(layout
, data
, strlen(data
));
3262 PangoRectangle rect
;
3263 pango_layout_get_extents(layout
, NULL
, &rect
);
3265 if (x
) (*x
) = (wxCoord
) PANGO_PIXELS(rect
.width
);
3266 if (y
) (*y
) = (wxCoord
) PANGO_PIXELS(rect
.height
);
3269 PangoLayoutIter
*iter
= pango_layout_get_iter(layout
);
3270 int baseline
= pango_layout_iter_get_baseline(iter
);
3271 pango_layout_iter_free(iter
);
3272 *descent
= *y
- PANGO_PIXELS(baseline
);
3274 if (externalLeading
) (*externalLeading
) = 0; // ??
3276 g_object_unref (layout
);
3279 bool wxWindowGTK::GTKSetDelayedFocusIfNeeded()
3281 if ( g_delayedFocus
== this )
3283 if ( GTK_WIDGET_REALIZED(m_widget
) )
3285 gtk_widget_grab_focus(m_widget
);
3286 g_delayedFocus
= NULL
;
3295 void wxWindowGTK::SetFocus()
3297 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3300 // don't do anything if we already have focus
3306 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow
))
3308 gtk_widget_grab_focus (m_wxwindow
);
3313 if (GTK_IS_CONTAINER(m_widget
))
3315 gtk_widget_child_focus( m_widget
, GTK_DIR_TAB_FORWARD
);
3318 if (GTK_WIDGET_CAN_FOCUS(m_widget
) && !GTK_WIDGET_HAS_FOCUS (m_widget
) )
3321 if (!GTK_WIDGET_REALIZED(m_widget
))
3323 // we can't set the focus to the widget now so we remember that
3324 // it should be focused and will do it later, during the idle
3325 // time, as soon as we can
3326 wxLogTrace(TRACE_FOCUS
,
3327 _T("Delaying setting focus to %s(%s)"),
3328 GetClassInfo()->GetClassName(), GetLabel().c_str());
3330 g_delayedFocus
= this;
3334 wxLogTrace(TRACE_FOCUS
,
3335 _T("Setting focus to %s(%s)"),
3336 GetClassInfo()->GetClassName(), GetLabel().c_str());
3338 gtk_widget_grab_focus (m_widget
);
3343 wxLogTrace(TRACE_FOCUS
,
3344 _T("Can't set focus to %s(%s)"),
3345 GetClassInfo()->GetClassName(), GetLabel().c_str());
3350 bool wxWindowGTK::AcceptsFocus() const
3352 return m_acceptsFocus
&& wxWindowBase::AcceptsFocus();
3355 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3357 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3359 wxWindowGTK
*oldParent
= m_parent
,
3360 *newParent
= (wxWindowGTK
*)newParentBase
;
3362 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3364 if ( !wxWindowBase::Reparent(newParent
) )
3367 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3369 /* prevent GTK from deleting the widget arbitrarily */
3370 gtk_widget_ref( m_widget
);
3374 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3377 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3381 if (GTK_WIDGET_VISIBLE (newParent
->m_widget
))
3383 m_showOnIdle
= true;
3384 gtk_widget_hide( m_widget
);
3387 /* insert GTK representation */
3388 (*(newParent
->m_insertCallback
))(newParent
, this);
3391 /* reverse: prevent GTK from deleting the widget arbitrarily */
3392 gtk_widget_unref( m_widget
);
3397 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3399 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3401 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3403 wxASSERT_MSG( (m_insertCallback
!= NULL
), wxT("invalid child insertion function") );
3408 /* insert GTK representation */
3409 (*m_insertCallback
)(this, child
);
3412 void wxWindowGTK::AddChild(wxWindowBase
*child
)
3414 wxWindowBase::AddChild(child
);
3415 m_dirtyTabOrder
= true;
3417 wxapp_install_idle_handler();
3420 void wxWindowGTK::RemoveChild(wxWindowBase
*child
)
3422 wxWindowBase::RemoveChild(child
);
3423 m_dirtyTabOrder
= true;
3425 wxapp_install_idle_handler();
3428 void wxWindowGTK::DoMoveInTabOrder(wxWindow
*win
, MoveKind move
)
3430 wxWindowBase::DoMoveInTabOrder(win
, move
);
3431 m_dirtyTabOrder
= true;
3433 wxapp_install_idle_handler();
3436 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3438 // none needed by default
3442 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget
* WXUNUSED(w
))
3444 // nothing to do by default since none is needed
3447 void wxWindowGTK::RealizeTabOrder()
3451 if ( !m_children
.empty() )
3453 // we don't only construct the correct focus chain but also use
3454 // this opportunity to update the mnemonic widgets for the widgets
3457 GList
*chain
= NULL
;
3458 wxWindowGTK
* mnemonicWindow
= NULL
;
3460 for ( wxWindowList::const_iterator i
= m_children
.begin();
3461 i
!= m_children
.end();
3464 wxWindowGTK
*win
= *i
;
3466 if ( mnemonicWindow
)
3468 if ( win
->AcceptsFocusFromKeyboard() )
3470 // wxComboBox et al. needs to focus on on a different
3471 // widget than m_widget, so if the main widget isn't
3472 // focusable try the connect widget
3473 GtkWidget
* w
= win
->m_widget
;
3474 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3476 w
= win
->GetConnectWidget();
3477 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3483 mnemonicWindow
->GTKWidgetDoSetMnemonic(w
);
3484 mnemonicWindow
= NULL
;
3488 else if ( win
->GTKWidgetNeedsMnemonic() )
3490 mnemonicWindow
= win
;
3493 chain
= g_list_prepend(chain
, win
->m_widget
);
3496 chain
= g_list_reverse(chain
);
3498 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow
), chain
);
3503 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow
));
3508 void wxWindowGTK::Raise()
3510 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3512 if (m_wxwindow
&& m_wxwindow
->window
)
3514 gdk_window_raise( m_wxwindow
->window
);
3516 else if (m_widget
->window
)
3518 gdk_window_raise( m_widget
->window
);
3522 void wxWindowGTK::Lower()
3524 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3526 if (m_wxwindow
&& m_wxwindow
->window
)
3528 gdk_window_lower( m_wxwindow
->window
);
3530 else if (m_widget
->window
)
3532 gdk_window_lower( m_widget
->window
);
3536 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3538 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3540 if (cursor
== m_cursor
)
3544 wxapp_install_idle_handler();
3546 return wxWindowBase::SetCursor( cursor
.Ok() ? cursor
: *wxSTANDARD_CURSOR
);
3549 void wxWindowGTK::GTKUpdateCursor()
3551 wxCursor
cursor(g_globalCursor
.Ok() ? g_globalCursor
: GetCursor());
3554 wxArrayGdkWindows windowsThis
;
3555 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3558 gdk_window_set_cursor(winThis
, cursor
.GetCursor());
3562 const size_t count
= windowsThis
.size();
3563 for ( size_t n
= 0; n
< count
; n
++ )
3565 gdk_window_set_cursor(windowsThis
[n
], cursor
.GetCursor());
3571 void wxWindowGTK::WarpPointer( int x
, int y
)
3573 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3575 // We provide this function ourselves as it is
3576 // missing in GDK (top of this file).
3578 GdkWindow
*window
= (GdkWindow
*) NULL
;
3580 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3582 window
= GetConnectWidget()->window
;
3585 gdk_window_warp_pointer( window
, x
, y
);
3588 wxWindowGTK::ScrollDir
wxWindowGTK::ScrollDirFromRange(GtkRange
*range
) const
3590 // find the scrollbar which generated the event
3591 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
3593 if ( range
== m_scrollBar
[dir
] )
3594 return (ScrollDir
)dir
;
3597 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3599 return ScrollDir_Max
;
3602 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir
, ScrollUnit unit
, int units
)
3604 bool changed
= false;
3605 GtkRange
* range
= m_scrollBar
[dir
];
3606 if ( range
&& units
)
3608 GtkAdjustment
* adj
= range
->adjustment
;
3609 gdouble inc
= unit
== ScrollUnit_Line
? adj
->step_increment
3610 : adj
->page_increment
;
3612 const int posOld
= int(adj
->value
+ 0.5);
3613 gtk_range_set_value(range
, posOld
+ units
*inc
);
3615 changed
= int(adj
->value
+ 0.5) != posOld
;
3621 bool wxWindowGTK::ScrollLines(int lines
)
3623 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Line
, lines
);
3626 bool wxWindowGTK::ScrollPages(int pages
)
3628 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Page
, pages
);
3631 void wxWindowGTK::Refresh( bool eraseBackground
, const wxRect
*rect
)
3635 if (!m_widget
->window
)
3640 if (!GTK_PIZZA(m_wxwindow
)->bin_window
) return;
3642 GdkRectangle gdk_rect
,
3646 gdk_rect
.x
= rect
->x
;
3647 gdk_rect
.y
= rect
->y
;
3648 gdk_rect
.width
= rect
->width
;
3649 gdk_rect
.height
= rect
->height
;
3652 else // invalidate everything
3657 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, p
, TRUE
);
3661 void wxWindowGTK::Update()
3665 // when we call Update() we really want to update the window immediately on
3666 // screen, even if it means flushing the entire queue and hence slowing down
3667 // everything -- but it should still be done, it's just that Update() should
3668 // be called very rarely
3672 void wxWindowGTK::GtkUpdate()
3674 if (m_wxwindow
&& GTK_PIZZA(m_wxwindow
)->bin_window
)
3675 gdk_window_process_updates( GTK_PIZZA(m_wxwindow
)->bin_window
, FALSE
);
3676 if (m_widget
&& m_widget
->window
)
3677 gdk_window_process_updates( m_widget
->window
, FALSE
);
3679 // for consistency with other platforms (and also because it's convenient
3680 // to be able to update an entire TLW by calling Update() only once), we
3681 // should also update all our children here
3682 for ( wxWindowList::compatibility_iterator node
= GetChildren().GetFirst();
3684 node
= node
->GetNext() )
3686 node
->GetData()->GtkUpdate();
3690 void wxWindowGTK::GtkSendPaintEvents()
3694 m_updateRegion
.Clear();
3698 // Clip to paint region in wxClientDC
3699 m_clipPaintRegion
= true;
3701 // widget to draw on
3702 GtkPizza
*pizza
= GTK_PIZZA (m_wxwindow
);
3704 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM
))
3706 // find ancestor from which to steal background
3707 wxWindow
*parent
= wxGetTopLevelParent((wxWindow
*)this);
3709 parent
= (wxWindow
*)this;
3711 if (GTK_WIDGET_MAPPED(parent
->m_widget
))
3713 wxRegionIterator
upd( m_updateRegion
);
3717 rect
.x
= upd
.GetX();
3718 rect
.y
= upd
.GetY();
3719 rect
.width
= upd
.GetWidth();
3720 rect
.height
= upd
.GetHeight();
3722 gtk_paint_flat_box( parent
->m_widget
->style
,
3724 (GtkStateType
)GTK_WIDGET_STATE(m_wxwindow
),
3738 wxWindowDC
dc( (wxWindow
*)this );
3739 dc
.SetClippingRegion( m_updateRegion
);
3741 wxEraseEvent
erase_event( GetId(), &dc
);
3742 erase_event
.SetEventObject( this );
3744 GetEventHandler()->ProcessEvent(erase_event
);
3747 wxNcPaintEvent
nc_paint_event( GetId() );
3748 nc_paint_event
.SetEventObject( this );
3749 GetEventHandler()->ProcessEvent( nc_paint_event
);
3751 wxPaintEvent
paint_event( GetId() );
3752 paint_event
.SetEventObject( this );
3753 GetEventHandler()->ProcessEvent( paint_event
);
3755 m_clipPaintRegion
= false;
3757 m_updateRegion
.Clear();
3760 void wxWindowGTK::SetDoubleBuffered( bool on
)
3762 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3765 gtk_widget_set_double_buffered( m_wxwindow
, on
);
3768 void wxWindowGTK::ClearBackground()
3770 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3774 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3776 wxWindowBase::DoSetToolTip(tip
);
3779 m_tooltip
->Apply( (wxWindow
*)this );
3782 void wxWindowGTK::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
3784 wxString
tmp( tip
);
3785 gtk_tooltips_set_tip( tips
, GetConnectWidget(), wxGTK_CONV(tmp
), (gchar
*) NULL
);
3787 #endif // wxUSE_TOOLTIPS
3789 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
3791 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3793 if (!wxWindowBase::SetBackgroundColour(colour
))
3798 // We need the pixel value e.g. for background clearing.
3799 m_backgroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3802 // apply style change (forceStyle=true so that new style is applied
3803 // even if the bg colour changed from valid to wxNullColour)
3804 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM
)
3805 ApplyWidgetStyle(true);
3810 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
3812 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3814 if (!wxWindowBase::SetForegroundColour(colour
))
3821 // We need the pixel value e.g. for background clearing.
3822 m_foregroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3825 // apply style change (forceStyle=true so that new style is applied
3826 // even if the bg colour changed from valid to wxNullColour):
3827 ApplyWidgetStyle(true);
3832 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
3834 return gtk_widget_get_pango_context( m_widget
);
3837 GtkRcStyle
*wxWindowGTK::CreateWidgetStyle(bool forceStyle
)
3839 // do we need to apply any changes at all?
3842 !m_foregroundColour
.Ok() && !m_backgroundColour
.Ok() )
3847 GtkRcStyle
*style
= gtk_rc_style_new();
3852 pango_font_description_copy( m_font
.GetNativeFontInfo()->description
);
3855 if ( m_foregroundColour
.Ok() )
3857 const GdkColor
*fg
= m_foregroundColour
.GetColor();
3859 style
->fg
[GTK_STATE_NORMAL
] = *fg
;
3860 style
->color_flags
[GTK_STATE_NORMAL
] = GTK_RC_FG
;
3862 style
->fg
[GTK_STATE_PRELIGHT
] = *fg
;
3863 style
->color_flags
[GTK_STATE_PRELIGHT
] = GTK_RC_FG
;
3865 style
->fg
[GTK_STATE_ACTIVE
] = *fg
;
3866 style
->color_flags
[GTK_STATE_ACTIVE
] = GTK_RC_FG
;
3869 if ( m_backgroundColour
.Ok() )
3871 const GdkColor
*bg
= m_backgroundColour
.GetColor();
3873 style
->bg
[GTK_STATE_NORMAL
] = *bg
;
3874 style
->base
[GTK_STATE_NORMAL
] = *bg
;
3875 style
->color_flags
[GTK_STATE_NORMAL
] = (GtkRcFlags
)
3876 (style
->color_flags
[GTK_STATE_NORMAL
] | GTK_RC_BG
| GTK_RC_BASE
);
3878 style
->bg
[GTK_STATE_PRELIGHT
] = *bg
;
3879 style
->base
[GTK_STATE_PRELIGHT
] = *bg
;
3880 style
->color_flags
[GTK_STATE_PRELIGHT
] = (GtkRcFlags
)
3881 (style
->color_flags
[GTK_STATE_PRELIGHT
] | GTK_RC_BG
| GTK_RC_BASE
);
3883 style
->bg
[GTK_STATE_ACTIVE
] = *bg
;
3884 style
->base
[GTK_STATE_ACTIVE
] = *bg
;
3885 style
->color_flags
[GTK_STATE_ACTIVE
] = (GtkRcFlags
)
3886 (style
->color_flags
[GTK_STATE_ACTIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
3888 style
->bg
[GTK_STATE_INSENSITIVE
] = *bg
;
3889 style
->base
[GTK_STATE_INSENSITIVE
] = *bg
;
3890 style
->color_flags
[GTK_STATE_INSENSITIVE
] = (GtkRcFlags
)
3891 (style
->color_flags
[GTK_STATE_INSENSITIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
3897 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle
)
3899 GtkRcStyle
*style
= CreateWidgetStyle(forceStyle
);
3902 DoApplyWidgetStyle(style
);
3903 gtk_rc_style_unref(style
);
3906 // Style change may affect GTK+'s size calculation:
3907 InvalidateBestSize();
3910 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle
*style
)
3913 gtk_widget_modify_style(m_wxwindow
, style
);
3915 gtk_widget_modify_style(m_widget
, style
);
3918 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style
)
3920 wxWindowBase::SetBackgroundStyle(style
);
3922 if (style
== wxBG_STYLE_CUSTOM
)
3924 GdkWindow
*window
= (GdkWindow
*) NULL
;
3926 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3928 window
= GetConnectWidget()->window
;
3932 // Make sure GDK/X11 doesn't refresh the window
3934 gdk_window_set_back_pixmap( window
, None
, False
);
3936 Display
* display
= GDK_WINDOW_DISPLAY(window
);
3939 m_needsStyleChange
= false;
3942 // Do in OnIdle, because the window is not yet available
3943 m_needsStyleChange
= true;
3945 // Don't apply widget style, or we get a grey background
3949 // apply style change (forceStyle=true so that new style is applied
3950 // even if the bg colour changed from valid to wxNullColour):
3951 ApplyWidgetStyle(true);
3956 #if wxUSE_DRAG_AND_DROP
3958 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
3960 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3962 GtkWidget
*dnd_widget
= GetConnectWidget();
3964 if (m_dropTarget
) m_dropTarget
->UnregisterWidget( dnd_widget
);
3966 if (m_dropTarget
) delete m_dropTarget
;
3967 m_dropTarget
= dropTarget
;
3969 if (m_dropTarget
) m_dropTarget
->RegisterWidget( dnd_widget
);
3972 #endif // wxUSE_DRAG_AND_DROP
3974 GtkWidget
* wxWindowGTK::GetConnectWidget()
3976 GtkWidget
*connect_widget
= m_widget
;
3977 if (m_wxwindow
) connect_widget
= m_wxwindow
;
3979 return connect_widget
;
3982 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow
*window
) const
3984 wxArrayGdkWindows windowsThis
;
3985 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3987 return winThis
? window
== winThis
3988 : windowsThis
.Index(window
) != wxNOT_FOUND
;
3991 GdkWindow
*wxWindowGTK::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const
3993 return m_wxwindow
? GTK_PIZZA(m_wxwindow
)->bin_window
: m_widget
->window
;
3996 bool wxWindowGTK::SetFont( const wxFont
&font
)
3998 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
4000 if (!wxWindowBase::SetFont(font
))
4003 // apply style change (forceStyle=true so that new style is applied
4004 // even if the font changed from valid to wxNullFont):
4005 ApplyWidgetStyle(true);
4010 void wxWindowGTK::DoCaptureMouse()
4012 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4014 GdkWindow
*window
= (GdkWindow
*) NULL
;
4016 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4018 window
= GetConnectWidget()->window
;
4020 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4022 const wxCursor
* cursor
= &m_cursor
;
4024 cursor
= wxSTANDARD_CURSOR
;
4026 gdk_pointer_grab( window
, FALSE
,
4028 (GDK_BUTTON_PRESS_MASK
|
4029 GDK_BUTTON_RELEASE_MASK
|
4030 GDK_POINTER_MOTION_HINT_MASK
|
4031 GDK_POINTER_MOTION_MASK
),
4033 cursor
->GetCursor(),
4034 (guint32
)GDK_CURRENT_TIME
);
4035 g_captureWindow
= this;
4036 g_captureWindowHasMouse
= true;
4039 void wxWindowGTK::DoReleaseMouse()
4041 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4043 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4045 g_captureWindow
= (wxWindowGTK
*) NULL
;
4047 GdkWindow
*window
= (GdkWindow
*) NULL
;
4049 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4051 window
= GetConnectWidget()->window
;
4056 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4060 wxWindow
*wxWindowBase::GetCapture()
4062 return (wxWindow
*)g_captureWindow
;
4065 bool wxWindowGTK::IsRetained() const
4070 void wxWindowGTK::BlockScrollEvent()
4072 wxASSERT(!m_blockScrollEvent
);
4073 m_blockScrollEvent
= true;
4076 void wxWindowGTK::UnblockScrollEvent()
4078 wxASSERT(m_blockScrollEvent
);
4079 m_blockScrollEvent
= false;
4082 void wxWindowGTK::SetScrollbar(int orient
,
4086 bool WXUNUSED(update
))
4088 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4089 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4093 m_hasScrolling
= true;
4097 // GtkRange requires upper > lower
4102 if (pos
> range
- thumbVisible
)
4103 pos
= range
- thumbVisible
;
4106 GtkAdjustment
* adj
= m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
;
4107 adj
->step_increment
= 1;
4108 adj
->page_increment
=
4109 adj
->page_size
= thumbVisible
;
4111 SetScrollPos(orient
, pos
);
4112 gtk_adjustment_changed(adj
);
4115 void wxWindowGTK::SetScrollPos(int orient
, int pos
, bool WXUNUSED(refresh
))
4117 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4118 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4120 // This check is more than an optimization. Without it, the slider
4121 // will not move smoothly while tracking when using wxScrollHelper.
4122 if (GetScrollPos(orient
) != pos
)
4124 const int dir
= ScrollDirFromOrient(orient
);
4125 GtkAdjustment
* adj
= m_scrollBar
[dir
]->adjustment
;
4126 const int max
= int(adj
->upper
- adj
->page_size
);
4133 // If a "value_changed" signal emission is not already in progress
4134 if (!m_blockValueChanged
[dir
])
4136 gtk_adjustment_value_changed(adj
);
4141 int wxWindowGTK::GetScrollThumb(int orient
) const
4143 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4144 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4146 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->page_size
);
4149 int wxWindowGTK::GetScrollPos( int orient
) const
4151 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4152 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4154 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->value
+ 0.5);
4157 int wxWindowGTK::GetScrollRange( int orient
) const
4159 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4160 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4162 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->upper
);
4165 // Determine if increment is the same as +/-x, allowing for some small
4166 // difference due to possible inexactness in floating point arithmetic
4167 static inline bool IsScrollIncrement(double increment
, double x
)
4169 wxASSERT(increment
> 0);
4170 const double tolerance
= 1.0 / 1024;
4171 return fabs(increment
- fabs(x
)) < tolerance
;
4174 wxEventType
wxWindowGTK::GetScrollEventType(GtkRange
* range
)
4179 wxapp_install_idle_handler();
4181 wxASSERT(range
== m_scrollBar
[0] || range
== m_scrollBar
[1]);
4183 const int barIndex
= range
== m_scrollBar
[1];
4184 GtkAdjustment
* adj
= range
->adjustment
;
4185 const int value
= int(adj
->value
+ 0.5);
4186 // save previous position
4187 const double oldPos
= m_scrollPos
[barIndex
];
4188 // update current position
4189 m_scrollPos
[barIndex
] = adj
->value
;
4190 // If event should be ignored, or integral position has not changed
4191 if (!m_hasVMT
|| g_blockEventsOnDrag
|| value
== int(oldPos
+ 0.5))
4196 wxEventType eventType
= wxEVT_SCROLL_THUMBTRACK
;
4199 // Difference from last change event
4200 const double diff
= adj
->value
- oldPos
;
4201 const bool isDown
= diff
> 0;
4203 if (IsScrollIncrement(adj
->step_increment
, diff
))
4205 eventType
= isDown
? wxEVT_SCROLL_LINEDOWN
: wxEVT_SCROLL_LINEUP
;
4207 else if (IsScrollIncrement(adj
->page_increment
, diff
))
4209 eventType
= isDown
? wxEVT_SCROLL_PAGEDOWN
: wxEVT_SCROLL_PAGEUP
;
4211 else if (m_mouseButtonDown
)
4213 // Assume track event
4214 m_isScrolling
= true;
4220 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4222 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4224 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4226 // No scrolling requested.
4227 if ((dx
== 0) && (dy
== 0)) return;
4229 m_clipPaintRegion
= true;
4231 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), -dx
, -dy
);
4233 m_clipPaintRegion
= false;
4236 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget
* w
, int wxstyle
)
4238 //RN: Note that static controls usually have no border on gtk, so maybe
4239 //it makes sense to treat that as simply no border at the wx level
4241 if (!(wxstyle
& wxNO_BORDER
) && !(wxstyle
& wxBORDER_STATIC
))
4243 GtkShadowType gtkstyle
;
4245 if(wxstyle
& wxBORDER_RAISED
)
4246 gtkstyle
= GTK_SHADOW_OUT
;
4247 else if (wxstyle
& wxBORDER_SUNKEN
)
4248 gtkstyle
= GTK_SHADOW_IN
;
4249 else if (wxstyle
& wxBORDER_DOUBLE
)
4250 gtkstyle
= GTK_SHADOW_ETCHED_IN
;
4252 gtkstyle
= GTK_SHADOW_IN
;
4254 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w
),
4259 void wxWindowGTK::SetWindowStyleFlag( long style
)
4261 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4262 wxWindowBase::SetWindowStyleFlag(style
);
4265 // Find the wxWindow at the current mouse position, also returning the mouse
4267 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4269 pt
= wxGetMousePosition();
4270 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4274 // Get the current mouse position.
4275 wxPoint
wxGetMousePosition()
4277 /* This crashes when used within wxHelpContext,
4278 so we have to use the X-specific implementation below.
4280 GdkModifierType *mask;
4281 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4283 return wxPoint(x, y);
4287 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4289 Display
*display
= windowAtPtr
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY();
4290 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4291 Window rootReturn
, childReturn
;
4292 int rootX
, rootY
, winX
, winY
;
4293 unsigned int maskReturn
;
4295 XQueryPointer (display
,
4299 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4300 return wxPoint(rootX
, rootY
);
4304 // Needed for implementing e.g. combobox on wxGTK within a modal dialog.
4305 void wxAddGrab(wxWindow
* window
)
4307 gtk_grab_add( (GtkWidget
*) window
->GetHandle() );
4310 void wxRemoveGrab(wxWindow
* window
)
4312 gtk_grab_remove( (GtkWidget
*) window
->GetHandle() );
4315 // ----------------------------------------------------------------------------
4317 // ----------------------------------------------------------------------------
4319 class wxWinModule
: public wxModule
4326 DECLARE_DYNAMIC_CLASS(wxWinModule
)
4329 IMPLEMENT_DYNAMIC_CLASS(wxWinModule
, wxModule
)
4331 bool wxWinModule::OnInit()
4333 // g_eraseGC = gdk_gc_new( gdk_get_default_root_window() );
4334 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
4339 void wxWinModule::OnExit()
4342 g_object_unref (g_eraseGC
);