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 wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
1517 wxWindowGTK
*child
= node
->GetData();
1519 node
= node
->GetNext();
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") );
1699 if (event_type
== wxEVT_LEFT_DCLICK
)
1701 // GTK 1.2 crashes when intercepting double
1702 // click events from both wxSpinButton and
1704 if (GTK_IS_SPIN_BUTTON(win
->m_widget
))
1706 // Just disable this event for now.
1712 if (win
->GetEventHandler()->ProcessEvent( event
))
1714 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "button_press_event" );
1721 //-----------------------------------------------------------------------------
1722 // "button_release_event"
1723 //-----------------------------------------------------------------------------
1725 static gint
gtk_window_button_release_callback( GtkWidget
*widget
,
1726 GdkEventButton
*gdk_event
,
1732 wxapp_install_idle_handler();
1734 if (!win
->m_hasVMT
) return FALSE
;
1735 if (g_blockEventsOnDrag
) return FALSE
;
1736 if (g_blockEventsOnScroll
) return FALSE
;
1738 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
1740 wxEventType event_type
= wxEVT_NULL
;
1742 switch (gdk_event
->button
)
1745 event_type
= wxEVT_LEFT_UP
;
1749 event_type
= wxEVT_MIDDLE_UP
;
1753 event_type
= wxEVT_RIGHT_UP
;
1757 // unknwon button, don't process
1761 wxMouseEvent
event( event_type
);
1762 InitMouseEvent( win
, event
, gdk_event
);
1764 AdjustEventButtonState(event
);
1766 // same wxListBox hack as above
1767 win
->FixUpMouseEvent(widget
, event
.m_x
, event
.m_y
);
1769 if ( event_type
== wxEVT_RIGHT_UP
)
1771 // generate a "context menu" event: this is similar to wxEVT_RIGHT_UP
1774 // (a) it's a command event and so is propagated to the parent
1775 // (b) under MSW it can be generated from kbd too
1776 // (c) it uses screen coords (because of (a))
1777 wxContextMenuEvent
evtCtx(wxEVT_CONTEXT_MENU
,
1779 win
->ClientToScreen(event
.GetPosition()));
1780 (void)win
->GetEventHandler()->ProcessEvent(evtCtx
);
1783 if ( !g_captureWindow
)
1784 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1786 if (win
->GetEventHandler()->ProcessEvent( event
))
1788 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "button_release_event" );
1795 //-----------------------------------------------------------------------------
1796 // "motion_notify_event"
1797 //-----------------------------------------------------------------------------
1799 static gint
gtk_window_motion_notify_callback( GtkWidget
*widget
,
1800 GdkEventMotion
*gdk_event
,
1806 wxapp_install_idle_handler();
1808 if (!win
->m_hasVMT
) return FALSE
;
1809 if (g_blockEventsOnDrag
) return FALSE
;
1810 if (g_blockEventsOnScroll
) return FALSE
;
1812 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
1814 if (gdk_event
->is_hint
)
1818 GdkModifierType state
;
1819 gdk_window_get_pointer(gdk_event
->window
, &x
, &y
, &state
);
1825 printf( "OnMotion from " );
1826 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1827 printf( win->GetClassInfo()->GetClassName() );
1831 wxMouseEvent
event( wxEVT_MOTION
);
1832 InitMouseEvent(win
, event
, gdk_event
);
1834 if ( g_captureWindow
)
1836 // synthetize a mouse enter or leave event if needed
1837 GdkWindow
*winUnderMouse
= gdk_window_at_pointer(NULL
, NULL
);
1838 // This seems to be necessary and actually been added to
1839 // GDK itself in version 2.0.X
1842 bool hasMouse
= winUnderMouse
== gdk_event
->window
;
1843 if ( hasMouse
!= g_captureWindowHasMouse
)
1845 // the mouse changed window
1846 g_captureWindowHasMouse
= hasMouse
;
1848 wxMouseEvent
event(g_captureWindowHasMouse
? wxEVT_ENTER_WINDOW
1849 : wxEVT_LEAVE_WINDOW
);
1850 InitMouseEvent(win
, event
, gdk_event
);
1851 event
.SetEventObject(win
);
1852 win
->GetEventHandler()->ProcessEvent(event
);
1857 win
= FindWindowForMouseEvent(win
, event
.m_x
, event
.m_y
);
1860 if (win
->GetEventHandler()->ProcessEvent( event
))
1862 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "motion_notify_event" );
1869 //-----------------------------------------------------------------------------
1871 //-----------------------------------------------------------------------------
1873 // send the wxChildFocusEvent and wxFocusEvent, common code of
1874 // gtk_window_focus_in_callback() and SetFocus()
1875 static bool DoSendFocusEvents(wxWindow
*win
)
1877 // Notify the parent keeping track of focus for the kbd navigation
1878 // purposes that we got it.
1879 wxChildFocusEvent
eventChildFocus(win
);
1880 (void)win
->GetEventHandler()->ProcessEvent(eventChildFocus
);
1882 wxFocusEvent
eventFocus(wxEVT_SET_FOCUS
, win
->GetId());
1883 eventFocus
.SetEventObject(win
);
1885 return win
->GetEventHandler()->ProcessEvent(eventFocus
);
1888 static gint
gtk_window_focus_in_callback( GtkWidget
*widget
,
1889 GdkEvent
*WXUNUSED(event
),
1895 wxapp_install_idle_handler();
1897 if (!win
->m_hasVMT
) return FALSE
;
1898 if (g_blockEventsOnDrag
) return FALSE
;
1900 switch ( g_sendActivateEvent
)
1903 // we've got focus from outside, synthetize wxActivateEvent
1904 g_sendActivateEvent
= 1;
1908 // another our window just lost focus, it was already ours before
1909 // - don't send any wxActivateEvent
1910 g_sendActivateEvent
= -1;
1915 g_focusWindow
= win
;
1917 wxLogTrace(TRACE_FOCUS
,
1918 _T("%s: focus in"), win
->GetName().c_str());
1922 gdk_im_begin(win
->m_ic
, win
->m_wxwindow
->window
);
1926 // caret needs to be informed about focus change
1927 wxCaret
*caret
= win
->GetCaret();
1930 caret
->OnSetFocus();
1932 #endif // wxUSE_CARET
1934 g_activeFrameLostFocus
= FALSE
;
1936 wxWindowGTK
*active
= wxGetTopLevelParent(win
);
1937 if ( active
!= g_activeFrame
)
1939 if ( g_activeFrame
)
1941 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from focus_in)"), g_activeFrame
);
1942 wxActivateEvent
event(wxEVT_ACTIVATE
, FALSE
, g_activeFrame
->GetId());
1943 event
.SetEventObject(g_activeFrame
);
1944 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
1947 wxLogTrace(wxT("activate"), wxT("Activating frame %p (from focus_in)"), active
);
1948 g_activeFrame
= active
;
1949 wxActivateEvent
event(wxEVT_ACTIVATE
, TRUE
, g_activeFrame
->GetId());
1950 event
.SetEventObject(g_activeFrame
);
1951 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
1953 // Don't send focus events in addition to activate
1954 // if (win == g_activeFrame)
1958 // does the window itself think that it has the focus?
1959 if ( !win
->m_hasFocus
)
1961 // not yet, notify it
1962 win
->m_hasFocus
= TRUE
;
1964 if ( DoSendFocusEvents(win
) )
1966 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "focus_in_event" );
1974 //-----------------------------------------------------------------------------
1975 // "focus_out_event"
1976 //-----------------------------------------------------------------------------
1978 static gint
gtk_window_focus_out_callback( GtkWidget
*widget
, GdkEventFocus
*gdk_event
, wxWindowGTK
*win
)
1983 wxapp_install_idle_handler();
1985 if (!win
->m_hasVMT
) return FALSE
;
1986 if (g_blockEventsOnDrag
) return FALSE
;
1988 wxLogTrace( TRACE_FOCUS
,
1989 _T("%s: focus out"), win
->GetName().c_str() );
1991 if ( !g_activeFrameLostFocus
&& g_activeFrame
)
1993 // VZ: commenting this out because it does happen (although not easy
1994 // to reproduce, I only see it when using wxMiniFrame and not
1995 // always) and makes using Mahogany quite annoying
1997 wxASSERT_MSG( wxGetTopLevelParent(win
) == g_activeFrame
,
1998 wxT("unfocusing window that hasn't gained focus properly") );
2001 g_activeFrameLostFocus
= TRUE
;
2004 // if the focus goes out of our app alltogether, OnIdle() will send
2005 // wxActivateEvent, otherwise gtk_window_focus_in_callback() will reset
2006 // g_sendActivateEvent to -1
2007 g_sendActivateEvent
= 0;
2009 wxWindowGTK
*winFocus
= wxFindFocusedChild(win
);
2013 g_focusWindow
= (wxWindowGTK
*)NULL
;
2021 // caret needs to be informed about focus change
2022 wxCaret
*caret
= win
->GetCaret();
2025 caret
->OnKillFocus();
2027 #endif // wxUSE_CARET
2029 // don't send the window a kill focus event if it thinks that it doesn't
2030 // have focus already
2031 if ( win
->m_hasFocus
)
2033 win
->m_hasFocus
= FALSE
;
2035 wxFocusEvent
event( wxEVT_KILL_FOCUS
, win
->GetId() );
2036 event
.SetEventObject( win
);
2038 if (win
->GetEventHandler()->ProcessEvent( event
))
2040 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "focus_out_event" );
2048 //-----------------------------------------------------------------------------
2049 // "enter_notify_event"
2050 //-----------------------------------------------------------------------------
2053 gint
gtk_window_enter_callback( GtkWidget
*widget
,
2054 GdkEventCrossing
*gdk_event
,
2060 wxapp_install_idle_handler();
2062 if (!win
->m_hasVMT
) return FALSE
;
2063 if (g_blockEventsOnDrag
) return FALSE
;
2065 // Event was emitted after a grab
2066 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
2068 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
2072 GdkModifierType state
= (GdkModifierType
)0;
2074 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
2076 wxMouseEvent
event( wxEVT_ENTER_WINDOW
);
2077 InitMouseEvent(win
, event
, gdk_event
);
2078 wxPoint pt
= win
->GetClientAreaOrigin();
2079 event
.m_x
= x
+ pt
.x
;
2080 event
.m_y
= y
+ pt
.y
;
2082 if (win
->GetEventHandler()->ProcessEvent( event
))
2084 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "enter_notify_event" );
2091 //-----------------------------------------------------------------------------
2092 // "leave_notify_event"
2093 //-----------------------------------------------------------------------------
2095 static gint
gtk_window_leave_callback( GtkWidget
*widget
, GdkEventCrossing
*gdk_event
, wxWindowGTK
*win
)
2100 wxapp_install_idle_handler();
2102 if (!win
->m_hasVMT
) return FALSE
;
2103 if (g_blockEventsOnDrag
) return FALSE
;
2105 // Event was emitted after an ungrab
2106 if (gdk_event
->mode
!= GDK_CROSSING_NORMAL
) return FALSE
;
2108 if (!win
->IsOwnGtkWindow( gdk_event
->window
)) return FALSE
;
2110 wxMouseEvent
event( wxEVT_LEAVE_WINDOW
);
2111 event
.SetTimestamp( gdk_event
->time
);
2112 event
.SetEventObject( win
);
2116 GdkModifierType state
= (GdkModifierType
)0;
2118 gdk_window_get_pointer( widget
->window
, &x
, &y
, &state
);
2120 event
.m_shiftDown
= (state
& GDK_SHIFT_MASK
) != 0;
2121 event
.m_controlDown
= (state
& GDK_CONTROL_MASK
) != 0;
2122 event
.m_altDown
= (state
& GDK_MOD1_MASK
) != 0;
2123 event
.m_metaDown
= (state
& GDK_MOD2_MASK
) != 0;
2124 event
.m_leftDown
= (state
& GDK_BUTTON1_MASK
) != 0;
2125 event
.m_middleDown
= (state
& GDK_BUTTON2_MASK
) != 0;
2126 event
.m_rightDown
= (state
& GDK_BUTTON3_MASK
) != 0;
2128 wxPoint pt
= win
->GetClientAreaOrigin();
2129 event
.m_x
= x
+ pt
.x
;
2130 event
.m_y
= y
+ pt
.y
;
2132 if (win
->GetEventHandler()->ProcessEvent( event
))
2134 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "leave_notify_event" );
2141 //-----------------------------------------------------------------------------
2142 // "value_changed" from m_vAdjust
2143 //-----------------------------------------------------------------------------
2145 static void gtk_window_vscroll_callback( GtkAdjustment
*adjust
,
2152 wxapp_install_idle_handler();
2154 if (g_blockEventsOnDrag
) return;
2156 if (!win
->m_hasVMT
) return;
2158 float diff
= adjust
->value
- win
->m_oldVerticalPos
;
2159 if (fabs(diff
) < 0.2) return;
2161 win
->m_oldVerticalPos
= adjust
->value
;
2164 GtkScrolledWindow
*sw
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2166 wxEventType command
= GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw
->vscrollbar
));
2168 int value
= (int)(adjust
->value
+0.5);
2170 wxScrollWinEvent
event( command
, value
, wxVERTICAL
);
2171 event
.SetEventObject( win
);
2172 win
->GetEventHandler()->ProcessEvent( event
);
2175 //-----------------------------------------------------------------------------
2176 // "value_changed" from m_hAdjust
2177 //-----------------------------------------------------------------------------
2179 static void gtk_window_hscroll_callback( GtkAdjustment
*adjust
,
2186 wxapp_install_idle_handler();
2188 if (g_blockEventsOnDrag
) return;
2189 if (!win
->m_hasVMT
) return;
2191 float diff
= adjust
->value
- win
->m_oldHorizontalPos
;
2192 if (fabs(diff
) < 0.2) return;
2195 GtkScrolledWindow
*sw
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2197 wxEventType command
= GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw
->hscrollbar
));
2199 win
->m_oldHorizontalPos
= adjust
->value
;
2201 int value
= (int)(adjust
->value
+0.5);
2203 wxScrollWinEvent
event( command
, value
, wxHORIZONTAL
);
2204 event
.SetEventObject( win
);
2205 win
->GetEventHandler()->ProcessEvent( event
);
2208 //-----------------------------------------------------------------------------
2209 // "button_press_event" from scrollbar
2210 //-----------------------------------------------------------------------------
2212 static gint
gtk_scrollbar_button_press_callback( GtkRange
*widget
,
2213 GdkEventButton
*gdk_event
,
2219 wxapp_install_idle_handler();
2222 g_blockEventsOnScroll
= TRUE
;
2224 // FIXME: there is no 'slider' field in GTK+ 2.0 any more
2226 win
->m_isScrolling
= (gdk_event
->window
== widget
->slider
);
2232 //-----------------------------------------------------------------------------
2233 // "button_release_event" from scrollbar
2234 //-----------------------------------------------------------------------------
2236 static gint
gtk_scrollbar_button_release_callback( GtkRange
*widget
,
2237 GdkEventButton
*WXUNUSED(gdk_event
),
2242 // don't test here as we can release the mouse while being over
2243 // a different window than the slider
2245 // if (gdk_event->window != widget->slider) return FALSE;
2247 g_blockEventsOnScroll
= FALSE
;
2249 if (win
->m_isScrolling
)
2251 wxEventType command
= wxEVT_SCROLLWIN_THUMBRELEASE
;
2255 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(win
->m_widget
);
2256 if (widget
== GTK_RANGE(scrolledWindow
->hscrollbar
))
2258 value
= (int)(win
->m_hAdjust
->value
+0.5);
2261 if (widget
== GTK_RANGE(scrolledWindow
->vscrollbar
))
2263 value
= (int)(win
->m_vAdjust
->value
+0.5);
2267 wxScrollWinEvent
event( command
, value
, dir
);
2268 event
.SetEventObject( win
);
2269 win
->GetEventHandler()->ProcessEvent( event
);
2272 win
->m_isScrolling
= FALSE
;
2277 // ----------------------------------------------------------------------------
2278 // this wxWindowBase function is implemented here (in platform-specific file)
2279 // because it is static and so couldn't be made virtual
2280 // ----------------------------------------------------------------------------
2282 wxWindow
*wxWindowBase::FindFocus()
2284 // the cast is necessary when we compile in wxUniversal mode
2285 return (wxWindow
*)g_focusWindow
;
2289 //-----------------------------------------------------------------------------
2290 // "realize" from m_widget
2291 //-----------------------------------------------------------------------------
2293 /* We cannot set colours and fonts before the widget has
2294 been realized, so we do this directly after realization. */
2297 gtk_window_realized_callback( GtkWidget
*m_widget
, wxWindow
*win
)
2302 wxapp_install_idle_handler();
2304 if (win
->m_delayedBackgroundColour
)
2305 win
->GtkSetBackgroundColour( win
->GetBackgroundColour() );
2307 if (win
->m_delayedForegroundColour
)
2308 win
->GtkSetForegroundColour( win
->GetForegroundColour() );
2311 if (win
->m_imContext
)
2313 GtkPizza
*pizza
= GTK_PIZZA( m_widget
);
2314 gtk_im_context_set_client_window( (GtkIMContext
*) win
->m_imContext
, pizza
->bin_window
);
2318 wxWindowCreateEvent
event( win
);
2319 event
.SetEventObject( win
);
2320 win
->GetEventHandler()->ProcessEvent( event
);
2325 //-----------------------------------------------------------------------------
2327 //-----------------------------------------------------------------------------
2330 void gtk_window_size_callback( GtkWidget
*WXUNUSED(widget
),
2331 GtkAllocation
*WXUNUSED(alloc
),
2335 wxapp_install_idle_handler();
2337 if (!win
->m_hasScrolling
) return;
2339 int client_width
= 0;
2340 int client_height
= 0;
2341 win
->GetClientSize( &client_width
, &client_height
);
2342 if ((client_width
== win
->m_oldClientWidth
) && (client_height
== win
->m_oldClientHeight
))
2345 win
->m_oldClientWidth
= client_width
;
2346 win
->m_oldClientHeight
= client_height
;
2348 if (!win
->m_nativeSizeEvent
)
2350 wxSizeEvent
event( win
->GetSize(), win
->GetId() );
2351 event
.SetEventObject( win
);
2352 win
->GetEventHandler()->ProcessEvent( event
);
2358 #define WXUNUSED_UNLESS_XIM(param) param
2360 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2363 /* Resize XIM window */
2366 void gtk_wxwindow_size_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2367 GtkAllocation
* WXUNUSED_UNLESS_XIM(alloc
),
2368 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2371 wxapp_install_idle_handler();
2377 if (gdk_ic_get_style (win
->m_ic
) & GDK_IM_PREEDIT_POSITION
)
2381 gdk_window_get_size (widget
->window
, &width
, &height
);
2382 win
->m_icattr
->preedit_area
.width
= width
;
2383 win
->m_icattr
->preedit_area
.height
= height
;
2384 gdk_ic_set_attr (win
->m_ic
, win
->m_icattr
, GDK_IC_PREEDIT_AREA
);
2389 //-----------------------------------------------------------------------------
2390 // "realize" from m_wxwindow
2391 //-----------------------------------------------------------------------------
2393 /* Initialize XIM support */
2396 gtk_wxwindow_realized_callback( GtkWidget
* WXUNUSED_UNLESS_XIM(widget
),
2397 wxWindowGTK
* WXUNUSED_UNLESS_XIM(win
) )
2400 wxapp_install_idle_handler();
2403 if (win
->m_ic
) return FALSE
;
2404 if (!widget
) return FALSE
;
2405 if (!gdk_im_ready()) return FALSE
;
2407 win
->m_icattr
= gdk_ic_attr_new();
2408 if (!win
->m_icattr
) return FALSE
;
2412 GdkColormap
*colormap
;
2413 GdkICAttr
*attr
= win
->m_icattr
;
2414 unsigned attrmask
= GDK_IC_ALL_REQ
;
2416 GdkIMStyle supported_style
= (GdkIMStyle
)
2417 (GDK_IM_PREEDIT_NONE
|
2418 GDK_IM_PREEDIT_NOTHING
|
2419 GDK_IM_PREEDIT_POSITION
|
2420 GDK_IM_STATUS_NONE
|
2421 GDK_IM_STATUS_NOTHING
);
2423 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2424 supported_style
= (GdkIMStyle
)(supported_style
& ~GDK_IM_PREEDIT_POSITION
);
2426 attr
->style
= style
= gdk_im_decide_style (supported_style
);
2427 attr
->client_window
= widget
->window
;
2429 if ((colormap
= gtk_widget_get_colormap (widget
)) !=
2430 gtk_widget_get_default_colormap ())
2432 attrmask
|= GDK_IC_PREEDIT_COLORMAP
;
2433 attr
->preedit_colormap
= colormap
;
2436 attrmask
|= GDK_IC_PREEDIT_FOREGROUND
;
2437 attrmask
|= GDK_IC_PREEDIT_BACKGROUND
;
2438 attr
->preedit_foreground
= widget
->style
->fg
[GTK_STATE_NORMAL
];
2439 attr
->preedit_background
= widget
->style
->base
[GTK_STATE_NORMAL
];
2441 switch (style
& GDK_IM_PREEDIT_MASK
)
2443 case GDK_IM_PREEDIT_POSITION
:
2444 if (widget
->style
&& widget
->style
->font
->type
!= GDK_FONT_FONTSET
)
2446 g_warning ("over-the-spot style requires fontset");
2450 gdk_window_get_size (widget
->window
, &width
, &height
);
2452 attrmask
|= GDK_IC_PREEDIT_POSITION_REQ
;
2453 attr
->spot_location
.x
= 0;
2454 attr
->spot_location
.y
= height
;
2455 attr
->preedit_area
.x
= 0;
2456 attr
->preedit_area
.y
= 0;
2457 attr
->preedit_area
.width
= width
;
2458 attr
->preedit_area
.height
= height
;
2459 attr
->preedit_fontset
= widget
->style
->font
;
2464 win
->m_ic
= gdk_ic_new (attr
, (GdkICAttributesType
)attrmask
);
2466 if (win
->m_ic
== NULL
)
2467 g_warning ("Can't create input context.");
2470 mask
= gdk_window_get_events (widget
->window
);
2471 mask
= (GdkEventMask
)(mask
| gdk_ic_get_events (win
->m_ic
));
2472 gdk_window_set_events (widget
->window
, mask
);
2474 if (GTK_WIDGET_HAS_FOCUS(widget
))
2475 gdk_im_begin (win
->m_ic
, widget
->window
);
2482 //-----------------------------------------------------------------------------
2483 // InsertChild for wxWindowGTK.
2484 //-----------------------------------------------------------------------------
2486 /* Callback for wxWindowGTK. This very strange beast has to be used because
2487 * C++ has no virtual methods in a constructor. We have to emulate a
2488 * virtual function here as wxNotebook requires a different way to insert
2489 * a child in it. I had opted for creating a wxNotebookPage window class
2490 * which would have made this superfluous (such in the MDI window system),
2491 * but no-one was listening to me... */
2493 static void wxInsertChildInWindow( wxWindowGTK
* parent
, wxWindowGTK
* child
)
2495 /* the window might have been scrolled already, do we
2496 have to adapt the position */
2497 GtkPizza
*pizza
= GTK_PIZZA(parent
->m_wxwindow
);
2498 child
->m_x
+= pizza
->xoffset
;
2499 child
->m_y
+= pizza
->yoffset
;
2501 gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
),
2502 GTK_WIDGET(child
->m_widget
),
2509 //-----------------------------------------------------------------------------
2511 //-----------------------------------------------------------------------------
2513 wxWindow
*wxGetActiveWindow()
2515 return wxWindow::FindFocus();
2518 //-----------------------------------------------------------------------------
2520 //-----------------------------------------------------------------------------
2522 // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2524 #ifdef __WXUNIVERSAL__
2525 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK
, wxWindowBase
)
2527 IMPLEMENT_DYNAMIC_CLASS(wxWindow
, wxWindowBase
)
2528 #endif // __WXUNIVERSAL__/__WXGTK__
2530 void wxWindowGTK::Init()
2536 m_widget
= (GtkWidget
*) NULL
;
2537 m_wxwindow
= (GtkWidget
*) NULL
;
2538 m_focusWidget
= (GtkWidget
*) NULL
;
2548 m_needParent
= TRUE
;
2549 m_isBeingDeleted
= FALSE
;
2552 m_nativeSizeEvent
= FALSE
;
2554 m_hasScrolling
= FALSE
;
2555 m_isScrolling
= FALSE
;
2557 m_hAdjust
= (GtkAdjustment
*) NULL
;
2558 m_vAdjust
= (GtkAdjustment
*) NULL
;
2559 m_oldHorizontalPos
= 0.0;
2560 m_oldVerticalPos
= 0.0;
2563 m_widgetStyle
= (GtkStyle
*) NULL
;
2565 m_insertCallback
= (wxInsertChildFunction
) NULL
;
2567 m_acceptsFocus
= FALSE
;
2570 m_clipPaintRegion
= FALSE
;
2572 m_cursor
= *wxSTANDARD_CURSOR
;
2574 m_delayedForegroundColour
= FALSE
;
2575 m_delayedBackgroundColour
= FALSE
;
2579 m_x11Context
= NULL
;
2582 m_ic
= (GdkIC
*) NULL
;
2583 m_icattr
= (GdkICAttr
*) NULL
;
2588 wxWindowGTK::wxWindowGTK()
2593 wxWindowGTK::wxWindowGTK( wxWindow
*parent
,
2598 const wxString
&name
)
2602 Create( parent
, id
, pos
, size
, style
, name
);
2605 bool wxWindowGTK::Create( wxWindow
*parent
,
2610 const wxString
&name
)
2612 if (!PreCreation( parent
, pos
, size
) ||
2613 !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name
))
2615 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
2619 m_insertCallback
= wxInsertChildInWindow
;
2621 // always needed for background clearing
2622 m_delayedBackgroundColour
= TRUE
;
2624 m_widget
= gtk_scrolled_window_new( (GtkAdjustment
*) NULL
, (GtkAdjustment
*) NULL
);
2625 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS
);
2627 GtkScrolledWindow
*scrolledWindow
= GTK_SCROLLED_WINDOW(m_widget
);
2629 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
2630 scroll_class
->scrollbar_spacing
= 0;
2632 gtk_scrolled_window_set_policy( scrolledWindow
, GTK_POLICY_AUTOMATIC
, GTK_POLICY_AUTOMATIC
);
2634 m_hAdjust
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->hscrollbar
) );
2635 m_vAdjust
= gtk_range_get_adjustment( GTK_RANGE(scrolledWindow
->vscrollbar
) );
2637 m_wxwindow
= gtk_pizza_new();
2639 #ifndef __WXUNIVERSAL__
2640 GtkPizza
*pizza
= GTK_PIZZA(m_wxwindow
);
2642 if (HasFlag(wxRAISED_BORDER
))
2644 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_OUT
);
2646 else if (HasFlag(wxSUNKEN_BORDER
))
2648 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_IN
);
2650 else if (HasFlag(wxSIMPLE_BORDER
))
2652 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_THIN
);
2656 gtk_pizza_set_shadow_type( pizza
, GTK_MYSHADOW_NONE
);
2658 #endif // __WXUNIVERSAL__
2660 gtk_container_add( GTK_CONTAINER(m_widget
), m_wxwindow
);
2662 GTK_WIDGET_SET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS
);
2663 m_acceptsFocus
= TRUE
;
2665 // I _really_ don't want scrollbars in the beginning
2666 m_vAdjust
->lower
= 0.0;
2667 m_vAdjust
->upper
= 1.0;
2668 m_vAdjust
->value
= 0.0;
2669 m_vAdjust
->step_increment
= 1.0;
2670 m_vAdjust
->page_increment
= 1.0;
2671 m_vAdjust
->page_size
= 5.0;
2672 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" );
2673 m_hAdjust
->lower
= 0.0;
2674 m_hAdjust
->upper
= 1.0;
2675 m_hAdjust
->value
= 0.0;
2676 m_hAdjust
->step_increment
= 1.0;
2677 m_hAdjust
->page_increment
= 1.0;
2678 m_hAdjust
->page_size
= 5.0;
2679 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" );
2681 // these handlers block mouse events to any window during scrolling such as
2682 // motion events and prevent GTK and wxWindows from fighting over where the
2685 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_press_event",
2686 (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this );
2688 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_press_event",
2689 (GtkSignalFunc
)gtk_scrollbar_button_press_callback
, (gpointer
) this );
2691 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->vscrollbar
), "button_release_event",
2692 (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this );
2694 gtk_signal_connect( GTK_OBJECT(scrolledWindow
->hscrollbar
), "button_release_event",
2695 (GtkSignalFunc
)gtk_scrollbar_button_release_callback
, (gpointer
) this );
2697 // these handlers get notified when screen updates are required either when
2698 // scrolling or when the window size (and therefore scrollbar configuration)
2701 gtk_signal_connect( GTK_OBJECT(m_hAdjust
), "value_changed",
2702 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
2703 gtk_signal_connect( GTK_OBJECT(m_vAdjust
), "value_changed",
2704 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
2707 // Create input method handler
2708 m_imContext
= (GtkIMMulticontext
*) gtk_im_multicontext_new ();
2710 // Cannot handle drawing preedited text yet
2711 gtk_im_context_set_use_preedit( (GtkIMContext
*) m_imContext
, FALSE
);
2713 g_signal_connect (G_OBJECT (m_imContext
), "commit",
2714 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2717 gtk_widget_show( m_wxwindow
);
2720 m_parent
->DoAddChild( this );
2722 m_focusWidget
= m_wxwindow
;
2731 wxWindowGTK::~wxWindowGTK()
2735 if (g_focusWindow
== this)
2736 g_focusWindow
= NULL
;
2738 if (g_activeFrame
== this)
2739 g_activeFrame
= NULL
;
2741 if ( g_delayedFocus
== this )
2742 g_delayedFocus
= NULL
;
2744 m_isBeingDeleted
= TRUE
;
2753 m_parent
->RemoveChild( this );
2757 gdk_ic_destroy (m_ic
);
2759 gdk_ic_attr_destroy (m_icattr
);
2764 #if DISABLE_STYLE_IF_BROKEN_THEME
2765 // don't delete if it's a pixmap theme style
2766 if (!m_widgetStyle
->engine_data
)
2767 gtk_style_unref( m_widgetStyle
);
2769 m_widgetStyle
= (GtkStyle
*) NULL
;
2774 gtk_widget_destroy( m_wxwindow
);
2775 m_wxwindow
= (GtkWidget
*) NULL
;
2780 gtk_widget_destroy( m_widget
);
2781 m_widget
= (GtkWidget
*) NULL
;
2785 bool wxWindowGTK::PreCreation( wxWindowGTK
*parent
, const wxPoint
&pos
, const wxSize
&size
)
2787 wxCHECK_MSG( !m_needParent
|| parent
, FALSE
, wxT("Need complete parent.") );
2789 // This turns -1 into 30 so that a minimal window is
2790 // visible even although -1,-1 has been given as the
2791 // size of the window. the same trick is used in other
2792 // ports and should make debugging easier.
2793 m_width
= WidthDefault(size
.x
) ;
2794 m_height
= HeightDefault(size
.y
);
2799 // some reasonable defaults
2804 m_x
= (gdk_screen_width () - m_width
) / 2;
2805 if (m_x
< 10) m_x
= 10;
2809 m_y
= (gdk_screen_height () - m_height
) / 2;
2810 if (m_y
< 10) m_y
= 10;
2817 void wxWindowGTK::PostCreation()
2819 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2825 // these get reported to wxWindows -> wxPaintEvent
2827 gtk_pizza_set_external( GTK_PIZZA(m_wxwindow
), TRUE
);
2829 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "expose_event",
2830 GTK_SIGNAL_FUNC(gtk_window_expose_callback
), (gpointer
)this );
2833 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "draw",
2834 GTK_SIGNAL_FUNC(gtk_window_draw_callback
), (gpointer
)this );
2836 if (HasFlag(wxNO_FULL_REPAINT_ON_RESIZE
))
2838 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "event",
2839 GTK_SIGNAL_FUNC(gtk_window_event_event_callback
), (gpointer
)this );
2842 // gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow), HasFlag( wxNO_FULL_REPAINT_ON_RESIZE ) );
2846 // Create input method handler
2847 m_imContext
= (GtkIMMulticontext
*) gtk_im_multicontext_new ();
2849 // Cannot handle drawing preedited text yet
2850 gtk_im_context_set_use_preedit( (GtkIMContext
*) m_imContext
, FALSE
);
2852 g_signal_connect (G_OBJECT (m_imContext
), "commit",
2853 G_CALLBACK (gtk_wxwindow_commit_cb
), this);
2857 // these are called when the "sunken" or "raised" borders are drawn
2858 gtk_signal_connect( GTK_OBJECT(m_widget
), "expose_event",
2859 GTK_SIGNAL_FUNC(gtk_window_own_expose_callback
), (gpointer
)this );
2862 gtk_signal_connect( GTK_OBJECT(m_widget
), "draw",
2863 GTK_SIGNAL_FUNC(gtk_window_own_draw_callback
), (gpointer
)this );
2869 if (m_focusWidget
== NULL
)
2870 m_focusWidget
= m_widget
;
2872 gtk_signal_connect( GTK_OBJECT(m_focusWidget
), "focus_in_event",
2873 GTK_SIGNAL_FUNC(gtk_window_focus_in_callback
), (gpointer
)this );
2875 gtk_signal_connect( GTK_OBJECT(m_focusWidget
), "focus_out_event",
2876 GTK_SIGNAL_FUNC(gtk_window_focus_out_callback
), (gpointer
)this );
2878 // connect to the various key and mouse handlers
2880 GtkWidget
*connect_widget
= GetConnectWidget();
2882 ConnectWidget( connect_widget
);
2884 /* We cannot set colours, fonts and cursors before the widget has
2885 been realized, so we do this directly after realization */
2886 gtk_signal_connect( GTK_OBJECT(connect_widget
), "realize",
2887 GTK_SIGNAL_FUNC(gtk_window_realized_callback
), (gpointer
) this );
2891 // Catch native resize events
2892 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "size_allocate",
2893 GTK_SIGNAL_FUNC(gtk_window_size_callback
), (gpointer
)this );
2895 // Initialize XIM support
2896 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "realize",
2897 GTK_SIGNAL_FUNC(gtk_wxwindow_realized_callback
), (gpointer
) this );
2899 // And resize XIM window
2900 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "size_allocate",
2901 GTK_SIGNAL_FUNC(gtk_wxwindow_size_callback
), (gpointer
)this );
2904 if (!GTK_IS_COMBO(m_widget
))
2906 // This is needed if we want to add our windows into native
2907 // GTK control, such as the toolbar. With this callback, the
2908 // toolbar gets to know the correct size (the one set by the
2909 // programmer). Sadly, it misbehaves for wxComboBox. FIXME
2910 // when moving to GTK 2.0.
2911 gtk_signal_connect( GTK_OBJECT(m_widget
), "size_request",
2912 GTK_SIGNAL_FUNC(gtk_window_size_request_callback
), (gpointer
) this );
2918 void wxWindowGTK::ConnectWidget( GtkWidget
*widget
)
2920 gtk_signal_connect( GTK_OBJECT(widget
), "key_press_event",
2921 GTK_SIGNAL_FUNC(gtk_window_key_press_callback
), (gpointer
)this );
2923 gtk_signal_connect( GTK_OBJECT(widget
), "key_release_event",
2924 GTK_SIGNAL_FUNC(gtk_window_key_release_callback
), (gpointer
)this );
2926 gtk_signal_connect( GTK_OBJECT(widget
), "button_press_event",
2927 GTK_SIGNAL_FUNC(gtk_window_button_press_callback
), (gpointer
)this );
2929 gtk_signal_connect( GTK_OBJECT(widget
), "button_release_event",
2930 GTK_SIGNAL_FUNC(gtk_window_button_release_callback
), (gpointer
)this );
2932 gtk_signal_connect( GTK_OBJECT(widget
), "motion_notify_event",
2933 GTK_SIGNAL_FUNC(gtk_window_motion_notify_callback
), (gpointer
)this );
2935 gtk_signal_connect( GTK_OBJECT(widget
), "enter_notify_event",
2936 GTK_SIGNAL_FUNC(gtk_window_enter_callback
), (gpointer
)this );
2938 gtk_signal_connect( GTK_OBJECT(widget
), "leave_notify_event",
2939 GTK_SIGNAL_FUNC(gtk_window_leave_callback
), (gpointer
)this );
2942 bool wxWindowGTK::Destroy()
2944 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2948 return wxWindowBase::Destroy();
2951 void wxWindowGTK::DoMoveWindow(int x
, int y
, int width
, int height
)
2953 gtk_pizza_set_size( GTK_PIZZA(m_parent
->m_wxwindow
), m_widget
, x
, y
, width
, height
);
2956 void wxWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags
)
2958 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
2959 wxASSERT_MSG( (m_parent
!= NULL
), wxT("wxWindowGTK::SetSize requires parent.\n") );
2962 printf( "DoSetSize: name %s, x,y,w,h: %d,%d,%d,%d \n", GetName().c_str(), x,y,width,height );
2965 if (m_resizing
) return; /* I don't like recursions */
2968 int currentX
, currentY
;
2969 GetPosition(¤tX
, ¤tY
);
2974 AdjustForParentClientOrigin(x
, y
, sizeFlags
);
2976 if (m_parent
->m_wxwindow
== NULL
) /* i.e. wxNotebook */
2978 /* don't set the size for children of wxNotebook, just take the values. */
2986 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
2987 if ((sizeFlags
& wxSIZE_ALLOW_MINUS_ONE
) == 0)
2989 if (x
!= -1) m_x
= x
+ pizza
->xoffset
;
2990 if (y
!= -1) m_y
= y
+ pizza
->yoffset
;
2994 m_x
= x
+ pizza
->xoffset
;
2995 m_y
= y
+ pizza
->yoffset
;
2997 if (width
!= -1) m_width
= width
;
2998 if (height
!= -1) m_height
= height
;
3000 if ((sizeFlags
& wxSIZE_AUTO_WIDTH
) == wxSIZE_AUTO_WIDTH
)
3002 if (width
== -1) m_width
= 80;
3005 if ((sizeFlags
& wxSIZE_AUTO_HEIGHT
) == wxSIZE_AUTO_HEIGHT
)
3007 if (height
== -1) m_height
= 26;
3010 int minWidth
= GetMinWidth(),
3011 minHeight
= GetMinHeight(),
3012 maxWidth
= GetMaxWidth(),
3013 maxHeight
= GetMaxHeight();
3015 if ((minWidth
!= -1) && (m_width
< minWidth
)) m_width
= minWidth
;
3016 if ((minHeight
!= -1) && (m_height
< minHeight
)) m_height
= minHeight
;
3017 if ((maxWidth
!= -1) && (m_width
> maxWidth
)) m_width
= maxWidth
;
3018 if ((maxHeight
!= -1) && (m_height
> maxHeight
)) m_height
= maxHeight
;
3021 int bottom_border
= 0;
3024 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
3026 /* the default button has a border around it */
3032 DoMoveWindow( m_x
-border
,
3035 m_height
+border
+bottom_border
);
3040 /* Sometimes the client area changes size without the
3041 whole windows's size changing, but if the whole
3042 windows's size doesn't change, no wxSizeEvent will
3043 normally be sent. Here we add an extra test if
3044 the client test has been changed and this will
3046 GetClientSize( &m_oldClientWidth
, &m_oldClientHeight
);
3050 wxPrintf( "OnSize sent from " );
3051 if (GetClassInfo() && GetClassInfo()->GetClassName())
3052 wxPrintf( GetClassInfo()->GetClassName() );
3053 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
3056 if (!m_nativeSizeEvent
)
3058 wxSizeEvent
event( wxSize(m_width
,m_height
), GetId() );
3059 event
.SetEventObject( this );
3060 GetEventHandler()->ProcessEvent( event
);
3066 void wxWindowGTK::OnInternalIdle()
3068 // Update invalidated regions.
3071 // Synthetize activate events.
3072 if ( g_sendActivateEvent
!= -1 )
3074 bool activate
= g_sendActivateEvent
!= 0;
3077 g_sendActivateEvent
= -1;
3079 wxTheApp
->SetActive(activate
, (wxWindow
*)g_focusWindowLast
);
3082 if ( g_activeFrameLostFocus
)
3084 if ( g_activeFrame
)
3086 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from idle)"), g_activeFrame
);
3087 wxActivateEvent
event(wxEVT_ACTIVATE
, FALSE
, g_activeFrame
->GetId());
3088 event
.SetEventObject(g_activeFrame
);
3089 g_activeFrame
->GetEventHandler()->ProcessEvent(event
);
3090 g_activeFrame
= NULL
;
3092 g_activeFrameLostFocus
= FALSE
;
3095 wxCursor cursor
= m_cursor
;
3096 if (g_globalCursor
.Ok()) cursor
= g_globalCursor
;
3100 /* I now set the cursor anew in every OnInternalIdle call
3101 as setting the cursor in a parent window also effects the
3102 windows above so that checking for the current cursor is
3107 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3109 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3111 if (!g_globalCursor
.Ok())
3112 cursor
= *wxSTANDARD_CURSOR
;
3114 window
= m_widget
->window
;
3115 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
3116 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3122 GdkWindow
*window
= m_widget
->window
;
3123 if ((window
) && !(GTK_WIDGET_NO_WINDOW(m_widget
)))
3124 gdk_window_set_cursor( window
, cursor
.GetCursor() );
3132 void wxWindowGTK::DoGetSize( int *width
, int *height
) const
3134 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3136 if (width
) (*width
) = m_width
;
3137 if (height
) (*height
) = m_height
;
3140 void wxWindowGTK::DoSetClientSize( int width
, int height
)
3142 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3146 SetSize( width
, height
);
3153 #ifndef __WXUNIVERSAL__
3154 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3156 /* when using GTK 1.2 we set the shadow border size to 2 */
3160 if (HasFlag(wxSIMPLE_BORDER
))
3162 /* when using GTK 1.2 we set the simple border size to 1 */
3166 #endif // __WXUNIVERSAL__
3170 GtkScrolledWindow
*scroll_window
= GTK_SCROLLED_WINDOW(m_widget
);
3172 GtkRequisition vscroll_req
;
3173 vscroll_req
.width
= 2;
3174 vscroll_req
.height
= 2;
3175 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
3176 (scroll_window
->vscrollbar
, &vscroll_req
);
3178 GtkRequisition hscroll_req
;
3179 hscroll_req
.width
= 2;
3180 hscroll_req
.height
= 2;
3181 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
3182 (scroll_window
->hscrollbar
, &hscroll_req
);
3184 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
3186 if (scroll_window
->vscrollbar_visible
)
3188 dw
+= vscroll_req
.width
;
3189 dw
+= scroll_class
->scrollbar_spacing
;
3192 if (scroll_window
->hscrollbar_visible
)
3194 dh
+= hscroll_req
.height
;
3195 dh
+= scroll_class
->scrollbar_spacing
;
3199 SetSize( width
+dw
, height
+dh
);
3203 void wxWindowGTK::DoGetClientSize( int *width
, int *height
) const
3205 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3209 if (width
) (*width
) = m_width
;
3210 if (height
) (*height
) = m_height
;
3217 #ifndef __WXUNIVERSAL__
3218 if (HasFlag(wxRAISED_BORDER
) || HasFlag(wxSUNKEN_BORDER
))
3220 /* when using GTK 1.2 we set the shadow border size to 2 */
3224 if (HasFlag(wxSIMPLE_BORDER
))
3226 /* when using GTK 1.2 we set the simple border size to 1 */
3230 #endif // __WXUNIVERSAL__
3234 GtkScrolledWindow
*scroll_window
= GTK_SCROLLED_WINDOW(m_widget
);
3236 GtkRequisition vscroll_req
;
3237 vscroll_req
.width
= 2;
3238 vscroll_req
.height
= 2;
3239 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->vscrollbar
) )->size_request
)
3240 (scroll_window
->vscrollbar
, &vscroll_req
);
3242 GtkRequisition hscroll_req
;
3243 hscroll_req
.width
= 2;
3244 hscroll_req
.height
= 2;
3245 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window
->hscrollbar
) )->size_request
)
3246 (scroll_window
->hscrollbar
, &hscroll_req
);
3248 GtkScrolledWindowClass
*scroll_class
= GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) );
3250 if (scroll_window
->vscrollbar_visible
)
3252 dw
+= vscroll_req
.width
;
3253 dw
+= scroll_class
->scrollbar_spacing
;
3256 if (scroll_window
->hscrollbar_visible
)
3258 dh
+= hscroll_req
.height
;
3259 dh
+= scroll_class
->scrollbar_spacing
;
3263 if (width
) (*width
) = m_width
- dw
;
3264 if (height
) (*height
) = m_height
- dh
;
3268 printf( "GetClientSize, name %s ", GetName().c_str() );
3269 if (width) printf( " width = %d", (*width) );
3270 if (height) printf( " height = %d", (*height) );
3275 void wxWindowGTK::DoGetPosition( int *x
, int *y
) const
3277 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3281 if (m_parent
&& m_parent
->m_wxwindow
)
3283 GtkPizza
*pizza
= GTK_PIZZA(m_parent
->m_wxwindow
);
3284 dx
= pizza
->xoffset
;
3285 dy
= pizza
->yoffset
;
3288 if (x
) (*x
) = m_x
- dx
;
3289 if (y
) (*y
) = m_y
- dy
;
3292 void wxWindowGTK::DoClientToScreen( int *x
, int *y
) const
3294 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3296 if (!m_widget
->window
) return;
3298 GdkWindow
*source
= (GdkWindow
*) NULL
;
3300 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3302 source
= m_widget
->window
;
3306 gdk_window_get_origin( source
, &org_x
, &org_y
);
3310 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3312 org_x
+= m_widget
->allocation
.x
;
3313 org_y
+= m_widget
->allocation
.y
;
3321 void wxWindowGTK::DoScreenToClient( int *x
, int *y
) const
3323 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3325 if (!m_widget
->window
) return;
3327 GdkWindow
*source
= (GdkWindow
*) NULL
;
3329 source
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3331 source
= m_widget
->window
;
3335 gdk_window_get_origin( source
, &org_x
, &org_y
);
3339 if (GTK_WIDGET_NO_WINDOW (m_widget
))
3341 org_x
+= m_widget
->allocation
.x
;
3342 org_y
+= m_widget
->allocation
.y
;
3350 bool wxWindowGTK::Show( bool show
)
3352 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3354 if (!wxWindowBase::Show(show
))
3361 gtk_widget_show( m_widget
);
3363 gtk_widget_hide( m_widget
);
3365 wxShowEvent
eventShow(GetId(), show
);
3366 eventShow
.m_eventObject
= this;
3368 GetEventHandler()->ProcessEvent(eventShow
);
3373 static void wxWindowNotifyEnable(wxWindowGTK
* win
, bool enable
)
3375 win
->OnParentEnable(enable
);
3377 // Recurse, so that children have the opportunity to Do The Right Thing
3378 // and reset colours that have been messed up by a parent's (really ancestor's)
3380 for ( wxWindowList::Node
*node
= win
->GetChildren().GetFirst();
3382 node
= node
->GetNext() )
3384 wxWindow
*child
= node
->GetData();
3385 if (!child
->IsKindOf(CLASSINFO(wxDialog
)) && !child
->IsKindOf(CLASSINFO(wxFrame
)))
3386 wxWindowNotifyEnable(child
, enable
);
3390 bool wxWindowGTK::Enable( bool enable
)
3392 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3394 if (!wxWindowBase::Enable(enable
))
3400 gtk_widget_set_sensitive( m_widget
, enable
);
3402 gtk_widget_set_sensitive( m_wxwindow
, enable
);
3404 wxWindowNotifyEnable(this, enable
);
3409 int wxWindowGTK::GetCharHeight() const
3411 wxCHECK_MSG( (m_widget
!= NULL
), 12, wxT("invalid window") );
3413 wxCHECK_MSG( m_font
.Ok(), 12, wxT("invalid font") );
3416 PangoContext
*context
= NULL
;
3418 context
= gtk_widget_get_pango_context( m_widget
);
3423 PangoFontDescription
*desc
= m_font
.GetNativeFontInfo()->description
;
3424 PangoLayout
*layout
= pango_layout_new(context
);
3425 pango_layout_set_font_description(layout
, desc
);
3426 pango_layout_set_text(layout
, "H", 1);
3427 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3429 PangoRectangle rect
;
3430 pango_layout_line_get_extents(line
, NULL
, &rect
);
3432 g_object_unref( G_OBJECT( layout
) );
3434 return (int) (rect
.height
/ PANGO_SCALE
);
3436 GdkFont
*font
= m_font
.GetInternalFont( 1.0 );
3438 return font
->ascent
+ font
->descent
;
3442 int wxWindowGTK::GetCharWidth() const
3444 wxCHECK_MSG( (m_widget
!= NULL
), 8, wxT("invalid window") );
3446 wxCHECK_MSG( m_font
.Ok(), 8, wxT("invalid font") );
3449 PangoContext
*context
= NULL
;
3451 context
= gtk_widget_get_pango_context( m_widget
);
3456 PangoFontDescription
*desc
= m_font
.GetNativeFontInfo()->description
;
3457 PangoLayout
*layout
= pango_layout_new(context
);
3458 pango_layout_set_font_description(layout
, desc
);
3459 pango_layout_set_text(layout
, "H", 1);
3460 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3462 PangoRectangle rect
;
3463 pango_layout_line_get_extents(line
, NULL
, &rect
);
3465 g_object_unref( G_OBJECT( layout
) );
3467 return (int) (rect
.width
/ PANGO_SCALE
);
3469 GdkFont
*font
= m_font
.GetInternalFont( 1.0 );
3471 return gdk_string_width( font
, "H" );
3475 void wxWindowGTK::GetTextExtent( const wxString
& string
,
3479 int *externalLeading
,
3480 const wxFont
*theFont
) const
3482 wxFont fontToUse
= m_font
;
3483 if (theFont
) fontToUse
= *theFont
;
3485 wxCHECK_RET( fontToUse
.Ok(), wxT("invalid font") );
3487 if (string
.IsEmpty())
3495 PangoContext
*context
= NULL
;
3497 context
= gtk_widget_get_pango_context( m_widget
);
3506 PangoFontDescription
*desc
= fontToUse
.GetNativeFontInfo()->description
;
3507 PangoLayout
*layout
= pango_layout_new(context
);
3508 pango_layout_set_font_description(layout
, desc
);
3511 const wxCharBuffer data
= wxConvUTF8
.cWC2MB( string
);
3512 pango_layout_set_text(layout
, (const char*) data
, strlen( (const char*) data
));
3514 const wxWCharBuffer wdata
= wxConvLocal
.cMB2WC( string
);
3515 const wxCharBuffer data
= wxConvUTF8
.cWC2MB( wdata
);
3516 pango_layout_set_text(layout
, (const char*) data
, strlen( (const char*) data
));
3519 PangoLayoutLine
*line
= (PangoLayoutLine
*)pango_layout_get_lines(layout
)->data
;
3521 PangoRectangle rect
;
3522 pango_layout_line_get_extents(line
, NULL
, &rect
);
3524 if (x
) (*x
) = (wxCoord
) (rect
.width
/ PANGO_SCALE
);
3525 if (y
) (*y
) = (wxCoord
) (rect
.height
/ PANGO_SCALE
);
3528 // Do something about metrics here
3531 if (externalLeading
) (*externalLeading
) = 0; // ??
3533 g_object_unref( G_OBJECT( layout
) );
3535 GdkFont
*font
= fontToUse
.GetInternalFont( 1.0 );
3536 if (x
) (*x
) = gdk_string_width( font
, wxGTK_CONV( string
) );
3537 if (y
) (*y
) = font
->ascent
+ font
->descent
;
3538 if (descent
) (*descent
) = font
->descent
;
3539 if (externalLeading
) (*externalLeading
) = 0; // ??
3543 void wxWindowGTK::SetFocus()
3545 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3549 // don't do anything if we already have focus
3555 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow
))
3557 gtk_widget_grab_focus (m_wxwindow
);
3562 if (GTK_WIDGET_CAN_FOCUS(m_widget
) && !GTK_WIDGET_HAS_FOCUS (m_widget
) )
3564 if (!GTK_WIDGET_REALIZED(m_widget
))
3566 // we can't set the focus to the widget now so we remember that
3567 // it should be focused and will do it later, during the idle
3568 // time, as soon as we can
3569 wxLogTrace(TRACE_FOCUS
,
3570 _T("Delaying setting focus to %s(%s)"),
3571 GetClassInfo()->GetClassName(), GetLabel().c_str());
3573 g_delayedFocus
= this;
3577 wxLogTrace(TRACE_FOCUS
,
3578 _T("Setting focus to %s(%s)"),
3579 GetClassInfo()->GetClassName(), GetLabel().c_str());
3581 gtk_widget_grab_focus (m_widget
);
3584 else if (GTK_IS_CONTAINER(m_widget
))
3586 SET_CONTAINER_FOCUS( m_widget
, GTK_DIR_TAB_FORWARD
);
3590 wxLogTrace(TRACE_FOCUS
,
3591 _T("Can't set focus to %s(%s)"),
3592 GetClassInfo()->GetClassName(), GetLabel().c_str());
3597 bool wxWindowGTK::AcceptsFocus() const
3599 return m_acceptsFocus
&& wxWindowBase::AcceptsFocus();
3602 bool wxWindowGTK::Reparent( wxWindowBase
*newParentBase
)
3604 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3606 wxWindowGTK
*oldParent
= m_parent
,
3607 *newParent
= (wxWindowGTK
*)newParentBase
;
3609 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3611 if ( !wxWindowBase::Reparent(newParent
) )
3614 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3616 /* prevent GTK from deleting the widget arbitrarily */
3617 gtk_widget_ref( m_widget
);
3621 gtk_container_remove( GTK_CONTAINER(m_widget
->parent
), m_widget
);
3624 wxASSERT( GTK_IS_WIDGET(m_widget
) );
3628 /* insert GTK representation */
3629 (*(newParent
->m_insertCallback
))(newParent
, this);
3632 /* reverse: prevent GTK from deleting the widget arbitrarily */
3633 gtk_widget_unref( m_widget
);
3638 void wxWindowGTK::DoAddChild(wxWindowGTK
*child
)
3640 wxASSERT_MSG( (m_widget
!= NULL
), wxT("invalid window") );
3642 wxASSERT_MSG( (child
!= NULL
), wxT("invalid child window") );
3644 wxASSERT_MSG( (m_insertCallback
!= NULL
), wxT("invalid child insertion function") );
3649 /* insert GTK representation */
3650 (*m_insertCallback
)(this, child
);
3653 void wxWindowGTK::Raise()
3655 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3657 if (!m_widget
->window
) return;
3659 gdk_window_raise( m_widget
->window
);
3662 void wxWindowGTK::Lower()
3664 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3666 if (!m_widget
->window
) return;
3668 gdk_window_lower( m_widget
->window
);
3671 bool wxWindowGTK::SetCursor( const wxCursor
&cursor
)
3673 wxCHECK_MSG( (m_widget
!= NULL
), FALSE
, wxT("invalid window") );
3675 if (cursor
== m_cursor
)
3679 wxapp_install_idle_handler();
3681 if (cursor
== wxNullCursor
)
3682 return wxWindowBase::SetCursor( *wxSTANDARD_CURSOR
);
3684 return wxWindowBase::SetCursor( cursor
);
3687 void wxWindowGTK::WarpPointer( int x
, int y
)
3689 wxCHECK_RET( (m_widget
!= NULL
), wxT("invalid window") );
3691 // We provide this function ourselves as it is
3692 // missing in GDK (top of this file).
3694 GdkWindow
*window
= (GdkWindow
*) NULL
;
3696 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3698 window
= GetConnectWidget()->window
;
3701 gdk_window_warp_pointer( window
, x
, y
);
3705 void wxWindowGTK::Refresh( bool eraseBackground
, const wxRect
*rect
)
3707 if (!m_widget
) return;
3708 if (!m_widget
->window
) return;
3712 wxapp_install_idle_handler();
3714 if (eraseBackground
&& m_wxwindow
&& m_wxwindow
->window
)
3718 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3719 m_clearRegion
.Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
3723 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3724 m_clearRegion
.Clear();
3725 m_clearRegion
.Union( 0, 0, m_wxwindow
->allocation
.width
, m_wxwindow
->allocation
.height
);
3733 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3734 m_updateRegion
.Union( rect
->x
, rect
->y
, rect
->width
, rect
->height
);
3738 GdkRectangle gdk_rect
;
3739 gdk_rect
.x
= rect
->x
;
3740 gdk_rect
.y
= rect
->y
;
3741 gdk_rect
.width
= rect
->width
;
3742 gdk_rect
.height
= rect
->height
;
3743 gtk_widget_draw( m_widget
, &gdk_rect
);
3750 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3751 m_updateRegion
.Clear();
3752 m_updateRegion
.Union( 0, 0, m_wxwindow
->allocation
.width
, m_wxwindow
->allocation
.height
);
3756 gtk_widget_draw( m_widget
, (GdkRectangle
*) NULL
);
3764 GdkRectangle gdk_rect
;
3765 gdk_rect
.x
= rect
->x
;
3766 gdk_rect
.y
= rect
->y
;
3767 gdk_rect
.width
= rect
->width
;
3768 gdk_rect
.height
= rect
->height
;
3769 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, &gdk_rect
, TRUE
);
3773 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow
)->bin_window
, NULL
, TRUE
);
3779 void wxWindowGTK::Update()
3784 void wxWindowGTK::GtkUpdate()
3787 if (m_wxwindow
&& GTK_PIZZA(m_wxwindow
)->bin_window
)
3788 gdk_window_process_updates( GTK_PIZZA(m_wxwindow
)->bin_window
, FALSE
);
3790 if (!m_updateRegion
.IsEmpty())
3791 GtkSendPaintEvents();
3795 void wxWindowGTK::GtkSendPaintEvents()
3800 m_clearRegion
.Clear();
3802 m_updateRegion
.Clear();
3806 // Clip to paint region in wxClientDC
3807 m_clipPaintRegion
= TRUE
;
3810 // widget to draw on
3811 GtkPizza
*pizza
= GTK_PIZZA (m_wxwindow
);
3813 // later for GTK 2.0, too.
3814 if (GetThemeEnabled())
3816 // find ancestor from which to steal background
3817 wxWindow
*parent
= GetParent();
3818 while (parent
&& !parent
->IsTopLevel())
3819 parent
= parent
->GetParent();
3821 parent
= (wxWindow
*)this;
3823 wxRegionIterator
upd( m_updateRegion
);
3827 rect
.x
= upd
.GetX();
3828 rect
.y
= upd
.GetY();
3829 rect
.width
= upd
.GetWidth();
3830 rect
.height
= upd
.GetHeight();
3832 gtk_paint_flat_box( parent
->m_widget
->style
,
3849 wxWindowDC
dc( (wxWindow
*)this );
3850 dc
.SetClippingRegion( m_updateRegion
);
3852 wxEraseEvent
erase_event( GetId(), &dc
);
3853 erase_event
.SetEventObject( this );
3855 GetEventHandler()->ProcessEvent(erase_event
);
3858 // if (!m_clearRegion.IsEmpty()) // Always send an erase event under GTK 1.2
3860 wxWindowDC
dc( (wxWindow
*)this );
3861 if (m_clearRegion
.IsEmpty())
3862 dc
.SetClippingRegion( m_updateRegion
);
3864 dc
.SetClippingRegion( m_clearRegion
);
3866 wxEraseEvent
erase_event( GetId(), &dc
);
3867 erase_event
.SetEventObject( this );
3869 if (!GetEventHandler()->ProcessEvent(erase_event
))
3873 g_eraseGC
= gdk_gc_new( pizza
->bin_window
);
3874 gdk_gc_set_fill( g_eraseGC
, GDK_SOLID
);
3876 gdk_gc_set_foreground( g_eraseGC
, m_backgroundColour
.GetColor() );
3878 wxRegionIterator
upd( m_clearRegion
);
3881 gdk_draw_rectangle( pizza
->bin_window
, g_eraseGC
, 1,
3882 upd
.GetX(), upd
.GetY(), upd
.GetWidth(), upd
.GetHeight() );
3886 m_clearRegion
.Clear();
3890 wxNcPaintEvent
nc_paint_event( GetId() );
3891 nc_paint_event
.SetEventObject( this );
3892 GetEventHandler()->ProcessEvent( nc_paint_event
);
3894 wxPaintEvent
paint_event( GetId() );
3895 paint_event
.SetEventObject( this );
3896 GetEventHandler()->ProcessEvent( paint_event
);
3898 m_clipPaintRegion
= FALSE
;
3900 #ifndef __WXUNIVERSAL__
3902 // The following code will result in all window-less widgets
3903 // being redrawn because the wxWindows class is allowed to
3904 // paint over the window-less widgets.
3906 GList
*children
= pizza
->children
;
3909 GtkPizzaChild
*child
= (GtkPizzaChild
*) children
->data
;
3910 children
= children
->next
;
3912 if (GTK_WIDGET_NO_WINDOW (child
->widget
) &&
3913 GTK_WIDGET_DRAWABLE (child
->widget
))
3915 // Get intersection of widget area and update region
3916 wxRegion
region( m_updateRegion
);
3918 GdkEventExpose gdk_event
;
3919 gdk_event
.type
= GDK_EXPOSE
;
3920 gdk_event
.window
= pizza
->bin_window
;
3921 gdk_event
.count
= 0;
3923 wxRegionIterator
upd( m_updateRegion
);
3927 rect
.x
= upd
.GetX();
3928 rect
.y
= upd
.GetY();
3929 rect
.width
= upd
.GetWidth();
3930 rect
.height
= upd
.GetHeight();
3932 if (gtk_widget_intersect (child
->widget
, &rect
, &gdk_event
.area
))
3934 gtk_widget_event (child
->widget
, (GdkEvent
*) &gdk_event
);
3944 m_updateRegion
.Clear();
3947 void wxWindowGTK::Clear()
3949 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
3952 if (m_wxwindow
&& m_wxwindow
->window
)
3954 m_clearRegion
.Clear();
3955 wxSize
size( GetClientSize() );
3956 m_clearRegion
.Union( 0,0,size
.x
,size
.y
);
3958 // Better do this in idle?
3965 void wxWindowGTK::DoSetToolTip( wxToolTip
*tip
)
3967 wxWindowBase::DoSetToolTip(tip
);
3970 m_tooltip
->Apply( (wxWindow
*)this );
3973 void wxWindowGTK::ApplyToolTip( GtkTooltips
*tips
, const wxChar
*tip
)
3975 gtk_tooltips_set_tip( tips
, GetConnectWidget(), wxConvCurrent
->cWX2MB(tip
), (gchar
*) NULL
);
3977 #endif // wxUSE_TOOLTIPS
3979 void wxWindowGTK::GtkSetBackgroundColour( const wxColour
&colour
)
3981 GdkWindow
*window
= (GdkWindow
*) NULL
;
3983 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
3985 window
= GetConnectWidget()->window
;
3989 // We need the pixel value e.g. for background clearing.
3990 m_backgroundColour
.CalcPixel( gdk_window_get_colormap( window
) );
3994 // wxMSW doesn't clear the window here, either.
3995 gdk_window_set_background( window
, m_backgroundColour
.GetColor() );
4001 bool wxWindowGTK::SetBackgroundColour( const wxColour
&colour
)
4003 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4005 if (!wxWindowBase::SetBackgroundColour(colour
))
4008 GdkWindow
*window
= (GdkWindow
*) NULL
;
4010 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4012 window
= GetConnectWidget()->window
;
4016 // indicate that a new style has been set
4017 // but it couldn't get applied as the
4018 // widget hasn't been realized yet.
4019 m_delayedBackgroundColour
= TRUE
;
4024 GtkSetBackgroundColour( colour
);
4030 void wxWindowGTK::GtkSetForegroundColour( const wxColour
&colour
)
4032 GdkWindow
*window
= (GdkWindow
*) NULL
;
4034 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4036 window
= GetConnectWidget()->window
;
4043 bool wxWindowGTK::SetForegroundColour( const wxColour
&colour
)
4045 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4047 if (!wxWindowBase::SetForegroundColour(colour
))
4049 // don't leave if the GTK widget has just
4051 if (!m_delayedForegroundColour
) return FALSE
;
4054 GdkWindow
*window
= (GdkWindow
*) NULL
;
4056 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4058 window
= GetConnectWidget()->window
;
4062 // indicate that a new style has been set
4063 // but it couldn't get applied as the
4064 // widget hasn't been realized yet.
4065 m_delayedForegroundColour
= TRUE
;
4069 GtkSetForegroundColour( colour
);
4076 PangoContext
*wxWindowGTK::GtkGetPangoDefaultContext()
4078 return gtk_widget_get_pango_context( m_widget
);
4081 PangoContext
*wxWindowGTK::GtkGetPangoX11Context()
4084 m_x11Context
= pango_x_get_context( gdk_display
);
4086 return m_x11Context
;
4090 GtkStyle
*wxWindowGTK::GetWidgetStyle()
4094 GtkStyle
*remake
= gtk_style_copy( m_widgetStyle
);
4096 // FIXME: no more klass in 2.0
4098 remake
->klass
= m_widgetStyle
->klass
;
4101 gtk_style_unref( m_widgetStyle
);
4102 m_widgetStyle
= remake
;
4106 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4109 def
= gtk_widget_get_default_style();
4111 m_widgetStyle
= gtk_style_copy( def
);
4113 // FIXME: no more klass in 2.0
4115 m_widgetStyle
->klass
= def
->klass
;
4119 return m_widgetStyle
;
4122 void wxWindowGTK::SetWidgetStyle()
4124 #if DISABLE_STYLE_IF_BROKEN_THEME
4125 if (m_widget
->style
->engine_data
)
4127 static bool s_warningPrinted
= FALSE
;
4128 if (!s_warningPrinted
)
4130 printf( "wxWindows warning: Widget styles disabled due to buggy GTK theme.\n" );
4131 s_warningPrinted
= TRUE
;
4133 m_widgetStyle
= m_widget
->style
;
4138 GtkStyle
*style
= GetWidgetStyle();
4140 if (m_font
!= wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT
))
4143 pango_font_description_free( style
->font_desc
);
4144 pango_font_description_copy( m_font
.GetNativeFontInfo()->description
);
4146 gdk_font_unref( style
->font
);
4147 style
->font
= gdk_font_ref( m_font
.GetInternalFont( 1.0 ) );
4151 if (m_foregroundColour
.Ok())
4153 m_foregroundColour
.CalcPixel( gtk_widget_get_colormap( m_widget
) );
4154 if (m_foregroundColour
!= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNTEXT
))
4156 style
->fg
[GTK_STATE_NORMAL
] = *m_foregroundColour
.GetColor();
4157 style
->fg
[GTK_STATE_PRELIGHT
] = *m_foregroundColour
.GetColor();
4158 style
->fg
[GTK_STATE_ACTIVE
] = *m_foregroundColour
.GetColor();
4162 // Try to restore the gtk default style. This is still a little
4163 // oversimplified for what is probably really needed here for controls
4164 // other than buttons, but is better than not being able to (re)set a
4165 // control's foreground colour to *wxBLACK -- RL
4166 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4169 def
= gtk_widget_get_default_style();
4171 style
->fg
[GTK_STATE_NORMAL
] = def
->fg
[GTK_STATE_NORMAL
];
4172 style
->fg
[GTK_STATE_PRELIGHT
] = def
->fg
[GTK_STATE_PRELIGHT
];
4173 style
->fg
[GTK_STATE_ACTIVE
] = def
->fg
[GTK_STATE_ACTIVE
];
4177 if (m_backgroundColour
.Ok())
4179 m_backgroundColour
.CalcPixel( gtk_widget_get_colormap( m_widget
) );
4180 if (m_backgroundColour
!= wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE
))
4182 style
->bg
[GTK_STATE_NORMAL
] = *m_backgroundColour
.GetColor();
4183 style
->base
[GTK_STATE_NORMAL
] = *m_backgroundColour
.GetColor();
4184 style
->bg
[GTK_STATE_PRELIGHT
] = *m_backgroundColour
.GetColor();
4185 style
->base
[GTK_STATE_PRELIGHT
] = *m_backgroundColour
.GetColor();
4186 style
->bg
[GTK_STATE_ACTIVE
] = *m_backgroundColour
.GetColor();
4187 style
->base
[GTK_STATE_ACTIVE
] = *m_backgroundColour
.GetColor();
4188 style
->bg
[GTK_STATE_INSENSITIVE
] = *m_backgroundColour
.GetColor();
4189 style
->base
[GTK_STATE_INSENSITIVE
] = *m_backgroundColour
.GetColor();
4193 // Try to restore the gtk default style. This is still a little
4194 // oversimplified for what is probably really needed here for controls
4195 // other than buttons, but is better than not being able to (re)set a
4196 // control's background colour to default grey and means resetting a
4197 // button to wxSYS_COLOUR_BTNFACE will restore its usual highlighting
4199 GtkStyle
*def
= gtk_rc_get_style( m_widget
);
4202 def
= gtk_widget_get_default_style();
4204 style
->bg
[GTK_STATE_NORMAL
] = def
->bg
[GTK_STATE_NORMAL
];
4205 style
->base
[GTK_STATE_NORMAL
] = def
->base
[GTK_STATE_NORMAL
];
4206 style
->bg
[GTK_STATE_PRELIGHT
] = def
->bg
[GTK_STATE_PRELIGHT
];
4207 style
->base
[GTK_STATE_PRELIGHT
] = def
->base
[GTK_STATE_PRELIGHT
];
4208 style
->bg
[GTK_STATE_ACTIVE
] = def
->bg
[GTK_STATE_ACTIVE
];
4209 style
->base
[GTK_STATE_ACTIVE
] = def
->base
[GTK_STATE_ACTIVE
];
4210 style
->bg
[GTK_STATE_INSENSITIVE
] = def
->bg
[GTK_STATE_INSENSITIVE
];
4211 style
->base
[GTK_STATE_INSENSITIVE
] = def
->base
[GTK_STATE_INSENSITIVE
];
4216 void wxWindowGTK::ApplyWidgetStyle()
4220 //-----------------------------------------------------------------------------
4221 // Pop-up menu stuff
4222 //-----------------------------------------------------------------------------
4224 #if wxUSE_MENUS_NATIVE
4227 void gtk_pop_hide_callback( GtkWidget
*WXUNUSED(widget
), bool* is_waiting
)
4229 *is_waiting
= FALSE
;
4232 static void SetInvokingWindow( wxMenu
*menu
, wxWindowGTK
*win
)
4234 menu
->SetInvokingWindow( win
);
4235 wxMenuItemList::Node
*node
= menu
->GetMenuItems().GetFirst();
4238 wxMenuItem
*menuitem
= node
->GetData();
4239 if (menuitem
->IsSubMenu())
4241 SetInvokingWindow( menuitem
->GetSubMenu(), win
);
4244 node
= node
->GetNext();
4248 // used to pass the coordinates from wxWindowGTK::DoPopupMenu() to
4249 // wxPopupMenuPositionCallback()
4251 // should be safe even in the MT case as the user can hardly popup 2 menus
4252 // simultaneously, can he?
4253 static gint gs_pop_x
= 0;
4254 static gint gs_pop_y
= 0;
4256 extern "C" void wxPopupMenuPositionCallback( GtkMenu
*menu
,
4259 gboolean
* WXUNUSED(whatever
),
4261 gpointer
WXUNUSED(user_data
) )
4263 // ensure that the menu appears entirely on screen
4265 gtk_widget_get_child_requisition(GTK_WIDGET(menu
), &req
);
4267 wxSize sizeScreen
= wxGetDisplaySize();
4269 gint xmax
= sizeScreen
.x
- req
.width
,
4270 ymax
= sizeScreen
.y
- req
.height
;
4272 *x
= gs_pop_x
< xmax
? gs_pop_x
: xmax
;
4273 *y
= gs_pop_y
< ymax
? gs_pop_y
: ymax
;
4276 bool wxWindowGTK::DoPopupMenu( wxMenu
*menu
, int x
, int y
)
4278 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4280 wxCHECK_MSG( menu
!= NULL
, FALSE
, wxT("invalid popup-menu") );
4282 SetInvokingWindow( menu
, this );
4288 ClientToScreen( &gs_pop_x
, &gs_pop_y
);
4290 bool is_waiting
= TRUE
;
4292 gtk_signal_connect( GTK_OBJECT(menu
->m_menu
),
4294 GTK_SIGNAL_FUNC(gtk_pop_hide_callback
),
4295 (gpointer
)&is_waiting
);
4298 GTK_MENU(menu
->m_menu
),
4299 (GtkWidget
*) NULL
, // parent menu shell
4300 (GtkWidget
*) NULL
, // parent menu item
4301 wxPopupMenuPositionCallback
, // function to position it
4302 NULL
, // client data
4303 0, // button used to activate it
4304 gs_timeLastClick
// the time of activation
4309 while (gtk_events_pending())
4310 gtk_main_iteration();
4316 #endif // wxUSE_MENUS_NATIVE
4318 #if wxUSE_DRAG_AND_DROP
4320 void wxWindowGTK::SetDropTarget( wxDropTarget
*dropTarget
)
4322 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4324 GtkWidget
*dnd_widget
= GetConnectWidget();
4326 if (m_dropTarget
) m_dropTarget
->UnregisterWidget( dnd_widget
);
4328 if (m_dropTarget
) delete m_dropTarget
;
4329 m_dropTarget
= dropTarget
;
4331 if (m_dropTarget
) m_dropTarget
->RegisterWidget( dnd_widget
);
4334 #endif // wxUSE_DRAG_AND_DROP
4336 GtkWidget
* wxWindowGTK::GetConnectWidget()
4338 GtkWidget
*connect_widget
= m_widget
;
4339 if (m_wxwindow
) connect_widget
= m_wxwindow
;
4341 return connect_widget
;
4344 bool wxWindowGTK::IsOwnGtkWindow( GdkWindow
*window
)
4347 return (window
== GTK_PIZZA(m_wxwindow
)->bin_window
);
4349 return (window
== m_widget
->window
);
4352 bool wxWindowGTK::SetFont( const wxFont
&font
)
4354 wxCHECK_MSG( m_widget
!= NULL
, FALSE
, wxT("invalid window") );
4356 if (!wxWindowBase::SetFont(font
))
4361 wxColour sysbg
= wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE
);
4362 if ( sysbg
== m_backgroundColour
)
4364 m_backgroundColour
= wxNullColour
;
4366 m_backgroundColour
= sysbg
;
4376 void wxWindowGTK::DoCaptureMouse()
4378 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4380 GdkWindow
*window
= (GdkWindow
*) NULL
;
4382 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4384 window
= GetConnectWidget()->window
;
4386 wxCHECK_RET( window
, _T("CaptureMouse() failed") );
4388 wxCursor
* cursor
= & m_cursor
;
4390 cursor
= wxSTANDARD_CURSOR
;
4392 gdk_pointer_grab( window
, FALSE
,
4394 (GDK_BUTTON_PRESS_MASK
|
4395 GDK_BUTTON_RELEASE_MASK
|
4396 GDK_POINTER_MOTION_HINT_MASK
|
4397 GDK_POINTER_MOTION_MASK
),
4399 cursor
->GetCursor(),
4400 (guint32
)GDK_CURRENT_TIME
);
4401 g_captureWindow
= this;
4402 g_captureWindowHasMouse
= TRUE
;
4405 void wxWindowGTK::DoReleaseMouse()
4407 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4409 wxCHECK_RET( g_captureWindow
, wxT("can't release mouse - not captured") );
4411 g_captureWindow
= (wxWindowGTK
*) NULL
;
4413 GdkWindow
*window
= (GdkWindow
*) NULL
;
4415 window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
4417 window
= GetConnectWidget()->window
;
4422 gdk_pointer_ungrab ( (guint32
)GDK_CURRENT_TIME
);
4426 wxWindow
*wxWindowBase::GetCapture()
4428 return (wxWindow
*)g_captureWindow
;
4431 bool wxWindowGTK::IsRetained() const
4436 void wxWindowGTK::SetScrollbar( int orient
, int pos
, int thumbVisible
,
4437 int range
, bool refresh
)
4439 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4441 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4443 m_hasScrolling
= TRUE
;
4445 if (orient
== wxHORIZONTAL
)
4447 float fpos
= (float)pos
;
4448 float frange
= (float)range
;
4449 float fthumb
= (float)thumbVisible
;
4450 if (fpos
> frange
-fthumb
) fpos
= frange
-fthumb
;
4451 if (fpos
< 0.0) fpos
= 0.0;
4453 if ((fabs(frange
-m_hAdjust
->upper
) < 0.2) &&
4454 (fabs(fthumb
-m_hAdjust
->page_size
) < 0.2))
4456 SetScrollPos( orient
, pos
, refresh
);
4460 m_oldHorizontalPos
= fpos
;
4462 m_hAdjust
->lower
= 0.0;
4463 m_hAdjust
->upper
= frange
;
4464 m_hAdjust
->value
= fpos
;
4465 m_hAdjust
->step_increment
= 1.0;
4466 m_hAdjust
->page_increment
= (float)(wxMax(fthumb
,0));
4467 m_hAdjust
->page_size
= fthumb
;
4471 float fpos
= (float)pos
;
4472 float frange
= (float)range
;
4473 float fthumb
= (float)thumbVisible
;
4474 if (fpos
> frange
-fthumb
) fpos
= frange
-fthumb
;
4475 if (fpos
< 0.0) fpos
= 0.0;
4477 if ((fabs(frange
-m_vAdjust
->upper
) < 0.2) &&
4478 (fabs(fthumb
-m_vAdjust
->page_size
) < 0.2))
4480 SetScrollPos( orient
, pos
, refresh
);
4484 m_oldVerticalPos
= fpos
;
4486 m_vAdjust
->lower
= 0.0;
4487 m_vAdjust
->upper
= frange
;
4488 m_vAdjust
->value
= fpos
;
4489 m_vAdjust
->step_increment
= 1.0;
4490 m_vAdjust
->page_increment
= (float)(wxMax(fthumb
,0));
4491 m_vAdjust
->page_size
= fthumb
;
4494 if (orient
== wxHORIZONTAL
)
4495 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "changed" );
4497 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "changed" );
4500 void wxWindowGTK::SetScrollPos( int orient
, int pos
, bool WXUNUSED(refresh
) )
4502 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4504 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4506 if (orient
== wxHORIZONTAL
)
4508 float fpos
= (float)pos
;
4509 if (fpos
> m_hAdjust
->upper
- m_hAdjust
->page_size
) fpos
= m_hAdjust
->upper
- m_hAdjust
->page_size
;
4510 if (fpos
< 0.0) fpos
= 0.0;
4511 m_oldHorizontalPos
= fpos
;
4513 if (fabs(fpos
-m_hAdjust
->value
) < 0.2) return;
4514 m_hAdjust
->value
= fpos
;
4518 float fpos
= (float)pos
;
4519 if (fpos
> m_vAdjust
->upper
- m_vAdjust
->page_size
) fpos
= m_vAdjust
->upper
- m_vAdjust
->page_size
;
4520 if (fpos
< 0.0) fpos
= 0.0;
4521 m_oldVerticalPos
= fpos
;
4523 if (fabs(fpos
-m_vAdjust
->value
) < 0.2) return;
4524 m_vAdjust
->value
= fpos
;
4527 if (m_wxwindow
->window
)
4529 if (orient
== wxHORIZONTAL
)
4531 gtk_signal_disconnect_by_func( GTK_OBJECT(m_hAdjust
),
4532 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
4534 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust
), "value_changed" );
4536 gtk_signal_connect( GTK_OBJECT(m_hAdjust
), "value_changed",
4537 (GtkSignalFunc
) gtk_window_hscroll_callback
, (gpointer
) this );
4541 gtk_signal_disconnect_by_func( GTK_OBJECT(m_vAdjust
),
4542 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
4544 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust
), "value_changed" );
4546 gtk_signal_connect( GTK_OBJECT(m_vAdjust
), "value_changed",
4547 (GtkSignalFunc
) gtk_window_vscroll_callback
, (gpointer
) this );
4552 int wxWindowGTK::GetScrollThumb( int orient
) const
4554 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4556 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4558 if (orient
== wxHORIZONTAL
)
4559 return (int)(m_hAdjust
->page_size
+0.5);
4561 return (int)(m_vAdjust
->page_size
+0.5);
4564 int wxWindowGTK::GetScrollPos( int orient
) const
4566 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4568 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4570 if (orient
== wxHORIZONTAL
)
4571 return (int)(m_hAdjust
->value
+0.5);
4573 return (int)(m_vAdjust
->value
+0.5);
4576 int wxWindowGTK::GetScrollRange( int orient
) const
4578 wxCHECK_MSG( m_widget
!= NULL
, 0, wxT("invalid window") );
4580 wxCHECK_MSG( m_wxwindow
!= NULL
, 0, wxT("window needs client area for scrolling") );
4582 if (orient
== wxHORIZONTAL
)
4583 return (int)(m_hAdjust
->upper
+0.5);
4585 return (int)(m_vAdjust
->upper
+0.5);
4588 void wxWindowGTK::ScrollWindow( int dx
, int dy
, const wxRect
* WXUNUSED(rect
) )
4590 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid window") );
4592 wxCHECK_RET( m_wxwindow
!= NULL
, wxT("window needs client area for scrolling") );
4594 // No scrolling requested.
4595 if ((dx
== 0) && (dy
== 0)) return;
4598 if (!m_updateRegion
.IsEmpty())
4600 m_updateRegion
.Offset( dx
, dy
);
4604 GetClientSize( &cw
, &ch
);
4605 m_updateRegion
.Intersect( 0, 0, cw
, ch
);
4608 if (!m_clearRegion
.IsEmpty())
4610 m_clearRegion
.Offset( dx
, dy
);
4614 GetClientSize( &cw
, &ch
);
4615 m_clearRegion
.Intersect( 0, 0, cw
, ch
);
4619 m_clipPaintRegion
= TRUE
;
4621 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow
), -dx
, -dy
);
4623 m_clipPaintRegion
= FALSE
;
4627 // Find the wxWindow at the current mouse position, also returning the mouse
4629 wxWindow
* wxFindWindowAtPointer(wxPoint
& pt
)
4631 pt
= wxGetMousePosition();
4632 wxWindow
* found
= wxFindWindowAtPoint(pt
);
4636 // Get the current mouse position.
4637 wxPoint
wxGetMousePosition()
4639 /* This crashes when used within wxHelpContext,
4640 so we have to use the X-specific implementation below.
4642 GdkModifierType *mask;
4643 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4645 return wxPoint(x, y);
4649 GdkWindow
* windowAtPtr
= gdk_window_at_pointer(& x
, & y
);
4651 Display
*display
= windowAtPtr
? GDK_WINDOW_XDISPLAY(windowAtPtr
) : GDK_DISPLAY();
4652 Window rootWindow
= RootWindowOfScreen (DefaultScreenOfDisplay(display
));
4653 Window rootReturn
, childReturn
;
4654 int rootX
, rootY
, winX
, winY
;
4655 unsigned int maskReturn
;
4657 XQueryPointer (display
,
4661 &rootX
, &rootY
, &winX
, &winY
, &maskReturn
);
4662 return wxPoint(rootX
, rootY
);
4666 // ----------------------------------------------------------------------------
4668 // ----------------------------------------------------------------------------
4670 class wxWinModule
: public wxModule
4677 DECLARE_DYNAMIC_CLASS(wxWinModule
)
4680 IMPLEMENT_DYNAMIC_CLASS(wxWinModule
, wxModule
)
4682 bool wxWinModule::OnInit()
4684 // g_eraseGC = gdk_gc_new( GDK_ROOT_PARENT() );
4685 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
4690 void wxWinModule::OnExit()
4693 gdk_gc_unref( g_eraseGC
);