]> git.saurik.com Git - wxWidgets.git/blame - src/gtk/window.cpp
always skip OnSize
[wxWidgets.git] / src / gtk / window.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
faa94f3e 2// Name: src/gtk/window.cpp
9a33c3ef 3// Purpose: wxWindowGTK implementation
c801d85f 4// Author: Robert Roebling
c67d8618 5// Id: $Id$
01111366 6// Copyright: (c) 1998 Robert Roebling, Julian Smart
65571936 7// Licence: wxWindows licence
c801d85f
KB
8/////////////////////////////////////////////////////////////////////////////
9
14f355c2
VS
10// For compilers that support precompilation, includes "wx.h".
11#include "wx/wxprec.h"
12
d02af7bb
JJ
13#ifdef __VMS
14#define XWarpPointer XWARPPOINTER
15#endif
16
c801d85f 17#include "wx/window.h"
88a7a4e1
WS
18
19#ifndef WX_PRECOMP
e4db172a 20 #include "wx/log.h"
670f9935 21 #include "wx/app.h"
cca410b3 22 #include "wx/toplevel.h"
ed4b0fdc 23 #include "wx/dcclient.h"
3b3dc801 24 #include "wx/menu.h"
9eddec69 25 #include "wx/settings.h"
246c5004 26 #include "wx/msgdlg.h"
18680f86 27 #include "wx/math.h"
88a7a4e1
WS
28#endif
29
bfeeb7f3
PC
30#include "wx/dnd.h"
31#include "wx/tooltip.h"
32#include "wx/caret.h"
48d011c8 33#include "wx/fontutil.h"
8ab7b4c5 34#include "wx/sysopt.h"
b4071e91 35
fab591c5 36#include <ctype.h>
c801d85f 37
9e691f46 38#include "wx/gtk/private.h"
bbd92d1d 39#include "wx/gtk/private/win_gtk.h"
14b44999 40
3ac8d3bc 41#include <gdk/gdkx.h>
6bc8a1c8 42
14b44999
VS
43#include <gdk/gdkkeysyms.h>
44#if GTK_CHECK_VERSION(3,0,0)
45#include <gdk/gdkkeysyms-compat.h>
46#endif
47
87ec008f
VS
48#if !GTK_CHECK_VERSION(2,10,0)
49 // GTK+ can reliably detect Meta key state only since 2.10 when
50 // GDK_META_MASK was introduced -- there wasn't any way to detect it
51 // in older versions. wxGTK used GDK_MOD2_MASK for this purpose, but
52 // GDK_MOD2_MASK is documented as:
53 //
54 // the fifth modifier key (it depends on the modifier mapping of the X
55 // server which key is interpreted as this modifier)
56 //
57 // In other words, it isn't guaranteed to map to Meta. This is a real
58 // problem: it is common to map NumLock to it (in fact, it's an exception
59 // if the X server _doesn't_ use it for NumLock). So the old code caused
60 // wxKeyEvent::MetaDown() to always return true as long as NumLock was on
61 // on many systems, which broke all applications using
62 // wxKeyEvent::GetModifiers() to check modifiers state (see e.g. here:
63 // http://tinyurl.com/56lsk2).
64 //
65 // Because of this, it's better to not detect Meta key state at all than
66 // to detect it incorrectly. Hence the following #define, which causes
67 // m_metaDown to be always set to false.
68 #define GDK_META_MASK 0
69#endif
70
868a2826
RR
71//-----------------------------------------------------------------------------
72// documentation on internals
73//-----------------------------------------------------------------------------
74
75/*
76 I have been asked several times about writing some documentation about
77ffb593 77 the GTK port of wxWidgets, especially its internal structures. Obviously,
868a2826 78 you cannot understand wxGTK without knowing a little about the GTK, but
47d67540 79 some more information about what the wxWindow, which is the base class
868a2826 80 for all other window classes, does seems required as well.
47d67540 81
30760ce7
RR
82 I)
83
868a2826 84 What does wxWindow do? It contains the common interface for the following
e380f72b 85 jobs of its descendants:
47d67540 86
868a2826 87 1) Define the rudimentary behaviour common to all window classes, such as
e380f72b
RR
88 resizing, intercepting user input (so as to make it possible to use these
89 events for special purposes in a derived class), window names etc.
868a2826
RR
90
91 2) Provide the possibility to contain and manage children, if the derived
92 class is allowed to contain children, which holds true for those window
e380f72b 93 classes which do not display a native GTK widget. To name them, these
868a2826 94 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
47d67540 95 work classes are a special case and are handled a bit differently from
e380f72b 96 the rest. The same holds true for the wxNotebook class.
47d67540 97
868a2826
RR
98 3) Provide the possibility to draw into a client area of a window. This,
99 too, only holds true for classes that do not display a native GTK widget
100 as above.
47d67540 101
e380f72b
RR
102 4) Provide the entire mechanism for scrolling widgets. This actual inter-
103 face for this is usually in wxScrolledWindow, but the GTK implementation
868a2826 104 is in this class.
47d67540 105
868a2826
RR
106 5) A multitude of helper or extra methods for special purposes, such as
107 Drag'n'Drop, managing validators etc.
47d67540 108
30760ce7
RR
109 6) Display a border (sunken, raised, simple or none).
110
77ffb593 111 Normally one might expect, that one wxWidgets window would always correspond
6a17b868 112 to one GTK widget. Under GTK, there is no such all-round widget that has all
868a2826
RR
113 the functionality. Moreover, the GTK defines a client area as a different
114 widget from the actual widget you are handling. Last but not least some
115 special classes (e.g. wxFrame) handle different categories of widgets and
116 still have the possibility to draw something in the client area.
117 It was therefore required to write a special purpose GTK widget, that would
77ffb593 118 represent a client area in the sense of wxWidgets capable to do the jobs
868a2826
RR
119 2), 3) and 4). I have written this class and it resides in win_gtk.c of
120 this directory.
47d67540 121
868a2826 122 All windows must have a widget, with which they interact with other under-
e380f72b 123 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
90e572f1 124 the wxWindow class has a member variable called m_widget which holds a
e380f72b
RR
125 pointer to this widget. When the window class represents a GTK native widget,
126 this is (in most cases) the only GTK widget the class manages. E.g. the
f8e045e2 127 wxStaticText class handles only a GtkLabel widget a pointer to which you
e380f72b 128 can find in m_widget (defined in wxWindow)
8bbe427f 129
e380f72b 130 When the class has a client area for drawing into and for containing children
08f53168
PC
131 it has to handle the client area widget (of the type wxPizza, defined in
132 win_gtk.cpp), but there could be any number of widgets, handled by a class.
8bbe427f
VZ
133 The common rule for all windows is only, that the widget that interacts with
134 the rest of GTK must be referenced in m_widget and all other widgets must be
135 children of this widget on the GTK level. The top-most widget, which also
136 represents the client area, must be in the m_wxwindow field and must be of
08f53168 137 the type wxPizza.
47d67540 138
868a2826
RR
139 As I said, the window classes that display a GTK native widget only have
140 one widget, so in the case of e.g. the wxButton class m_widget holds a
141 pointer to a GtkButton widget. But windows with client areas (for drawing
142 and children) have a m_widget field that is a pointer to a GtkScrolled-
08f53168 143 Window and a m_wxwindow field that is pointer to a wxPizza and this
868a2826 144 one is (in the GTK sense) a child of the GtkScrolledWindow.
47d67540 145
868a2826 146 If the m_wxwindow field is set, then all input to this widget is inter-
77ffb593 147 cepted and sent to the wxWidgets class. If not, all input to the widget
868a2826 148 that gets pointed to by m_widget gets intercepted and sent to the class.
148cd9b6 149
30760ce7 150 II)
148cd9b6 151
77ffb593 152 The design of scrolling in wxWidgets is markedly different from that offered
30760ce7
RR
153 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
154 clicking on a scrollbar belonging to scrolled window will inevitably move
77ffb593 155 the window. In wxWidgets, the scrollbar will only emit an event, send this
30760ce7 156 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
08f53168 157 which actually moves the window and its sub-windows. Note that wxPizza
77ffb593 158 memorizes how much it has been scrolled but that wxWidgets forgets this
30760ce7 159 so that the two coordinates systems have to be kept in synch. This is done
08f53168 160 in various places using the pizza->m_scroll_x and pizza->m_scroll_y values.
148cd9b6
VZ
161
162 III)
163
6a17b868 164 Singularly the most broken code in GTK is the code that is supposed to
30760ce7
RR
165 inform subwindows (child windows) about new positions. Very often, duplicate
166 events are sent without changes in size or position, equally often no
167 events are sent at all (All this is due to a bug in the GtkContainer code
168 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
169 GTK's own system and it simply waits for size events for toplevel windows
170 and then iterates down the respective size events to all window. This has
90e572f1 171 the disadvantage that windows might get size events before the GTK widget
30760ce7 172 actually has the reported size. This doesn't normally pose any problem, but
90e572f1 173 the OpenGL drawing routines rely on correct behaviour. Therefore, I have
30760ce7
RR
174 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
175 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
90e572f1 176 window that is used for OpenGL output really has that size (as reported by
30760ce7
RR
177 GTK).
178
179 IV)
148cd9b6 180
30760ce7 181 If someone at some point of time feels the immense desire to have a look at,
90e572f1
MR
182 change or attempt to optimise the Refresh() logic, this person will need an
183 intimate understanding of what "draw" and "expose" events are and what
184 they are used for, in particular when used in connection with GTK's
30760ce7 185 own windowless widgets. Beware.
148cd9b6 186
30760ce7 187 V)
148cd9b6 188
30760ce7
RR
189 Cursors, too, have been a constant source of pleasure. The main difficulty
190 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
c2246a38
RR
191 for the parent. To prevent this from doing too much harm, SetCursor calls
192 GTKUpdateCursor, which will recursively re-set the cursors of all child windows.
30760ce7
RR
193 Also don't forget that cursors (like much else) are connected to GdkWindows,
194 not GtkWidgets and that the "window" field of a GtkWidget might very well
90e572f1 195 point to the GdkWindow of the parent widget (-> "window-less widget") and
30760ce7 196 that the two obviously have very different meanings.
868a2826
RR
197*/
198
f03fc89f
VZ
199//-----------------------------------------------------------------------------
200// data
201//-----------------------------------------------------------------------------
202
b541538f
PC
203// Don't allow event propagation during drag
204bool g_blockEventsOnDrag;
205// Don't allow mouse event propagation during scroll
206bool g_blockEventsOnScroll;
238d735d 207extern wxCursor g_globalCursor;
f68586e5 208
1e6feb95
VZ
209// mouse capture state: the window which has it and if the mouse is currently
210// inside it
d3b9f782 211static wxWindowGTK *g_captureWindow = NULL;
0a164d4c 212static bool g_captureWindowHasMouse = false;
1e6feb95 213
22f43cb5
VS
214// The window that currently has focus:
215static wxWindowGTK *gs_currentFocus = NULL;
216// The window that is scheduled to get focus in the next event loop iteration
217// or NULL if there's no pending focus change:
218static wxWindowGTK *gs_pendingFocus = NULL;
1e6feb95 219
bd2e08d0
VS
220// the window that has deferred focus-out event pending, if any (see
221// GTKAddDeferredFocusOut() for details)
222static wxWindowGTK *gs_deferredFocusOut = NULL;
d7fa7eaa 223
5e513780
RR
224// global variables because GTK+ DnD want to have the
225// mouse event that caused it
d3b9f782 226GdkEvent *g_lastMouseEvent = NULL;
5e513780 227int g_lastButtonNumber = 0;
8f9850dd 228
2e563988
RR
229//-----------------------------------------------------------------------------
230// debug
231//-----------------------------------------------------------------------------
232
6cad4f1b 233// the trace mask used for the focus debugging messages
9a83f860 234#define TRACE_FOCUS wxT("focus")
6cad4f1b 235
85eb36c2
RR
236//-----------------------------------------------------------------------------
237// missing gdk functions
238//-----------------------------------------------------------------------------
239
240void
241gdk_window_warp_pointer (GdkWindow *window,
c50f1fb9
VZ
242 gint x,
243 gint y)
85eb36c2 244{
85eb36c2 245 if (!window)
2454dc8a 246 window = gdk_get_default_root_window();
c50f1fb9 247
dd00f3f6
OK
248 if (!GDK_WINDOW_DESTROYED(window))
249 {
250 XWarpPointer (GDK_WINDOW_XDISPLAY(window),
251 None, /* not source window -> move from anywhere */
252 GDK_WINDOW_XID(window), /* dest window */
253 0, 0, 0, 0, /* not source window -> move from anywhere */
254 x, y );
255 }
85eb36c2
RR
256}
257
f6bcfd97 258
47c93b63
RR
259//-----------------------------------------------------------------------------
260// "size_request" of m_widget
261//-----------------------------------------------------------------------------
262
865bb325 263extern "C" {
9800347d
FM
264static void
265wxgtk_window_size_request_callback(GtkWidget * WXUNUSED(widget),
266 GtkRequisition *requisition,
267 wxWindow * win)
47c93b63 268{
e1f448ee 269 int w, h;
47c93b63 270 win->GetSize( &w, &h );
e1f448ee
VZ
271 if (w < 2)
272 w = 2;
273 if (h < 2)
274 h = 2;
1e6feb95 275
47c93b63
RR
276 requisition->height = h;
277 requisition->width = w;
278}
865bb325 279}
47c93b63 280
c801d85f 281//-----------------------------------------------------------------------------
034be888 282// "expose_event" of m_wxwindow
c801d85f
KB
283//-----------------------------------------------------------------------------
284
865bb325 285extern "C" {
7f7beb1d 286static gboolean
f089940f 287gtk_window_expose_callback( GtkWidget*,
7f7beb1d
MR
288 GdkEventExpose *gdk_event,
289 wxWindow *win )
47d67540 290{
f089940f 291 if (gdk_event->window == win->GTKGetDrawingWindow())
3d2d8da1 292 {
8d49fda6
PC
293 win->GetUpdateRegion() = wxRegion( gdk_event->region );
294 win->GtkSendPaintEvents();
3d2d8da1 295 }
90e572f1 296 // Let parent window draw window-less widgets
2109c024 297 return FALSE;
b6fa52db 298}
865bb325 299}
b6fa52db 300
75f661bb 301#ifndef __WXUNIVERSAL__
08f53168 302//-----------------------------------------------------------------------------
75f661bb 303// "expose_event" from m_wxwindow->parent, for drawing border
08f53168
PC
304//-----------------------------------------------------------------------------
305
08f53168
PC
306extern "C" {
307static gboolean
6de67633 308expose_event_border(GtkWidget* widget, GdkEventExpose* gdk_event, wxWindow* win)
08f53168 309{
f089940f 310 if (gdk_event->window != gtk_widget_get_parent_window(win->m_wxwindow))
75f661bb
PC
311 return false;
312
6e7cf3bd
PC
313 if (!win->IsShown())
314 return false;
315
75f661bb
PC
316 const GtkAllocation& alloc = win->m_wxwindow->allocation;
317 const int x = alloc.x;
318 const int y = alloc.y;
319 const int w = alloc.width;
320 const int h = alloc.height;
321
322 if (w <= 0 || h <= 0)
08f53168
PC
323 return false;
324
08f53168
PC
325 if (win->HasFlag(wxBORDER_SIMPLE))
326 {
830910cc
PC
327 gdk_draw_rectangle(gdk_event->window,
328 widget->style->black_gc, false, x, y, w - 1, h - 1);
08f53168
PC
329 }
330 else
331 {
332 GtkShadowType shadow = GTK_SHADOW_IN;
333 if (win->HasFlag(wxBORDER_RAISED))
334 shadow = GTK_SHADOW_OUT;
03eaa52a
PC
335
336 // Style detail to use
337 const char* detail;
75f661bb 338 if (win->m_widget == win->m_wxwindow)
03eaa52a
PC
339 // for non-scrollable wxWindows
340 detail = "entry";
341 else
342 // for scrollable ones
343 detail = "viewport";
344
08f53168 345 gtk_paint_shadow(
830910cc
PC
346 win->m_wxwindow->style, gdk_event->window, GTK_STATE_NORMAL,
347 shadow, NULL, wxGTKPrivate::GetEntryWidget(), detail, x, y, w, h);
08f53168 348 }
75f661bb
PC
349 return false;
350}
351}
352
353//-----------------------------------------------------------------------------
354// "parent_set" from m_wxwindow
355//-----------------------------------------------------------------------------
08f53168 356
75f661bb
PC
357extern "C" {
358static void
359parent_set(GtkWidget* widget, GtkObject* old_parent, wxWindow* win)
360{
361 if (old_parent)
362 {
363 g_signal_handlers_disconnect_by_func(
364 old_parent, (void*)expose_event_border, win);
365 }
366 if (widget->parent)
367 {
368 g_signal_connect_after(widget->parent, "expose_event",
369 G_CALLBACK(expose_event_border), win);
370 }
08f53168
PC
371}
372}
373#endif // !__WXUNIVERSAL__
374
c801d85f 375//-----------------------------------------------------------------------------
b292e2f5 376// "key_press_event" from any window
c801d85f 377//-----------------------------------------------------------------------------
c801d85f 378
7da1b5f9
RD
379// These are used when transforming Ctrl-alpha to ascii values 1-26
380inline bool wxIsLowerChar(int code)
381{
382 return (code >= 'a' && code <= 'z' );
383}
384
385inline bool wxIsUpperChar(int code)
386{
387 return (code >= 'A' && code <= 'Z' );
388}
389
390
74710601 391// set WXTRACE to this to see the key event codes on the console
9a83f860 392#define TRACE_KEYS wxT("keyevent")
f17393f1 393
1c6896d7
VZ
394// translates an X key symbol to WXK_XXX value
395//
396// if isChar is true it means that the value returned will be used for EVT_CHAR
397// event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
398// for example, while if it is false it means that the value is going to be
399// used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
400// WXK_NUMPAD_DIVIDE
401static long wxTranslateKeySymToWXKey(KeySym keysym, bool isChar)
402{
403 long key_code;
404
405 switch ( keysym )
406 {
407 // Shift, Control and Alt don't generate the CHAR events at all
408 case GDK_Shift_L:
409 case GDK_Shift_R:
410 key_code = isChar ? 0 : WXK_SHIFT;
411 break;
412 case GDK_Control_L:
413 case GDK_Control_R:
414 key_code = isChar ? 0 : WXK_CONTROL;
415 break;
416 case GDK_Meta_L:
417 case GDK_Meta_R:
418 case GDK_Alt_L:
419 case GDK_Alt_R:
420 case GDK_Super_L:
421 case GDK_Super_R:
422 key_code = isChar ? 0 : WXK_ALT;
423 break;
424
425 // neither do the toggle modifies
426 case GDK_Scroll_Lock:
427 key_code = isChar ? 0 : WXK_SCROLL;
428 break;
429
430 case GDK_Caps_Lock:
431 key_code = isChar ? 0 : WXK_CAPITAL;
432 break;
433
434 case GDK_Num_Lock:
435 key_code = isChar ? 0 : WXK_NUMLOCK;
436 break;
437
438
439 // various other special keys
440 case GDK_Menu:
441 key_code = WXK_MENU;
442 break;
443
444 case GDK_Help:
445 key_code = WXK_HELP;
446 break;
447
448 case GDK_BackSpace:
449 key_code = WXK_BACK;
450 break;
451
452 case GDK_ISO_Left_Tab:
453 case GDK_Tab:
454 key_code = WXK_TAB;
455 break;
456
457 case GDK_Linefeed:
458 case GDK_Return:
459 key_code = WXK_RETURN;
460 break;
461
462 case GDK_Clear:
463 key_code = WXK_CLEAR;
464 break;
465
466 case GDK_Pause:
467 key_code = WXK_PAUSE;
468 break;
469
470 case GDK_Select:
471 key_code = WXK_SELECT;
472 break;
473
474 case GDK_Print:
475 key_code = WXK_PRINT;
476 break;
477
478 case GDK_Execute:
479 key_code = WXK_EXECUTE;
480 break;
481
482 case GDK_Escape:
483 key_code = WXK_ESCAPE;
484 break;
485
486 // cursor and other extended keyboard keys
487 case GDK_Delete:
488 key_code = WXK_DELETE;
489 break;
490
491 case GDK_Home:
492 key_code = WXK_HOME;
493 break;
494
495 case GDK_Left:
496 key_code = WXK_LEFT;
497 break;
498
499 case GDK_Up:
500 key_code = WXK_UP;
501 break;
502
503 case GDK_Right:
504 key_code = WXK_RIGHT;
505 break;
506
507 case GDK_Down:
508 key_code = WXK_DOWN;
509 break;
510
511 case GDK_Prior: // == GDK_Page_Up
faa94f3e 512 key_code = WXK_PAGEUP;
1c6896d7
VZ
513 break;
514
515 case GDK_Next: // == GDK_Page_Down
faa94f3e 516 key_code = WXK_PAGEDOWN;
1c6896d7
VZ
517 break;
518
519 case GDK_End:
520 key_code = WXK_END;
521 break;
522
523 case GDK_Begin:
524 key_code = WXK_HOME;
525 break;
526
527 case GDK_Insert:
528 key_code = WXK_INSERT;
529 break;
530
531
532 // numpad keys
533 case GDK_KP_0:
534 case GDK_KP_1:
535 case GDK_KP_2:
536 case GDK_KP_3:
537 case GDK_KP_4:
538 case GDK_KP_5:
539 case GDK_KP_6:
540 case GDK_KP_7:
541 case GDK_KP_8:
542 case GDK_KP_9:
2a230426 543 key_code = (isChar ? '0' : int(WXK_NUMPAD0)) + keysym - GDK_KP_0;
1c6896d7
VZ
544 break;
545
546 case GDK_KP_Space:
2a230426 547 key_code = isChar ? ' ' : int(WXK_NUMPAD_SPACE);
1c6896d7
VZ
548 break;
549
550 case GDK_KP_Tab:
551 key_code = isChar ? WXK_TAB : WXK_NUMPAD_TAB;
552 break;
553
554 case GDK_KP_Enter:
555 key_code = isChar ? WXK_RETURN : WXK_NUMPAD_ENTER;
556 break;
557
558 case GDK_KP_F1:
559 key_code = isChar ? WXK_F1 : WXK_NUMPAD_F1;
560 break;
561
562 case GDK_KP_F2:
563 key_code = isChar ? WXK_F2 : WXK_NUMPAD_F2;
564 break;
565
566 case GDK_KP_F3:
567 key_code = isChar ? WXK_F3 : WXK_NUMPAD_F3;
568 break;
569
570 case GDK_KP_F4:
571 key_code = isChar ? WXK_F4 : WXK_NUMPAD_F4;
572 break;
573
574 case GDK_KP_Home:
575 key_code = isChar ? WXK_HOME : WXK_NUMPAD_HOME;
576 break;
577
578 case GDK_KP_Left:
579 key_code = isChar ? WXK_LEFT : WXK_NUMPAD_LEFT;
580 break;
581
582 case GDK_KP_Up:
583 key_code = isChar ? WXK_UP : WXK_NUMPAD_UP;
584 break;
585
586 case GDK_KP_Right:
587 key_code = isChar ? WXK_RIGHT : WXK_NUMPAD_RIGHT;
588 break;
589
590 case GDK_KP_Down:
591 key_code = isChar ? WXK_DOWN : WXK_NUMPAD_DOWN;
592 break;
593
594 case GDK_KP_Prior: // == GDK_KP_Page_Up
faa94f3e 595 key_code = isChar ? WXK_PAGEUP : WXK_NUMPAD_PAGEUP;
1c6896d7
VZ
596 break;
597
598 case GDK_KP_Next: // == GDK_KP_Page_Down
faa94f3e 599 key_code = isChar ? WXK_PAGEDOWN : WXK_NUMPAD_PAGEDOWN;
1c6896d7
VZ
600 break;
601
602 case GDK_KP_End:
603 key_code = isChar ? WXK_END : WXK_NUMPAD_END;
604 break;
605
606 case GDK_KP_Begin:
607 key_code = isChar ? WXK_HOME : WXK_NUMPAD_BEGIN;
608 break;
609
610 case GDK_KP_Insert:
611 key_code = isChar ? WXK_INSERT : WXK_NUMPAD_INSERT;
612 break;
613
614 case GDK_KP_Delete:
615 key_code = isChar ? WXK_DELETE : WXK_NUMPAD_DELETE;
616 break;
617
618 case GDK_KP_Equal:
2a230426 619 key_code = isChar ? '=' : int(WXK_NUMPAD_EQUAL);
1c6896d7
VZ
620 break;
621
622 case GDK_KP_Multiply:
2a230426 623 key_code = isChar ? '*' : int(WXK_NUMPAD_MULTIPLY);
1c6896d7
VZ
624 break;
625
626 case GDK_KP_Add:
2a230426 627 key_code = isChar ? '+' : int(WXK_NUMPAD_ADD);
1c6896d7
VZ
628 break;
629
630 case GDK_KP_Separator:
631 // FIXME: what is this?
2a230426 632 key_code = isChar ? '.' : int(WXK_NUMPAD_SEPARATOR);
1c6896d7
VZ
633 break;
634
635 case GDK_KP_Subtract:
2a230426 636 key_code = isChar ? '-' : int(WXK_NUMPAD_SUBTRACT);
1c6896d7
VZ
637 break;
638
639 case GDK_KP_Decimal:
2a230426 640 key_code = isChar ? '.' : int(WXK_NUMPAD_DECIMAL);
1c6896d7
VZ
641 break;
642
643 case GDK_KP_Divide:
2a230426 644 key_code = isChar ? '/' : int(WXK_NUMPAD_DIVIDE);
1c6896d7
VZ
645 break;
646
647
648 // function keys
649 case GDK_F1:
650 case GDK_F2:
651 case GDK_F3:
652 case GDK_F4:
653 case GDK_F5:
654 case GDK_F6:
655 case GDK_F7:
656 case GDK_F8:
657 case GDK_F9:
658 case GDK_F10:
659 case GDK_F11:
660 case GDK_F12:
661 key_code = WXK_F1 + keysym - GDK_F1;
662 break;
663
664 default:
665 key_code = 0;
666 }
667
668 return key_code;
669}
670
671static inline bool wxIsAsciiKeysym(KeySym ks)
672{
673 return ks < 256;
674}
675
a3c15d89
VS
676static void wxFillOtherKeyEventFields(wxKeyEvent& event,
677 wxWindowGTK *win,
678 GdkEventKey *gdk_event)
679{
680 int x = 0;
681 int y = 0;
682 GdkModifierType state;
683 if (gdk_event->window)
684 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
685
686 event.SetTimestamp( gdk_event->time );
cfa8c7d6 687 event.SetId(win->GetId());
d0fb62a6 688
a3c15d89
VS
689 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0;
690 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0;
691 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0;
87ec008f 692 event.m_metaDown = (gdk_event->state & GDK_META_MASK) != 0;
d0fb62a6
VZ
693
694 // Normally we take the state of modifiers directly from the low level GDK
695 // event but unfortunately GDK uses a different convention from MSW for the
696 // key events corresponding to the modifier keys themselves: in it, when
697 // e.g. Shift key is pressed, GDK_SHIFT_MASK is not set while it is set
698 // when Shift is released. Under MSW the situation is exactly reversed and
699 // the modifier corresponding to the key is set when it is pressed and
700 // unset when it is released. To ensure consistent behaviour between
701 // platforms (and because it seems to make slightly more sense, although
702 // arguably both behaviours are reasonable) we follow MSW here.
703 //
704 // Final notice: we set the flags to the desired value instead of just
705 // inverting them because they are not set correctly (i.e. in the same way
706 // as for the real events generated by the user) for wxUIActionSimulator-
707 // produced events and it seems better to keep that class code the same
708 // among all platforms and fix the discrepancy here instead of adding
709 // wxGTK-specific code to wxUIActionSimulator.
710 const bool isPress = gdk_event->type == GDK_KEY_PRESS;
711 switch ( gdk_event->keyval )
712 {
713 case GDK_Shift_L:
714 case GDK_Shift_R:
715 event.m_shiftDown = isPress;
716 break;
717
718 case GDK_Control_L:
719 case GDK_Control_R:
720 event.m_controlDown = isPress;
721 break;
722
723 case GDK_Alt_L:
724 case GDK_Alt_R:
725 event.m_altDown = isPress;
726 break;
727
728 case GDK_Meta_L:
729 case GDK_Meta_R:
730 case GDK_Super_L:
731 case GDK_Super_R:
732 event.m_metaDown = isPress;
733 break;
734 }
735
a3c15d89 736 event.m_rawCode = (wxUint32) gdk_event->keyval;
5995a84f 737 event.m_rawFlags = gdk_event->hardware_keycode;
d0fb62a6 738
c958d025
RR
739 wxGetMousePosition( &x, &y );
740 win->ScreenToClient( &x, &y );
a3c15d89
VS
741 event.m_x = x;
742 event.m_y = y;
743 event.SetEventObject( win );
744}
745
746
74710601
VZ
747static bool
748wxTranslateGTKKeyEventToWx(wxKeyEvent& event,
749 wxWindowGTK *win,
750 GdkEventKey *gdk_event)
47d67540 751{
1c6896d7
VZ
752 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
753 // but only event->keyval which is quite useless to us, so remember
754 // the last character from GDK_KEY_PRESS and reuse it as last resort
755 //
756 // NB: should be MT-safe as we're always called from the main thread only
757 static struct
758 {
759 KeySym keysym;
760 long keycode;
761 } s_lastKeyPress = { 0, 0 };
762
763 KeySym keysym = gdk_event->keyval;
0a62b197 764
9a83f860
VZ
765 wxLogTrace(TRACE_KEYS, wxT("Key %s event: keysym = %ld"),
766 event.GetEventType() == wxEVT_KEY_UP ? wxT("release")
767 : wxT("press"),
0a62b197
VZ
768 keysym);
769
0a164d4c 770 long key_code = wxTranslateKeySymToWXKey(keysym, false /* !isChar */);
1c6896d7
VZ
771
772 if ( !key_code )
773 {
774 // do we have the translation or is it a plain ASCII character?
775 if ( (gdk_event->length == 1) || wxIsAsciiKeysym(keysym) )
776 {
777 // we should use keysym if it is ASCII as X does some translations
778 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
779 // which we don't want here (but which we do use for OnChar())
780 if ( !wxIsAsciiKeysym(keysym) )
781 {
782 keysym = (KeySym)gdk_event->string[0];
783 }
a2053b27 784
1c6896d7 785 // we want to always get the same key code when the same key is
90e572f1 786 // pressed regardless of the state of the modifiers, i.e. on a
1c6896d7
VZ
787 // standard US keyboard pressing '5' or '%' ('5' key with
788 // Shift) should result in the same key code in OnKeyDown():
789 // '5' (although OnChar() will get either '5' or '%').
790 //
791 // to do it we first translate keysym to keycode (== scan code)
792 // and then back but always using the lower register
793 Display *dpy = (Display *)wxGetDisplay();
794 KeyCode keycode = XKeysymToKeycode(dpy, keysym);
0a62b197 795
9a83f860 796 wxLogTrace(TRACE_KEYS, wxT("\t-> keycode %d"), keycode);
0a62b197 797
1c6896d7
VZ
798 KeySym keysymNormalized = XKeycodeToKeysym(dpy, keycode, 0);
799
800 // use the normalized, i.e. lower register, keysym if we've
801 // got one
802 key_code = keysymNormalized ? keysymNormalized : keysym;
803
804 // as explained above, we want to have lower register key codes
805 // normally but for the letter keys we want to have the upper ones
806 //
807 // NB: don't use XConvertCase() here, we want to do it for letters
808 // only
809 key_code = toupper(key_code);
810 }
811 else // non ASCII key, what to do?
812 {
813 // by default, ignore it
814 key_code = 0;
815
816 // but if we have cached information from the last KEY_PRESS
817 if ( gdk_event->type == GDK_KEY_RELEASE )
818 {
819 // then reuse it
820 if ( keysym == s_lastKeyPress.keysym )
821 {
822 key_code = s_lastKeyPress.keycode;
823 }
824 }
825 }
826
827 if ( gdk_event->type == GDK_KEY_PRESS )
828 {
829 // remember it to be reused for KEY_UP event later
830 s_lastKeyPress.keysym = keysym;
831 s_lastKeyPress.keycode = key_code;
832 }
833 }
834
9a83f860 835 wxLogTrace(TRACE_KEYS, wxT("\t-> wxKeyCode %ld"), key_code);
c801d85f 836
74710601 837 // sending unknown key events doesn't really make sense
1c6896d7 838 if ( !key_code )
0a164d4c 839 return false;
3d6f7261 840
276f883f
VZ
841 event.m_keyCode = key_code;
842
7333c0ef
VZ
843#if wxUSE_UNICODE
844 event.m_uniChar = gdk_keyval_to_unicode(key_code ? key_code : keysym);
845 if ( !event.m_uniChar && event.m_keyCode <= WXK_DELETE )
846 {
847 // Set Unicode key code to the ASCII equivalent for compatibility. E.g.
848 // let RETURN generate the key event with both key and Unicode key
849 // codes of 13.
850 event.m_uniChar = event.m_keyCode;
851 }
852#endif // wxUSE_UNICODE
853
1c6896d7 854 // now fill all the other fields
a3c15d89 855 wxFillOtherKeyEventFields(event, win, gdk_event);
0a164d4c 856
0a164d4c 857 return true;
74710601
VZ
858}
859
7c5e6fc6 860
a3c15d89
VS
861struct wxGtkIMData
862{
863 GtkIMContext *context;
864 GdkEventKey *lastKeyEvent;
865
866 wxGtkIMData()
867 {
868 context = gtk_im_multicontext_new();
869 lastKeyEvent = NULL;
870 }
871 ~wxGtkIMData()
872 {
3fe39b0c 873 g_object_unref (context);
a3c15d89
VS
874 }
875};
a3c15d89 876
2933f70a
VZ
877namespace
878{
879
880// Send wxEVT_CHAR_HOOK event to the parent of the window and if it wasn't
881// processed, send wxEVT_CHAR to the window itself. Return true if either of
882// them was handled.
883bool
884SendCharHookAndCharEvents(const wxKeyEvent& event, wxWindow *win)
885{
886 // wxEVT_CHAR_HOOK must be sent to the top level parent window to allow it
5c16a699
VZ
887 // to handle key events in all of its children unless the mouse is captured
888 // in which case we consider that the keyboard should be "captured" too.
889 if ( !g_captureWindow )
2933f70a 890 {
5c16a699
VZ
891 wxWindow * const parent = wxGetTopLevelParent(win);
892 if ( parent )
893 {
894 // We need to make a copy of the event object because it is
895 // modified while it's handled, notably its WasProcessed() flag
896 // is set after it had been processed once.
897 wxKeyEvent eventCharHook(event);
898 eventCharHook.SetEventType(wxEVT_CHAR_HOOK);
899 if ( parent->HandleWindowEvent(eventCharHook) )
900 return true;
901 }
2933f70a
VZ
902 }
903
904 // As above, make a copy of the event first.
905 wxKeyEvent eventChar(event);
906 eventChar.SetEventType(wxEVT_CHAR);
907 return win->HandleWindowEvent(eventChar);
908}
909
910} // anonymous namespace
911
865bb325 912extern "C" {
7f7beb1d 913static gboolean
3591d10f 914gtk_window_key_press_callback( GtkWidget *WXUNUSED(widget),
7f7beb1d
MR
915 GdkEventKey *gdk_event,
916 wxWindow *win )
74710601 917{
1c6896d7
VZ
918 if (!win->m_hasVMT)
919 return FALSE;
920 if (g_blockEventsOnDrag)
921 return FALSE;
f1272160 922
f1272160
RR
923 wxKeyEvent event( wxEVT_KEY_DOWN );
924 bool ret = false;
925 bool return_after_IM = false;
926
84dc821c
JS
927 if( wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
928 {
929 // Emit KEY_DOWN event
937013e0 930 ret = win->HandleWindowEvent( event );
84dc821c
JS
931 }
932 else
f1272160
RR
933 {
934 // Return after IM processing as we cannot do
935 // anything with it anyhow.
936 return_after_IM = true;
937 }
938
9cbe96d0 939 if (!ret && win->m_imData)
f6fca1f8 940 {
9cbe96d0
VZ
941 win->m_imData->lastKeyEvent = gdk_event;
942
f1272160
RR
943 // We should let GTK+ IM filter key event first. According to GTK+ 2.0 API
944 // docs, if IM filter returns true, no further processing should be done.
0a164d4c 945 // we should send the key_down event anyway.
f1272160 946 bool intercepted_by_IM = gtk_im_context_filter_keypress(win->m_imData->context, gdk_event);
f6fca1f8 947 win->m_imData->lastKeyEvent = NULL;
f1272160
RR
948 if (intercepted_by_IM)
949 {
9a83f860 950 wxLogTrace(TRACE_KEYS, wxT("Key event intercepted by IM"));
7f7beb1d 951 return TRUE;
f1272160 952 }
f6fca1f8 953 }
68567a96 954
f1272160 955 if (return_after_IM)
7f7beb1d 956 return FALSE;
68567a96 957
50b58dec
RD
958#if wxUSE_ACCEL
959 if (!ret)
960 {
961 wxWindowGTK *ancestor = win;
962 while (ancestor)
963 {
964 int command = ancestor->GetAcceleratorTable()->GetCommand( event );
965 if (command != -1)
966 {
b1bb04c5
VS
967 wxCommandEvent menu_event( wxEVT_COMMAND_MENU_SELECTED, command );
968 ret = ancestor->HandleWindowEvent( menu_event );
969
970 if ( !ret )
971 {
972 // if the accelerator wasn't handled as menu event, try
973 // it as button click (for compatibility with other
974 // platforms):
975 wxCommandEvent button_event( wxEVT_COMMAND_BUTTON_CLICKED, command );
976 ret = ancestor->HandleWindowEvent( button_event );
977 }
978
50b58dec
RD
979 break;
980 }
981 if (ancestor->IsTopLevel())
982 break;
983 ancestor = ancestor->GetParent();
984 }
985 }
986#endif // wxUSE_ACCEL
987
1ec3a984
RR
988 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
989 // will only be sent if it is not in an accelerator table.
2dde25b9 990 if (!ret)
d728dd40 991 {
7c5e6fc6 992 long key_code;
1c6896d7 993 KeySym keysym = gdk_event->keyval;
36025bcc 994 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
0a164d4c 995 key_code = wxTranslateKeySymToWXKey(keysym, true /* isChar */);
36025bcc 996 if ( !key_code )
1c6896d7 997 {
0b187670 998 if ( wxIsAsciiKeysym(keysym) )
1ec3a984 999 {
36025bcc
VS
1000 // ASCII key
1001 key_code = (unsigned char)keysym;
1002 }
0b187670
RR
1003 // gdk_event->string is actually deprecated
1004 else if ( gdk_event->length == 1 )
1005 {
1006 key_code = (unsigned char)gdk_event->string[0];
1007 }
36025bcc 1008 }
7c5e6fc6 1009
36025bcc
VS
1010 if ( key_code )
1011 {
9a83f860 1012 wxLogTrace(TRACE_KEYS, wxT("Char event: %ld"), key_code);
7c5e6fc6 1013
36025bcc 1014 event.m_keyCode = key_code;
7c5e6fc6 1015
1dabdced
RD
1016 // To conform to the docs we need to translate Ctrl-alpha
1017 // characters to values in the range 1-26.
7da1b5f9
RD
1018 if ( event.ControlDown() &&
1019 ( wxIsLowerChar(key_code) || wxIsUpperChar(key_code) ))
1dabdced 1020 {
7da1b5f9
RD
1021 if ( wxIsLowerChar(key_code) )
1022 event.m_keyCode = key_code - 'a' + 1;
1023 if ( wxIsUpperChar(key_code) )
1024 event.m_keyCode = key_code - 'A' + 1;
1dabdced
RD
1025#if wxUSE_UNICODE
1026 event.m_uniChar = event.m_keyCode;
1027#endif
7f7beb1d 1028 }
1dabdced 1029
2933f70a 1030 ret = SendCharHookAndCharEvents(event, win);
f17393f1 1031 }
d728dd40 1032 }
4d3ab2a0 1033
c7b2e494 1034 return ret;
2b5f62a0 1035}
865bb325 1036}
2b5f62a0 1037
865bb325 1038extern "C" {
7f7beb1d 1039static void
e4161a2a 1040gtk_wxwindow_commit_cb (GtkIMContext * WXUNUSED(context),
7f7beb1d
MR
1041 const gchar *str,
1042 wxWindow *window)
2b5f62a0 1043{
2b5f62a0
VZ
1044 wxKeyEvent event( wxEVT_KEY_DOWN );
1045
a3c15d89
VS
1046 // take modifiers, cursor position, timestamp etc. from the last
1047 // key_press_event that was fed into Input Method:
1048 if (window->m_imData->lastKeyEvent)
1049 {
1050 wxFillOtherKeyEventFields(event,
1051 window, window->m_imData->lastKeyEvent);
1052 }
28c513cb
RR
1053 else
1054 {
1055 event.SetEventObject( window );
1056 }
a3c15d89 1057
6256849f
VS
1058 const wxString data(wxGTK_CONV_BACK_SYS(str));
1059 if( data.empty() )
5bfaca1b 1060 return;
7c5e6fc6 1061
6256849f 1062 for( wxString::const_iterator pstr = data.begin(); pstr != data.end(); ++pstr )
7c5e6fc6 1063 {
5bfaca1b
VS
1064#if wxUSE_UNICODE
1065 event.m_uniChar = *pstr;
f6fca1f8 1066 // Backward compatible for ISO-8859-1
5bfaca1b 1067 event.m_keyCode = *pstr < 256 ? event.m_uniChar : 0;
9a83f860 1068 wxLogTrace(TRACE_KEYS, wxT("IM sent character '%c'"), event.m_uniChar);
5bfaca1b 1069#else
6256849f 1070 event.m_keyCode = (char)*pstr;
5bfaca1b 1071#endif // wxUSE_UNICODE
1dabdced
RD
1072
1073 // To conform to the docs we need to translate Ctrl-alpha
1074 // characters to values in the range 1-26.
7da1b5f9
RD
1075 if ( event.ControlDown() &&
1076 ( wxIsLowerChar(*pstr) || wxIsUpperChar(*pstr) ))
1dabdced 1077 {
7da1b5f9
RD
1078 if ( wxIsLowerChar(*pstr) )
1079 event.m_keyCode = *pstr - 'a' + 1;
1080 if ( wxIsUpperChar(*pstr) )
1081 event.m_keyCode = *pstr - 'A' + 1;
1082
1dabdced
RD
1083 event.m_keyCode = *pstr - 'a' + 1;
1084#if wxUSE_UNICODE
1085 event.m_uniChar = event.m_keyCode;
7f7beb1d
MR
1086#endif
1087 }
1dabdced 1088
2933f70a 1089 SendCharHookAndCharEvents(event, window);
2b5f62a0
VZ
1090 }
1091}
865bb325 1092}
2b5f62a0
VZ
1093
1094
b666df2c
RR
1095//-----------------------------------------------------------------------------
1096// "key_release_event" from any window
1097//-----------------------------------------------------------------------------
1098
865bb325 1099extern "C" {
7f7beb1d 1100static gboolean
e4161a2a 1101gtk_window_key_release_callback( GtkWidget * WXUNUSED(widget),
7f7beb1d
MR
1102 GdkEventKey *gdk_event,
1103 wxWindowGTK *win )
b666df2c 1104{
74710601
VZ
1105 if (!win->m_hasVMT)
1106 return FALSE;
b666df2c 1107
74710601
VZ
1108 if (g_blockEventsOnDrag)
1109 return FALSE;
b666df2c
RR
1110
1111 wxKeyEvent event( wxEVT_KEY_UP );
74710601 1112 if ( !wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
b666df2c 1113 {
90e572f1 1114 // unknown key pressed, ignore (the event would be useless anyhow)
74710601 1115 return FALSE;
b666df2c
RR
1116 }
1117
97687291 1118 return win->GTKProcessEvent(event);
b666df2c 1119}
865bb325 1120}
b666df2c 1121
c5f9d156
VS
1122// ============================================================================
1123// the mouse events
1124// ============================================================================
1125
d1f2ac45
VZ
1126// ----------------------------------------------------------------------------
1127// mouse event processing helpers
1128// ----------------------------------------------------------------------------
1129
50f00d0c
VS
1130// init wxMouseEvent with the info from GdkEventXXX struct
1131template<typename T> void InitMouseEvent(wxWindowGTK *win,
1132 wxMouseEvent& event,
1133 T *gdk_event)
1134{
1135 event.SetTimestamp( gdk_event->time );
c4021a79
PC
1136 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0;
1137 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0;
1138 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0;
1139 event.m_metaDown = (gdk_event->state & GDK_META_MASK) != 0;
1140 event.m_leftDown = (gdk_event->state & GDK_BUTTON1_MASK) != 0;
1141 event.m_middleDown = (gdk_event->state & GDK_BUTTON2_MASK) != 0;
1142 event.m_rightDown = (gdk_event->state & GDK_BUTTON3_MASK) != 0;
1143 event.m_aux1Down = (gdk_event->state & GDK_BUTTON4_MASK) != 0;
1144 event.m_aux2Down = (gdk_event->state & GDK_BUTTON5_MASK) != 0;
50f00d0c
VS
1145
1146 wxPoint pt = win->GetClientAreaOrigin();
1147 event.m_x = (wxCoord)gdk_event->x - pt.x;
1148 event.m_y = (wxCoord)gdk_event->y - pt.y;
fcb29b23 1149
847dfdb4 1150 if ((win->m_wxwindow) && (win->GetLayoutDirection() == wxLayout_RightToLeft))
428f4657
RR
1151 {
1152 // origin in the upper right corner
08f53168 1153 int window_width = win->m_wxwindow->allocation.width;
720a0aab 1154 event.m_x = window_width - event.m_x;
428f4657 1155 }
50f00d0c
VS
1156
1157 event.SetEventObject( win );
1158 event.SetId( win->GetId() );
1159 event.SetTimestamp( gdk_event->time );
1160}
c5f9d156 1161
2daa0ce9
VZ
1162static void AdjustEventButtonState(wxMouseEvent& event)
1163{
1164 // GDK reports the old state of the button for a button press event, but
1165 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1166 // for a LEFT_DOWN event, not FALSE, so we will invert
1167 // left/right/middleDown for the corresponding click events
1e6feb95 1168
1a8caf94
RR
1169 if ((event.GetEventType() == wxEVT_LEFT_DOWN) ||
1170 (event.GetEventType() == wxEVT_LEFT_DCLICK) ||
1171 (event.GetEventType() == wxEVT_LEFT_UP))
1172 {
1173 event.m_leftDown = !event.m_leftDown;
1174 return;
1175 }
1176
1177 if ((event.GetEventType() == wxEVT_MIDDLE_DOWN) ||
1178 (event.GetEventType() == wxEVT_MIDDLE_DCLICK) ||
1179 (event.GetEventType() == wxEVT_MIDDLE_UP))
2daa0ce9 1180 {
1a8caf94
RR
1181 event.m_middleDown = !event.m_middleDown;
1182 return;
1183 }
1184
1185 if ((event.GetEventType() == wxEVT_RIGHT_DOWN) ||
1186 (event.GetEventType() == wxEVT_RIGHT_DCLICK) ||
1187 (event.GetEventType() == wxEVT_RIGHT_UP))
1188 {
1189 event.m_rightDown = !event.m_rightDown;
1190 return;
2daa0ce9
VZ
1191 }
1192}
1193
d1f2ac45
VZ
1194// find the window to send the mouse event too
1195static
1196wxWindowGTK *FindWindowForMouseEvent(wxWindowGTK *win, wxCoord& x, wxCoord& y)
1197{
7d4909b2
RR
1198 wxCoord xx = x;
1199 wxCoord yy = y;
1200
d1f2ac45
VZ
1201 if (win->m_wxwindow)
1202 {
08f53168
PC
1203 wxPizza* pizza = WX_PIZZA(win->m_wxwindow);
1204 xx += pizza->m_scroll_x;
1205 yy += pizza->m_scroll_y;
d1f2ac45
VZ
1206 }
1207
222ed1d6 1208 wxWindowList::compatibility_iterator node = win->GetChildren().GetFirst();
d1f2ac45
VZ
1209 while (node)
1210 {
b1d4dd7a 1211 wxWindowGTK *child = node->GetData();
d1f2ac45 1212
b1d4dd7a 1213 node = node->GetNext();
d1f2ac45
VZ
1214 if (!child->IsShown())
1215 continue;
1216
71ead4bf 1217 if (child->GTKIsTransparentForMouse())
d1f2ac45
VZ
1218 {
1219 // wxStaticBox is transparent in the box itself
1220 int xx1 = child->m_x;
1221 int yy1 = child->m_y;
1222 int xx2 = child->m_x + child->m_width;
7408cf7f 1223 int yy2 = child->m_y + child->m_height;
d1f2ac45
VZ
1224
1225 // left
7d4909b2 1226 if (((xx >= xx1) && (xx <= xx1+10) && (yy >= yy1) && (yy <= yy2)) ||
d1f2ac45 1227 // right
7d4909b2 1228 ((xx >= xx2-10) && (xx <= xx2) && (yy >= yy1) && (yy <= yy2)) ||
d1f2ac45 1229 // top
7d4909b2 1230 ((xx >= xx1) && (xx <= xx2) && (yy >= yy1) && (yy <= yy1+10)) ||
d1f2ac45 1231 // bottom
7d4909b2 1232 ((xx >= xx1) && (xx <= xx2) && (yy >= yy2-1) && (yy <= yy2)))
d1f2ac45
VZ
1233 {
1234 win = child;
1235 x -= child->m_x;
1236 y -= child->m_y;
1237 break;
1238 }
1239
1240 }
1241 else
1242 {
d3b9f782 1243 if ((child->m_wxwindow == NULL) &&
7d4909b2
RR
1244 (child->m_x <= xx) &&
1245 (child->m_y <= yy) &&
af3653dd
RR
1246 (child->m_x+child->m_width >= xx) &&
1247 (child->m_y+child->m_height >= yy))
d1f2ac45
VZ
1248 {
1249 win = child;
1250 x -= child->m_x;
1251 y -= child->m_y;
1252 break;
1253 }
1254 }
1255 }
1256
1257 return win;
1258}
1259
ef5c70f9
VZ
1260// ----------------------------------------------------------------------------
1261// common event handlers helpers
1262// ----------------------------------------------------------------------------
1263
97687291
VZ
1264bool wxWindowGTK::GTKProcessEvent(wxEvent& event) const
1265{
1266 // nothing special at this level
937013e0 1267 return HandleWindowEvent(event);
97687291
VZ
1268}
1269
02bad830
VZ
1270bool wxWindowGTK::GTKShouldIgnoreEvent() const
1271{
1272 return !m_hasVMT || g_blockEventsOnDrag;
1273}
1274
5478f221 1275int wxWindowGTK::GTKCallbackCommonPrologue(GdkEventAny *event) const
ef5c70f9 1276{
ef5c70f9 1277 if (!m_hasVMT)
5478f221 1278 return FALSE;
ef5c70f9 1279 if (g_blockEventsOnDrag)
5478f221 1280 return TRUE;
ef5c70f9 1281 if (g_blockEventsOnScroll)
5478f221 1282 return TRUE;
ef5c70f9
VZ
1283
1284 if (!GTKIsOwnWindow(event->window))
5478f221 1285 return FALSE;
ef5c70f9 1286
5478f221 1287 return -1;
ef5c70f9
VZ
1288}
1289
1290// overloads for all GDK event types we use here: we need to have this as
1291// GdkEventXXX can't be implicitly cast to GdkEventAny even if it, in fact,
1292// derives from it in the sense that the structs have the same layout
1293#define wxDEFINE_COMMON_PROLOGUE_OVERLOAD(T) \
5478f221 1294 static int wxGtkCallbackCommonPrologue(T *event, wxWindowGTK *win) \
ef5c70f9
VZ
1295 { \
1296 return win->GTKCallbackCommonPrologue((GdkEventAny *)event); \
1297 }
1298
1299wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventButton)
1300wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventMotion)
1301wxDEFINE_COMMON_PROLOGUE_OVERLOAD(GdkEventCrossing)
1302
1303#undef wxDEFINE_COMMON_PROLOGUE_OVERLOAD
1304
5478f221
VZ
1305#define wxCOMMON_CALLBACK_PROLOGUE(event, win) \
1306 const int rc = wxGtkCallbackCommonPrologue(event, win); \
1307 if ( rc != -1 ) \
1308 return rc
1309
ef5c70f9
VZ
1310// all event handlers must have C linkage as they're called from GTK+ C code
1311extern "C"
1312{
1313
c801d85f 1314//-----------------------------------------------------------------------------
2f2aa628
RR
1315// "button_press_event"
1316//-----------------------------------------------------------------------------
c801d85f 1317
7f7beb1d
MR
1318static gboolean
1319gtk_window_button_press_callback( GtkWidget *widget,
1320 GdkEventButton *gdk_event,
1321 wxWindowGTK *win )
903f689b 1322{
5478f221 1323 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
034be888 1324
8f9850dd
RR
1325 g_lastButtonNumber = gdk_event->button;
1326
90e572f1 1327 // GDK sends surplus button down events
2b5f62a0
VZ
1328 // before a double click event. We
1329 // need to filter these out.
1e89eecd 1330 if ((gdk_event->type == GDK_BUTTON_PRESS) && (win->m_wxwindow))
2b5f62a0
VZ
1331 {
1332 GdkEvent *peek_event = gdk_event_peek();
1333 if (peek_event)
1334 {
8b8a8e0e
RR
1335 if ((peek_event->type == GDK_2BUTTON_PRESS) ||
1336 (peek_event->type == GDK_3BUTTON_PRESS))
2b5f62a0
VZ
1337 {
1338 gdk_event_free( peek_event );
1339 return TRUE;
1340 }
1341 else
1342 {
1343 gdk_event_free( peek_event );
1344 }
1345 }
1346 }
1347
2daa0ce9 1348 wxEventType event_type = wxEVT_NULL;
47d67540 1349
15475ced
VZ
1350 if ( gdk_event->type == GDK_2BUTTON_PRESS &&
1351 gdk_event->button >= 1 && gdk_event->button <= 3 )
1352 {
1353 // Reset GDK internal timestamp variables in order to disable GDK
1354 // triple click events. GDK will then next time believe no button has
1355 // been clicked just before, and send a normal button click event.
1356 GdkDisplay* display = gtk_widget_get_display (widget);
1357 display->button_click_time[1] = 0;
1358 display->button_click_time[0] = 0;
1359 }
15475ced 1360
f5e27805 1361 if (gdk_event->button == 1)
c801d85f 1362 {
f3f0d961 1363 // note that GDK generates triple click events which are not supported
77ffb593 1364 // by wxWidgets but still have to be passed to the app as otherwise
f3f0d961 1365 // clicks would simply go missing
f5e27805
RR
1366 switch (gdk_event->type)
1367 {
15475ced
VZ
1368 // we shouldn't get triple clicks at all for GTK2 because we
1369 // suppress them artificially using the code above but we still
1370 // should map them to something for GTK1 and not just ignore them
1371 // as this would lose clicks
f3f0d961
VZ
1372 case GDK_3BUTTON_PRESS: // we could also map this to DCLICK...
1373 case GDK_BUTTON_PRESS:
1374 event_type = wxEVT_LEFT_DOWN;
1375 break;
1376
1377 case GDK_2BUTTON_PRESS:
1378 event_type = wxEVT_LEFT_DCLICK;
1379 break;
b1f50e65
VZ
1380
1381 default:
1382 // just to silence gcc warnings
1383 ;
f5e27805 1384 }
362c6693 1385 }
f5e27805 1386 else if (gdk_event->button == 2)
c801d85f 1387 {
f5e27805
RR
1388 switch (gdk_event->type)
1389 {
15475ced 1390 case GDK_3BUTTON_PRESS:
f3f0d961
VZ
1391 case GDK_BUTTON_PRESS:
1392 event_type = wxEVT_MIDDLE_DOWN;
1393 break;
1394
1395 case GDK_2BUTTON_PRESS:
1396 event_type = wxEVT_MIDDLE_DCLICK;
1397 break;
b1f50e65
VZ
1398
1399 default:
1400 ;
f5e27805 1401 }
362c6693 1402 }
f5e27805 1403 else if (gdk_event->button == 3)
c801d85f 1404 {
f5e27805
RR
1405 switch (gdk_event->type)
1406 {
15475ced 1407 case GDK_3BUTTON_PRESS:
f3f0d961
VZ
1408 case GDK_BUTTON_PRESS:
1409 event_type = wxEVT_RIGHT_DOWN;
1410 break;
1411
1412 case GDK_2BUTTON_PRESS:
1413 event_type = wxEVT_RIGHT_DCLICK;
1414 break;
b1f50e65
VZ
1415
1416 default:
1417 ;
2b5f62a0
VZ
1418 }
1419 }
47d67540 1420
2daa0ce9
VZ
1421 if ( event_type == wxEVT_NULL )
1422 {
1423 // unknown mouse button or click type
1424 return FALSE;
1425 }
1426
8f9850dd 1427 g_lastMouseEvent = (GdkEvent*) gdk_event;
fcb29b23 1428
f5e27805 1429 wxMouseEvent event( event_type );
c5f9d156 1430 InitMouseEvent( win, event, gdk_event );
47d67540 1431
2daa0ce9 1432 AdjustEventButtonState(event);
94633ad9 1433
90e572f1
MR
1434 // find the correct window to send the event to: it may be a different one
1435 // from the one which got it at GTK+ level because some controls don't have
d1f2ac45
VZ
1436 // their own X window and thus cannot get any events.
1437 if ( !g_captureWindow )
1438 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
ff8bfdbb 1439
ccb9fa0d
RD
1440 // reset the event object and id in case win changed.
1441 event.SetEventObject( win );
1442 event.SetId( win->GetId() );
670f9935 1443
fcb29b23 1444 bool ret = win->GTKProcessEvent( event );
5e513780 1445 g_lastMouseEvent = NULL;
fcb29b23
VZ
1446 if ( ret )
1447 return TRUE;
47d67540 1448
78cd9c69 1449 if ((event_type == wxEVT_LEFT_DOWN) && !win->IsOfStandardClass() &&
22f43cb5 1450 (gs_currentFocus != win) /* && win->IsFocusable() */)
8622e8cd 1451 {
c7bfb76a 1452 win->SetFocus();
8622e8cd
RR
1453 }
1454
ac103441
RR
1455 if (event_type == wxEVT_RIGHT_DOWN)
1456 {
1457 // generate a "context menu" event: this is similar to right mouse
1458 // click under many GUIs except that it is generated differently
1459 // (right up under MSW, ctrl-click under Mac, right down here) and
1460 //
1461 // (a) it's a command event and so is propagated to the parent
1462 // (b) under some ports it can be generated from kbd too
1463 // (c) it uses screen coords (because of (a))
1464 wxContextMenuEvent evtCtx(
1465 wxEVT_CONTEXT_MENU,
1466 win->GetId(),
1467 win->ClientToScreen(event.GetPosition()));
1468 evtCtx.SetEventObject(win);
97687291 1469 return win->GTKProcessEvent(evtCtx);
ac103441
RR
1470 }
1471
034be888 1472 return FALSE;
362c6693 1473}
c801d85f
KB
1474
1475//-----------------------------------------------------------------------------
97b3455a 1476// "button_release_event"
2f2aa628 1477//-----------------------------------------------------------------------------
c801d85f 1478
7f7beb1d 1479static gboolean
71ead4bf 1480gtk_window_button_release_callback( GtkWidget *WXUNUSED(widget),
7f7beb1d
MR
1481 GdkEventButton *gdk_event,
1482 wxWindowGTK *win )
47d67540 1483{
5478f221 1484 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
47d67540 1485
8f9850dd
RR
1486 g_lastButtonNumber = 0;
1487
f5e27805 1488 wxEventType event_type = wxEVT_NULL;
47d67540 1489
f5e27805
RR
1490 switch (gdk_event->button)
1491 {
2b5f62a0
VZ
1492 case 1:
1493 event_type = wxEVT_LEFT_UP;
1494 break;
1495
1496 case 2:
1497 event_type = wxEVT_MIDDLE_UP;
1498 break;
1499
1500 case 3:
1501 event_type = wxEVT_RIGHT_UP;
1502 break;
1503
1504 default:
6a17b868 1505 // unknown button, don't process
2b5f62a0 1506 return FALSE;
f5e27805 1507 }
47d67540 1508
8f9850dd 1509 g_lastMouseEvent = (GdkEvent*) gdk_event;
fcb29b23 1510
f5e27805 1511 wxMouseEvent event( event_type );
c5f9d156 1512 InitMouseEvent( win, event, gdk_event );
f5e27805 1513
2daa0ce9
VZ
1514 AdjustEventButtonState(event);
1515
d1f2ac45
VZ
1516 if ( !g_captureWindow )
1517 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
47d67540 1518
ccb9fa0d
RD
1519 // reset the event object and id in case win changed.
1520 event.SetEventObject( win );
1521 event.SetId( win->GetId() );
1522
e6cac2df 1523 bool ret = win->GTKProcessEvent(event);
f4322df6 1524
e6cac2df 1525 g_lastMouseEvent = NULL;
f4322df6 1526
e6cac2df 1527 return ret;
362c6693 1528}
c801d85f
KB
1529
1530//-----------------------------------------------------------------------------
2f2aa628
RR
1531// "motion_notify_event"
1532//-----------------------------------------------------------------------------
c801d85f 1533
7f7beb1d 1534static gboolean
e4161a2a 1535gtk_window_motion_notify_callback( GtkWidget * WXUNUSED(widget),
7f7beb1d
MR
1536 GdkEventMotion *gdk_event,
1537 wxWindowGTK *win )
47d67540 1538{
5478f221 1539 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
034be888 1540
ff8bfdbb 1541 if (gdk_event->is_hint)
aae24d21 1542 {
f7a11f8c
RR
1543 int x = 0;
1544 int y = 0;
1545 GdkModifierType state;
1546 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
1547 gdk_event->x = x;
1548 gdk_event->y = y;
aae24d21 1549 }
ff8bfdbb 1550
8f9850dd
RR
1551 g_lastMouseEvent = (GdkEvent*) gdk_event;
1552
e380f72b 1553 wxMouseEvent event( wxEVT_MOTION );
c5f9d156 1554 InitMouseEvent(win, event, gdk_event);
e380f72b 1555
50382578 1556 if ( g_captureWindow )
2f2aa628 1557 {
6a17b868 1558 // synthesise a mouse enter or leave event if needed
1e6feb95 1559 GdkWindow *winUnderMouse = gdk_window_at_pointer(NULL, NULL);
7c5e6fc6 1560 // This seems to be necessary and actually been added to
50382578
RR
1561 // GDK itself in version 2.0.X
1562 gdk_flush();
7c5e6fc6 1563
1e6feb95
VZ
1564 bool hasMouse = winUnderMouse == gdk_event->window;
1565 if ( hasMouse != g_captureWindowHasMouse )
1566 {
1567 // the mouse changed window
1568 g_captureWindowHasMouse = hasMouse;
1569
17a1ebd1
VZ
1570 wxMouseEvent eventM(g_captureWindowHasMouse ? wxEVT_ENTER_WINDOW
1571 : wxEVT_LEAVE_WINDOW);
1572 InitMouseEvent(win, eventM, gdk_event);
1573 eventM.SetEventObject(win);
97687291 1574 win->GTKProcessEvent(eventM);
1e6feb95
VZ
1575 }
1576 }
1577 else // no capture
1578 {
d1f2ac45 1579 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
ccb9fa0d
RD
1580
1581 // reset the event object and id in case win changed.
1582 event.SetEventObject( win );
1583 event.SetId( win->GetId() );
2f2aa628 1584 }
47d67540 1585
38f69be1
RR
1586 if ( !g_captureWindow )
1587 {
1588 wxSetCursorEvent cevent( event.m_x, event.m_y );
97687291 1589 if (win->GTKProcessEvent( cevent ))
38f69be1 1590 {
ec1e0c66 1591 win->SetCursor( cevent.GetCursor() );
38f69be1
RR
1592 }
1593 }
2e1f5012 1594
5e513780 1595 bool ret = win->GTKProcessEvent(event);
fcb29b23 1596
5e513780
RR
1597 g_lastMouseEvent = NULL;
1598
1599 return ret;
362c6693 1600}
c801d85f 1601
557c9f5b 1602//-----------------------------------------------------------------------------
40e5ebbf 1603// "scroll_event" (mouse wheel event)
557c9f5b
JS
1604//-----------------------------------------------------------------------------
1605
7f251559
RR
1606static gboolean
1607window_scroll_event_hscrollbar(GtkWidget*, GdkEventScroll* gdk_event, wxWindow* win)
1608{
1609 if (gdk_event->direction != GDK_SCROLL_LEFT &&
1610 gdk_event->direction != GDK_SCROLL_RIGHT)
1611 {
1612 return false;
1613 }
1614
1615 wxMouseEvent event(wxEVT_MOUSEWHEEL);
1616 InitMouseEvent(win, event, gdk_event);
1617
1618 GtkRange *range = win->m_scrollBar[wxWindow::ScrollDir_Horz];
1619 if (!range) return FALSE;
1620
1621 if (range && GTK_WIDGET_VISIBLE (range))
1622 {
1623 GtkAdjustment *adj = range->adjustment;
1624 gdouble delta = adj->step_increment * 3;
1625 if (gdk_event->direction == GDK_SCROLL_LEFT)
1626 delta = -delta;
1627
1628 gdouble new_value = CLAMP (adj->value + delta, adj->lower, adj->upper - adj->page_size);
35510e48 1629
7f251559
RR
1630 gtk_adjustment_set_value (adj, new_value);
1631
1632 return TRUE;
1633 }
1634
1635 return FALSE;
1636}
1637
7f7beb1d 1638static gboolean
76e4be8e 1639window_scroll_event(GtkWidget*, GdkEventScroll* gdk_event, wxWindow* win)
557c9f5b 1640{
76e4be8e
PC
1641 if (gdk_event->direction != GDK_SCROLL_UP &&
1642 gdk_event->direction != GDK_SCROLL_DOWN)
1643 {
1644 return false;
1645 }
0a164d4c 1646
76e4be8e 1647 wxMouseEvent event(wxEVT_MOUSEWHEEL);
b4e43132 1648 InitMouseEvent(win, event, gdk_event);
c53c7fad
RD
1649
1650 // FIXME: Get these values from GTK or GDK
58d185f7 1651 event.m_linesPerAction = 3;
60773911 1652 event.m_wheelDelta = 120;
557c9f5b
JS
1653 if (gdk_event->direction == GDK_SCROLL_UP)
1654 event.m_wheelRotation = 120;
1655 else
1656 event.m_wheelRotation = -120;
1657
7f251559
RR
1658 if (win->GTKProcessEvent(event))
1659 return TRUE;
35510e48 1660
7f251559
RR
1661 GtkRange *range = win->m_scrollBar[wxWindow::ScrollDir_Vert];
1662 if (!range) return FALSE;
1663
1664 if (range && GTK_WIDGET_VISIBLE (range))
1665 {
1666 GtkAdjustment *adj = range->adjustment;
1667 gdouble delta = adj->step_increment * 3;
1668 if (gdk_event->direction == GDK_SCROLL_UP)
1669 delta = -delta;
1670
1671 gdouble new_value = CLAMP (adj->value + delta, adj->lower, adj->upper - adj->page_size);
35510e48 1672
7f251559
RR
1673 gtk_adjustment_set_value (adj, new_value);
1674
1675 return TRUE;
1676 }
1677
1678 return FALSE;
557c9f5b 1679}
ac103441
RR
1680
1681//-----------------------------------------------------------------------------
1682// "popup-menu"
1683//-----------------------------------------------------------------------------
ef5c70f9 1684
ac103441
RR
1685static gboolean wxgtk_window_popup_menu_callback(GtkWidget*, wxWindowGTK* win)
1686{
ac9d38d8 1687 wxContextMenuEvent event(wxEVT_CONTEXT_MENU, win->GetId(), wxPoint(-1, -1));
ac103441 1688 event.SetEventObject(win);
97687291 1689 return win->GTKProcessEvent(event);
ac103441 1690}
557c9f5b 1691
c801d85f 1692//-----------------------------------------------------------------------------
2f2aa628
RR
1693// "focus_in_event"
1694//-----------------------------------------------------------------------------
c801d85f 1695
7f7beb1d 1696static gboolean
e4161a2a 1697gtk_window_focus_in_callback( GtkWidget * WXUNUSED(widget),
7f7beb1d 1698 GdkEventFocus *WXUNUSED(event),
bd2e08d0 1699 wxWindowGTK *win )
c801d85f 1700{
bd2e08d0 1701 return win->GTKHandleFocusIn();
362c6693 1702}
c801d85f
KB
1703
1704//-----------------------------------------------------------------------------
2f2aa628
RR
1705// "focus_out_event"
1706//-----------------------------------------------------------------------------
c801d85f 1707
7f7beb1d 1708static gboolean
e4161a2a
VZ
1709gtk_window_focus_out_callback( GtkWidget * WXUNUSED(widget),
1710 GdkEventFocus * WXUNUSED(gdk_event),
7f7beb1d 1711 wxWindowGTK *win )
c801d85f 1712{
bd2e08d0 1713 return win->GTKHandleFocusOut();
362c6693 1714}
c801d85f 1715
bd2e08d0
VS
1716//-----------------------------------------------------------------------------
1717// "focus"
1718//-----------------------------------------------------------------------------
1719
28e88942
VZ
1720static gboolean
1721wx_window_focus_callback(GtkWidget *widget,
e4161a2a 1722 GtkDirectionType WXUNUSED(direction),
28e88942
VZ
1723 wxWindowGTK *win)
1724{
08f53168 1725 // the default handler for focus signal in GtkScrolledWindow sets
28e88942
VZ
1726 // focus to the window itself even if it doesn't accept focus, i.e. has no
1727 // GTK_CAN_FOCUS in its style -- work around this by forcibly preventing
1728 // the signal from reaching gtk_scrolled_window_focus() if we don't have
1729 // any children which might accept focus (we know we don't accept the focus
1730 // ourselves as this signal is only connected in this case)
1731 if ( win->GetChildren().empty() )
1732 g_signal_stop_emission_by_name(widget, "focus");
1733
1734 // we didn't change the focus
1735 return FALSE;
1736}
1737
b4071e91
RR
1738//-----------------------------------------------------------------------------
1739// "enter_notify_event"
1740//-----------------------------------------------------------------------------
1741
f3a5f83a
MR
1742static gboolean
1743gtk_window_enter_callback( GtkWidget *widget,
1744 GdkEventCrossing *gdk_event,
1745 wxWindowGTK *win )
b4071e91 1746{
5478f221 1747 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
47d67540 1748
7f5f144a
RR
1749 // Event was emitted after a grab
1750 if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE;
7c5e6fc6 1751
4a33eba6
RR
1752 int x = 0;
1753 int y = 0;
1754 GdkModifierType state = (GdkModifierType)0;
ff8bfdbb 1755
a2053b27 1756 gdk_window_get_pointer( widget->window, &x, &y, &state );
ff8bfdbb 1757
edc1d330 1758 wxMouseEvent event( wxEVT_ENTER_WINDOW );
c5f9d156
VS
1759 InitMouseEvent(win, event, gdk_event);
1760 wxPoint pt = win->GetClientAreaOrigin();
1761 event.m_x = x + pt.x;
1762 event.m_y = y + pt.y;
ff8bfdbb 1763
38f69be1
RR
1764 if ( !g_captureWindow )
1765 {
1766 wxSetCursorEvent cevent( event.m_x, event.m_y );
97687291 1767 if (win->GTKProcessEvent( cevent ))
38f69be1 1768 {
ec1e0c66 1769 win->SetCursor( cevent.GetCursor() );
38f69be1
RR
1770 }
1771 }
2e1f5012 1772
97687291 1773 return win->GTKProcessEvent(event);
b4071e91 1774}
47d67540 1775
b4071e91
RR
1776//-----------------------------------------------------------------------------
1777// "leave_notify_event"
1778//-----------------------------------------------------------------------------
1779
f3a5f83a
MR
1780static gboolean
1781gtk_window_leave_callback( GtkWidget *widget,
1782 GdkEventCrossing *gdk_event,
1783 wxWindowGTK *win )
b4071e91 1784{
5478f221 1785 wxCOMMON_CALLBACK_PROLOGUE(gdk_event, win);
b292e2f5 1786
7f5f144a
RR
1787 // Event was emitted after an ungrab
1788 if (gdk_event->mode != GDK_CROSSING_NORMAL) return FALSE;
7c5e6fc6 1789
e380f72b 1790 wxMouseEvent event( wxEVT_LEAVE_WINDOW );
47d67540 1791
4a33eba6
RR
1792 int x = 0;
1793 int y = 0;
1794 GdkModifierType state = (GdkModifierType)0;
ff8bfdbb 1795
4a33eba6 1796 gdk_window_get_pointer( widget->window, &x, &y, &state );
ff8bfdbb 1797
b4e43132 1798 InitMouseEvent(win, event, gdk_event);
ff8bfdbb 1799
97687291 1800 return win->GTKProcessEvent(event);
b4071e91 1801}
47d67540 1802
c801d85f 1803//-----------------------------------------------------------------------------
add7cadd 1804// "value_changed" from scrollbar
2f2aa628 1805//-----------------------------------------------------------------------------
c801d85f 1806
add7cadd
PC
1807static void
1808gtk_scrollbar_value_changed(GtkRange* range, wxWindow* win)
47d67540 1809{
71ead4bf 1810 wxEventType eventType = win->GTKGetScrollEventType(range);
add7cadd
PC
1811 if (eventType != wxEVT_NULL)
1812 {
1813 // Convert scroll event type to scrollwin event type
1814 eventType += wxEVT_SCROLLWIN_TOP - wxEVT_SCROLL_TOP;
22c9b211
VZ
1815
1816 // find the scrollbar which generated the event
1817 wxWindowGTK::ScrollDir dir = win->ScrollDirFromRange(range);
35510e48 1818
22c9b211 1819 // generate the corresponding wx event
bfeeb7f3 1820 const int orient = wxWindow::OrientFromScrollDir(dir);
add7cadd
PC
1821 wxScrollWinEvent event(eventType, win->GetScrollPos(orient), orient);
1822 event.SetEventObject(win);
22c9b211 1823
97687291 1824 win->GTKProcessEvent(event);
add7cadd 1825 }
362c6693 1826}
c801d85f 1827
cb43b372
RR
1828//-----------------------------------------------------------------------------
1829// "button_press_event" from scrollbar
1830//-----------------------------------------------------------------------------
1831
7f7beb1d 1832static gboolean
add7cadd 1833gtk_scrollbar_button_press_event(GtkRange*, GdkEventButton*, wxWindow* win)
cb43b372 1834{
0a164d4c 1835 g_blockEventsOnScroll = true;
add7cadd 1836 win->m_mouseButtonDown = true;
9e691f46 1837
add7cadd 1838 return false;
cb43b372
RR
1839}
1840
c918b2cd
PC
1841//-----------------------------------------------------------------------------
1842// "event_after" from scrollbar
1843//-----------------------------------------------------------------------------
1844
c918b2cd
PC
1845static void
1846gtk_scrollbar_event_after(GtkRange* range, GdkEvent* event, wxWindow* win)
1847{
1848 if (event->type == GDK_BUTTON_RELEASE)
1849 {
1850 g_signal_handlers_block_by_func(range, (void*)gtk_scrollbar_event_after, win);
1851
bfeeb7f3 1852 const int orient = wxWindow::OrientFromScrollDir(
22c9b211 1853 win->ScrollDirFromRange(range));
74ab5f5b
VZ
1854 wxScrollWinEvent evt(wxEVT_SCROLLWIN_THUMBRELEASE,
1855 win->GetScrollPos(orient), orient);
1856 evt.SetEventObject(win);
1857 win->GTKProcessEvent(evt);
c918b2cd
PC
1858 }
1859}
c918b2cd 1860
cb43b372
RR
1861//-----------------------------------------------------------------------------
1862// "button_release_event" from scrollbar
1863//-----------------------------------------------------------------------------
1864
7f7beb1d 1865static gboolean
add7cadd 1866gtk_scrollbar_button_release_event(GtkRange* range, GdkEventButton*, wxWindow* win)
cb43b372 1867{
0a164d4c 1868 g_blockEventsOnScroll = false;
add7cadd
PC
1869 win->m_mouseButtonDown = false;
1870 // If thumb tracking
2a23d363 1871 if (win->m_isScrolling)
88413fec 1872 {
add7cadd 1873 win->m_isScrolling = false;
c918b2cd 1874 // Hook up handler to send thumb release event after this emission is finished.
8ea30e36
PC
1875 // To allow setting scroll position from event handler, sending event must
1876 // be deferred until after the GtkRange handler for this signal has run
c918b2cd 1877 g_signal_handlers_unblock_by_func(range, (void*)gtk_scrollbar_event_after, win);
88413fec
RR
1878 }
1879
add7cadd 1880 return false;
cb43b372 1881}
ca298c88 1882
a2053b27
RR
1883//-----------------------------------------------------------------------------
1884// "realize" from m_widget
1885//-----------------------------------------------------------------------------
1886
7f7beb1d 1887static void
08f53168 1888gtk_window_realized_callback(GtkWidget* widget, wxWindow* win)
a2053b27 1889{
a3c15d89 1890 if (win->m_imData)
2b5f62a0 1891 {
a3c15d89 1892 gtk_im_context_set_client_window( win->m_imData->context,
f089940f 1893 win->m_wxwindow ? win->GTKGetDrawingWindow() : widget->window);
2b5f62a0 1894 }
03647350 1895
8622e8cd
RR
1896 // We cannot set colours and fonts before the widget
1897 // been realized, so we do this directly after realization
1898 // or otherwise in idle time
1899
1900 if (win->m_needsStyleChange)
1901 {
1902 win->SetBackgroundStyle(win->GetBackgroundStyle());
1903 win->m_needsStyleChange = false;
1904 }
1905
3c679789
RR
1906 wxWindowCreateEvent event( win );
1907 event.SetEventObject( win );
97687291 1908 win->GTKProcessEvent( event );
9c76b9af
RR
1909
1910 win->GTKUpdateCursor(true, false);
a2053b27
RR
1911}
1912
b79395c5 1913//-----------------------------------------------------------------------------
cca410b3 1914// "size_allocate" from m_wxwindow or m_widget
b79395c5
RR
1915//-----------------------------------------------------------------------------
1916
cca410b3
PC
1917static void
1918size_allocate(GtkWidget*, GtkAllocation* alloc, wxWindow* win)
8f75cb6c 1919{
cca410b3
PC
1920 int w = alloc->width;
1921 int h = alloc->height;
1922 if (win->m_wxwindow)
5b8a521e 1923 {
08f53168
PC
1924 int border_x, border_y;
1925 WX_PIZZA(win->m_wxwindow)->get_border_widths(border_x, border_y);
1926 w -= 2 * border_x;
1927 h -= 2 * border_y;
cca410b3
PC
1928 if (w < 0) w = 0;
1929 if (h < 0) h = 0;
1930 }
1931 if (win->m_oldClientWidth != w || win->m_oldClientHeight != h)
1932 {
1933 win->m_oldClientWidth = w;
1934 win->m_oldClientHeight = h;
1935 // this callback can be connected to m_wxwindow,
1936 // so always get size from m_widget->allocation
1937 win->m_width = win->m_widget->allocation.width;
1938 win->m_height = win->m_widget->allocation.height;
1939 if (!win->m_nativeSizeEvent)
1940 {
1941 wxSizeEvent event(win->GetSize(), win->GetId());
1942 event.SetEventObject(win);
1943 win->GTKProcessEvent(event);
1944 }
5b8a521e 1945 }
8f75cb6c
RR
1946}
1947
7738af59 1948//-----------------------------------------------------------------------------
78cd9c69 1949// "grab_broken"
7738af59
VZ
1950//-----------------------------------------------------------------------------
1951
db885f2a
PC
1952#if GTK_CHECK_VERSION(2, 8, 0)
1953static gboolean
1954gtk_window_grab_broken( GtkWidget*,
7738af59
VZ
1955 GdkEventGrabBroken *event,
1956 wxWindow *win )
1957{
1958 // Mouse capture has been lost involuntarily, notify the application
db885f2a 1959 if(!event->keyboard && wxWindow::GetCapture() == win)
7738af59
VZ
1960 {
1961 wxMouseCaptureLostEvent evt( win->GetId() );
1962 evt.SetEventObject( win );
937013e0 1963 win->HandleWindowEvent( evt );
7738af59 1964 }
db885f2a 1965 return false;
7738af59 1966}
4b859ff4 1967#endif
7738af59 1968
013151c7
JS
1969//-----------------------------------------------------------------------------
1970// "style_set"
1971//-----------------------------------------------------------------------------
1972
1973static
1974void gtk_window_style_set_callback( GtkWidget *WXUNUSED(widget),
1975 GtkStyle *previous_style,
1976 wxWindow* win )
1977{
013151c7
JS
1978 if (win && previous_style)
1979 {
013151c7
JS
1980 wxSysColourChangedEvent event;
1981 event.SetEventObject(win);
1982
1983 win->GTKProcessEvent( event );
1984 }
1985}
1986
ef5c70f9
VZ
1987} // extern "C"
1988
1989// ----------------------------------------------------------------------------
1990// this wxWindowBase function is implemented here (in platform-specific file)
1991// because it is static and so couldn't be made virtual
1992// ----------------------------------------------------------------------------
1993
1994wxWindow *wxWindowBase::DoFindFocus()
1995{
22f43cb5 1996 wxWindowGTK *focus = gs_pendingFocus ? gs_pendingFocus : gs_currentFocus;
ef5c70f9 1997 // the cast is necessary when we compile in wxUniversal mode
5c33522f 1998 return static_cast<wxWindow*>(focus);
865bb325 1999}
63081513 2000
48200154 2001void wxWindowGTK::AddChildGTK(wxWindowGTK* child)
6ca41e57 2002{
9a33c3ef 2003 wxASSERT_MSG(m_wxwindow, "Cannot add a child to a window without a client area");
03647350 2004
9a33c3ef
FM
2005 // the window might have been scrolled already, we
2006 // have to adapt the position
48200154 2007 wxPizza* pizza = WX_PIZZA(m_wxwindow);
08f53168
PC
2008 child->m_x += pizza->m_scroll_x;
2009 child->m_y += pizza->m_scroll_y;
148cd9b6 2010
08f53168
PC
2011 gtk_widget_set_size_request(
2012 child->m_widget, child->m_width, child->m_height);
f089940f 2013 pizza->put(child->m_widget, child->m_x, child->m_y);
6ca41e57
RR
2014}
2015
bbe0af5b
RR
2016//-----------------------------------------------------------------------------
2017// global functions
2018//-----------------------------------------------------------------------------
2019
1e6feb95 2020wxWindow *wxGetActiveWindow()
bbe0af5b 2021{
6cad4f1b 2022 return wxWindow::FindFocus();
bbe0af5b
RR
2023}
2024
7dd40b6f
RD
2025
2026wxMouseState wxGetMouseState()
2027{
2028 wxMouseState ms;
2029
2030 gint x;
2031 gint y;
2032 GdkModifierType mask;
2033
2034 gdk_window_get_pointer(NULL, &x, &y, &mask);
2035
2036 ms.SetX(x);
2037 ms.SetY(y);
c4021a79
PC
2038 ms.SetLeftDown((mask & GDK_BUTTON1_MASK) != 0);
2039 ms.SetMiddleDown((mask & GDK_BUTTON2_MASK) != 0);
2040 ms.SetRightDown((mask & GDK_BUTTON3_MASK) != 0);
2041 ms.SetAux1Down((mask & GDK_BUTTON4_MASK) != 0);
2042 ms.SetAux2Down((mask & GDK_BUTTON5_MASK) != 0);
2043
2044 ms.SetControlDown((mask & GDK_CONTROL_MASK) != 0);
2045 ms.SetShiftDown((mask & GDK_SHIFT_MASK) != 0);
2046 ms.SetAltDown((mask & GDK_MOD1_MASK) != 0);
2047 ms.SetMetaDown((mask & GDK_META_MASK) != 0);
3d257b8d 2048
7dd40b6f
RD
2049 return ms;
2050}
3d257b8d 2051
c801d85f 2052//-----------------------------------------------------------------------------
1e6feb95 2053// wxWindowGTK
c801d85f
KB
2054//-----------------------------------------------------------------------------
2055
6522713c
VZ
2056// in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2057// method
1e6feb95 2058#ifdef __WXUNIVERSAL__
6522713c 2059 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK, wxWindowBase)
28953245 2060#endif // __WXUNIVERSAL__
c801d85f 2061
1e6feb95 2062void wxWindowGTK::Init()
c801d85f 2063{
f03fc89f 2064 // GTK specific
d3b9f782
VZ
2065 m_widget = NULL;
2066 m_wxwindow = NULL;
2067 m_focusWidget = NULL;
8bbe427f 2068
f03fc89f 2069 // position/size
a2053b27
RR
2070 m_x = 0;
2071 m_y = 0;
2072 m_width = 0;
e380f72b 2073 m_height = 0;
8bbe427f 2074
0a164d4c 2075 m_hasVMT = false;
02761f6c 2076
c6212a0c 2077 m_showOnIdle = false;
148cd9b6 2078
0a164d4c
WS
2079 m_noExpose = false;
2080 m_nativeSizeEvent = false;
94633ad9 2081
0a164d4c 2082 m_isScrolling = false;
add7cadd 2083 m_mouseButtonDown = false;
add7cadd 2084
22c9b211
VZ
2085 // initialize scrolling stuff
2086 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
2087 {
2088 m_scrollBar[dir] = NULL;
2089 m_scrollPos[dir] = 0;
22c9b211 2090 }
f03fc89f 2091
815ac4a7
VZ
2092 m_oldClientWidth =
2093 m_oldClientHeight = 0;
8bbe427f 2094
0a164d4c 2095 m_clipPaintRegion = false;
b6fa52db 2096
c7382f91
JS
2097 m_needsStyleChange = false;
2098
5e014a0c 2099 m_cursor = *wxSTANDARD_CURSOR;
2daa0ce9 2100
a3c15d89 2101 m_imData = NULL;
a589495e 2102 m_dirtyTabOrder = false;
362c6693 2103}
c801d85f 2104
1e6feb95 2105wxWindowGTK::wxWindowGTK()
68995f26
VZ
2106{
2107 Init();
2108}
2109
1e6feb95
VZ
2110wxWindowGTK::wxWindowGTK( wxWindow *parent,
2111 wxWindowID id,
2112 const wxPoint &pos,
2113 const wxSize &size,
2114 long style,
2115 const wxString &name )
6ca41e57 2116{
68995f26
VZ
2117 Init();
2118
e380f72b 2119 Create( parent, id, pos, size, style, name );
6ca41e57 2120}
8bbe427f 2121
1e6feb95
VZ
2122bool wxWindowGTK::Create( wxWindow *parent,
2123 wxWindowID id,
2124 const wxPoint &pos,
2125 const wxSize &size,
2126 long style,
2127 const wxString &name )
c801d85f 2128{
e5048854
JS
2129 // Get default border
2130 wxBorder border = GetBorder(style);
9800347d 2131
e5048854
JS
2132 style &= ~wxBORDER_MASK;
2133 style |= border;
ac9d38d8 2134
4dcaf11a
RR
2135 if (!PreCreation( parent, pos, size ) ||
2136 !CreateBase( parent, id, pos, size, style, wxDefaultValidator, name ))
2137 {
1e6feb95 2138 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
0a164d4c 2139 return false;
4dcaf11a 2140 }
47d67540 2141
7da4a9cf
RR
2142 // We should accept the native look
2143#if 0
2144 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
2145 scroll_class->scrollbar_spacing = 0;
2146#endif
2147
e5048854 2148
0f52f610 2149 m_wxwindow = wxPizza::New(m_windowStyle);
75f661bb
PC
2150#ifndef __WXUNIVERSAL__
2151 if (HasFlag(wxPizza::BORDER_STYLES))
2152 {
2153 g_signal_connect(m_wxwindow, "parent_set",
2154 G_CALLBACK(parent_set), this);
2155 }
2156#endif
3e09bcfd 2157 if (!HasFlag(wxHSCROLL) && !HasFlag(wxVSCROLL))
c9192212 2158 m_widget = m_wxwindow;
6552c7af
RR
2159 else
2160 {
d3b9f782 2161 m_widget = gtk_scrolled_window_new( NULL, NULL );
ff8bfdbb 2162
6552c7af 2163 GtkScrolledWindow *scrolledWindow = GTK_SCROLLED_WINDOW(m_widget);
47d67540 2164
6552c7af
RR
2165 // There is a conflict with default bindings at GTK+
2166 // level between scrolled windows and notebooks both of which want to use
2167 // Ctrl-PageUp/Down: scrolled windows for scrolling in the horizontal
2168 // direction and notebooks for changing pages -- we decide that if we don't
2169 // have wxHSCROLL style we can safely sacrifice horizontal scrolling if it
2170 // means we can get working keyboard navigation in notebooks
2171 if ( !HasFlag(wxHSCROLL) )
fd4081aa 2172 {
6552c7af
RR
2173 GtkBindingSet *
2174 bindings = gtk_binding_set_by_class(G_OBJECT_GET_CLASS(m_widget));
2175 if ( bindings )
2176 {
2177 gtk_binding_entry_remove(bindings, GDK_Page_Up, GDK_CONTROL_MASK);
2178 gtk_binding_entry_remove(bindings, GDK_Page_Down, GDK_CONTROL_MASK);
2179 }
fd4081aa 2180 }
fd4081aa 2181
6552c7af
RR
2182 if (HasFlag(wxALWAYS_SHOW_SB))
2183 {
2184 gtk_scrolled_window_set_policy( scrolledWindow, GTK_POLICY_ALWAYS, GTK_POLICY_ALWAYS );
d2288928 2185
6552c7af
RR
2186 scrolledWindow->hscrollbar_visible = TRUE;
2187 scrolledWindow->vscrollbar_visible = TRUE;
2188 }
2189 else
2190 {
2191 gtk_scrolled_window_set_policy( scrolledWindow, GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC );
2192 }
47d67540 2193
6552c7af
RR
2194 m_scrollBar[ScrollDir_Horz] = GTK_RANGE(scrolledWindow->hscrollbar);
2195 m_scrollBar[ScrollDir_Vert] = GTK_RANGE(scrolledWindow->vscrollbar);
2196 if (GetLayoutDirection() == wxLayout_RightToLeft)
2197 gtk_range_set_inverted( m_scrollBar[ScrollDir_Horz], TRUE );
47d67540 2198
6552c7af 2199 gtk_container_add( GTK_CONTAINER(m_widget), m_wxwindow );
4e5a4c69 2200
6552c7af
RR
2201 // connect various scroll-related events
2202 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
2203 {
2204 // these handlers block mouse events to any window during scrolling
2205 // such as motion events and prevent GTK and wxWidgets from fighting
2206 // over where the slider should be
2207 g_signal_connect(m_scrollBar[dir], "button_press_event",
22c9b211 2208 G_CALLBACK(gtk_scrollbar_button_press_event), this);
6552c7af 2209 g_signal_connect(m_scrollBar[dir], "button_release_event",
22c9b211
VZ
2210 G_CALLBACK(gtk_scrollbar_button_release_event), this);
2211
6552c7af 2212 gulong handler_id = g_signal_connect(m_scrollBar[dir], "event_after",
22c9b211 2213 G_CALLBACK(gtk_scrollbar_event_after), this);
6552c7af 2214 g_signal_handler_block(m_scrollBar[dir], handler_id);
22c9b211 2215
6552c7af 2216 // these handlers get notified when scrollbar slider moves
40e5ebbf 2217 g_signal_connect_after(m_scrollBar[dir], "value_changed",
22c9b211 2218 G_CALLBACK(gtk_scrollbar_value_changed), this);
6552c7af 2219 }
76ed8f8d 2220
6552c7af
RR
2221 gtk_widget_show( m_wxwindow );
2222 }
9ff9d30c 2223 g_object_ref(m_widget);
47d67540 2224
f03fc89f
VZ
2225 if (m_parent)
2226 m_parent->DoAddChild( this );
94633ad9 2227
76fcf0f2 2228 m_focusWidget = m_wxwindow;
8bbe427f 2229
29901843
VS
2230 SetCanFocus(AcceptsFocus());
2231
e380f72b 2232 PostCreation();
8bbe427f 2233
0a164d4c 2234 return true;
362c6693 2235}
c801d85f 2236
1e6feb95 2237wxWindowGTK::~wxWindowGTK()
c801d85f 2238{
7de59551
RD
2239 SendDestroyEvent();
2240
22f43cb5
VS
2241 if (gs_currentFocus == this)
2242 gs_currentFocus = NULL;
2243 if (gs_pendingFocus == this)
2244 gs_pendingFocus = NULL;
44cd54c2 2245
bd2e08d0
VS
2246 if ( gs_deferredFocusOut == this )
2247 gs_deferredFocusOut = NULL;
3e679f01 2248
0a164d4c 2249 m_hasVMT = false;
47d67540 2250
02c3e53b
JS
2251 // destroy children before destroying this window itself
2252 DestroyChildren();
2253
2254 // unhook focus handlers to prevent stray events being
2255 // propagated to this (soon to be) dead object
2256 if (m_focusWidget != NULL)
2257 {
9fa72bd2
MR
2258 g_signal_handlers_disconnect_by_func (m_focusWidget,
2259 (gpointer) gtk_window_focus_in_callback,
2260 this);
2261 g_signal_handlers_disconnect_by_func (m_focusWidget,
2262 (gpointer) gtk_window_focus_out_callback,
2263 this);
02c3e53b
JS
2264 }
2265
f03fc89f 2266 if (m_widget)
0a164d4c 2267 Show( false );
8bbe427f 2268
f6551618
MW
2269 // delete before the widgets to avoid a crash on solaris
2270 delete m_imData;
f6551618 2271
2f55b88c
PC
2272 // avoid problem with GTK+ 2.18 where a frozen window causes the whole
2273 // TLW to be frozen, and if the window is then destroyed, nothing ever
2274 // gets painted again
2275 if (IsFrozen())
2276 DoThaw();
2277
f03fc89f 2278 if (m_widget)
a2053b27 2279 {
9ff9d30c
PC
2280 // Note that gtk_widget_destroy() does not destroy the widget, it just
2281 // emits the "destroy" signal. The widget is not actually destroyed
2282 // until its reference count drops to zero.
2283 gtk_widget_destroy(m_widget);
2284 // Release our reference, should be the last one
2285 g_object_unref(m_widget);
2286 m_widget = NULL;
a2053b27 2287 }
9ff9d30c 2288 m_wxwindow = NULL;
362c6693 2289}
c801d85f 2290
1e6feb95 2291bool wxWindowGTK::PreCreation( wxWindowGTK *parent, const wxPoint &pos, const wxSize &size )
c801d85f 2292{
e8375af8
VZ
2293 if ( GTKNeedsParent() )
2294 {
2295 wxCHECK_MSG( parent, false, wxT("Must have non-NULL parent") );
2296 }
8bbe427f 2297
a7c26d10
RD
2298 // Use either the given size, or the default if -1 is given.
2299 // See wxWindowBase for these functions.
3013a903 2300 m_width = WidthDefault(size.x) ;
f03fc89f 2301 m_height = HeightDefault(size.y);
8bbe427f 2302
43a18898
RR
2303 m_x = (int)pos.x;
2304 m_y = (int)pos.y;
8bbe427f 2305
0a164d4c 2306 return true;
c801d85f
KB
2307}
2308
1e6feb95 2309void wxWindowGTK::PostCreation()
c801d85f 2310{
82b978d7
RD
2311 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
2312
43a18898
RR
2313 if (m_wxwindow)
2314 {
147bc491 2315 if (!m_noExpose)
b02da6b1 2316 {
77ffb593 2317 // these get reported to wxWidgets -> wxPaintEvent
1e6feb95 2318
9fa72bd2
MR
2319 g_signal_connect (m_wxwindow, "expose_event",
2320 G_CALLBACK (gtk_window_expose_callback), this);
147bc491 2321
847dfdb4 2322 if (GetLayoutDirection() == wxLayout_LeftToRight)
08f53168 2323 gtk_widget_set_redraw_on_allocate(m_wxwindow, HasFlag(wxFULL_REPAINT_ON_RESIZE));
93d23d8f 2324 }
2b5f62a0 2325
ed56a258
JS
2326 // Create input method handler
2327 m_imData = new wxGtkIMData;
2328
2b5f62a0 2329 // Cannot handle drawing preedited text yet
a3c15d89 2330 gtk_im_context_set_use_preedit( m_imData->context, FALSE );
2b5f62a0 2331
9fa72bd2 2332 g_signal_connect (m_imData->context, "commit",
da210120 2333 G_CALLBACK (gtk_wxwindow_commit_cb), this);
43a18898 2334 }
47d67540 2335
76fcf0f2 2336 // focus handling
63081513 2337
06fda9e8
RR
2338 if (!GTK_IS_WINDOW(m_widget))
2339 {
2340 if (m_focusWidget == NULL)
2341 m_focusWidget = m_widget;
0a164d4c 2342
4c20ee63
RR
2343 if (m_wxwindow)
2344 {
2345 g_signal_connect (m_focusWidget, "focus_in_event",
9fa72bd2 2346 G_CALLBACK (gtk_window_focus_in_callback), this);
4c20ee63 2347 g_signal_connect (m_focusWidget, "focus_out_event",
f1e57cb9 2348 G_CALLBACK (gtk_window_focus_out_callback), this);
4c20ee63
RR
2349 }
2350 else
2351 {
2352 g_signal_connect_after (m_focusWidget, "focus_in_event",
2353 G_CALLBACK (gtk_window_focus_in_callback), this);
2354 g_signal_connect_after (m_focusWidget, "focus_out_event",
2355 G_CALLBACK (gtk_window_focus_out_callback), this);
2356 }
06fda9e8 2357 }
76fcf0f2 2358
28e88942
VZ
2359 if ( !AcceptsFocusFromKeyboard() )
2360 {
80332672 2361 SetCanFocus(false);
28e88942
VZ
2362
2363 g_signal_connect(m_widget, "focus",
2364 G_CALLBACK(wx_window_focus_callback), this);
2365 }
2366
76fcf0f2 2367 // connect to the various key and mouse handlers
63081513 2368
a2053b27 2369 GtkWidget *connect_widget = GetConnectWidget();
f03fc89f 2370
a2053b27 2371 ConnectWidget( connect_widget );
47d67540 2372
63081513 2373 /* We cannot set colours, fonts and cursors before the widget has
a2053b27 2374 been realized, so we do this directly after realization */
9fa72bd2
MR
2375 g_signal_connect (connect_widget, "realize",
2376 G_CALLBACK (gtk_window_realized_callback), this);
2daa0ce9 2377
cca410b3
PC
2378 if (!IsTopLevel())
2379 {
2380 g_signal_connect(m_wxwindow ? m_wxwindow : m_widget, "size_allocate",
2381 G_CALLBACK(size_allocate), this);
2382 }
2383
db885f2a 2384#if GTK_CHECK_VERSION(2, 8, 0)
bb8afd65
VZ
2385 if ( gtk_check_version(2,8,0) == NULL )
2386 {
2387 // Make sure we can notify the app when mouse capture is lost
2388 if ( m_wxwindow )
4b859ff4 2389 {
4b859ff4 2390 g_signal_connect (m_wxwindow, "grab_broken_event",
7738af59 2391 G_CALLBACK (gtk_window_grab_broken), this);
4b859ff4 2392 }
7738af59 2393
bb8afd65 2394 if ( connect_widget != m_wxwindow )
4b859ff4 2395 {
4b859ff4 2396 g_signal_connect (connect_widget, "grab_broken_event",
7738af59 2397 G_CALLBACK (gtk_window_grab_broken), this);
4b859ff4 2398 }
63081513 2399 }
bb8afd65 2400#endif // GTK+ >= 2.8
2daa0ce9 2401
92153555 2402 if ( GTKShouldConnectSizeRequest() )
47c93b63
RR
2403 {
2404 // This is needed if we want to add our windows into native
024e9a4c 2405 // GTK controls, such as the toolbar. With this callback, the
47c93b63 2406 // toolbar gets to know the correct size (the one set by the
024e9a4c 2407 // programmer). Sadly, it misbehaves for wxComboBox.
9fa72bd2
MR
2408 g_signal_connect (m_widget, "size_request",
2409 G_CALLBACK (wxgtk_window_size_request_callback),
2410 this);
47c93b63 2411 }
1e6feb95 2412
40bab631
VS
2413 InheritAttributes();
2414
0a164d4c 2415 m_hasVMT = true;
e892f2fd 2416
978af864 2417 SetLayoutDirection(wxLayout_Default);
a433fbd5
VZ
2418
2419 // unless the window was created initially hidden (i.e. Hide() had been
2420 // called before Create()), we should show it at GTK+ level as well
2421 if ( IsShown() )
2422 gtk_widget_show( m_widget );
b4071e91
RR
2423}
2424
a0c8bb73
VZ
2425gulong wxWindowGTK::GTKConnectWidget(const char *signal, void (*callback)())
2426{
2427 return g_signal_connect(m_widget, signal, callback, this);
2428}
2429
1e6feb95 2430void wxWindowGTK::ConnectWidget( GtkWidget *widget )
b4071e91 2431{
9fa72bd2
MR
2432 g_signal_connect (widget, "key_press_event",
2433 G_CALLBACK (gtk_window_key_press_callback), this);
2434 g_signal_connect (widget, "key_release_event",
2435 G_CALLBACK (gtk_window_key_release_callback), this);
2436 g_signal_connect (widget, "button_press_event",
2437 G_CALLBACK (gtk_window_button_press_callback), this);
2438 g_signal_connect (widget, "button_release_event",
2439 G_CALLBACK (gtk_window_button_release_callback), this);
2440 g_signal_connect (widget, "motion_notify_event",
2441 G_CALLBACK (gtk_window_motion_notify_callback), this);
35510e48 2442
9fa72bd2 2443 g_signal_connect (widget, "scroll_event",
76e4be8e 2444 G_CALLBACK (window_scroll_event), this);
7f251559
RR
2445 if (m_scrollBar[ScrollDir_Horz])
2446 g_signal_connect (m_scrollBar[ScrollDir_Horz], "scroll_event",
2447 G_CALLBACK (window_scroll_event_hscrollbar), this);
2448 if (m_scrollBar[ScrollDir_Vert])
2449 g_signal_connect (m_scrollBar[ScrollDir_Vert], "scroll_event",
2450 G_CALLBACK (window_scroll_event), this);
35510e48 2451
9fa72bd2
MR
2452 g_signal_connect (widget, "popup_menu",
2453 G_CALLBACK (wxgtk_window_popup_menu_callback), this);
2454 g_signal_connect (widget, "enter_notify_event",
2455 G_CALLBACK (gtk_window_enter_callback), this);
2456 g_signal_connect (widget, "leave_notify_event",
2457 G_CALLBACK (gtk_window_leave_callback), this);
013151c7
JS
2458
2459 if (IsTopLevel() && m_wxwindow)
2460 g_signal_connect (m_wxwindow, "style_set",
2461 G_CALLBACK (gtk_window_style_set_callback), this);
362c6693 2462}
c801d85f 2463
1e6feb95 2464bool wxWindowGTK::Destroy()
c801d85f 2465{
82b978d7 2466 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
47d67540 2467
0a164d4c 2468 m_hasVMT = false;
c801d85f 2469
f03fc89f 2470 return wxWindowBase::Destroy();
362c6693 2471}
c801d85f 2472
1e6feb95 2473void wxWindowGTK::DoMoveWindow(int x, int y, int width, int height)
23efdd02 2474{
08f53168 2475 gtk_widget_set_size_request(m_widget, width, height);
03647350 2476
69597639 2477 // inform the parent to perform the move
03647350 2478 wxASSERT_MSG(m_parent && m_parent->m_wxwindow,
9a33c3ef 2479 "the parent window has no client area?");
08f53168 2480 WX_PIZZA(m_parent->m_wxwindow)->move(m_widget, x, y);
23efdd02 2481}
2daa0ce9 2482
82008f15
PC
2483void wxWindowGTK::ConstrainSize()
2484{
2485#ifdef __WXGPE__
2486 // GPE's window manager doesn't like size hints at all, esp. when the user
2487 // has to use the virtual keyboard, so don't constrain size there
2488 if (!IsTopLevel())
2489#endif
2490 {
2491 const wxSize minSize = GetMinSize();
2492 const wxSize maxSize = GetMaxSize();
2493 if (minSize.x > 0 && m_width < minSize.x) m_width = minSize.x;
2494 if (minSize.y > 0 && m_height < minSize.y) m_height = minSize.y;
2495 if (maxSize.x > 0 && m_width > maxSize.x) m_width = maxSize.x;
2496 if (maxSize.y > 0 && m_height > maxSize.y) m_height = maxSize.y;
2497 }
2498}
2499
1e6feb95 2500void wxWindowGTK::DoSetSize( int x, int y, int width, int height, int sizeFlags )
c801d85f 2501{
82b978d7 2502 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
1e6feb95 2503 wxASSERT_MSG( (m_parent != NULL), wxT("wxWindowGTK::SetSize requires parent.\n") );
8bbe427f 2504
b9f29261
VS
2505 int currentX, currentY;
2506 GetPosition(&currentX, &currentY);
443c834d 2507 if (x == -1 && !(sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
b9f29261 2508 x = currentX;
443c834d 2509 if (y == -1 && !(sizeFlags & wxSIZE_ALLOW_MINUS_ONE))
b9f29261 2510 y = currentY;
a200c35e
VS
2511 AdjustForParentClientOrigin(x, y, sizeFlags);
2512
fe39b16a
RR
2513 // calculate the best size if we should auto size the window
2514 if ( ((sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1) ||
2515 ((sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1) )
2516 {
2517 const wxSize sizeBest = GetBestSize();
2518 if ( (sizeFlags & wxSIZE_AUTO_WIDTH) && width == -1 )
2519 width = sizeBest.x;
2520 if ( (sizeFlags & wxSIZE_AUTO_HEIGHT) && height == -1 )
2521 height = sizeBest.y;
2522 }
2523
cca410b3 2524 const wxSize oldSize(m_width, m_height);
fe39b16a
RR
2525 if (width != -1)
2526 m_width = width;
2527 if (height != -1)
2528 m_height = height;
2529
cca410b3 2530 if (m_parent->m_wxwindow)
fb1585ae 2531 {
08f53168
PC
2532 wxPizza* pizza = WX_PIZZA(m_parent->m_wxwindow);
2533 m_x = x + pizza->m_scroll_x;
2534 m_y = y + pizza->m_scroll_y;
47d67540 2535
863e0817
RR
2536 int left_border = 0;
2537 int right_border = 0;
2538 int top_border = 0;
c50f1fb9 2539 int bottom_border = 0;
f03fc89f 2540
863e0817 2541 /* the default button has a border around it */
29f538ce 2542 if (GTK_WIDGET_CAN_DEFAULT(m_widget))
c50f1fb9 2543 {
f893066b
RR
2544 GtkBorder *default_border = NULL;
2545 gtk_widget_style_get( m_widget, "default_border", &default_border, NULL );
2546 if (default_border)
863e0817 2547 {
f893066b
RR
2548 left_border += default_border->left;
2549 right_border += default_border->right;
2550 top_border += default_border->top;
2551 bottom_border += default_border->bottom;
6cfdfed8 2552 gtk_border_free( default_border );
863e0817 2553 }
863e0817 2554 }
c50f1fb9 2555
18ac7bef
PC
2556 DoMoveWindow( m_x - left_border,
2557 m_y - top_border,
863e0817
RR
2558 m_width+left_border+right_border,
2559 m_height+top_border+bottom_border );
54517652 2560 }
148cd9b6 2561
cca410b3 2562 if (m_width != oldSize.x || m_height != oldSize.y)
5b8a521e 2563 {
cca410b3
PC
2564 // update these variables to keep size_allocate handler
2565 // from sending another size event for this change
5b8a521e 2566 GetClientSize( &m_oldClientWidth, &m_oldClientHeight );
6d693bb4 2567
cca410b3
PC
2568 gtk_widget_queue_resize(m_widget);
2569 if (!m_nativeSizeEvent)
2570 {
2571 wxSizeEvent event( wxSize(m_width,m_height), GetId() );
2572 event.SetEventObject( this );
937013e0 2573 HandleWindowEvent( event );
cca410b3 2574 }
03647350 2575 } else
e47e063a
RR
2576 if (sizeFlags & wxSIZE_FORCE_EVENT)
2577 {
2578 wxSizeEvent event( wxSize(m_width,m_height), GetId() );
2579 event.SetEventObject( this );
2580 HandleWindowEvent( event );
30760ce7 2581 }
362c6693 2582}
c801d85f 2583
71ead4bf 2584bool wxWindowGTK::GTKShowFromOnIdle()
9390a202 2585{
f46ad98f
RR
2586 if (IsShown() && m_showOnIdle && !GTK_WIDGET_VISIBLE (m_widget))
2587 {
2588 GtkAllocation alloc;
2589 alloc.x = m_x;
2590 alloc.y = m_y;
2591 alloc.width = m_width;
2592 alloc.height = m_height;
2593 gtk_widget_size_allocate( m_widget, &alloc );
2594 gtk_widget_show( m_widget );
2595 wxShowEvent eventShow(GetId(), true);
2596 eventShow.SetEventObject(this);
937013e0 2597 HandleWindowEvent(eventShow);
f46ad98f 2598 m_showOnIdle = false;
7317857d 2599 return true;
f46ad98f 2600 }
02761f6c 2601
7317857d
RR
2602 return false;
2603}
2604
2605void wxWindowGTK::OnInternalIdle()
2606{
bd2e08d0
VS
2607 if ( gs_deferredFocusOut )
2608 GTKHandleDeferredFocusOut();
2609
7317857d 2610 // Check if we have to show window now
71ead4bf 2611 if (GTKShowFromOnIdle()) return;
7317857d
RR
2612
2613 if ( m_dirtyTabOrder )
2614 {
2615 m_dirtyTabOrder = false;
2616 RealizeTabOrder();
2617 }
2618
9c61c5b0
VZ
2619 // Update style if the window was not yet realized when
2620 // SetBackgroundStyle() was called
7317857d
RR
2621 if (m_needsStyleChange)
2622 {
2623 SetBackgroundStyle(GetBackgroundStyle());
2624 m_needsStyleChange = false;
2625 }
02761f6c 2626
6a50a2c4 2627 wxWindowBase::OnInternalIdle();
9390a202
RR
2628}
2629
1e6feb95 2630void wxWindowGTK::DoGetSize( int *width, int *height ) const
c801d85f 2631{
82b978d7 2632 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 2633
fb1585ae
RR
2634 if (width) (*width) = m_width;
2635 if (height) (*height) = m_height;
362c6693 2636}
c801d85f 2637
1e6feb95 2638void wxWindowGTK::DoSetClientSize( int width, int height )
c801d85f 2639{
82b978d7
RD
2640 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
2641
949ff63e
PC
2642 const wxSize size = GetSize();
2643 const wxSize clientSize = GetClientSize();
2644 SetSize(width + (size.x - clientSize.x), height + (size.y - clientSize.y));
362c6693 2645}
c801d85f 2646
1e6feb95 2647void wxWindowGTK::DoGetClientSize( int *width, int *height ) const
c801d85f 2648{
82b978d7 2649 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 2650
09bf8378
PC
2651 int w = m_width;
2652 int h = m_height;
2653
fcdd5335 2654 if ( m_wxwindow )
c801d85f 2655 {
8466fc74 2656 // if window is scrollable, account for scrollbars
fcdd5335 2657 if ( GTK_IS_SCROLLED_WINDOW(m_widget) )
8466fc74 2658 {
fcdd5335
VZ
2659 GtkPolicyType policy[ScrollDir_Max];
2660 gtk_scrolled_window_get_policy(GTK_SCROLLED_WINDOW(m_widget),
2661 &policy[ScrollDir_Horz],
2662 &policy[ScrollDir_Vert]);
2663
2664 for ( int i = 0; i < ScrollDir_Max; i++ )
8466fc74 2665 {
fcdd5335
VZ
2666 // don't account for the scrollbars we don't have
2667 GtkRange * const range = m_scrollBar[i];
2668 if ( !range )
2669 continue;
2670
2671 // nor for the ones we have but don't current show
2672 switch ( policy[i] )
2673 {
2674 case GTK_POLICY_NEVER:
2675 // never shown so doesn't take any place
2676 continue;
2677
2678 case GTK_POLICY_ALWAYS:
2679 // no checks necessary
2680 break;
2681
2682 case GTK_POLICY_AUTOMATIC:
2683 // may be shown or not, check
2684 GtkAdjustment *adj = gtk_range_get_adjustment(range);
2685 if ( adj->upper <= adj->page_size )
2686 continue;
2687 }
2688
9800347d 2689 GtkScrolledWindowClass *scroll_class =
7da4a9cf
RR
2690 GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
2691
fcdd5335
VZ
2692 GtkRequisition req;
2693 gtk_widget_size_request(GTK_WIDGET(range), &req);
8466fc74 2694 if (i == ScrollDir_Horz)
7da4a9cf 2695 h -= req.height + scroll_class->scrollbar_spacing;
8466fc74 2696 else
7da4a9cf 2697 w -= req.width + scroll_class->scrollbar_spacing;
8466fc74
PC
2698 }
2699 }
09bf8378 2700
b3554952
VZ
2701 const wxSize sizeBorders = DoGetBorderSize();
2702 w -= sizeBorders.x;
2703 h -= sizeBorders.y;
9000c624 2704
69346023
PC
2705 if (w < 0)
2706 w = 0;
2707 if (h < 0)
2708 h = 0;
1ecc4d80 2709 }
1e6feb95 2710
09bf8378
PC
2711 if (width) *width = w;
2712 if (height) *height = h;
362c6693 2713}
c801d85f 2714
b3554952
VZ
2715wxSize wxWindowGTK::DoGetBorderSize() const
2716{
2717 if ( !m_wxwindow )
2718 return wxWindowBase::DoGetBorderSize();
2719
2720 int x, y;
2721 WX_PIZZA(m_wxwindow)->get_border_widths(x, y);
2722
2723 return 2*wxSize(x, y);
2724}
2725
1e6feb95 2726void wxWindowGTK::DoGetPosition( int *x, int *y ) const
c801d85f 2727{
82b978d7 2728 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 2729
bf0c00c6
RR
2730 int dx = 0;
2731 int dy = 0;
fd8a7b0b 2732 if (!IsTopLevel() && m_parent && m_parent->m_wxwindow)
bf0c00c6 2733 {
08f53168
PC
2734 wxPizza* pizza = WX_PIZZA(m_parent->m_wxwindow);
2735 dx = pizza->m_scroll_x;
2736 dy = pizza->m_scroll_y;
bf0c00c6 2737 }
94633ad9 2738
96c8547e
JS
2739 if (m_x == -1 && m_y == -1)
2740 {
d3b9f782 2741 GdkWindow *source = NULL;
96c8547e 2742 if (m_wxwindow)
08f53168 2743 source = m_wxwindow->window;
96c8547e
JS
2744 else
2745 source = m_widget->window;
2746
2747 if (source)
2748 {
2749 int org_x = 0;
2750 int org_y = 0;
2751 gdk_window_get_origin( source, &org_x, &org_y );
2752
bb5521a9
VZ
2753 if (m_parent)
2754 m_parent->ScreenToClient(&org_x, &org_y);
96c8547e 2755
5c33522f
VZ
2756 const_cast<wxWindowGTK*>(this)->m_x = org_x;
2757 const_cast<wxWindowGTK*>(this)->m_y = org_y;
bfeeb7f3 2758 }
96c8547e
JS
2759 }
2760
496beb3f
VS
2761 if (x) (*x) = m_x - dx;
2762 if (y) (*y) = m_y - dy;
362c6693 2763}
c801d85f 2764
1e6feb95 2765void wxWindowGTK::DoClientToScreen( int *x, int *y ) const
c801d85f 2766{
82b978d7 2767 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 2768
a2053b27
RR
2769 if (!m_widget->window) return;
2770
d3b9f782 2771 GdkWindow *source = NULL;
43a18898 2772 if (m_wxwindow)
08f53168 2773 source = m_wxwindow->window;
43a18898
RR
2774 else
2775 source = m_widget->window;
47d67540 2776
43a18898
RR
2777 int org_x = 0;
2778 int org_y = 0;
2779 gdk_window_get_origin( source, &org_x, &org_y );
c801d85f 2780
43a18898 2781 if (!m_wxwindow)
c801d85f 2782 {
43a18898
RR
2783 if (GTK_WIDGET_NO_WINDOW (m_widget))
2784 {
2785 org_x += m_widget->allocation.x;
2786 org_y += m_widget->allocation.y;
2787 }
362c6693 2788 }
47d67540 2789
49e74855
RR
2790
2791 if (x)
2792 {
2793 if (GetLayoutDirection() == wxLayout_RightToLeft)
2794 *x = (GetClientSize().x - *x) + org_x;
fcb29b23 2795 else
49e74855
RR
2796 *x += org_x;
2797 }
fcb29b23 2798
43a18898 2799 if (y) *y += org_y;
362c6693 2800}
c801d85f 2801
1e6feb95 2802void wxWindowGTK::DoScreenToClient( int *x, int *y ) const
c801d85f 2803{
82b978d7 2804 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 2805
a2053b27
RR
2806 if (!m_widget->window) return;
2807
d3b9f782 2808 GdkWindow *source = NULL;
1ecc4d80 2809 if (m_wxwindow)
08f53168 2810 source = m_wxwindow->window;
1ecc4d80
RR
2811 else
2812 source = m_widget->window;
47d67540 2813
1ecc4d80
RR
2814 int org_x = 0;
2815 int org_y = 0;
2816 gdk_window_get_origin( source, &org_x, &org_y );
c801d85f 2817
1ecc4d80 2818 if (!m_wxwindow)
c801d85f 2819 {
1ecc4d80
RR
2820 if (GTK_WIDGET_NO_WINDOW (m_widget))
2821 {
2822 org_x += m_widget->allocation.x;
2823 org_y += m_widget->allocation.y;
2824 }
362c6693 2825 }
47d67540 2826
49e74855
RR
2827 if (x)
2828 {
2829 if (GetLayoutDirection() == wxLayout_RightToLeft)
2830 *x = (GetClientSize().x - *x) - org_x;
fcb29b23 2831 else
49e74855
RR
2832 *x -= org_x;
2833 }
1ecc4d80 2834 if (y) *y -= org_y;
362c6693 2835}
c801d85f 2836
1e6feb95 2837bool wxWindowGTK::Show( bool show )
c801d85f 2838{
3cc57044 2839 if ( !wxWindowBase::Show(show) )
739730ca
RR
2840 {
2841 // nothing to do
0a164d4c 2842 return false;
739730ca 2843 }
8bbe427f 2844
3cc57044
VZ
2845 // notice that we may call Hide() before the window is created and this is
2846 // actually useful to create it hidden initially -- but we can't call
2847 // Show() before it is created
2848 if ( !m_widget )
f46ad98f 2849 {
3cc57044
VZ
2850 wxASSERT_MSG( !show, "can't show invalid window" );
2851 return true;
f46ad98f 2852 }
3cc57044
VZ
2853
2854 if ( show )
f46ad98f 2855 {
3cc57044
VZ
2856 if ( m_showOnIdle )
2857 {
2858 // defer until later
2859 return true;
2860 }
2861
2862 gtk_widget_show(m_widget);
f46ad98f 2863 }
3cc57044
VZ
2864 else // hide
2865 {
2866 gtk_widget_hide(m_widget);
2867 }
2868
2869 wxShowEvent eventShow(GetId(), show);
2870 eventShow.SetEventObject(this);
2871 HandleWindowEvent(eventShow);
2b5f62a0 2872
0a164d4c 2873 return true;
362c6693 2874}
c801d85f 2875
47a8a4d5 2876void wxWindowGTK::DoEnable( bool enable )
fdca68a6 2877{
47a8a4d5 2878 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
1ecc4d80 2879
f03fc89f 2880 gtk_widget_set_sensitive( m_widget, enable );
6552c7af 2881 if (m_wxwindow && (m_wxwindow != m_widget))
f03fc89f 2882 gtk_widget_set_sensitive( m_wxwindow, enable );
362c6693 2883}
c801d85f 2884
1e6feb95 2885int wxWindowGTK::GetCharHeight() const
2f2aa628 2886{
82b978d7 2887 wxCHECK_MSG( (m_widget != NULL), 12, wxT("invalid window") );
47d67540 2888
cc402e64
VZ
2889 wxFont font = GetFont();
2890 wxCHECK_MSG( font.Ok(), 12, wxT("invalid font") );
2f2aa628 2891
db885f2a 2892 PangoContext* context = gtk_widget_get_pango_context(m_widget);
bbd006c0
RR
2893
2894 if (!context)
2895 return 0;
2896
cc402e64 2897 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
bbd006c0
RR
2898 PangoLayout *layout = pango_layout_new(context);
2899 pango_layout_set_font_description(layout, desc);
2900 pango_layout_set_text(layout, "H", 1);
2901 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
2902
2903 PangoRectangle rect;
2904 pango_layout_line_get_extents(line, NULL, &rect);
2905
3fe39b0c 2906 g_object_unref (layout);
7de59551 2907
f69e2009 2908 return (int) PANGO_PIXELS(rect.height);
362c6693 2909}
c801d85f 2910
1e6feb95 2911int wxWindowGTK::GetCharWidth() const
c33c4050 2912{
82b978d7 2913 wxCHECK_MSG( (m_widget != NULL), 8, wxT("invalid window") );
47d67540 2914
cc402e64
VZ
2915 wxFont font = GetFont();
2916 wxCHECK_MSG( font.Ok(), 8, wxT("invalid font") );
47d67540 2917
db885f2a 2918 PangoContext* context = gtk_widget_get_pango_context(m_widget);
bbd006c0
RR
2919
2920 if (!context)
2921 return 0;
2922
cc402e64 2923 PangoFontDescription *desc = font.GetNativeFontInfo()->description;
bbd006c0
RR
2924 PangoLayout *layout = pango_layout_new(context);
2925 pango_layout_set_font_description(layout, desc);
95c430aa 2926 pango_layout_set_text(layout, "g", 1);
bbd006c0
RR
2927 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
2928
2929 PangoRectangle rect;
2930 pango_layout_line_get_extents(line, NULL, &rect);
2931
3fe39b0c 2932 g_object_unref (layout);
7de59551 2933
f69e2009 2934 return (int) PANGO_PIXELS(rect.width);
c33c4050
RR
2935}
2936
6de70470
VZ
2937void wxWindowGTK::DoGetTextExtent( const wxString& string,
2938 int *x,
2939 int *y,
2940 int *descent,
2941 int *externalLeading,
2942 const wxFont *theFont ) const
c33c4050 2943{
cc402e64 2944 wxFont fontToUse = theFont ? *theFont : GetFont();
47d67540 2945
223d09f6 2946 wxCHECK_RET( fontToUse.Ok(), wxT("invalid font") );
2b5f62a0 2947
0a164d4c 2948 if (string.empty())
48d011c8 2949 {
b15ed747
RR
2950 if (x) (*x) = 0;
2951 if (y) (*y) = 0;
48d011c8
RR
2952 return;
2953 }
47d67540 2954
48d011c8
RR
2955 PangoContext *context = NULL;
2956 if (m_widget)
2b5f62a0
VZ
2957 context = gtk_widget_get_pango_context( m_widget );
2958
48d011c8
RR
2959 if (!context)
2960 {
b15ed747
RR
2961 if (x) (*x) = 0;
2962 if (y) (*y) = 0;
48d011c8
RR
2963 return;
2964 }
2b5f62a0 2965
48d011c8
RR
2966 PangoFontDescription *desc = fontToUse.GetNativeFontInfo()->description;
2967 PangoLayout *layout = pango_layout_new(context);
2968 pango_layout_set_font_description(layout, desc);
2969 {
a3669332
VZ
2970 const wxCharBuffer data = wxGTK_CONV( string );
2971 if ( data )
2972 pango_layout_set_text(layout, data, strlen(data));
48d011c8 2973 }
2b5f62a0 2974
48d011c8 2975 PangoRectangle rect;
fd43b1b3 2976 pango_layout_get_extents(layout, NULL, &rect);
2b5f62a0 2977
f69e2009
VS
2978 if (x) (*x) = (wxCoord) PANGO_PIXELS(rect.width);
2979 if (y) (*y) = (wxCoord) PANGO_PIXELS(rect.height);
48d011c8
RR
2980 if (descent)
2981 {
f69e2009
VS
2982 PangoLayoutIter *iter = pango_layout_get_iter(layout);
2983 int baseline = pango_layout_iter_get_baseline(iter);
2984 pango_layout_iter_free(iter);
2985 *descent = *y - PANGO_PIXELS(baseline);
48d011c8
RR
2986 }
2987 if (externalLeading) (*externalLeading) = 0; // ??
2b5f62a0 2988
3fe39b0c 2989 g_object_unref (layout);
c33c4050
RR
2990}
2991
36a845fe
RR
2992void wxWindowGTK::GTKDisableFocusOutEvent()
2993{
2994 g_signal_handlers_block_by_func( m_focusWidget,
2995 (gpointer) gtk_window_focus_out_callback, this);
2996}
2997
2998void wxWindowGTK::GTKEnableFocusOutEvent()
2999{
3000 g_signal_handlers_unblock_by_func( m_focusWidget,
3001 (gpointer) gtk_window_focus_out_callback, this);
3002}
bd2e08d0
VS
3003
3004bool wxWindowGTK::GTKHandleFocusIn()
ef5c70f9 3005{
bd2e08d0
VS
3006 // Disable default focus handling for custom windows since the default GTK+
3007 // handler issues a repaint
3008 const bool retval = m_wxwindow ? true : false;
3009
3010
3011 // NB: if there's still unprocessed deferred focus-out event (see
3012 // GTKHandleFocusOut() for explanation), we need to process it first so
3013 // that the order of focus events -- focus-out first, then focus-in
3014 // elsewhere -- is preserved
3015 if ( gs_deferredFocusOut )
ef5c70f9 3016 {
bd2e08d0
VS
3017 if ( GTKNeedsToFilterSameWindowFocus() &&
3018 gs_deferredFocusOut == this )
ef5c70f9 3019 {
bd2e08d0
VS
3020 // GTK+ focus changed from this wxWindow back to itself, so don't
3021 // emit any events at all
3022 wxLogTrace(TRACE_FOCUS,
3023 "filtered out spurious focus change within %s(%p, %s)",
3024 GetClassInfo()->GetClassName(), this, GetLabel());
3025 gs_deferredFocusOut = NULL;
3026 return retval;
ef5c70f9 3027 }
bd2e08d0
VS
3028
3029 // otherwise we need to send focus-out first
3030 wxASSERT_MSG ( gs_deferredFocusOut != this,
3031 "GTKHandleFocusIn(GTKFocus_Normal) called even though focus changed back to itself - derived class should handle this" );
3032 GTKHandleDeferredFocusOut();
ef5c70f9
VZ
3033 }
3034
bd2e08d0
VS
3035
3036 wxLogTrace(TRACE_FOCUS,
3037 "handling focus_in event for %s(%p, %s)",
3038 GetClassInfo()->GetClassName(), this, GetLabel());
3039
3040 if (m_imData)
3041 gtk_im_context_focus_in(m_imData->context);
3042
22f43cb5
VS
3043 gs_currentFocus = this;
3044 gs_pendingFocus = NULL;
bd2e08d0
VS
3045
3046#if wxUSE_CARET
3047 // caret needs to be informed about focus change
3048 wxCaret *caret = GetCaret();
3049 if ( caret )
3050 {
3051 caret->OnSetFocus();
3052 }
3053#endif // wxUSE_CARET
3054
3055 // Notify the parent keeping track of focus for the kbd navigation
3056 // purposes that we got it.
b22bb1a0 3057 wxChildFocusEvent eventChildFocus(static_cast<wxWindow*>(this));
bd2e08d0
VS
3058 GTKProcessEvent(eventChildFocus);
3059
3060 wxFocusEvent eventFocus(wxEVT_SET_FOCUS, GetId());
3061 eventFocus.SetEventObject(this);
3062 GTKProcessEvent(eventFocus);
3063
3064 return retval;
ef5c70f9
VZ
3065}
3066
bd2e08d0 3067bool wxWindowGTK::GTKHandleFocusOut()
7cec1c9e 3068{
bd2e08d0
VS
3069 // Disable default focus handling for custom windows since the default GTK+
3070 // handler issues a repaint
3071 const bool retval = m_wxwindow ? true : false;
3072
3073
3074 // NB: If a control is composed of several GtkWidgets and when focus
3075 // changes from one of them to another within the same wxWindow, we get
3076 // a focus-out event followed by focus-in for another GtkWidget owned
3077 // by the same wx control. We don't want to generate two spurious
3078 // wxEVT_SET_FOCUS events in this case, so we defer sending wx events
3079 // from GTKHandleFocusOut() until we know for sure it's not coming back
3080 // (i.e. in GTKHandleFocusIn() or at idle time).
3081 if ( GTKNeedsToFilterSameWindowFocus() )
7cec1c9e 3082 {
bd2e08d0
VS
3083 wxASSERT_MSG( gs_deferredFocusOut == NULL,
3084 "deferred focus out event already pending" );
3085 wxLogTrace(TRACE_FOCUS,
3086 "deferring focus_out event for %s(%p, %s)",
3087 GetClassInfo()->GetClassName(), this, GetLabel());
3088 gs_deferredFocusOut = this;
3089 return retval;
7cec1c9e
RR
3090 }
3091
bd2e08d0
VS
3092 GTKHandleFocusOutNoDeferring();
3093
3094 return retval;
3095}
3096
3097void wxWindowGTK::GTKHandleFocusOutNoDeferring()
3098{
3099 wxLogTrace(TRACE_FOCUS,
3100 "handling focus_out event for %s(%p, %s)",
3101 GetClassInfo()->GetClassName(), this, GetLabel());
3102
3103 if (m_imData)
3104 gtk_im_context_focus_out(m_imData->context);
3105
22f43cb5 3106 if ( gs_currentFocus != this )
7cec1c9e 3107 {
22f43cb5 3108 // Something is terribly wrong, gs_currentFocus is out of sync with the
bd2e08d0
VS
3109 // real focus. We will reset it to NULL anyway, because after this
3110 // focus-out event is handled, one of the following with happen:
3111 //
3112 // * either focus will go out of the app altogether, in which case
22f43cb5 3113 // gs_currentFocus _should_ be NULL
bd2e08d0
VS
3114 //
3115 // * or it goes to another control, in which case focus-in event will
22f43cb5 3116 // follow immediately and it will set gs_currentFocus to the right
bd2e08d0
VS
3117 // value
3118 wxLogDebug("window %s(%p, %s) lost focus even though it didn't have it",
3119 GetClassInfo()->GetClassName(), this, GetLabel());
3120 }
22f43cb5 3121 gs_currentFocus = NULL;
db885f2a 3122
bd2e08d0
VS
3123#if wxUSE_CARET
3124 // caret needs to be informed about focus change
3125 wxCaret *caret = GetCaret();
3126 if ( caret )
3127 {
3128 caret->OnKillFocus();
354aa1e3 3129 }
bd2e08d0
VS
3130#endif // wxUSE_CARET
3131
3132 wxFocusEvent event( wxEVT_KILL_FOCUS, GetId() );
3133 event.SetEventObject( this );
3134 GTKProcessEvent( event );
3135}
3136
3137/*static*/
3138void wxWindowGTK::GTKHandleDeferredFocusOut()
3139{
3140 // NB: See GTKHandleFocusOut() for explanation. This function is called
3141 // from either GTKHandleFocusIn() or OnInternalIdle() to process
3142 // deferred event.
3143 if ( gs_deferredFocusOut )
c801d85f 3144 {
bd2e08d0
VS
3145 wxWindowGTK *win = gs_deferredFocusOut;
3146 gs_deferredFocusOut = NULL;
db885f2a 3147
bd2e08d0
VS
3148 wxLogTrace(TRACE_FOCUS,
3149 "processing deferred focus_out event for %s(%p, %s)",
3150 win->GetClassInfo()->GetClassName(), win, win->GetLabel());
fcb29b23 3151
bd2e08d0
VS
3152 win->GTKHandleFocusOutNoDeferring();
3153 }
3154}
0a164d4c 3155
bd2e08d0
VS
3156void wxWindowGTK::SetFocus()
3157{
3158 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
6aeb6f2a 3159
bd2e08d0
VS
3160 // Setting "physical" focus is not immediate in GTK+ and while
3161 // gtk_widget_is_focus ("determines if the widget is the focus widget
3162 // within its toplevel", i.e. returns true for one widget per TLW, not
3163 // globally) returns true immediately after grabbing focus,
3164 // GTK_WIDGET_HAS_FOCUS (which returns true only for the one widget that
3165 // has focus at the moment) takes affect only after the window is shown
3166 // (if it was hidden at the moment of the call) or at the next event loop
3167 // iteration.
3168 //
3169 // Because we want to FindFocus() call immediately following
3170 // foo->SetFocus() to return foo, we have to keep track of "pending" focus
3171 // ourselves.
22f43cb5 3172 gs_pendingFocus = this;
bd2e08d0
VS
3173
3174 GtkWidget *widget = m_wxwindow ? m_wxwindow : m_focusWidget;
3175
3176 if ( GTK_IS_CONTAINER(widget) &&
3177 !GTK_WIDGET_CAN_FOCUS(widget) )
3178 {
3179 wxLogTrace(TRACE_FOCUS,
9a83f860 3180 wxT("Setting focus to a child of %s(%p, %s)"),
bd2e08d0
VS
3181 GetClassInfo()->GetClassName(), this, GetLabel().c_str());
3182 gtk_widget_child_focus(widget, GTK_DIR_TAB_FORWARD);
3183 }
3184 else
3185 {
3186 wxLogTrace(TRACE_FOCUS,
9a83f860 3187 wxT("Setting focus to %s(%p, %s)"),
bd2e08d0
VS
3188 GetClassInfo()->GetClassName(), this, GetLabel().c_str());
3189 gtk_widget_grab_focus(widget);
362c6693 3190 }
362c6693 3191}
c801d85f 3192
80332672
VZ
3193void wxWindowGTK::SetCanFocus(bool canFocus)
3194{
3195 if ( canFocus )
3196 GTK_WIDGET_SET_FLAGS(m_widget, GTK_CAN_FOCUS);
3197 else
3198 GTK_WIDGET_UNSET_FLAGS(m_widget, GTK_CAN_FOCUS);
3199
3200 if ( m_wxwindow && (m_widget != m_wxwindow) )
3201 {
3202 if ( canFocus )
3203 GTK_WIDGET_SET_FLAGS(m_wxwindow, GTK_CAN_FOCUS);
3204 else
3205 GTK_WIDGET_UNSET_FLAGS(m_wxwindow, GTK_CAN_FOCUS);
3206 }
3207}
3208
1e6feb95 3209bool wxWindowGTK::Reparent( wxWindowBase *newParentBase )
463c1fa1 3210{
0a164d4c 3211 wxCHECK_MSG( (m_widget != NULL), false, wxT("invalid window") );
c50f1fb9 3212
1e6feb95
VZ
3213 wxWindowGTK *oldParent = m_parent,
3214 *newParent = (wxWindowGTK *)newParentBase;
a2053b27 3215
5fd11f09
RR
3216 wxASSERT( GTK_IS_WIDGET(m_widget) );
3217
f03fc89f 3218 if ( !wxWindowBase::Reparent(newParent) )
0a164d4c 3219 return false;
8bbe427f 3220
5fd11f09
RR
3221 wxASSERT( GTK_IS_WIDGET(m_widget) );
3222
8ce63e9d 3223 if (oldParent)
3017f78d 3224 gtk_container_remove( GTK_CONTAINER(m_widget->parent), m_widget );
c50f1fb9 3225
5fd11f09
RR
3226 wxASSERT( GTK_IS_WIDGET(m_widget) );
3227
8ce63e9d
RR
3228 if (newParent)
3229 {
f46ad98f
RR
3230 if (GTK_WIDGET_VISIBLE (newParent->m_widget))
3231 {
3232 m_showOnIdle = true;
3233 gtk_widget_hide( m_widget );
3234 }
8ce63e9d 3235 /* insert GTK representation */
48200154 3236 newParent->AddChildGTK(this);
8ce63e9d 3237 }
c50f1fb9 3238
978af864 3239 SetLayoutDirection(wxLayout_Default);
148cd9b6 3240
0a164d4c 3241 return true;
362c6693 3242}
c801d85f 3243
1e6feb95 3244void wxWindowGTK::DoAddChild(wxWindowGTK *child)
ddb6bc71 3245{
223d09f6 3246 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
223d09f6 3247 wxASSERT_MSG( (child != NULL), wxT("invalid child window") );
ddb6bc71 3248
ddb6bc71
RR
3249 /* add to list */
3250 AddChild( child );
c50f1fb9 3251
ddb6bc71 3252 /* insert GTK representation */
48200154 3253 AddChildGTK(child);
ddb6bc71
RR
3254}
3255
a589495e
VS
3256void wxWindowGTK::AddChild(wxWindowBase *child)
3257{
3258 wxWindowBase::AddChild(child);
3259 m_dirtyTabOrder = true;
a1abca32 3260 wxTheApp->WakeUpIdle();
a589495e
VS
3261}
3262
3263void wxWindowGTK::RemoveChild(wxWindowBase *child)
3264{
3265 wxWindowBase::RemoveChild(child);
3266 m_dirtyTabOrder = true;
a1abca32 3267 wxTheApp->WakeUpIdle();
a589495e 3268}
0a164d4c 3269
978af864
VZ
3270/* static */
3271wxLayoutDirection wxWindowGTK::GTKGetLayout(GtkWidget *widget)
3272{
10bd1f7d 3273 return gtk_widget_get_direction(widget) == GTK_TEXT_DIR_RTL
978af864
VZ
3274 ? wxLayout_RightToLeft
3275 : wxLayout_LeftToRight;
3276}
3277
3278/* static */
3279void wxWindowGTK::GTKSetLayout(GtkWidget *widget, wxLayoutDirection dir)
3280{
9a83f860 3281 wxASSERT_MSG( dir != wxLayout_Default, wxT("invalid layout direction") );
978af864 3282
10bd1f7d 3283 gtk_widget_set_direction(widget,
978af864
VZ
3284 dir == wxLayout_RightToLeft ? GTK_TEXT_DIR_RTL
3285 : GTK_TEXT_DIR_LTR);
3286}
3287
3288wxLayoutDirection wxWindowGTK::GetLayoutDirection() const
3289{
3290 return GTKGetLayout(m_widget);
3291}
3292
3293void wxWindowGTK::SetLayoutDirection(wxLayoutDirection dir)
3294{
3295 if ( dir == wxLayout_Default )
3296 {
3297 const wxWindow *const parent = GetParent();
3298 if ( parent )
3299 {
3300 // inherit layout from parent.
3301 dir = parent->GetLayoutDirection();
3302 }
3303 else // no parent, use global default layout
3304 {
3305 dir = wxTheApp->GetLayoutDirection();
3306 }
3307 }
3308
3309 if ( dir == wxLayout_Default )
3310 return;
3311
3312 GTKSetLayout(m_widget, dir);
fcb29b23 3313
6552c7af 3314 if (m_wxwindow && (m_wxwindow != m_widget))
69597639
RR
3315 GTKSetLayout(m_wxwindow, dir);
3316}
3317
3318wxCoord
3319wxWindowGTK::AdjustForLayoutDirection(wxCoord x,
3320 wxCoord WXUNUSED(width),
3321 wxCoord WXUNUSED(widthTotal)) const
3322{
08f53168 3323 // We now mirror the coordinates of RTL windows in wxPizza
69597639 3324 return x;
978af864
VZ
3325}
3326
915bd4e4 3327void wxWindowGTK::DoMoveInTabOrder(wxWindow *win, WindowOrder move)
a589495e
VS
3328{
3329 wxWindowBase::DoMoveInTabOrder(win, move);
3330 m_dirtyTabOrder = true;
a1abca32 3331 wxTheApp->WakeUpIdle();
a589495e
VS
3332}
3333
5644933f
VZ
3334bool wxWindowGTK::DoNavigateIn(int flags)
3335{
3336 if ( flags & wxNavigationKeyEvent::WinChange )
3337 {
9a83f860 3338 wxFAIL_MSG( wxT("not implemented") );
5644933f
VZ
3339
3340 return false;
3341 }
3342 else // navigate inside the container
3343 {
7a3ba5af 3344 wxWindow *parent = wxGetTopLevelParent((wxWindow *)this);
9a83f860 3345 wxCHECK_MSG( parent, false, wxT("every window must have a TLW parent") );
5644933f
VZ
3346
3347 GtkDirectionType dir;
3348 dir = flags & wxNavigationKeyEvent::IsForward ? GTK_DIR_TAB_FORWARD
3349 : GTK_DIR_TAB_BACKWARD;
3350
3351 gboolean rc;
3352 g_signal_emit_by_name(parent->m_widget, "focus", dir, &rc);
3353
3354 return rc == TRUE;
3355 }
3356}
3357
2e1f5012
VZ
3358bool wxWindowGTK::GTKWidgetNeedsMnemonic() const
3359{
3360 // none needed by default
3361 return false;
3362}
3363
3364void wxWindowGTK::GTKWidgetDoSetMnemonic(GtkWidget* WXUNUSED(w))
3365{
3366 // nothing to do by default since none is needed
3367}
3368
a589495e
VS
3369void wxWindowGTK::RealizeTabOrder()
3370{
3371 if (m_wxwindow)
3372 {
12848fda 3373 if ( !m_children.empty() )
a589495e 3374 {
259858fc 3375 // we don't only construct the correct focus chain but also use
2e1f5012
VZ
3376 // this opportunity to update the mnemonic widgets for the widgets
3377 // that need them
259858fc 3378
a589495e 3379 GList *chain = NULL;
2e1f5012 3380 wxWindowGTK* mnemonicWindow = NULL;
0a164d4c 3381
12848fda
VZ
3382 for ( wxWindowList::const_iterator i = m_children.begin();
3383 i != m_children.end();
3384 ++i )
a589495e 3385 {
259858fc 3386 wxWindowGTK *win = *i;
2e1f5012
VZ
3387
3388 if ( mnemonicWindow )
259858fc
VZ
3389 {
3390 if ( win->AcceptsFocusFromKeyboard() )
3391 {
2e1f5012
VZ
3392 // wxComboBox et al. needs to focus on on a different
3393 // widget than m_widget, so if the main widget isn't
3394 // focusable try the connect widget
3395 GtkWidget* w = win->m_widget;
3396 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3397 {
3398 w = win->GetConnectWidget();
3399 if ( !GTK_WIDGET_CAN_FOCUS(w) )
3400 w = NULL;
3401 }
3402
3403 if ( w )
3404 {
3405 mnemonicWindow->GTKWidgetDoSetMnemonic(w);
3406 mnemonicWindow = NULL;
3407 }
259858fc
VZ
3408 }
3409 }
2e1f5012 3410 else if ( win->GTKWidgetNeedsMnemonic() )
259858fc 3411 {
2e1f5012 3412 mnemonicWindow = win;
259858fc 3413 }
259858fc
VZ
3414
3415 chain = g_list_prepend(chain, win->m_widget);
a589495e 3416 }
0a164d4c 3417
a589495e 3418 chain = g_list_reverse(chain);
0a164d4c 3419
a589495e
VS
3420 gtk_container_set_focus_chain(GTK_CONTAINER(m_wxwindow), chain);
3421 g_list_free(chain);
3422 }
12848fda 3423 else // no children
a589495e
VS
3424 {
3425 gtk_container_unset_focus_chain(GTK_CONTAINER(m_wxwindow));
3426 }
3427 }
a589495e
VS
3428}
3429
1e6feb95 3430void wxWindowGTK::Raise()
362c6693 3431{
82b978d7
RD
3432 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3433
fdfb8475
RR
3434 if (m_wxwindow && m_wxwindow->window)
3435 {
3436 gdk_window_raise( m_wxwindow->window );
3437 }
0a164d4c 3438 else if (m_widget->window)
fdfb8475
RR
3439 {
3440 gdk_window_raise( m_widget->window );
3441 }
362c6693
RR
3442}
3443
1e6feb95 3444void wxWindowGTK::Lower()
362c6693 3445{
82b978d7
RD
3446 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3447
fdfb8475
RR
3448 if (m_wxwindow && m_wxwindow->window)
3449 {
3450 gdk_window_lower( m_wxwindow->window );
3451 }
0a164d4c 3452 else if (m_widget->window)
fdfb8475
RR
3453 {
3454 gdk_window_lower( m_widget->window );
3455 }
362c6693 3456}
c801d85f 3457
1e6feb95 3458bool wxWindowGTK::SetCursor( const wxCursor &cursor )
86b29a61 3459{
fe282c08 3460 if ( !wxWindowBase::SetCursor(cursor.Ok() ? cursor : *wxSTANDARD_CURSOR) )
2f262021 3461 return false;
f6bcfd97 3462
2f262021 3463 GTKUpdateCursor();
1e6feb95 3464
2f262021 3465 return true;
ef5c70f9
VZ
3466}
3467
c2246a38 3468void wxWindowGTK::GTKUpdateCursor(bool update_self /*=true*/, bool recurse /*=true*/)
ef5c70f9 3469{
c2246a38 3470 if (update_self)
ef5c70f9 3471 {
c2246a38
RR
3472 wxCursor cursor(g_globalCursor.Ok() ? g_globalCursor : GetCursor());
3473 if ( cursor.Ok() )
ef5c70f9 3474 {
9c76b9af
RR
3475 wxArrayGdkWindows windowsThis;
3476 GdkWindow* window = GTKGetWindow(windowsThis);
3477 if (window)
3478 gdk_window_set_cursor( window, cursor.GetCursor() );
3479 else
ef5c70f9 3480 {
9c76b9af
RR
3481 const size_t count = windowsThis.size();
3482 for ( size_t n = 0; n < count; n++ )
5dd7eef7 3483 {
9c76b9af
RR
3484 GdkWindow *win = windowsThis[n];
3485 // It can be zero if the window has not been realized yet.
3486 if ( win )
c2246a38 3487 {
c2246a38
RR
3488 gdk_window_set_cursor(win, cursor.GetCursor());
3489 }
5dd7eef7 3490 }
c2246a38
RR
3491 }
3492 }
3493 }
3494
3495 if (recurse)
3496 {
3497 for (wxWindowList::iterator it = GetChildren().begin(); it != GetChildren().end(); ++it)
3498 {
3499 (*it)->GTKUpdateCursor( true );
ef5c70f9
VZ
3500 }
3501 }
362c6693 3502}
c801d85f 3503
1e6feb95 3504void wxWindowGTK::WarpPointer( int x, int y )
4f22cf8d 3505{
82b978d7
RD
3506 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3507
3bcc8d15
RR
3508 // We provide this function ourselves as it is
3509 // missing in GDK (top of this file).
148cd9b6 3510
d3b9f782 3511 GdkWindow *window = NULL;
ed673c6a 3512 if (m_wxwindow)
08f53168 3513 window = m_wxwindow->window;
ed673c6a
RR
3514 else
3515 window = GetConnectWidget()->window;
148cd9b6 3516
ed673c6a
RR
3517 if (window)
3518 gdk_window_warp_pointer( window, x, y );
4f22cf8d
RR
3519}
3520
22c9b211 3521wxWindowGTK::ScrollDir wxWindowGTK::ScrollDirFromRange(GtkRange *range) const
0c131a5a 3522{
22c9b211
VZ
3523 // find the scrollbar which generated the event
3524 for ( int dir = 0; dir < ScrollDir_Max; dir++ )
0c131a5a 3525 {
22c9b211
VZ
3526 if ( range == m_scrollBar[dir] )
3527 return (ScrollDir)dir;
0c131a5a 3528 }
22c9b211 3529
9a83f860 3530 wxFAIL_MSG( wxT("event from unknown scrollbar received") );
22c9b211
VZ
3531
3532 return ScrollDir_Max;
0c131a5a
VZ
3533}
3534
22c9b211 3535bool wxWindowGTK::DoScrollByUnits(ScrollDir dir, ScrollUnit unit, int units)
0c131a5a 3536{
add7cadd 3537 bool changed = false;
22c9b211
VZ
3538 GtkRange* range = m_scrollBar[dir];
3539 if ( range && units )
add7cadd
PC
3540 {
3541 GtkAdjustment* adj = range->adjustment;
22c9b211
VZ
3542 gdouble inc = unit == ScrollUnit_Line ? adj->step_increment
3543 : adj->page_increment;
3544
3545 const int posOld = int(adj->value + 0.5);
3546 gtk_range_set_value(range, posOld + units*inc);
3547
3548 changed = int(adj->value + 0.5) != posOld;
add7cadd 3549 }
22c9b211 3550
add7cadd 3551 return changed;
0c131a5a 3552}
3013a903 3553
22c9b211
VZ
3554bool wxWindowGTK::ScrollLines(int lines)
3555{
3556 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Line, lines);
3557}
3558
3559bool wxWindowGTK::ScrollPages(int pages)
3560{
3561 return DoScrollByUnits(ScrollDir_Vert, ScrollUnit_Page, pages);
3562}
3563
e4161a2a
VZ
3564void wxWindowGTK::Refresh(bool WXUNUSED(eraseBackground),
3565 const wxRect *rect)
c801d85f 3566{
b4857be3
VZ
3567 if ( !m_widget )
3568 {
3569 // it is valid to call Refresh() for a window which hasn't been created
3570 // yet, it simply doesn't do anything in this case
3571 return;
3572 }
08bbdb08 3573
d0285203 3574 if (!m_wxwindow)
ba86da30 3575 {
d0285203
RR
3576 if (rect)
3577 gtk_widget_queue_draw_area( m_widget, rect->x, rect->y, rect->width, rect->height );
3578 else
3579 gtk_widget_queue_draw( m_widget );
ba86da30 3580 }
6cab4fca 3581 else
4e5a4c69 3582 {
d0285203
RR
3583 // Just return if the widget or one of its ancestors isn't mapped
3584 GtkWidget *w;
3585 for (w = m_wxwindow; w != NULL; w = w->parent)
3586 if (!GTK_WIDGET_MAPPED (w))
3587 return;
3588
f089940f 3589 GdkWindow* window = GTKGetDrawingWindow();
d0285203
RR
3590 if (rect)
3591 {
3592 int x = rect->x;
3593 if (GetLayoutDirection() == wxLayout_RightToLeft)
3594 x = GetClientSize().x - x - rect->width;
3595 GdkRectangle r;
3596 r.x = rect->x;
3597 r.y = rect->y;
3598 r.width = rect->width;
3599 r.height = rect->height;
f089940f 3600 gdk_window_invalidate_rect(window, &r, true);
d0285203
RR
3601 }
3602 else
f089940f 3603 gdk_window_invalidate_rect(window, NULL, true);
4e5a4c69 3604 }
362c6693 3605}
c801d85f 3606
beab25bd 3607void wxWindowGTK::Update()
010afced 3608{
f089940f 3609 if (m_widget && GTK_WIDGET_MAPPED(m_widget))
ab0c1a3c
PC
3610 {
3611 GdkDisplay* display = gtk_widget_get_display(m_widget);
3612 // Flush everything out to the server, and wait for it to finish.
3613 // This ensures nothing will overwrite the drawing we are about to do.
3614 gdk_display_sync(display);
010afced 3615
f089940f
PC
3616 GdkWindow* window = GTKGetDrawingWindow();
3617 if (window == NULL)
3618 window = m_widget->window;
3619 gdk_window_process_updates(window, true);
03647350 3620
6cab4fca
PC
3621 // Flush again, but no need to wait for it to finish
3622 gdk_display_flush(display);
a67f1484 3623 }
beab25bd
RR
3624}
3625
657b4fd4 3626bool wxWindowGTK::DoIsExposed( int x, int y ) const
847dfdb4
RR
3627{
3628 return m_updateRegion.Contains(x, y) != wxOutRegion;
3629}
3630
657b4fd4 3631bool wxWindowGTK::DoIsExposed( int x, int y, int w, int h ) const
847dfdb4
RR
3632{
3633 if (GetLayoutDirection() == wxLayout_RightToLeft)
3634 return m_updateRegion.Contains(x-w, y, w, h) != wxOutRegion;
3635 else
3636 return m_updateRegion.Contains(x, y, w, h) != wxOutRegion;
3637}
3638
beab25bd
RR
3639void wxWindowGTK::GtkSendPaintEvents()
3640{
3bcc8d15
RR
3641 if (!m_wxwindow)
3642 {
3bcc8d15
RR
3643 m_updateRegion.Clear();
3644 return;
3645 }
beab25bd 3646
f90566f5 3647 // Clip to paint region in wxClientDC
0a164d4c 3648 m_clipPaintRegion = true;
fab591c5 3649
bcb614b3
RR
3650 m_nativeUpdateRegion = m_updateRegion;
3651
847dfdb4
RR
3652 if (GetLayoutDirection() == wxLayout_RightToLeft)
3653 {
bcb614b3
RR
3654 // Transform m_updateRegion under RTL
3655 m_updateRegion.Clear();
fcb29b23 3656
847dfdb4 3657 gint width;
08f53168 3658 gdk_drawable_get_size(m_wxwindow->window, &width, NULL);
fcb29b23 3659
bcb614b3 3660 wxRegionIterator upd( m_nativeUpdateRegion );
847dfdb4
RR
3661 while (upd)
3662 {
3663 wxRect rect;
3664 rect.x = upd.GetX();
3665 rect.y = upd.GetY();
3666 rect.width = upd.GetWidth();
3667 rect.height = upd.GetHeight();
fcb29b23 3668
847dfdb4 3669 rect.x = width - rect.x - rect.width;
bcb614b3 3670 m_updateRegion.Union( rect );
fcb29b23 3671
847dfdb4
RR
3672 ++upd;
3673 }
3674 }
fcb29b23 3675
9c61c5b0 3676 switch ( GetBackgroundStyle() )
f90566f5 3677 {
9c61c5b0 3678 case wxBG_STYLE_ERASE:
822cf31c 3679 {
9c61c5b0
VZ
3680 wxWindowDC dc( (wxWindow*)this );
3681 dc.SetDeviceClippingRegion( m_updateRegion );
3682
3683 // Work around gtk-qt <= 0.60 bug whereby the window colour
3684 // remains grey
3685 if ( UseBgCol() &&
3686 wxSystemOptions::
3687 GetOptionInt("gtk.window.force-background-colour") )
3688 {
3689 dc.SetBackground(GetBackgroundColour());
3690 dc.Clear();
3691 }
3692
3693 wxEraseEvent erase_event( GetId(), &dc );
3694 erase_event.SetEventObject( this );
3695
3696 if ( HandleWindowEvent(erase_event) )
3697 {
3698 // background erased, don't do it again
3699 break;
3700 }
822cf31c 3701 }
9c61c5b0 3702 // fall through
b15ed747 3703
9c61c5b0
VZ
3704 case wxBG_STYLE_SYSTEM:
3705 if ( GetThemeEnabled() )
3706 {
3707 // find ancestor from which to steal background
3708 wxWindow *parent = wxGetTopLevelParent((wxWindow *)this);
3709 if (!parent)
3710 parent = (wxWindow*)this;
3711
3712 if (GTK_WIDGET_MAPPED(parent->m_widget))
3713 {
3714 wxRegionIterator upd( m_nativeUpdateRegion );
3715 while (upd)
3716 {
3717 GdkRectangle rect;
3718 rect.x = upd.GetX();
3719 rect.y = upd.GetY();
3720 rect.width = upd.GetWidth();
3721 rect.height = upd.GetHeight();
3722
3723 gtk_paint_flat_box( parent->m_widget->style,
f089940f 3724 GTKGetDrawingWindow(),
9c61c5b0
VZ
3725 (GtkStateType)GTK_WIDGET_STATE(m_wxwindow),
3726 GTK_SHADOW_NONE,
3727 &rect,
3728 parent->m_widget,
3729 (char *)"base",
3730 0, 0, -1, -1 );
3731
3732 ++upd;
3733 }
3734 }
3735 }
3736 break;
8ab7b4c5 3737
9c61c5b0
VZ
3738 case wxBG_STYLE_PAINT:
3739 // nothing to do: window will be painted over in EVT_PAINT
3740 break;
b15ed747 3741
9c61c5b0
VZ
3742 default:
3743 wxFAIL_MSG( "unsupported background style" );
b15ed747 3744 }
03eaa52a 3745
beab25bd
RR
3746 wxNcPaintEvent nc_paint_event( GetId() );
3747 nc_paint_event.SetEventObject( this );
937013e0 3748 HandleWindowEvent( nc_paint_event );
03eaa52a 3749
beab25bd
RR
3750 wxPaintEvent paint_event( GetId() );
3751 paint_event.SetEventObject( this );
937013e0 3752 HandleWindowEvent( paint_event );
beab25bd 3753
0a164d4c 3754 m_clipPaintRegion = false;
c89f5c02 3755
c89f5c02 3756 m_updateRegion.Clear();
bcb614b3 3757 m_nativeUpdateRegion.Clear();
beab25bd
RR
3758}
3759
8e1a5bf9
VZ
3760void wxWindowGTK::SetDoubleBuffered( bool on )
3761{
3762 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
3763
3764 if ( m_wxwindow )
3765 gtk_widget_set_double_buffered( m_wxwindow, on );
3766}
3767
2e992e06
VZ
3768bool wxWindowGTK::IsDoubleBuffered() const
3769{
3770 return GTK_WIDGET_DOUBLE_BUFFERED( m_wxwindow );
3771}
3772
596f1d11 3773void wxWindowGTK::ClearBackground()
c801d85f 3774{
82b978d7 3775 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
362c6693 3776}
c801d85f 3777
ff8bfdbb 3778#if wxUSE_TOOLTIPS
1e6feb95 3779void wxWindowGTK::DoSetToolTip( wxToolTip *tip )
b1170810 3780{
f03fc89f 3781 wxWindowBase::DoSetToolTip(tip);
ff8bfdbb 3782
f03fc89f 3783 if (m_tooltip)
fb4b0165 3784 {
7fc8b9a4 3785 m_tooltip->GTKApply( (wxWindow *)this );
fb4b0165
MW
3786 }
3787 else
3788 {
3789 GtkWidget *w = GetConnectWidget();
3790 wxToolTip::GTKApply(w, NULL);
fb4b0165 3791 }
b1170810
RR
3792}
3793
7fc8b9a4 3794void wxWindowGTK::GTKApplyToolTip( GtkTooltips *tips, const gchar *tip )
b1170810 3795{
614a87df 3796 GtkWidget *w = GetConnectWidget();
614a87df
JS
3797
3798#if GTK_CHECK_VERSION(2, 12, 0)
85314957 3799 if (!gtk_check_version(2, 12, 0))
614a87df 3800 {
85314957 3801 gtk_widget_set_tooltip_text (w, tip);
614a87df 3802 }
85314957 3803 else
614a87df 3804#endif
85314957
VZ
3805 {
3806 gtk_tooltips_set_tip(tips, w, tip, NULL);
3807 }
301cd871 3808}
ff8bfdbb 3809#endif // wxUSE_TOOLTIPS
b1170810 3810
1e6feb95 3811bool wxWindowGTK::SetBackgroundColour( const wxColour &colour )
c801d85f 3812{
0a164d4c 3813 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
8bbe427f 3814
739730ca 3815 if (!wxWindowBase::SetBackgroundColour(colour))
44dfb5ce 3816 return false;
c50f1fb9 3817
5edef14e 3818 if (colour.Ok())
994bc575 3819 {
5edef14e
VS
3820 // We need the pixel value e.g. for background clearing.
3821 m_backgroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
994bc575 3822 }
ca298c88 3823
5edef14e 3824 // apply style change (forceStyle=true so that new style is applied
c7382f91 3825 // even if the bg colour changed from valid to wxNullColour)
9c61c5b0 3826 GTKApplyWidgetStyle(true);
ea323db3 3827
5edef14e 3828 return true;
6de97a3b
RR
3829}
3830
1e6feb95 3831bool wxWindowGTK::SetForegroundColour( const wxColour &colour )
6de97a3b 3832{
0a164d4c 3833 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
8bbe427f 3834
739730ca
RR
3835 if (!wxWindowBase::SetForegroundColour(colour))
3836 {
5edef14e 3837 return false;
739730ca 3838 }
0a164d4c 3839
5edef14e 3840 if (colour.Ok())
ea323db3 3841 {
5edef14e
VS
3842 // We need the pixel value e.g. for background clearing.
3843 m_foregroundColour.CalcPixel(gtk_widget_get_colormap(m_widget));
ea323db3 3844 }
f03fc89f 3845
5edef14e
VS
3846 // apply style change (forceStyle=true so that new style is applied
3847 // even if the bg colour changed from valid to wxNullColour):
496e7ec6 3848 GTKApplyWidgetStyle(true);
5edef14e 3849
44dfb5ce 3850 return true;
58614078
RR
3851}
3852
496e7ec6 3853PangoContext *wxWindowGTK::GTKGetPangoDefaultContext()
2b5f62a0
VZ
3854{
3855 return gtk_widget_get_pango_context( m_widget );
3856}
0a164d4c 3857
496e7ec6 3858GtkRcStyle *wxWindowGTK::GTKCreateWidgetStyle(bool forceStyle)
58614078 3859{
f40fdaa3 3860 // do we need to apply any changes at all?
5edef14e 3861 if ( !forceStyle &&
984e8d0b
VS
3862 !m_font.Ok() &&
3863 !m_foregroundColour.Ok() && !m_backgroundColour.Ok() )
fb65642c 3864 {
f40fdaa3 3865 return NULL;
fb65642c
RR
3866 }
3867
f40fdaa3 3868 GtkRcStyle *style = gtk_rc_style_new();
1ecc4d80 3869
984e8d0b 3870 if ( m_font.Ok() )
db434467 3871 {
0a164d4c 3872 style->font_desc =
f40fdaa3 3873 pango_font_description_copy( m_font.GetNativeFontInfo()->description );
288059b2 3874 }
1ecc4d80 3875
532ae0f6
VZ
3876 int flagsNormal = 0,
3877 flagsPrelight = 0,
3878 flagsActive = 0,
3879 flagsInsensitive = 0;
3880
fe161a26 3881 if ( m_foregroundColour.Ok() )
1ecc4d80 3882 {
c6685317 3883 const GdkColor *fg = m_foregroundColour.GetColor();
0a164d4c 3884
532ae0f6
VZ
3885 style->fg[GTK_STATE_NORMAL] =
3886 style->text[GTK_STATE_NORMAL] = *fg;
3887 flagsNormal |= GTK_RC_FG | GTK_RC_TEXT;
0a164d4c 3888
532ae0f6
VZ
3889 style->fg[GTK_STATE_PRELIGHT] =
3890 style->text[GTK_STATE_PRELIGHT] = *fg;
3891 flagsPrelight |= GTK_RC_FG | GTK_RC_TEXT;
0a164d4c 3892
532ae0f6
VZ
3893 style->fg[GTK_STATE_ACTIVE] =
3894 style->text[GTK_STATE_ACTIVE] = *fg;
3895 flagsActive |= GTK_RC_FG | GTK_RC_TEXT;
1ecc4d80
RR
3896 }
3897
fe161a26 3898 if ( m_backgroundColour.Ok() )
1ecc4d80 3899 {
c6685317 3900 const GdkColor *bg = m_backgroundColour.GetColor();
5edef14e 3901
532ae0f6 3902 style->bg[GTK_STATE_NORMAL] =
5edef14e 3903 style->base[GTK_STATE_NORMAL] = *bg;
532ae0f6 3904 flagsNormal |= GTK_RC_BG | GTK_RC_BASE;
0a164d4c 3905
532ae0f6 3906 style->bg[GTK_STATE_PRELIGHT] =
5edef14e 3907 style->base[GTK_STATE_PRELIGHT] = *bg;
532ae0f6 3908 flagsPrelight |= GTK_RC_BG | GTK_RC_BASE;
0a164d4c 3909
532ae0f6 3910 style->bg[GTK_STATE_ACTIVE] =
5edef14e 3911 style->base[GTK_STATE_ACTIVE] = *bg;
532ae0f6 3912 flagsActive |= GTK_RC_BG | GTK_RC_BASE;
0a164d4c 3913
532ae0f6 3914 style->bg[GTK_STATE_INSENSITIVE] =
5edef14e 3915 style->base[GTK_STATE_INSENSITIVE] = *bg;
532ae0f6 3916 flagsInsensitive |= GTK_RC_BG | GTK_RC_BASE;
1ecc4d80 3917 }
0a164d4c 3918
532ae0f6
VZ
3919 style->color_flags[GTK_STATE_NORMAL] = (GtkRcFlags)flagsNormal;
3920 style->color_flags[GTK_STATE_PRELIGHT] = (GtkRcFlags)flagsPrelight;
3921 style->color_flags[GTK_STATE_ACTIVE] = (GtkRcFlags)flagsActive;
3922 style->color_flags[GTK_STATE_INSENSITIVE] = (GtkRcFlags)flagsInsensitive;
3923
f40fdaa3 3924 return style;
a81258be
RR
3925}
3926
496e7ec6 3927void wxWindowGTK::GTKApplyWidgetStyle(bool forceStyle)
a81258be 3928{
496e7ec6 3929 GtkRcStyle *style = GTKCreateWidgetStyle(forceStyle);
f8e045e2
RD
3930 if ( style )
3931 {
7074ce35 3932 DoApplyWidgetStyle(style);
f8e045e2
RD
3933 gtk_rc_style_unref(style);
3934 }
6dd18972
VS
3935
3936 // Style change may affect GTK+'s size calculation:
3937 InvalidateBestSize();
6de97a3b
RR
3938}
3939
7074ce35
VS
3940void wxWindowGTK::DoApplyWidgetStyle(GtkRcStyle *style)
3941{
0d013e46
VZ
3942 if ( m_wxwindow )
3943 {
3944 // block the signal temporarily to avoid sending
3945 // wxSysColourChangedEvents when we change the colours ourselves
3946 bool unblock = false;
3947 if ( IsTopLevel() )
3948 {
3949 unblock = true;
3950 g_signal_handlers_block_by_func(
3951 m_wxwindow, (void *)gtk_window_style_set_callback, this);
3952 }
013151c7 3953
7074ce35 3954 gtk_widget_modify_style(m_wxwindow, style);
0d013e46
VZ
3955
3956 if ( unblock )
3957 {
3958 g_signal_handlers_unblock_by_func(
3959 m_wxwindow, (void *)gtk_window_style_set_callback, this);
3960 }
3961 }
7cb93e45 3962 else
0d013e46 3963 {
7cb93e45 3964 gtk_widget_modify_style(m_widget, style);
0d013e46 3965 }
7074ce35
VS
3966}
3967
c7382f91
JS
3968bool wxWindowGTK::SetBackgroundStyle(wxBackgroundStyle style)
3969{
3970 wxWindowBase::SetBackgroundStyle(style);
0a164d4c 3971
9c61c5b0 3972 if ( style == wxBG_STYLE_PAINT )
c7382f91 3973 {
788f9633
VZ
3974 GdkWindow *window;
3975 if ( m_wxwindow )
3976 {
f089940f 3977 window = GTKGetDrawingWindow();
788f9633 3978 }
c7382f91 3979 else
788f9633
VZ
3980 {
3981 GtkWidget * const w = GetConnectWidget();
3982 window = w ? w->window : NULL;
3983 }
c7382f91
JS
3984
3985 if (window)
3986 {
3987 // Make sure GDK/X11 doesn't refresh the window
3988 // automatically.
3989 gdk_window_set_back_pixmap( window, None, False );
3990#ifdef __X__
3991 Display* display = GDK_WINDOW_DISPLAY(window);
3992 XFlush(display);
3993#endif
3994 m_needsStyleChange = false;
3995 }
788f9633
VZ
3996 else // window not realized yet
3997 {
c7382f91
JS
3998 // Do in OnIdle, because the window is not yet available
3999 m_needsStyleChange = true;
788f9633 4000 }
0a164d4c 4001
c7382f91
JS
4002 // Don't apply widget style, or we get a grey background
4003 }
4004 else
4005 {
4006 // apply style change (forceStyle=true so that new style is applied
4007 // even if the bg colour changed from valid to wxNullColour):
496e7ec6 4008 GTKApplyWidgetStyle(true);
c7382f91 4009 }
9c61c5b0 4010
c7382f91
JS
4011 return true;
4012}
7074ce35 4013
bcf7614c
PC
4014// ----------------------------------------------------------------------------
4015// Pop-up menu stuff
4016// ----------------------------------------------------------------------------
4017
4018#if wxUSE_MENUS_NATIVE
4019
edd6813c
PC
4020extern "C" {
4021static
bcf7614c
PC
4022void wxPopupMenuPositionCallback( GtkMenu *menu,
4023 gint *x, gint *y,
4024 gboolean * WXUNUSED(whatever),
4025 gpointer user_data )
4026{
4027 // ensure that the menu appears entirely on screen
4028 GtkRequisition req;
4029 gtk_widget_get_child_requisition(GTK_WIDGET(menu), &req);
4030
4031 wxSize sizeScreen = wxGetDisplaySize();
4032 wxPoint *pos = (wxPoint*)user_data;
4033
4034 gint xmax = sizeScreen.x - req.width,
4035 ymax = sizeScreen.y - req.height;
4036
4037 *x = pos->x < xmax ? pos->x : xmax;
4038 *y = pos->y < ymax ? pos->y : ymax;
4039}
edd6813c
PC
4040}
4041
bcf7614c
PC
4042bool wxWindowGTK::DoPopupMenu( wxMenu *menu, int x, int y )
4043{
4044 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
4045
a1c6f069 4046 menu->UpdateUI();
bcf7614c
PC
4047
4048 wxPoint pos;
4049 gpointer userdata;
4050 GtkMenuPositionFunc posfunc;
4051 if ( x == -1 && y == -1 )
4052 {
4053 // use GTK's default positioning algorithm
4054 userdata = NULL;
4055 posfunc = NULL;
4056 }
4057 else
4058 {
4059 pos = ClientToScreen(wxPoint(x, y));
4060 userdata = &pos;
4061 posfunc = wxPopupMenuPositionCallback;
4062 }
4063
4064 menu->m_popupShown = true;
4065 gtk_menu_popup(
4066 GTK_MENU(menu->m_menu),
d3b9f782
VZ
4067 NULL, // parent menu shell
4068 NULL, // parent menu item
bcf7614c
PC
4069 posfunc, // function to position it
4070 userdata, // client data
4071 0, // button used to activate it
4072 gtk_get_current_event_time()
4073 );
4074
4075 while (menu->m_popupShown)
4076 {
4077 gtk_main_iteration();
4078 }
4079
4080 return true;
4081}
4082
4083#endif // wxUSE_MENUS_NATIVE
4084
06cfab17 4085#if wxUSE_DRAG_AND_DROP
ac57418f 4086
1e6feb95 4087void wxWindowGTK::SetDropTarget( wxDropTarget *dropTarget )
c801d85f 4088{
82b978d7
RD
4089 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4090
1ecc4d80 4091 GtkWidget *dnd_widget = GetConnectWidget();
47d67540 4092
1fe91d70 4093 if (m_dropTarget) m_dropTarget->GtkUnregisterWidget( dnd_widget );
47d67540 4094
1ecc4d80
RR
4095 if (m_dropTarget) delete m_dropTarget;
4096 m_dropTarget = dropTarget;
47d67540 4097
1fe91d70 4098 if (m_dropTarget) m_dropTarget->GtkRegisterWidget( dnd_widget );
362c6693 4099}
c801d85f 4100
f03fc89f 4101#endif // wxUSE_DRAG_AND_DROP
ac57418f 4102
1e6feb95 4103GtkWidget* wxWindowGTK::GetConnectWidget()
e3e65dac 4104{
1ecc4d80
RR
4105 GtkWidget *connect_widget = m_widget;
4106 if (m_wxwindow) connect_widget = m_wxwindow;
47d67540 4107
1ecc4d80 4108 return connect_widget;
e3e65dac 4109}
47d67540 4110
ef5c70f9 4111bool wxWindowGTK::GTKIsOwnWindow(GdkWindow *window) const
903f689b 4112{
ef5c70f9
VZ
4113 wxArrayGdkWindows windowsThis;
4114 GdkWindow * const winThis = GTKGetWindow(windowsThis);
148cd9b6 4115
ef5c70f9
VZ
4116 return winThis ? window == winThis
4117 : windowsThis.Index(window) != wxNOT_FOUND;
4118}
4119
4120GdkWindow *wxWindowGTK::GTKGetWindow(wxArrayGdkWindows& WXUNUSED(windows)) const
4121{
f089940f 4122 return m_wxwindow ? GTKGetDrawingWindow() : m_widget->window;
903f689b
RR
4123}
4124
1e6feb95 4125bool wxWindowGTK::SetFont( const wxFont &font )
c801d85f 4126{
0a164d4c 4127 wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
9c288e4d 4128
5edef14e
VS
4129 if (!wxWindowBase::SetFont(font))
4130 return false;
c801d85f 4131
5edef14e
VS
4132 // apply style change (forceStyle=true so that new style is applied
4133 // even if the font changed from valid to wxNullFont):
496e7ec6 4134 GTKApplyWidgetStyle(true);
5edef14e
VS
4135
4136 return true;
362c6693 4137}
c801d85f 4138
94633ad9 4139void wxWindowGTK::DoCaptureMouse()
c801d85f 4140{
82b978d7
RD
4141 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4142
d3b9f782 4143 GdkWindow *window = NULL;
b231914f 4144 if (m_wxwindow)
f089940f 4145 window = GTKGetDrawingWindow();
ed673c6a 4146 else
b231914f 4147 window = GetConnectWidget()->window;
148cd9b6 4148
9a83f860 4149 wxCHECK_RET( window, wxT("CaptureMouse() failed") );
c50f1fb9 4150
f516d986 4151 const wxCursor* cursor = &m_cursor;
cca602ac
JS
4152 if (!cursor->Ok())
4153 cursor = wxSTANDARD_CURSOR;
4154
ed673c6a 4155 gdk_pointer_grab( window, FALSE,
1ecc4d80
RR
4156 (GdkEventMask)
4157 (GDK_BUTTON_PRESS_MASK |
4158 GDK_BUTTON_RELEASE_MASK |
148cd9b6 4159 GDK_POINTER_MOTION_HINT_MASK |
1ecc4d80 4160 GDK_POINTER_MOTION_MASK),
d3b9f782 4161 NULL,
cca602ac 4162 cursor->GetCursor(),
b02da6b1 4163 (guint32)GDK_CURRENT_TIME );
b231914f 4164 g_captureWindow = this;
0a164d4c 4165 g_captureWindowHasMouse = true;
362c6693 4166}
c801d85f 4167
94633ad9 4168void wxWindowGTK::DoReleaseMouse()
c801d85f 4169{
82b978d7
RD
4170 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4171
e4606ed9 4172 wxCHECK_RET( g_captureWindow, wxT("can't release mouse - not captured") );
47d67540 4173
d3b9f782 4174 g_captureWindow = NULL;
c43430bb 4175
d3b9f782 4176 GdkWindow *window = NULL;
b231914f 4177 if (m_wxwindow)
f089940f 4178 window = GTKGetDrawingWindow();
ed673c6a 4179 else
b231914f 4180 window = GetConnectWidget()->window;
148cd9b6 4181
b02da6b1
VZ
4182 if (!window)
4183 return;
c50f1fb9 4184
b02da6b1 4185 gdk_pointer_ungrab ( (guint32)GDK_CURRENT_TIME );
362c6693 4186}
c801d85f 4187
7738af59
VZ
4188void wxWindowGTK::GTKReleaseMouseAndNotify()
4189{
4190 DoReleaseMouse();
4191 wxMouseCaptureLostEvent evt(GetId());
4192 evt.SetEventObject( this );
937013e0 4193 HandleWindowEvent( evt );
7738af59
VZ
4194}
4195
1e6feb95
VZ
4196/* static */
4197wxWindow *wxWindowBase::GetCapture()
4198{
4199 return (wxWindow *)g_captureWindow;
4200}
4201
4202bool wxWindowGTK::IsRetained() const
c801d85f 4203{
0a164d4c 4204 return false;
362c6693 4205}
c801d85f 4206
22c9b211
VZ
4207void wxWindowGTK::SetScrollbar(int orient,
4208 int pos,
4209 int thumbVisible,
4210 int range,
4211 bool WXUNUSED(update))
c801d85f 4212{
63c95f27
PC
4213 const int dir = ScrollDirFromOrient(orient);
4214 GtkRange* const sb = m_scrollBar[dir];
9a83f860 4215 wxCHECK_RET( sb, wxT("this window is not scrollable") );
c801d85f 4216
8466fc74 4217 if (range <= 0)
de7bb802
PC
4218 {
4219 // GtkRange requires upper > lower
4220 range =
4221 thumbVisible = 1;
4222 }
47d67540 4223
b9e7f011 4224 GtkAdjustment * const adj = sb->adjustment;
add7cadd 4225 adj->step_increment = 1;
35510e48 4226 adj->page_increment =
add7cadd 4227 adj->page_size = thumbVisible;
63c95f27
PC
4228 adj->value = pos;
4229
4230 g_signal_handlers_block_by_func(
4231 sb, (void*)gtk_scrollbar_value_changed, this);
4232
4233 gtk_range_set_range(sb, 0, range);
4234 m_scrollPos[dir] = sb->adjustment->value;
4235
4236 g_signal_handlers_unblock_by_func(
4237 sb, (void*)gtk_scrollbar_value_changed, this);
87a3ebe9
VZ
4238}
4239
22c9b211 4240void wxWindowGTK::SetScrollPos(int orient, int pos, bool WXUNUSED(refresh))
c801d85f 4241{
b9e7f011
VZ
4242 const int dir = ScrollDirFromOrient(orient);
4243 GtkRange * const sb = m_scrollBar[dir];
9a83f860 4244 wxCHECK_RET( sb, wxT("this window is not scrollable") );
1ecc4d80 4245
add7cadd
PC
4246 // This check is more than an optimization. Without it, the slider
4247 // will not move smoothly while tracking when using wxScrollHelper.
4248 if (GetScrollPos(orient) != pos)
47d67540 4249 {
63c95f27
PC
4250 g_signal_handlers_block_by_func(
4251 sb, (void*)gtk_scrollbar_value_changed, this);
40e5ebbf 4252
63c95f27
PC
4253 gtk_range_set_value(sb, pos);
4254 m_scrollPos[dir] = sb->adjustment->value;
98264520 4255
63c95f27
PC
4256 g_signal_handlers_unblock_by_func(
4257 sb, (void*)gtk_scrollbar_value_changed, this);
cb43b372 4258 }
362c6693 4259}
c801d85f 4260
22c9b211 4261int wxWindowGTK::GetScrollThumb(int orient) const
c801d85f 4262{
b9e7f011 4263 GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)];
9a83f860 4264 wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") );
47d67540 4265
40b849aa 4266 return wxRound(sb->adjustment->page_size);
362c6693 4267}
c801d85f 4268
1e6feb95 4269int wxWindowGTK::GetScrollPos( int orient ) const
c801d85f 4270{
b9e7f011 4271 GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)];
9a83f860 4272 wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") );
c801d85f 4273
40b849aa 4274 return wxRound(sb->adjustment->value);
362c6693 4275}
c801d85f 4276
1e6feb95 4277int wxWindowGTK::GetScrollRange( int orient ) const
c801d85f 4278{
b9e7f011 4279 GtkRange * const sb = m_scrollBar[ScrollDirFromOrient(orient)];
9a83f860 4280 wxCHECK_MSG( sb, 0, wxT("this window is not scrollable") );
c801d85f 4281
40b849aa 4282 return wxRound(sb->adjustment->upper);
add7cadd
PC
4283}
4284
4285// Determine if increment is the same as +/-x, allowing for some small
4286// difference due to possible inexactness in floating point arithmetic
4287static inline bool IsScrollIncrement(double increment, double x)
4288{
4289 wxASSERT(increment > 0);
4290 const double tolerance = 1.0 / 1024;
4291 return fabs(increment - fabs(x)) < tolerance;
4292}
4293
71ead4bf 4294wxEventType wxWindowGTK::GTKGetScrollEventType(GtkRange* range)
add7cadd 4295{
add7cadd
PC
4296 wxASSERT(range == m_scrollBar[0] || range == m_scrollBar[1]);
4297
4298 const int barIndex = range == m_scrollBar[1];
4299 GtkAdjustment* adj = range->adjustment;
fcb29b23 4300
40b849aa 4301 const int value = wxRound(adj->value);
fcb29b23 4302
add7cadd
PC
4303 // save previous position
4304 const double oldPos = m_scrollPos[barIndex];
4305 // update current position
4306 m_scrollPos[barIndex] = adj->value;
4307 // If event should be ignored, or integral position has not changed
40b849aa 4308 if (!m_hasVMT || g_blockEventsOnDrag || value == wxRound(oldPos))
add7cadd
PC
4309 {
4310 return wxEVT_NULL;
4311 }
4312
4313 wxEventType eventType = wxEVT_SCROLL_THUMBTRACK;
4314 if (!m_isScrolling)
4315 {
4316 // Difference from last change event
4317 const double diff = adj->value - oldPos;
4318 const bool isDown = diff > 0;
4319
4320 if (IsScrollIncrement(adj->step_increment, diff))
4321 {
4322 eventType = isDown ? wxEVT_SCROLL_LINEDOWN : wxEVT_SCROLL_LINEUP;
4323 }
4324 else if (IsScrollIncrement(adj->page_increment, diff))
4325 {
4326 eventType = isDown ? wxEVT_SCROLL_PAGEDOWN : wxEVT_SCROLL_PAGEUP;
4327 }
4328 else if (m_mouseButtonDown)
4329 {
4330 // Assume track event
4331 m_isScrolling = true;
4332 }
4333 }
4334 return eventType;
362c6693 4335}
c801d85f 4336
1e6feb95 4337void wxWindowGTK::ScrollWindow( int dx, int dy, const wxRect* WXUNUSED(rect) )
c801d85f 4338{
82b978d7
RD
4339 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4340
223d09f6 4341 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
2b5f62a0 4342
f47ae6e7 4343 // No scrolling requested.
8e217128 4344 if ((dx == 0) && (dy == 0)) return;
35510e48 4345
0a164d4c 4346 m_clipPaintRegion = true;
0fc5dbf5 4347
08f53168 4348 WX_PIZZA(m_wxwindow)->scroll(dx, dy);
0fc5dbf5 4349
0a164d4c 4350 m_clipPaintRegion = false;
113faca1 4351
231018bd 4352#if wxUSE_CARET
113faca1
JS
4353 bool restoreCaret = (GetCaret() != NULL && GetCaret()->IsVisible());
4354 if (restoreCaret)
4355 {
4356 wxRect caretRect(GetCaret()->GetPosition(), GetCaret()->GetSize());
4357 if (dx > 0)
4358 caretRect.width += dx;
4359 else
fcb29b23 4360 {
113faca1 4361 caretRect.x += dx; caretRect.width -= dx;
fcb29b23 4362 }
113faca1
JS
4363 if (dy > 0)
4364 caretRect.height += dy;
4365 else
fcb29b23 4366 {
113faca1 4367 caretRect.y += dy; caretRect.height -= dy;
fcb29b23
VZ
4368 }
4369
113faca1
JS
4370 RefreshRect(caretRect);
4371 }
fcb29b23 4372#endif // wxUSE_CARET
c801d85f 4373}
3723b7b1 4374
496e7ec6 4375void wxWindowGTK::GTKScrolledWindowSetBorder(GtkWidget* w, int wxstyle)
6493aaca
VZ
4376{
4377 //RN: Note that static controls usually have no border on gtk, so maybe
88a7a4e1 4378 //it makes sense to treat that as simply no border at the wx level
6493aaca
VZ
4379 //as well...
4380 if (!(wxstyle & wxNO_BORDER) && !(wxstyle & wxBORDER_STATIC))
4381 {
4382 GtkShadowType gtkstyle;
88a7a4e1 4383
6493aaca
VZ
4384 if(wxstyle & wxBORDER_RAISED)
4385 gtkstyle = GTK_SHADOW_OUT;
ec2d6790 4386 else if ((wxstyle & wxBORDER_SUNKEN) || (wxstyle & wxBORDER_THEME))
6493aaca 4387 gtkstyle = GTK_SHADOW_IN;
78cd9c69
JS
4388#if 0
4389 // Now obsolete
6493aaca
VZ
4390 else if (wxstyle & wxBORDER_DOUBLE)
4391 gtkstyle = GTK_SHADOW_ETCHED_IN;
78cd9c69 4392#endif
6493aaca
VZ
4393 else //default
4394 gtkstyle = GTK_SHADOW_IN;
4395
88a7a4e1 4396 gtk_scrolled_window_set_shadow_type( GTK_SCROLLED_WINDOW(w),
6493aaca
VZ
4397 gtkstyle );
4398 }
4399}
4400
015dca24
MR
4401void wxWindowGTK::SetWindowStyleFlag( long style )
4402{
4403 // Updates the internal variable. NB: Now m_windowStyle bits carry the _new_ style values already
4404 wxWindowBase::SetWindowStyleFlag(style);
4405}
4e5a4c69 4406
3723b7b1
JS
4407// Find the wxWindow at the current mouse position, also returning the mouse
4408// position.
4409wxWindow* wxFindWindowAtPointer(wxPoint& pt)
4410{
59a12e90
JS
4411 pt = wxGetMousePosition();
4412 wxWindow* found = wxFindWindowAtPoint(pt);
4413 return found;
3723b7b1
JS
4414}
4415
4416// Get the current mouse position.
4417wxPoint wxGetMousePosition()
4418{
59a12e90
JS
4419 /* This crashes when used within wxHelpContext,
4420 so we have to use the X-specific implementation below.
4421 gint x, y;
4422 GdkModifierType *mask;
4423 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4424
4425 return wxPoint(x, y);
4426 */
4427
3723b7b1
JS
4428 int x, y;
4429 GdkWindow* windowAtPtr = gdk_window_at_pointer(& x, & y);
59a12e90 4430
37d81cc2 4431 Display *display = windowAtPtr ? GDK_WINDOW_XDISPLAY(windowAtPtr) : GDK_DISPLAY();
59a12e90
JS
4432 Window rootWindow = RootWindowOfScreen (DefaultScreenOfDisplay(display));
4433 Window rootReturn, childReturn;
4434 int rootX, rootY, winX, winY;
4435 unsigned int maskReturn;
4436
4437 XQueryPointer (display,
5cd09f0b
RR
4438 rootWindow,
4439 &rootReturn,
59a12e90
JS
4440 &childReturn,
4441 &rootX, &rootY, &winX, &winY, &maskReturn);
4442 return wxPoint(rootX, rootY);
4443
3723b7b1
JS
4444}
4445
08f53168
PC
4446GdkWindow* wxWindowGTK::GTKGetDrawingWindow() const
4447{
4448 GdkWindow* window = NULL;
4449 if (m_wxwindow)
375efc1f 4450 window = m_wxwindow->window;
08f53168
PC
4451 return window;
4452}
5f346ddc
VS
4453
4454// ----------------------------------------------------------------------------
4455// freeze/thaw
4456// ----------------------------------------------------------------------------
4457
89267fe5
VS
4458extern "C"
4459{
4460
4461// this is called if we attempted to freeze unrealized widget when it finally
4462// is realized (and so can be frozen):
f089940f 4463static void wx_frozen_widget_realize(GtkWidget* w, wxWindowGTK* win)
89267fe5
VS
4464{
4465 wxASSERT( w && !GTK_WIDGET_NO_WINDOW(w) );
4466 wxASSERT( GTK_WIDGET_REALIZED(w) );
4467
4468 g_signal_handlers_disconnect_by_func
4469 (
4470 w,
4471 (void*)wx_frozen_widget_realize,
f089940f 4472 win
89267fe5
VS
4473 );
4474
9914bfbb
PC
4475 GdkWindow* window = w->window;
4476 if (w == win->m_wxwindow)
4477 window = win->GTKGetDrawingWindow();
f089940f 4478 gdk_window_freeze_updates(window);
89267fe5
VS
4479}
4480
4481} // extern "C"
4482
5f346ddc
VS
4483void wxWindowGTK::GTKFreezeWidget(GtkWidget *w)
4484{
89267fe5
VS
4485 if ( !w || GTK_WIDGET_NO_WINDOW(w) )
4486 return; // window-less widget, cannot be frozen
4487
4488 if ( !GTK_WIDGET_REALIZED(w) )
4489 {
4490 // we can't thaw unrealized widgets because they don't have GdkWindow,
4491 // so set it up to be done immediately after realization:
4492 g_signal_connect_after
4493 (
4494 w,
4495 "realize",
4496 G_CALLBACK(wx_frozen_widget_realize),
f089940f 4497 this
89267fe5
VS
4498 );
4499 return;
4500 }
4501
9914bfbb
PC
4502 GdkWindow* window = w->window;
4503 if (w == m_wxwindow)
4504 window = GTKGetDrawingWindow();
f089940f 4505 gdk_window_freeze_updates(window);
5f346ddc
VS
4506}
4507
4508void wxWindowGTK::GTKThawWidget(GtkWidget *w)
4509{
89267fe5
VS
4510 if ( !w || GTK_WIDGET_NO_WINDOW(w) )
4511 return; // window-less widget, cannot be frozen
4512
4513 if ( !GTK_WIDGET_REALIZED(w) )
4514 {
4515 // the widget wasn't realized yet, no need to thaw
4516 g_signal_handlers_disconnect_by_func
4517 (
4518 w,
4519 (void*)wx_frozen_widget_realize,
f089940f 4520 this
89267fe5
VS
4521 );
4522 return;
4523 }
4524
9914bfbb
PC
4525 GdkWindow* window = w->window;
4526 if (w == m_wxwindow)
4527 window = GTKGetDrawingWindow();
f089940f 4528 gdk_window_thaw_updates(window);
5f346ddc
VS
4529}
4530
4531void wxWindowGTK::DoFreeze()
4532{
4533 GTKFreezeWidget(m_widget);
4534 if ( m_wxwindow && m_widget != m_wxwindow )
4535 GTKFreezeWidget(m_wxwindow);
4536}
4537
4538void wxWindowGTK::DoThaw()
4539{
4540 GTKThawWidget(m_widget);
4541 if ( m_wxwindow && m_widget != m_wxwindow )
4542 GTKThawWidget(m_wxwindow);
4543}