Files
wxWidgets/src/gtk/window.cpp
Paul Cornett a050652c6a Use "notify::gtk-theme-name" from GtkSettings to generate wxSysColourChangedEvent
"style-updated" occurs frequently for other reasons, such as switching focus between TLWs

(cherry picked from commit 61c8a7ca60)
2016-12-13 09:07:50 -08:00

5152 lines
156 KiB
C++

/////////////////////////////////////////////////////////////////////////////
// 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 <ctype.h>
#include <gtk/gtk.h>
#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 <gdk/gdkx.h>
#include "wx/x11/private/wrapxkb.h"
#else
typedef guint KeySym;
#endif
#include <gdk/gdkkeysyms.h>
#ifdef __WXGTK3__
#include <gdk/gdkkeysyms-compat.h>
#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<unsigned long>(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<wxWindow*>(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<wxWindow*>(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<wxWindow*>(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<wxWindowGTK*>(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<wxWindowGTK*>(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<const wxWindow*>(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<wxWindow*>(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<wxWindow*>(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<wxWindow*>(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<wxWindow*>(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<wxWindow*>(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<wxWindow*>(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);
}