1 /////////////////////////////////////////////////////////////////////////////
2 // Name: gtk/window.cpp
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling, Julian Smart
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
12 #pragma implementation "window.h"
16 #define XWarpPointer XWARPPOINTER
20 #include "wx/window.h"
21 #include "wx/dcclient.h"
24 #include "wx/layout.h"
26 #include "wx/dialog.h"
27 #include "wx/msgdlg.h"
28 #include "wx/module.h"
30 #if wxUSE_DRAG_AND_DROP
35 #include "wx/tooltip.h"
43 #include "wx/textctrl.h"
47 #include "wx/statusbr.h"
49 #include "wx/settings.h"
51 #include "wx/fontutil.h"
54 #include "wx/thread.h"
60 #include "wx/gtk/private.h"
61 #include <gdk/gdkprivate.h>
62 #include <gdk/gdkkeysyms.h>
66 #include <gtk/gtkprivate.h>
68 #include "wx/gtk/win_gtk.h"
71 #include <pango/pangox.h>
75 #define SET_CONTAINER_FOCUS(w, d) gtk_widget_child_focus((w), (d))
77 #define SET_CONTAINER_FOCUS(w, d) gtk_container_focus(GTK_CONTAINER(w), (d))
87 extern GtkContainerClass
*pizza_parent_class
;
90 //-----------------------------------------------------------------------------
91 // documentation on internals
92 //-----------------------------------------------------------------------------
95 I have been asked several times about writing some documentation about
96 the GTK port of wxWindows, especially its internal structures. Obviously,
97 you cannot understand wxGTK without knowing a little about the GTK, but
98 some more information about what the wxWindow, which is the base class
99 for all other window classes, does seems required as well.
103 What does wxWindow do? It contains the common interface for the following
104 jobs of its descendants:
106 1) Define the rudimentary behaviour common to all window classes, such as
107 resizing, intercepting user input (so as to make it possible to use these
108 events for special purposes in a derived class), window names etc.
110 2) Provide the possibility to contain and manage children, if the derived
111 class is allowed to contain children, which holds true for those window
112 classes which do not display a native GTK widget. To name them, these
113 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
114 work classes are a special case and are handled a bit differently from
115 the rest. The same holds true for the wxNotebook class.
117 3) Provide the possibility to draw into a client area of a window. This,
118 too, only holds true for classes that do not display a native GTK widget
121 4) Provide the entire mechanism for scrolling widgets. This actual inter-
122 face for this is usually in wxScrolledWindow, but the GTK implementation
125 5) A multitude of helper or extra methods for special purposes, such as
126 Drag'n'Drop, managing validators etc.
128 6) Display a border (sunken, raised, simple or none).
130 Normally one might expect, that one wxWindows window would always correspond
131 to one GTK widget. Under GTK, there is no such allround widget that has all
132 the functionality. Moreover, the GTK defines a client area as a different
133 widget from the actual widget you are handling. Last but not least some
134 special classes (e.g. wxFrame) handle different categories of widgets and
135 still have the possibility to draw something in the client area.
136 It was therefore required to write a special purpose GTK widget, that would
137 represent a client area in the sense of wxWindows capable to do the jobs
138 2), 3) and 4). I have written this class and it resides in win_gtk.c of
141 All windows must have a widget, with which they interact with other under-
142 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
143 thw wxWindow class has a member variable called m_widget which holds a
144 pointer to this widget. When the window class represents a GTK native widget,
145 this is (in most cases) the only GTK widget the class manages. E.g. the
146 wxStatitText class handles only a GtkLabel widget a pointer to which you
147 can find in m_widget (defined in wxWindow)
149 When the class has a client area for drawing into and for containing children
150 it has to handle the client area widget (of the type GtkPizza, defined in
151 win_gtk.c), but there could be any number of widgets, handled by a class
152 The common rule for all windows is only, that the widget that interacts with
153 the rest of GTK must be referenced in m_widget and all other widgets must be
154 children of this widget on the GTK level. The top-most widget, which also
155 represents the client area, must be in the m_wxwindow field and must be of
158 As I said, the window classes that display a GTK native widget only have
159 one widget, so in the case of e.g. the wxButton class m_widget holds a
160 pointer to a GtkButton widget. But windows with client areas (for drawing
161 and children) have a m_widget field that is a pointer to a GtkScrolled-
162 Window and a m_wxwindow field that is pointer to a GtkPizza and this
163 one is (in the GTK sense) a child of the GtkScrolledWindow.
165 If the m_wxwindow field is set, then all input to this widget is inter-
166 cepted and sent to the wxWindows class. If not, all input to the widget
167 that gets pointed to by m_widget gets intercepted and sent to the class.
171 The design of scrolling in wxWindows is markedly different from that offered
172 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
173 clicking on a scrollbar belonging to scrolled window will inevitably move
174 the window. In wxWindows, the scrollbar will only emit an event, send this
175 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
176 which actually moves the window and its subchildren. Note that GtkPizza
177 memorizes how much it has been scrolled but that wxWindows forgets this
178 so that the two coordinates systems have to be kept in synch. This is done
179 in various places using the pizza->xoffset and pizza->yoffset values.
183 Singularily the most broken code in GTK is the code that is supposes to
184 inform subwindows (child windows) about new positions. Very often, duplicate
185 events are sent without changes in size or position, equally often no
186 events are sent at all (All this is due to a bug in the GtkContainer code
187 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
188 GTK's own system and it simply waits for size events for toplevel windows
189 and then iterates down the respective size events to all window. This has
190 the disadvantage, that windows might get size events before the GTK widget
191 actually has the reported size. This doesn't normally pose any problem, but
192 the OpenGl drawing routines rely on correct behaviour. Therefore, I have
193 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
194 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
195 window that is used for OpenGl output really has that size (as reported by
200 If someone at some point of time feels the immense desire to have a look at,
201 change or attempt to optimse the Refresh() logic, this person will need an
202 intimate understanding of what a "draw" and what an "expose" events are and
203 what there are used for, in particular when used in connection with GTK's
204 own windowless widgets. Beware.
208 Cursors, too, have been a constant source of pleasure. The main difficulty
209 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
210 for the parent. To prevent this from doing too much harm, I use idle time
211 to set the cursor over and over again, starting from the toplevel windows
212 and ending with the youngest generation (speaking of parent and child windows).
213 Also don't forget that cursors (like much else) are connected to GdkWindows,
214 not GtkWidgets and that the "window" field of a GtkWidget might very well
215 point to the GdkWindow of the parent widget (-> "window less widget") and
216 that the two obviously have very different meanings.
220 //-----------------------------------------------------------------------------
222 //-----------------------------------------------------------------------------
224 extern wxList wxPendingDelete
;
225 extern bool g_blockEventsOnDrag
;
226 extern bool g_blockEventsOnScroll
;
227 extern wxCursor g_globalCursor
;
229 static GdkGC
*g_eraseGC
= NULL
;
231 // mouse capture state: the window which has it and if the mouse is currently
233 static wxWindowGTK
*g_captureWindow
= (wxWindowGTK
*) NULL
;
234 static bool g_captureWindowHasMouse
= FALSE
;
236 /* extern */ wxWindowGTK
*g_focusWindow
= (wxWindowGTK
*) NULL
;
238 // the last window which had the focus - this is normally never NULL (except
239 // if we never had focus at all) as even when g_focusWindow is NULL it still
240 // keeps its previous value
241 static wxWindowGTK
*g_focusWindowLast
= (wxWindowGTK
*) NULL
;
243 // the frame that is currently active (i.e. its child has focus). It is
244 // used to generate wxActivateEvents
245 static wxWindowGTK
*g_activeFrame
= (wxWindowGTK
*) NULL
;
246 static bool g_activeFrameLostFocus
= FALSE
;
248 // If a window get the focus set but has not been realized
249 // yet, defer setting the focus to idle time.
250 wxWindowGTK
*g_delayedFocus
= (wxWindowGTK
*) NULL
;
252 // if we detect that the app has got/lost the focus, we set this variable to
253 // either TRUE or FALSE and an activate event will be sent during the next
254 // OnIdle() call and it is reset to -1: this value means that we shouldn't
255 // send any activate events at all
256 static int g_sendActivateEvent
= -1;
258 /* hack: we need something to pass to gtk_menu_popup, so we store the time of
259 the last click here */
260 static guint32 gs_timeLastClick
= 0;
262 extern bool g_mainThreadLocked
;
264 //-----------------------------------------------------------------------------
266 //-----------------------------------------------------------------------------
269 #define DISABLE_STYLE_IF_BROKEN_THEME 0
275 # define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance");
277 # define DEBUG_MAIN_THREAD
280 #define DEBUG_MAIN_THREAD
283 // the trace mask used for the focus debugging messages
284 #define TRACE_FOCUS _T("focus")
286 //-----------------------------------------------------------------------------
287 // missing gdk functions
288 //-----------------------------------------------------------------------------
291 gdk_window_warp_pointer (GdkWindow
*window
,
296 GdkWindowPrivate
*priv
;
300 window
= GDK_ROOT_PARENT();
303 if (!GDK_WINDOW_DESTROYED(window
))
305 XWarpPointer (GDK_WINDOW_XDISPLAY(window
),
306 None
, /* not source window -> move from anywhere */
307 GDK_WINDOW_XID(window
), /* dest window */
308 0, 0, 0, 0, /* not source window -> move from anywhere */
312 priv
= (GdkWindowPrivate
*) window
;
314 if (!priv
->destroyed
)
316 XWarpPointer (priv
->xdisplay
,
317 None
, /* not source window -> move from anywhere */
318 priv
->xwindow
, /* dest window */
319 0, 0, 0, 0, /* not source window -> move from anywhere */
325 //-----------------------------------------------------------------------------
327 //-----------------------------------------------------------------------------
329 extern void wxapp_install_idle_handler();
330 extern bool g_isIdle
;
332 //-----------------------------------------------------------------------------
333 // local code (see below)
334 //-----------------------------------------------------------------------------
336 // returns the child of win which currently has focus or NULL if not found
338 // Note: can't be static, needed by textctrl.cpp.
339 wxWindow
*wxFindFocusedChild(wxWindowGTK
*win
)
341 wxWindow
*winFocus
= wxWindowGTK::FindFocus();
343 return (wxWindow
*)NULL
;
345 if ( winFocus
== win
)
346 return (wxWindow
*)win
;
348 for ( wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
350 node
= node
->GetNext() )
352 wxWindow
*child
= wxFindFocusedChild(node
->GetData());
357 return (wxWindow
*)NULL
;
360 static void draw_frame( GtkWidget
*widget
, wxWindowGTK
*win
)
362 // wxUniversal widgets draw the borders and scrollbars themselves
363 #ifndef __WXUNIVERSAL__
370 if (win
->m_hasScrolling
)
372 GtkScrolledWindow
*scroll_window
= GTK_SCROLLED_WINDOW(widget
);
374 GtkRequisition vscroll_req
;
375 vscroll_req
.width
= 2;
376 vscroll_req
.height
= 2;
377 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
378 (scroll_window
->vscrollbar
, &vscroll_req
);
380 GtkRequisition hscroll_req
;
381 hscroll_req
.width
= 2;
382 hscroll_req
.height
= 2;
383 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
384 (scroll_window
->hscrollbar
, &hscroll_req
);
386 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(widget
) );
388 if (scroll_window
->vscrollbar_visible
)
390 dw
+= vscroll_req
.width
;
391 dw
+= scroll_class
->scrollbar_spacing
;
394 if (scroll_window
->hscrollbar_visible
)
396 dh
+= hscroll_req
.height
;
397 dh
+= scroll_class
->scrollbar_spacing
;
403 if (GTK_WIDGET_NO_WINDOW (widget
))
405 dx
+= widget
->allocation
.x
;
406 dy
+= widget
->allocation
.y
;
409 if (win
->HasFlag(wxRAISED_BORDER
))
411 gtk_draw_shadow( widget
->style
,
416 widget
->allocation
.width
-dw
, widget
->allocation
.height
-dh
);
420 if (win
->HasFlag(wxSUNKEN_BORDER
))
422 gtk_draw_shadow( widget
->style
,
427 widget
->allocation
.width
-dw
, widget
->allocation
.height
-dh
);
431 if (win
->HasFlag(wxSIMPLE_BORDER
))
434 gc
= gdk_gc_new( widget
->window
);
435 gdk_gc_set_foreground( gc
, &widget
->style
->black
);
436 gdk_draw_rectangle( widget
->window
, gc
, FALSE
,
438 widget
->allocation
.width
-dw
-1, widget
->allocation
.height
-dh
-1 );
442 #endif // __WXUNIVERSAL__
445 //-----------------------------------------------------------------------------
446 // "expose_event" of m_widget
447 //-----------------------------------------------------------------------------
449 gint
gtk_window_own_expose_callback( GtkWidget
*widget
, GdkEventExpose
*gdk_event
, wxWindowGTK
*win
)
451 if (gdk_event
->count
> 0) return FALSE
;
453 draw_frame( widget
, win
);
457 (* GTK_WIDGET_CLASS (pizza_parent_class
)->expose_event
) (widget
, gdk_event
);
463 //-----------------------------------------------------------------------------
464 // "draw" of m_widget
465 //-----------------------------------------------------------------------------
469 static void gtk_window_own_draw_callback( GtkWidget
*widget
, GdkRectangle
*WXUNUSED(rect
), wxWindowGTK
*win
)
471 draw_frame( widget
, win
);
476 //-----------------------------------------------------------------------------
477 // "size_request" of m_widget
478 //-----------------------------------------------------------------------------
480 static void gtk_window_size_request_callback( GtkWidget
*widget
, GtkRequisition
*requisition
, wxWindow
*win
)
483 win
->GetSize( &w
, &h
);
487 requisition
->height
= h
;
488 requisition
->width
= w
;
491 //-----------------------------------------------------------------------------
492 // "expose_event" of m_wxwindow
493 //-----------------------------------------------------------------------------
495 static int gtk_window_expose_callback( GtkWidget
*widget
,
496 GdkEventExpose
*gdk_event
,
502 wxapp_install_idle_handler();
505 // This callback gets called in drawing-idle time under
506 // GTK 2.0, so we don't need to defer anything to idle
509 GtkPizza
*pizza
= GTK_PIZZA( widget
);
510 if (gdk_event
->window
!= pizza
->bin_window
) return FALSE
;
515 wxPrintf( wxT("OnExpose from ") );
516 if (win
->GetClassInfo() && win
->GetClassInfo()->GetClassName())
517 wxPrintf( win
->GetClassInfo()->GetClassName() );
518 wxPrintf( wxT(" %d %d %d %d\n"), (int)gdk_event
->area
.x
,
519 (int)gdk_event
->area
.y
,
520 (int)gdk_event
->area
.width
,
521 (int)gdk_event
->area
.height
);
525 win
->GetUpdateRegion() = wxRegion( gdk_event
->region
);
527 win
->GtkSendPaintEvents();
530 // Let parent window draw window less widgets
531 (* GTK_WIDGET_CLASS (pizza_parent_class
)->expose_event
) (widget
, gdk_event
);
533 // This gets called immediately after an expose event
534 // under GTK 1.2 so we collect the calls and wait for
535 // the idle handler to pick things up.
537 win
->GetUpdateRegion().Union( gdk_event
->area
.x
,
539 gdk_event
->area
.width
,
540 gdk_event
->area
.height
);
541 win
->m_clearRegion
.Union( gdk_event
->area
.x
,
543 gdk_event
->area
.width
,
544 gdk_event
->area
.height
);
546 // Actual redrawing takes place in idle time.
553 //-----------------------------------------------------------------------------
554 // "event" of m_wxwindow
555 //-----------------------------------------------------------------------------
557 // GTK thinks it is clever and filters out a certain amount of "unneeded"
558 // expose events. We need them, of course, so we override the main event
559 // procedure in GtkWidget by giving our own handler for all system events.
560 // There, we look for expose events ourselves whereas all other events are
563 gint
gtk_window_event_event_callback( GtkWidget
*widget
,
564 GdkEventExpose
*event
,
567 if (event
->type
== GDK_EXPOSE
)
569 gint ret
= gtk_window_expose_callback( widget
, event
, win
);
576 //-----------------------------------------------------------------------------
577 // "draw" of m_wxwindow
578 //-----------------------------------------------------------------------------
582 // This callback is a complete replacement of the gtk_pizza_draw() function,
583 // which is disabled.
585 static void gtk_window_draw_callback( GtkWidget
*widget
,
592 wxapp_install_idle_handler();
594 // The wxNO_FULL_REPAINT_ON_RESIZE flag only works if
595 // there are no child windows.
596 if ((win
->HasFlag(wxNO_FULL_REPAINT_ON_RESIZE
)) &&
597 (win
->GetChildren().GetCount() == 0))
605 wxPrintf( wxT("OnDraw from ") );
606 if (win
->GetClassInfo() && win
->GetClassInfo()->GetClassName())
607 wxPrintf( win
->GetClassInfo()->GetClassName() );
608 wxPrintf( wxT(" %d %d %d %d\n"), (int)rect
->x
,
615 #ifndef __WXUNIVERSAL__
616 GtkPizza
*pizza
= GTK_PIZZA (widget
);
618 if (win
->GetThemeEnabled())
620 wxWindow
*parent
= win
->GetParent();
621 while (parent
&& !parent
->IsTopLevel())
622 parent
= parent
->GetParent();
626 gtk_paint_flat_box (parent
->m_widget
->style
,
637 win
->m_clearRegion
.Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
638 win
->GetUpdateRegion().Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
640 // Update immediately, not in idle time.
643 #ifndef __WXUNIVERSAL__
644 // Redraw child widgets
645 GList
*children
= pizza
->children
;
648 GtkPizzaChild
*child
= (GtkPizzaChild
*) children
->data
;
649 children
= children
->next
;
651 GdkRectangle child_area
;
652 if (gtk_widget_intersect (child
->widget
, rect
, &child_area
))
654 gtk_widget_draw (child
->widget
, &child_area
/* (GdkRectangle*) NULL*/ );
662 //-----------------------------------------------------------------------------
663 // "key_press_event" from any window
664 //-----------------------------------------------------------------------------
666 // set WXTRACE to this to see the key event codes on the console
667 #define TRACE_KEYS _T("keyevent")
669 // translates an X key symbol to WXK_XXX value
671 // if isChar is true it means that the value returned will be used for EVT_CHAR
672 // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
673 // for example, while if it is false it means that the value is going to be
674 // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
676 static long wxTranslateKeySymToWXKey(KeySym keysym
, bool isChar
)
682 // Shift, Control and Alt don't generate the CHAR events at all
685 key_code
= isChar
? 0 : WXK_SHIFT
;
689 key_code
= isChar
? 0 : WXK_CONTROL
;
697 key_code
= isChar
? 0 : WXK_ALT
;
700 // neither do the toggle modifies
701 case GDK_Scroll_Lock
:
702 key_code
= isChar
? 0 : WXK_SCROLL
;
706 key_code
= isChar
? 0 : WXK_CAPITAL
;
710 key_code
= isChar
? 0 : WXK_NUMLOCK
;
714 // various other special keys
727 case GDK_ISO_Left_Tab
:
734 key_code
= WXK_RETURN
;
738 key_code
= WXK_CLEAR
;
742 key_code
= WXK_PAUSE
;
746 key_code
= WXK_SELECT
;
750 key_code
= WXK_PRINT
;
754 key_code
= WXK_EXECUTE
;
758 key_code
= WXK_ESCAPE
;
761 // cursor and other extended keyboard keys
763 key_code
= WXK_DELETE
;
779 key_code
= WXK_RIGHT
;
786 case GDK_Prior
: // == GDK_Page_Up
787 key_code
= WXK_PRIOR
;
790 case GDK_Next
: // == GDK_Page_Down
803 key_code
= WXK_INSERT
;
818 key_code
= (isChar
? '0' : WXK_NUMPAD0
) + keysym
- GDK_KP_0
;
822 key_code
= isChar
? ' ' : WXK_NUMPAD_SPACE
;
826 key_code
= isChar
? WXK_TAB
: WXK_NUMPAD_TAB
;
830 key_code
= isChar
? WXK_RETURN
: WXK_NUMPAD_ENTER
;
834 key_code
= isChar
? WXK_F1
: WXK_NUMPAD_F1
;
838 key_code
= isChar
? WXK_F2
: WXK_NUMPAD_F2
;
842 key_code
= isChar
? WXK_F3
: WXK_NUMPAD_F3
;
846 key_code
= isChar
? WXK_F4
: WXK_NUMPAD_F4
;
850 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_HOME
;
854 key_code
= isChar
? WXK_LEFT
: WXK_NUMPAD_LEFT
;
858 key_code
= isChar
? WXK_UP
: WXK_NUMPAD_UP
;
862 key_code
= isChar
? WXK_RIGHT
: WXK_NUMPAD_RIGHT
;
866 key_code
= isChar
? WXK_DOWN
: WXK_NUMPAD_DOWN
;
869 case GDK_KP_Prior
: // == GDK_KP_Page_Up
870 key_code
= isChar
? WXK_PRIOR
: WXK_NUMPAD_PRIOR
;
873 case GDK_KP_Next
: // == GDK_KP_Page_Down
874 key_code
= isChar
? WXK_NEXT
: WXK_NUMPAD_NEXT
;
878 key_code
= isChar
? WXK_END
: WXK_NUMPAD_END
;
882 key_code
= isChar
? WXK_HOME
: WXK_NUMPAD_BEGIN
;
886 key_code
= isChar
? WXK_INSERT
: WXK_NUMPAD_INSERT
;
890 key_code
= isChar
? WXK_DELETE
: WXK_NUMPAD_DELETE
;
894 key_code
= isChar
? '=' : WXK_NUMPAD_EQUAL
;
897 case GDK_KP_Multiply
:
898 key_code
= isChar
? '*' : WXK_NUMPAD_MULTIPLY
;
902 key_code
= isChar
? '+' : WXK_NUMPAD_ADD
;
905 case GDK_KP_Separator
:
906 // FIXME: what is this?
907 key_code
= isChar
? '.' : WXK_NUMPAD_SEPARATOR
;
910 case GDK_KP_Subtract
:
911 key_code
= isChar
? '-' : WXK_NUMPAD_SUBTRACT
;
915 key_code
= isChar
? '.' : WXK_NUMPAD_DECIMAL
;
919 key_code
= isChar
? '/' : WXK_NUMPAD_DIVIDE
;
936 key_code
= WXK_F1
+ keysym
- GDK_F1
;
946 static inline bool wxIsAsciiKeysym(KeySym ks
)
952 wxTranslateGTKKeyEventToWx(wxKeyEvent
& event
,
954 GdkEventKey
*gdk_event
)
956 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
957 // but only event->keyval which is quite useless to us, so remember
958 // the last character from GDK_KEY_PRESS and reuse it as last resort
960 // NB: should be MT-safe as we're always called from the main thread only
965 } s_lastKeyPress
= { 0, 0 };
967 KeySym keysym
= gdk_event
->keyval
;
969 wxLogTrace(TRACE_KEYS
, _T("Key %s event: keysym = %ld"),
970 event
.GetEventType() == wxEVT_KEY_UP
? _T("release")
974 long key_code
= wxTranslateKeySymToWXKey(keysym
, FALSE
/* !isChar */);
978 // do we have the translation or is it a plain ASCII character?
979 if ( (gdk_event
->length
== 1) || wxIsAsciiKeysym(keysym
) )
981 // we should use keysym if it is ASCII as X does some translations
982 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
983 // which we don't want here (but which we do use for OnChar())
984 if ( !wxIsAsciiKeysym(keysym
) )
986 keysym
= (KeySym
)gdk_event
->string
[0];
989 // we want to always get the same key code when the same key is
990 // pressed regardless of the state of the modifies, i.e. on a
991 // standard US keyboard pressing '5' or '%' ('5' key with
992 // Shift) should result in the same key code in OnKeyDown():
993 // '5' (although OnChar() will get either '5' or '%').
995 // to do it we first translate keysym to keycode (== scan code)
996 // and then back but always using the lower register
997 Display
*dpy
= (Display
*)wxGetDisplay();
998 KeyCode keycode
= XKeysymToKeycode(dpy
, keysym
);
1000 wxLogTrace(TRACE_KEYS
, _T("\t-> keycode %d"), keycode
);
1002 KeySym keysymNormalized
= XKeycodeToKeysym(dpy
, keycode
, 0);
1004 // use the normalized, i.e. lower register, keysym if we've
1006 key_code
= keysymNormalized
? keysymNormalized
: keysym
;
1008 // as explained above, we want to have lower register key codes
1009 // normally but for the letter keys we want to have the upper ones
1011 // NB: don't use XConvertCase() here, we want to do it for letters
1013 key_code
= toupper(key_code
);
1015 else // non ASCII key, what to do?
1017 // by default, ignore it
1020 // but if we have cached information from the last KEY_PRESS
1021 if ( gdk_event
->type
== GDK_KEY_RELEASE
)
1024 if ( keysym
== s_lastKeyPress
.keysym
)
1026 key_code
= s_lastKeyPress
.keycode
;
1031 if ( gdk_event
->type
== GDK_KEY_PRESS
)
1033 // remember it to be reused for KEY_UP event later
1034 s_lastKeyPress
.keysym
= keysym
;
1035 s_lastKeyPress
.keycode
= key_code
;
1039 wxLogTrace(TRACE_KEYS
, _T("\t-> wxKeyCode %ld"), key_code
);
1041 // sending unknown key events doesn't really make sense
1045 // now fill all the other fields
1048 GdkModifierType state
;
1049 if (gdk_event
->window
)
1050 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1052 event
.SetTimestamp( gdk_event
->time
);
1053 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
) != 0;
1054 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
) != 0;
1055 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
) != 0;
1056 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
) != 0;
1057 event
.m_keyCode
= key_code
;
1058 event
.m_scanCode
= gdk_event
->keyval
;
1059 event
.m_rawCode
= (wxUint32
) gdk_event
->keyval
;
1060 event
.m_rawFlags
= 0;
1063 event
.SetEventObject( win
);
1069 static gint
gtk_window_key_press_callback( GtkWidget
*widget
,
1070 GdkEventKey
*gdk_event
,
1076 wxapp_install_idle_handler();
1080 if (g_blockEventsOnDrag
)
1084 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1085 if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1087 // unknown key pressed, ignore (the event would be useless anyhow)
1091 // Emit KEY_DOWN event
1092 bool ret
= win
->GetEventHandler()->ProcessEvent( event
);
1097 wxWindowGTK
*ancestor
= win
;
1100 int command
= ancestor
->GetAcceleratorTable()->GetCommand( event
);
1103 wxCommandEvent
command_event( wxEVT_COMMAND_MENU_SELECTED
, command
);
1104 ret
= ancestor
->GetEventHandler()->ProcessEvent( command_event
);
1107 if (ancestor
->IsTopLevel())
1109 ancestor
= ancestor
->GetParent();
1112 #endif // wxUSE_ACCEL
1114 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
1115 // will only be sent if it is not in an accelerator table.
1118 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
1119 KeySym keysym
= gdk_event
->keyval
;
1120 long key_code
= wxTranslateKeySymToWXKey(keysym
, TRUE
/* isChar */);
1123 if ( gdk_event
->length
== 1 )
1125 key_code
= (unsigned char)gdk_event
->string
[0];
1127 else if ( wxIsAsciiKeysym(keysym
) )
1130 key_code
= (unsigned char)keysym
;
1136 wxLogTrace(TRACE_KEYS
, _T("Char event: %ld"), key_code
);
1138 event
.m_keyCode
= key_code
;
1140 // Implement OnCharHook by checking ancesteror top level windows
1141 wxWindow
*parent
= win
;
1142 while (parent
&& !parent
->IsTopLevel())
1143 parent
= parent
->GetParent();
1146 event
.SetEventType( wxEVT_CHAR_HOOK
);
1147 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1152 event
.SetEventType(wxEVT_CHAR
);
1153 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1158 // win is a control: tab can be propagated up
1160 ((gdk_event
->keyval
== GDK_Tab
) || (gdk_event
->keyval
== GDK_ISO_Left_Tab
)) &&
1161 // VZ: testing for wxTE_PROCESS_TAB shouldn't be done here the control may
1162 // have this style, yet choose not to process this particular TAB in which
1163 // case TAB must still work as a navigational character
1165 !win
->HasFlag(wxTE_PROCESS_TAB
) &&
1167 win
->GetParent() && (win
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) )
1169 wxNavigationKeyEvent new_event
;
1170 new_event
.SetEventObject( win
->GetParent() );
1171 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1172 new_event
.SetDirection( (gdk_event
->keyval
== GDK_Tab
) );
1173 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1174 new_event
.SetWindowChange( (gdk_event
->state
& GDK_CONTROL_MASK
) );
1175 new_event
.SetCurrentFocus( win
);
1176 ret
= win
->GetParent()->GetEventHandler()->ProcessEvent( new_event
);
1179 // generate wxID_CANCEL if <esc> has been pressed (typically in dialogs)
1181 (gdk_event
->keyval
== GDK_Escape
) )
1183 // however only do it if we have a Cancel button in the dialog,
1184 // otherwise the user code may get confused by the events from a
1185 // non-existing button and, worse, a wxButton might get button event
1186 // from another button which is not really expected
1187 wxWindow
*winForCancel
= win
,
1189 while ( winForCancel
)
1191 btnCancel
= winForCancel
->FindWindow(wxID_CANCEL
);
1194 // found a cancel button
1198 if ( winForCancel
->IsTopLevel() )
1200 // no need to look further
1204 // maybe our parent has a cancel button?
1205 winForCancel
= winForCancel
->GetParent();
1210 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, wxID_CANCEL
);
1211 event
.SetEventObject(btnCancel
);
1212 ret
= btnCancel
->GetEventHandler()->ProcessEvent(event
);
1218 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "key_press_event" );
1227 static gint
gtk_window_key_press_callback( GtkWidget
*widget
,
1228 GdkEventKey
*gdk_event
,
1232 wxapp_install_idle_handler();
1236 if (g_blockEventsOnDrag
)
1240 bool dont_use_IM
= FALSE
;
1242 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1243 long keycode
= wxTranslateKeySymToWXKey( gdk_event
->keyval
, FALSE
);
1246 // We were able to decode the key press without
1247 // any input method, so don't use it.
1250 // now fill all the other fields
1253 GdkModifierType state
;
1254 if (gdk_event
->window
)
1255 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1257 event
.SetTimestamp( gdk_event
->time
);
1258 event
.m_shiftDown
= (gdk_event
->state
& GDK_SHIFT_MASK
) != 0;
1259 event
.m_controlDown
= (gdk_event
->state
& GDK_CONTROL_MASK
) != 0;
1260 event
.m_altDown
= (gdk_event
->state
& GDK_MOD1_MASK
) != 0;
1261 event
.m_metaDown
= (gdk_event
->state
& GDK_MOD2_MASK
) != 0;
1262 event
.m_keyCode
= keycode
;
1263 event
.m_scanCode
= gdk_event
->keyval
;
1264 event
.m_rawCode
= (wxUint32
) gdk_event
->keyval
;
1265 event
.m_rawFlags
= 0;
1268 event
.SetEventObject( win
);
1270 // send key down event
1271 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1275 // Implement OnCharHook by checking ancesteror top level windows
1276 wxWindow
*parent
= win
;
1277 while (parent
&& !parent
->IsTopLevel())
1278 parent
= parent
->GetParent();
1281 event
.SetEventType( wxEVT_CHAR_HOOK
);
1282 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1287 event
.SetEventType(wxEVT_CHAR
);
1288 ret
= win
->GetEventHandler()->ProcessEvent( event
);
1292 // win is a control: tab can be propagated up
1294 ((gdk_event
->keyval
== GDK_Tab
) || (gdk_event
->keyval
== GDK_ISO_Left_Tab
)) &&
1295 win
->GetParent() && (win
->GetParent()->HasFlag( wxTAB_TRAVERSAL
)) )
1297 wxNavigationKeyEvent new_event
;
1298 new_event
.SetEventObject( win
->GetParent() );
1299 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1300 new_event
.SetDirection( (gdk_event
->keyval
== GDK_Tab
) );
1301 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1302 new_event
.SetWindowChange( (gdk_event
->state
& GDK_CONTROL_MASK
) );
1303 new_event
.SetCurrentFocus( win
);
1304 ret
= win
->GetParent()->GetEventHandler()->ProcessEvent( new_event
);
1308 (gdk_event
->keyval
== GDK_Escape
) )
1310 wxWindow
*winForCancel
= win
, *btnCancel
= NULL
;
1311 while ( winForCancel
)
1313 btnCancel
= winForCancel
->FindWindow(wxID_CANCEL
);
1314 if ( btnCancel
) break;
1316 if ( winForCancel
->IsTopLevel() ) break;
1318 winForCancel
= winForCancel
->GetParent();
1323 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, wxID_CANCEL
);
1324 event
.SetEventObject(btnCancel
);
1325 ret
= btnCancel
->GetEventHandler()->ProcessEvent(event
);
1332 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "key_press_event" );
1336 if (!dont_use_IM
&& win
->m_imContext
)
1338 // In GTK 2.0, we need to hand over the key
1339 // event to an input method and the IM will
1340 // emit a "commit" event containing the
1341 // actual utf8 character.
1342 gtk_im_context_filter_keypress ( (GtkIMContext
*) win
->m_imContext
, gdk_event
);
1350 static void gtk_wxwindow_commit_cb (GtkIMContext
*context
,
1356 wxKeyEvent
event( wxEVT_KEY_DOWN
);
1359 event
.m_uniChar
= g_utf8_get_char( str
);
1361 // Backward compatible for ISO-8859
1362 if (event
.m_uniChar
< 256)
1363 event
.m_keyCode
= event
.m_uniChar
;
1365 gunichar uniChar
= g_utf8_get_char( str
);
1366 // We cannot handle Unicode in non-Unicode mode
1367 if (uniChar
> 255) return;
1369 event
.m_keyCode
= uniChar
;
1374 // Implement OnCharHook by checking ancesteror top level windows
1375 wxWindow
*parent
= window
;
1376 while (parent
&& !parent
->IsTopLevel())
1377 parent
= parent
->GetParent();
1380 event
.SetEventType( wxEVT_CHAR_HOOK
);
1381 ret
= parent
->GetEventHandler()->ProcessEvent( event
);
1386 event
.SetEventType(wxEVT_CHAR
);
1387 ret
= window
->GetEventHandler()->ProcessEvent( event
);
1394 //-----------------------------------------------------------------------------
1395 // "key_release_event" from any window
1396 //-----------------------------------------------------------------------------
1398 static gint
gtk_window_key_release_callback( GtkWidget
*widget
,
1399 GdkEventKey
*gdk_event
,
1405 wxapp_install_idle_handler();
1410 if (g_blockEventsOnDrag
)
1413 wxKeyEvent
event( wxEVT_KEY_UP
);
1414 if ( !wxTranslateGTKKeyEventToWx(event
, win
, gdk_event
) )
1416 // unknown key pressed, ignore (the event would be useless anyhow
1420 if ( !win
->GetEventHandler()->ProcessEvent( event
) )
1423 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "key_release_event" );
1427 // ============================================================================
1429 // ============================================================================
1431 // ----------------------------------------------------------------------------
1432 // mouse event processing helpers
1433 // ----------------------------------------------------------------------------
1435 // init wxMouseEvent with the info from gdk_event
1437 // NB: this has to be a macro as gdk_event type is different for different
1438 // events we're used with
1439 #define InitMouseEvent(/* wxWindowGTK * */ win, \
1440 /* wxMouseEvent& */ event, \
1441 /* GdkEventXXX * */ gdk_event) \
1443 event.SetTimestamp( gdk_event->time ); \
1444 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK); \
1445 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK); \
1446 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK); \
1447 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK); \
1448 event.m_leftDown = (gdk_event->state & GDK_BUTTON1_MASK); \
1449 event.m_middleDown = (gdk_event->state & GDK_BUTTON2_MASK); \
1450 event.m_rightDown = (gdk_event->state & GDK_BUTTON3_MASK); \
1451 if (event.GetEventType()==wxEVT_MOUSEWHEEL) \
1453 if (((GdkEventButton*)gdk_event)->button == 4) \
1454 event.m_wheelRotation = 120; \
1455 else if (((GdkEventButton*)gdk_event)->button == 5) \
1456 event.m_wheelRotation = -120; \
1459 wxPoint pt = win->GetClientAreaOrigin(); \
1460 event.m_x = (wxCoord)gdk_event->x - pt.x; \
1461 event.m_y = (wxCoord)gdk_event->y - pt.y; \
1463 event.SetEventObject( win ); \
1464 event.SetId( win->GetId() ); \
1465 event.SetTimestamp( gdk_event->time ); \
1468 static void AdjustEventButtonState(wxMouseEvent& event)
1470 // GDK reports the old state of the button for a button press event, but
1471 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1472 // for a LEFT_DOWN event, not FALSE, so we will invert
1473 // left/right/middleDown for the corresponding click events
1475 if ((event
.GetEventType() == wxEVT_LEFT_DOWN
) ||
1476 (event
.GetEventType() == wxEVT_LEFT_DCLICK
) ||
1477 (event
.GetEventType() == wxEVT_LEFT_UP
))
1479 event
.m_leftDown
= !event
.m_leftDown
;
1483 if ((event
.GetEventType() == wxEVT_MIDDLE_DOWN
) ||
1484 (event
.GetEventType() == wxEVT_MIDDLE_DCLICK
) ||
1485 (event
.GetEventType() == wxEVT_MIDDLE_UP
))
1487 event
.m_middleDown
= !event
.m_middleDown
;
1491 if ((event
.GetEventType() == wxEVT_RIGHT_DOWN
) ||
1492 (event
.GetEventType() == wxEVT_RIGHT_DCLICK
) ||
1493 (event
.GetEventType() == wxEVT_RIGHT_UP
))
1495 event
.m_rightDown
= !event
.m_rightDown
;
1500 // find the window to send the mouse event too
1502 wxWindowGTK
*FindWindowForMouseEvent(wxWindowGTK
*win
, wxCoord
& x
, wxCoord
& y
)
1507 if (win
->m_wxwindow
)
1509 GtkPizza
*pizza
= GTK_PIZZA(win
->m_wxwindow
);
1510 xx
+= pizza
->xoffset
;
1511 yy
+= pizza
->yoffset
;
1514 wxNode
*node
= win
->GetChildren().First();
1517 wxWindowGTK
*child
= (wxWindowGTK
*)node
->Data();
1519 node
= node
->Next();
1520 if (!child
->IsShown())
1523 if (child
->IsTransparentForMouse())
1525 // wxStaticBox is transparent in the box itself
1526 int xx1
= child
->m_x
;
1527 int yy1
= child
->m_y
;
1528 int xx2
= child
->m_x
+ child
->m_width
;
1529 int yy2
= child
->m_x
+ child
->m_height
;
1532 if (((xx
>= xx1
) && (xx
<= xx1
+10) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1534 ((xx
>= xx2
-10) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy2
)) ||
1536 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy1
) && (yy
<= yy1
+10)) ||
1538 ((xx
>= xx1
) && (xx
<= xx2
) && (yy
>= yy2
-1) && (yy
<= yy2
)))
1549 if ((child
->m_wxwindow
== (GtkWidget
*) NULL
) &&
1550 (child
->m_x
<= xx
) &&
1551 (child
->m_y
<= yy
) &&
1552 (child
->m_x
+child
->m_width
>= xx
) &&
1553 (child
->m_y
+child
->m_height
>= yy
))
1566 //-----------------------------------------------------------------------------
1567 // "button_press_event"
1568 //-----------------------------------------------------------------------------
1570 static gint
gtk_window_button_press_callback( GtkWidget
*widget
,
1571 GdkEventButton
*gdk_event
,
1577 wxapp_install_idle_handler();
1580 wxPrintf( wxT("1) OnButtonPress from ") );
1581 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1582 wxPrintf( win->GetClassInfo()->GetClassName() );
1583 wxPrintf( wxT(".\n") );
1585 if (!win
->m_hasVMT
) return FALSE
;
1586 if (g_blockEventsOnDrag
) return TRUE
;
1587 if (g_blockEventsOnScroll
) return TRUE
;
1589 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
1591 if (win
->m_wxwindow
&& (g_focusWindow
!= win
) && win
->AcceptsFocus())
1593 gtk_widget_grab_focus( win
->m_wxwindow
);
1595 wxPrintf( wxT("GrabFocus from ") );
1596 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1597 wxPrintf( win->GetClassInfo()->GetClassName() );
1598 wxPrintf( wxT(".\n") );
1602 // GDK sends surplus button down event
1603 // before a double click event. We
1604 // need to filter these out.
1605 if (gdk_event
->type
== GDK_BUTTON_PRESS
)
1607 GdkEvent
*peek_event
= gdk_event_peek();
1610 if (peek_event
->type
== GDK_2BUTTON_PRESS
)
1612 gdk_event_free( peek_event
);
1617 gdk_event_free( peek_event
);
1622 wxEventType event_type
= wxEVT_NULL
;
1624 if (gdk_event
->button
== 1)
1626 switch (gdk_event
->type
)
1628 case GDK_BUTTON_PRESS
: event_type
= wxEVT_LEFT_DOWN
; break;
1629 case GDK_2BUTTON_PRESS
: event_type
= wxEVT_LEFT_DCLICK
; break;
1633 else if (gdk_event
->button
== 2)
1635 switch (gdk_event
->type
)
1637 case GDK_BUTTON_PRESS
: event_type
= wxEVT_MIDDLE_DOWN
; break;
1638 case GDK_2BUTTON_PRESS
: event_type
= wxEVT_MIDDLE_DCLICK
; break;
1642 else if (gdk_event
->button
== 3)
1644 switch (gdk_event
->type
)
1646 case GDK_BUTTON_PRESS
: event_type
= wxEVT_RIGHT_DOWN
; break;
1647 case GDK_2BUTTON_PRESS
: event_type
= wxEVT_RIGHT_DCLICK
; break;
1651 else if (gdk_event
->button
== 4)
1653 switch (gdk_event
->type
)
1655 case GDK_BUTTON_PRESS
: event_type
= wxEVT_MOUSEWHEEL
; break;
1659 else if (gdk_event
->button
== 5)
1661 switch (gdk_event
->type
)
1663 case GDK_BUTTON_PRESS
: event_type
= wxEVT_MOUSEWHEEL
; break;
1668 if ( event_type
== wxEVT_NULL
)
1670 // unknown mouse button or click type
1674 wxMouseEvent
event( event_type
);
1675 InitMouseEvent( win
, event
, gdk_event
);
1677 AdjustEventButtonState(event
);
1679 // wxListBox actually get mouse events from the item, so we need to give it
1680 // a chance to correct this
1681 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1683 // find the correct window to send the event too: it may be a different one
1684 // from the one which got it at GTK+ level because some control don't have
1685 // their own X window and thus cannot get any events.
1686 if ( !g_captureWindow
)
1687 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1689 gs_timeLastClick
= gdk_event
->time
;
1692 wxPrintf( wxT("2) OnButtonPress from ") );
1693 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1694 wxPrintf( win->GetClassInfo()->GetClassName() );
1695 wxPrintf( wxT(".\n") );
1698 if (win
->GetEventHandler()->ProcessEvent( event
))
1700 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "button_press_event" );
1707 //-----------------------------------------------------------------------------
1708 // "button_release_event"
1709 //-----------------------------------------------------------------------------
1711 static gint
gtk_window_button_release_callback( GtkWidget
*widget
,
1712 GdkEventButton
*gdk_event
,
1718 wxapp_install_idle_handler();
1720 if (!win
->m_hasVMT
) return FALSE
;
1721 if (g_blockEventsOnDrag
) return FALSE
;
1722 if (g_blockEventsOnScroll
) return FALSE
;
1724 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
1726 wxEventType event_type
= wxEVT_NULL
;
1728 switch (gdk_event
->button
)
1731 event_type
= wxEVT_LEFT_UP
;
1735 event_type
= wxEVT_MIDDLE_UP
;
1739 event_type
= wxEVT_RIGHT_UP
;
1743 // unknwon button, don't process
1747 wxMouseEvent
event( event_type
);
1748 InitMouseEvent( win
, event
, gdk_event
);
1750 AdjustEventButtonState(event
);
1752 // same wxListBox hack as above
1753 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1755 if ( event_type
== wxEVT_RIGHT_UP
)
1757 // generate a "context menu" event: this is similar to wxEVT_RIGHT_UP
1760 // (a) it's a command event and so is propagated to the parent
1761 // (b) under MSW it can be generated from kbd too
1762 // (c) it uses screen coords (because of (a))
1763 wxContextMenuEvent
evtCtx(wxEVT_CONTEXT_MENU
,
1765 win
->ClientToScreen(event
.GetPosition()));
1766 (void)win
->GetEventHandler()->ProcessEvent(evtCtx
);
1769 if ( !g_captureWindow
)
1770 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1772 if (win
->GetEventHandler()->ProcessEvent( event
))
1774 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "button_release_event" );
1781 //-----------------------------------------------------------------------------
1782 // "motion_notify_event"
1783 //-----------------------------------------------------------------------------
1785 static gint
gtk_window_motion_notify_callback( GtkWidget
*widget
,
1786 GdkEventMotion
*gdk_event
,
1792 wxapp_install_idle_handler();
1794 if (!win
->m_hasVMT
) return FALSE
;
1795 if (g_blockEventsOnDrag
) return FALSE
;
1796 if (g_blockEventsOnScroll
) return FALSE
;
1798 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
1800 if (gdk_event
->is_hint
)
1804 GdkModifierType state
;
1805 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1811 printf( "OnMotion from " );
1812 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1813 printf( win->GetClassInfo()->GetClassName() );
1817 wxMouseEvent
event( wxEVT_MOTION
);
1818 InitMouseEvent(win
, event
, gdk_event
);
1820 if ( g_captureWindow
)
1822 // synthetize a mouse enter or leave event if needed
1823 GdkWindow
*winUnderMouse
= gdk_window_at_pointer(NULL
, NULL
);
1824 bool hasMouse
= winUnderMouse
== gdk_event
->window
;
1825 if ( hasMouse
!= g_captureWindowHasMouse
)
1827 // the mouse changed window
1828 g_captureWindowHasMouse
= hasMouse
;
1830 wxMouseEvent
event(g_captureWindowHasMouse
? wxEVT_ENTER_WINDOW
1831 : wxEVT_LEAVE_WINDOW
);
1832 InitMouseEvent(win
, event
, gdk_event
);
1833 event
.SetEventObject(win
);
1834 win
->GetEventHandler()->ProcessEvent(event
);
1839 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1842 if (win
->GetEventHandler()->ProcessEvent( event
))
1844 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "motion_notify_event" );
1851 //-----------------------------------------------------------------------------
1853 //-----------------------------------------------------------------------------
1855 // send the wxChildFocusEvent and wxFocusEvent, common code of
1856 // gtk_window_focus_in_callback() and SetFocus()
1857 static bool DoSendFocusEvents(wxWindow
*win
)
1859 // Notify the parent keeping track of focus for the kbd navigation
1860 // purposes that we got it.
1861 wxChildFocusEvent
eventChildFocus(win
);
1862 (void)win
->GetEventHandler()->ProcessEvent(eventChildFocus
);
1864 wxFocusEvent
eventFocus(wxEVT_SET_FOCUS
, win
->GetId());
1865 eventFocus
.SetEventObject(win
);
1867 return win
->GetEventHandler()->ProcessEvent(eventFocus
);
1870 static gint
gtk_window_focus_in_callback( GtkWidget
*widget
,
1871 GdkEvent
*WXUNUSED(event
),
1877 wxapp_install_idle_handler();
1879 if (!win
->m_hasVMT
) return FALSE
;
1880 if (g_blockEventsOnDrag
) return FALSE
;
1882 switch ( g_sendActivateEvent
)
1885 // we've got focus from outside, synthetize wxActivateEvent
1886 g_sendActivateEvent
= 1;
1890 // another our window just lost focus, it was already ours before
1891 // - don't send any wxActivateEvent
1892 g_sendActivateEvent
= -1;
1897 g_focusWindow
= win
;
1899 wxLogTrace(TRACE_FOCUS
,
1900 _T("%s: focus in"), win
->GetName().c_str());
1904 gdk_im_begin(win
->m_ic
, win
->m_wxwindow
->window
);
1908 // caret needs to be informed about focus change
1909 wxCaret
*caret
= win
->GetCaret();
1912 caret
->OnSetFocus();
1914 #endif // wxUSE_CARET
1916 g_activeFrameLostFocus
= FALSE
;
1918 wxWindowGTK
*active
= wxGetTopLevelParent(win
);
1919 if ( active
!= g_activeFrame
)
1921 if ( g_activeFrame
)
1923 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from focus_in)"), g_activeFrame
);
1924 wxActivateEvent
event(wxEVT_ACTIVATE
, FALSE
, g_activeFrame
->GetId());
1925 event
.SetEventObject(g_activeFrame
);
1926 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
1929 wxLogTrace(wxT("activate"), wxT("Activating frame %p (from focus_in)"), active
);
1930 g_activeFrame
= active
;
1931 wxActivateEvent
event(wxEVT_ACTIVATE
, TRUE
, g_activeFrame
->GetId());
1932 event
.SetEventObject(g_activeFrame
);
1933 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
1935 // Don't send focus events in addition to activate
1936 // if (win == g_activeFrame)
1940 // does the window itself think that it has the focus?
1941 if ( !win
->m_hasFocus
)
1943 // not yet, notify it
1944 win
->m_hasFocus
= TRUE
;
1946 if ( DoSendFocusEvents(win
) )
1948 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "focus_in_event" );
1956 //-----------------------------------------------------------------------------
1957 // "focus_out_event"
1958 //-----------------------------------------------------------------------------
1960 static gint
gtk_window_focus_out_callback( GtkWidget
*widget
, GdkEventFocus
*gdk_event
, wxWindowGTK
*win
)
1965 wxapp_install_idle_handler();
1967 if (!win
->m_hasVMT
) return FALSE
;
1968 if (g_blockEventsOnDrag
) return FALSE
;
1970 wxLogTrace( TRACE_FOCUS
,
1971 _T("%s: focus out"), win
->GetName().c_str() );
1973 if ( !g_activeFrameLostFocus
&& g_activeFrame
)
1975 // VZ: commenting this out because it does happen (although not easy
1976 // to reproduce, I only see it when using wxMiniFrame and not
1977 // always) and makes using Mahogany quite annoying
1979 wxASSERT_MSG( wxGetTopLevelParent(win
) == g_activeFrame
,
1980 wxT("unfocusing window that hasn't gained focus properly") );
1983 g_activeFrameLostFocus
= TRUE
;
1986 // if the focus goes out of our app alltogether, OnIdle() will send
1987 // wxActivateEvent, otherwise gtk_window_focus_in_callback() will reset
1988 // g_sendActivateEvent to -1
1989 g_sendActivateEvent
= 0;
1991 wxWindowGTK
*winFocus
= wxFindFocusedChild(win
);
1995 g_focusWindow
= (wxWindowGTK
*)NULL
;
2003 // caret needs to be informed about focus change
2004 wxCaret
*caret
= win
->GetCaret();
2007 caret
->OnKillFocus();
2009 #endif // wxUSE_CARET
2011 // don't send the window a kill focus event if it thinks that it doesn't
2012 // have focus already
2013 if ( win
->m_hasFocus
)
2015 win
->m_hasFocus
= FALSE
;
2017 wxFocusEvent
event( wxEVT_KILL_FOCUS
, win
->GetId() );
2018 event
.SetEventObject( win
);
2020 if (win
->GetEventHandler()->ProcessEvent( event
))
2022 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "focus_out_event" );
2030 //-----------------------------------------------------------------------------
2031 // "enter_notify_event"
2032 //-----------------------------------------------------------------------------
2035 gint
gtk_window_enter_callback( GtkWidget
*widget
,
2036 GdkEventCrossing
*gdk_event
,
2042 wxapp_install_idle_handler();
2044 if (!win
->m_hasVMT
) return FALSE
;
2045 if (g_blockEventsOnDrag
) return FALSE
;
2047 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
2051 GdkModifierType state
= (GdkModifierType
)0;
2053 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
2055 wxMouseEvent
event( wxEVT_ENTER_WINDOW
);
2056 InitMouseEvent(win
, event
, gdk_event
);
2057 wxPoint pt
= win
->GetClientAreaOrigin();
2058 event
.m_x
= x
+ pt
.x
;
2059 event
.m_y
= y
+ pt
.y
;
2061 if (win
->GetEventHandler()->ProcessEvent( event
))
2063 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "enter_notify_event" );
2070 //-----------------------------------------------------------------------------
2071 // "leave_notify_event"
2072 //-----------------------------------------------------------------------------
2074 static gint
gtk_window_leave_callback( GtkWidget
*widget
, GdkEventCrossing
*gdk_event
, wxWindowGTK
*win
)
2079 wxapp_install_idle_handler();
2081 if (!win
->m_hasVMT
) return FALSE
;
2082 if (g_blockEventsOnDrag
) return FALSE
;
2084 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
2086 wxMouseEvent
event( wxEVT_LEAVE_WINDOW
);
2087 event
.SetTimestamp( gdk_event
->time
);
2088 event
.SetEventObject( win
);
2092 GdkModifierType state
= (GdkModifierType
)0;
2094 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
2096 event
.m_shiftDown
= (state
& GDK_SHIFT_MASK
) != 0;
2097 event
.m_controlDown
= (state
& GDK_CONTROL_MASK
) != 0;
2098 event
.m_altDown
= (state
& GDK_MOD1_MASK
) != 0;
2099 event
.m_metaDown
= (state
& GDK_MOD2_MASK
) != 0;
2100 event
.m_leftDown
= (state
& GDK_BUTTON1_MASK
) != 0;
2101 event
.m_middleDown
= (state
& GDK_BUTTON2_MASK
) != 0;
2102 event
.m_rightDown
= (state
& GDK_BUTTON3_MASK
) != 0;
2104 wxPoint pt
= win
->GetClientAreaOrigin();
2105 event
.m_x
= x
+ pt
.x
;
2106 event
.m_y
= y
+ pt
.y
;
2108 if (win
->GetEventHandler()->ProcessEvent( event
))
2110 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "leave_notify_event" );
2117 //-----------------------------------------------------------------------------
2118 // "value_changed" from m_vAdjust
2119 //-----------------------------------------------------------------------------
2121 static void gtk_window_vscroll_callback( GtkAdjustment
*adjust
,
2128 wxapp_install_idle_handler();
2130 if (g_blockEventsOnDrag
) return;
2132 if (!win
->m_hasVMT
) return;
2134 float diff
= adjust
->value
- win
->m_oldVerticalPos
;
2135 if (fabs(diff
) < 0.2) return;
2137 win
->m_oldVerticalPos
= adjust
->value
;
2140 GtkScrolledWindow
*sw
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2142 wxEventType command
= GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw
->vscrollbar
));
2144 int value
= (int)(adjust
->value
+0.5);
2146 wxScrollWinEvent
event( command
, value
, wxVERTICAL
);
2147 event
.SetEventObject( win
);
2148 win
->GetEventHandler()->ProcessEvent( event
);
2151 //-----------------------------------------------------------------------------
2152 // "value_changed" from m_hAdjust
2153 //-----------------------------------------------------------------------------
2155 static void gtk_window_hscroll_callback( GtkAdjustment
*adjust
,
2162 wxapp_install_idle_handler();
2164 if (g_blockEventsOnDrag
) return;
2165 if (!win
->m_hasVMT
) return;
2167 float diff
= adjust
->value
- win
->m_oldHorizontalPos
;
2168 if (fabs(diff
) < 0.2) return;
2171 GtkScrolledWindow
*sw
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2173 wxEventType command
= GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw
->hscrollbar
));
2175 win
->m_oldHorizontalPos
= adjust
->value
;
2177 int value
= (int)(adjust
->value
+0.5);
2179 wxScrollWinEvent
event( command
, value
, wxHORIZONTAL
);
2180 event
.SetEventObject( win
);
2181 win
->GetEventHandler()->ProcessEvent( event
);
2184 //-----------------------------------------------------------------------------
2185 // "button_press_event" from scrollbar
2186 //-----------------------------------------------------------------------------
2188 static gint
gtk_scrollbar_button_press_callback( GtkRange
*widget
,
2189 GdkEventButton
*gdk_event
,
2195 wxapp_install_idle_handler();
2198 g_blockEventsOnScroll
= TRUE
;
2200 // FIXME: there is no 'slider' field in GTK+ 2.0 any more
2202 win
->m_isScrolling
= (gdk_event
->window
== widget
->slider
);
2208 //-----------------------------------------------------------------------------
2209 // "button_release_event" from scrollbar
2210 //-----------------------------------------------------------------------------
2212 static gint
gtk_scrollbar_button_release_callback( GtkRange
*widget
,
2213 GdkEventButton
*WXUNUSED(gdk_event
),
2218 // don't test here as we can release the mouse while being over
2219 // a different window than the slider
2221 // if (gdk_event->window != widget->slider) return FALSE;
2223 g_blockEventsOnScroll
= FALSE
;
2225 if (win
->m_isScrolling
)
2227 wxEventType command
= wxEVT_SCROLLWIN_THUMBRELEASE
;
2231 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2232 if (widget
== GTK_RANGE(scrolledWindow
->hscrollbar
))
2234 value
= (int)(win
->m_hAdjust
->value
+0.5);
2237 if (widget
== GTK_RANGE(scrolledWindow
->vscrollbar
))
2239 value
= (int)(win
->m_vAdjust
->value
+0.5);
2243 wxScrollWinEvent
event( command
, value
, dir
);
2244 event
.SetEventObject( win
);
2245 win
->GetEventHandler()->ProcessEvent( event
);
2248 win
->m_isScrolling
= FALSE
;
2253 // ----------------------------------------------------------------------------
2254 // this wxWindowBase function is implemented here (in platform-specific file)
2255 // because it is static and so couldn't be made virtual
2256 // ----------------------------------------------------------------------------
2258 wxWindow
*wxWindowBase::FindFocus()
2260 // the cast is necessary when we compile in wxUniversal mode
2261 return (wxWindow
*)g_focusWindow
;
2264 //-----------------------------------------------------------------------------
2266 //-----------------------------------------------------------------------------
2268 // VZ: Robert commented the code using out so it generates warnings: should
2269 // be either fixed or removed completely
2272 static void gtk_window_destroy_callback( GtkWidget
* widget
, wxWindow
*win
)
2274 wxWindowDestroyEvent
event(win
);
2275 win
->GetEventHandler()->ProcessEvent(event
);
2280 //-----------------------------------------------------------------------------
2281 // "realize" from m_widget
2282 //-----------------------------------------------------------------------------
2284 /* We cannot set colours and fonts before the widget has
2285 been realized, so we do this directly after realization. */
2288 gtk_window_realized_callback( GtkWidget
*m_widget
, wxWindow
*win
)
2293 wxapp_install_idle_handler();
2295 if (win
->m_delayedBackgroundColour
)
2296 win
->GtkSetBackgroundColour( win
->GetBackgroundColour() );
2298 if (win
->m_delayedForegroundColour
)
2299 win
->GtkSetForegroundColour( win
->GetForegroundColour() );
2302 if (win
->m_imContext
)
2304 GtkPizza
*pizza
= GTK_PIZZA( m_widget
);
2305 gtk_im_context_set_client_window( (GtkIMContext
*) win
->m_imContext
, pizza
->bin_window
);
2309 wxWindowCreateEvent
event( win
);
2310 event
.SetEventObject( win
);
2311 win
->GetEventHandler()->ProcessEvent( event
);
2316 //-----------------------------------------------------------------------------
2318 //-----------------------------------------------------------------------------
2321 void gtk_window_size_callback( GtkWidget
*WXUNUSED(widget
),
2322 GtkAllocation
*WXUNUSED(alloc
),
2326 wxapp_install_idle_handler();
2328 if (!win
->m_hasScrolling
) return;
2330 int client_width
= 0;
2331 int client_height
= 0;
2332 win
->GetClientSize( &client_width
, &client_height
);
2333 if ((client_width
== win
->m_oldClientWidth
) && (client_height
== win
->m_oldClientHeight
))
2336 win
->m_oldClientWidth
= client_width
;
2337 win
->m_oldClientHeight
= client_height
;
2339 if (!win
->m_nativeSizeEvent
)
2341 wxSizeEvent
event( win
->GetSize(), win
->GetId() );
2342 event
.SetEventObject( win
);
2343 win
->GetEventHandler()->ProcessEvent( event
);
2349 #define WXUNUSED_UNLESS_XIM(param) param
2351 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2354 /* Resize XIM window */
2357 void gtk_wxwindow_size_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2358 GtkAllocation
* WXUNUSED_UNLESS_XIM(alloc
),
2359 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2362 wxapp_install_idle_handler();
2368 if (gdk_ic_get_style (win
->m_ic
) & GDK_IM_PREEDIT_POSITION
)
2372 gdk_window_get_size (widget
->window
, &width
, &height
);
2373 win
->m_icattr
->preedit_area
.width
= width
;
2374 win
->m_icattr
->preedit_area
.height
= height
;
2375 gdk_ic_set_attr (win
->m_ic
, win
->m_icattr
, GDK_IC_PREEDIT_AREA
);
2380 //-----------------------------------------------------------------------------
2381 // "realize" from m_wxwindow
2382 //-----------------------------------------------------------------------------
2384 /* Initialize XIM support */
2387 gtk_wxwindow_realized_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2388 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2391 wxapp_install_idle_handler();
2394 if (win
->m_ic
) return FALSE
;
2395 if (!widget
) return FALSE
;
2396 if (!gdk_im_ready()) return FALSE
;
2398 win
->m_icattr
= gdk_ic_attr_new();
2399 if (!win
->m_icattr
) return FALSE
;
2403 GdkColormap
*colormap
;
2404 GdkICAttr
*attr
= win
->m_icattr
;
2405 unsigned attrmask
= GDK_IC_ALL_REQ
;
2407 GdkIMStyle supported_style
= (GdkIMStyle
)
2408 (GDK_IM_PREEDIT_NONE
|
2409 GDK_IM_PREEDIT_NOTHING
|
2410 GDK_IM_PREEDIT_POSITION
|
2411 GDK_IM_STATUS_NONE
|
2412 GDK_IM_STATUS_NOTHING
);
2414 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2415 supported_style
= (GdkIMStyle
)(supported_style
& ~GDK_IM_PREEDIT_POSITION
);
2417 attr
->style
= style
= gdk_im_decide_style (supported_style
);
2418 attr
->client_window
= widget
->window
;
2420 if ((colormap
= gtk_widget_get_colormap (widget
)) !=
2421 gtk_widget_get_default_colormap ())
2423 attrmask
|= GDK_IC_PREEDIT_COLORMAP
;
2424 attr
->preedit_colormap
= colormap
;
2427 attrmask
|= GDK_IC_PREEDIT_FOREGROUND
;
2428 attrmask
|= GDK_IC_PREEDIT_BACKGROUND
;
2429 attr
->preedit_foreground
= widget
->style
->fg
[GTK_STATE_NORMAL
];
2430 attr
->preedit_background
= widget
->style
->base
[GTK_STATE_NORMAL
];
2432 switch (style
& GDK_IM_PREEDIT_MASK
)
2434 case GDK_IM_PREEDIT_POSITION
:
2435 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2437 g_warning ("over-the-spot style requires fontset");
2441 gdk_window_get_size (widget
->window
, &width
, &height
);
2443 attrmask
|= GDK_IC_PREEDIT_POSITION_REQ
;
2444 attr
->spot_location
.x
= 0;
2445 attr
->spot_location
.y
= height
;
2446 attr
->preedit_area
.x
= 0;
2447 attr
->preedit_area
.y
= 0;
2448 attr
->preedit_area
.width
= width
;
2449 attr
->preedit_area
.height
= height
;
2450 attr
->preedit_fontset
= widget
->style
->font
;
2455 win
->m_ic
= gdk_ic_new (attr
, (GdkICAttributesType
)attrmask
);
2457 if (win
->m_ic
== NULL
)
2458 g_warning ("Can't create input context.");
2461 mask
= gdk_window_get_events (widget
->window
);
2462 mask
= (GdkEventMask
)(mask
| gdk_ic_get_events (win
->m_ic
));
2463 gdk_window_set_events (widget
->window
, mask
);
2465 if (GTK_WIDGET_HAS_FOCUS(widget
))
2466 gdk_im_begin (win
->m_ic
, widget
->window
);
2473 //-----------------------------------------------------------------------------
2474 // InsertChild for wxWindowGTK.
2475 //-----------------------------------------------------------------------------
2477 /* Callback for wxWindowGTK. This very strange beast has to be used because
2478 * C++ has no virtual methods in a constructor. We have to emulate a
2479 * virtual function here as wxNotebook requires a different way to insert
2480 * a child in it. I had opted for creating a wxNotebookPage window class
2481 * which would have made this superfluous (such in the MDI window system),
2482 * but no-one was listening to me... */
2484 static void wxInsertChildInWindow( wxWindowGTK
* parent
, wxWindowGTK
* child
)
2486 /* the window might have been scrolled already, do we
2487 have to adapt the position */
2488 GtkPizza
*pizza
= GTK_PIZZA(parent
->m_wxwindow
);
2489 child
->m_x
+= pizza
->xoffset
;
2490 child
->m_y
+= pizza
->yoffset
;
2492 gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
),
2493 GTK_WIDGET(child
->m_widget
),
2500 //-----------------------------------------------------------------------------
2502 //-----------------------------------------------------------------------------
2504 wxWindow
*wxGetActiveWindow()
2506 return wxWindow::FindFocus();
2509 //-----------------------------------------------------------------------------
2511 //-----------------------------------------------------------------------------
2513 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2515 #ifdef __WXUNIVERSAL__
2516 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
)
2518 IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
)
2519 #endif // __WXUNIVERSAL__/__WXGTK__
2521 void wxWindowGTK::Init()
2527 m_widget
= (GtkWidget
*) NULL
;
2528 m_wxwindow
= (GtkWidget
*) NULL
;
2529 m_focusWidget
= (GtkWidget
*) NULL
;
2539 m_needParent
= TRUE
;
2540 m_isBeingDeleted
= FALSE
;
2543 m_nativeSizeEvent
= FALSE
;
2545 m_hasScrolling
= FALSE
;
2546 m_isScrolling
= FALSE
;
2548 m_hAdjust
= (GtkAdjustment
*) NULL
;
2549 m_vAdjust
= (GtkAdjustment
*) NULL
;
2550 m_oldHorizontalPos
= 0.0;
2551 m_oldVerticalPos
= 0.0;
2554 m_widgetStyle
= (GtkStyle
*) NULL
;
2556 m_insertCallback
= (wxInsertChildFunction
) NULL
;
2558 m_acceptsFocus
= FALSE
;
2561 m_clipPaintRegion
= FALSE
;
2563 m_cursor
= *wxSTANDARD_CURSOR
;
2565 m_delayedForegroundColour
= FALSE
;
2566 m_delayedBackgroundColour
= FALSE
;
2570 m_x11Context
= NULL
;
2573 m_ic
= (GdkIC
*) NULL
;
2574 m_icattr
= (GdkICAttr
*) NULL
;
2579 wxWindowGTK::wxWindowGTK()
2584 wxWindowGTK::wxWindowGTK( wxWindow
*parent
,
2589 const wxString
&name
)
2593 Create( parent
, id
, pos
, size
, style
, name
);
2596 bool wxWindowGTK::Create( wxWindow
*parent
,
2601 const wxString
&name
)
2603 if (!PreCreation( parent
, pos
, size
) ||
2604 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
2606 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2610 m_insertCallback
= wxInsertChildInWindow
;
2612 // always needed for background clearing
2613 m_delayedBackgroundColour
= TRUE
;
2615 m_widget
= gtk_scrolled_window_new( (GtkAdjustment
*) NULL
, (GtkAdjustment
*) NULL
);
2616 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS
);
2618 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(m_widget
);
2620 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2621 scroll_class
->scrollbar_spacing
= 0;
2623 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
2625 m_hAdjust
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->hscrollbar
) );
2626 m_vAdjust
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->vscrollbar
) );
2628 m_wxwindow
= gtk_pizza_new();
2630 #ifndef __WXUNIVERSAL__
2631 GtkPizza
*pizza
= GTK_PIZZA(m_wxwindow
);
2633 if (HasFlag(wxRAISED_BORDER
))
2635 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_OUT
);
2637 else if (HasFlag(wxSUNKEN_BORDER
))
2639 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_IN
);
2641 else if (HasFlag(wxSIMPLE_BORDER
))
2643 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_THIN
);
2647 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_NONE
);
2649 #endif // __WXUNIVERSAL__
2651 gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow
);
2653 GTK_WIDGET_SET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS
);
2654 m_acceptsFocus
= TRUE
;
2656 // I _really_ don't want scrollbars in the beginning
2657 m_vAdjust
->lower
= 0.0;
2658 m_vAdjust
->upper
= 1.0;
2659 m_vAdjust
->value
= 0.0;
2660 m_vAdjust
->step_increment
= 1.0;
2661 m_vAdjust
->page_increment
= 1.0;
2662 m_vAdjust
->page_size
= 5.0;
2663 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" );
2664 m_hAdjust
->lower
= 0.0;
2665 m_hAdjust
->upper
= 1.0;
2666 m_hAdjust
->value
= 0.0;
2667 m_hAdjust
->step_increment
= 1.0;
2668 m_hAdjust
->page_increment
= 1.0;
2669 m_hAdjust
->page_size
= 5.0;
2670 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" );
2672 // these handlers block mouse events to any window during scrolling such as
2673 // motion events and prevent GTK and wxWindows from fighting over where the
2676 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_press_event",
2677 (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this );
2679 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_press_event",
2680 (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this );
2682 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_release_event",
2683 (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this );
2685 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_release_event",
2686 (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this );
2688 // these handlers get notified when screen updates are required either when
2689 // scrolling or when the window size (and therefore scrollbar configuration)
2692 gtk_signal_connect( GTK_OBJECT(m_hAdjust
), "value_changed",
2693 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
2694 gtk_signal_connect( GTK_OBJECT(m_vAdjust
), "value_changed",
2695 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
2698 // Create input method handler
2699 m_imContext
= (GtkIMMulticontext
*) gtk_im_multicontext_new ();
2701 // Cannot handle drawing preedited text yet
2702 gtk_im_context_set_use_preedit( (GtkIMContext
*) m_imContext
, FALSE
);
2704 g_signal_connect (G_OBJECT (m_imContext
), "commit",
2705 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2708 gtk_widget_show( m_wxwindow
);
2711 m_parent
->DoAddChild( this );
2713 m_focusWidget
= m_wxwindow
;
2722 wxWindowGTK::~wxWindowGTK()
2724 if (g_focusWindow
== this)
2725 g_focusWindow
= NULL
;
2727 if (g_activeFrame
== this)
2728 g_activeFrame
= NULL
;
2730 if ( g_delayedFocus
== this )
2731 g_delayedFocus
= NULL
;
2733 m_isBeingDeleted
= TRUE
;
2742 m_parent
->RemoveChild( this );
2746 gdk_ic_destroy (m_ic
);
2748 gdk_ic_attr_destroy (m_icattr
);
2753 #if DISABLE_STYLE_IF_BROKEN_THEME
2754 // don't delete if it's a pixmap theme style
2755 if (!m_widgetStyle
->engine_data
)
2756 gtk_style_unref( m_widgetStyle
);
2758 m_widgetStyle
= (GtkStyle
*) NULL
;
2763 gtk_widget_destroy( m_wxwindow
);
2764 m_wxwindow
= (GtkWidget
*) NULL
;
2769 gtk_widget_destroy( m_widget
);
2770 m_widget
= (GtkWidget
*) NULL
;
2774 bool wxWindowGTK::PreCreation( wxWindowGTK
*parent
, const wxPoint
&pos
, const wxSize
&size
)
2776 wxCHECK_MSG( !m_needParent
|| parent
, FALSE
, wxT("Need complete parent.") );
2778 // This turns -1 into 30 so that a minimal window is
2779 // visible even although -1,-1 has been given as the
2780 // size of the window. the same trick is used in other
2781 // ports and should make debugging easier.
2782 m_width
= WidthDefault(size
.x
) ;
2783 m_height
= HeightDefault(size
.y
);
2788 // some reasonable defaults
2793 m_x
= (gdk_screen_width () - m_width
) / 2;
2794 if (m_x
< 10) m_x
= 10;
2798 m_y
= (gdk_screen_height () - m_height
) / 2;
2799 if (m_y
< 10) m_y
= 10;
2806 void wxWindowGTK::PostCreation()
2808 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2814 // these get reported to wxWindows -> wxPaintEvent
2816 gtk_pizza_set_external( GTK_PIZZA(m_wxwindow
), TRUE
);
2818 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "expose_event",
2819 GTK_SIGNAL_FUNC(gtk_window_expose_callback
), (gpointer
)this );
2822 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "draw",
2823 GTK_SIGNAL_FUNC(gtk_window_draw_callback
), (gpointer
)this );
2825 if (HasFlag(wxNO_FULL_REPAINT_ON_RESIZE
))
2827 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "event",
2828 GTK_SIGNAL_FUNC(gtk_window_event_event_callback
), (gpointer
)this );
2831 // gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow), HasFlag( wxNO_FULL_REPAINT_ON_RESIZE ) );
2835 // Create input method handler
2836 m_imContext
= (GtkIMMulticontext
*) gtk_im_multicontext_new ();
2838 // Cannot handle drawing preedited text yet
2839 gtk_im_context_set_use_preedit( (GtkIMContext
*) m_imContext
, FALSE
);
2841 g_signal_connect (G_OBJECT (m_imContext
), "commit",
2842 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2846 // these are called when the "sunken" or "raised" borders are drawn
2847 gtk_signal_connect( GTK_OBJECT(m_widget
), "expose_event",
2848 GTK_SIGNAL_FUNC(gtk_window_own_expose_callback
), (gpointer
)this );
2851 gtk_signal_connect( GTK_OBJECT(m_widget
), "draw",
2852 GTK_SIGNAL_FUNC(gtk_window_own_draw_callback
), (gpointer
)this );
2858 if (m_focusWidget
== NULL
)
2859 m_focusWidget
= m_widget
;
2861 gtk_signal_connect( GTK_OBJECT(m_focusWidget
), "focus_in_event",
2862 GTK_SIGNAL_FUNC(gtk_window_focus_in_callback
), (gpointer
)this );
2864 gtk_signal_connect( GTK_OBJECT(m_focusWidget
), "focus_out_event",
2865 GTK_SIGNAL_FUNC(gtk_window_focus_out_callback
), (gpointer
)this );
2867 // connect to the various key and mouse handlers
2869 GtkWidget
*connect_widget
= GetConnectWidget();
2871 ConnectWidget( connect_widget
);
2873 /* We cannot set colours, fonts and cursors before the widget has
2874 been realized, so we do this directly after realization */
2875 gtk_signal_connect( GTK_OBJECT(connect_widget
), "realize",
2876 GTK_SIGNAL_FUNC(gtk_window_realized_callback
), (gpointer
) this );
2880 // Catch native resize events
2881 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "size_allocate",
2882 GTK_SIGNAL_FUNC(gtk_window_size_callback
), (gpointer
)this );
2884 // Initialize XIM support
2885 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "realize",
2886 GTK_SIGNAL_FUNC(gtk_wxwindow_realized_callback
), (gpointer
) this );
2888 // And resize XIM window
2889 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "size_allocate",
2890 GTK_SIGNAL_FUNC(gtk_wxwindow_size_callback
), (gpointer
)this );
2893 if (!GTK_IS_COMBO(m_widget
))
2895 // This is needed if we want to add our windows into native
2896 // GTK control, such as the toolbar. With this callback, the
2897 // toolbar gets to know the correct size (the one set by the
2898 // programmer). Sadly, it misbehaves for wxComboBox. FIXME
2899 // when moving to GTK 2.0.
2900 gtk_signal_connect( GTK_OBJECT(m_widget
), "size_request",
2901 GTK_SIGNAL_FUNC(gtk_window_size_request_callback
), (gpointer
) this );
2907 void wxWindowGTK::ConnectWidget( GtkWidget
*widget
)
2909 gtk_signal_connect( GTK_OBJECT(widget
), "key_press_event",
2910 GTK_SIGNAL_FUNC(gtk_window_key_press_callback
), (gpointer
)this );
2912 gtk_signal_connect( GTK_OBJECT(widget
), "key_release_event",
2913 GTK_SIGNAL_FUNC(gtk_window_key_release_callback
), (gpointer
)this );
2915 gtk_signal_connect( GTK_OBJECT(widget
), "button_press_event",
2916 GTK_SIGNAL_FUNC(gtk_window_button_press_callback
), (gpointer
)this );
2918 gtk_signal_connect( GTK_OBJECT(widget
), "button_release_event",
2919 GTK_SIGNAL_FUNC(gtk_window_button_release_callback
), (gpointer
)this );
2921 gtk_signal_connect( GTK_OBJECT(widget
), "motion_notify_event",
2922 GTK_SIGNAL_FUNC(gtk_window_motion_notify_callback
), (gpointer
)this );
2924 gtk_signal_connect( GTK_OBJECT(widget
), "enter_notify_event",
2925 GTK_SIGNAL_FUNC(gtk_window_enter_callback
), (gpointer
)this );
2927 gtk_signal_connect( GTK_OBJECT(widget
), "leave_notify_event",
2928 GTK_SIGNAL_FUNC(gtk_window_leave_callback
), (gpointer
)this );
2930 // This keeps crashing on me. RR.
2932 // gtk_signal_connect( GTK_OBJECT(widget), "destroy",
2933 // GTK_SIGNAL_FUNC(gtk_window_destroy_callback), (gpointer)this );
2936 bool wxWindowGTK::Destroy()
2938 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2942 return wxWindowBase::Destroy();
2945 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
)
2947 gtk_pizza_set_size( GTK_PIZZA(m_parent
->m_wxwindow
), m_widget
, x
, y
, width
, height
);
2950 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
2952 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2953 wxASSERT_MSG( (m_parent
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") );
2956 printf( "DoSetSize: name %s, x,y,w,h: %d,%d,%d,%d \n", GetName().c_str(), x,y,width,height );
2959 if (m_resizing
) return; /* I don't like recursions */
2962 int currentX
, currentY
;
2963 GetPosition(¤tX
, ¤tY
);
2968 AdjustForParentClientOrigin(x
, y
, sizeFlags
);
2970 if (m_parent
->m_wxwindow
== NULL
) /* i.e. wxNotebook */
2972 /* don't set the size for children of wxNotebook, just take the values. */
2980 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
2981 if ((sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) == 0)
2983 if (x
!= -1) m_x
= x
+ pizza
->xoffset
;
2984 if (y
!= -1) m_y
= y
+ pizza
->yoffset
;
2988 m_x
= x
+ pizza
->xoffset
;
2989 m_y
= y
+ pizza
->yoffset
;
2991 if (width
!= -1) m_width
= width
;
2992 if (height
!= -1) m_height
= height
;
2994 if ((sizeFlags
& wxSIZE_AUTO_WIDTH
) == wxSIZE_AUTO_WIDTH
)
2996 if (width
== -1) m_width
= 80;
2999 if ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) == wxSIZE_AUTO_HEIGHT
)
3001 if (height
== -1) m_height
= 26;
3004 int minWidth
= GetMinWidth(),
3005 minHeight
= GetMinHeight(),
3006 maxWidth
= GetMaxWidth(),
3007 maxHeight
= GetMaxHeight();
3009 if ((minWidth
!= -1) && (m_width
< minWidth
)) m_width
= minWidth
;
3010 if ((minHeight
!= -1) && (m_height
< minHeight
)) m_height
= minHeight
;
3011 if ((maxWidth
!= -1) && (m_width
> maxWidth
)) m_width
= maxWidth
;
3012 if ((maxHeight
!= -1) && (m_height
> maxHeight
)) m_height
= maxHeight
;
3015 int bottom_border
= 0;
3018 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
3020 /* the default button has a border around it */
3026 DoMoveWindow( m_x
-border
,
3029 m_height
+border
+bottom_border
);
3034 /* Sometimes the client area changes size without the
3035 whole windows's size changing, but if the whole
3036 windows's size doesn't change, no wxSizeEvent will
3037 normally be sent. Here we add an extra test if
3038 the client test has been changed and this will
3040 GetClientSize( &m_oldClientWidth
, &m_oldClientHeight
);
3044 wxPrintf( "OnSize sent from " );
3045 if (GetClassInfo() && GetClassInfo()->GetClassName())
3046 wxPrintf( GetClassInfo()->GetClassName() );
3047 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
3050 if (!m_nativeSizeEvent
)
3052 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
3053 event
.SetEventObject( this );
3054 GetEventHandler()->ProcessEvent( event
);
3060 void wxWindowGTK::OnInternalIdle()
3062 // Update invalidated regions.
3065 // Synthetize activate events.
3066 if ( g_sendActivateEvent
!= -1 )
3068 bool activate
= g_sendActivateEvent
!= 0;
3071 g_sendActivateEvent
= -1;
3073 wxTheApp
->SetActive(activate
, (wxWindow
*)g_focusWindowLast
);
3076 if ( g_activeFrameLostFocus
)
3078 if ( g_activeFrame
)
3080 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from idle)"), g_activeFrame
);
3081 wxActivateEvent
event(wxEVT_ACTIVATE
, FALSE
, g_activeFrame
->GetId());
3082 event
.SetEventObject(g_activeFrame
);
3083 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
3084 g_activeFrame
= NULL
;
3086 g_activeFrameLostFocus
= FALSE
;
3089 wxCursor cursor
= m_cursor
;
3090 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
3094 /* I now set the cursor anew in every OnInternalIdle call
3095 as setting the cursor in a parent window also effects the
3096 windows above so that checking for the current cursor is
3101 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3103 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3105 if (!g_globalCursor
.Ok())
3106 cursor
= *wxSTANDARD_CURSOR
;
3108 window
= m_widget
->window
;
3109 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
3110 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3116 GdkWindow
*window
= m_widget
->window
;
3117 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
3118 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3126 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
3128 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3130 if (width
) (*width
) = m_width
;
3131 if (height
) (*height
) = m_height
;
3134 void wxWindowGTK::DoSetClientSize( int width
, int height
)
3136 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3140 SetSize( width
, height
);
3147 #ifndef __WXUNIVERSAL__
3148 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3150 /* when using GTK 1.2 we set the shadow border size to 2 */
3154 if (HasFlag(wxSIMPLE_BORDER
))
3156 /* when using GTK 1.2 we set the simple border size to 1 */
3160 #endif // __WXUNIVERSAL__
3164 GtkScrolledWindow
*scroll_window
= GTK_SCROLLED_WINDOW(m_widget
);
3166 GtkRequisition vscroll_req
;
3167 vscroll_req
.width
= 2;
3168 vscroll_req
.height
= 2;
3169 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
3170 (scroll_window
->vscrollbar
, &vscroll_req
);
3172 GtkRequisition hscroll_req
;
3173 hscroll_req
.width
= 2;
3174 hscroll_req
.height
= 2;
3175 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
3176 (scroll_window
->hscrollbar
, &hscroll_req
);
3178 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
3180 if (scroll_window
->vscrollbar_visible
)
3182 dw
+= vscroll_req
.width
;
3183 dw
+= scroll_class
->scrollbar_spacing
;
3186 if (scroll_window
->hscrollbar_visible
)
3188 dh
+= hscroll_req
.height
;
3189 dh
+= scroll_class
->scrollbar_spacing
;
3193 SetSize( width
+dw
, height
+dh
);
3197 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
3199 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3203 if (width
) (*width
) = m_width
;
3204 if (height
) (*height
) = m_height
;
3211 #ifndef __WXUNIVERSAL__
3212 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3214 /* when using GTK 1.2 we set the shadow border size to 2 */
3218 if (HasFlag(wxSIMPLE_BORDER
))
3220 /* when using GTK 1.2 we set the simple border size to 1 */
3224 #endif // __WXUNIVERSAL__
3228 GtkScrolledWindow
*scroll_window
= GTK_SCROLLED_WINDOW(m_widget
);
3230 GtkRequisition vscroll_req
;
3231 vscroll_req
.width
= 2;
3232 vscroll_req
.height
= 2;
3233 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
3234 (scroll_window
->vscrollbar
, &vscroll_req
);
3236 GtkRequisition hscroll_req
;
3237 hscroll_req
.width
= 2;
3238 hscroll_req
.height
= 2;
3239 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
3240 (scroll_window
->hscrollbar
, &hscroll_req
);
3242 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
3244 if (scroll_window
->vscrollbar_visible
)
3246 dw
+= vscroll_req
.width
;
3247 dw
+= scroll_class
->scrollbar_spacing
;
3250 if (scroll_window
->hscrollbar_visible
)
3252 dh
+= hscroll_req
.height
;
3253 dh
+= scroll_class
->scrollbar_spacing
;
3257 if (width
) (*width
) = m_width
- dw
;
3258 if (height
) (*height
) = m_height
- dh
;
3262 printf( "GetClientSize, name %s ", GetName().c_str() );
3263 if (width) printf( " width = %d", (*width) );
3264 if (height) printf( " height = %d", (*height) );
3269 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
3271 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3275 if (m_parent
&& m_parent
->m_wxwindow
)
3277 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
3278 dx
= pizza
->xoffset
;
3279 dy
= pizza
->yoffset
;
3282 if (x
) (*x
) = m_x
- dx
;
3283 if (y
) (*y
) = m_y
- dy
;
3286 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
3288 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3290 if (!m_widget
->window
) return;
3292 GdkWindow
*source
= (GdkWindow
*) NULL
;
3294 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3296 source
= m_widget
->window
;
3300 gdk_window_get_origin( source
, &org_x
, &org_y
);
3304 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3306 org_x
+= m_widget
->allocation
.x
;
3307 org_y
+= m_widget
->allocation
.y
;
3315 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
3317 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3319 if (!m_widget
->window
) return;
3321 GdkWindow
*source
= (GdkWindow
*) NULL
;
3323 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3325 source
= m_widget
->window
;
3329 gdk_window_get_origin( source
, &org_x
, &org_y
);
3333 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3335 org_x
+= m_widget
->allocation
.x
;
3336 org_y
+= m_widget
->allocation
.y
;
3344 bool wxWindowGTK::Show( bool show
)
3346 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3348 if (!wxWindowBase::Show(show
))
3355 gtk_widget_show( m_widget
);
3357 gtk_widget_hide( m_widget
);
3359 wxShowEvent
eventShow(GetId(), show
);
3360 eventShow
.m_eventObject
= this;
3362 GetEventHandler()->ProcessEvent(eventShow
);
3367 static void wxWindowNotifyEnable(wxWindowGTK
* win
, bool enable
)
3369 win
->OnParentEnable(enable
);
3371 // Recurse, so that children have the opportunity to Do The Right Thing
3372 // and reset colours that have been messed up by a parent's (really ancestor's)
3374 for ( wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
3376 node
= node
->GetNext() )
3378 wxWindow
*child
= node
->GetData();
3379 if (!child
->IsKindOf(CLASSINFO(wxDialog
)) && !child
->IsKindOf(CLASSINFO(wxFrame
)))
3380 wxWindowNotifyEnable(child
, enable
);
3384 bool wxWindowGTK::Enable( bool enable
)
3386 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3388 if (!wxWindowBase::Enable(enable
))
3394 gtk_widget_set_sensitive( m_widget
, enable
);
3396 gtk_widget_set_sensitive( m_wxwindow
, enable
);
3398 wxWindowNotifyEnable(this, enable
);
3403 int wxWindowGTK::GetCharHeight() const
3405 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
3407 wxCHECK_MSG( m_font
.Ok(), 12, wxT("invalid font") );
3409 GdkFont
*font
= m_font
.GetInternalFont( 1.0 );
3411 return font
->ascent
+ font
->descent
;
3414 int wxWindowGTK::GetCharWidth() const
3416 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
3418 wxCHECK_MSG( m_font
.Ok(), 8, wxT("invalid font") );
3420 GdkFont
*font
= m_font
.GetInternalFont( 1.0 );
3422 return gdk_string_width( font
, "H" );
3425 void wxWindowGTK::GetTextExtent( const wxString
& string
,
3429 int *externalLeading
,
3430 const wxFont
*theFont
) const
3432 wxFont fontToUse
= m_font
;
3433 if (theFont
) fontToUse
= *theFont
;
3435 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
3437 if (string
.IsEmpty())
3445 PangoContext
*context
= NULL
;
3447 context
= gtk_widget_get_pango_context( m_widget
);
3456 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
3457 PangoLayout
*layout
= pango_layout_new(context
);
3458 pango_layout_set_font_description(layout
, desc
);
3461 const wxCharBuffer data
= wxConvUTF8
.cWC2MB( string
);
3462 pango_layout_set_text(layout
, (const char*) data
, strlen( (const char*) data
));
3464 const wxWCharBuffer wdata
= wxConvLocal
.cMB2WC( string
);
3465 const wxCharBuffer data
= wxConvUTF8
.cWC2MB( wdata
);
3466 pango_layout_set_text(layout
, (const char*) data
, strlen( (const char*) data
));
3469 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3471 PangoRectangle rect
;
3472 pango_layout_line_get_extents(line
, NULL
, &rect
);
3474 if (x
) (*x
) = (wxCoord
) (rect
.width
/ PANGO_SCALE
);
3475 if (y
) (*y
) = (wxCoord
) (rect
.height
/ PANGO_SCALE
);
3478 // Do something about metrics here
3481 if (externalLeading
) (*externalLeading
) = 0; // ??
3483 g_object_unref( G_OBJECT( layout
) );
3485 GdkFont
*font
= fontToUse
.GetInternalFont( 1.0 );
3486 if (x
) (*x
) = gdk_string_width( font
, wxGTK_CONV( string
) );
3487 if (y
) (*y
) = font
->ascent
+ font
->descent
;
3488 if (descent
) (*descent
) = font
->descent
;
3489 if (externalLeading
) (*externalLeading
) = 0; // ??
3493 void wxWindowGTK::SetFocus()
3495 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3499 // don't do anything if we already have focus
3505 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow
))
3507 gtk_widget_grab_focus (m_wxwindow
);
3512 if (GTK_WIDGET_CAN_FOCUS(m_widget
) && !GTK_WIDGET_HAS_FOCUS (m_widget
) )
3514 if (!GTK_WIDGET_REALIZED(m_widget
))
3516 // we can't set the focus to the widget now so we remember that
3517 // it should be focused and will do it later, during the idle
3518 // time, as soon as we can
3519 wxLogTrace(TRACE_FOCUS
,
3520 _T("Delaying setting focus to %s(%s)"),
3521 GetClassInfo()->GetClassName(), GetLabel().c_str());
3523 g_delayedFocus
= this;
3527 wxLogTrace(TRACE_FOCUS
,
3528 _T("Setting focus to %s(%s)"),
3529 GetClassInfo()->GetClassName(), GetLabel().c_str());
3531 gtk_widget_grab_focus (m_widget
);
3534 else if (GTK_IS_CONTAINER(m_widget
))
3536 SET_CONTAINER_FOCUS( m_widget
, GTK_DIR_TAB_FORWARD
);
3540 wxLogTrace(TRACE_FOCUS
,
3541 _T("Can't set focus to %s(%s)"),
3542 GetClassInfo()->GetClassName(), GetLabel().c_str());
3547 bool wxWindowGTK::AcceptsFocus() const
3549 return m_acceptsFocus
&& wxWindowBase::AcceptsFocus();
3552 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3554 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3556 wxWindowGTK
*oldParent
= m_parent
,
3557 *newParent
= (wxWindowGTK
*)newParentBase
;
3559 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3561 if ( !wxWindowBase::Reparent(newParent
) )
3564 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3566 /* prevent GTK from deleting the widget arbitrarily */
3567 gtk_widget_ref( m_widget
);
3571 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3574 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3578 /* insert GTK representation */
3579 (*(newParent
->m_insertCallback
))(newParent
, this);
3582 /* reverse: prevent GTK from deleting the widget arbitrarily */
3583 gtk_widget_unref( m_widget
);
3588 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3590 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3592 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3594 wxASSERT_MSG( (m_insertCallback
!= NULL
), wxT("invalid child insertion function") );
3599 /* insert GTK representation */
3600 (*m_insertCallback
)(this, child
);
3603 void wxWindowGTK::Raise()
3605 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3607 if (!m_widget
->window
) return;
3609 gdk_window_raise( m_widget
->window
);
3612 void wxWindowGTK::Lower()
3614 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3616 if (!m_widget
->window
) return;
3618 gdk_window_lower( m_widget
->window
);
3621 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3623 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3625 if (cursor
== m_cursor
)
3629 wxapp_install_idle_handler();
3631 if (cursor
== wxNullCursor
)
3632 return wxWindowBase::SetCursor( *wxSTANDARD_CURSOR
);
3634 return wxWindowBase::SetCursor( cursor
);
3637 void wxWindowGTK::WarpPointer( int x
, int y
)
3639 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3641 // We provide this function ourselves as it is
3642 // missing in GDK (top of this file).
3644 GdkWindow
*window
= (GdkWindow
*) NULL
;
3646 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3648 window
= GetConnectWidget()->window
;
3651 gdk_window_warp_pointer( window
, x
, y
);
3655 void wxWindowGTK::Refresh( bool eraseBackground
, const wxRect
*rect
)
3657 if (!m_widget
) return;
3658 if (!m_widget
->window
) return;
3662 wxapp_install_idle_handler();
3664 if (eraseBackground
&& m_wxwindow
&& m_wxwindow
->window
)
3668 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3669 m_clearRegion
.Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
3673 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3674 m_clearRegion
.Clear();
3675 m_clearRegion
.Union( 0, 0, m_wxwindow
->allocation
.width
, m_wxwindow
->allocation
.height
);
3683 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3684 m_updateRegion
.Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
3688 GdkRectangle gdk_rect
;
3689 gdk_rect
.x
= rect
->x
;
3690 gdk_rect
.y
= rect
->y
;
3691 gdk_rect
.width
= rect
->width
;
3692 gdk_rect
.height
= rect
->height
;
3693 gtk_widget_draw( m_widget
, &gdk_rect
);
3700 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3701 m_updateRegion
.Clear();
3702 m_updateRegion
.Union( 0, 0, m_wxwindow
->allocation
.width
, m_wxwindow
->allocation
.height
);
3706 gtk_widget_draw( m_widget
, (GdkRectangle
*) NULL
);
3714 GdkRectangle gdk_rect
;
3715 gdk_rect
.x
= rect
->x
;
3716 gdk_rect
.y
= rect
->y
;
3717 gdk_rect
.width
= rect
->width
;
3718 gdk_rect
.height
= rect
->height
;
3719 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, &gdk_rect
, TRUE
);
3723 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, NULL
, TRUE
);
3729 void wxWindowGTK::Update()
3734 void wxWindowGTK::GtkUpdate()
3737 if (m_wxwindow
&& GTK_PIZZA(m_wxwindow
)->bin_window
)
3738 gdk_window_process_updates( GTK_PIZZA(m_wxwindow
)->bin_window
, FALSE
);
3740 if (!m_updateRegion
.IsEmpty())
3741 GtkSendPaintEvents();
3745 void wxWindowGTK::GtkSendPaintEvents()
3750 m_clearRegion
.Clear();
3752 m_updateRegion
.Clear();
3756 // Clip to paint region in wxClientDC
3757 m_clipPaintRegion
= TRUE
;
3760 // widget to draw on
3761 GtkPizza
*pizza
= GTK_PIZZA (m_wxwindow
);
3763 // later for GTK 2.0, too.
3764 if (GetThemeEnabled())
3766 // find ancestor from which to steal background
3767 wxWindow
*parent
= GetParent();
3768 while (parent
&& !parent
->IsTopLevel())
3769 parent
= parent
->GetParent();
3771 parent
= (wxWindow
*)this;
3773 wxRegionIterator
upd( m_updateRegion
);
3777 rect
.x
= upd
.GetX();
3778 rect
.y
= upd
.GetY();
3779 rect
.width
= upd
.GetWidth();
3780 rect
.height
= upd
.GetHeight();
3782 gtk_paint_flat_box( parent
->m_widget
->style
,
3799 wxWindowDC
dc( (wxWindow
*)this );
3800 dc
.SetClippingRegion( m_updateRegion
);
3802 wxEraseEvent
erase_event( GetId(), &dc
);
3803 erase_event
.SetEventObject( this );
3805 GetEventHandler()->ProcessEvent(erase_event
);
3808 // if (!m_clearRegion.IsEmpty()) // Always send an erase event under GTK 1.2
3810 wxWindowDC
dc( (wxWindow
*)this );
3811 if (m_clearRegion
.IsEmpty())
3812 dc
.SetClippingRegion( m_updateRegion
);
3814 dc
.SetClippingRegion( m_clearRegion
);
3816 wxEraseEvent
erase_event( GetId(), &dc
);
3817 erase_event
.SetEventObject( this );
3819 if (!GetEventHandler()->ProcessEvent(erase_event
))
3823 g_eraseGC
= gdk_gc_new( pizza
->bin_window
);
3824 gdk_gc_set_fill( g_eraseGC
, GDK_SOLID
);
3826 gdk_gc_set_foreground( g_eraseGC
, m_backgroundColour
.GetColor() );
3828 wxRegionIterator
upd( m_clearRegion
);
3831 gdk_draw_rectangle( pizza
->bin_window
, g_eraseGC
, 1,
3832 upd
.GetX(), upd
.GetY(), upd
.GetWidth(), upd
.GetHeight() );
3836 m_clearRegion
.Clear();
3840 wxNcPaintEvent
nc_paint_event( GetId() );
3841 nc_paint_event
.SetEventObject( this );
3842 GetEventHandler()->ProcessEvent( nc_paint_event
);
3844 wxPaintEvent
paint_event( GetId() );
3845 paint_event
.SetEventObject( this );
3846 GetEventHandler()->ProcessEvent( paint_event
);
3848 m_clipPaintRegion
= FALSE
;
3850 #ifndef __WXUNIVERSAL__
3852 // The following code will result in all window-less widgets
3853 // being redrawn because the wxWindows class is allowed to
3854 // paint over the window-less widgets.
3856 GList
*children
= pizza
->children
;
3859 GtkPizzaChild
*child
= (GtkPizzaChild
*) children
->data
;
3860 children
= children
->next
;
3862 if (GTK_WIDGET_NO_WINDOW (child
->widget
) &&
3863 GTK_WIDGET_DRAWABLE (child
->widget
))
3865 // Get intersection of widget area and update region
3866 wxRegion
region( m_updateRegion
);
3868 GdkEventExpose gdk_event
;
3869 gdk_event
.type
= GDK_EXPOSE
;
3870 gdk_event
.window
= pizza
->bin_window
;
3871 gdk_event
.count
= 0;
3873 wxRegionIterator
upd( m_updateRegion
);
3877 rect
.x
= upd
.GetX();
3878 rect
.y
= upd
.GetY();
3879 rect
.width
= upd
.GetWidth();
3880 rect
.height
= upd
.GetHeight();
3882 if (gtk_widget_intersect (child
->widget
, &rect
, &gdk_event
.area
))
3884 gtk_widget_event (child
->widget
, (GdkEvent
*) &gdk_event
);
3894 m_updateRegion
.Clear();
3897 void wxWindowGTK::Clear()
3899 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3902 if (m_wxwindow
&& m_wxwindow
->window
)
3904 m_clearRegion
.Clear();
3905 wxSize
size( GetClientSize() );
3906 m_clearRegion
.Union( 0,0,size
.x
,size
.y
);
3908 // Better do this in idle?
3915 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3917 wxWindowBase::DoSetToolTip(tip
);
3920 m_tooltip
->Apply( (wxWindow
*)this );
3923 void wxWindowGTK::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
3925 gtk_tooltips_set_tip( tips
, GetConnectWidget(), wxConvCurrent
->cWX2MB(tip
), (gchar
*) NULL
);
3927 #endif // wxUSE_TOOLTIPS
3929 void wxWindowGTK::GtkSetBackgroundColour( const wxColour
&colour
)
3931 GdkWindow
*window
= (GdkWindow
*) NULL
;
3933 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3935 window
= GetConnectWidget()->window
;
3939 // We need the pixel value e.g. for background clearing.
3940 m_backgroundColour
.CalcPixel( gdk_window_get_colormap( window
) );
3944 // wxMSW doesn't clear the window here, either.
3945 gdk_window_set_background( window
, m_backgroundColour
.GetColor() );
3951 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
3953 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
3955 if (!wxWindowBase::SetBackgroundColour(colour
))
3958 GdkWindow
*window
= (GdkWindow
*) NULL
;
3960 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3962 window
= GetConnectWidget()->window
;
3966 // indicate that a new style has been set
3967 // but it couldn't get applied as the
3968 // widget hasn't been realized yet.
3969 m_delayedBackgroundColour
= TRUE
;
3974 GtkSetBackgroundColour( colour
);
3980 void wxWindowGTK::GtkSetForegroundColour( const wxColour
&colour
)
3982 GdkWindow
*window
= (GdkWindow
*) NULL
;
3984 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3986 window
= GetConnectWidget()->window
;
3993 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
3995 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
3997 if (!wxWindowBase::SetForegroundColour(colour
))
3999 // don't leave if the GTK widget has just
4001 if (!m_delayedForegroundColour
) return FALSE
;
4004 GdkWindow
*window
= (GdkWindow
*) NULL
;
4006 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4008 window
= GetConnectWidget()->window
;
4012 // indicate that a new style has been set
4013 // but it couldn't get applied as the
4014 // widget hasn't been realized yet.
4015 m_delayedForegroundColour
= TRUE
;
4019 GtkSetForegroundColour( colour
);
4026 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
4028 return gtk_widget_get_pango_context( m_widget
);
4031 PangoContext
*wxWindowGTK::GtkGetPangoX11Context()
4034 m_x11Context
= pango_x_get_context( gdk_display
);
4036 return m_x11Context
;
4040 GtkStyle
*wxWindowGTK::GetWidgetStyle()
4044 GtkStyle
*remake
= gtk_style_copy( m_widgetStyle
);
4046 // FIXME: no more klass in 2.0
4048 remake
->klass
= m_widgetStyle
->klass
;
4051 gtk_style_unref( m_widgetStyle
);
4052 m_widgetStyle
= remake
;
4056 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4059 def
= gtk_widget_get_default_style();
4061 m_widgetStyle
= gtk_style_copy( def
);
4063 // FIXME: no more klass in 2.0
4065 m_widgetStyle
->klass
= def
->klass
;
4069 return m_widgetStyle
;
4072 void wxWindowGTK::SetWidgetStyle()
4074 #if DISABLE_STYLE_IF_BROKEN_THEME
4075 if (m_widget
->style
->engine_data
)
4077 static bool s_warningPrinted
= FALSE
;
4078 if (!s_warningPrinted
)
4080 printf( "wxWindows warning: Widget styles disabled due to buggy GTK theme.\n" );
4081 s_warningPrinted
= TRUE
;
4083 m_widgetStyle
= m_widget
->style
;
4088 GtkStyle
*style
= GetWidgetStyle();
4090 if (m_font
!= wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT
))
4092 SET_STYLE_FONT(style
, m_font
.GetInternalFont( 1.0 ));
4095 if (m_foregroundColour
.Ok())
4097 m_foregroundColour
.CalcPixel( gtk_widget_get_colormap( m_widget
) );
4098 if (m_foregroundColour
!= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNTEXT
))
4100 style
->fg
[GTK_STATE_NORMAL
] = *m_foregroundColour
.GetColor();
4101 style
->fg
[GTK_STATE_PRELIGHT
] = *m_foregroundColour
.GetColor();
4102 style
->fg
[GTK_STATE_ACTIVE
] = *m_foregroundColour
.GetColor();
4106 // Try to restore the gtk default style. This is still a little
4107 // oversimplified for what is probably really needed here for controls
4108 // other than buttons, but is better than not being able to (re)set a
4109 // control's foreground colour to *wxBLACK -- RL
4110 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4113 def
= gtk_widget_get_default_style();
4115 style
->fg
[GTK_STATE_NORMAL
] = def
->fg
[GTK_STATE_NORMAL
];
4116 style
->fg
[GTK_STATE_PRELIGHT
] = def
->fg
[GTK_STATE_PRELIGHT
];
4117 style
->fg
[GTK_STATE_ACTIVE
] = def
->fg
[GTK_STATE_ACTIVE
];
4121 if (m_backgroundColour
.Ok())
4123 m_backgroundColour
.CalcPixel( gtk_widget_get_colormap( m_widget
) );
4124 if (m_backgroundColour
!= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
))
4126 style
->bg
[GTK_STATE_NORMAL
] = *m_backgroundColour
.GetColor();
4127 style
->base
[GTK_STATE_NORMAL
] = *m_backgroundColour
.GetColor();
4128 style
->bg
[GTK_STATE_PRELIGHT
] = *m_backgroundColour
.GetColor();
4129 style
->base
[GTK_STATE_PRELIGHT
] = *m_backgroundColour
.GetColor();
4130 style
->bg
[GTK_STATE_ACTIVE
] = *m_backgroundColour
.GetColor();
4131 style
->base
[GTK_STATE_ACTIVE
] = *m_backgroundColour
.GetColor();
4132 style
->bg
[GTK_STATE_INSENSITIVE
] = *m_backgroundColour
.GetColor();
4133 style
->base
[GTK_STATE_INSENSITIVE
] = *m_backgroundColour
.GetColor();
4137 // Try to restore the gtk default style. This is still a little
4138 // oversimplified for what is probably really needed here for controls
4139 // other than buttons, but is better than not being able to (re)set a
4140 // control's background colour to default grey and means resetting a
4141 // button to wxSYS_COLOUR_BTNFACE will restore its usual highlighting
4143 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4146 def
= gtk_widget_get_default_style();
4148 style
->bg
[GTK_STATE_NORMAL
] = def
->bg
[GTK_STATE_NORMAL
];
4149 style
->base
[GTK_STATE_NORMAL
] = def
->base
[GTK_STATE_NORMAL
];
4150 style
->bg
[GTK_STATE_PRELIGHT
] = def
->bg
[GTK_STATE_PRELIGHT
];
4151 style
->base
[GTK_STATE_PRELIGHT
] = def
->base
[GTK_STATE_PRELIGHT
];
4152 style
->bg
[GTK_STATE_ACTIVE
] = def
->bg
[GTK_STATE_ACTIVE
];
4153 style
->base
[GTK_STATE_ACTIVE
] = def
->base
[GTK_STATE_ACTIVE
];
4154 style
->bg
[GTK_STATE_INSENSITIVE
] = def
->bg
[GTK_STATE_INSENSITIVE
];
4155 style
->base
[GTK_STATE_INSENSITIVE
] = def
->base
[GTK_STATE_INSENSITIVE
];
4160 void wxWindowGTK::ApplyWidgetStyle()
4164 //-----------------------------------------------------------------------------
4165 // Pop-up menu stuff
4166 //-----------------------------------------------------------------------------
4168 #if wxUSE_MENUS_NATIVE
4171 void gtk_pop_hide_callback( GtkWidget
*WXUNUSED(widget
), bool* is_waiting
)
4173 *is_waiting
= FALSE
;
4176 static void SetInvokingWindow( wxMenu
*menu
, wxWindowGTK
*win
)
4178 menu
->SetInvokingWindow( win
);
4179 wxMenuItemList::Node
*node
= menu
->GetMenuItems().GetFirst();
4182 wxMenuItem
*menuitem
= node
->GetData();
4183 if (menuitem
->IsSubMenu())
4185 SetInvokingWindow( menuitem
->GetSubMenu(), win
);
4188 node
= node
->GetNext();
4192 // used to pass the coordinates from wxWindowGTK::DoPopupMenu() to
4193 // wxPopupMenuPositionCallback()
4195 // should be safe even in the MT case as the user can hardly popup 2 menus
4196 // simultaneously, can he?
4197 static gint gs_pop_x
= 0;
4198 static gint gs_pop_y
= 0;
4200 extern "C" void wxPopupMenuPositionCallback( GtkMenu
*menu
,
4203 gboolean
* WXUNUSED(whatever
),
4205 gpointer
WXUNUSED(user_data
) )
4207 // ensure that the menu appears entirely on screen
4209 gtk_widget_get_child_requisition(GTK_WIDGET(menu
), &req
);
4211 wxSize sizeScreen
= wxGetDisplaySize();
4213 gint xmax
= sizeScreen
.x
- req
.width
,
4214 ymax
= sizeScreen
.y
- req
.height
;
4216 *x
= gs_pop_x
< xmax
? gs_pop_x
: xmax
;
4217 *y
= gs_pop_y
< ymax
? gs_pop_y
: ymax
;
4220 bool wxWindowGTK::DoPopupMenu( wxMenu
*menu
, int x
, int y
)
4222 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4224 wxCHECK_MSG( menu
!= NULL
, FALSE
, wxT("invalid popup-menu") );
4226 SetInvokingWindow( menu
, this );
4232 ClientToScreen( &gs_pop_x
, &gs_pop_y
);
4234 bool is_waiting
= TRUE
;
4236 gtk_signal_connect( GTK_OBJECT(menu
->m_menu
),
4238 GTK_SIGNAL_FUNC(gtk_pop_hide_callback
),
4239 (gpointer
)&is_waiting
);
4242 GTK_MENU(menu
->m_menu
),
4243 (GtkWidget
*) NULL
, // parent menu shell
4244 (GtkWidget
*) NULL
, // parent menu item
4245 wxPopupMenuPositionCallback
, // function to position it
4246 NULL
, // client data
4247 0, // button used to activate it
4248 gs_timeLastClick
// the time of activation
4253 while (gtk_events_pending())
4254 gtk_main_iteration();
4260 #endif // wxUSE_MENUS_NATIVE
4262 #if wxUSE_DRAG_AND_DROP
4264 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
4266 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4268 GtkWidget
*dnd_widget
= GetConnectWidget();
4270 if (m_dropTarget
) m_dropTarget
->UnregisterWidget( dnd_widget
);
4272 if (m_dropTarget
) delete m_dropTarget
;
4273 m_dropTarget
= dropTarget
;
4275 if (m_dropTarget
) m_dropTarget
->RegisterWidget( dnd_widget
);
4278 #endif // wxUSE_DRAG_AND_DROP
4280 GtkWidget
* wxWindowGTK::GetConnectWidget()
4282 GtkWidget
*connect_widget
= m_widget
;
4283 if (m_wxwindow
) connect_widget
= m_wxwindow
;
4285 return connect_widget
;
4288 bool wxWindowGTK::IsOwnGtkWindow( GdkWindow
*window
)
4291 return (window
== GTK_PIZZA(m_wxwindow
)->bin_window
);
4293 return (window
== m_widget
->window
);
4296 bool wxWindowGTK::SetFont( const wxFont
&font
)
4298 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4300 if (!wxWindowBase::SetFont(font
))
4305 wxColour sysbg
= wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE
);
4306 if ( sysbg
== m_backgroundColour
)
4308 m_backgroundColour
= wxNullColour
;
4310 m_backgroundColour
= sysbg
;
4320 void wxWindowGTK::DoCaptureMouse()
4322 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4324 GdkWindow
*window
= (GdkWindow
*) NULL
;
4326 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4328 window
= GetConnectWidget()->window
;
4330 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4332 wxCursor
* cursor
= & m_cursor
;
4334 cursor
= wxSTANDARD_CURSOR
;
4336 gdk_pointer_grab( window
, FALSE
,
4338 (GDK_BUTTON_PRESS_MASK
|
4339 GDK_BUTTON_RELEASE_MASK
|
4340 GDK_POINTER_MOTION_HINT_MASK
|
4341 GDK_POINTER_MOTION_MASK
),
4343 cursor
->GetCursor(),
4344 (guint32
)GDK_CURRENT_TIME
);
4345 g_captureWindow
= this;
4346 g_captureWindowHasMouse
= TRUE
;
4349 void wxWindowGTK::DoReleaseMouse()
4351 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4353 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4355 g_captureWindow
= (wxWindowGTK
*) NULL
;
4357 GdkWindow
*window
= (GdkWindow
*) NULL
;
4359 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4361 window
= GetConnectWidget()->window
;
4366 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4370 wxWindow
*wxWindowBase::GetCapture()
4372 return (wxWindow
*)g_captureWindow
;
4375 bool wxWindowGTK::IsRetained() const
4380 void wxWindowGTK::SetScrollbar( int orient
, int pos
, int thumbVisible
,
4381 int range
, bool refresh
)
4383 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4385 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4387 m_hasScrolling
= TRUE
;
4389 if (orient
== wxHORIZONTAL
)
4391 float fpos
= (float)pos
;
4392 float frange
= (float)range
;
4393 float fthumb
= (float)thumbVisible
;
4394 if (fpos
> frange
-fthumb
) fpos
= frange
-fthumb
;
4395 if (fpos
< 0.0) fpos
= 0.0;
4397 if ((fabs(frange
-m_hAdjust
->upper
) < 0.2) &&
4398 (fabs(fthumb
-m_hAdjust
->page_size
) < 0.2))
4400 SetScrollPos( orient
, pos
, refresh
);
4404 m_oldHorizontalPos
= fpos
;
4406 m_hAdjust
->lower
= 0.0;
4407 m_hAdjust
->upper
= frange
;
4408 m_hAdjust
->value
= fpos
;
4409 m_hAdjust
->step_increment
= 1.0;
4410 m_hAdjust
->page_increment
= (float)(wxMax(fthumb
,0));
4411 m_hAdjust
->page_size
= fthumb
;
4415 float fpos
= (float)pos
;
4416 float frange
= (float)range
;
4417 float fthumb
= (float)thumbVisible
;
4418 if (fpos
> frange
-fthumb
) fpos
= frange
-fthumb
;
4419 if (fpos
< 0.0) fpos
= 0.0;
4421 if ((fabs(frange
-m_vAdjust
->upper
) < 0.2) &&
4422 (fabs(fthumb
-m_vAdjust
->page_size
) < 0.2))
4424 SetScrollPos( orient
, pos
, refresh
);
4428 m_oldVerticalPos
= fpos
;
4430 m_vAdjust
->lower
= 0.0;
4431 m_vAdjust
->upper
= frange
;
4432 m_vAdjust
->value
= fpos
;
4433 m_vAdjust
->step_increment
= 1.0;
4434 m_vAdjust
->page_increment
= (float)(wxMax(fthumb
,0));
4435 m_vAdjust
->page_size
= fthumb
;
4438 if (orient
== wxHORIZONTAL
)
4439 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" );
4441 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" );
4444 void wxWindowGTK::SetScrollPos( int orient
, int pos
, bool WXUNUSED(refresh
) )
4446 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4448 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4450 if (orient
== wxHORIZONTAL
)
4452 float fpos
= (float)pos
;
4453 if (fpos
> m_hAdjust
->upper
- m_hAdjust
->page_size
) fpos
= m_hAdjust
->upper
- m_hAdjust
->page_size
;
4454 if (fpos
< 0.0) fpos
= 0.0;
4455 m_oldHorizontalPos
= fpos
;
4457 if (fabs(fpos
-m_hAdjust
->value
) < 0.2) return;
4458 m_hAdjust
->value
= fpos
;
4462 float fpos
= (float)pos
;
4463 if (fpos
> m_vAdjust
->upper
- m_vAdjust
->page_size
) fpos
= m_vAdjust
->upper
- m_vAdjust
->page_size
;
4464 if (fpos
< 0.0) fpos
= 0.0;
4465 m_oldVerticalPos
= fpos
;
4467 if (fabs(fpos
-m_vAdjust
->value
) < 0.2) return;
4468 m_vAdjust
->value
= fpos
;
4471 if (m_wxwindow
->window
)
4473 if (orient
== wxHORIZONTAL
)
4475 gtk_signal_disconnect_by_func( GTK_OBJECT(m_hAdjust
),
4476 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
4478 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "value_changed" );
4480 gtk_signal_connect( GTK_OBJECT(m_hAdjust
), "value_changed",
4481 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
4485 gtk_signal_disconnect_by_func( GTK_OBJECT(m_vAdjust
),
4486 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
4488 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "value_changed" );
4490 gtk_signal_connect( GTK_OBJECT(m_vAdjust
), "value_changed",
4491 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
4496 int wxWindowGTK::GetScrollThumb( int orient
) const
4498 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4500 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4502 if (orient
== wxHORIZONTAL
)
4503 return (int)(m_hAdjust
->page_size
+0.5);
4505 return (int)(m_vAdjust
->page_size
+0.5);
4508 int wxWindowGTK::GetScrollPos( int orient
) const
4510 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4512 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4514 if (orient
== wxHORIZONTAL
)
4515 return (int)(m_hAdjust
->value
+0.5);
4517 return (int)(m_vAdjust
->value
+0.5);
4520 int wxWindowGTK::GetScrollRange( int orient
) const
4522 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4524 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4526 if (orient
== wxHORIZONTAL
)
4527 return (int)(m_hAdjust
->upper
+0.5);
4529 return (int)(m_vAdjust
->upper
+0.5);
4532 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4534 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4536 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4538 // No scrolling requested.
4539 if ((dx
== 0) && (dy
== 0)) return;
4542 if (!m_updateRegion
.IsEmpty())
4544 m_updateRegion
.Offset( dx
, dy
);
4548 GetClientSize( &cw
, &ch
);
4549 m_updateRegion
.Intersect( 0, 0, cw
, ch
);
4552 if (!m_clearRegion
.IsEmpty())
4554 m_clearRegion
.Offset( dx
, dy
);
4558 GetClientSize( &cw
, &ch
);
4559 m_clearRegion
.Intersect( 0, 0, cw
, ch
);
4563 m_clipPaintRegion
= TRUE
;
4565 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), -dx
, -dy
);
4567 m_clipPaintRegion
= FALSE
;
4571 // Find the wxWindow at the current mouse position, also returning the mouse
4573 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4575 pt
= wxGetMousePosition();
4576 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4580 // Get the current mouse position.
4581 wxPoint
wxGetMousePosition()
4583 /* This crashes when used within wxHelpContext,
4584 so we have to use the X-specific implementation below.
4586 GdkModifierType *mask;
4587 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4589 return wxPoint(x, y);
4593 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4595 return wxPoint(-999, -999);
4597 Display
*display
= GDK_WINDOW_XDISPLAY(windowAtPtr
);
4598 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4599 Window rootReturn
, childReturn
;
4600 int rootX
, rootY
, winX
, winY
;
4601 unsigned int maskReturn
;
4603 XQueryPointer (display
,
4607 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4608 return wxPoint(rootX
, rootY
);
4612 // ----------------------------------------------------------------------------
4614 // ----------------------------------------------------------------------------
4616 class wxWinModule
: public wxModule
4623 DECLARE_DYNAMIC_CLASS(wxWinModule
)
4626 IMPLEMENT_DYNAMIC_CLASS(wxWinModule
, wxModule
)
4628 bool wxWinModule::OnInit()
4630 // g_eraseGC = gdk_gc_new( GDK_ROOT_PARENT() );
4631 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
4636 void wxWinModule::OnExit()
4639 gdk_gc_unref( g_eraseGC
);