///////////////////////////////////////////////////////////////////////////// // Name: src/gtk/window.cpp // Purpose: wxWindowGTK implementation // Author: Robert Roebling // Copyright: (c) 1998 Robert Roebling, Julian Smart // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #ifdef __VMS #define XWarpPointer XWARPPOINTER #endif #include "wx/window.h" #ifndef WX_PRECOMP #include "wx/log.h" #include "wx/app.h" #include "wx/toplevel.h" #include "wx/dcclient.h" #include "wx/menu.h" #include "wx/settings.h" #include "wx/msgdlg.h" #include "wx/math.h" #endif #include "wx/dnd.h" #include "wx/tooltip.h" #include "wx/caret.h" #include "wx/fontutil.h" #include "wx/sysopt.h" #ifdef __WXGTK3__ #include "wx/gtk/dc.h" #endif #include #include #include "wx/gtk/private.h" #include "wx/gtk/private/gtk2-compat.h" #include "wx/gtk/private/event.h" #include "wx/gtk/private/win_gtk.h" #include "wx/private/textmeasure.h" using namespace wxGTKImpl; #ifdef GDK_WINDOWING_X11 #include #include "wx/x11/private/wrapxkb.h" #else typedef guint KeySym; #endif #include #ifdef __WXGTK3__ #include #endif // gdk_window_set_composited() is only supported since 2.12 #define wxGTK_VERSION_REQUIRED_FOR_COMPOSITING 2,12,0 #define wxGTK_HAS_COMPOSITING_SUPPORT (GTK_CHECK_VERSION(2,12,0) && wxUSE_CAIRO) //----------------------------------------------------------------------------- // documentation on internals //----------------------------------------------------------------------------- /* I have been asked several times about writing some documentation about the GTK port of wxWidgets, especially its internal structures. Obviously, you cannot understand wxGTK without knowing a little about the GTK, but some more information about what the wxWindow, which is the base class for all other window classes, does seems required as well. I) What does wxWindow do? It contains the common interface for the following jobs of its descendants: 1) Define the rudimentary behaviour common to all window classes, such as resizing, intercepting user input (so as to make it possible to use these events for special purposes in a derived class), window names etc. 2) Provide the possibility to contain and manage children, if the derived class is allowed to contain children, which holds true for those window classes which do not display a native GTK widget. To name them, these classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame- work classes are a special case and are handled a bit differently from the rest. The same holds true for the wxNotebook class. 3) Provide the possibility to draw into a client area of a window. This, too, only holds true for classes that do not display a native GTK widget as above. 4) Provide the entire mechanism for scrolling widgets. This actual inter- face for this is usually in wxScrolledWindow, but the GTK implementation is in this class. 5) A multitude of helper or extra methods for special purposes, such as Drag'n'Drop, managing validators etc. 6) Display a border (sunken, raised, simple or none). Normally one might expect, that one wxWidgets window would always correspond to one GTK widget. Under GTK, there is no such all-round widget that has all the functionality. Moreover, the GTK defines a client area as a different widget from the actual widget you are handling. Last but not least some special classes (e.g. wxFrame) handle different categories of widgets and still have the possibility to draw something in the client area. It was therefore required to write a special purpose GTK widget, that would represent a client area in the sense of wxWidgets capable to do the jobs 2), 3) and 4). I have written this class and it resides in win_gtk.c of this directory. All windows must have a widget, with which they interact with other under- lying GTK widgets. It is this widget, e.g. that has to be resized etc and the wxWindow class has a member variable called m_widget which holds a pointer to this widget. When the window class represents a GTK native widget, this is (in most cases) the only GTK widget the class manages. E.g. the wxStaticText class handles only a GtkLabel widget a pointer to which you can find in m_widget (defined in wxWindow) When the class has a client area for drawing into and for containing children it has to handle the client area widget (of the type wxPizza, defined in win_gtk.cpp), but there could be any number of widgets, handled by a class. The common rule for all windows is only, that the widget that interacts with the rest of GTK must be referenced in m_widget and all other widgets must be children of this widget on the GTK level. The top-most widget, which also represents the client area, must be in the m_wxwindow field and must be of the type wxPizza. As I said, the window classes that display a GTK native widget only have one widget, so in the case of e.g. the wxButton class m_widget holds a pointer to a GtkButton widget. But windows with client areas (for drawing and children) have a m_widget field that is a pointer to a GtkScrolled- Window and a m_wxwindow field that is pointer to a wxPizza and this one is (in the GTK sense) a child of the GtkScrolledWindow. If the m_wxwindow field is set, then all input to this widget is inter- cepted and sent to the wxWidgets class. If not, all input to the widget that gets pointed to by m_widget gets intercepted and sent to the class. II) The design of scrolling in wxWidgets is markedly different from that offered by the GTK itself and therefore we cannot simply take it as it is. In GTK, clicking on a scrollbar belonging to scrolled window will inevitably move the window. In wxWidgets, the scrollbar will only emit an event, send this to (normally) a wxScrolledWindow and that class will call ScrollWindow() which actually moves the window and its sub-windows. Note that wxPizza memorizes how much it has been scrolled but that wxWidgets forgets this so that the two coordinates systems have to be kept in synch. This is done in various places using the pizza->m_scroll_x and pizza->m_scroll_y values. III) Singularly the most broken code in GTK is the code that is supposed to inform subwindows (child windows) about new positions. Very often, duplicate events are sent without changes in size or position, equally often no events are sent at all (All this is due to a bug in the GtkContainer code which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores GTK's own system and it simply waits for size events for toplevel windows and then iterates down the respective size events to all window. This has the disadvantage that windows might get size events before the GTK widget actually has the reported size. This doesn't normally pose any problem, but the OpenGL drawing routines rely on correct behaviour. Therefore, I have added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas, i.e. the wxGLCanvas will emit a size event, when (and not before) the X11 window that is used for OpenGL output really has that size (as reported by GTK). IV) If someone at some point of time feels the immense desire to have a look at, change or attempt to optimise the Refresh() logic, this person will need an intimate understanding of what "draw" and "expose" events are and what they are used for, in particular when used in connection with GTK's own windowless widgets. Beware. V) Cursors, too, have been a constant source of pleasure. The main difficulty is that a GdkWindow inherits a cursor if the programmer sets a new cursor for the parent. To prevent this from doing too much harm, SetCursor calls GTKUpdateCursor, which will recursively re-set the cursors of all child windows. Also don't forget that cursors (like much else) are connected to GdkWindows, not GtkWidgets and that the "window" field of a GtkWidget might very well point to the GdkWindow of the parent widget (-> "window-less widget") and that the two obviously have very different meanings. */ //----------------------------------------------------------------------------- // data //----------------------------------------------------------------------------- // Don't allow event propagation during drag bool g_blockEventsOnDrag; // Don't allow mouse event propagation during scroll bool g_blockEventsOnScroll; extern wxCursor g_globalCursor; // mouse capture state: the window which has it and if the mouse is currently // inside it static wxWindowGTK *g_captureWindow = NULL; static bool g_captureWindowHasMouse = false; // The window that currently has focus: static wxWindowGTK *gs_currentFocus = NULL; // The window that is scheduled to get focus in the next event loop iteration // or NULL if there's no pending focus change: static wxWindowGTK *gs_pendingFocus = NULL; // the window that has deferred focus-out event pending, if any (see // GTKAddDeferredFocusOut() for details) static wxWindowGTK *gs_deferredFocusOut = NULL; // global variables because GTK+ DnD want to have the // mouse event that caused it GdkEvent *g_lastMouseEvent = NULL; int g_lastButtonNumber = 0; #ifdef __WXGTK3__ static GList* gs_sizeRevalidateList; GList* wx_sizeEventList; static bool gs_inSizeAllocate; void wxGTKSizeRevalidate(wxWindow*); #endif //----------------------------------------------------------------------------- // debug //----------------------------------------------------------------------------- // the trace mask used for the focus debugging messages #define TRACE_FOCUS wxT("focus") // A handy function to run from under gdb to show information about the given // GtkWidget. Right now it only shows its type, we could enhance it to show // more information later but this is already pretty useful. const char* wxDumpGtkWidget(GtkWidget* w) { static wxString s; s.Printf("GtkWidget %p, type \"%s\"", w, G_OBJECT_TYPE_NAME(w)); return s.c_str(); } //----------------------------------------------------------------------------- // "expose_event"/"draw" from m_wxwindow //----------------------------------------------------------------------------- extern "C" { #ifdef __WXGTK3__ static gboolean draw(GtkWidget*, cairo_t* cr, wxWindow* win) { if (gtk_cairo_should_draw_window(cr, win->GTKGetDrawingWindow())) win->GTKSendPaintEvents(cr); return false; } #else // !__WXGTK3__ static gboolean expose_event(GtkWidget*, GdkEventExpose* gdk_event, wxWindow* win) { if (gdk_event->window == win->GTKGetDrawingWindow()) win->GTKSendPaintEvents(gdk_event->region); return false; } #endif // !__WXGTK3__ } #ifndef __WXUNIVERSAL__ //----------------------------------------------------------------------------- // "expose_event"/"draw" from m_wxwindow->parent, for drawing border //----------------------------------------------------------------------------- extern "C" { static gboolean #ifdef __WXGTK3__ draw_border(GtkWidget*, cairo_t* cr, wxWindow* win) #else draw_border(GtkWidget* widget, GdkEventExpose* gdk_event, wxWindow* win) #endif { #ifdef __WXGTK3__ if (!gtk_cairo_should_draw_window(cr, gtk_widget_get_parent_window(win->m_wxwindow))) #else if (gdk_event->window != gtk_widget_get_parent_window(win->m_wxwindow)) #endif return false; if (!win->IsShown()) return false; GtkAllocation alloc; gtk_widget_get_allocation(win->m_wxwindow, &alloc); const int x = alloc.x; const int y = alloc.y; const int w = alloc.width; const int h = alloc.height; if (w <= 0 || h <= 0) return false; if (win->HasFlag(wxBORDER_SIMPLE)) { #ifdef __WXGTK3__ GtkStyleContext* sc = gtk_widget_get_style_context(win->m_wxwindow); GdkRGBA c; gtk_style_context_save(sc); gtk_style_context_set_state(sc, GTK_STATE_FLAG_NORMAL); gtk_style_context_get_border_color(sc, GTK_STATE_FLAG_NORMAL, &c); gtk_style_context_restore(sc); gdk_cairo_set_source_rgba(cr, &c); cairo_set_line_width(cr, 1); cairo_rectangle(cr, x + 0.5, y + 0.5, w - 1, h - 1); cairo_stroke(cr); #else gdk_draw_rectangle(gdk_event->window, gtk_widget_get_style(widget)->black_gc, false, x, y, w - 1, h - 1); #endif } else if (win->HasFlag(wxBORDER_RAISED | wxBORDER_SUNKEN | wxBORDER_THEME)) { #ifdef __WXGTK3__ //TODO: wxBORDER_RAISED/wxBORDER_SUNKEN GtkStyleContext* sc; if (win->HasFlag(wxHSCROLL | wxVSCROLL)) sc = gtk_widget_get_style_context(wxGTKPrivate::GetTreeWidget()); else sc = gtk_widget_get_style_context(wxGTKPrivate::GetEntryWidget()); gtk_render_frame(sc, cr, x, y, w, h); #else // !__WXGTK3__ GtkShadowType shadow = GTK_SHADOW_IN; if (win->HasFlag(wxBORDER_RAISED)) shadow = GTK_SHADOW_OUT; GtkStyle* style; const char* detail; if (win->HasFlag(wxHSCROLL | wxVSCROLL)) { style = gtk_widget_get_style(wxGTKPrivate::GetTreeWidget()); detail = "viewport"; } else { style = gtk_widget_get_style(wxGTKPrivate::GetEntryWidget()); detail = "entry"; } // clip rect is required to avoid painting background // over upper left (w,h) of parent window GdkRectangle clipRect = { x, y, w, h }; gtk_paint_shadow( style, gdk_event->window, GTK_STATE_NORMAL, shadow, &clipRect, widget, detail, x, y, w, h); #endif // !__WXGTK3__ } return false; } } //----------------------------------------------------------------------------- // "parent_set" from m_wxwindow //----------------------------------------------------------------------------- extern "C" { static void parent_set(GtkWidget* widget, GtkWidget* old_parent, wxWindow* win) { if (old_parent) { g_signal_handlers_disconnect_by_func( old_parent, (void*)draw_border, win); } GtkWidget* parent = gtk_widget_get_parent(widget); if (parent) { #ifdef __WXGTK3__ g_signal_connect_after(parent, "draw", G_CALLBACK(draw_border), win); #else g_signal_connect_after(parent, "expose_event", G_CALLBACK(draw_border), win); #endif } } } #endif // !__WXUNIVERSAL__ //----------------------------------------------------------------------------- // "key_press_event" from any window //----------------------------------------------------------------------------- // set WXTRACE to this to see the key event codes on the console #define TRACE_KEYS wxT("keyevent") // translates an X key symbol to WXK_XXX value // // if isChar is true it means that the value returned will be used for EVT_CHAR // event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide, // for example, while if it is false it means that the value is going to be // used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to // WXK_NUMPAD_DIVIDE static long wxTranslateKeySymToWXKey(KeySym keysym, bool isChar) { long key_code; switch ( keysym ) { // Shift, Control and Alt don't generate the CHAR events at all case GDK_Shift_L: case GDK_Shift_R: key_code = isChar ? 0 : WXK_SHIFT; break; case GDK_Control_L: case GDK_Control_R: key_code = isChar ? 0 : WXK_CONTROL; break; case GDK_Meta_L: case GDK_Meta_R: case GDK_Alt_L: case GDK_Alt_R: case GDK_Super_L: case GDK_Super_R: key_code = isChar ? 0 : WXK_ALT; break; // neither do the toggle modifies case GDK_Scroll_Lock: key_code = isChar ? 0 : WXK_SCROLL; break; case GDK_Caps_Lock: key_code = isChar ? 0 : WXK_CAPITAL; break; case GDK_Num_Lock: key_code = isChar ? 0 : WXK_NUMLOCK; break; // various other special keys case GDK_Menu: key_code = WXK_MENU; break; case GDK_Help: key_code = WXK_HELP; break; case GDK_BackSpace: key_code = WXK_BACK; break; case GDK_ISO_Left_Tab: case GDK_Tab: key_code = WXK_TAB; break; case GDK_Linefeed: case GDK_Return: key_code = WXK_RETURN; break; case GDK_Clear: key_code = WXK_CLEAR; break; case GDK_Pause: key_code = WXK_PAUSE; break; case GDK_Select: key_code = WXK_SELECT; break; case GDK_Print: key_code = WXK_PRINT; break; case GDK_Execute: key_code = WXK_EXECUTE; break; case GDK_Escape: key_code = WXK_ESCAPE; break; // cursor and other extended keyboard keys case GDK_Delete: key_code = WXK_DELETE; break; case GDK_Home: key_code = WXK_HOME; break; case GDK_Left: key_code = WXK_LEFT; break; case GDK_Up: key_code = WXK_UP; break; case GDK_Right: key_code = WXK_RIGHT; break; case GDK_Down: key_code = WXK_DOWN; break; case GDK_Prior: // == GDK_Page_Up key_code = WXK_PAGEUP; break; case GDK_Next: // == GDK_Page_Down key_code = WXK_PAGEDOWN; break; case GDK_End: key_code = WXK_END; break; case GDK_Begin: key_code = WXK_HOME; break; case GDK_Insert: key_code = WXK_INSERT; break; // numpad keys case GDK_KP_0: case GDK_KP_1: case GDK_KP_2: case GDK_KP_3: case GDK_KP_4: case GDK_KP_5: case GDK_KP_6: case GDK_KP_7: case GDK_KP_8: case GDK_KP_9: key_code = (isChar ? '0' : int(WXK_NUMPAD0)) + keysym - GDK_KP_0; break; case GDK_KP_Space: key_code = isChar ? ' ' : int(WXK_NUMPAD_SPACE); break; case GDK_KP_Tab: key_code = isChar ? WXK_TAB : WXK_NUMPAD_TAB; break; case GDK_KP_Enter: key_code = isChar ? WXK_RETURN : WXK_NUMPAD_ENTER; break; case GDK_KP_F1: key_code = isChar ? WXK_F1 : WXK_NUMPAD_F1; break; case GDK_KP_F2: key_code = isChar ? WXK_F2 : WXK_NUMPAD_F2; break; case GDK_KP_F3: key_code = isChar ? WXK_F3 : WXK_NUMPAD_F3; break; case GDK_KP_F4: key_code = isChar ? WXK_F4 : WXK_NUMPAD_F4; break; case GDK_KP_Home: key_code = isChar ? WXK_HOME : WXK_NUMPAD_HOME; break; case GDK_KP_Left: key_code = isChar ? WXK_LEFT : WXK_NUMPAD_LEFT; break; case GDK_KP_Up: key_code = isChar ? WXK_UP : WXK_NUMPAD_UP; break; case GDK_KP_Right: key_code = isChar ? WXK_RIGHT : WXK_NUMPAD_RIGHT; break; case GDK_KP_Down: key_code = isChar ? WXK_DOWN : WXK_NUMPAD_DOWN; break; case GDK_KP_Prior: // == GDK_KP_Page_Up key_code = isChar ? WXK_PAGEUP : WXK_NUMPAD_PAGEUP; break; case GDK_KP_Next: // == GDK_KP_Page_Down key_code = isChar ? WXK_PAGEDOWN : WXK_NUMPAD_PAGEDOWN; break; case GDK_KP_End: key_code = isChar ? WXK_END : WXK_NUMPAD_END; break; case GDK_KP_Begin: key_code = isChar ? WXK_HOME : WXK_NUMPAD_BEGIN; break; case GDK_KP_Insert: key_code = isChar ? WXK_INSERT : WXK_NUMPAD_INSERT; break; case GDK_KP_Delete: key_code = isChar ? WXK_DELETE : WXK_NUMPAD_DELETE; break; case GDK_KP_Equal: key_code = isChar ? '=' : int(WXK_NUMPAD_EQUAL); break; case GDK_KP_Multiply: key_code = isChar ? '*' : int(WXK_NUMPAD_MULTIPLY); break; case GDK_KP_Add: key_code = isChar ? '+' : int(WXK_NUMPAD_ADD); break; case GDK_KP_Separator: // FIXME: what is this? key_code = isChar ? '.' : int(WXK_NUMPAD_SEPARATOR); break; case GDK_KP_Subtract: key_code = isChar ? '-' : int(WXK_NUMPAD_SUBTRACT); break; case GDK_KP_Decimal: key_code = isChar ? '.' : int(WXK_NUMPAD_DECIMAL); break; case GDK_KP_Divide: key_code = isChar ? '/' : int(WXK_NUMPAD_DIVIDE); break; // function keys case GDK_F1: case GDK_F2: case GDK_F3: case GDK_F4: case GDK_F5: case GDK_F6: case GDK_F7: case GDK_F8: case GDK_F9: case GDK_F10: case GDK_F11: case GDK_F12: key_code = WXK_F1 + keysym - GDK_F1; break; default: key_code = 0; } return key_code; } static inline bool wxIsAsciiKeysym(KeySym ks) { return ks < 256; } static void wxFillOtherKeyEventFields(wxKeyEvent& event, wxWindowGTK *win, GdkEventKey *gdk_event) { event.SetTimestamp( gdk_event->time ); event.SetId(win->GetId()); event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0; event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0; event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0; event.m_metaDown = (gdk_event->state & GDK_META_MASK) != 0; // At least with current Linux systems, MOD5 corresponds to AltGr key and // we represent it, for consistency with Windows, which really allows to // use Ctrl+Alt as a replacement for AltGr if this key is not present, as a // combination of these two modifiers. if ( gdk_event->state & GDK_MOD5_MASK ) { event.m_controlDown = event.m_altDown = true; } // Normally we take the state of modifiers directly from the low level GDK // event but unfortunately GDK uses a different convention from MSW for the // key events corresponding to the modifier keys themselves: in it, when // e.g. Shift key is pressed, GDK_SHIFT_MASK is not set while it is set // when Shift is released. Under MSW the situation is exactly reversed and // the modifier corresponding to the key is set when it is pressed and // unset when it is released. To ensure consistent behaviour between // platforms (and because it seems to make slightly more sense, although // arguably both behaviours are reasonable) we follow MSW here. // // Final notice: we set the flags to the desired value instead of just // inverting them because they are not set correctly (i.e. in the same way // as for the real events generated by the user) for wxUIActionSimulator- // produced events and it seems better to keep that class code the same // among all platforms and fix the discrepancy here instead of adding // wxGTK-specific code to wxUIActionSimulator. const bool isPress = gdk_event->type == GDK_KEY_PRESS; switch ( gdk_event->keyval ) { case GDK_Shift_L: case GDK_Shift_R: event.m_shiftDown = isPress; break; case GDK_Control_L: case GDK_Control_R: event.m_controlDown = isPress; break; case GDK_Alt_L: case GDK_Alt_R: event.m_altDown = isPress; break; case GDK_Meta_L: case GDK_Meta_R: case GDK_Super_L: case GDK_Super_R: event.m_metaDown = isPress; break; } event.m_rawCode = (wxUint32) gdk_event->keyval; event.m_rawFlags = gdk_event->hardware_keycode; event.SetEventObject( win ); } static bool wxTranslateGTKKeyEventToWx(wxKeyEvent& event, wxWindowGTK *win, GdkEventKey *gdk_event) { // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string // but only event->keyval which is quite useless to us, so remember // the last character from GDK_KEY_PRESS and reuse it as last resort // // NB: should be MT-safe as we're always called from the main thread only static struct { KeySym keysym; long keycode; } s_lastKeyPress = { 0, 0 }; KeySym keysym = gdk_event->keyval; wxLogTrace(TRACE_KEYS, wxT("Key %s event: keysym = %lu"), event.GetEventType() == wxEVT_KEY_UP ? wxT("release") : wxT("press"), static_cast(keysym)); long key_code = wxTranslateKeySymToWXKey(keysym, false /* !isChar */); if ( !key_code ) { // do we have the translation or is it a plain ASCII character? if ( (gdk_event->length == 1) || wxIsAsciiKeysym(keysym) ) { // we should use keysym if it is ASCII as X does some translations // like "I pressed while Control is down" => "Ctrl-I" == "TAB" // which we don't want here (but which we do use for OnChar()) if ( !wxIsAsciiKeysym(keysym) ) { keysym = (KeySym)gdk_event->string[0]; } #ifdef GDK_WINDOWING_X11 if (GDK_IS_X11_DISPLAY(gdk_window_get_display(gdk_event->window))) { // we want to always get the same key code when the same key is // pressed regardless of the state of the modifiers, i.e. on a // standard US keyboard pressing '5' or '%' ('5' key with // Shift) should result in the same key code in OnKeyDown(): // '5' (although OnChar() will get either '5' or '%'). // // to do it we first translate keysym to keycode (== scan code) // and then back but always using the lower register Display *dpy = (Display *)wxGetDisplay(); KeyCode keycode = XKeysymToKeycode(dpy, keysym); wxLogTrace(TRACE_KEYS, wxT("\t-> keycode %d"), keycode); #ifdef HAVE_X11_XKBLIB_H KeySym keysymNormalized = XkbKeycodeToKeysym(dpy, keycode, 0, 0); #else KeySym keysymNormalized = XKeycodeToKeysym(dpy, keycode, 0); #endif // use the normalized, i.e. lower register, keysym if we've // got one key_code = keysymNormalized ? keysymNormalized : keysym; } #else key_code = keysym; #endif // as explained above, we want to have lower register key codes // normally but for the letter keys we want to have the upper ones // // NB: don't use XConvertCase() here, we want to do it for letters // only key_code = toupper(key_code); } else // non ASCII key, what to do? { // by default, ignore it key_code = 0; // but if we have cached information from the last KEY_PRESS if ( gdk_event->type == GDK_KEY_RELEASE ) { // then reuse it if ( keysym == s_lastKeyPress.keysym ) { key_code = s_lastKeyPress.keycode; } } } if ( gdk_event->type == GDK_KEY_PRESS ) { // remember it to be reused for KEY_UP event later s_lastKeyPress.keysym = keysym; s_lastKeyPress.keycode = key_code; } } wxLogTrace(TRACE_KEYS, wxT("\t-> wxKeyCode %ld"), key_code); // sending unknown key events doesn't really make sense if ( !key_code ) return false; event.m_keyCode = key_code; #if wxUSE_UNICODE event.m_uniChar = gdk_keyval_to_unicode(key_code); if ( !event.m_uniChar && event.m_keyCode <= WXK_DELETE ) { // Set Unicode key code to the ASCII equivalent for compatibility. E.g. // let RETURN generate the key event with both key and Unicode key // codes of 13. event.m_uniChar = event.m_keyCode; } #endif // wxUSE_UNICODE // now fill all the other fields wxFillOtherKeyEventFields(event, win, gdk_event); return true; } namespace { // Send wxEVT_CHAR_HOOK event to the parent of the window and return true only // if it was processed (and not skipped). bool SendCharHookEvent(const wxKeyEvent& event, wxWindow *win) { // wxEVT_CHAR_HOOK must be sent to allow the parent windows (e.g. a dialog // which typically closes when Esc key is pressed in any of its controls) // to handle key events in all of its children unless the mouse is captured // in which case we consider that the keyboard should be "captured" too. if ( !g_captureWindow ) { wxKeyEvent eventCharHook(wxEVT_CHAR_HOOK, event); if ( win->HandleWindowEvent(eventCharHook) && !event.IsNextEventAllowed() ) return true; } return false; } // Adjust wxEVT_CHAR event key code fields. This function takes care of two // conventions: // (a) Ctrl-letter key presses generate key codes in range 1..26 // (b) Unicode key codes are same as key codes for the codes in 1..255 range void AdjustCharEventKeyCodes(wxKeyEvent& event) { const int code = event.m_keyCode; // Check for (a) above. if ( event.ControlDown() ) { // We intentionally don't use isupper/lower() here, we really need // ASCII letters only as it doesn't make sense to translate any other // ones into this range which has only 26 slots. if ( code >= 'a' && code <= 'z' ) event.m_keyCode = code - 'a' + 1; else if ( code >= 'A' && code <= 'Z' ) event.m_keyCode = code - 'A' + 1; #if wxUSE_UNICODE // Adjust the Unicode equivalent in the same way too. if ( event.m_keyCode != code ) event.m_uniChar = event.m_keyCode; #endif // wxUSE_UNICODE } #if wxUSE_UNICODE // Check for (b) from above. // // FIXME: Should we do it for key codes up to 255? if ( !event.m_uniChar && code < WXK_DELETE ) event.m_uniChar = code; #endif // wxUSE_UNICODE } } // anonymous namespace // If a widget does not handle a key or mouse event, GTK+ sends it up the // parent chain until it is handled. These events are not supposed to propagate // in wxWidgets, so this code avoids handling them in any parent wxWindow, // while still allowing the event to propagate so things like native keyboard // navigation will work. #define wxPROCESS_EVENT_ONCE(EventType, event) \ static EventType eventPrev; \ if (!gs_isNewEvent && memcmp(&eventPrev, event, sizeof(EventType)) == 0) \ return false; \ gs_isNewEvent = false; \ eventPrev = *event static bool gs_isNewEvent; extern "C" { static gboolean gtk_window_key_press_callback( GtkWidget *WXUNUSED(widget), GdkEventKey *gdk_event, wxWindow *win ) { if (g_blockEventsOnDrag) return FALSE; wxPROCESS_EVENT_ONCE(GdkEventKey, gdk_event); wxKeyEvent event( wxEVT_KEY_DOWN ); bool ret = false; bool return_after_IM = false; if( wxTranslateGTKKeyEventToWx(event, win, gdk_event) ) { // Send the CHAR_HOOK event first if ( SendCharHookEvent(event, win) ) { // Don't do anything at all with this event any more. return TRUE; } // Next check for accelerators. #if wxUSE_ACCEL wxWindowGTK *ancestor = win; while (ancestor) { int command = ancestor->GetAcceleratorTable()->GetCommand( event ); if (command != -1) { wxCommandEvent menu_event( wxEVT_MENU, command ); ret = ancestor->HandleWindowEvent( menu_event ); if ( !ret ) { // if the accelerator wasn't handled as menu event, try // it as button click (for compatibility with other // platforms): wxCommandEvent button_event( wxEVT_BUTTON, command ); ret = ancestor->HandleWindowEvent( button_event ); } break; } if (ancestor->IsTopLevel()) break; ancestor = ancestor->GetParent(); } #endif // wxUSE_ACCEL // If not an accelerator, then emit KEY_DOWN event if ( !ret ) ret = win->HandleWindowEvent( event ); } else { // Return after IM processing as we cannot do // anything with it anyhow. return_after_IM = true; } if ( !ret ) { // Indicate that IM handling is in process by setting this pointer // (which will remain valid for all the code called during IM key // handling). win->m_imKeyEvent = gdk_event; // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API // docs, if IM filter returns true, no further processing should be done. // we should send the key_down event anyway. const int intercepted_by_IM = win->GTKIMFilterKeypress(gdk_event); win->m_imKeyEvent = NULL; if ( intercepted_by_IM ) { wxLogTrace(TRACE_KEYS, wxT("Key event intercepted by IM")); return TRUE; } } if (return_after_IM) return FALSE; // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x // will only be sent if it is not in an accelerator table. if (!ret) { KeySym keysym = gdk_event->keyval; // Find key code for EVT_CHAR and EVT_CHAR_HOOK events long key_code = wxTranslateKeySymToWXKey(keysym, true /* isChar */); if ( !key_code ) { if ( wxIsAsciiKeysym(keysym) ) { // ASCII key key_code = (unsigned char)keysym; } // gdk_event->string is actually deprecated else if ( gdk_event->length == 1 ) { key_code = (unsigned char)gdk_event->string[0]; } } if ( key_code ) { wxKeyEvent eventChar(wxEVT_CHAR, event); wxLogTrace(TRACE_KEYS, wxT("Char event: %ld"), key_code); eventChar.m_keyCode = key_code; #if wxUSE_UNICODE eventChar.m_uniChar = gdk_keyval_to_unicode(key_code); #endif // wxUSE_UNICODE AdjustCharEventKeyCodes(eventChar); ret = win->HandleWindowEvent(eventChar); } } return ret; } } int wxWindowGTK::GTKIMFilterKeypress(GdkEventKey* event) const { return m_imContext ? gtk_im_context_filter_keypress(m_imContext, event) : FALSE; } extern "C" { static void gtk_wxwindow_commit_cb (GtkIMContext * WXUNUSED(context), const gchar *str, wxWindow *window) { // Ignore the return value here, it doesn't matter for the "commit" signal. window->GTKDoInsertTextFromIM(str); } } bool wxWindowGTK::GTKDoInsertTextFromIM(const char* str) { wxKeyEvent event( wxEVT_CHAR ); // take modifiers, cursor position, timestamp etc. from the last // key_press_event that was fed into Input Method: if ( m_imKeyEvent ) { wxFillOtherKeyEventFields(event, this, m_imKeyEvent); } else { event.SetEventObject(this); } const wxString data(wxGTK_CONV_BACK_SYS(str)); if( data.empty() ) return false; bool processed = false; for( wxString::const_iterator pstr = data.begin(); pstr != data.end(); ++pstr ) { #if wxUSE_UNICODE event.m_uniChar = *pstr; // Backward compatible for ISO-8859-1 event.m_keyCode = *pstr < 256 ? event.m_uniChar : 0; wxLogTrace(TRACE_KEYS, wxT("IM sent character '%c'"), event.m_uniChar); #else event.m_keyCode = (char)*pstr; #endif // wxUSE_UNICODE AdjustCharEventKeyCodes(event); if ( HandleWindowEvent(event) ) processed = true; } return processed; } bool wxWindowGTK::GTKOnInsertText(const char* text) { if ( !m_imKeyEvent ) { // We're not inside IM key handling at all. return false; } return GTKDoInsertTextFromIM(text); } //----------------------------------------------------------------------------- // "key_release_event" from any window //----------------------------------------------------------------------------- extern "C" { static gboolean gtk_window_key_release_callback( GtkWidget * WXUNUSED(widget), GdkEventKey *gdk_event, wxWindowGTK *win ) { if (g_blockEventsOnDrag) return FALSE; wxPROCESS_EVENT_ONCE(GdkEventKey, gdk_event); wxKeyEvent event( wxEVT_KEY_UP ); if ( !wxTranslateGTKKeyEventToWx(event, win, gdk_event) ) { // unknown key pressed, ignore (the event would be useless anyhow) return FALSE; } return win->GTKProcessEvent(event); } } // ============================================================================ // the mouse events // ============================================================================ // ---------------------------------------------------------------------------- // mouse event processing helpers // ---------------------------------------------------------------------------- static void AdjustEventButtonState(wxMouseEvent& event) { // GDK reports the old state of the button for a button press event, but // for compatibility with MSW and common sense we want m_leftDown be TRUE // for a LEFT_DOWN event, not FALSE, so we will invert // left/right/middleDown for the corresponding click events if ((event.GetEventType() == wxEVT_LEFT_DOWN) || (event.GetEventType() == wxEVT_LEFT_DCLICK) || (event.GetEventType() == wxEVT_LEFT_UP)) { event.m_leftDown = !event.m_leftDown; return; } if ((event.GetEventType() == wxEVT_MIDDLE_DOWN) || (event.GetEventType() == wxEVT_MIDDLE_DCLICK) || (event.GetEventType() == wxEVT_MIDDLE_UP)) { event.m_middleDown = !event.m_middleDown; return; } if ((event.GetEventType() == wxEVT_RIGHT_DOWN) || (event.GetEventType() == wxEVT_RIGHT_DCLICK) || (event.GetEventType() == wxEVT_RIGHT_UP)) { event.m_rightDown = !event.m_rightDown; return; } if ((event.GetEventType() == wxEVT_AUX1_DOWN) || (event.GetEventType() == wxEVT_AUX1_DCLICK)) { event.m_aux1Down = true; return; } if ((event.GetEventType() == wxEVT_AUX2_DOWN) || (event.GetEventType() == wxEVT_AUX2_DCLICK)) { event.m_aux2Down = true; return; } } // find the window to send the mouse event to static wxWindowGTK *FindWindowForMouseEvent(wxWindowGTK *win, wxCoord& x, wxCoord& y) { wxCoord xx = x; wxCoord yy = y; if (win->m_wxwindow) { wxPizza* pizza = WX_PIZZA(win->m_wxwindow); xx += pizza->m_scroll_x; yy += pizza->m_scroll_y; } wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst(); while (node) { wxWindow* child = static_cast(node->GetData()); node = node->GetNext(); if (!child->IsShown()) continue; if (child->GTKIsTransparentForMouse()) { // wxStaticBox is transparent in the box itself int xx1 = child->m_x; int yy1 = child->m_y; int xx2 = child->m_x + child->m_width; int yy2 = child->m_y + child->m_height; // left if (((xx >= xx1) && (xx <= xx1+10) && (yy >= yy1) && (yy <= yy2)) || // right ((xx >= xx2-10) && (xx <= xx2) && (yy >= yy1) && (yy <= yy2)) || // top ((xx >= xx1) && (xx <= xx2) && (yy >= yy1) && (yy <= yy1+10)) || // bottom ((xx >= xx1) && (xx <= xx2) && (yy >= yy2-1) && (yy <= yy2))) { win = child; x -= child->m_x; y -= child->m_y; break; } } else { if ((child->m_wxwindow == NULL) && win->IsClientAreaChild(child) && (child->m_x <= xx) && (child->m_y <= yy) && (child->m_x+child->m_width >= xx) && (child->m_y+child->m_height >= yy)) { win = child; x -= child->m_x; y -= child->m_y; break; } } } return win; } // ---------------------------------------------------------------------------- // common event handlers helpers // ---------------------------------------------------------------------------- bool wxWindowGTK::GTKProcessEvent(wxEvent& event) const { // nothing special at this level return HandleWindowEvent(event); } bool wxWindowGTK::GTKShouldIgnoreEvent() const { return g_blockEventsOnDrag; } int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny *event) const { if (g_blockEventsOnDrag) return TRUE; if (g_blockEventsOnScroll) return TRUE; if (!GTKIsOwnWindow(event->window)) return FALSE; return -1; } // overloads for all GDK event types we use here: we need to have this as // GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact, // derives from it in the sense that the structs have the same layout #define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \ static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \ { \ return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \ } wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton) wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion) wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing) #undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD #define wxCOMMON_CALLBACK_PROLOGUE(event, win) \ const int rc = wxGtkCallbackCommonPrologue(event, win); \ if ( rc != -1 ) \ return rc // all event handlers must have C linkage as they're called from GTK+ C code extern "C" { //----------------------------------------------------------------------------- // "button_press_event" //----------------------------------------------------------------------------- static gboolean gtk_window_button_press_callback( GtkWidget* WXUNUSED_IN_GTK3(widget), GdkEventButton *gdk_event, wxWindowGTK *win ) { wxPROCESS_EVENT_ONCE(GdkEventButton, gdk_event); wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win); g_lastButtonNumber = gdk_event->button; wxEventType event_type; wxEventType down; wxEventType dclick; switch (gdk_event->button) { case 1: down = wxEVT_LEFT_DOWN; dclick = wxEVT_LEFT_DCLICK; break; case 2: down = wxEVT_MIDDLE_DOWN; dclick = wxEVT_MIDDLE_DCLICK; break; case 3: down = wxEVT_RIGHT_DOWN; dclick = wxEVT_RIGHT_DCLICK; break; case 8: down = wxEVT_AUX1_DOWN; dclick = wxEVT_AUX1_DCLICK; break; case 9: down = wxEVT_AUX2_DOWN; dclick = wxEVT_AUX2_DCLICK; break; default: return false; } switch (gdk_event->type) { case GDK_BUTTON_PRESS: event_type = down; // GDK sends surplus button down events // before a double click event. We // need to filter these out. if (win->m_wxwindow) { GdkEvent* peek_event = gdk_event_peek(); if (peek_event) { const GdkEventType peek_event_type = peek_event->type; gdk_event_free(peek_event); if (peek_event_type == GDK_2BUTTON_PRESS || peek_event_type == GDK_3BUTTON_PRESS) { return true; } } } break; case GDK_2BUTTON_PRESS: event_type = dclick; #ifndef __WXGTK3__ if (gdk_event->button >= 1 && gdk_event->button <= 3) { // Reset GDK internal timestamp variables in order to disable GDK // triple click events. GDK will then next time believe no button has // been clicked just before, and send a normal button click event. GdkDisplay* display = gtk_widget_get_display(widget); display->button_click_time[1] = 0; display->button_click_time[0] = 0; } #endif // !__WXGTK3__ break; // we shouldn't get triple clicks at all for GTK2 because we // suppress them artificially using the code above but we still // should map them to something for GTK3 and not just ignore them // as this would lose clicks case GDK_3BUTTON_PRESS: event_type = down; break; default: return false; } g_lastMouseEvent = (GdkEvent*) gdk_event; wxMouseEvent event( event_type ); InitMouseEvent( win, event, gdk_event ); AdjustEventButtonState(event); // find the correct window to send the event to: it may be a different one // from the one which got it at GTK+ level because some controls don't have // their own X window and thus cannot get any events. if ( !g_captureWindow ) win = FindWindowForMouseEvent(win, event.m_x, event.m_y); // reset the event object and id in case win changed. event.SetEventObject( win ); event.SetId( win->GetId() ); bool ret = win->GTKProcessEvent( event ); g_lastMouseEvent = NULL; if ( ret ) return TRUE; if ((event_type == wxEVT_LEFT_DOWN) && !win->IsOfStandardClass() && (gs_currentFocus != win) /* && win->IsFocusable() */) { win->SetFocus(); } if (event_type == wxEVT_RIGHT_DOWN) { // generate a "context menu" event: this is similar to right mouse // click under many GUIs except that it is generated differently // (right up under MSW, ctrl-click under Mac, right down here) and // // (a) it's a command event and so is propagated to the parent // (b) under some ports it can be generated from kbd too // (c) it uses screen coords (because of (a)) wxContextMenuEvent evtCtx( wxEVT_CONTEXT_MENU, win->GetId(), win->ClientToScreen(event.GetPosition())); evtCtx.SetEventObject(win); return win->GTKProcessEvent(evtCtx); } return FALSE; } //----------------------------------------------------------------------------- // "button_release_event" //----------------------------------------------------------------------------- static gboolean gtk_window_button_release_callback( GtkWidget *WXUNUSED(widget), GdkEventButton *gdk_event, wxWindowGTK *win ) { wxPROCESS_EVENT_ONCE(GdkEventButton, gdk_event); wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win); g_lastButtonNumber = 0; wxEventType event_type = wxEVT_NULL; switch (gdk_event->button) { case 1: event_type = wxEVT_LEFT_UP; break; case 2: event_type = wxEVT_MIDDLE_UP; break; case 3: event_type = wxEVT_RIGHT_UP; break; case 8: event_type = wxEVT_AUX1_UP; break; case 9: event_type = wxEVT_AUX2_UP; break; default: // unknown button, don't process return FALSE; } g_lastMouseEvent = (GdkEvent*) gdk_event; wxMouseEvent event( event_type ); InitMouseEvent( win, event, gdk_event ); AdjustEventButtonState(event); if ( !g_captureWindow ) win = FindWindowForMouseEvent(win, event.m_x, event.m_y); // reset the event object and id in case win changed. event.SetEventObject( win ); event.SetId( win->GetId() ); // We ignore the result of the event processing here as we don't really // want to prevent the other handlers from running even if we did process // this event ourselves, there is no real advantage in doing this and it // could actually be harmful, see #16055. (void)win->GTKProcessEvent(event); g_lastMouseEvent = NULL; return FALSE; } //----------------------------------------------------------------------------- WX_DECLARE_VOIDPTR_HASH_MAP(bool, wxVoidPtrBoolMap); static wxVoidPtrBoolMap gs_needCursorResetMap; static const wxCursor* gs_overrideCursor; static void SendSetCursorEvent(wxWindowGTK* win, int x, int y) { wxPoint posClient(x, y); const wxPoint posScreen = win->ClientToScreen(posClient); wxWindowGTK* w = win; for ( ;; ) { wxSetCursorEvent event(posClient.x, posClient.y); if (w->GTKProcessEvent(event)) { gs_overrideCursor = &event.GetCursor(); win->GTKUpdateCursor(); gs_overrideCursor = NULL; gs_needCursorResetMap[win] = true; return; } // this is how wxMSW works... if (w->GetCursor().IsOk()) break; w = w->GetParent(); if (w == NULL || w->m_widget == NULL || !gtk_widget_get_visible(w->m_widget)) break; posClient = w->ScreenToClient(posScreen); } if (gs_needCursorResetMap[win]) win->GTKUpdateCursor(); } //----------------------------------------------------------------------------- // "motion_notify_event" //----------------------------------------------------------------------------- static gboolean gtk_window_motion_notify_callback( GtkWidget * WXUNUSED(widget), GdkEventMotion *gdk_event, wxWindowGTK *win ) { wxPROCESS_EVENT_ONCE(GdkEventMotion, gdk_event); wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win); if (gdk_event->is_hint) { int x = 0; int y = 0; #ifdef __WXGTK3__ gdk_window_get_device_position(gdk_event->window, gdk_event->device, &x, &y, NULL); #else gdk_window_get_pointer(gdk_event->window, &x, &y, NULL); #endif gdk_event->x = x; gdk_event->y = y; } g_lastMouseEvent = (GdkEvent*) gdk_event; wxMouseEvent event( wxEVT_MOTION ); InitMouseEvent(win, event, gdk_event); if ( g_captureWindow ) { // synthesise a mouse enter or leave event if needed GdkWindow* winUnderMouse = #ifdef __WXGTK3__ gdk_device_get_window_at_position(gdk_event->device, NULL, NULL); #else gdk_window_at_pointer(NULL, NULL); #endif // This seems to be necessary and actually been added to // GDK itself in version 2.0.X gdk_flush(); bool hasMouse = winUnderMouse == gdk_event->window; if ( hasMouse != g_captureWindowHasMouse ) { // the mouse changed window g_captureWindowHasMouse = hasMouse; wxMouseEvent eventM(g_captureWindowHasMouse ? wxEVT_ENTER_WINDOW : wxEVT_LEAVE_WINDOW); InitMouseEvent(win, eventM, gdk_event); eventM.SetEventObject(win); win->GTKProcessEvent(eventM); } } else // no capture { win = FindWindowForMouseEvent(win, event.m_x, event.m_y); // reset the event object and id in case win changed. event.SetEventObject( win ); event.SetId( win->GetId() ); } if ( !g_captureWindow ) SendSetCursorEvent(win, event.m_x, event.m_y); bool ret = win->GTKProcessEvent(event); g_lastMouseEvent = NULL; return ret; } //----------------------------------------------------------------------------- // "scroll_event" (mouse wheel event) //----------------------------------------------------------------------------- static void AdjustRangeValue(GtkRange* range, double step) { if (gtk_widget_get_visible(GTK_WIDGET(range))) { GtkAdjustment* adj = gtk_range_get_adjustment(range); double value = gtk_adjustment_get_value(adj); value += step * gtk_adjustment_get_step_increment(adj); gtk_range_set_value(range, value); } } static gboolean scroll_event(GtkWidget* widget, GdkEventScroll* gdk_event, wxWindow* win) { wxMouseEvent event(wxEVT_MOUSEWHEEL); InitMouseEvent(win, event, gdk_event); event.m_wheelDelta = 120; event.m_linesPerAction = 3; event.m_columnsPerAction = 3; GtkRange* range_h = win->m_scrollBar[wxWindow::ScrollDir_Horz]; GtkRange* range_v = win->m_scrollBar[wxWindow::ScrollDir_Vert]; const bool is_range_h = (void*)widget == range_h; const bool is_range_v = (void*)widget == range_v; GdkScrollDirection direction = gdk_event->direction; switch (direction) { case GDK_SCROLL_UP: if (is_range_h) direction = GDK_SCROLL_LEFT; break; case GDK_SCROLL_DOWN: if (is_range_h) direction = GDK_SCROLL_RIGHT; break; case GDK_SCROLL_LEFT: if (is_range_v) direction = GDK_SCROLL_UP; break; case GDK_SCROLL_RIGHT: if (is_range_v) direction = GDK_SCROLL_DOWN; break; default: break; #if GTK_CHECK_VERSION(3,4,0) case GDK_SCROLL_SMOOTH: double delta_x = gdk_event->delta_x; double delta_y = gdk_event->delta_y; if (delta_x == 0) { if (is_range_h) { delta_x = delta_y; delta_y = 0; } } else if (delta_y == 0) { if (is_range_v) { delta_y = delta_x; delta_x = 0; } } bool handled = false; if (delta_x) { event.m_wheelAxis = wxMOUSE_WHEEL_HORIZONTAL; event.m_wheelRotation = int(event.m_wheelDelta * delta_x); handled = win->GTKProcessEvent(event); if (!handled && range_h) { AdjustRangeValue(range_h, event.m_columnsPerAction * delta_x); handled = true; } } if (delta_y) { event.m_wheelAxis = wxMOUSE_WHEEL_VERTICAL; event.m_wheelRotation = int(event.m_wheelDelta * -delta_y); handled = win->GTKProcessEvent(event); if (!handled && range_v) { AdjustRangeValue(range_v, event.m_linesPerAction * delta_y); handled = true; } } return handled; #endif // GTK_CHECK_VERSION(3,4,0) } GtkRange *range; double step; switch (direction) { case GDK_SCROLL_UP: case GDK_SCROLL_DOWN: range = range_v; event.m_wheelAxis = wxMOUSE_WHEEL_VERTICAL; step = event.m_linesPerAction; break; case GDK_SCROLL_LEFT: case GDK_SCROLL_RIGHT: range = range_h; event.m_wheelAxis = wxMOUSE_WHEEL_HORIZONTAL; step = event.m_columnsPerAction; break; default: return false; } event.m_wheelRotation = event.m_wheelDelta; if (direction == GDK_SCROLL_DOWN || direction == GDK_SCROLL_LEFT) event.m_wheelRotation = -event.m_wheelRotation; if (!win->GTKProcessEvent(event)) { if (!range) return false; if (direction == GDK_SCROLL_UP || direction == GDK_SCROLL_LEFT) step = -step; AdjustRangeValue(range, step); } return true; } //----------------------------------------------------------------------------- // "popup-menu" //----------------------------------------------------------------------------- static gboolean wxgtk_window_popup_menu_callback(GtkWidget*, wxWindowGTK* win) { wxContextMenuEvent event(wxEVT_CONTEXT_MENU, win->GetId(), wxPoint(-1, -1)); event.SetEventObject(win); return win->GTKProcessEvent(event); } //----------------------------------------------------------------------------- // "focus_in_event" //----------------------------------------------------------------------------- static gboolean gtk_window_focus_in_callback( GtkWidget * WXUNUSED(widget), GdkEventFocus *WXUNUSED(event), wxWindowGTK *win ) { return win->GTKHandleFocusIn(); } //----------------------------------------------------------------------------- // "focus_out_event" //----------------------------------------------------------------------------- static gboolean gtk_window_focus_out_callback( GtkWidget * WXUNUSED(widget), GdkEventFocus * WXUNUSED(gdk_event), wxWindowGTK *win ) { return win->GTKHandleFocusOut(); } //----------------------------------------------------------------------------- // "focus" //----------------------------------------------------------------------------- static gboolean wx_window_focus_callback(GtkWidget *widget, GtkDirectionType WXUNUSED(direction), wxWindowGTK *win) { // the default handler for focus signal in GtkScrolledWindow sets // focus to the window itself even if it doesn't accept focus, i.e. has no // GTK_CAN_FOCUS in its style -- work around this by forcibly preventing // the signal from reaching gtk_scrolled_window_focus() if we don't have // any children which might accept focus (we know we don't accept the focus // ourselves as this signal is only connected in this case) if ( win->GetChildren().empty() ) g_signal_stop_emission_by_name(widget, "focus"); // we didn't change the focus return FALSE; } //----------------------------------------------------------------------------- // "enter_notify_event" //----------------------------------------------------------------------------- static gboolean gtk_window_enter_callback( GtkWidget*, GdkEventCrossing *gdk_event, wxWindowGTK *win ) { wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win); // Event was emitted after a grab if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE; wxMouseEvent event( wxEVT_ENTER_WINDOW ); InitMouseEvent(win, event, gdk_event); if ( !g_captureWindow ) SendSetCursorEvent(win, event.m_x, event.m_y); return win->GTKProcessEvent(event); } //----------------------------------------------------------------------------- // "leave_notify_event" //----------------------------------------------------------------------------- static gboolean gtk_window_leave_callback( GtkWidget*, GdkEventCrossing *gdk_event, wxWindowGTK *win ) { wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win); if (gs_needCursorResetMap[win]) win->GTKUpdateCursor(); // Event was emitted after an ungrab if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE; wxMouseEvent event( wxEVT_LEAVE_WINDOW ); InitMouseEvent(win, event, gdk_event); return win->GTKProcessEvent(event); } //----------------------------------------------------------------------------- // "value_changed" from scrollbar //----------------------------------------------------------------------------- static void gtk_scrollbar_value_changed(GtkRange* range, wxWindow* win) { wxEventType eventType = win->GTKGetScrollEventType(range); if (eventType != wxEVT_NULL) { // Convert scroll event type to scrollwin event type eventType += wxEVT_SCROLLWIN_TOP - wxEVT_SCROLL_TOP; // find the scrollbar which generated the event wxWindowGTK::ScrollDir dir = win->ScrollDirFromRange(range); // generate the corresponding wx event const int orient = wxWindow::OrientFromScrollDir(dir); wxScrollWinEvent event(eventType, win->GetScrollPos(orient), orient); event.SetEventObject(win); win->GTKProcessEvent(event); } } //----------------------------------------------------------------------------- // "button_press_event" from scrollbar //----------------------------------------------------------------------------- static gboolean gtk_scrollbar_button_press_event(GtkRange*, GdkEventButton*, wxWindow* win) { g_blockEventsOnScroll = true; win->m_mouseButtonDown = true; return false; } //----------------------------------------------------------------------------- // "event_after" from scrollbar //----------------------------------------------------------------------------- static void gtk_scrollbar_event_after(GtkRange* range, GdkEvent* event, wxWindow* win) { if (event->type == GDK_BUTTON_RELEASE) { g_signal_handlers_block_by_func(range, (void*)gtk_scrollbar_event_after, win); const int orient = wxWindow::OrientFromScrollDir( win->ScrollDirFromRange(range)); wxScrollWinEvent evt(wxEVT_SCROLLWIN_THUMBRELEASE, win->GetScrollPos(orient), orient); evt.SetEventObject(win); win->GTKProcessEvent(evt); } } //----------------------------------------------------------------------------- // "button_release_event" from scrollbar //----------------------------------------------------------------------------- static gboolean gtk_scrollbar_button_release_event(GtkRange* range, GdkEventButton*, wxWindow* win) { g_blockEventsOnScroll = false; win->m_mouseButtonDown = false; // If thumb tracking if (win->m_isScrolling) { win->m_isScrolling = false; // Hook up handler to send thumb release event after this emission is finished. // To allow setting scroll position from event handler, sending event must // be deferred until after the GtkRange handler for this signal has run g_signal_handlers_unblock_by_func(range, (void*)gtk_scrollbar_event_after, win); } return false; } //----------------------------------------------------------------------------- // "realize" from m_widget //----------------------------------------------------------------------------- static void gtk_window_realized_callback(GtkWidget* WXUNUSED(widget), wxWindowGTK* win) { win->GTKHandleRealized(); } //----------------------------------------------------------------------------- // "size_allocate" from m_wxwindow or m_widget //----------------------------------------------------------------------------- static void size_allocate(GtkWidget*, GtkAllocation* alloc, wxWindow* win) { int w = alloc->width; int h = alloc->height; if (win->m_wxwindow) { GtkBorder border; WX_PIZZA(win->m_wxwindow)->get_border(border); w -= border.left + border.right; h -= border.top + border.bottom; if (w < 0) w = 0; if (h < 0) h = 0; } GtkAllocation a; gtk_widget_get_allocation(win->m_widget, &a); // update position for widgets in native containers, such as wxToolBar if (!WX_IS_PIZZA(gtk_widget_get_parent(win->m_widget))) { win->m_x = a.x; win->m_y = a.y; } win->m_useCachedClientSize = true; if (win->m_clientWidth != w || win->m_clientHeight != h) { win->m_clientWidth = w; win->m_clientHeight = h; // this callback can be connected to m_wxwindow, // so always get size from m_widget->allocation win->m_width = a.width; win->m_height = a.height; if (!win->m_nativeSizeEvent) { wxSizeEvent event(win->GetSize(), win->GetId()); event.SetEventObject(win); win->GTKProcessEvent(event); } } } //----------------------------------------------------------------------------- // "grab_broken" //----------------------------------------------------------------------------- #if GTK_CHECK_VERSION(2, 8, 0) static gboolean gtk_window_grab_broken( GtkWidget*, GdkEventGrabBroken *event, wxWindow *win ) { // Mouse capture has been lost involuntarily, notify the application if(!event->keyboard && wxWindow::GetCapture() == win) { wxWindowGTK::GTKHandleCaptureLost(); } return false; } #endif //----------------------------------------------------------------------------- // "unrealize" //----------------------------------------------------------------------------- static void unrealize(GtkWidget*, wxWindow* win) { win->GTKHandleUnrealize(); } #if GTK_CHECK_VERSION(3,8,0) //----------------------------------------------------------------------------- // "layout" from GdkFrameClock //----------------------------------------------------------------------------- static void frame_clock_layout(GdkFrameClock*, wxWindow* win) { wxGTKSizeRevalidate(win); } #endif // GTK_CHECK_VERSION(3,8,0) #ifdef __WXGTK3__ //----------------------------------------------------------------------------- // "check-resize" //----------------------------------------------------------------------------- static void check_resize(GtkContainer*, wxWindow*) { gs_inSizeAllocate = true; } static void check_resize_after(GtkContainer*, wxWindow*) { gs_inSizeAllocate = false; } #endif // __WXGTK3__ } // extern "C" void wxWindowGTK::GTKHandleRealized() { GdkWindow* const window = GTKGetDrawingWindow(); if (m_wxwindow) { if (m_imContext == NULL) { // Create input method handler m_imContext = gtk_im_multicontext_new(); // Cannot handle drawing preedited text yet gtk_im_context_set_use_preedit(m_imContext, false); g_signal_connect(m_imContext, "commit", G_CALLBACK(gtk_wxwindow_commit_cb), this); } gtk_im_context_set_client_window(m_imContext, window); } // Use composited window if background is transparent, if supported. if (m_backgroundStyle == wxBG_STYLE_TRANSPARENT) { #if wxGTK_HAS_COMPOSITING_SUPPORT if (IsTransparentBackgroundSupported()) { if (window) gdk_window_set_composited(window, true); } else #endif // wxGTK_HAS_COMPOSITING_SUPPORT { // We revert to erase mode if transparency is not supported m_backgroundStyle = wxBG_STYLE_ERASE; } } #ifndef __WXGTK3__ if (window && ( m_backgroundStyle == wxBG_STYLE_PAINT || m_backgroundStyle == wxBG_STYLE_TRANSPARENT)) { gdk_window_set_back_pixmap(window, NULL, false); } #endif const bool isTopLevel = IsTopLevel(); #if GTK_CHECK_VERSION(3,8,0) if (isTopLevel && gtk_check_version(3,8,0) == NULL) { GdkFrameClock* clock = gtk_widget_get_frame_clock(m_widget); if (clock && !g_signal_handler_find(clock, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, this)) { g_signal_connect(clock, "layout", G_CALLBACK(frame_clock_layout), this); } } #endif wxWindowCreateEvent event(static_cast(this)); event.SetEventObject( this ); GTKProcessEvent( event ); GTKUpdateCursor(false, true); } void wxWindowGTK::GTKHandleUnrealize() { if (m_wxwindow) { if (m_imContext) gtk_im_context_set_client_window(m_imContext, NULL); } } // ---------------------------------------------------------------------------- // this wxWindowBase function is implemented here (in platform-specific file) // because it is static and so couldn't be made virtual // ---------------------------------------------------------------------------- wxWindow *wxWindowBase::DoFindFocus() { #if wxUSE_MENUS // For compatibility with wxMSW, pretend that showing a popup menu doesn't // change the focus and that it remains on the window showing it, even // though the real focus does change in GTK. extern wxMenu *wxCurrentPopupMenu; if ( wxCurrentPopupMenu ) return wxCurrentPopupMenu->GetInvokingWindow(); #endif // wxUSE_MENUS wxWindowGTK *focus = gs_pendingFocus ? gs_pendingFocus : gs_currentFocus; // the cast is necessary when we compile in wxUniversal mode return static_cast(focus); } void wxWindowGTK::AddChildGTK(wxWindowGTK* child) { wxASSERT_MSG(m_wxwindow, "Cannot add a child to a window without a client area"); // the window might have been scrolled already, we // have to adapt the position wxPizza* pizza = WX_PIZZA(m_wxwindow); child->m_x += pizza->m_scroll_x; child->m_y += pizza->m_scroll_y; pizza->put(child->m_widget, child->m_x, child->m_y, child->m_width, child->m_height); } //----------------------------------------------------------------------------- // global functions //----------------------------------------------------------------------------- wxWindow *wxGetActiveWindow() { return wxWindow::FindFocus(); } // Under Unix this is implemented using X11 functions in utilsx11.cpp but we // need to have this function under Windows too, so provide at least a stub. #ifndef GDK_WINDOWING_X11 bool wxGetKeyState(wxKeyCode WXUNUSED(key)) { wxFAIL_MSG(wxS("Not implemented under Windows")); return false; } #endif // __WINDOWS__ static GdkDisplay* GetDisplay() { wxWindow* tlw = NULL; if (!wxTopLevelWindows.empty()) tlw = wxTopLevelWindows.front(); GdkDisplay* display; if (tlw && tlw->m_widget) display = gtk_widget_get_display(tlw->m_widget); else display = gdk_display_get_default(); return display; } wxMouseState wxGetMouseState() { wxMouseState ms; gint x; gint y; GdkModifierType mask; GdkDisplay* display = GetDisplay(); #ifdef __WXGTK3__ GdkDeviceManager* manager = gdk_display_get_device_manager(display); GdkDevice* device = gdk_device_manager_get_client_pointer(manager); GdkScreen* screen; gdk_device_get_position(device, &screen, &x, &y); GdkWindow* window = gdk_screen_get_root_window(screen); gdk_device_get_state(device, window, NULL, &mask); #else gdk_display_get_pointer(display, NULL, &x, &y, &mask); #endif ms.SetX(x); ms.SetY(y); ms.SetLeftDown((mask & GDK_BUTTON1_MASK) != 0); ms.SetMiddleDown((mask & GDK_BUTTON2_MASK) != 0); ms.SetRightDown((mask & GDK_BUTTON3_MASK) != 0); // see the comment in InitMouseEvent() ms.SetAux1Down((mask & GDK_BUTTON4_MASK) != 0); ms.SetAux2Down((mask & GDK_BUTTON5_MASK) != 0); ms.SetControlDown((mask & GDK_CONTROL_MASK) != 0); ms.SetShiftDown((mask & GDK_SHIFT_MASK) != 0); ms.SetAltDown((mask & GDK_MOD1_MASK) != 0); ms.SetMetaDown((mask & GDK_META_MASK) != 0); return ms; } //----------------------------------------------------------------------------- // wxWindowGTK //----------------------------------------------------------------------------- // in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu() // method #ifdef __WXUNIVERSAL__ IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK, wxWindowBase) #endif // __WXUNIVERSAL__ void wxWindowGTK::Init() { // GTK specific m_widget = NULL; m_wxwindow = NULL; m_focusWidget = NULL; // position/size m_x = 0; m_y = 0; m_width = 0; m_height = 0; m_showOnIdle = false; m_noExpose = false; m_nativeSizeEvent = false; #ifdef __WXGTK3__ m_paintContext = NULL; m_styleProvider = NULL; #endif m_isScrolling = false; m_mouseButtonDown = false; // initialize scrolling stuff for ( int dir = 0; dir < ScrollDir_Max; dir++ ) { m_scrollBar[dir] = NULL; m_scrollPos[dir] = 0; } m_clientWidth = m_clientHeight = 0; m_useCachedClientSize = false; m_clipPaintRegion = false; m_imContext = NULL; m_imKeyEvent = NULL; m_dirtyTabOrder = false; } wxWindowGTK::wxWindowGTK() { Init(); } wxWindowGTK::wxWindowGTK( wxWindow *parent, wxWindowID id, const wxPoint &pos, const wxSize &size, long style, const wxString &name ) { Init(); Create( parent, id, pos, size, style, name ); } void wxWindowGTK::GTKCreateScrolledWindowWith(GtkWidget* view) { wxASSERT_MSG( HasFlag(wxHSCROLL) || HasFlag(wxVSCROLL), wxS("Must not be called if scrolling is not needed.") ); m_widget = gtk_scrolled_window_new( NULL, NULL ); GtkScrolledWindow *scrolledWindow = GTK_SCROLLED_WINDOW(m_widget); // There is a conflict with default bindings at GTK+ // level between scrolled windows and notebooks both of which want to use // Ctrl-PageUp/Down: scrolled windows for scrolling in the horizontal // direction and notebooks for changing pages -- we decide that if we don't // have wxHSCROLL style we can safely sacrifice horizontal scrolling if it // means we can get working keyboard navigation in notebooks if ( !HasFlag(wxHSCROLL) ) { GtkBindingSet * bindings = gtk_binding_set_by_class(G_OBJECT_GET_CLASS(m_widget)); if ( bindings ) { gtk_binding_entry_remove(bindings, GDK_Page_Up, GDK_CONTROL_MASK); gtk_binding_entry_remove(bindings, GDK_Page_Down, GDK_CONTROL_MASK); } } // If wx[HV]SCROLL is not given, the corresponding scrollbar is not shown // at all. Otherwise it may be shown only on demand (default) or always, if // the wxALWAYS_SHOW_SB is specified. GtkPolicyType horzPolicy = HasFlag(wxHSCROLL) ? HasFlag(wxALWAYS_SHOW_SB) ? GTK_POLICY_ALWAYS : GTK_POLICY_AUTOMATIC : GTK_POLICY_NEVER; GtkPolicyType vertPolicy = HasFlag(wxVSCROLL) ? HasFlag(wxALWAYS_SHOW_SB) ? GTK_POLICY_ALWAYS : GTK_POLICY_AUTOMATIC : GTK_POLICY_NEVER; gtk_scrolled_window_set_policy( scrolledWindow, horzPolicy, vertPolicy ); m_scrollBar[ScrollDir_Horz] = GTK_RANGE(gtk_scrolled_window_get_hscrollbar(scrolledWindow)); m_scrollBar[ScrollDir_Vert] = GTK_RANGE(gtk_scrolled_window_get_vscrollbar(scrolledWindow)); if (GetLayoutDirection() == wxLayout_RightToLeft) gtk_range_set_inverted( m_scrollBar[ScrollDir_Horz], TRUE ); gtk_container_add( GTK_CONTAINER(m_widget), view ); // connect various scroll-related events for ( int dir = 0; dir < ScrollDir_Max; dir++ ) { // these handlers block mouse events to any window during scrolling // such as motion events and prevent GTK and wxWidgets from fighting // over where the slider should be g_signal_connect(m_scrollBar[dir], "button_press_event", G_CALLBACK(gtk_scrollbar_button_press_event), this); g_signal_connect(m_scrollBar[dir], "button_release_event", G_CALLBACK(gtk_scrollbar_button_release_event), this); gulong handler_id = g_signal_connect(m_scrollBar[dir], "event_after", G_CALLBACK(gtk_scrollbar_event_after), this); g_signal_handler_block(m_scrollBar[dir], handler_id); // these handlers get notified when scrollbar slider moves g_signal_connect_after(m_scrollBar[dir], "value_changed", G_CALLBACK(gtk_scrollbar_value_changed), this); } gtk_widget_show( view ); } bool wxWindowGTK::Create( wxWindow *parent, wxWindowID id, const wxPoint &pos, const wxSize &size, long style, const wxString &name ) { // Get default border wxBorder border = GetBorder(style); style &= ~wxBORDER_MASK; style |= border; if (!PreCreation( parent, pos, size ) || !CreateBase( parent, id, pos, size, style, wxDefaultValidator, name )) { wxFAIL_MSG( wxT("wxWindowGTK creation failed") ); return false; } // We should accept the native look #if 0 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) ); scroll_class->scrollbar_spacing = 0; #endif m_wxwindow = wxPizza::New(m_windowStyle); #ifndef __WXUNIVERSAL__ if (HasFlag(wxPizza::BORDER_STYLES)) { g_signal_connect(m_wxwindow, "parent_set", G_CALLBACK(parent_set), this); } #endif if (!HasFlag(wxHSCROLL) && !HasFlag(wxVSCROLL)) m_widget = m_wxwindow; else GTKCreateScrolledWindowWith(m_wxwindow); g_object_ref(m_widget); if (m_parent) m_parent->DoAddChild( this ); m_focusWidget = m_wxwindow; SetCanFocus(AcceptsFocus()); PostCreation(); return true; } void wxWindowGTK::GTKDisconnect(void* instance) { g_signal_handlers_disconnect_matched(instance, GSignalMatchType(G_SIGNAL_MATCH_DATA), 0, 0, NULL, NULL, this); } wxWindowGTK::~wxWindowGTK() { SendDestroyEvent(); if (gs_currentFocus == this) gs_currentFocus = NULL; if (gs_pendingFocus == this) gs_pendingFocus = NULL; if ( gs_deferredFocusOut == this ) gs_deferredFocusOut = NULL; // Unlike the above cases, which can happen in normal circumstances, a // window shouldn't be destroyed while it still has capture, so even though // we still reset the global pointer to avoid leaving it dangling and // crashing afterwards, also complain about it. if ( g_captureWindow == this ) { wxFAIL_MSG( wxS("Destroying window with mouse capture") ); g_captureWindow = NULL; } if (m_wxwindow) { GTKDisconnect(m_wxwindow); GtkWidget* parent = gtk_widget_get_parent(m_wxwindow); if (parent) GTKDisconnect(parent); } if (m_widget && m_widget != m_wxwindow) GTKDisconnect(m_widget); // destroy children before destroying this window itself DestroyChildren(); // delete before the widgets to avoid a crash on solaris if ( m_imContext ) { g_object_unref(m_imContext); m_imContext = NULL; } #ifdef __WXGTK3__ if (m_styleProvider) g_object_unref(m_styleProvider); gs_sizeRevalidateList = g_list_remove_all(gs_sizeRevalidateList, this); wx_sizeEventList = g_list_remove(wx_sizeEventList, this); #endif gs_needCursorResetMap.erase(this); if (m_widget) { // Note that gtk_widget_destroy() does not destroy the widget, it just // emits the "destroy" signal. The widget is not actually destroyed // until its reference count drops to zero. gtk_widget_destroy(m_widget); // Release our reference, should be the last one g_object_unref(m_widget); m_widget = NULL; } m_wxwindow = NULL; } bool wxWindowGTK::PreCreation( wxWindowGTK *parent, const wxPoint &pos, const wxSize &size ) { if ( GTKNeedsParent() ) { wxCHECK_MSG( parent, false, wxT("Must have non-NULL parent") ); } // Use either the given size, or the default if -1 is given. // See wxWindowBase for these functions. m_width = WidthDefault(size.x) ; m_height = HeightDefault(size.y); if (pos != wxDefaultPosition) { m_x = pos.x; m_y = pos.y; } return true; } void wxWindowGTK::PostCreation() { wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") ); GTKConnectFreezeWidget(m_widget); if (m_wxwindow && m_wxwindow != m_widget) GTKConnectFreezeWidget(m_wxwindow); #if wxGTK_HAS_COMPOSITING_SUPPORT // Set RGBA visual as soon as possible to minimize the possibility that // somebody uses the wrong one. if ( m_backgroundStyle == wxBG_STYLE_TRANSPARENT && IsTransparentBackgroundSupported() ) { GdkScreen *screen = gtk_widget_get_screen (m_widget); #ifdef __WXGTK3__ gtk_widget_set_visual(m_widget, gdk_screen_get_rgba_visual(screen)); #else GdkColormap *rgba_colormap = gdk_screen_get_rgba_colormap (screen); if (rgba_colormap) gtk_widget_set_colormap(m_widget, rgba_colormap); #endif } #endif // wxGTK_HAS_COMPOSITING_SUPPORT if (m_wxwindow) { if (!m_noExpose) { // these get reported to wxWidgets -> wxPaintEvent #ifdef __WXGTK3__ g_signal_connect(m_wxwindow, "draw", G_CALLBACK(draw), this); #else g_signal_connect(m_wxwindow, "expose_event", G_CALLBACK(expose_event), this); #endif if (GetLayoutDirection() == wxLayout_LeftToRight) gtk_widget_set_redraw_on_allocate(m_wxwindow, HasFlag(wxFULL_REPAINT_ON_RESIZE)); } } // focus handling if (!GTK_IS_WINDOW(m_widget)) { if (m_focusWidget == NULL) m_focusWidget = m_widget; if (m_wxwindow) { g_signal_connect (m_focusWidget, "focus_in_event", G_CALLBACK (gtk_window_focus_in_callback), this); g_signal_connect (m_focusWidget, "focus_out_event", G_CALLBACK (gtk_window_focus_out_callback), this); } else { g_signal_connect_after (m_focusWidget, "focus_in_event", G_CALLBACK (gtk_window_focus_in_callback), this); g_signal_connect_after (m_focusWidget, "focus_out_event", G_CALLBACK (gtk_window_focus_out_callback), this); } } if ( !AcceptsFocusFromKeyboard() ) { SetCanFocus(false); g_signal_connect(m_widget, "focus", G_CALLBACK(wx_window_focus_callback), this); } // connect to the various key and mouse handlers GtkWidget *connect_widget = GetConnectWidget(); ConnectWidget( connect_widget ); // We cannot set colours, fonts and cursors before the widget has been // realized, so we do this directly after realization -- unless the widget // was in fact realized already. if ( gtk_widget_get_realized(connect_widget) ) { GTKHandleRealized(); } else { g_signal_connect (connect_widget, "realize", G_CALLBACK (gtk_window_realized_callback), this); } g_signal_connect(connect_widget, "unrealize", G_CALLBACK(unrealize), this); if (!IsTopLevel()) { g_signal_connect(m_wxwindow ? m_wxwindow : m_widget, "size_allocate", G_CALLBACK(size_allocate), this); } #ifdef __WXGTK3__ else { g_signal_connect(m_widget, "check-resize", G_CALLBACK(check_resize), this); g_signal_connect_after(m_widget, "check-resize", G_CALLBACK(check_resize_after), this); } #endif #if GTK_CHECK_VERSION(2, 8, 0) #ifndef __WXGTK3__ if ( gtk_check_version(2,8,0) == NULL ) #endif { // Make sure we can notify the app when mouse capture is lost if ( m_wxwindow ) { g_signal_connect (m_wxwindow, "grab_broken_event", G_CALLBACK (gtk_window_grab_broken), this); } if ( connect_widget != m_wxwindow ) { g_signal_connect (connect_widget, "grab_broken_event", G_CALLBACK (gtk_window_grab_broken), this); } } #endif // GTK+ >= 2.8 if (!WX_IS_PIZZA(gtk_widget_get_parent(m_widget)) && !GTK_IS_WINDOW(m_widget)) gtk_widget_set_size_request(m_widget, m_width, m_height); // apply any font or color changes made before creation GTKApplyWidgetStyle(); InheritAttributes(); SetLayoutDirection(wxLayout_Default); // unless the window was created initially hidden (i.e. Hide() had been // called before Create()), we should show it at GTK+ level as well if (m_isShown) gtk_widget_show( m_widget ); } unsigned long wxWindowGTK::GTKConnectWidget(const char *signal, wxGTKCallback callback) { return g_signal_connect(m_widget, signal, callback, this); } // GSource callback functions for source used to detect new GDK events extern "C" { static gboolean source_prepare(GSource*, int*) { return !gs_isNewEvent; } static gboolean source_check(GSource*) { // 'check' will only be called if 'prepare' returned false return false; } static gboolean source_dispatch(GSource*, GSourceFunc, void*) { gs_isNewEvent = true; // don't remove this source return true; } } void wxWindowGTK::ConnectWidget( GtkWidget *widget ) { static bool isSourceAttached; if (!isSourceAttached) { // attach GSource to detect new GDK events isSourceAttached = true; static GSourceFuncs funcs = { source_prepare, source_check, source_dispatch, NULL, NULL, NULL }; GSource* source = g_source_new(&funcs, sizeof(GSource)); // priority slightly higher than GDK_PRIORITY_EVENTS g_source_set_priority(source, GDK_PRIORITY_EVENTS - 1); g_source_attach(source, NULL); } g_signal_connect (widget, "key_press_event", G_CALLBACK (gtk_window_key_press_callback), this); g_signal_connect (widget, "key_release_event", G_CALLBACK (gtk_window_key_release_callback), this); g_signal_connect (widget, "button_press_event", G_CALLBACK (gtk_window_button_press_callback), this); g_signal_connect (widget, "button_release_event", G_CALLBACK (gtk_window_button_release_callback), this); g_signal_connect (widget, "motion_notify_event", G_CALLBACK (gtk_window_motion_notify_callback), this); g_signal_connect(widget, "scroll_event", G_CALLBACK(scroll_event), this); GtkRange* range = m_scrollBar[ScrollDir_Horz]; if (range) g_signal_connect(range, "scroll_event", G_CALLBACK(scroll_event), this); range = m_scrollBar[ScrollDir_Vert]; if (range) g_signal_connect(range, "scroll_event", G_CALLBACK(scroll_event), this); g_signal_connect (widget, "popup_menu", G_CALLBACK (wxgtk_window_popup_menu_callback), this); g_signal_connect (widget, "enter_notify_event", G_CALLBACK (gtk_window_enter_callback), this); g_signal_connect (widget, "leave_notify_event", G_CALLBACK (gtk_window_leave_callback), this); } void wxWindowGTK::DoMoveWindow(int x, int y, int width, int height) { GtkWidget* parent = gtk_widget_get_parent(m_widget); wxPizza* pizza = NULL; if (WX_IS_PIZZA(parent)) { pizza = WX_PIZZA(parent); pizza->move(m_widget, x, y, width, height); if ( #ifdef __WXGTK3__ !gs_inSizeAllocate && #endif gtk_widget_get_visible(m_widget)) { // in case only the position is changing gtk_widget_queue_resize(m_widget); } } gtk_widget_set_size_request(m_widget, width, height); #ifdef __WXGTK3__ // With GTK3, gtk_widget_queue_resize() is ignored while a size-allocate // is in progress. This situation is common in wxWidgets, since // size-allocate can generate wxSizeEvent and size event handlers often // call SetSize(), directly or indirectly. It should be fine to call // gtk_widget_size_allocate() immediately in this case. if (gs_inSizeAllocate && gtk_widget_get_visible(m_widget) && width > 0 && height > 0) { // obligatory size request before size allocate to avoid GTK3 warnings GtkRequisition req; gtk_widget_get_preferred_size(m_widget, &req, NULL); GtkAllocation alloc = { x, y, width, height }; if (pizza) { alloc.x -= pizza->m_scroll_x; alloc.y -= pizza->m_scroll_y; if (gtk_widget_get_direction(parent) == GTK_TEXT_DIR_RTL) { GtkBorder border; pizza->get_border(border); GtkAllocation parent_alloc; gtk_widget_get_allocation(parent, &parent_alloc); int w = parent_alloc.width - border.left - border.right; alloc.x = w - alloc.x - alloc.width; } } gtk_widget_size_allocate(m_widget, &alloc); } #endif // __WXGTK3__ } void wxWindowGTK::ConstrainSize() { #ifdef __WXGPE__ // GPE's window manager doesn't like size hints at all, esp. when the user // has to use the virtual keyboard, so don't constrain size there if (!IsTopLevel()) #endif { const wxSize minSize = GetMinSize(); const wxSize maxSize = GetMaxSize(); if (minSize.x > 0 && m_width < minSize.x) m_width = minSize.x; if (minSize.y > 0 && m_height < minSize.y) m_height = minSize.y; if (maxSize.x > 0 && m_width > maxSize.x) m_width = maxSize.x; if (maxSize.y > 0 && m_height > maxSize.y) m_height = maxSize.y; } } void wxWindowGTK::DoSetSize( int x, int y, int width, int height, int sizeFlags ) { wxCHECK_RET(m_widget, "invalid window"); int scrollX = 0, scrollY = 0; GtkWidget* parent = gtk_widget_get_parent(m_widget); if (WX_IS_PIZZA(parent)) { wxPizza* pizza = WX_PIZZA(parent); scrollX = pizza->m_scroll_x; scrollY = pizza->m_scroll_y; } if (x != -1 || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE)) x += scrollX; else x = m_x; if (y != -1 || (sizeFlags & wxSIZE_ALLOW_MINUS_ONE)) y += scrollY; else y = m_y; // calculate the best size if we should auto size the window if ( ((sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1) || ((sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1) ) { const wxSize sizeBest = GetBestSize(); if ( (sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1 ) width = sizeBest.x; if ( (sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1 ) height = sizeBest.y; } if (width == -1) width = m_width; if (height == -1) height = m_height; bool sizeChange = m_width != width || m_height != height; if (sizeChange) m_useCachedClientSize = false; #ifdef __WXGTK3__ if (GList* p = g_list_find(wx_sizeEventList, this)) { sizeChange = true; wx_sizeEventList = g_list_delete_link(wx_sizeEventList, p); } #endif if (sizeChange || m_x != x || m_y != y) { m_x = x; m_y = y; m_width = width; m_height = height; /* the default button has a border around it */ if (gtk_widget_get_can_default(m_widget)) { GtkBorder *default_border = NULL; gtk_widget_style_get( m_widget, "default_border", &default_border, NULL ); if (default_border) { x -= default_border->left; y -= default_border->top; width += default_border->left + default_border->right; height += default_border->top + default_border->bottom; gtk_border_free( default_border ); } } DoMoveWindow(x, y, width, height); } if ((sizeChange && !m_nativeSizeEvent) || (sizeFlags & wxSIZE_FORCE_EVENT)) { // update these variables to keep size_allocate handler // from sending another size event for this change DoGetClientSize(&m_clientWidth, &m_clientHeight); wxSizeEvent event( wxSize(m_width,m_height), GetId() ); event.SetEventObject( this ); HandleWindowEvent( event ); } } bool wxWindowGTK::GTKShowFromOnIdle() { if (m_isShown && m_showOnIdle && !gtk_widget_get_visible(m_widget)) { GtkAllocation alloc; alloc.x = m_x; alloc.y = m_y; alloc.width = m_width; alloc.height = m_height; gtk_widget_size_allocate( m_widget, &alloc ); gtk_widget_show( m_widget ); wxShowEvent eventShow(GetId(), true); eventShow.SetEventObject(this); HandleWindowEvent(eventShow); m_showOnIdle = false; return true; } return false; } void wxWindowGTK::OnInternalIdle() { if ( gs_deferredFocusOut ) GTKHandleDeferredFocusOut(); // Check if we have to show window now if (GTKShowFromOnIdle()) return; if ( m_dirtyTabOrder ) { m_dirtyTabOrder = false; RealizeTabOrder(); } wxWindowBase::OnInternalIdle(); } void wxWindowGTK::DoGetSize( int *width, int *height ) const { if (width) (*width) = m_width; if (height) (*height) = m_height; } void wxWindowGTK::DoSetClientSize( int width, int height ) { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); const wxSize size = GetSize(); const wxSize clientSize = GetClientSize(); SetSize(width + (size.x - clientSize.x), height + (size.y - clientSize.y)); } void wxWindowGTK::DoGetClientSize( int *width, int *height ) const { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); if (m_useCachedClientSize) { if (width) *width = m_clientWidth; if (height) *height = m_clientHeight; return; } int w = m_width; int h = m_height; if ( m_wxwindow ) { // if window is scrollable, account for scrollbars if ( GTK_IS_SCROLLED_WINDOW(m_widget) ) { GtkPolicyType policy[ScrollDir_Max]; gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(m_widget), &policy[ScrollDir_Horz], &policy[ScrollDir_Vert]); // get scrollbar spacing the same way the GTK-private function // _gtk_scrolled_window_get_scrollbar_spacing() does it int scrollbar_spacing = GTK_SCROLLED_WINDOW_GET_CLASS(m_widget)->scrollbar_spacing; if (scrollbar_spacing < 0) { gtk_widget_style_get( m_widget, "scrollbar-spacing", &scrollbar_spacing, NULL); } for ( int i = 0; i < ScrollDir_Max; i++ ) { // don't account for the scrollbars we don't have GtkRange * const range = m_scrollBar[i]; if ( !range ) continue; // nor for the ones we have but don't current show switch ( policy[i] ) { #if GTK_CHECK_VERSION(3,16,0) case GTK_POLICY_EXTERNAL: #endif case GTK_POLICY_NEVER: // never shown so doesn't take any place continue; case GTK_POLICY_ALWAYS: // no checks necessary break; case GTK_POLICY_AUTOMATIC: // may be shown or not, check GtkAdjustment *adj = gtk_range_get_adjustment(range); if (gtk_adjustment_get_upper(adj) <= gtk_adjustment_get_page_size(adj)) continue; } GtkRequisition req; #ifdef __WXGTK3__ GtkWidget* widget = GTK_WIDGET(range); if (i == ScrollDir_Horz) { if (height) { gtk_widget_get_preferred_height(widget, NULL, &req.height); h -= req.height + scrollbar_spacing; } } else { if (width) { gtk_widget_get_preferred_width(widget, NULL, &req.width); w -= req.width + scrollbar_spacing; } } #else // !__WXGTK3__ gtk_widget_size_request(GTK_WIDGET(range), &req); if (i == ScrollDir_Horz) h -= req.height + scrollbar_spacing; else w -= req.width + scrollbar_spacing; #endif // !__WXGTK3__ } } const wxSize sizeBorders = DoGetBorderSize(); w -= sizeBorders.x; h -= sizeBorders.y; if (w < 0) w = 0; if (h < 0) h = 0; } if (width) *width = w; if (height) *height = h; } wxSize wxWindowGTK::DoGetBorderSize() const { if ( !m_wxwindow ) return wxWindowBase::DoGetBorderSize(); GtkBorder border; WX_PIZZA(m_wxwindow)->get_border(border); return wxSize(border.left + border.right, border.top + border.bottom); } void wxWindowGTK::DoGetPosition( int *x, int *y ) const { int dx = 0; int dy = 0; GtkWidget* parent = NULL; if (m_widget) parent = gtk_widget_get_parent(m_widget); if (WX_IS_PIZZA(parent)) { wxPizza* pizza = WX_PIZZA(parent); dx = pizza->m_scroll_x; dy = pizza->m_scroll_y; } if (x) (*x) = m_x - dx; if (y) (*y) = m_y - dy; } void wxWindowGTK::DoClientToScreen( int *x, int *y ) const { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); GtkWidget* widget = m_widget; if (m_wxwindow) widget = m_wxwindow; GdkWindow* source = gtk_widget_get_window(widget); if ((!m_useCachedClientSize || source == NULL) && !IsTopLevel() && m_parent) { m_parent->DoClientToScreen(x, y); int xx, yy; DoGetPosition(&xx, &yy); if (m_wxwindow) { GtkBorder border; WX_PIZZA(m_wxwindow)->get_border(border); xx += border.left; yy += border.top; } if (y) *y += yy; if (x) { if (GetLayoutDirection() != wxLayout_RightToLeft) *x += xx; else { int w; // undo RTL conversion done by parent static_cast(m_parent)->DoGetClientSize(&w, NULL); *x = w - *x; DoGetClientSize(&w, NULL); *x += xx; *x = w - *x; } } return; } if (source == NULL) { wxLogDebug("ClientToScreen cannot work when toplevel window is not shown"); return; } int org_x = 0; int org_y = 0; gdk_window_get_origin( source, &org_x, &org_y ); if (!m_wxwindow) { if (!gtk_widget_get_has_window(m_widget)) { GtkAllocation a; gtk_widget_get_allocation(m_widget, &a); org_x += a.x; org_y += a.y; } } if (x) { if (GetLayoutDirection() == wxLayout_RightToLeft) *x = (GetClientSize().x - *x) + org_x; else *x += org_x; } if (y) *y += org_y; } void wxWindowGTK::DoScreenToClient( int *x, int *y ) const { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); GtkWidget* widget = m_widget; if (m_wxwindow) widget = m_wxwindow; GdkWindow* source = gtk_widget_get_window(widget); if ((!m_useCachedClientSize || source == NULL) && !IsTopLevel() && m_parent) { m_parent->DoScreenToClient(x, y); int xx, yy; DoGetPosition(&xx, &yy); if (m_wxwindow) { GtkBorder border; WX_PIZZA(m_wxwindow)->get_border(border); xx += border.left; yy += border.top; } if (y) *y -= yy; if (x) { if (GetLayoutDirection() != wxLayout_RightToLeft) *x -= xx; else { int w; // undo RTL conversion done by parent static_cast(m_parent)->DoGetClientSize(&w, NULL); *x = w - *x; DoGetClientSize(&w, NULL); *x -= xx; *x = w - *x; } } return; } if (source == NULL) { wxLogDebug("ScreenToClient cannot work when toplevel window is not shown"); return; } int org_x = 0; int org_y = 0; gdk_window_get_origin( source, &org_x, &org_y ); if (!m_wxwindow) { if (!gtk_widget_get_has_window(m_widget)) { GtkAllocation a; gtk_widget_get_allocation(m_widget, &a); org_x += a.x; org_y += a.y; } } if (x) { if (GetLayoutDirection() == wxLayout_RightToLeft) *x = (GetClientSize().x - *x) - org_x; else *x -= org_x; } if (y) *y -= org_y; } bool wxWindowGTK::Show( bool show ) { if ( !wxWindowBase::Show(show) ) { // nothing to do return false; } // notice that we may call Hide() before the window is created and this is // actually useful to create it hidden initially -- but we can't call // Show() before it is created if ( !m_widget ) { wxASSERT_MSG( !show, "can't show invalid window" ); return true; } if ( show ) { if ( m_showOnIdle ) { // defer until later return true; } gtk_widget_show(m_widget); } else // hide { gtk_widget_hide(m_widget); } wxShowEvent eventShow(GetId(), show); eventShow.SetEventObject(this); HandleWindowEvent(eventShow); return true; } bool wxWindowGTK::IsShown() const { // return false for non-selected wxNotebook pages return m_isShown && (m_widget == NULL || gtk_widget_get_child_visible(m_widget)); } void wxWindowGTK::DoEnable( bool enable ) { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); gtk_widget_set_sensitive( m_widget, enable ); if (m_wxwindow && (m_wxwindow != m_widget)) gtk_widget_set_sensitive( m_wxwindow, enable ); } int wxWindowGTK::GetCharHeight() const { wxCHECK_MSG( (m_widget != NULL), 12, wxT("invalid window") ); wxFont font = GetFont(); wxCHECK_MSG( font.IsOk(), 12, wxT("invalid font") ); PangoContext* context = gtk_widget_get_pango_context(m_widget); if (!context) return 0; PangoFontDescription *desc = font.GetNativeFontInfo()->description; PangoLayout *layout = pango_layout_new(context); pango_layout_set_font_description(layout, desc); pango_layout_set_text(layout, "H", 1); PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data; PangoRectangle rect; pango_layout_line_get_extents(line, NULL, &rect); g_object_unref (layout); return (int) PANGO_PIXELS(rect.height); } int wxWindowGTK::GetCharWidth() const { wxCHECK_MSG( (m_widget != NULL), 8, wxT("invalid window") ); wxFont font = GetFont(); wxCHECK_MSG( font.IsOk(), 8, wxT("invalid font") ); PangoContext* context = gtk_widget_get_pango_context(m_widget); if (!context) return 0; PangoFontDescription *desc = font.GetNativeFontInfo()->description; PangoLayout *layout = pango_layout_new(context); pango_layout_set_font_description(layout, desc); pango_layout_set_text(layout, "g", 1); PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data; PangoRectangle rect; pango_layout_line_get_extents(line, NULL, &rect); g_object_unref (layout); return (int) PANGO_PIXELS(rect.width); } void wxWindowGTK::DoGetTextExtent( const wxString& string, int *x, int *y, int *descent, int *externalLeading, const wxFont *theFont ) const { // ensure we work with a valid font wxFont fontToUse; if ( !theFont || !theFont->IsOk() ) fontToUse = GetFont(); else fontToUse = *theFont; wxCHECK_RET( fontToUse.IsOk(), wxT("invalid font") ); const wxWindow* win = static_cast(this); wxTextMeasure txm(win, &fontToUse); txm.GetTextExtent(string, x, y, descent, externalLeading); } void wxWindowGTK::GTKDisableFocusOutEvent() { g_signal_handlers_block_by_func( m_focusWidget, (gpointer) gtk_window_focus_out_callback, this); } void wxWindowGTK::GTKEnableFocusOutEvent() { g_signal_handlers_unblock_by_func( m_focusWidget, (gpointer) gtk_window_focus_out_callback, this); } bool wxWindowGTK::GTKHandleFocusIn() { // Disable default focus handling for custom windows since the default GTK+ // handler issues a repaint const bool retval = m_wxwindow ? true : false; // NB: if there's still unprocessed deferred focus-out event (see // GTKHandleFocusOut() for explanation), we need to process it first so // that the order of focus events -- focus-out first, then focus-in // elsewhere -- is preserved if ( gs_deferredFocusOut ) { if ( GTKNeedsToFilterSameWindowFocus() && gs_deferredFocusOut == this ) { // GTK+ focus changed from this wxWindow back to itself, so don't // emit any events at all wxLogTrace(TRACE_FOCUS, "filtered out spurious focus change within %s(%p, %s)", GetClassInfo()->GetClassName(), this, GetLabel()); gs_deferredFocusOut = NULL; return retval; } // otherwise we need to send focus-out first wxASSERT_MSG ( gs_deferredFocusOut != this, "GTKHandleFocusIn(GTKFocus_Normal) called even though focus changed back to itself - derived class should handle this" ); GTKHandleDeferredFocusOut(); } wxLogTrace(TRACE_FOCUS, "handling focus_in event for %s(%p, %s)", GetClassInfo()->GetClassName(), this, GetLabel()); if (m_imContext) gtk_im_context_focus_in(m_imContext); gs_currentFocus = this; gs_pendingFocus = NULL; #if wxUSE_CARET // caret needs to be informed about focus change wxCaret *caret = GetCaret(); if ( caret ) { caret->OnSetFocus(); } #endif // wxUSE_CARET // Notify the parent keeping track of focus for the kbd navigation // purposes that we got it. wxChildFocusEvent eventChildFocus(static_cast(this)); GTKProcessEvent(eventChildFocus); wxFocusEvent eventFocus(wxEVT_SET_FOCUS, GetId()); eventFocus.SetEventObject(this); GTKProcessEvent(eventFocus); return retval; } bool wxWindowGTK::GTKHandleFocusOut() { // Disable default focus handling for custom windows since the default GTK+ // handler issues a repaint const bool retval = m_wxwindow ? true : false; // NB: If a control is composed of several GtkWidgets and when focus // changes from one of them to another within the same wxWindow, we get // a focus-out event followed by focus-in for another GtkWidget owned // by the same wx control. We don't want to generate two spurious // wxEVT_SET_FOCUS events in this case, so we defer sending wx events // from GTKHandleFocusOut() until we know for sure it's not coming back // (i.e. in GTKHandleFocusIn() or at idle time). if ( GTKNeedsToFilterSameWindowFocus() ) { wxASSERT_MSG( gs_deferredFocusOut == NULL, "deferred focus out event already pending" ); wxLogTrace(TRACE_FOCUS, "deferring focus_out event for %s(%p, %s)", GetClassInfo()->GetClassName(), this, GetLabel()); gs_deferredFocusOut = this; return retval; } GTKHandleFocusOutNoDeferring(); return retval; } void wxWindowGTK::GTKHandleFocusOutNoDeferring() { wxLogTrace(TRACE_FOCUS, "handling focus_out event for %s(%p, %s)", GetClassInfo()->GetClassName(), this, GetLabel()); if (m_imContext) gtk_im_context_focus_out(m_imContext); if ( gs_currentFocus != this ) { // Something is terribly wrong, gs_currentFocus is out of sync with the // real focus. We will reset it to NULL anyway, because after this // focus-out event is handled, one of the following with happen: // // * either focus will go out of the app altogether, in which case // gs_currentFocus _should_ be NULL // // * or it goes to another control, in which case focus-in event will // follow immediately and it will set gs_currentFocus to the right // value wxLogDebug("window %s(%p, %s) lost focus even though it didn't have it", GetClassInfo()->GetClassName(), this, GetLabel()); } gs_currentFocus = NULL; #if wxUSE_CARET // caret needs to be informed about focus change wxCaret *caret = GetCaret(); if ( caret ) { caret->OnKillFocus(); } #endif // wxUSE_CARET wxFocusEvent event( wxEVT_KILL_FOCUS, GetId() ); event.SetEventObject( this ); event.SetWindow( FindFocus() ); GTKProcessEvent( event ); } /*static*/ void wxWindowGTK::GTKHandleDeferredFocusOut() { // NB: See GTKHandleFocusOut() for explanation. This function is called // from either GTKHandleFocusIn() or OnInternalIdle() to process // deferred event. if ( gs_deferredFocusOut ) { wxWindowGTK *win = gs_deferredFocusOut; gs_deferredFocusOut = NULL; wxLogTrace(TRACE_FOCUS, "processing deferred focus_out event for %s(%p, %s)", win->GetClassInfo()->GetClassName(), win, win->GetLabel()); win->GTKHandleFocusOutNoDeferring(); } } void wxWindowGTK::SetFocus() { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); // Setting "physical" focus is not immediate in GTK+ and while // gtk_widget_is_focus ("determines if the widget is the focus widget // within its toplevel", i.e. returns true for one widget per TLW, not // globally) returns true immediately after grabbing focus, // GTK_WIDGET_HAS_FOCUS (which returns true only for the one widget that // has focus at the moment) takes effect only after the window is shown // (if it was hidden at the moment of the call) or at the next event loop // iteration. // // Because we want to FindFocus() call immediately following // foo->SetFocus() to return foo, we have to keep track of "pending" focus // ourselves. gs_pendingFocus = this; GtkWidget *widget = m_wxwindow ? m_wxwindow : m_focusWidget; if ( GTK_IS_CONTAINER(widget) && !gtk_widget_get_can_focus(widget) ) { wxLogTrace(TRACE_FOCUS, wxT("Setting focus to a child of %s(%p, %s)"), GetClassInfo()->GetClassName(), this, GetLabel().c_str()); gtk_widget_child_focus(widget, GTK_DIR_TAB_FORWARD); } else { wxLogTrace(TRACE_FOCUS, wxT("Setting focus to %s(%p, %s)"), GetClassInfo()->GetClassName(), this, GetLabel().c_str()); gtk_widget_grab_focus(widget); } } void wxWindowGTK::SetCanFocus(bool canFocus) { wxCHECK_RET(m_widget, "invalid window"); gtk_widget_set_can_focus(m_widget, canFocus); if ( m_wxwindow && (m_widget != m_wxwindow) ) { gtk_widget_set_can_focus(m_wxwindow, canFocus); } } bool wxWindowGTK::Reparent( wxWindowBase *newParentBase ) { wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") ); wxWindowGTK * const newParent = (wxWindowGTK *)newParentBase; wxASSERT( GTK_IS_WIDGET(m_widget) ); if ( !wxWindowBase::Reparent(newParent) ) return false; wxASSERT( GTK_IS_WIDGET(m_widget) ); // Notice that old m_parent pointer might be non-NULL here but the widget // still not have any parent at GTK level if it's a notebook page that had // been removed from the notebook so test this at GTK level and not wx one. if ( GtkWidget *parentGTK = gtk_widget_get_parent(m_widget) ) gtk_container_remove(GTK_CONTAINER(parentGTK), m_widget); wxASSERT( GTK_IS_WIDGET(m_widget) ); if (newParent) { if (gtk_widget_get_visible (newParent->m_widget)) { m_showOnIdle = true; gtk_widget_hide( m_widget ); } /* insert GTK representation */ newParent->AddChildGTK(this); } SetLayoutDirection(wxLayout_Default); return true; } void wxWindowGTK::DoAddChild(wxWindowGTK *child) { wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") ); wxASSERT_MSG( (child != NULL), wxT("invalid child window") ); /* add to list */ AddChild( child ); /* insert GTK representation */ AddChildGTK(child); } void wxWindowGTK::AddChild(wxWindowBase *child) { wxWindowBase::AddChild(child); m_dirtyTabOrder = true; wxTheApp->WakeUpIdle(); } void wxWindowGTK::RemoveChild(wxWindowBase *child) { wxWindowBase::RemoveChild(child); m_dirtyTabOrder = true; wxTheApp->WakeUpIdle(); } /* static */ wxLayoutDirection wxWindowGTK::GTKGetLayout(GtkWidget *widget) { return gtk_widget_get_direction(widget) == GTK_TEXT_DIR_RTL ? wxLayout_RightToLeft : wxLayout_LeftToRight; } /* static */ void wxWindowGTK::GTKSetLayout(GtkWidget *widget, wxLayoutDirection dir) { wxASSERT_MSG( dir != wxLayout_Default, wxT("invalid layout direction") ); gtk_widget_set_direction(widget, dir == wxLayout_RightToLeft ? GTK_TEXT_DIR_RTL : GTK_TEXT_DIR_LTR); } wxLayoutDirection wxWindowGTK::GetLayoutDirection() const { return GTKGetLayout(m_widget); } void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir) { if ( dir == wxLayout_Default ) { const wxWindow *const parent = GetParent(); if ( parent ) { // inherit layout from parent. dir = parent->GetLayoutDirection(); } else // no parent, use global default layout { dir = wxTheApp->GetLayoutDirection(); } } if ( dir == wxLayout_Default ) return; GTKSetLayout(m_widget, dir); if (m_wxwindow && (m_wxwindow != m_widget)) GTKSetLayout(m_wxwindow, dir); } wxCoord wxWindowGTK::AdjustForLayoutDirection(wxCoord x, wxCoord WXUNUSED(width), wxCoord WXUNUSED(widthTotal)) const { // We now mirror the coordinates of RTL windows in wxPizza return x; } void wxWindowGTK::DoMoveInTabOrder(wxWindow *win, WindowOrder move) { wxWindowBase::DoMoveInTabOrder(win, move); // Update the TAB order at GTK+ level too, but do it slightly later in case // we're changing the TAB order of several controls at once, as is common. wxWindow* const parent = GetParent(); if ( parent ) { parent->m_dirtyTabOrder = true; wxTheApp->WakeUpIdle(); } } bool wxWindowGTK::DoNavigateIn(int flags) { wxWindow *parent = wxGetTopLevelParent((wxWindow *)this); wxCHECK_MSG( parent, false, wxT("every window must have a TLW parent") ); GtkDirectionType dir; dir = flags & wxNavigationKeyEvent::IsForward ? GTK_DIR_TAB_FORWARD : GTK_DIR_TAB_BACKWARD; gboolean rc; g_signal_emit_by_name(parent->m_widget, "focus", dir, &rc); return rc != 0; } bool wxWindowGTK::GTKWidgetNeedsMnemonic() const { // none needed by default return false; } void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget* WXUNUSED(w)) { // nothing to do by default since none is needed } void wxWindowGTK::RealizeTabOrder() { if (m_wxwindow) { if ( !m_children.empty() ) { // we don't only construct the correct focus chain but also use // this opportunity to update the mnemonic widgets for the widgets // that need them GList *chain = NULL; wxWindowGTK* mnemonicWindow = NULL; for ( wxWindowList::const_iterator i = m_children.begin(); i != m_children.end(); ++i ) { wxWindowGTK *win = *i; bool focusableFromKeyboard = win->AcceptsFocusFromKeyboard(); if ( mnemonicWindow ) { if ( focusableFromKeyboard ) { // wxComboBox et al. needs to focus on on a different // widget than m_widget, so if the main widget isn't // focusable try the connect widget GtkWidget* w = win->m_widget; if ( !gtk_widget_get_can_focus(w) ) { w = win->GetConnectWidget(); if ( !gtk_widget_get_can_focus(w) ) w = NULL; } if ( w ) { mnemonicWindow->GTKWidgetDoSetMnemonic(w); mnemonicWindow = NULL; } } } else if ( win->GTKWidgetNeedsMnemonic() ) { mnemonicWindow = win; } if ( focusableFromKeyboard ) chain = g_list_prepend(chain, win->m_widget); } chain = g_list_reverse(chain); gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow), chain); g_list_free(chain); } else // no children { gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow)); } } } void wxWindowGTK::Raise() { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); if (m_wxwindow && gtk_widget_get_window(m_wxwindow)) { gdk_window_raise(gtk_widget_get_window(m_wxwindow)); } else if (gtk_widget_get_window(m_widget)) { gdk_window_raise(gtk_widget_get_window(m_widget)); } } void wxWindowGTK::Lower() { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); if (m_wxwindow && gtk_widget_get_window(m_wxwindow)) { gdk_window_lower(gtk_widget_get_window(m_wxwindow)); } else if (gtk_widget_get_window(m_widget)) { gdk_window_lower(gtk_widget_get_window(m_widget)); } } bool wxWindowGTK::SetCursor( const wxCursor &cursor ) { if (!wxWindowBase::SetCursor(cursor)) return false; GTKUpdateCursor(); return true; } void wxWindowGTK::GTKUpdateCursor(bool isBusyOrGlobalCursor, bool isRealize) { gs_needCursorResetMap[this] = false; if (m_widget == NULL || !gtk_widget_get_realized(m_widget)) return; // if we don't already know there is a busy/global cursor, we have to check for one if (!isBusyOrGlobalCursor) { if (g_globalCursor.IsOk()) isBusyOrGlobalCursor = true; else if (wxIsBusy()) { wxWindow* win = wxGetTopLevelParent(static_cast(this)); if (win && win->m_widget && !gtk_window_get_modal(GTK_WINDOW(win->m_widget))) isBusyOrGlobalCursor = true; } } GdkCursor* cursor = NULL; if (!isBusyOrGlobalCursor) { const wxCursor* overrideCursor = gs_overrideCursor; gs_overrideCursor = NULL; cursor = (overrideCursor ? *overrideCursor : m_cursor).GetCursor(); } GdkWindow* window = NULL; if (cursor || isBusyOrGlobalCursor || !isRealize) { wxArrayGdkWindows windows; window = GTKGetWindow(windows); if (window) gdk_window_set_cursor(window, cursor); else { for (size_t i = windows.size(); i--;) { window = windows[i]; if (window) gdk_window_set_cursor(window, cursor); } } } if (window && cursor == NULL && m_wxwindow == NULL && !isBusyOrGlobalCursor && !isRealize) { void* data; gdk_window_get_user_data(window, &data); if (data) { #ifdef __WXGTK3__ const char sig_name[] = "state-flags-changed"; GtkStateFlags state = gtk_widget_get_state_flags(GTK_WIDGET(data)); #else const char sig_name[] = "state-changed"; GtkStateType state = gtk_widget_get_state(GTK_WIDGET(data)); #endif static unsigned sig_id = g_signal_lookup(sig_name, GTK_TYPE_WIDGET); // encourage native widget to restore any non-default cursors g_signal_emit(data, sig_id, 0, state); } } } void wxWindowGTK::WarpPointer( int x, int y ) { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); ClientToScreen(&x, &y); GdkDisplay* display = gtk_widget_get_display(m_widget); GdkScreen* screen = gtk_widget_get_screen(m_widget); #ifdef __WXGTK3__ GdkDeviceManager* manager = gdk_display_get_device_manager(display); gdk_device_warp(gdk_device_manager_get_client_pointer(manager), screen, x, y); #else #ifdef GDK_WINDOWING_X11 XWarpPointer(GDK_DISPLAY_XDISPLAY(display), None, GDK_WINDOW_XID(gdk_screen_get_root_window(screen)), 0, 0, 0, 0, x, y); #endif #endif } wxWindowGTK::ScrollDir wxWindowGTK::ScrollDirFromRange(GtkRange *range) const { // find the scrollbar which generated the event for ( int dir = 0; dir < ScrollDir_Max; dir++ ) { if ( range == m_scrollBar[dir] ) return (ScrollDir)dir; } wxFAIL_MSG( wxT("event from unknown scrollbar received") ); return ScrollDir_Max; } bool wxWindowGTK::DoScrollByUnits(ScrollDir dir, ScrollUnit unit, int units) { bool changed = false; GtkRange* range = m_scrollBar[dir]; if ( range && units ) { GtkAdjustment* adj = gtk_range_get_adjustment(range); double inc = unit == ScrollUnit_Line ? gtk_adjustment_get_step_increment(adj) : gtk_adjustment_get_page_increment(adj); const int posOld = wxRound(gtk_adjustment_get_value(adj)); gtk_range_set_value(range, posOld + units*inc); changed = wxRound(gtk_adjustment_get_value(adj)) != posOld; } return changed; } bool wxWindowGTK::ScrollLines(int lines) { return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Line, lines); } bool wxWindowGTK::ScrollPages(int pages) { return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Page, pages); } void wxWindowGTK::Refresh(bool WXUNUSED(eraseBackground), const wxRect *rect) { if (m_wxwindow) { if (gtk_widget_get_mapped(m_wxwindow)) { GdkWindow* window = gtk_widget_get_window(m_wxwindow); if (rect) { GdkRectangle r = { rect->x, rect->y, rect->width, rect->height }; if (GetLayoutDirection() == wxLayout_RightToLeft) r.x = gdk_window_get_width(window) - r.x - rect->width; gdk_window_invalidate_rect(window, &r, true); } else gdk_window_invalidate_rect(window, NULL, true); } } else if (m_widget) { if (gtk_widget_get_mapped(m_widget)) { if (rect) gtk_widget_queue_draw_area(m_widget, rect->x, rect->y, rect->width, rect->height); else gtk_widget_queue_draw(m_widget); } } } void wxWindowGTK::Update() { if (m_widget && gtk_widget_get_mapped(m_widget) && m_width > 0 && m_height > 0) { GdkDisplay* display = gtk_widget_get_display(m_widget); // Flush everything out to the server, and wait for it to finish. // This ensures nothing will overwrite the drawing we are about to do. gdk_display_sync(display); GdkWindow* window = GTKGetDrawingWindow(); if (window == NULL) window = gtk_widget_get_window(m_widget); gdk_window_process_updates(window, true); // Flush again, but no need to wait for it to finish gdk_display_flush(display); } } bool wxWindowGTK::DoIsExposed( int x, int y ) const { return m_updateRegion.Contains(x, y) != wxOutRegion; } bool wxWindowGTK::DoIsExposed( int x, int y, int w, int h ) const { if (GetLayoutDirection() == wxLayout_RightToLeft) return m_updateRegion.Contains(x-w, y, w, h) != wxOutRegion; else return m_updateRegion.Contains(x, y, w, h) != wxOutRegion; } #ifdef __WXGTK3__ void wxWindowGTK::GTKSendPaintEvents(cairo_t* cr) #else void wxWindowGTK::GTKSendPaintEvents(const GdkRegion* region) #endif { #ifdef __WXGTK3__ { cairo_region_t* region = gdk_window_get_clip_region(gtk_widget_get_window(m_wxwindow)); cairo_rectangle_int_t rect; cairo_region_get_extents(region, &rect); cairo_region_destroy(region); cairo_rectangle(cr, rect.x, rect.y, rect.width, rect.height); cairo_clip(cr); } double x1, y1, x2, y2; cairo_clip_extents(cr, &x1, &y1, &x2, &y2); if (x1 >= x2 || y1 >= y2) return; m_paintContext = cr; m_updateRegion = wxRegion(int(x1), int(y1), int(x2 - x1), int(y2 - y1)); #else // !__WXGTK3__ m_updateRegion = wxRegion(region); #if wxGTK_HAS_COMPOSITING_SUPPORT cairo_t* cr = NULL; #endif #endif // !__WXGTK3__ // Clip to paint region in wxClientDC m_clipPaintRegion = true; m_nativeUpdateRegion = m_updateRegion; if (GetLayoutDirection() == wxLayout_RightToLeft) { // Transform m_updateRegion under RTL m_updateRegion.Clear(); const int width = gdk_window_get_width(GTKGetDrawingWindow()); wxRegionIterator upd( m_nativeUpdateRegion ); while (upd) { wxRect rect; rect.x = upd.GetX(); rect.y = upd.GetY(); rect.width = upd.GetWidth(); rect.height = upd.GetHeight(); rect.x = width - rect.x - rect.width; m_updateRegion.Union( rect ); ++upd; } } switch ( GetBackgroundStyle() ) { case wxBG_STYLE_TRANSPARENT: #if wxGTK_HAS_COMPOSITING_SUPPORT if (IsTransparentBackgroundSupported()) { // Set a transparent background, so that overlaying in parent // might indeed let see through where this child did not // explicitly paint. // NB: it works also for top level windows (but this is the // windows manager which then does the compositing job) #ifndef __WXGTK3__ cr = gdk_cairo_create(m_wxwindow->window); gdk_cairo_region(cr, m_nativeUpdateRegion.GetRegion()); cairo_clip(cr); #endif cairo_set_operator(cr, CAIRO_OPERATOR_CLEAR); cairo_paint(cr); cairo_set_operator(cr, CAIRO_OPERATOR_OVER); #ifndef __WXGTK3__ cairo_surface_flush(cairo_get_target(cr)); #endif } #endif // wxGTK_HAS_COMPOSITING_SUPPORT break; case wxBG_STYLE_ERASE: { #ifdef __WXGTK3__ wxGTKCairoDC dc(cr); #else wxWindowDC dc( (wxWindow*)this ); dc.SetDeviceClippingRegion( m_updateRegion ); // Work around gtk-qt <= 0.60 bug whereby the window colour // remains grey if ( UseBgCol() && wxSystemOptions:: GetOptionInt("gtk.window.force-background-colour") ) { dc.SetBackground(GetBackgroundColour()); dc.Clear(); } #endif // !__WXGTK3__ wxEraseEvent erase_event( GetId(), &dc ); erase_event.SetEventObject( this ); if ( HandleWindowEvent(erase_event) ) { // background erased, don't do it again break; } } // fall through case wxBG_STYLE_SYSTEM: if ( GetThemeEnabled() ) { GdkWindow* gdkWindow = GTKGetDrawingWindow(); const int w = gdk_window_get_width(gdkWindow); const int h = gdk_window_get_height(gdkWindow); #ifdef __WXGTK3__ GtkStyleContext* sc = gtk_widget_get_style_context(m_wxwindow); gtk_render_background(sc, cr, 0, 0, w, h); #else // find ancestor from which to steal background wxWindow *parent = wxGetTopLevelParent((wxWindow *)this); if (!parent) parent = (wxWindow*)this; GdkRectangle rect; m_nativeUpdateRegion.GetBox(rect.x, rect.y, rect.width, rect.height); gtk_paint_flat_box(gtk_widget_get_style(parent->m_widget), gdkWindow, gtk_widget_get_state(m_wxwindow), GTK_SHADOW_NONE, &rect, parent->m_widget, (char *)"base", 0, 0, w, h); #endif // !__WXGTK3__ } #ifdef __WXGTK3__ else if (m_backgroundColour.IsOk() && gtk_check_version(3,20,0) == NULL) { cairo_save(cr); gdk_cairo_set_source_rgba(cr, m_backgroundColour); cairo_paint(cr); cairo_restore(cr); } #endif break; case wxBG_STYLE_PAINT: // nothing to do: window will be painted over in EVT_PAINT break; default: wxFAIL_MSG( "unsupported background style" ); } wxNcPaintEvent nc_paint_event( GetId() ); nc_paint_event.SetEventObject( this ); HandleWindowEvent( nc_paint_event ); wxPaintEvent paint_event( GetId() ); paint_event.SetEventObject( this ); HandleWindowEvent( paint_event ); #if wxGTK_HAS_COMPOSITING_SUPPORT if (IsTransparentBackgroundSupported()) { // now composite children which need it // Overlay all our composite children on top of the painted area wxWindowList::compatibility_iterator node; for ( node = m_children.GetFirst(); node ; node = node->GetNext() ) { wxWindow *compositeChild = node->GetData(); if (compositeChild->GetBackgroundStyle() == wxBG_STYLE_TRANSPARENT) { #ifndef __WXGTK3__ if (cr == NULL) { cr = gdk_cairo_create(m_wxwindow->window); gdk_cairo_region(cr, m_nativeUpdateRegion.GetRegion()); cairo_clip(cr); } #endif // !__WXGTK3__ GtkWidget *child = compositeChild->m_wxwindow; GtkAllocation alloc; gtk_widget_get_allocation(child, &alloc); // The source data is the (composited) child gdk_cairo_set_source_window( cr, gtk_widget_get_window(child), alloc.x, alloc.y); cairo_paint(cr); } } #ifndef __WXGTK3__ if (cr) cairo_destroy(cr); #endif } #endif // wxGTK_HAS_COMPOSITING_SUPPORT m_clipPaintRegion = false; #ifdef __WXGTK3__ m_paintContext = NULL; #endif m_updateRegion.Clear(); m_nativeUpdateRegion.Clear(); } void wxWindowGTK::SetDoubleBuffered( bool on ) { wxCHECK_RET( (m_widget != NULL), wxT("invalid window") ); if ( m_wxwindow ) gtk_widget_set_double_buffered( m_wxwindow, on ); } bool wxWindowGTK::IsDoubleBuffered() const { return gtk_widget_get_double_buffered( m_wxwindow ) != 0; } void wxWindowGTK::ClearBackground() { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); } #if wxUSE_TOOLTIPS void wxWindowGTK::DoSetToolTip( wxToolTip *tip ) { if (m_tooltip != tip) { wxWindowBase::DoSetToolTip(tip); if (m_tooltip) m_tooltip->GTKSetWindow(static_cast(this)); else GTKApplyToolTip(NULL); } } void wxWindowGTK::GTKApplyToolTip(const char* tip) { wxToolTip::GTKApply(GetConnectWidget(), tip); } #endif // wxUSE_TOOLTIPS bool wxWindowGTK::SetBackgroundColour( const wxColour &colour ) { if (!wxWindowBase::SetBackgroundColour(colour)) return false; if (m_widget) { #ifndef __WXGTK3__ if (colour.IsOk()) { // We need the pixel value e.g. for background clearing. m_backgroundColour.CalcPixel(gtk_widget_get_colormap(m_widget)); } #endif // apply style change (forceStyle=true so that new style is applied // even if the bg colour changed from valid to wxNullColour) GTKApplyWidgetStyle(true); } return true; } bool wxWindowGTK::SetForegroundColour( const wxColour &colour ) { if (!wxWindowBase::SetForegroundColour(colour)) return false; if (m_widget) { #ifndef __WXGTK3__ if (colour.IsOk()) { // We need the pixel value e.g. for background clearing. m_foregroundColour.CalcPixel(gtk_widget_get_colormap(m_widget)); } #endif // apply style change (forceStyle=true so that new style is applied // even if the bg colour changed from valid to wxNullColour): GTKApplyWidgetStyle(true); } return true; } PangoContext *wxWindowGTK::GTKGetPangoDefaultContext() { return gtk_widget_get_pango_context( m_widget ); } #ifndef __WXGTK3__ GtkRcStyle* wxWindowGTK::GTKCreateWidgetStyle() { GtkRcStyle *style = gtk_rc_style_new(); if ( m_font.IsOk() ) { style->font_desc = pango_font_description_copy( m_font.GetNativeFontInfo()->description ); } int flagsNormal = 0, flagsPrelight = 0, flagsActive = 0, flagsInsensitive = 0; if ( m_foregroundColour.IsOk() ) { const GdkColor *fg = m_foregroundColour.GetColor(); style->fg[GTK_STATE_NORMAL] = style->text[GTK_STATE_NORMAL] = *fg; flagsNormal |= GTK_RC_FG | GTK_RC_TEXT; style->fg[GTK_STATE_PRELIGHT] = style->text[GTK_STATE_PRELIGHT] = *fg; flagsPrelight |= GTK_RC_FG | GTK_RC_TEXT; style->fg[GTK_STATE_ACTIVE] = style->text[GTK_STATE_ACTIVE] = *fg; flagsActive |= GTK_RC_FG | GTK_RC_TEXT; } if ( m_backgroundColour.IsOk() ) { const GdkColor *bg = m_backgroundColour.GetColor(); style->bg[GTK_STATE_NORMAL] = style->base[GTK_STATE_NORMAL] = *bg; flagsNormal |= GTK_RC_BG | GTK_RC_BASE; style->bg[GTK_STATE_PRELIGHT] = style->base[GTK_STATE_PRELIGHT] = *bg; flagsPrelight |= GTK_RC_BG | GTK_RC_BASE; style->bg[GTK_STATE_ACTIVE] = style->base[GTK_STATE_ACTIVE] = *bg; flagsActive |= GTK_RC_BG | GTK_RC_BASE; style->bg[GTK_STATE_INSENSITIVE] = style->base[GTK_STATE_INSENSITIVE] = *bg; flagsInsensitive |= GTK_RC_BG | GTK_RC_BASE; } style->color_flags[GTK_STATE_NORMAL] = (GtkRcFlags)flagsNormal; style->color_flags[GTK_STATE_PRELIGHT] = (GtkRcFlags)flagsPrelight; style->color_flags[GTK_STATE_ACTIVE] = (GtkRcFlags)flagsActive; style->color_flags[GTK_STATE_INSENSITIVE] = (GtkRcFlags)flagsInsensitive; return style; } #endif // !__WXGTK3__ void wxWindowGTK::GTKApplyWidgetStyle(bool forceStyle) { if (forceStyle || m_font.IsOk() || m_foregroundColour.IsOk() || m_backgroundColour.IsOk()) { #ifdef __WXGTK3__ if (m_backgroundColour.IsOk()) { // create a GtkStyleProvider to override "background-image" if (m_styleProvider == NULL) m_styleProvider = GTK_STYLE_PROVIDER(gtk_css_provider_new()); const char css[] = "*{background-image:-gtk-gradient(linear,0 0,0 1," "from(rgba(%u,%u,%u,%g)),to(rgba(%u,%u,%u,%g)))}"; char buf[sizeof(css) + 20]; const unsigned r = m_backgroundColour.Red(); const unsigned g = m_backgroundColour.Green(); const unsigned b = m_backgroundColour.Blue(); const double a = m_backgroundColour.Alpha() / 255.0; g_snprintf(buf, sizeof(buf), css, r, g, b, a, r, g, b, a); gtk_css_provider_load_from_data(GTK_CSS_PROVIDER(m_styleProvider), buf, -1, NULL); } DoApplyWidgetStyle(NULL); #else GtkRcStyle* style = GTKCreateWidgetStyle(); DoApplyWidgetStyle(style); g_object_unref(style); #endif } } void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle *style) { GtkWidget* widget = m_wxwindow ? m_wxwindow : m_widget; GTKApplyStyle(widget, style); } void wxWindowGTK::GTKApplyStyle(GtkWidget* widget, GtkRcStyle* WXUNUSED_IN_GTK3(style)) { #ifdef __WXGTK3__ const PangoFontDescription* pfd = NULL; if (m_font.IsOk()) pfd = m_font.GetNativeFontInfo()->description; gtk_widget_override_font(widget, pfd); gtk_widget_override_color(widget, GTK_STATE_FLAG_NORMAL, m_foregroundColour); gtk_widget_override_background_color(widget, GTK_STATE_FLAG_NORMAL, m_backgroundColour); // setting background color has no effect with some themes when the widget style // has a "background-image" property, so we need to override that as well GtkStyleContext* context = gtk_widget_get_style_context(widget); if (m_styleProvider) gtk_style_context_remove_provider(context, m_styleProvider); cairo_pattern_t* pattern = NULL; if (m_backgroundColour.IsOk()) { gtk_style_context_save(context); gtk_style_context_set_state(context, GTK_STATE_FLAG_NORMAL); gtk_style_context_get(context, GTK_STATE_FLAG_NORMAL, "background-image", &pattern, NULL); gtk_style_context_restore(context); } if (pattern) { cairo_pattern_destroy(pattern); gtk_style_context_add_provider(context, m_styleProvider, GTK_STYLE_PROVIDER_PRIORITY_APPLICATION); } #else gtk_widget_modify_style(widget, style); #endif } bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style) { if (!wxWindowBase::SetBackgroundStyle(style)) return false; #ifndef __WXGTK3__ GdkWindow *window; if ((style == wxBG_STYLE_PAINT || style == wxBG_STYLE_TRANSPARENT) && (window = GTKGetDrawingWindow())) { gdk_window_set_back_pixmap(window, NULL, false); } #endif // !__WXGTK3__ return true; } bool wxWindowGTK::IsTransparentBackgroundSupported(wxString* reason) const { #if wxGTK_HAS_COMPOSITING_SUPPORT #ifndef __WXGTK3__ if (gtk_check_version(wxGTK_VERSION_REQUIRED_FOR_COMPOSITING) != NULL) { if (reason) { *reason = _("GTK+ installed on this machine is too old to " "support screen compositing, please install " "GTK+ 2.12 or later."); } return false; } #endif // !__WXGTK3__ // NB: We don't check here if the particular kind of widget supports // transparency, we check only if it would be possible for a generic window wxCHECK_MSG ( m_widget, false, "Window must be created first" ); if (!gdk_screen_is_composited(gtk_widget_get_screen(m_widget))) { if (reason) { *reason = _("Compositing not supported by this system, " "please enable it in your Window Manager."); } return false; } return true; #else if (reason) { *reason = _("This program was compiled with a too old version of GTK+, " "please rebuild with GTK+ 2.12 or newer."); } return false; #endif // wxGTK_HAS_COMPOSITING_SUPPORT/!wxGTK_HAS_COMPOSITING_SUPPORT } #ifdef __WXGTK3__ GdkWindow* wxGTKFindWindow(GtkWidget* widget) { GdkWindow* window = gtk_widget_get_window(widget); for (const GList* p = gdk_window_peek_children(window); p; p = p->next) { window = GDK_WINDOW(p->data); void* data; gdk_window_get_user_data(window, &data); if (data == widget) return window; } return NULL; } void wxGTKFindWindow(GtkWidget* widget, wxArrayGdkWindows& windows) { GdkWindow* window = gtk_widget_get_window(widget); for (const GList* p = gdk_window_peek_children(window); p; p = p->next) { window = GDK_WINDOW(p->data); void* data; gdk_window_get_user_data(window, &data); if (data == widget) windows.push_back(window); } } #endif // __WXGTK3__ // ---------------------------------------------------------------------------- // Pop-up menu stuff // ---------------------------------------------------------------------------- #if wxUSE_MENUS_NATIVE extern "C" { static void wxPopupMenuPositionCallback( GtkMenu *menu, gint *x, gint *y, gboolean * WXUNUSED(whatever), gpointer user_data ) { // ensure that the menu appears entirely on screen GtkRequisition req; #ifdef __WXGTK3__ gtk_widget_get_preferred_size(GTK_WIDGET(menu), &req, NULL); #else gtk_widget_get_child_requisition(GTK_WIDGET(menu), &req); #endif wxSize sizeScreen = wxGetDisplaySize(); wxPoint *pos = (wxPoint*)user_data; gint xmax = sizeScreen.x - req.width, ymax = sizeScreen.y - req.height; *x = pos->x < xmax ? pos->x : xmax; *y = pos->y < ymax ? pos->y : ymax; } } bool wxWindowGTK::DoPopupMenu( wxMenu *menu, int x, int y ) { wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") ); menu->UpdateUI(); wxPoint pos; gpointer userdata; GtkMenuPositionFunc posfunc; if ( x == -1 && y == -1 ) { // use GTK's default positioning algorithm userdata = NULL; posfunc = NULL; } else { pos = ClientToScreen(wxPoint(x, y)); userdata = &pos; posfunc = wxPopupMenuPositionCallback; } menu->m_popupShown = true; gtk_menu_popup( GTK_MENU(menu->m_menu), NULL, // parent menu shell NULL, // parent menu item posfunc, // function to position it userdata, // client data 0, // button used to activate it gtk_get_current_event_time() ); // it is possible for gtk_menu_popup() to fail if (!gtk_widget_get_visible(GTK_WIDGET(menu->m_menu))) { menu->m_popupShown = false; return false; } while (menu->m_popupShown) { gtk_main_iteration(); } return true; } #endif // wxUSE_MENUS_NATIVE #if wxUSE_DRAG_AND_DROP void wxWindowGTK::SetDropTarget( wxDropTarget *dropTarget ) { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); GtkWidget *dnd_widget = GetConnectWidget(); if (m_dropTarget) m_dropTarget->GtkUnregisterWidget( dnd_widget ); if (m_dropTarget) delete m_dropTarget; m_dropTarget = dropTarget; if (m_dropTarget) m_dropTarget->GtkRegisterWidget( dnd_widget ); } #endif // wxUSE_DRAG_AND_DROP GtkWidget* wxWindowGTK::GetConnectWidget() { GtkWidget *connect_widget = m_widget; if (m_wxwindow) connect_widget = m_wxwindow; return connect_widget; } bool wxWindowGTK::GTKIsOwnWindow(GdkWindow *window) const { wxArrayGdkWindows windowsThis; GdkWindow * const winThis = GTKGetWindow(windowsThis); return winThis ? window == winThis : windowsThis.Index(window) != wxNOT_FOUND; } GdkWindow *wxWindowGTK::GTKGetWindow(wxArrayGdkWindows& WXUNUSED(windows)) const { return m_wxwindow ? GTKGetDrawingWindow() : gtk_widget_get_window(m_widget); } #ifdef __WXGTK3__ void wxGTKSizeRevalidate(wxWindow* tlw) { GList* next; for (GList* p = gs_sizeRevalidateList; p; p = next) { next = p->next; wxWindow* win = static_cast(p->data); if (wxGetTopLevelParent(win) == tlw) { win->InvalidateBestSize(); gs_sizeRevalidateList = g_list_delete_link(gs_sizeRevalidateList, p); for (;;) { win = win->GetParent(); if (win == NULL || g_list_find(wx_sizeEventList, win)) break; wx_sizeEventList = g_list_prepend(wx_sizeEventList, win); if (win->IsTopLevel()) break; } } } } extern "C" { static gboolean before_resize(void* data) { wxWindow* win = static_cast(data); win->InvalidateBestSize(); return false; } } #endif // __WXGTK3__ bool wxWindowGTK::SetFont( const wxFont &font ) { if (!wxWindowBase::SetFont(font)) return false; if (m_widget) { // apply style change (forceStyle=true so that new style is applied // even if the font changed from valid to wxNullFont): GTKApplyWidgetStyle(true); InvalidateBestSize(); } #ifdef __WXGTK3__ // Starting with GTK 3.6, style information is cached, and the cache is only // updated before resizing, or when showing a TLW. If a different size font // is set, our best size calculation will be wrong. All we can do is // invalidate the best size right before the style cache is updated, so any // subsequent best size requests use the correct font. if (gtk_check_version(3,8,0) == NULL) gs_sizeRevalidateList = g_list_prepend(gs_sizeRevalidateList, this); else if (gtk_check_version(3,6,0) == NULL) { wxWindow* tlw = wxGetTopLevelParent(static_cast(this)); if (tlw->m_widget && gtk_widget_get_visible(tlw->m_widget)) g_idle_add_full(GTK_PRIORITY_RESIZE - 1, before_resize, this, NULL); else gs_sizeRevalidateList = g_list_prepend(gs_sizeRevalidateList, this); } #endif return true; } void wxWindowGTK::DoCaptureMouse() { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); GdkWindow *window = NULL; if (m_wxwindow) window = GTKGetDrawingWindow(); else window = gtk_widget_get_window(GetConnectWidget()); wxCHECK_RET( window, wxT("CaptureMouse() failed") ); const GdkEventMask mask = GdkEventMask( GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_HINT_MASK | GDK_POINTER_MOTION_MASK); #ifdef __WXGTK3__ GdkDisplay* display = gdk_window_get_display(window); GdkDeviceManager* manager = gdk_display_get_device_manager(display); GdkDevice* device = gdk_device_manager_get_client_pointer(manager); gdk_device_grab( device, window, GDK_OWNERSHIP_NONE, false, mask, NULL, unsigned(GDK_CURRENT_TIME)); #else gdk_pointer_grab( window, FALSE, mask, NULL, NULL, (guint32)GDK_CURRENT_TIME ); #endif g_captureWindow = this; g_captureWindowHasMouse = true; } void wxWindowGTK::DoReleaseMouse() { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); wxCHECK_RET( g_captureWindow, wxT("can't release mouse - not captured") ); g_captureWindow = NULL; GdkWindow *window = NULL; if (m_wxwindow) window = GTKGetDrawingWindow(); else window = gtk_widget_get_window(GetConnectWidget()); if (!window) return; #ifdef __WXGTK3__ GdkDisplay* display = gdk_window_get_display(window); GdkDeviceManager* manager = gdk_display_get_device_manager(display); GdkDevice* device = gdk_device_manager_get_client_pointer(manager); gdk_device_ungrab(device, unsigned(GDK_CURRENT_TIME)); #else gdk_pointer_ungrab ( (guint32)GDK_CURRENT_TIME ); #endif } void wxWindowGTK::GTKReleaseMouseAndNotify() { GdkDisplay* display = gtk_widget_get_display(m_widget); #ifdef __WXGTK3__ GdkDeviceManager* manager = gdk_display_get_device_manager(display); GdkDevice* device = gdk_device_manager_get_client_pointer(manager); gdk_device_ungrab(device, unsigned(GDK_CURRENT_TIME)); #else gdk_display_pointer_ungrab(display, unsigned(GDK_CURRENT_TIME)); #endif g_captureWindow = NULL; NotifyCaptureLost(); } void wxWindowGTK::GTKHandleCaptureLost() { g_captureWindow = NULL; NotifyCaptureLost(); } /* static */ wxWindow *wxWindowBase::GetCapture() { return (wxWindow *)g_captureWindow; } bool wxWindowGTK::IsRetained() const { return false; } void wxWindowGTK::SetScrollbar(int orient, int pos, int thumbVisible, int range, bool WXUNUSED(update)) { const int dir = ScrollDirFromOrient(orient); GtkRange* const sb = m_scrollBar[dir]; wxCHECK_RET( sb, wxT("this window is not scrollable") ); if (range <= 0) { // GtkRange requires upper > lower range = thumbVisible = 1; } g_signal_handlers_block_by_func( sb, (void*)gtk_scrollbar_value_changed, this); GtkAdjustment* adj = gtk_range_get_adjustment(sb); const bool wasVisible = gtk_adjustment_get_upper(adj) > gtk_adjustment_get_page_size(adj); g_object_freeze_notify(G_OBJECT(adj)); gtk_range_set_increments(sb, 1, thumbVisible); gtk_adjustment_set_page_size(adj, thumbVisible); gtk_range_set_range(sb, 0, range); g_object_thaw_notify(G_OBJECT(adj)); gtk_range_set_value(sb, pos); m_scrollPos[dir] = gtk_range_get_value(sb); const bool isVisible = gtk_adjustment_get_upper(adj) > gtk_adjustment_get_page_size(adj); if (isVisible != wasVisible) m_useCachedClientSize = false; g_signal_handlers_unblock_by_func( sb, (void*)gtk_scrollbar_value_changed, this); } void wxWindowGTK::SetScrollPos(int orient, int pos, bool WXUNUSED(refresh)) { const int dir = ScrollDirFromOrient(orient); GtkRange * const sb = m_scrollBar[dir]; wxCHECK_RET( sb, wxT("this window is not scrollable") ); // This check is more than an optimization. Without it, the slider // will not move smoothly while tracking when using wxScrollHelper. if (GetScrollPos(orient) != pos) { g_signal_handlers_block_by_func( sb, (void*)gtk_scrollbar_value_changed, this); gtk_range_set_value(sb, pos); m_scrollPos[dir] = gtk_range_get_value(sb); g_signal_handlers_unblock_by_func( sb, (void*)gtk_scrollbar_value_changed, this); } } int wxWindowGTK::GetScrollThumb(int orient) const { GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)]; wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") ); return wxRound(gtk_adjustment_get_page_size(gtk_range_get_adjustment(sb))); } int wxWindowGTK::GetScrollPos( int orient ) const { GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)]; wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") ); return wxRound(gtk_range_get_value(sb)); } int wxWindowGTK::GetScrollRange( int orient ) const { GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)]; wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") ); return wxRound(gtk_adjustment_get_upper(gtk_range_get_adjustment(sb))); } // Determine if increment is the same as +/-x, allowing for some small // difference due to possible inexactness in floating point arithmetic static inline bool IsScrollIncrement(double increment, double x) { wxASSERT(increment > 0); const double tolerance = 1.0 / 1024; return fabs(increment - fabs(x)) < tolerance; } wxEventType wxWindowGTK::GTKGetScrollEventType(GtkRange* range) { wxASSERT(range == m_scrollBar[0] || range == m_scrollBar[1]); const int barIndex = range == m_scrollBar[1]; GtkAdjustment* adj = gtk_range_get_adjustment(range); const double value = gtk_adjustment_get_value(adj); // save previous position const double oldPos = m_scrollPos[barIndex]; // update current position m_scrollPos[barIndex] = value; // If event should be ignored, or integral position has not changed // or scrollbar is disabled (webkitgtk is known to cause a "value-changed" // by setting the GtkAdjustment to all zeros) if (g_blockEventsOnDrag || wxRound(value) == wxRound(oldPos) || gtk_adjustment_get_upper(adj) <= gtk_adjustment_get_page_size(adj)) { return wxEVT_NULL; } wxEventType eventType = wxEVT_SCROLL_THUMBTRACK; if (!m_isScrolling) { // Difference from last change event const double diff = value - oldPos; const bool isDown = diff > 0; if (IsScrollIncrement(gtk_adjustment_get_step_increment(adj), diff)) { eventType = isDown ? wxEVT_SCROLL_LINEDOWN : wxEVT_SCROLL_LINEUP; } else if (IsScrollIncrement(gtk_adjustment_get_page_increment(adj), diff)) { eventType = isDown ? wxEVT_SCROLL_PAGEDOWN : wxEVT_SCROLL_PAGEUP; } else if (m_mouseButtonDown) { // Assume track event m_isScrolling = true; } } return eventType; } void wxWindowGTK::ScrollWindow( int dx, int dy, const wxRect* WXUNUSED(rect) ) { wxCHECK_RET( m_widget != NULL, wxT("invalid window") ); wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") ); // No scrolling requested. if ((dx == 0) && (dy == 0)) return; m_clipPaintRegion = true; WX_PIZZA(m_wxwindow)->scroll(dx, dy); m_clipPaintRegion = false; #if wxUSE_CARET bool restoreCaret = (GetCaret() != NULL && GetCaret()->IsVisible()); if (restoreCaret) { wxRect caretRect(GetCaret()->GetPosition(), GetCaret()->GetSize()); if (dx > 0) caretRect.width += dx; else { caretRect.x += dx; caretRect.width -= dx; } if (dy > 0) caretRect.height += dy; else { caretRect.y += dy; caretRect.height -= dy; } RefreshRect(caretRect); } #endif // wxUSE_CARET } void wxWindowGTK::GTKScrolledWindowSetBorder(GtkWidget* w, int wxstyle) { //RN: Note that static controls usually have no border on gtk, so maybe //it makes sense to treat that as simply no border at the wx level //as well... if (!(wxstyle & wxNO_BORDER) && !(wxstyle & wxBORDER_STATIC)) { GtkShadowType gtkstyle; if(wxstyle & wxBORDER_RAISED) gtkstyle = GTK_SHADOW_OUT; else if ((wxstyle & wxBORDER_SUNKEN) || (wxstyle & wxBORDER_THEME)) gtkstyle = GTK_SHADOW_IN; #if 0 // Now obsolete else if (wxstyle & wxBORDER_DOUBLE) gtkstyle = GTK_SHADOW_ETCHED_IN; #endif else //default gtkstyle = GTK_SHADOW_IN; gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w), gtkstyle ); } } // Find the wxWindow at the current mouse position, also returning the mouse // position. wxWindow* wxFindWindowAtPointer(wxPoint& pt) { pt = wxGetMousePosition(); wxWindow* found = wxFindWindowAtPoint(pt); return found; } // Get the current mouse position. void wxGetMousePosition(int* x, int* y) { GdkDisplay* display = GetDisplay(); #ifdef __WXGTK3__ GdkDeviceManager* manager = gdk_display_get_device_manager(display); GdkDevice* device = gdk_device_manager_get_client_pointer(manager); gdk_device_get_position(device, NULL, x, y); #else gdk_display_get_pointer(display, NULL, x, y, NULL); #endif } wxPoint wxGetMousePosition() { wxPoint pt; wxGetMousePosition(&pt.x, &pt.y); return pt; } GdkWindow* wxWindowGTK::GTKGetDrawingWindow() const { GdkWindow* window = NULL; if (m_wxwindow) window = gtk_widget_get_window(m_wxwindow); return window; } // ---------------------------------------------------------------------------- // freeze/thaw // ---------------------------------------------------------------------------- extern "C" { static gboolean draw_freeze(GtkWidget*, void*, wxWindow*) { // stop other handlers from being invoked return true; } } void wxWindowGTK::GTKConnectFreezeWidget(GtkWidget* widget) { #ifdef __WXGTK3__ gulong id = g_signal_connect(widget, "draw", G_CALLBACK(draw_freeze), this); #else gulong id = g_signal_connect(widget, "expose-event", G_CALLBACK(draw_freeze), this); #endif g_signal_handler_block(widget, id); } void wxWindowGTK::GTKFreezeWidget(GtkWidget* widget) { g_signal_handlers_unblock_by_func(widget, (void*)draw_freeze, this); } void wxWindowGTK::GTKThawWidget(GtkWidget* widget) { g_signal_handlers_block_by_func(widget, (void*)draw_freeze, this); gtk_widget_queue_draw(widget); } void wxWindowGTK::DoFreeze() { wxCHECK_RET(m_widget, "invalid window"); GTKFreezeWidget(m_widget); if (m_wxwindow && m_wxwindow != m_widget) GTKFreezeWidget(m_wxwindow); } void wxWindowGTK::DoThaw() { wxCHECK_RET(m_widget, "invalid window"); GTKThawWidget(m_widget); if (m_wxwindow && m_wxwindow != m_widget) GTKThawWidget(m_wxwindow); }