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 // Update invalidated regions.
2909 wxCursor cursor
= m_cursor
;
2910 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
2914 /* I now set the cursor anew in every OnInternalIdle call
2915 as setting the cursor in a parent window also effects the
2916 windows above so that checking for the current cursor is
2921 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
2923 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2925 if (!g_globalCursor
.Ok())
2926 cursor
= *wxSTANDARD_CURSOR
;
2928 window
= m_widget
->window
;
2929 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
2930 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2933 else if ( m_widget
)
2935 GdkWindow
*window
= m_widget
->window
;
2936 if ( window
&& !GTK_WIDGET_NO_WINDOW(m_widget
) )
2937 gdk_window_set_cursor( window
, cursor
.GetCursor() );
2941 if (wxUpdateUIEvent::CanUpdate(this))
2942 UpdateWindowUI(wxUPDATE_UI_FROMIDLE
);
2945 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
2947 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2949 if (width
) (*width
) = m_width
;
2950 if (height
) (*height
) = m_height
;
2953 void wxWindowGTK::DoSetClientSize( int width
, int height
)
2955 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
2964 GetScrollbarWidth(m_widget
, dw
, dh
);
2967 #ifndef __WXUNIVERSAL__
2968 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
2970 // shadow border size is 2
2974 if (HasFlag(wxSIMPLE_BORDER
))
2976 // simple border size is 1
2980 #endif // __WXUNIVERSAL__
2986 SetSize(width
, height
);
2989 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
2991 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3003 GetScrollbarWidth(m_widget
, dw
, dh
);
3006 #ifndef __WXUNIVERSAL__
3007 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3009 // shadow border size is 2
3013 if (HasFlag(wxSIMPLE_BORDER
))
3015 // simple border size is 1
3019 #endif // __WXUNIVERSAL__
3025 if (width
) *width
= w
;
3026 if (height
) *height
= h
;
3029 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
3031 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3035 if (m_parent
&& m_parent
->m_wxwindow
)
3037 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
3038 dx
= pizza
->xoffset
;
3039 dy
= pizza
->yoffset
;
3042 if (x
) (*x
) = m_x
- dx
;
3043 if (y
) (*y
) = m_y
- dy
;
3046 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
3048 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3050 if (!m_widget
->window
) return;
3052 GdkWindow
*source
= (GdkWindow
*) NULL
;
3054 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3056 source
= m_widget
->window
;
3060 gdk_window_get_origin( source
, &org_x
, &org_y
);
3064 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3066 org_x
+= m_widget
->allocation
.x
;
3067 org_y
+= m_widget
->allocation
.y
;
3075 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
3077 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3079 if (!m_widget
->window
) return;
3081 GdkWindow
*source
= (GdkWindow
*) NULL
;
3083 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3085 source
= m_widget
->window
;
3089 gdk_window_get_origin( source
, &org_x
, &org_y
);
3093 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3095 org_x
+= m_widget
->allocation
.x
;
3096 org_y
+= m_widget
->allocation
.y
;
3104 bool wxWindowGTK::Show( bool show
)
3106 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3108 if (!wxWindowBase::Show(show
))
3118 gtk_widget_show( m_widget
);
3119 wxShowEvent
eventShow(GetId(), show
);
3120 eventShow
.SetEventObject(this);
3121 GetEventHandler()->ProcessEvent(eventShow
);
3126 gtk_widget_hide( m_widget
);
3127 wxShowEvent
eventShow(GetId(), show
);
3128 eventShow
.SetEventObject(this);
3129 GetEventHandler()->ProcessEvent(eventShow
);
3135 static void wxWindowNotifyEnable(wxWindowGTK
* win
, bool enable
)
3137 win
->OnParentEnable(enable
);
3139 // Recurse, so that children have the opportunity to Do The Right Thing
3140 // and reset colours that have been messed up by a parent's (really ancestor's)
3142 for ( wxWindowList::compatibility_iterator node
= win
->GetChildren().GetFirst();
3144 node
= node
->GetNext() )
3146 wxWindow
*child
= node
->GetData();
3147 if (!child
->IsKindOf(CLASSINFO(wxDialog
)) && !child
->IsKindOf(CLASSINFO(wxFrame
)))
3148 wxWindowNotifyEnable(child
, enable
);
3152 bool wxWindowGTK::Enable( bool enable
)
3154 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3156 if (!wxWindowBase::Enable(enable
))
3162 gtk_widget_set_sensitive( m_widget
, enable
);
3164 gtk_widget_set_sensitive( m_wxwindow
, enable
);
3166 wxWindowNotifyEnable(this, enable
);
3171 int wxWindowGTK::GetCharHeight() const
3173 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
3175 wxFont font
= GetFont();
3176 wxCHECK_MSG( font
.Ok(), 12, wxT("invalid font") );
3178 PangoContext
*context
= NULL
;
3180 context
= gtk_widget_get_pango_context( m_widget
);
3185 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3186 PangoLayout
*layout
= pango_layout_new(context
);
3187 pango_layout_set_font_description(layout
, desc
);
3188 pango_layout_set_text(layout
, "H", 1);
3189 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3191 PangoRectangle rect
;
3192 pango_layout_line_get_extents(line
, NULL
, &rect
);
3194 g_object_unref (layout
);
3196 return (int) PANGO_PIXELS(rect
.height
);
3199 int wxWindowGTK::GetCharWidth() const
3201 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
3203 wxFont font
= GetFont();
3204 wxCHECK_MSG( font
.Ok(), 8, wxT("invalid font") );
3206 PangoContext
*context
= NULL
;
3208 context
= gtk_widget_get_pango_context( m_widget
);
3213 PangoFontDescription
*desc
= font
.GetNativeFontInfo()->description
;
3214 PangoLayout
*layout
= pango_layout_new(context
);
3215 pango_layout_set_font_description(layout
, desc
);
3216 pango_layout_set_text(layout
, "g", 1);
3217 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3219 PangoRectangle rect
;
3220 pango_layout_line_get_extents(line
, NULL
, &rect
);
3222 g_object_unref (layout
);
3224 return (int) PANGO_PIXELS(rect
.width
);
3227 void wxWindowGTK::GetTextExtent( const wxString
& string
,
3231 int *externalLeading
,
3232 const wxFont
*theFont
) const
3234 wxFont fontToUse
= theFont
? *theFont
: GetFont();
3236 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
3245 PangoContext
*context
= NULL
;
3247 context
= gtk_widget_get_pango_context( m_widget
);
3256 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
3257 PangoLayout
*layout
= pango_layout_new(context
);
3258 pango_layout_set_font_description(layout
, desc
);
3260 const wxCharBuffer data
= wxGTK_CONV( string
);
3262 pango_layout_set_text(layout
, data
, strlen(data
));
3265 PangoRectangle rect
;
3266 pango_layout_get_extents(layout
, NULL
, &rect
);
3268 if (x
) (*x
) = (wxCoord
) PANGO_PIXELS(rect
.width
);
3269 if (y
) (*y
) = (wxCoord
) PANGO_PIXELS(rect
.height
);
3272 PangoLayoutIter
*iter
= pango_layout_get_iter(layout
);
3273 int baseline
= pango_layout_iter_get_baseline(iter
);
3274 pango_layout_iter_free(iter
);
3275 *descent
= *y
- PANGO_PIXELS(baseline
);
3277 if (externalLeading
) (*externalLeading
) = 0; // ??
3279 g_object_unref (layout
);
3282 bool wxWindowGTK::GTKSetDelayedFocusIfNeeded()
3284 if ( g_delayedFocus
== this )
3286 if ( GTK_WIDGET_REALIZED(m_widget
) )
3288 gtk_widget_grab_focus(m_widget
);
3289 g_delayedFocus
= NULL
;
3298 void wxWindowGTK::SetFocus()
3300 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3303 // don't do anything if we already have focus
3309 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow
))
3311 gtk_widget_grab_focus (m_wxwindow
);
3316 if (GTK_IS_CONTAINER(m_widget
))
3318 gtk_widget_child_focus( m_widget
, GTK_DIR_TAB_FORWARD
);
3321 if (GTK_WIDGET_CAN_FOCUS(m_widget
) && !GTK_WIDGET_HAS_FOCUS (m_widget
) )
3324 if (!GTK_WIDGET_REALIZED(m_widget
))
3326 // we can't set the focus to the widget now so we remember that
3327 // it should be focused and will do it later, during the idle
3328 // time, as soon as we can
3329 wxLogTrace(TRACE_FOCUS
,
3330 _T("Delaying setting focus to %s(%s)"),
3331 GetClassInfo()->GetClassName(), GetLabel().c_str());
3333 g_delayedFocus
= this;
3337 wxLogTrace(TRACE_FOCUS
,
3338 _T("Setting focus to %s(%s)"),
3339 GetClassInfo()->GetClassName(), GetLabel().c_str());
3341 gtk_widget_grab_focus (m_widget
);
3346 wxLogTrace(TRACE_FOCUS
,
3347 _T("Can't set focus to %s(%s)"),
3348 GetClassInfo()->GetClassName(), GetLabel().c_str());
3353 bool wxWindowGTK::AcceptsFocus() const
3355 return m_acceptsFocus
&& wxWindowBase::AcceptsFocus();
3358 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3360 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3362 wxWindowGTK
*oldParent
= m_parent
,
3363 *newParent
= (wxWindowGTK
*)newParentBase
;
3365 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3367 if ( !wxWindowBase::Reparent(newParent
) )
3370 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3372 /* prevent GTK from deleting the widget arbitrarily */
3373 gtk_widget_ref( m_widget
);
3377 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3380 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3384 if (GTK_WIDGET_VISIBLE (newParent
->m_widget
))
3386 m_showOnIdle
= true;
3387 gtk_widget_hide( m_widget
);
3390 /* insert GTK representation */
3391 (*(newParent
->m_insertCallback
))(newParent
, this);
3394 /* reverse: prevent GTK from deleting the widget arbitrarily */
3395 gtk_widget_unref( m_widget
);
3400 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3402 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3404 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3406 wxASSERT_MSG( (m_insertCallback
!= NULL
), wxT("invalid child insertion function") );
3411 /* insert GTK representation */
3412 (*m_insertCallback
)(this, child
);
3415 void wxWindowGTK::AddChild(wxWindowBase
*child
)
3417 wxWindowBase::AddChild(child
);
3418 m_dirtyTabOrder
= true;
3420 wxapp_install_idle_handler();
3423 void wxWindowGTK::RemoveChild(wxWindowBase
*child
)
3425 wxWindowBase::RemoveChild(child
);
3426 m_dirtyTabOrder
= true;
3428 wxapp_install_idle_handler();
3431 void wxWindowGTK::DoMoveInTabOrder(wxWindow
*win
, MoveKind move
)
3433 wxWindowBase::DoMoveInTabOrder(win
, move
);
3434 m_dirtyTabOrder
= true;
3436 wxapp_install_idle_handler();
3439 bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3441 // none needed by default
3445 void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget
* WXUNUSED(w
))
3447 // nothing to do by default since none is needed
3450 void wxWindowGTK::RealizeTabOrder()
3454 if ( !m_children
.empty() )
3456 // we don't only construct the correct focus chain but also use
3457 // this opportunity to update the mnemonic widgets for the widgets
3460 GList
*chain
= NULL
;
3461 wxWindowGTK
* mnemonicWindow
= NULL
;
3463 for ( wxWindowList::const_iterator i
= m_children
.begin();
3464 i
!= m_children
.end();
3467 wxWindowGTK
*win
= *i
;
3469 if ( mnemonicWindow
)
3471 if ( win
->AcceptsFocusFromKeyboard() )
3473 // wxComboBox et al. needs to focus on on a different
3474 // widget than m_widget, so if the main widget isn't
3475 // focusable try the connect widget
3476 GtkWidget
* w
= win
->m_widget
;
3477 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3479 w
= win
->GetConnectWidget();
3480 if ( !GTK_WIDGET_CAN_FOCUS(w
) )
3486 mnemonicWindow
->GTKWidgetDoSetMnemonic(w
);
3487 mnemonicWindow
= NULL
;
3491 else if ( win
->GTKWidgetNeedsMnemonic() )
3493 mnemonicWindow
= win
;
3496 chain
= g_list_prepend(chain
, win
->m_widget
);
3499 chain
= g_list_reverse(chain
);
3501 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow
), chain
);
3506 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow
));
3511 void wxWindowGTK::Raise()
3513 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3515 if (m_wxwindow
&& m_wxwindow
->window
)
3517 gdk_window_raise( m_wxwindow
->window
);
3519 else if (m_widget
->window
)
3521 gdk_window_raise( m_widget
->window
);
3525 void wxWindowGTK::Lower()
3527 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3529 if (m_wxwindow
&& m_wxwindow
->window
)
3531 gdk_window_lower( m_wxwindow
->window
);
3533 else if (m_widget
->window
)
3535 gdk_window_lower( m_widget
->window
);
3539 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3541 wxCHECK_MSG( (m_widget
!= NULL
), false, wxT("invalid window") );
3543 if (cursor
== m_cursor
)
3547 wxapp_install_idle_handler();
3549 return wxWindowBase::SetCursor( cursor
.Ok() ? cursor
: *wxSTANDARD_CURSOR
);
3552 void wxWindowGTK::GTKUpdateCursor()
3554 wxCursor
cursor(g_globalCursor
.Ok() ? g_globalCursor
: GetCursor());
3557 wxArrayGdkWindows windowsThis
;
3558 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3561 gdk_window_set_cursor(winThis
, cursor
.GetCursor());
3565 const size_t count
= windowsThis
.size();
3566 for ( size_t n
= 0; n
< count
; n
++ )
3568 gdk_window_set_cursor(windowsThis
[n
], cursor
.GetCursor());
3574 void wxWindowGTK::WarpPointer( int x
, int y
)
3576 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3578 // We provide this function ourselves as it is
3579 // missing in GDK (top of this file).
3581 GdkWindow
*window
= (GdkWindow
*) NULL
;
3583 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3585 window
= GetConnectWidget()->window
;
3588 gdk_window_warp_pointer( window
, x
, y
);
3591 wxWindowGTK::ScrollDir
wxWindowGTK::ScrollDirFromRange(GtkRange
*range
) const
3593 // find the scrollbar which generated the event
3594 for ( int dir
= 0; dir
< ScrollDir_Max
; dir
++ )
3596 if ( range
== m_scrollBar
[dir
] )
3597 return (ScrollDir
)dir
;
3600 wxFAIL_MSG( _T("event from unknown scrollbar received") );
3602 return ScrollDir_Max
;
3605 bool wxWindowGTK::DoScrollByUnits(ScrollDir dir
, ScrollUnit unit
, int units
)
3607 bool changed
= false;
3608 GtkRange
* range
= m_scrollBar
[dir
];
3609 if ( range
&& units
)
3611 GtkAdjustment
* adj
= range
->adjustment
;
3612 gdouble inc
= unit
== ScrollUnit_Line
? adj
->step_increment
3613 : adj
->page_increment
;
3615 const int posOld
= int(adj
->value
+ 0.5);
3616 gtk_range_set_value(range
, posOld
+ units
*inc
);
3618 changed
= int(adj
->value
+ 0.5) != posOld
;
3624 bool wxWindowGTK::ScrollLines(int lines
)
3626 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Line
, lines
);
3629 bool wxWindowGTK::ScrollPages(int pages
)
3631 return DoScrollByUnits(ScrollDir_Vert
, ScrollUnit_Page
, pages
);
3634 void wxWindowGTK::Refresh( bool eraseBackground
, const wxRect
*rect
)
3638 if (!m_widget
->window
)
3643 if (!GTK_PIZZA(m_wxwindow
)->bin_window
) return;
3645 GdkRectangle gdk_rect
,
3649 gdk_rect
.x
= rect
->x
;
3650 gdk_rect
.y
= rect
->y
;
3651 gdk_rect
.width
= rect
->width
;
3652 gdk_rect
.height
= rect
->height
;
3655 else // invalidate everything
3660 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, p
, TRUE
);
3664 void wxWindowGTK::Update()
3668 // when we call Update() we really want to update the window immediately on
3669 // screen, even if it means flushing the entire queue and hence slowing down
3670 // everything -- but it should still be done, it's just that Update() should
3671 // be called very rarely
3675 void wxWindowGTK::GtkUpdate()
3677 if (m_wxwindow
&& GTK_PIZZA(m_wxwindow
)->bin_window
)
3678 gdk_window_process_updates( GTK_PIZZA(m_wxwindow
)->bin_window
, FALSE
);
3679 if (m_widget
&& m_widget
->window
)
3680 gdk_window_process_updates( m_widget
->window
, FALSE
);
3682 // for consistency with other platforms (and also because it's convenient
3683 // to be able to update an entire TLW by calling Update() only once), we
3684 // should also update all our children here
3685 for ( wxWindowList::compatibility_iterator node
= GetChildren().GetFirst();
3687 node
= node
->GetNext() )
3689 node
->GetData()->GtkUpdate();
3693 void wxWindowGTK::GtkSendPaintEvents()
3697 m_updateRegion
.Clear();
3701 // Clip to paint region in wxClientDC
3702 m_clipPaintRegion
= true;
3704 // widget to draw on
3705 GtkPizza
*pizza
= GTK_PIZZA (m_wxwindow
);
3707 if (GetThemeEnabled() && (GetBackgroundStyle() == wxBG_STYLE_SYSTEM
))
3709 // find ancestor from which to steal background
3710 wxWindow
*parent
= wxGetTopLevelParent((wxWindow
*)this);
3712 parent
= (wxWindow
*)this;
3714 if (GTK_WIDGET_MAPPED(parent
->m_widget
))
3716 wxRegionIterator
upd( m_updateRegion
);
3720 rect
.x
= upd
.GetX();
3721 rect
.y
= upd
.GetY();
3722 rect
.width
= upd
.GetWidth();
3723 rect
.height
= upd
.GetHeight();
3725 gtk_paint_flat_box( parent
->m_widget
->style
,
3727 (GtkStateType
)GTK_WIDGET_STATE(m_wxwindow
),
3741 wxWindowDC
dc( (wxWindow
*)this );
3742 dc
.SetClippingRegion( m_updateRegion
);
3744 wxEraseEvent
erase_event( GetId(), &dc
);
3745 erase_event
.SetEventObject( this );
3747 GetEventHandler()->ProcessEvent(erase_event
);
3750 wxNcPaintEvent
nc_paint_event( GetId() );
3751 nc_paint_event
.SetEventObject( this );
3752 GetEventHandler()->ProcessEvent( nc_paint_event
);
3754 wxPaintEvent
paint_event( GetId() );
3755 paint_event
.SetEventObject( this );
3756 GetEventHandler()->ProcessEvent( paint_event
);
3758 m_clipPaintRegion
= false;
3760 m_updateRegion
.Clear();
3763 void wxWindowGTK::SetDoubleBuffered( bool on
)
3765 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3768 gtk_widget_set_double_buffered( m_wxwindow
, on
);
3771 void wxWindowGTK::ClearBackground()
3773 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3777 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3779 wxWindowBase::DoSetToolTip(tip
);
3782 m_tooltip
->Apply( (wxWindow
*)this );
3785 void wxWindowGTK::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
3787 wxString
tmp( tip
);
3788 gtk_tooltips_set_tip( tips
, GetConnectWidget(), wxGTK_CONV(tmp
), (gchar
*) NULL
);
3790 #endif // wxUSE_TOOLTIPS
3792 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
3794 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3796 if (!wxWindowBase::SetBackgroundColour(colour
))
3801 // We need the pixel value e.g. for background clearing.
3802 m_backgroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3805 // apply style change (forceStyle=true so that new style is applied
3806 // even if the bg colour changed from valid to wxNullColour)
3807 if (GetBackgroundStyle() != wxBG_STYLE_CUSTOM
)
3808 ApplyWidgetStyle(true);
3813 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
3815 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
3817 if (!wxWindowBase::SetForegroundColour(colour
))
3824 // We need the pixel value e.g. for background clearing.
3825 m_foregroundColour
.CalcPixel(gtk_widget_get_colormap(m_widget
));
3828 // apply style change (forceStyle=true so that new style is applied
3829 // even if the bg colour changed from valid to wxNullColour):
3830 ApplyWidgetStyle(true);
3835 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
3837 return gtk_widget_get_pango_context( m_widget
);
3840 GtkRcStyle
*wxWindowGTK::CreateWidgetStyle(bool forceStyle
)
3842 // do we need to apply any changes at all?
3845 !m_foregroundColour
.Ok() && !m_backgroundColour
.Ok() )
3850 GtkRcStyle
*style
= gtk_rc_style_new();
3855 pango_font_description_copy( m_font
.GetNativeFontInfo()->description
);
3858 if ( m_foregroundColour
.Ok() )
3860 const GdkColor
*fg
= m_foregroundColour
.GetColor();
3862 style
->fg
[GTK_STATE_NORMAL
] = *fg
;
3863 style
->color_flags
[GTK_STATE_NORMAL
] = GTK_RC_FG
;
3865 style
->fg
[GTK_STATE_PRELIGHT
] = *fg
;
3866 style
->color_flags
[GTK_STATE_PRELIGHT
] = GTK_RC_FG
;
3868 style
->fg
[GTK_STATE_ACTIVE
] = *fg
;
3869 style
->color_flags
[GTK_STATE_ACTIVE
] = GTK_RC_FG
;
3872 if ( m_backgroundColour
.Ok() )
3874 const GdkColor
*bg
= m_backgroundColour
.GetColor();
3876 style
->bg
[GTK_STATE_NORMAL
] = *bg
;
3877 style
->base
[GTK_STATE_NORMAL
] = *bg
;
3878 style
->color_flags
[GTK_STATE_NORMAL
] = (GtkRcFlags
)
3879 (style
->color_flags
[GTK_STATE_NORMAL
] | GTK_RC_BG
| GTK_RC_BASE
);
3881 style
->bg
[GTK_STATE_PRELIGHT
] = *bg
;
3882 style
->base
[GTK_STATE_PRELIGHT
] = *bg
;
3883 style
->color_flags
[GTK_STATE_PRELIGHT
] = (GtkRcFlags
)
3884 (style
->color_flags
[GTK_STATE_PRELIGHT
] | GTK_RC_BG
| GTK_RC_BASE
);
3886 style
->bg
[GTK_STATE_ACTIVE
] = *bg
;
3887 style
->base
[GTK_STATE_ACTIVE
] = *bg
;
3888 style
->color_flags
[GTK_STATE_ACTIVE
] = (GtkRcFlags
)
3889 (style
->color_flags
[GTK_STATE_ACTIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
3891 style
->bg
[GTK_STATE_INSENSITIVE
] = *bg
;
3892 style
->base
[GTK_STATE_INSENSITIVE
] = *bg
;
3893 style
->color_flags
[GTK_STATE_INSENSITIVE
] = (GtkRcFlags
)
3894 (style
->color_flags
[GTK_STATE_INSENSITIVE
] | GTK_RC_BG
| GTK_RC_BASE
);
3900 void wxWindowGTK::ApplyWidgetStyle(bool forceStyle
)
3902 GtkRcStyle
*style
= CreateWidgetStyle(forceStyle
);
3905 DoApplyWidgetStyle(style
);
3906 gtk_rc_style_unref(style
);
3909 // Style change may affect GTK+'s size calculation:
3910 InvalidateBestSize();
3913 void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle
*style
)
3916 gtk_widget_modify_style(m_wxwindow
, style
);
3918 gtk_widget_modify_style(m_widget
, style
);
3921 bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style
)
3923 wxWindowBase::SetBackgroundStyle(style
);
3925 if (style
== wxBG_STYLE_CUSTOM
)
3927 GdkWindow
*window
= (GdkWindow
*) NULL
;
3929 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3931 window
= GetConnectWidget()->window
;
3935 // Make sure GDK/X11 doesn't refresh the window
3937 gdk_window_set_back_pixmap( window
, None
, False
);
3939 Display
* display
= GDK_WINDOW_DISPLAY(window
);
3942 m_needsStyleChange
= false;
3945 // Do in OnIdle, because the window is not yet available
3946 m_needsStyleChange
= true;
3948 // Don't apply widget style, or we get a grey background
3952 // apply style change (forceStyle=true so that new style is applied
3953 // even if the bg colour changed from valid to wxNullColour):
3954 ApplyWidgetStyle(true);
3959 #if wxUSE_DRAG_AND_DROP
3961 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
3963 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3965 GtkWidget
*dnd_widget
= GetConnectWidget();
3967 if (m_dropTarget
) m_dropTarget
->UnregisterWidget( dnd_widget
);
3969 if (m_dropTarget
) delete m_dropTarget
;
3970 m_dropTarget
= dropTarget
;
3972 if (m_dropTarget
) m_dropTarget
->RegisterWidget( dnd_widget
);
3975 #endif // wxUSE_DRAG_AND_DROP
3977 GtkWidget
* wxWindowGTK::GetConnectWidget()
3979 GtkWidget
*connect_widget
= m_widget
;
3980 if (m_wxwindow
) connect_widget
= m_wxwindow
;
3982 return connect_widget
;
3985 bool wxWindowGTK::GTKIsOwnWindow(GdkWindow
*window
) const
3987 wxArrayGdkWindows windowsThis
;
3988 GdkWindow
* const winThis
= GTKGetWindow(windowsThis
);
3990 return winThis
? window
== winThis
3991 : windowsThis
.Index(window
) != wxNOT_FOUND
;
3994 GdkWindow
*wxWindowGTK::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const
3996 return m_wxwindow
? GTK_PIZZA(m_wxwindow
)->bin_window
: m_widget
->window
;
3999 bool wxWindowGTK::SetFont( const wxFont
&font
)
4001 wxCHECK_MSG( m_widget
!= NULL
, false, wxT("invalid window") );
4003 if (!wxWindowBase::SetFont(font
))
4006 // apply style change (forceStyle=true so that new style is applied
4007 // even if the font changed from valid to wxNullFont):
4008 ApplyWidgetStyle(true);
4013 void wxWindowGTK::DoCaptureMouse()
4015 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4017 GdkWindow
*window
= (GdkWindow
*) NULL
;
4019 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4021 window
= GetConnectWidget()->window
;
4023 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4025 const wxCursor
* cursor
= &m_cursor
;
4027 cursor
= wxSTANDARD_CURSOR
;
4029 gdk_pointer_grab( window
, FALSE
,
4031 (GDK_BUTTON_PRESS_MASK
|
4032 GDK_BUTTON_RELEASE_MASK
|
4033 GDK_POINTER_MOTION_HINT_MASK
|
4034 GDK_POINTER_MOTION_MASK
),
4036 cursor
->GetCursor(),
4037 (guint32
)GDK_CURRENT_TIME
);
4038 g_captureWindow
= this;
4039 g_captureWindowHasMouse
= true;
4042 void wxWindowGTK::DoReleaseMouse()
4044 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4046 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4048 g_captureWindow
= (wxWindowGTK
*) NULL
;
4050 GdkWindow
*window
= (GdkWindow
*) NULL
;
4052 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4054 window
= GetConnectWidget()->window
;
4059 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4063 wxWindow
*wxWindowBase::GetCapture()
4065 return (wxWindow
*)g_captureWindow
;
4068 bool wxWindowGTK::IsRetained() const
4073 void wxWindowGTK::BlockScrollEvent()
4075 wxASSERT(!m_blockScrollEvent
);
4076 m_blockScrollEvent
= true;
4079 void wxWindowGTK::UnblockScrollEvent()
4081 wxASSERT(m_blockScrollEvent
);
4082 m_blockScrollEvent
= false;
4085 void wxWindowGTK::SetScrollbar(int orient
,
4089 bool WXUNUSED(update
))
4091 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4092 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4096 m_hasScrolling
= true;
4100 // GtkRange requires upper > lower
4105 if (pos
> range
- thumbVisible
)
4106 pos
= range
- thumbVisible
;
4109 GtkAdjustment
* adj
= m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
;
4110 adj
->step_increment
= 1;
4111 adj
->page_increment
=
4112 adj
->page_size
= thumbVisible
;
4114 SetScrollPos(orient
, pos
);
4115 gtk_adjustment_changed(adj
);
4118 void wxWindowGTK::SetScrollPos(int orient
, int pos
, bool WXUNUSED(refresh
))
4120 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4121 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4123 // This check is more than an optimization. Without it, the slider
4124 // will not move smoothly while tracking when using wxScrollHelper.
4125 if (GetScrollPos(orient
) != pos
)
4127 const int dir
= ScrollDirFromOrient(orient
);
4128 GtkAdjustment
* adj
= m_scrollBar
[dir
]->adjustment
;
4129 const int max
= int(adj
->upper
- adj
->page_size
);
4136 // If a "value_changed" signal emission is not already in progress
4137 if (!m_blockValueChanged
[dir
])
4139 gtk_adjustment_value_changed(adj
);
4144 int wxWindowGTK::GetScrollThumb(int orient
) const
4146 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4147 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4149 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->page_size
);
4152 int wxWindowGTK::GetScrollPos( int orient
) const
4154 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4155 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4157 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->value
+ 0.5);
4160 int wxWindowGTK::GetScrollRange( int orient
) const
4162 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4163 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4165 return int(m_scrollBar
[ScrollDirFromOrient(orient
)]->adjustment
->upper
);
4168 // Determine if increment is the same as +/-x, allowing for some small
4169 // difference due to possible inexactness in floating point arithmetic
4170 static inline bool IsScrollIncrement(double increment
, double x
)
4172 wxASSERT(increment
> 0);
4173 const double tolerance
= 1.0 / 1024;
4174 return fabs(increment
- fabs(x
)) < tolerance
;
4177 wxEventType
wxWindowGTK::GetScrollEventType(GtkRange
* range
)
4182 wxapp_install_idle_handler();
4184 wxASSERT(range
== m_scrollBar
[0] || range
== m_scrollBar
[1]);
4186 const int barIndex
= range
== m_scrollBar
[1];
4187 GtkAdjustment
* adj
= range
->adjustment
;
4188 const int value
= int(adj
->value
+ 0.5);
4189 // save previous position
4190 const double oldPos
= m_scrollPos
[barIndex
];
4191 // update current position
4192 m_scrollPos
[barIndex
] = adj
->value
;
4193 // If event should be ignored, or integral position has not changed
4194 if (!m_hasVMT
|| g_blockEventsOnDrag
|| value
== int(oldPos
+ 0.5))
4199 wxEventType eventType
= wxEVT_SCROLL_THUMBTRACK
;
4202 // Difference from last change event
4203 const double diff
= adj
->value
- oldPos
;
4204 const bool isDown
= diff
> 0;
4206 if (IsScrollIncrement(adj
->step_increment
, diff
))
4208 eventType
= isDown
? wxEVT_SCROLL_LINEDOWN
: wxEVT_SCROLL_LINEUP
;
4210 else if (IsScrollIncrement(adj
->page_increment
, diff
))
4212 eventType
= isDown
? wxEVT_SCROLL_PAGEDOWN
: wxEVT_SCROLL_PAGEUP
;
4214 else if (m_mouseButtonDown
)
4216 // Assume track event
4217 m_isScrolling
= true;
4223 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4225 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4227 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4229 // No scrolling requested.
4230 if ((dx
== 0) && (dy
== 0)) return;
4232 m_clipPaintRegion
= true;
4234 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), -dx
, -dy
);
4236 m_clipPaintRegion
= false;
4239 void wxWindowGTK::GtkScrolledWindowSetBorder(GtkWidget
* w
, int wxstyle
)
4241 //RN: Note that static controls usually have no border on gtk, so maybe
4242 //it makes sense to treat that as simply no border at the wx level
4244 if (!(wxstyle
& wxNO_BORDER
) && !(wxstyle
& wxBORDER_STATIC
))
4246 GtkShadowType gtkstyle
;
4248 if(wxstyle
& wxBORDER_RAISED
)
4249 gtkstyle
= GTK_SHADOW_OUT
;
4250 else if (wxstyle
& wxBORDER_SUNKEN
)
4251 gtkstyle
= GTK_SHADOW_IN
;
4252 else if (wxstyle
& wxBORDER_DOUBLE
)
4253 gtkstyle
= GTK_SHADOW_ETCHED_IN
;
4255 gtkstyle
= GTK_SHADOW_IN
;
4257 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w
),
4262 void wxWindowGTK::SetWindowStyleFlag( long style
)
4264 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4265 wxWindowBase::SetWindowStyleFlag(style
);
4268 // Find the wxWindow at the current mouse position, also returning the mouse
4270 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4272 pt
= wxGetMousePosition();
4273 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4277 // Get the current mouse position.
4278 wxPoint
wxGetMousePosition()
4280 /* This crashes when used within wxHelpContext,
4281 so we have to use the X-specific implementation below.
4283 GdkModifierType *mask;
4284 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4286 return wxPoint(x, y);
4290 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4292 Display
*display
= windowAtPtr
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY();
4293 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4294 Window rootReturn
, childReturn
;
4295 int rootX
, rootY
, winX
, winY
;
4296 unsigned int maskReturn
;
4298 XQueryPointer (display
,
4302 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4303 return wxPoint(rootX
, rootY
);
4307 // Needed for implementing e.g. combobox on wxGTK within a modal dialog.
4308 void wxAddGrab(wxWindow
* window
)
4310 gtk_grab_add( (GtkWidget
*) window
->GetHandle() );
4313 void wxRemoveGrab(wxWindow
* window
)
4315 gtk_grab_remove( (GtkWidget
*) window
->GetHandle() );
4318 // ----------------------------------------------------------------------------
4320 // ----------------------------------------------------------------------------
4322 class wxWinModule
: public wxModule
4329 DECLARE_DYNAMIC_CLASS(wxWinModule
)
4332 IMPLEMENT_DYNAMIC_CLASS(wxWinModule
, wxModule
)
4334 bool wxWinModule::OnInit()
4336 // g_eraseGC = gdk_gc_new( gdk_get_default_root_window() );
4337 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
4342 void wxWinModule::OnExit()
4345 g_object_unref (g_eraseGC
);