]> git.saurik.com Git - wxWidgets.git/blame - src/gtk/window.cpp
OS/2 updates to catch the port up with the rest of the library.
[wxWidgets.git] / src / gtk / window.cpp
CommitLineData
c801d85f 1/////////////////////////////////////////////////////////////////////////////
f6bcfd97 2// Name: gtk/window.cpp
c801d85f
KB
3// Purpose:
4// Author: Robert Roebling
c67d8618 5// Id: $Id$
01111366 6// Copyright: (c) 1998 Robert Roebling, Julian Smart
5e0aa05a 7// Licence: wxWindows licence
c801d85f
KB
8/////////////////////////////////////////////////////////////////////////////
9
10
11#ifdef __GNUG__
bfc6fde4 12 #pragma implementation "window.h"
c801d85f
KB
13#endif
14
d02af7bb
JJ
15#ifdef __VMS
16#define XWarpPointer XWARPPOINTER
17#endif
18
c801d85f
KB
19#include "wx/defs.h"
20#include "wx/window.h"
33611ebb 21#include "wx/dcclient.h"
c801d85f
KB
22#include "wx/frame.h"
23#include "wx/app.h"
24#include "wx/layout.h"
25#include "wx/utils.h"
26#include "wx/dialog.h"
27#include "wx/msgdlg.h"
0fc5dbf5 28#include "wx/module.h"
bfc6fde4 29
06cfab17 30#if wxUSE_DRAG_AND_DROP
bfc6fde4 31 #include "wx/dnd.h"
ac57418f 32#endif
bfc6fde4 33
cad880f5 34#if wxUSE_TOOLTIPS
bfc6fde4 35 #include "wx/tooltip.h"
cad880f5 36#endif
bfc6fde4 37
f6bcfd97
BP
38#if wxUSE_CARET
39 #include "wx/caret.h"
40#endif // wxUSE_CARET
41
ab93a576 42#if wxUSE_TEXTCTRL
0fc5dbf5 43 #include "wx/textctrl.h"
ab93a576
RD
44#endif
45
30dea054 46#include "wx/menu.h"
d4c99d6f 47#include "wx/statusbr.h"
b4071e91 48#include "wx/intl.h"
3bc755fc 49#include "wx/settings.h"
3069ac4e 50#include "wx/log.h"
48d011c8 51#include "wx/fontutil.h"
b4071e91 52
3ac8d3bc
RR
53#ifdef __WXDEBUG__
54 #include "wx/thread.h"
55#endif
56
b4071e91 57#include <math.h>
fab591c5 58#include <ctype.h>
c801d85f 59
9e691f46 60#include "wx/gtk/private.h"
3ac8d3bc
RR
61#include <gdk/gdkprivate.h>
62#include <gdk/gdkkeysyms.h>
3ac8d3bc 63#include <gdk/gdkx.h>
6bc8a1c8 64
8cb9f0d0
RR
65#include <gtk/gtk.h>
66#include <gtk/gtkprivate.h>
67
68#include "wx/gtk/win_gtk.h"
69
2b5f62a0
VZ
70#ifdef __WXGTK20__
71#include <pango/pangox.h>
72#endif
73
9e691f46
VZ
74#ifdef __WXGTK20__
75 #define SET_CONTAINER_FOCUS(w, d) gtk_widget_child_focus((w), (d))
76#else
77 #define SET_CONTAINER_FOCUS(w, d) gtk_container_focus(GTK_CONTAINER(w), (d))
78#endif
79
4e5a4c69
RR
80#ifdef __WXGTK20__
81 #ifdef HAVE_XIM
82 #undef HAVE_XIM
83 #endif
84#endif
85
af1d24da
RR
86#ifdef __WXGTK20__
87extern GtkContainerClass *pizza_parent_class;
88#endif
89
868a2826
RR
90//-----------------------------------------------------------------------------
91// documentation on internals
92//-----------------------------------------------------------------------------
93
94/*
95 I have been asked several times about writing some documentation about
96 the GTK port of wxWindows, especially its internal structures. Obviously,
97 you cannot understand wxGTK without knowing a little about the GTK, but
47d67540 98 some more information about what the wxWindow, which is the base class
868a2826 99 for all other window classes, does seems required as well.
47d67540 100
30760ce7
RR
101 I)
102
868a2826 103 What does wxWindow do? It contains the common interface for the following
e380f72b 104 jobs of its descendants:
47d67540 105
868a2826 106 1) Define the rudimentary behaviour common to all window classes, such as
e380f72b
RR
107 resizing, intercepting user input (so as to make it possible to use these
108 events for special purposes in a derived class), window names etc.
868a2826
RR
109
110 2) Provide the possibility to contain and manage children, if the derived
111 class is allowed to contain children, which holds true for those window
e380f72b 112 classes which do not display a native GTK widget. To name them, these
868a2826 113 classes are wxPanel, wxScrolledWindow, wxDialog, wxFrame. The MDI frame-
47d67540 114 work classes are a special case and are handled a bit differently from
e380f72b 115 the rest. The same holds true for the wxNotebook class.
47d67540 116
868a2826
RR
117 3) Provide the possibility to draw into a client area of a window. This,
118 too, only holds true for classes that do not display a native GTK widget
119 as above.
47d67540 120
e380f72b
RR
121 4) Provide the entire mechanism for scrolling widgets. This actual inter-
122 face for this is usually in wxScrolledWindow, but the GTK implementation
868a2826 123 is in this class.
47d67540 124
868a2826
RR
125 5) A multitude of helper or extra methods for special purposes, such as
126 Drag'n'Drop, managing validators etc.
47d67540 127
30760ce7
RR
128 6) Display a border (sunken, raised, simple or none).
129
e380f72b
RR
130 Normally one might expect, that one wxWindows window would always correspond
131 to one GTK widget. Under GTK, there is no such allround widget that has all
868a2826
RR
132 the functionality. Moreover, the GTK defines a client area as a different
133 widget from the actual widget you are handling. Last but not least some
134 special classes (e.g. wxFrame) handle different categories of widgets and
135 still have the possibility to draw something in the client area.
136 It was therefore required to write a special purpose GTK widget, that would
137 represent a client area in the sense of wxWindows capable to do the jobs
138 2), 3) and 4). I have written this class and it resides in win_gtk.c of
139 this directory.
47d67540 140
868a2826 141 All windows must have a widget, with which they interact with other under-
e380f72b 142 lying GTK widgets. It is this widget, e.g. that has to be resized etc and
868a2826 143 thw wxWindow class has a member variable called m_widget which holds a
e380f72b
RR
144 pointer to this widget. When the window class represents a GTK native widget,
145 this is (in most cases) the only GTK widget the class manages. E.g. the
146 wxStatitText class handles only a GtkLabel widget a pointer to which you
147 can find in m_widget (defined in wxWindow)
8bbe427f 148
e380f72b 149 When the class has a client area for drawing into and for containing children
da048e3d 150 it has to handle the client area widget (of the type GtkPizza, defined in
8bbe427f
VZ
151 win_gtk.c), but there could be any number of widgets, handled by a class
152 The common rule for all windows is only, that the widget that interacts with
153 the rest of GTK must be referenced in m_widget and all other widgets must be
154 children of this widget on the GTK level. The top-most widget, which also
155 represents the client area, must be in the m_wxwindow field and must be of
da048e3d 156 the type GtkPizza.
47d67540 157
868a2826
RR
158 As I said, the window classes that display a GTK native widget only have
159 one widget, so in the case of e.g. the wxButton class m_widget holds a
160 pointer to a GtkButton widget. But windows with client areas (for drawing
161 and children) have a m_widget field that is a pointer to a GtkScrolled-
da048e3d 162 Window and a m_wxwindow field that is pointer to a GtkPizza and this
868a2826 163 one is (in the GTK sense) a child of the GtkScrolledWindow.
47d67540 164
868a2826
RR
165 If the m_wxwindow field is set, then all input to this widget is inter-
166 cepted and sent to the wxWindows class. If not, all input to the widget
167 that gets pointed to by m_widget gets intercepted and sent to the class.
148cd9b6 168
30760ce7 169 II)
148cd9b6 170
30760ce7
RR
171 The design of scrolling in wxWindows is markedly different from that offered
172 by the GTK itself and therefore we cannot simply take it as it is. In GTK,
173 clicking on a scrollbar belonging to scrolled window will inevitably move
174 the window. In wxWindows, the scrollbar will only emit an event, send this
175 to (normally) a wxScrolledWindow and that class will call ScrollWindow()
da048e3d 176 which actually moves the window and its subchildren. Note that GtkPizza
30760ce7
RR
177 memorizes how much it has been scrolled but that wxWindows forgets this
178 so that the two coordinates systems have to be kept in synch. This is done
da048e3d 179 in various places using the pizza->xoffset and pizza->yoffset values.
148cd9b6
VZ
180
181 III)
182
30760ce7
RR
183 Singularily the most broken code in GTK is the code that is supposes to
184 inform subwindows (child windows) about new positions. Very often, duplicate
185 events are sent without changes in size or position, equally often no
186 events are sent at all (All this is due to a bug in the GtkContainer code
187 which got fixed in GTK 1.2.6). For that reason, wxGTK completely ignores
188 GTK's own system and it simply waits for size events for toplevel windows
189 and then iterates down the respective size events to all window. This has
190 the disadvantage, that windows might get size events before the GTK widget
191 actually has the reported size. This doesn't normally pose any problem, but
3017f78d 192 the OpenGl drawing routines rely on correct behaviour. Therefore, I have
30760ce7
RR
193 added the m_nativeSizeEvents flag, which is true only for the OpenGL canvas,
194 i.e. the wxGLCanvas will emit a size event, when (and not before) the X11
195 window that is used for OpenGl output really has that size (as reported by
196 GTK).
197
198 IV)
148cd9b6 199
30760ce7
RR
200 If someone at some point of time feels the immense desire to have a look at,
201 change or attempt to optimse the Refresh() logic, this person will need an
202 intimate understanding of what a "draw" and what an "expose" events are and
203 what there are used for, in particular when used in connection with GTK's
204 own windowless widgets. Beware.
148cd9b6 205
30760ce7 206 V)
148cd9b6 207
30760ce7
RR
208 Cursors, too, have been a constant source of pleasure. The main difficulty
209 is that a GdkWindow inherits a cursor if the programmer sets a new cursor
210 for the parent. To prevent this from doing too much harm, I use idle time
211 to set the cursor over and over again, starting from the toplevel windows
212 and ending with the youngest generation (speaking of parent and child windows).
213 Also don't forget that cursors (like much else) are connected to GdkWindows,
214 not GtkWidgets and that the "window" field of a GtkWidget might very well
148cd9b6 215 point to the GdkWindow of the parent widget (-> "window less widget") and
30760ce7 216 that the two obviously have very different meanings.
868a2826
RR
217
218*/
219
f03fc89f
VZ
220//-----------------------------------------------------------------------------
221// data
222//-----------------------------------------------------------------------------
223
224extern wxList wxPendingDelete;
225extern bool g_blockEventsOnDrag;
226extern bool g_blockEventsOnScroll;
238d735d 227extern wxCursor g_globalCursor;
f68586e5 228
4e5a4c69
RR
229static GdkGC *g_eraseGC = NULL;
230
1e6feb95
VZ
231// mouse capture state: the window which has it and if the mouse is currently
232// inside it
233static wxWindowGTK *g_captureWindow = (wxWindowGTK*) NULL;
234static bool g_captureWindowHasMouse = FALSE;
235
236/* extern */ wxWindowGTK *g_focusWindow = (wxWindowGTK*) NULL;
237
238// the last window which had the focus - this is normally never NULL (except
239// if we never had focus at all) as even when g_focusWindow is NULL it still
240// keeps its previous value
d7fa7eaa 241static wxWindowGTK *g_focusWindowLast = (wxWindowGTK*) NULL;
148cd9b6 242
e8c12a53
VS
243// the frame that is currently active (i.e. its child has focus). It is
244// used to generate wxActivateEvents
d7fa7eaa 245static wxWindowGTK *g_activeFrame = (wxWindowGTK*) NULL;
e8c12a53
VS
246static bool g_activeFrameLostFocus = FALSE;
247
d7fa7eaa
RR
248// If a window get the focus set but has not been realized
249// yet, defer setting the focus to idle time.
250wxWindowGTK *g_delayedFocus = (wxWindowGTK*) NULL;
251
148cd9b6
VZ
252// if we detect that the app has got/lost the focus, we set this variable to
253// either TRUE or FALSE and an activate event will be sent during the next
254// OnIdle() call and it is reset to -1: this value means that we shouldn't
255// send any activate events at all
256static int g_sendActivateEvent = -1;
f03fc89f
VZ
257
258/* hack: we need something to pass to gtk_menu_popup, so we store the time of
259 the last click here */
260static guint32 gs_timeLastClick = 0;
261
3ac8d3bc
RR
262extern bool g_mainThreadLocked;
263
2e563988
RR
264//-----------------------------------------------------------------------------
265// debug
266//-----------------------------------------------------------------------------
267
d52e91c9 268#ifndef __WXGTK20__
2b5f62a0 269#define DISABLE_STYLE_IF_BROKEN_THEME 0
d52e91c9 270#endif
a56fcaaf 271
2e563988
RR
272#ifdef __WXDEBUG__
273
c6e62f74
KB
274#if wxUSE_THREADS
275# define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance");
276#else
277# define DEBUG_MAIN_THREAD
278#endif
559d79aa
BJ
279#else
280#define DEBUG_MAIN_THREAD
f03fc89f 281#endif // Debug
ff8bfdbb 282
6cad4f1b
VZ
283// the trace mask used for the focus debugging messages
284#define TRACE_FOCUS _T("focus")
285
85eb36c2
RR
286//-----------------------------------------------------------------------------
287// missing gdk functions
288//-----------------------------------------------------------------------------
289
290void
291gdk_window_warp_pointer (GdkWindow *window,
c50f1fb9
VZ
292 gint x,
293 gint y)
85eb36c2 294{
dd00f3f6 295#ifndef __WXGTK20__
85eb36c2 296 GdkWindowPrivate *priv;
dd00f3f6 297#endif
c50f1fb9 298
85eb36c2 299 if (!window)
27df579a 300 window = GDK_ROOT_PARENT();
c50f1fb9 301
dd00f3f6
OK
302#ifdef __WXGTK20__
303 if (!GDK_WINDOW_DESTROYED(window))
304 {
305 XWarpPointer (GDK_WINDOW_XDISPLAY(window),
306 None, /* not source window -> move from anywhere */
307 GDK_WINDOW_XID(window), /* dest window */
308 0, 0, 0, 0, /* not source window -> move from anywhere */
309 x, y );
310 }
311#else
85eb36c2 312 priv = (GdkWindowPrivate*) window;
c50f1fb9 313
85eb36c2
RR
314 if (!priv->destroyed)
315 {
c50f1fb9 316 XWarpPointer (priv->xdisplay,
85eb36c2 317 None, /* not source window -> move from anywhere */
c50f1fb9 318 priv->xwindow, /* dest window */
85eb36c2 319 0, 0, 0, 0, /* not source window -> move from anywhere */
c50f1fb9 320 x, y );
85eb36c2 321 }
dd00f3f6 322#endif
85eb36c2
RR
323}
324
acfd422a 325//-----------------------------------------------------------------------------
a2053b27 326// idle system
acfd422a
RR
327//-----------------------------------------------------------------------------
328
a2053b27
RR
329extern void wxapp_install_idle_handler();
330extern bool g_isIdle;
331
ed673c6a
RR
332//-----------------------------------------------------------------------------
333// local code (see below)
334//-----------------------------------------------------------------------------
335
f6bcfd97 336// returns the child of win which currently has focus or NULL if not found
1e6feb95 337//
fc71ef6e 338// Note: can't be static, needed by textctrl.cpp.
3379ed37 339wxWindow *wxFindFocusedChild(wxWindowGTK *win)
f6bcfd97 340{
3379ed37 341 wxWindow *winFocus = wxWindowGTK::FindFocus();
f6bcfd97 342 if ( !winFocus )
3379ed37 343 return (wxWindow *)NULL;
f6bcfd97
BP
344
345 if ( winFocus == win )
3379ed37 346 return (wxWindow *)win;
f6bcfd97
BP
347
348 for ( wxWindowList::Node *node = win->GetChildren().GetFirst();
349 node;
350 node = node->GetNext() )
351 {
3379ed37 352 wxWindow *child = wxFindFocusedChild(node->GetData());
f6bcfd97
BP
353 if ( child )
354 return child;
355 }
356
3379ed37 357 return (wxWindow *)NULL;
f6bcfd97
BP
358}
359
1e6feb95 360static void draw_frame( GtkWidget *widget, wxWindowGTK *win )
ed673c6a 361{
1e6feb95
VZ
362 // wxUniversal widgets draw the borders and scrollbars themselves
363#ifndef __WXUNIVERSAL__
ed673c6a
RR
364 if (!win->m_hasVMT)
365 return;
366
367 int dw = 0;
368 int dh = 0;
369
5b8a521e 370 if (win->m_hasScrolling)
ed673c6a 371 {
beab25bd 372 GtkScrolledWindow *scroll_window = GTK_SCROLLED_WINDOW(widget);
2daa0ce9 373
beab25bd
RR
374 GtkRequisition vscroll_req;
375 vscroll_req.width = 2;
376 vscroll_req.height = 2;
377 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->vscrollbar) )->size_request )
378 (scroll_window->vscrollbar, &vscroll_req );
2daa0ce9 379
beab25bd
RR
380 GtkRequisition hscroll_req;
381 hscroll_req.width = 2;
382 hscroll_req.height = 2;
383 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->hscrollbar) )->size_request )
384 (scroll_window->hscrollbar, &hscroll_req );
9000c624 385
beab25bd 386 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(widget) );
ed673c6a 387
beab25bd
RR
388 if (scroll_window->vscrollbar_visible)
389 {
390 dw += vscroll_req.width;
391 dw += scroll_class->scrollbar_spacing;
392 }
ed673c6a 393
beab25bd
RR
394 if (scroll_window->hscrollbar_visible)
395 {
396 dh += hscroll_req.height;
397 dh += scroll_class->scrollbar_spacing;
398 }
399}
ed673c6a
RR
400
401 int dx = 0;
402 int dy = 0;
403 if (GTK_WIDGET_NO_WINDOW (widget))
404 {
405 dx += widget->allocation.x;
406 dy += widget->allocation.y;
407 }
408
409 if (win->HasFlag(wxRAISED_BORDER))
410 {
411 gtk_draw_shadow( widget->style,
412 widget->window,
413 GTK_STATE_NORMAL,
414 GTK_SHADOW_OUT,
415 dx, dy,
07f5b19a 416 widget->allocation.width-dw, widget->allocation.height-dh );
ed673c6a
RR
417 return;
418 }
419
420 if (win->HasFlag(wxSUNKEN_BORDER))
421 {
422 gtk_draw_shadow( widget->style,
423 widget->window,
424 GTK_STATE_NORMAL,
425 GTK_SHADOW_IN,
426 dx, dy,
07f5b19a 427 widget->allocation.width-dw, widget->allocation.height-dh );
ed673c6a
RR
428 return;
429 }
148cd9b6 430
ed673c6a
RR
431 if (win->HasFlag(wxSIMPLE_BORDER))
432 {
433 GdkGC *gc;
b02da6b1 434 gc = gdk_gc_new( widget->window );
ed673c6a 435 gdk_gc_set_foreground( gc, &widget->style->black );
148cd9b6 436 gdk_draw_rectangle( widget->window, gc, FALSE,
ed673c6a 437 dx, dy,
07f5b19a 438 widget->allocation.width-dw-1, widget->allocation.height-dh-1 );
ed673c6a
RR
439 gdk_gc_unref( gc );
440 return;
441 }
1e6feb95 442#endif // __WXUNIVERSAL__
ed673c6a
RR
443}
444
445//-----------------------------------------------------------------------------
446// "expose_event" of m_widget
447//-----------------------------------------------------------------------------
448
1e6feb95 449gint gtk_window_own_expose_callback( GtkWidget *widget, GdkEventExpose *gdk_event, wxWindowGTK *win )
ed673c6a 450{
b6fa52db 451 if (gdk_event->count > 0) return FALSE;
1e6feb95 452
ed673c6a 453 draw_frame( widget, win );
1e6feb95 454
af1d24da
RR
455#ifdef __WXGTK20__
456
457 (* GTK_WIDGET_CLASS (pizza_parent_class)->expose_event) (widget, gdk_event);
0fc5dbf5
VZ
458
459#endif
b6fa52db 460 return TRUE;
ed673c6a
RR
461}
462
463//-----------------------------------------------------------------------------
147bc491 464// "draw" of m_widget
ed673c6a
RR
465//-----------------------------------------------------------------------------
466
67d78217
RR
467#ifndef __WXGTK20__
468
1e6feb95 469static void gtk_window_own_draw_callback( GtkWidget *widget, GdkRectangle *WXUNUSED(rect), wxWindowGTK *win )
ed673c6a
RR
470{
471 draw_frame( widget, win );
472}
473
1c6896d7 474#endif // GTK+ < 2.0
c5b42c87 475
47c93b63
RR
476//-----------------------------------------------------------------------------
477// "size_request" of m_widget
478//-----------------------------------------------------------------------------
479
480static void gtk_window_size_request_callback( GtkWidget *widget, GtkRequisition *requisition, wxWindow *win )
481{
482 int w,h;
483 win->GetSize( &w, &h );
484 if (w < 2) w = 2;
485 if (h < 2) h = 2;
1e6feb95 486
47c93b63
RR
487 requisition->height = h;
488 requisition->width = w;
489}
490
c801d85f 491//-----------------------------------------------------------------------------
034be888 492// "expose_event" of m_wxwindow
c801d85f
KB
493//-----------------------------------------------------------------------------
494
1e6feb95
VZ
495static int gtk_window_expose_callback( GtkWidget *widget,
496 GdkEventExpose *gdk_event,
497 wxWindow *win )
47d67540 498{
3ac8d3bc
RR
499 DEBUG_MAIN_THREAD
500
b6fa52db
RR
501 if (g_isIdle)
502 wxapp_install_idle_handler();
1e6feb95 503
3fc6e5fa
RR
504#ifdef __WXGTK20__
505 // This callback gets called in drawing-idle time under
506 // GTK 2.0, so we don't need to defer anything to idle
507 // time anymore.
2b5f62a0 508
3fc6e5fa
RR
509 GtkPizza *pizza = GTK_PIZZA( widget );
510 if (gdk_event->window != pizza->bin_window) return FALSE;
2b5f62a0 511
d7fa7eaa
RR
512#if 0
513 if (win->GetName())
3d2d8da1
RR
514 {
515 wxPrintf( wxT("OnExpose from ") );
516 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
517 wxPrintf( win->GetClassInfo()->GetClassName() );
518 wxPrintf( wxT(" %d %d %d %d\n"), (int)gdk_event->area.x,
519 (int)gdk_event->area.y,
520 (int)gdk_event->area.width,
521 (int)gdk_event->area.height );
522 }
d7fa7eaa 523#endif
1e6feb95 524
b15ed747
RR
525 win->GetUpdateRegion() = wxRegion( gdk_event->region );
526
527 win->GtkSendPaintEvents();
528
2b5f62a0 529
3fc6e5fa 530 // Let parent window draw window less widgets
b15ed747
RR
531 (* GTK_WIDGET_CLASS (pizza_parent_class)->expose_event) (widget, gdk_event);
532#else
533 // This gets called immediately after an expose event
534 // under GTK 1.2 so we collect the calls and wait for
535 // the idle handler to pick things up.
2b5f62a0 536
2bc6945a
RR
537 win->GetUpdateRegion().Union( gdk_event->area.x,
538 gdk_event->area.y,
539 gdk_event->area.width,
540 gdk_event->area.height );
d665f50b
JS
541 win->m_clearRegion.Union( gdk_event->area.x,
542 gdk_event->area.y,
543 gdk_event->area.width,
544 gdk_event->area.height );
0e09f76e 545
8f3e7ecc 546 // Actual redrawing takes place in idle time.
62cb3cd8 547 // win->GtkUpdate();
0fc5dbf5 548#endif
f7a11f8c 549
3fc6e5fa 550 return FALSE;
b6fa52db
RR
551}
552
553//-----------------------------------------------------------------------------
554// "event" of m_wxwindow
555//-----------------------------------------------------------------------------
556
beab25bd
RR
557// GTK thinks it is clever and filters out a certain amount of "unneeded"
558// expose events. We need them, of course, so we override the main event
559// procedure in GtkWidget by giving our own handler for all system events.
560// There, we look for expose events ourselves whereas all other events are
561// handled normally.
b6fa52db 562
1e6feb95
VZ
563gint gtk_window_event_event_callback( GtkWidget *widget,
564 GdkEventExpose *event,
565 wxWindow *win )
b6fa52db
RR
566{
567 if (event->type == GDK_EXPOSE)
568 {
569 gint ret = gtk_window_expose_callback( widget, event, win );
570 return ret;
571 }
572
b6fa52db 573 return FALSE;
362c6693 574}
c801d85f
KB
575
576//-----------------------------------------------------------------------------
034be888 577// "draw" of m_wxwindow
2f2aa628 578//-----------------------------------------------------------------------------
c801d85f 579
67d78217
RR
580#ifndef __WXGTK20__
581
beab25bd 582// This callback is a complete replacement of the gtk_pizza_draw() function,
67d78217 583// which is disabled.
b6fa52db 584
1e6feb95
VZ
585static void gtk_window_draw_callback( GtkWidget *widget,
586 GdkRectangle *rect,
587 wxWindow *win )
47d67540 588{
3ac8d3bc
RR
589 DEBUG_MAIN_THREAD
590
c50f1fb9 591 if (g_isIdle)
a2053b27 592 wxapp_install_idle_handler();
2d19ad25 593
beab25bd
RR
594 // The wxNO_FULL_REPAINT_ON_RESIZE flag only works if
595 // there are no child windows.
e22454be
RR
596 if ((win->HasFlag(wxNO_FULL_REPAINT_ON_RESIZE)) &&
597 (win->GetChildren().GetCount() == 0))
598 {
599 return;
600 }
2d19ad25 601
d7fa7eaa
RR
602#if 0
603 if (win->GetName())
3d2d8da1
RR
604 {
605 wxPrintf( wxT("OnDraw from ") );
606 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
607 wxPrintf( win->GetClassInfo()->GetClassName() );
608 wxPrintf( wxT(" %d %d %d %d\n"), (int)rect->x,
609 (int)rect->y,
610 (int)rect->width,
611 (int)rect->height );
612 }
d7fa7eaa 613#endif
1e6feb95 614
2e09dc2d 615#ifndef __WXUNIVERSAL__
b420fb6a 616 GtkPizza *pizza = GTK_PIZZA (widget);
2bc6945a 617
9cc7121f 618 if (win->GetThemeEnabled())
789dbcd4 619 {
9cc7121f
RR
620 wxWindow *parent = win->GetParent();
621 while (parent && !parent->IsTopLevel())
789dbcd4 622 parent = parent->GetParent();
9cc7121f
RR
623 if (!parent)
624 parent = win;
625
90350682
VZ
626 gtk_paint_flat_box (parent->m_widget->style,
627 pizza->bin_window,
628 GTK_STATE_NORMAL,
629 GTK_SHADOW_NONE,
630 rect,
631 parent->m_widget,
632 (char *)"base",
633 0, 0, -1, -1);
789dbcd4 634 }
2e09dc2d 635#endif
1e6feb95 636
62cb3cd8 637 win->m_clearRegion.Union( rect->x, rect->y, rect->width, rect->height );
9cc7121f 638 win->GetUpdateRegion().Union( rect->x, rect->y, rect->width, rect->height );
148cd9b6 639
62cb3cd8 640 // Update immediately, not in idle time.
010afced 641 win->GtkUpdate();
1e6feb95 642
8f3e7ecc 643#ifndef __WXUNIVERSAL__
beab25bd 644 // Redraw child widgets
e22454be
RR
645 GList *children = pizza->children;
646 while (children)
647 {
beab25bd
RR
648 GtkPizzaChild *child = (GtkPizzaChild*) children->data;
649 children = children->next;
b420fb6a 650
beab25bd
RR
651 GdkRectangle child_area;
652 if (gtk_widget_intersect (child->widget, rect, &child_area))
653 {
654 gtk_widget_draw (child->widget, &child_area /* (GdkRectangle*) NULL*/ );
655 }
e22454be 656 }
8f3e7ecc 657#endif
362c6693 658}
c801d85f 659
67d78217
RR
660#endif
661
c801d85f 662//-----------------------------------------------------------------------------
b292e2f5 663// "key_press_event" from any window
c801d85f 664//-----------------------------------------------------------------------------
c801d85f 665
74710601
VZ
666// set WXTRACE to this to see the key event codes on the console
667#define TRACE_KEYS _T("keyevent")
f17393f1 668
1c6896d7
VZ
669// translates an X key symbol to WXK_XXX value
670//
671// if isChar is true it means that the value returned will be used for EVT_CHAR
672// event and then we choose the logical WXK_XXX, i.e. '/' for GDK_KP_Divide,
673// for example, while if it is false it means that the value is going to be
674// used for KEY_DOWN/UP events and then we translate GDK_KP_Divide to
675// WXK_NUMPAD_DIVIDE
676static long wxTranslateKeySymToWXKey(KeySym keysym, bool isChar)
677{
678 long key_code;
679
680 switch ( keysym )
681 {
682 // Shift, Control and Alt don't generate the CHAR events at all
683 case GDK_Shift_L:
684 case GDK_Shift_R:
685 key_code = isChar ? 0 : WXK_SHIFT;
686 break;
687 case GDK_Control_L:
688 case GDK_Control_R:
689 key_code = isChar ? 0 : WXK_CONTROL;
690 break;
691 case GDK_Meta_L:
692 case GDK_Meta_R:
693 case GDK_Alt_L:
694 case GDK_Alt_R:
695 case GDK_Super_L:
696 case GDK_Super_R:
697 key_code = isChar ? 0 : WXK_ALT;
698 break;
699
700 // neither do the toggle modifies
701 case GDK_Scroll_Lock:
702 key_code = isChar ? 0 : WXK_SCROLL;
703 break;
704
705 case GDK_Caps_Lock:
706 key_code = isChar ? 0 : WXK_CAPITAL;
707 break;
708
709 case GDK_Num_Lock:
710 key_code = isChar ? 0 : WXK_NUMLOCK;
711 break;
712
713
714 // various other special keys
715 case GDK_Menu:
716 key_code = WXK_MENU;
717 break;
718
719 case GDK_Help:
720 key_code = WXK_HELP;
721 break;
722
723 case GDK_BackSpace:
724 key_code = WXK_BACK;
725 break;
726
727 case GDK_ISO_Left_Tab:
728 case GDK_Tab:
729 key_code = WXK_TAB;
730 break;
731
732 case GDK_Linefeed:
733 case GDK_Return:
734 key_code = WXK_RETURN;
735 break;
736
737 case GDK_Clear:
738 key_code = WXK_CLEAR;
739 break;
740
741 case GDK_Pause:
742 key_code = WXK_PAUSE;
743 break;
744
745 case GDK_Select:
746 key_code = WXK_SELECT;
747 break;
748
749 case GDK_Print:
750 key_code = WXK_PRINT;
751 break;
752
753 case GDK_Execute:
754 key_code = WXK_EXECUTE;
755 break;
756
757 case GDK_Escape:
758 key_code = WXK_ESCAPE;
759 break;
760
761 // cursor and other extended keyboard keys
762 case GDK_Delete:
763 key_code = WXK_DELETE;
764 break;
765
766 case GDK_Home:
767 key_code = WXK_HOME;
768 break;
769
770 case GDK_Left:
771 key_code = WXK_LEFT;
772 break;
773
774 case GDK_Up:
775 key_code = WXK_UP;
776 break;
777
778 case GDK_Right:
779 key_code = WXK_RIGHT;
780 break;
781
782 case GDK_Down:
783 key_code = WXK_DOWN;
784 break;
785
786 case GDK_Prior: // == GDK_Page_Up
787 key_code = WXK_PRIOR;
788 break;
789
790 case GDK_Next: // == GDK_Page_Down
791 key_code = WXK_NEXT;
792 break;
793
794 case GDK_End:
795 key_code = WXK_END;
796 break;
797
798 case GDK_Begin:
799 key_code = WXK_HOME;
800 break;
801
802 case GDK_Insert:
803 key_code = WXK_INSERT;
804 break;
805
806
807 // numpad keys
808 case GDK_KP_0:
809 case GDK_KP_1:
810 case GDK_KP_2:
811 case GDK_KP_3:
812 case GDK_KP_4:
813 case GDK_KP_5:
814 case GDK_KP_6:
815 case GDK_KP_7:
816 case GDK_KP_8:
817 case GDK_KP_9:
818 key_code = (isChar ? '0' : WXK_NUMPAD0) + keysym - GDK_KP_0;
819 break;
820
821 case GDK_KP_Space:
822 key_code = isChar ? ' ' : WXK_NUMPAD_SPACE;
823 break;
824
825 case GDK_KP_Tab:
826 key_code = isChar ? WXK_TAB : WXK_NUMPAD_TAB;
827 break;
828
829 case GDK_KP_Enter:
830 key_code = isChar ? WXK_RETURN : WXK_NUMPAD_ENTER;
831 break;
832
833 case GDK_KP_F1:
834 key_code = isChar ? WXK_F1 : WXK_NUMPAD_F1;
835 break;
836
837 case GDK_KP_F2:
838 key_code = isChar ? WXK_F2 : WXK_NUMPAD_F2;
839 break;
840
841 case GDK_KP_F3:
842 key_code = isChar ? WXK_F3 : WXK_NUMPAD_F3;
843 break;
844
845 case GDK_KP_F4:
846 key_code = isChar ? WXK_F4 : WXK_NUMPAD_F4;
847 break;
848
849 case GDK_KP_Home:
850 key_code = isChar ? WXK_HOME : WXK_NUMPAD_HOME;
851 break;
852
853 case GDK_KP_Left:
854 key_code = isChar ? WXK_LEFT : WXK_NUMPAD_LEFT;
855 break;
856
857 case GDK_KP_Up:
858 key_code = isChar ? WXK_UP : WXK_NUMPAD_UP;
859 break;
860
861 case GDK_KP_Right:
862 key_code = isChar ? WXK_RIGHT : WXK_NUMPAD_RIGHT;
863 break;
864
865 case GDK_KP_Down:
866 key_code = isChar ? WXK_DOWN : WXK_NUMPAD_DOWN;
867 break;
868
869 case GDK_KP_Prior: // == GDK_KP_Page_Up
870 key_code = isChar ? WXK_PRIOR : WXK_NUMPAD_PRIOR;
871 break;
872
873 case GDK_KP_Next: // == GDK_KP_Page_Down
874 key_code = isChar ? WXK_NEXT : WXK_NUMPAD_NEXT;
875 break;
876
877 case GDK_KP_End:
878 key_code = isChar ? WXK_END : WXK_NUMPAD_END;
879 break;
880
881 case GDK_KP_Begin:
882 key_code = isChar ? WXK_HOME : WXK_NUMPAD_BEGIN;
883 break;
884
885 case GDK_KP_Insert:
886 key_code = isChar ? WXK_INSERT : WXK_NUMPAD_INSERT;
887 break;
888
889 case GDK_KP_Delete:
890 key_code = isChar ? WXK_DELETE : WXK_NUMPAD_DELETE;
891 break;
892
893 case GDK_KP_Equal:
894 key_code = isChar ? '=' : WXK_NUMPAD_EQUAL;
895 break;
896
897 case GDK_KP_Multiply:
898 key_code = isChar ? '*' : WXK_NUMPAD_MULTIPLY;
899 break;
900
901 case GDK_KP_Add:
902 key_code = isChar ? '+' : WXK_NUMPAD_ADD;
903 break;
904
905 case GDK_KP_Separator:
906 // FIXME: what is this?
907 key_code = isChar ? '.' : WXK_NUMPAD_SEPARATOR;
908 break;
909
910 case GDK_KP_Subtract:
911 key_code = isChar ? '-' : WXK_NUMPAD_SUBTRACT;
912 break;
913
914 case GDK_KP_Decimal:
915 key_code = isChar ? '.' : WXK_NUMPAD_DECIMAL;
916 break;
917
918 case GDK_KP_Divide:
919 key_code = isChar ? '/' : WXK_NUMPAD_DIVIDE;
920 break;
921
922
923 // function keys
924 case GDK_F1:
925 case GDK_F2:
926 case GDK_F3:
927 case GDK_F4:
928 case GDK_F5:
929 case GDK_F6:
930 case GDK_F7:
931 case GDK_F8:
932 case GDK_F9:
933 case GDK_F10:
934 case GDK_F11:
935 case GDK_F12:
936 key_code = WXK_F1 + keysym - GDK_F1;
937 break;
938
939 default:
940 key_code = 0;
941 }
942
943 return key_code;
944}
945
946static inline bool wxIsAsciiKeysym(KeySym ks)
947{
948 return ks < 256;
949}
950
74710601
VZ
951static bool
952wxTranslateGTKKeyEventToWx(wxKeyEvent& event,
953 wxWindowGTK *win,
954 GdkEventKey *gdk_event)
47d67540 955{
1c6896d7
VZ
956 // VZ: it seems that GDK_KEY_RELEASE event doesn't set event->string
957 // but only event->keyval which is quite useless to us, so remember
958 // the last character from GDK_KEY_PRESS and reuse it as last resort
959 //
960 // NB: should be MT-safe as we're always called from the main thread only
961 static struct
962 {
963 KeySym keysym;
964 long keycode;
965 } s_lastKeyPress = { 0, 0 };
966
967 KeySym keysym = gdk_event->keyval;
0a62b197 968
ada7d2c0 969 wxLogTrace(TRACE_KEYS, _T("Key %s event: keysym = %ld"),
0a62b197
VZ
970 event.GetEventType() == wxEVT_KEY_UP ? _T("release")
971 : _T("press"),
972 keysym);
973
1c6896d7
VZ
974 long key_code = wxTranslateKeySymToWXKey(keysym, FALSE /* !isChar */);
975
976 if ( !key_code )
977 {
978 // do we have the translation or is it a plain ASCII character?
979 if ( (gdk_event->length == 1) || wxIsAsciiKeysym(keysym) )
980 {
981 // we should use keysym if it is ASCII as X does some translations
982 // like "I pressed while Control is down" => "Ctrl-I" == "TAB"
983 // which we don't want here (but which we do use for OnChar())
984 if ( !wxIsAsciiKeysym(keysym) )
985 {
986 keysym = (KeySym)gdk_event->string[0];
987 }
a2053b27 988
1c6896d7
VZ
989 // we want to always get the same key code when the same key is
990 // pressed regardless of the state of the modifies, i.e. on a
991 // standard US keyboard pressing '5' or '%' ('5' key with
992 // Shift) should result in the same key code in OnKeyDown():
993 // '5' (although OnChar() will get either '5' or '%').
994 //
995 // to do it we first translate keysym to keycode (== scan code)
996 // and then back but always using the lower register
997 Display *dpy = (Display *)wxGetDisplay();
998 KeyCode keycode = XKeysymToKeycode(dpy, keysym);
0a62b197
VZ
999
1000 wxLogTrace(TRACE_KEYS, _T("\t-> keycode %d"), keycode);
1001
1c6896d7
VZ
1002 KeySym keysymNormalized = XKeycodeToKeysym(dpy, keycode, 0);
1003
1004 // use the normalized, i.e. lower register, keysym if we've
1005 // got one
1006 key_code = keysymNormalized ? keysymNormalized : keysym;
1007
1008 // as explained above, we want to have lower register key codes
1009 // normally but for the letter keys we want to have the upper ones
1010 //
1011 // NB: don't use XConvertCase() here, we want to do it for letters
1012 // only
1013 key_code = toupper(key_code);
1014 }
1015 else // non ASCII key, what to do?
1016 {
1017 // by default, ignore it
1018 key_code = 0;
1019
1020 // but if we have cached information from the last KEY_PRESS
1021 if ( gdk_event->type == GDK_KEY_RELEASE )
1022 {
1023 // then reuse it
1024 if ( keysym == s_lastKeyPress.keysym )
1025 {
1026 key_code = s_lastKeyPress.keycode;
1027 }
1028 }
1029 }
1030
1031 if ( gdk_event->type == GDK_KEY_PRESS )
1032 {
1033 // remember it to be reused for KEY_UP event later
1034 s_lastKeyPress.keysym = keysym;
1035 s_lastKeyPress.keycode = key_code;
1036 }
1037 }
1038
ada7d2c0 1039 wxLogTrace(TRACE_KEYS, _T("\t-> wxKeyCode %ld"), key_code);
c801d85f 1040
74710601 1041 // sending unknown key events doesn't really make sense
1c6896d7 1042 if ( !key_code )
74710601 1043 return FALSE;
3d6f7261 1044
1c6896d7 1045 // now fill all the other fields
b0e813a0
RR
1046 int x = 0;
1047 int y = 0;
1048 GdkModifierType state;
f17393f1
VZ
1049 if (gdk_event->window)
1050 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
47d67540 1051
d1367c3d 1052 event.SetTimestamp( gdk_event->time );
74710601
VZ
1053 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0;
1054 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0;
1055 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0;
1056 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK) != 0;
f5e27805 1057 event.m_keyCode = key_code;
b0e813a0 1058 event.m_scanCode = gdk_event->keyval;
9c7df356
VZ
1059 event.m_rawCode = (wxUint32) gdk_event->keyval;
1060 event.m_rawFlags = 0;
b0e813a0
RR
1061 event.m_x = x;
1062 event.m_y = y;
f5e27805 1063 event.SetEventObject( win );
74710601
VZ
1064
1065 return TRUE;
1066}
1067
2b5f62a0 1068#ifndef __WXGTK20__
74710601
VZ
1069static gint gtk_window_key_press_callback( GtkWidget *widget,
1070 GdkEventKey *gdk_event,
1071 wxWindow *win )
1072{
1073 DEBUG_MAIN_THREAD
1074
1075 if (g_isIdle)
1076 wxapp_install_idle_handler();
1077
1c6896d7
VZ
1078 if (!win->m_hasVMT)
1079 return FALSE;
1080 if (g_blockEventsOnDrag)
1081 return FALSE;
74710601 1082
1ec3a984
RR
1083
1084 wxKeyEvent event( wxEVT_KEY_DOWN );
74710601
VZ
1085 if ( !wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
1086 {
1c6896d7 1087 // unknown key pressed, ignore (the event would be useless anyhow)
74710601
VZ
1088 return FALSE;
1089 }
1090
1ec3a984
RR
1091 // Emit KEY_DOWN event
1092 bool ret = win->GetEventHandler()->ProcessEvent( event );
c50f1fb9 1093
88ac883a 1094#if wxUSE_ACCEL
f5e27805 1095 if (!ret)
47d67540 1096 {
1e6feb95 1097 wxWindowGTK *ancestor = win;
f5e27805
RR
1098 while (ancestor)
1099 {
1100 int command = ancestor->GetAcceleratorTable()->GetCommand( event );
1101 if (command != -1)
1102 {
1103 wxCommandEvent command_event( wxEVT_COMMAND_MENU_SELECTED, command );
1104 ret = ancestor->GetEventHandler()->ProcessEvent( command_event );
1105 break;
1106 }
f6bcfd97 1107 if (ancestor->IsTopLevel())
b02da6b1 1108 break;
f5e27805
RR
1109 ancestor = ancestor->GetParent();
1110 }
bcf1fa6b 1111 }
88ac883a 1112#endif // wxUSE_ACCEL
3017f78d 1113
1ec3a984
RR
1114 // Only send wxEVT_CHAR event if not processed yet. Thus, ALT-x
1115 // will only be sent if it is not in an accelerator table.
2dde25b9 1116 if (!ret)
d728dd40 1117 {
1ec3a984 1118 // Find key code for EVT_CHAR and EVT_CHAR_HOOK events
1c6896d7
VZ
1119 KeySym keysym = gdk_event->keyval;
1120 long key_code = wxTranslateKeySymToWXKey(keysym, TRUE /* isChar */);
1121 if ( !key_code )
1122 {
1123 if ( gdk_event->length == 1 )
1124 {
486ac244 1125 key_code = (unsigned char)gdk_event->string[0];
1c6896d7 1126 }
486ac244 1127 else if ( wxIsAsciiKeysym(keysym) )
1c6896d7
VZ
1128 {
1129 // ASCII key
486ac244 1130 key_code = (unsigned char)keysym;
1c6896d7
VZ
1131 }
1132 }
f17393f1
VZ
1133
1134 if ( key_code )
1135 {
74710601 1136 wxLogTrace(TRACE_KEYS, _T("Char event: %ld"), key_code);
f17393f1 1137
f17393f1 1138 event.m_keyCode = key_code;
2b5f62a0 1139
1ec3a984
RR
1140 // Implement OnCharHook by checking ancesteror top level windows
1141 wxWindow *parent = win;
1142 while (parent && !parent->IsTopLevel())
1143 parent = parent->GetParent();
1144 if (parent)
1145 {
1146 event.SetEventType( wxEVT_CHAR_HOOK );
1147 ret = parent->GetEventHandler()->ProcessEvent( event );
1148 }
f17393f1 1149
1ec3a984
RR
1150 if (!ret)
1151 {
1152 event.SetEventType(wxEVT_CHAR);
1153 ret = win->GetEventHandler()->ProcessEvent( event );
1154 }
f17393f1 1155 }
d728dd40
KB
1156 }
1157
1ec3a984 1158 // win is a control: tab can be propagated up
f17393f1 1159 if ( !ret &&
5664fc32 1160 ((gdk_event->keyval == GDK_Tab) || (gdk_event->keyval == GDK_ISO_Left_Tab)) &&
f6bcfd97
BP
1161// VZ: testing for wxTE_PROCESS_TAB shouldn't be done here the control may
1162// have this style, yet choose not to process this particular TAB in which
1163// case TAB must still work as a navigational character
1164#if 0
f17393f1 1165 !win->HasFlag(wxTE_PROCESS_TAB) &&
f6bcfd97 1166#endif // 0
f17393f1 1167 win->GetParent() && (win->GetParent()->HasFlag( wxTAB_TRAVERSAL)) )
b292e2f5
RR
1168 {
1169 wxNavigationKeyEvent new_event;
8253c7fd 1170 new_event.SetEventObject( win->GetParent() );
1ec3a984 1171 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
5664fc32 1172 new_event.SetDirection( (gdk_event->keyval == GDK_Tab) );
1ec3a984 1173 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
b98d804b 1174 new_event.SetWindowChange( (gdk_event->state & GDK_CONTROL_MASK) );
b292e2f5 1175 new_event.SetCurrentFocus( win );
8253c7fd 1176 ret = win->GetParent()->GetEventHandler()->ProcessEvent( new_event );
b292e2f5 1177 }
ff8bfdbb 1178
1ec3a984 1179 // generate wxID_CANCEL if <esc> has been pressed (typically in dialogs)
f17393f1 1180 if ( !ret &&
b98d804b
RR
1181 (gdk_event->keyval == GDK_Escape) )
1182 {
6ce16bdb
VZ
1183 // however only do it if we have a Cancel button in the dialog,
1184 // otherwise the user code may get confused by the events from a
1185 // non-existing button and, worse, a wxButton might get button event
1186 // from another button which is not really expected
1187 wxWindow *winForCancel = win,
1188 *btnCancel = NULL;
1189 while ( winForCancel )
1190 {
1191 btnCancel = winForCancel->FindWindow(wxID_CANCEL);
1192 if ( btnCancel )
1193 {
1194 // found a cancel button
1195 break;
1196 }
1197
1198 if ( winForCancel->IsTopLevel() )
1199 {
1200 // no need to look further
1201 break;
1202 }
1203
1204 // maybe our parent has a cancel button?
1205 winForCancel = winForCancel->GetParent();
1206 }
1207
1208 if ( btnCancel )
1209 {
1210 wxCommandEvent event(wxEVT_COMMAND_BUTTON_CLICKED, wxID_CANCEL);
1211 event.SetEventObject(btnCancel);
1212 ret = btnCancel->GetEventHandler()->ProcessEvent(event);
1213 }
b98d804b 1214 }
c50f1fb9 1215
2b5f62a0 1216 if (ret)
801aa178 1217 {
2b5f62a0
VZ
1218 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "key_press_event" );
1219 return TRUE;
1220 }
1221
1222 return FALSE;
1223}
1224#endif
1225
1226#ifdef __WXGTK20__
1227static gint gtk_window_key_press_callback( GtkWidget *widget,
1228 GdkEventKey *gdk_event,
1229 wxWindow *win )
1230{
1231 if (g_isIdle)
1232 wxapp_install_idle_handler();
1233
1234 if (!win->m_hasVMT)
1235 return FALSE;
1236 if (g_blockEventsOnDrag)
1237 return FALSE;
1238
1239 bool ret = FALSE;
1240 bool dont_use_IM = FALSE;
1241
1242 wxKeyEvent event( wxEVT_KEY_DOWN );
1243 long keycode = wxTranslateKeySymToWXKey( gdk_event->keyval, FALSE );
1244 if (keycode)
1245 {
1246 // We were able to decode the key press without
1247 // any input method, so don't use it.
1248 dont_use_IM = TRUE;
1249
1250 // now fill all the other fields
1251 int x = 0;
1252 int y = 0;
1253 GdkModifierType state;
1254 if (gdk_event->window)
1255 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
1256
1257 event.SetTimestamp( gdk_event->time );
1258 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK) != 0;
1259 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK) != 0;
1260 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK) != 0;
1261 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK) != 0;
1262 event.m_keyCode = keycode;
1263 event.m_scanCode = gdk_event->keyval;
1264 event.m_rawCode = (wxUint32) gdk_event->keyval;
1265 event.m_rawFlags = 0;
1266 event.m_x = x;
1267 event.m_y = y;
1268 event.SetEventObject( win );
1269
1270 // send key down event
1271 ret = win->GetEventHandler()->ProcessEvent( event );
1272
1273 if (!ret)
801aa178 1274 {
2b5f62a0
VZ
1275 // Implement OnCharHook by checking ancesteror top level windows
1276 wxWindow *parent = win;
1277 while (parent && !parent->IsTopLevel())
1278 parent = parent->GetParent();
1279 if (parent)
c50f1fb9 1280 {
2b5f62a0
VZ
1281 event.SetEventType( wxEVT_CHAR_HOOK );
1282 ret = parent->GetEventHandler()->ProcessEvent( event );
1283 }
1284
1285 if (!ret)
1286 {
1287 event.SetEventType(wxEVT_CHAR);
1288 ret = win->GetEventHandler()->ProcessEvent( event );
1289 }
1290 }
1291
1292 // win is a control: tab can be propagated up
1293 if ( !ret &&
1294 ((gdk_event->keyval == GDK_Tab) || (gdk_event->keyval == GDK_ISO_Left_Tab)) &&
1295 win->GetParent() && (win->GetParent()->HasFlag( wxTAB_TRAVERSAL)) )
1296 {
1297 wxNavigationKeyEvent new_event;
1298 new_event.SetEventObject( win->GetParent() );
1299 // GDK reports GDK_ISO_Left_Tab for SHIFT-TAB
1300 new_event.SetDirection( (gdk_event->keyval == GDK_Tab) );
1301 // CTRL-TAB changes the (parent) window, i.e. switch notebook page
1302 new_event.SetWindowChange( (gdk_event->state & GDK_CONTROL_MASK) );
1303 new_event.SetCurrentFocus( win );
1304 ret = win->GetParent()->GetEventHandler()->ProcessEvent( new_event );
1305 }
1306
1307 if ( !ret &&
1308 (gdk_event->keyval == GDK_Escape) )
1309 {
1310 wxWindow *winForCancel = win, *btnCancel = NULL;
1311 while ( winForCancel )
1312 {
1313 btnCancel = winForCancel->FindWindow(wxID_CANCEL);
1314 if ( btnCancel ) break;
1315
1316 if ( winForCancel->IsTopLevel() ) break;
1317
1318 winForCancel = winForCancel->GetParent();
1319 }
1320
1321 if ( btnCancel )
1322 {
1323 wxCommandEvent event(wxEVT_COMMAND_BUTTON_CLICKED, wxID_CANCEL);
1324 event.SetEventObject(btnCancel);
1325 ret = btnCancel->GetEventHandler()->ProcessEvent(event);
c50f1fb9 1326 }
801aa178
RR
1327 }
1328 }
ca298c88 1329
f5e27805
RR
1330 if (ret)
1331 {
b292e2f5 1332 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "key_press_event" );
f03fc89f 1333 return TRUE;
f5e27805 1334 }
2b5f62a0
VZ
1335
1336 if (!dont_use_IM && win->m_imContext)
1337 {
1338 // In GTK 2.0, we need to hand over the key
1339 // event to an input method and the IM will
1340 // emit a "commit" event containing the
1341 // actual utf8 character.
1342 gtk_im_context_filter_keypress ( (GtkIMContext*) win->m_imContext, gdk_event );
1343
1344 return TRUE;
1345 }
1346
034be888 1347 return FALSE;
362c6693 1348}
c801d85f 1349
2b5f62a0
VZ
1350static void gtk_wxwindow_commit_cb (GtkIMContext *context,
1351 const gchar *str,
1352 wxWindow *window)
1353{
1354 bool ret = FALSE;
1355
1356 wxKeyEvent event( wxEVT_KEY_DOWN );
1357
1358#if wxUSE_UNICODE
1359 event.m_uniChar = g_utf8_get_char( str );
1360
1361 // Backward compatible for ISO-8859
1362 if (event.m_uniChar < 256)
1363 event.m_keyCode = event.m_uniChar;
1364#else
1365 gunichar uniChar = g_utf8_get_char( str );
1366 // We cannot handle Unicode in non-Unicode mode
1367 if (uniChar > 255) return;
1368
1369 event.m_keyCode = uniChar;
1370#endif
1371
1372 if (!ret)
1373 {
1374 // Implement OnCharHook by checking ancesteror top level windows
1375 wxWindow *parent = window;
1376 while (parent && !parent->IsTopLevel())
1377 parent = parent->GetParent();
1378 if (parent)
1379 {
1380 event.SetEventType( wxEVT_CHAR_HOOK );
1381 ret = parent->GetEventHandler()->ProcessEvent( event );
1382 }
1383
1384 if (!ret)
1385 {
1386 event.SetEventType(wxEVT_CHAR);
1387 ret = window->GetEventHandler()->ProcessEvent( event );
1388 }
1389 }
1390}
1391#endif
1392
1393
b666df2c
RR
1394//-----------------------------------------------------------------------------
1395// "key_release_event" from any window
1396//-----------------------------------------------------------------------------
1397
74710601
VZ
1398static gint gtk_window_key_release_callback( GtkWidget *widget,
1399 GdkEventKey *gdk_event,
1400 wxWindowGTK *win )
b666df2c 1401{
3ac8d3bc
RR
1402 DEBUG_MAIN_THREAD
1403
c50f1fb9 1404 if (g_isIdle)
a2053b27
RR
1405 wxapp_install_idle_handler();
1406
74710601
VZ
1407 if (!win->m_hasVMT)
1408 return FALSE;
b666df2c 1409
74710601
VZ
1410 if (g_blockEventsOnDrag)
1411 return FALSE;
b666df2c
RR
1412
1413 wxKeyEvent event( wxEVT_KEY_UP );
74710601 1414 if ( !wxTranslateGTKKeyEventToWx(event, win, gdk_event) )
b666df2c 1415 {
74710601
VZ
1416 // unknown key pressed, ignore (the event would be useless anyhow
1417 return FALSE;
b666df2c
RR
1418 }
1419
74710601
VZ
1420 if ( !win->GetEventHandler()->ProcessEvent( event ) )
1421 return FALSE;
1422
1423 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "key_release_event" );
1424 return TRUE;
b666df2c
RR
1425}
1426
c5f9d156
VS
1427// ============================================================================
1428// the mouse events
1429// ============================================================================
1430
d1f2ac45
VZ
1431// ----------------------------------------------------------------------------
1432// mouse event processing helpers
1433// ----------------------------------------------------------------------------
1434
c5f9d156 1435// init wxMouseEvent with the info from gdk_event
edc1d330
VZ
1436//
1437// NB: this has to be a macro as gdk_event type is different for different
1438// events we're used with
1439#define InitMouseEvent(/* wxWindowGTK * */ win, \
1440 /* wxMouseEvent& */ event, \
1441 /* GdkEventXXX * */ gdk_event) \
1442{ \
1443 event.SetTimestamp( gdk_event->time ); \
1444 event.m_shiftDown = (gdk_event->state & GDK_SHIFT_MASK); \
1445 event.m_controlDown = (gdk_event->state & GDK_CONTROL_MASK); \
1446 event.m_altDown = (gdk_event->state & GDK_MOD1_MASK); \
1447 event.m_metaDown = (gdk_event->state & GDK_MOD2_MASK); \
1448 event.m_leftDown = (gdk_event->state & GDK_BUTTON1_MASK); \
1449 event.m_middleDown = (gdk_event->state & GDK_BUTTON2_MASK); \
1450 event.m_rightDown = (gdk_event->state & GDK_BUTTON3_MASK); \
2b5f62a0
VZ
1451 if (event.GetEventType()==wxEVT_MOUSEWHEEL) \
1452 { \
1453 if (((GdkEventButton*)gdk_event)->button == 4) \
1454 event.m_wheelRotation = 120; \
1455 else if (((GdkEventButton*)gdk_event)->button == 5) \
1456 event.m_wheelRotation = -120; \
1457 } \
edc1d330
VZ
1458 \
1459 wxPoint pt = win->GetClientAreaOrigin(); \
1460 event.m_x = (wxCoord)gdk_event->x - pt.x; \
1461 event.m_y = (wxCoord)gdk_event->y - pt.y; \
1462 \
1463 event.SetEventObject( win ); \
1464 event.SetId( win->GetId() ); \
1465 event.SetTimestamp( gdk_event->time ); \
1466} \
c5f9d156 1467
2daa0ce9
VZ
1468static void AdjustEventButtonState(wxMouseEvent& event)
1469{
1470 // GDK reports the old state of the button for a button press event, but
1471 // for compatibility with MSW and common sense we want m_leftDown be TRUE
1472 // for a LEFT_DOWN event, not FALSE, so we will invert
1473 // left/right/middleDown for the corresponding click events
1e6feb95 1474
1a8caf94
RR
1475 if ((event.GetEventType() == wxEVT_LEFT_DOWN) ||
1476 (event.GetEventType() == wxEVT_LEFT_DCLICK) ||
1477 (event.GetEventType() == wxEVT_LEFT_UP))
1478 {
1479 event.m_leftDown = !event.m_leftDown;
1480 return;
1481 }
1482
1483 if ((event.GetEventType() == wxEVT_MIDDLE_DOWN) ||
1484 (event.GetEventType() == wxEVT_MIDDLE_DCLICK) ||
1485 (event.GetEventType() == wxEVT_MIDDLE_UP))
2daa0ce9 1486 {
1a8caf94
RR
1487 event.m_middleDown = !event.m_middleDown;
1488 return;
1489 }
1490
1491 if ((event.GetEventType() == wxEVT_RIGHT_DOWN) ||
1492 (event.GetEventType() == wxEVT_RIGHT_DCLICK) ||
1493 (event.GetEventType() == wxEVT_RIGHT_UP))
1494 {
1495 event.m_rightDown = !event.m_rightDown;
1496 return;
2daa0ce9
VZ
1497 }
1498}
1499
d1f2ac45
VZ
1500// find the window to send the mouse event too
1501static
1502wxWindowGTK *FindWindowForMouseEvent(wxWindowGTK *win, wxCoord& x, wxCoord& y)
1503{
7d4909b2
RR
1504 wxCoord xx = x;
1505 wxCoord yy = y;
1506
d1f2ac45
VZ
1507 if (win->m_wxwindow)
1508 {
1509 GtkPizza *pizza = GTK_PIZZA(win->m_wxwindow);
7d4909b2
RR
1510 xx += pizza->xoffset;
1511 yy += pizza->yoffset;
d1f2ac45
VZ
1512 }
1513
b1d4dd7a 1514 wxWindowList::Node *node = win->GetChildren().GetFirst();
d1f2ac45
VZ
1515 while (node)
1516 {
b1d4dd7a 1517 wxWindowGTK *child = node->GetData();
d1f2ac45 1518
b1d4dd7a 1519 node = node->GetNext();
d1f2ac45
VZ
1520 if (!child->IsShown())
1521 continue;
1522
1523 if (child->IsTransparentForMouse())
1524 {
1525 // wxStaticBox is transparent in the box itself
1526 int xx1 = child->m_x;
1527 int yy1 = child->m_y;
1528 int xx2 = child->m_x + child->m_width;
1529 int yy2 = child->m_x + child->m_height;
1530
1531 // left
7d4909b2 1532 if (((xx >= xx1) && (xx <= xx1+10) && (yy >= yy1) && (yy <= yy2)) ||
d1f2ac45 1533 // right
7d4909b2 1534 ((xx >= xx2-10) && (xx <= xx2) && (yy >= yy1) && (yy <= yy2)) ||
d1f2ac45 1535 // top
7d4909b2 1536 ((xx >= xx1) && (xx <= xx2) && (yy >= yy1) && (yy <= yy1+10)) ||
d1f2ac45 1537 // bottom
7d4909b2 1538 ((xx >= xx1) && (xx <= xx2) && (yy >= yy2-1) && (yy <= yy2)))
d1f2ac45
VZ
1539 {
1540 win = child;
1541 x -= child->m_x;
1542 y -= child->m_y;
1543 break;
1544 }
1545
1546 }
1547 else
1548 {
1549 if ((child->m_wxwindow == (GtkWidget*) NULL) &&
7d4909b2
RR
1550 (child->m_x <= xx) &&
1551 (child->m_y <= yy) &&
af3653dd
RR
1552 (child->m_x+child->m_width >= xx) &&
1553 (child->m_y+child->m_height >= yy))
d1f2ac45
VZ
1554 {
1555 win = child;
1556 x -= child->m_x;
1557 y -= child->m_y;
1558 break;
1559 }
1560 }
1561 }
1562
1563 return win;
1564}
1565
c801d85f 1566//-----------------------------------------------------------------------------
2f2aa628
RR
1567// "button_press_event"
1568//-----------------------------------------------------------------------------
c801d85f 1569
d1f2ac45
VZ
1570static gint gtk_window_button_press_callback( GtkWidget *widget,
1571 GdkEventButton *gdk_event,
1572 wxWindowGTK *win )
903f689b 1573{
3ac8d3bc
RR
1574 DEBUG_MAIN_THREAD
1575
c50f1fb9 1576 if (g_isIdle)
a2053b27
RR
1577 wxapp_install_idle_handler();
1578
1579/*
223d09f6 1580 wxPrintf( wxT("1) OnButtonPress from ") );
a2053b27
RR
1581 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1582 wxPrintf( win->GetClassInfo()->GetClassName() );
223d09f6 1583 wxPrintf( wxT(".\n") );
a2053b27 1584*/
a2053b27 1585 if (!win->m_hasVMT) return FALSE;
f5e27805 1586 if (g_blockEventsOnDrag) return TRUE;
76ed8f8d 1587 if (g_blockEventsOnScroll) return TRUE;
c801d85f 1588
034be888
RR
1589 if (!win->IsOwnGtkWindow( gdk_event->window )) return FALSE;
1590
afbe906a 1591 if (win->m_wxwindow && (g_focusWindow != win) && win->AcceptsFocus())
c801d85f 1592 {
afbe906a 1593 gtk_widget_grab_focus( win->m_wxwindow );
c801d85f 1594/*
afbe906a
RR
1595 wxPrintf( wxT("GrabFocus from ") );
1596 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1597 wxPrintf( win->GetClassInfo()->GetClassName() );
1598 wxPrintf( wxT(".\n") );
c801d85f 1599*/
362c6693 1600 }
47d67540 1601
2b5f62a0
VZ
1602 // GDK sends surplus button down event
1603 // before a double click event. We
1604 // need to filter these out.
1605 if (gdk_event->type == GDK_BUTTON_PRESS)
1606 {
1607 GdkEvent *peek_event = gdk_event_peek();
1608 if (peek_event)
1609 {
1610 if (peek_event->type == GDK_2BUTTON_PRESS)
1611 {
1612 gdk_event_free( peek_event );
1613 return TRUE;
1614 }
1615 else
1616 {
1617 gdk_event_free( peek_event );
1618 }
1619 }
1620 }
1621
2daa0ce9 1622 wxEventType event_type = wxEVT_NULL;
47d67540 1623
f5e27805 1624 if (gdk_event->button == 1)
c801d85f 1625 {
f5e27805
RR
1626 switch (gdk_event->type)
1627 {
1628 case GDK_BUTTON_PRESS: event_type = wxEVT_LEFT_DOWN; break;
1629 case GDK_2BUTTON_PRESS: event_type = wxEVT_LEFT_DCLICK; break;
1630 default: break;
1631 }
362c6693 1632 }
f5e27805 1633 else if (gdk_event->button == 2)
c801d85f 1634 {
f5e27805
RR
1635 switch (gdk_event->type)
1636 {
1637 case GDK_BUTTON_PRESS: event_type = wxEVT_MIDDLE_DOWN; break;
1638 case GDK_2BUTTON_PRESS: event_type = wxEVT_MIDDLE_DCLICK; break;
1639 default: break;
1640 }
362c6693 1641 }
f5e27805 1642 else if (gdk_event->button == 3)
c801d85f 1643 {
f5e27805
RR
1644 switch (gdk_event->type)
1645 {
1646 case GDK_BUTTON_PRESS: event_type = wxEVT_RIGHT_DOWN; break;
1647 case GDK_2BUTTON_PRESS: event_type = wxEVT_RIGHT_DCLICK; break;
1648 default: break;
1649 }
362c6693 1650 }
2b5f62a0
VZ
1651 else if (gdk_event->button == 4)
1652 {
1653 switch (gdk_event->type)
1654 {
1655 case GDK_BUTTON_PRESS: event_type = wxEVT_MOUSEWHEEL; break;
1656 default: break;
1657 }
1658 }
1659 else if (gdk_event->button == 5)
1660 {
1661 switch (gdk_event->type)
1662 {
1663 case GDK_BUTTON_PRESS: event_type = wxEVT_MOUSEWHEEL; break;
1664 default: break;
1665 }
1666 }
47d67540 1667
2daa0ce9
VZ
1668 if ( event_type == wxEVT_NULL )
1669 {
1670 // unknown mouse button or click type
1671 return FALSE;
1672 }
1673
f5e27805 1674 wxMouseEvent event( event_type );
c5f9d156 1675 InitMouseEvent( win, event, gdk_event );
47d67540 1676
2daa0ce9 1677 AdjustEventButtonState(event);
94633ad9 1678
3ae4c570
VZ
1679 // wxListBox actually get mouse events from the item, so we need to give it
1680 // a chance to correct this
1681 win->FixUpMouseEvent(widget, event.m_x, event.m_y);
2daa0ce9 1682
d1f2ac45
VZ
1683 // find the correct window to send the event too: it may be a different one
1684 // from the one which got it at GTK+ level because some control don't have
1685 // their own X window and thus cannot get any events.
1686 if ( !g_captureWindow )
1687 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
ff8bfdbb 1688
f5e27805 1689 gs_timeLastClick = gdk_event->time;
47d67540 1690
2f073eb2
RR
1691/*
1692 wxPrintf( wxT("2) OnButtonPress from ") );
1693 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1694 wxPrintf( win->GetClassInfo()->GetClassName() );
1695 wxPrintf( wxT(".\n") );
1696*/
1697
df135f2b
RR
1698#ifndef __WXGTK20__
1699 if (event_type == wxEVT_LEFT_DCLICK)
1700 {
1701 // GTK 1.2 crashes when intercepting double
1702 // click events from both wxSpinButton and
1703 // wxSpinCtrl
1704 if (GTK_IS_SPIN_BUTTON(win->m_widget))
1705 {
1706 // Just disable this event for now.
1707 return FALSE;
1708 }
1709 }
1710#endif
1711
f5e27805 1712 if (win->GetEventHandler()->ProcessEvent( event ))
034be888 1713 {
f5e27805 1714 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "button_press_event" );
f03fc89f 1715 return TRUE;
034be888 1716 }
47d67540 1717
034be888 1718 return FALSE;
362c6693 1719}
c801d85f
KB
1720
1721//-----------------------------------------------------------------------------
97b3455a 1722// "button_release_event"
2f2aa628 1723//-----------------------------------------------------------------------------
c801d85f 1724
2b5f62a0
VZ
1725static gint gtk_window_button_release_callback( GtkWidget *widget,
1726 GdkEventButton *gdk_event,
1727 wxWindowGTK *win )
47d67540 1728{
3ac8d3bc
RR
1729 DEBUG_MAIN_THREAD
1730
c50f1fb9 1731 if (g_isIdle)
a2053b27
RR
1732 wxapp_install_idle_handler();
1733
1734 if (!win->m_hasVMT) return FALSE;
034be888
RR
1735 if (g_blockEventsOnDrag) return FALSE;
1736 if (g_blockEventsOnScroll) return FALSE;
c801d85f 1737
034be888 1738 if (!win->IsOwnGtkWindow( gdk_event->window )) return FALSE;
47d67540 1739
f5e27805 1740 wxEventType event_type = wxEVT_NULL;
47d67540 1741
f5e27805
RR
1742 switch (gdk_event->button)
1743 {
2b5f62a0
VZ
1744 case 1:
1745 event_type = wxEVT_LEFT_UP;
1746 break;
1747
1748 case 2:
1749 event_type = wxEVT_MIDDLE_UP;
1750 break;
1751
1752 case 3:
1753 event_type = wxEVT_RIGHT_UP;
1754 break;
1755
1756 default:
1757 // unknwon button, don't process
1758 return FALSE;
f5e27805 1759 }
47d67540 1760
f5e27805 1761 wxMouseEvent event( event_type );
c5f9d156 1762 InitMouseEvent( win, event, gdk_event );
f5e27805 1763
2daa0ce9
VZ
1764 AdjustEventButtonState(event);
1765
3ae4c570
VZ
1766 // same wxListBox hack as above
1767 win->FixUpMouseEvent(widget, event.m_x, event.m_y);
e2762ff0 1768
2b5f62a0
VZ
1769 if ( event_type == wxEVT_RIGHT_UP )
1770 {
1771 // generate a "context menu" event: this is similar to wxEVT_RIGHT_UP
1772 // except that:
1773 //
1774 // (a) it's a command event and so is propagated to the parent
1775 // (b) under MSW it can be generated from kbd too
1776 // (c) it uses screen coords (because of (a))
1777 wxContextMenuEvent evtCtx(wxEVT_CONTEXT_MENU,
1778 win->GetId(),
1779 win->ClientToScreen(event.GetPosition()));
1780 (void)win->GetEventHandler()->ProcessEvent(evtCtx);
1781 }
1782
d1f2ac45
VZ
1783 if ( !g_captureWindow )
1784 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
47d67540 1785
f5e27805 1786 if (win->GetEventHandler()->ProcessEvent( event ))
034be888 1787 {
f5e27805 1788 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "button_release_event" );
f03fc89f 1789 return TRUE;
034be888 1790 }
47d67540 1791
034be888 1792 return FALSE;
362c6693 1793}
c801d85f
KB
1794
1795//-----------------------------------------------------------------------------
2f2aa628
RR
1796// "motion_notify_event"
1797//-----------------------------------------------------------------------------
c801d85f 1798
1e6feb95
VZ
1799static gint gtk_window_motion_notify_callback( GtkWidget *widget,
1800 GdkEventMotion *gdk_event,
1801 wxWindowGTK *win )
47d67540 1802{
3ac8d3bc
RR
1803 DEBUG_MAIN_THREAD
1804
c50f1fb9 1805 if (g_isIdle)
a2053b27
RR
1806 wxapp_install_idle_handler();
1807
1808 if (!win->m_hasVMT) return FALSE;
034be888
RR
1809 if (g_blockEventsOnDrag) return FALSE;
1810 if (g_blockEventsOnScroll) return FALSE;
148cd9b6 1811
034be888
RR
1812 if (!win->IsOwnGtkWindow( gdk_event->window )) return FALSE;
1813
ff8bfdbb 1814 if (gdk_event->is_hint)
aae24d21 1815 {
f7a11f8c
RR
1816 int x = 0;
1817 int y = 0;
1818 GdkModifierType state;
1819 gdk_window_get_pointer(gdk_event->window, &x, &y, &state);
1820 gdk_event->x = x;
1821 gdk_event->y = y;
aae24d21 1822 }
ff8bfdbb 1823
c801d85f 1824/*
e380f72b
RR
1825 printf( "OnMotion from " );
1826 if (win->GetClassInfo() && win->GetClassInfo()->GetClassName())
1827 printf( win->GetClassInfo()->GetClassName() );
1828 printf( ".\n" );
aae24d21 1829*/
47d67540 1830
e380f72b 1831 wxMouseEvent event( wxEVT_MOTION );
c5f9d156 1832 InitMouseEvent(win, event, gdk_event);
e380f72b 1833
1e6feb95 1834 if ( g_captureWindow )
2f2aa628 1835 {
1e6feb95
VZ
1836 // synthetize a mouse enter or leave event if needed
1837 GdkWindow *winUnderMouse = gdk_window_at_pointer(NULL, NULL);
1838 bool hasMouse = winUnderMouse == gdk_event->window;
1839 if ( hasMouse != g_captureWindowHasMouse )
1840 {
1841 // the mouse changed window
1842 g_captureWindowHasMouse = hasMouse;
1843
1844 wxMouseEvent event(g_captureWindowHasMouse ? wxEVT_ENTER_WINDOW
1845 : wxEVT_LEAVE_WINDOW);
c5f9d156 1846 InitMouseEvent(win, event, gdk_event);
1e6feb95
VZ
1847 event.SetEventObject(win);
1848 win->GetEventHandler()->ProcessEvent(event);
1849 }
1850 }
1851 else // no capture
1852 {
d1f2ac45 1853 win = FindWindowForMouseEvent(win, event.m_x, event.m_y);
2f2aa628 1854 }
47d67540 1855
e380f72b 1856 if (win->GetEventHandler()->ProcessEvent( event ))
034be888 1857 {
e380f72b 1858 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "motion_notify_event" );
f03fc89f 1859 return TRUE;
034be888 1860 }
47d67540 1861
034be888 1862 return FALSE;
362c6693 1863}
c801d85f
KB
1864
1865//-----------------------------------------------------------------------------
2f2aa628
RR
1866// "focus_in_event"
1867//-----------------------------------------------------------------------------
c801d85f 1868
6aeb6f2a
VZ
1869// send the wxChildFocusEvent and wxFocusEvent, common code of
1870// gtk_window_focus_in_callback() and SetFocus()
1871static bool DoSendFocusEvents(wxWindow *win)
1872{
1873 // Notify the parent keeping track of focus for the kbd navigation
1874 // purposes that we got it.
1875 wxChildFocusEvent eventChildFocus(win);
1876 (void)win->GetEventHandler()->ProcessEvent(eventChildFocus);
1877
1878 wxFocusEvent eventFocus(wxEVT_SET_FOCUS, win->GetId());
1879 eventFocus.SetEventObject(win);
1880
1881 return win->GetEventHandler()->ProcessEvent(eventFocus);
1882}
1883
1e6feb95
VZ
1884static gint gtk_window_focus_in_callback( GtkWidget *widget,
1885 GdkEvent *WXUNUSED(event),
1886 wxWindow *win )
c801d85f 1887{
3ac8d3bc
RR
1888 DEBUG_MAIN_THREAD
1889
c50f1fb9 1890 if (g_isIdle)
a2053b27
RR
1891 wxapp_install_idle_handler();
1892
1893 if (!win->m_hasVMT) return FALSE;
034be888 1894 if (g_blockEventsOnDrag) return FALSE;
ff8bfdbb 1895
148cd9b6
VZ
1896 switch ( g_sendActivateEvent )
1897 {
1898 case -1:
76fcf0f2 1899 // we've got focus from outside, synthetize wxActivateEvent
148cd9b6
VZ
1900 g_sendActivateEvent = 1;
1901 break;
1902
1903 case 0:
1904 // another our window just lost focus, it was already ours before
1905 // - don't send any wxActivateEvent
1906 g_sendActivateEvent = -1;
1907 break;
1908 }
1909
1e6feb95 1910 g_focusWindowLast =
b292e2f5 1911 g_focusWindow = win;
ff8bfdbb 1912
6cad4f1b
VZ
1913 wxLogTrace(TRACE_FOCUS,
1914 _T("%s: focus in"), win->GetName().c_str());
7de59551 1915
b79395c5
RR
1916#ifdef HAVE_XIM
1917 if (win->m_ic)
1918 gdk_im_begin(win->m_ic, win->m_wxwindow->window);
1919#endif
1920
1e6feb95 1921#if wxUSE_CARET
f6bcfd97
BP
1922 // caret needs to be informed about focus change
1923 wxCaret *caret = win->GetCaret();
1924 if ( caret )
1925 {
1926 caret->OnSetFocus();
1927 }
1928#endif // wxUSE_CARET
1929
d7fa7eaa 1930 g_activeFrameLostFocus = FALSE;
2b5f62a0 1931
e8c12a53
VS
1932 wxWindowGTK *active = wxGetTopLevelParent(win);
1933 if ( active != g_activeFrame )
942bef71 1934 {
e8c12a53
VS
1935 if ( g_activeFrame )
1936 {
45eb5249 1937 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from focus_in)"), g_activeFrame);
e8c12a53
VS
1938 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, g_activeFrame->GetId());
1939 event.SetEventObject(g_activeFrame);
1940 g_activeFrame->GetEventHandler()->ProcessEvent(event);
1941 }
47d67540 1942
45eb5249 1943 wxLogTrace(wxT("activate"), wxT("Activating frame %p (from focus_in)"), active);
e8c12a53
VS
1944 g_activeFrame = active;
1945 wxActivateEvent event(wxEVT_ACTIVATE, TRUE, g_activeFrame->GetId());
1946 event.SetEventObject(g_activeFrame);
1947 g_activeFrame->GetEventHandler()->ProcessEvent(event);
2b5f62a0 1948
d7fa7eaa
RR
1949 // Don't send focus events in addition to activate
1950 // if (win == g_activeFrame)
1951 // return TRUE;
942bef71 1952 }
e8c12a53 1953
6cad4f1b
VZ
1954 // does the window itself think that it has the focus?
1955 if ( !win->m_hasFocus )
5cd09f0b 1956 {
6cad4f1b
VZ
1957 // not yet, notify it
1958 win->m_hasFocus = TRUE;
1959
1960 if ( DoSendFocusEvents(win) )
1961 {
1962 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "focus_in_event" );
1963 return TRUE;
1964 }
034be888 1965 }
ca298c88 1966
034be888 1967 return FALSE;
362c6693 1968}
c801d85f
KB
1969
1970//-----------------------------------------------------------------------------
2f2aa628
RR
1971// "focus_out_event"
1972//-----------------------------------------------------------------------------
c801d85f 1973
afbe906a 1974static gint gtk_window_focus_out_callback( GtkWidget *widget, GdkEventFocus *gdk_event, wxWindowGTK *win )
c801d85f 1975{
3ac8d3bc
RR
1976 DEBUG_MAIN_THREAD
1977
c50f1fb9 1978 if (g_isIdle)
a2053b27
RR
1979 wxapp_install_idle_handler();
1980
1981 if (!win->m_hasVMT) return FALSE;
034be888 1982 if (g_blockEventsOnDrag) return FALSE;
ca298c88 1983
6cad4f1b
VZ
1984 wxLogTrace( TRACE_FOCUS,
1985 _T("%s: focus out"), win->GetName().c_str() );
b231914f 1986
45eb5249
VS
1987 if ( !g_activeFrameLostFocus && g_activeFrame )
1988 {
c0ad3d42
VZ
1989 // VZ: commenting this out because it does happen (although not easy
1990 // to reproduce, I only see it when using wxMiniFrame and not
1991 // always) and makes using Mahogany quite annoying
1992#if 0
1993 wxASSERT_MSG( wxGetTopLevelParent(win) == g_activeFrame,
63fd5f0b 1994 wxT("unfocusing window that hasn't gained focus properly") );
c0ad3d42
VZ
1995#endif // 0
1996
45eb5249
VS
1997 g_activeFrameLostFocus = TRUE;
1998 }
1999
148cd9b6
VZ
2000 // if the focus goes out of our app alltogether, OnIdle() will send
2001 // wxActivateEvent, otherwise gtk_window_focus_in_callback() will reset
2002 // g_sendActivateEvent to -1
2003 g_sendActivateEvent = 0;
2004
3379ed37 2005 wxWindowGTK *winFocus = wxFindFocusedChild(win);
f6bcfd97
BP
2006 if ( winFocus )
2007 win = winFocus;
2008
1e6feb95 2009 g_focusWindow = (wxWindowGTK *)NULL;
148cd9b6 2010
b79395c5
RR
2011#ifdef HAVE_XIM
2012 if (win->m_ic)
2013 gdk_im_end();
2014#endif
2015
1e6feb95 2016#if wxUSE_CARET
f6bcfd97
BP
2017 // caret needs to be informed about focus change
2018 wxCaret *caret = win->GetCaret();
2019 if ( caret )
2020 {
2021 caret->OnKillFocus();
2022 }
2023#endif // wxUSE_CARET
2024
6cad4f1b
VZ
2025 // don't send the window a kill focus event if it thinks that it doesn't
2026 // have focus already
2027 if ( win->m_hasFocus )
5cd09f0b 2028 {
6cad4f1b
VZ
2029 win->m_hasFocus = FALSE;
2030
2031 wxFocusEvent event( wxEVT_KILL_FOCUS, win->GetId() );
2032 event.SetEventObject( win );
2033
2034 if (win->GetEventHandler()->ProcessEvent( event ))
2035 {
2036 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "focus_out_event" );
2037 return TRUE;
2038 }
034be888 2039 }
ca298c88 2040
034be888 2041 return FALSE;
362c6693 2042}
c801d85f 2043
b4071e91
RR
2044//-----------------------------------------------------------------------------
2045// "enter_notify_event"
2046//-----------------------------------------------------------------------------
2047
edc1d330
VZ
2048static
2049gint gtk_window_enter_callback( GtkWidget *widget,
2050 GdkEventCrossing *gdk_event,
2051 wxWindowGTK *win )
b4071e91 2052{
3ac8d3bc
RR
2053 DEBUG_MAIN_THREAD
2054
c50f1fb9 2055 if (g_isIdle)
a2053b27 2056 wxapp_install_idle_handler();
ca298c88 2057
a2053b27
RR
2058 if (!win->m_hasVMT) return FALSE;
2059 if (g_blockEventsOnDrag) return FALSE;
47d67540 2060
a2053b27 2061 if (!win->IsOwnGtkWindow( gdk_event->window )) return FALSE;
b292e2f5 2062
4a33eba6
RR
2063 int x = 0;
2064 int y = 0;
2065 GdkModifierType state = (GdkModifierType)0;
ff8bfdbb 2066
a2053b27 2067 gdk_window_get_pointer( widget->window, &x, &y, &state );
ff8bfdbb 2068
edc1d330 2069 wxMouseEvent event( wxEVT_ENTER_WINDOW );
c5f9d156
VS
2070 InitMouseEvent(win, event, gdk_event);
2071 wxPoint pt = win->GetClientAreaOrigin();
2072 event.m_x = x + pt.x;
2073 event.m_y = y + pt.y;
ff8bfdbb 2074
e380f72b 2075 if (win->GetEventHandler()->ProcessEvent( event ))
034be888 2076 {
e380f72b 2077 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "enter_notify_event" );
034be888
RR
2078 return TRUE;
2079 }
ca298c88 2080
034be888 2081 return FALSE;
b4071e91 2082}
47d67540 2083
b4071e91
RR
2084//-----------------------------------------------------------------------------
2085// "leave_notify_event"
2086//-----------------------------------------------------------------------------
2087
1e6feb95 2088static gint gtk_window_leave_callback( GtkWidget *widget, GdkEventCrossing *gdk_event, wxWindowGTK *win )
b4071e91 2089{
3ac8d3bc
RR
2090 DEBUG_MAIN_THREAD
2091
c50f1fb9 2092 if (g_isIdle)
a2053b27 2093 wxapp_install_idle_handler();
acfd422a 2094
a2053b27
RR
2095 if (!win->m_hasVMT) return FALSE;
2096 if (g_blockEventsOnDrag) return FALSE;
b292e2f5 2097
a2053b27 2098 if (!win->IsOwnGtkWindow( gdk_event->window )) return FALSE;
c50f1fb9 2099
e380f72b 2100 wxMouseEvent event( wxEVT_LEAVE_WINDOW );
d1367c3d 2101 event.SetTimestamp( gdk_event->time );
e380f72b 2102 event.SetEventObject( win );
47d67540 2103
4a33eba6
RR
2104 int x = 0;
2105 int y = 0;
2106 GdkModifierType state = (GdkModifierType)0;
ff8bfdbb 2107
4a33eba6 2108 gdk_window_get_pointer( widget->window, &x, &y, &state );
ff8bfdbb 2109
74710601
VZ
2110 event.m_shiftDown = (state & GDK_SHIFT_MASK) != 0;
2111 event.m_controlDown = (state & GDK_CONTROL_MASK) != 0;
2112 event.m_altDown = (state & GDK_MOD1_MASK) != 0;
2113 event.m_metaDown = (state & GDK_MOD2_MASK) != 0;
2114 event.m_leftDown = (state & GDK_BUTTON1_MASK) != 0;
2115 event.m_middleDown = (state & GDK_BUTTON2_MASK) != 0;
2116 event.m_rightDown = (state & GDK_BUTTON3_MASK) != 0;
4a33eba6 2117
c5f9d156
VS
2118 wxPoint pt = win->GetClientAreaOrigin();
2119 event.m_x = x + pt.x;
2120 event.m_y = y + pt.y;
ff8bfdbb 2121
e380f72b 2122 if (win->GetEventHandler()->ProcessEvent( event ))
034be888 2123 {
e380f72b 2124 gtk_signal_emit_stop_by_name( GTK_OBJECT(widget), "leave_notify_event" );
034be888
RR
2125 return TRUE;
2126 }
ca298c88 2127
034be888 2128 return FALSE;
b4071e91 2129}
47d67540 2130
c801d85f 2131//-----------------------------------------------------------------------------
2f2aa628
RR
2132// "value_changed" from m_vAdjust
2133//-----------------------------------------------------------------------------
c801d85f 2134
9e691f46
VZ
2135static void gtk_window_vscroll_callback( GtkAdjustment *adjust,
2136 SCROLLBAR_CBACK_ARG
2137 wxWindowGTK *win )
c801d85f 2138{
3ac8d3bc
RR
2139 DEBUG_MAIN_THREAD
2140
c50f1fb9 2141 if (g_isIdle)
a2053b27 2142 wxapp_install_idle_handler();
c801d85f 2143
a2053b27 2144 if (g_blockEventsOnDrag) return;
47d67540 2145
a2053b27 2146 if (!win->m_hasVMT) return;
148cd9b6 2147
5e014a0c 2148 float diff = adjust->value - win->m_oldVerticalPos;
e380f72b 2149 if (fabs(diff) < 0.2) return;
148cd9b6 2150
5e014a0c 2151 win->m_oldVerticalPos = adjust->value;
47d67540 2152
a8bf1826 2153#ifndef __WXGTK20__
6728fb61 2154 GtkScrolledWindow *sw = GTK_SCROLLED_WINDOW(win->m_widget);
a8bf1826 2155#endif
6728fb61 2156 wxEventType command = GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw->vscrollbar));
148cd9b6 2157
5e014a0c 2158 int value = (int)(adjust->value+0.5);
c801d85f 2159
c5b42c87 2160 wxScrollWinEvent event( command, value, wxVERTICAL );
e380f72b
RR
2161 event.SetEventObject( win );
2162 win->GetEventHandler()->ProcessEvent( event );
362c6693 2163}
c801d85f
KB
2164
2165//-----------------------------------------------------------------------------
2f2aa628
RR
2166// "value_changed" from m_hAdjust
2167//-----------------------------------------------------------------------------
c801d85f 2168
9e691f46
VZ
2169static void gtk_window_hscroll_callback( GtkAdjustment *adjust,
2170 SCROLLBAR_CBACK_ARG
2171 wxWindowGTK *win )
47d67540 2172{
3ac8d3bc
RR
2173 DEBUG_MAIN_THREAD
2174
c50f1fb9 2175 if (g_isIdle)
a2053b27 2176 wxapp_install_idle_handler();
47d67540 2177
a2053b27
RR
2178 if (g_blockEventsOnDrag) return;
2179 if (!win->m_hasVMT) return;
47d67540 2180
5e014a0c 2181 float diff = adjust->value - win->m_oldHorizontalPos;
e380f72b 2182 if (fabs(diff) < 0.2) return;
148cd9b6 2183
a8bf1826 2184#ifndef __WXGTK20__
6728fb61 2185 GtkScrolledWindow *sw = GTK_SCROLLED_WINDOW(win->m_widget);
a8bf1826 2186#endif
6728fb61 2187 wxEventType command = GtkScrollWinTypeToWx(GET_SCROLL_TYPE(sw->hscrollbar));
148cd9b6 2188
9e691f46 2189 win->m_oldHorizontalPos = adjust->value;
148cd9b6 2190
5e014a0c 2191 int value = (int)(adjust->value+0.5);
47d67540 2192
c5b42c87 2193 wxScrollWinEvent event( command, value, wxHORIZONTAL );
e380f72b
RR
2194 event.SetEventObject( win );
2195 win->GetEventHandler()->ProcessEvent( event );
362c6693 2196}
c801d85f 2197
cb43b372
RR
2198//-----------------------------------------------------------------------------
2199// "button_press_event" from scrollbar
2200//-----------------------------------------------------------------------------
2201
2a23d363
RR
2202static gint gtk_scrollbar_button_press_callback( GtkRange *widget,
2203 GdkEventButton *gdk_event,
1e6feb95 2204 wxWindowGTK *win)
cb43b372 2205{
3ac8d3bc
RR
2206 DEBUG_MAIN_THREAD
2207
c50f1fb9 2208 if (g_isIdle)
a2053b27
RR
2209 wxapp_install_idle_handler();
2210
d6d26e04 2211
5b8a521e 2212 g_blockEventsOnScroll = TRUE;
9e691f46
VZ
2213
2214 // FIXME: there is no 'slider' field in GTK+ 2.0 any more
2215#ifndef __WXGTK20__
2a23d363 2216 win->m_isScrolling = (gdk_event->window == widget->slider);
9e691f46 2217#endif
47d67540 2218
e380f72b 2219 return FALSE;
cb43b372
RR
2220}
2221
2222//-----------------------------------------------------------------------------
2223// "button_release_event" from scrollbar
2224//-----------------------------------------------------------------------------
2225
88413fec 2226static gint gtk_scrollbar_button_release_callback( GtkRange *widget,
8bbe427f 2227 GdkEventButton *WXUNUSED(gdk_event),
1e6feb95 2228 wxWindowGTK *win)
cb43b372 2229{
3ac8d3bc
RR
2230 DEBUG_MAIN_THREAD
2231
1ecc4d80 2232// don't test here as we can release the mouse while being over
5e014a0c 2233// a different window than the slider
76ed8f8d
RR
2234//
2235// if (gdk_event->window != widget->slider) return FALSE;
cb43b372 2236
5b8a521e 2237 g_blockEventsOnScroll = FALSE;
47d67540 2238
2a23d363 2239 if (win->m_isScrolling)
88413fec 2240 {
d6d26e04 2241 wxEventType command = wxEVT_SCROLLWIN_THUMBRELEASE;
2a23d363
RR
2242 int value = -1;
2243 int dir = -1;
2daa0ce9 2244
2a23d363
RR
2245 GtkScrolledWindow *scrolledWindow = GTK_SCROLLED_WINDOW(win->m_widget);
2246 if (widget == GTK_RANGE(scrolledWindow->hscrollbar))
2247 {
2248 value = (int)(win->m_hAdjust->value+0.5);
2249 dir = wxHORIZONTAL;
2250 }
2251 if (widget == GTK_RANGE(scrolledWindow->vscrollbar))
2252 {
2253 value = (int)(win->m_vAdjust->value+0.5);
2254 dir = wxVERTICAL;
2255 }
2daa0ce9 2256
2a23d363 2257 wxScrollWinEvent event( command, value, dir );
2a23d363
RR
2258 event.SetEventObject( win );
2259 win->GetEventHandler()->ProcessEvent( event );
88413fec
RR
2260 }
2261
2a23d363 2262 win->m_isScrolling = FALSE;
2daa0ce9 2263
e380f72b 2264 return FALSE;
cb43b372
RR
2265}
2266
f03fc89f
VZ
2267// ----------------------------------------------------------------------------
2268// this wxWindowBase function is implemented here (in platform-specific file)
2269// because it is static and so couldn't be made virtual
2270// ----------------------------------------------------------------------------
2b07d713 2271
f03fc89f 2272wxWindow *wxWindowBase::FindFocus()
2b07d713 2273{
1e6feb95
VZ
2274 // the cast is necessary when we compile in wxUniversal mode
2275 return (wxWindow *)g_focusWindow;
2b07d713 2276}
ca298c88 2277
1a685a59 2278
a2053b27
RR
2279//-----------------------------------------------------------------------------
2280// "realize" from m_widget
2281//-----------------------------------------------------------------------------
2282
b79395c5
RR
2283/* We cannot set colours and fonts before the widget has
2284 been realized, so we do this directly after realization. */
a2053b27
RR
2285
2286static gint
2b5f62a0 2287gtk_window_realized_callback( GtkWidget *m_widget, wxWindow *win )
a2053b27 2288{
3ac8d3bc
RR
2289 DEBUG_MAIN_THREAD
2290
c50f1fb9 2291 if (g_isIdle)
a2053b27
RR
2292 wxapp_install_idle_handler();
2293
739730ca 2294 if (win->m_delayedBackgroundColour)
ea323db3 2295 win->GtkSetBackgroundColour( win->GetBackgroundColour() );
a2053b27 2296
739730ca 2297 if (win->m_delayedForegroundColour)
ea323db3 2298 win->GtkSetForegroundColour( win->GetForegroundColour() );
a2053b27 2299
2b5f62a0
VZ
2300#ifdef __WXGTK20__
2301 if (win->m_imContext)
2302 {
2303 GtkPizza *pizza = GTK_PIZZA( m_widget );
2304 gtk_im_context_set_client_window( (GtkIMContext*) win->m_imContext, pizza->bin_window );
2305 }
2306#endif
2307
3c679789
RR
2308 wxWindowCreateEvent event( win );
2309 event.SetEventObject( win );
2310 win->GetEventHandler()->ProcessEvent( event );
1e6feb95 2311
a2053b27
RR
2312 return FALSE;
2313}
2314
b79395c5
RR
2315//-----------------------------------------------------------------------------
2316// "size_allocate"
2317//-----------------------------------------------------------------------------
2318
8f75cb6c 2319static
adc1999b
RR
2320void gtk_window_size_callback( GtkWidget *WXUNUSED(widget),
2321 GtkAllocation *WXUNUSED(alloc),
8f75cb6c
RR
2322 wxWindow *win )
2323{
2324 if (g_isIdle)
2325 wxapp_install_idle_handler();
2daa0ce9 2326
5b8a521e 2327 if (!win->m_hasScrolling) return;
2daa0ce9 2328
5b8a521e
RR
2329 int client_width = 0;
2330 int client_height = 0;
2331 win->GetClientSize( &client_width, &client_height );
2332 if ((client_width == win->m_oldClientWidth) && (client_height == win->m_oldClientHeight))
2333 return;
2daa0ce9 2334
5b8a521e
RR
2335 win->m_oldClientWidth = client_width;
2336 win->m_oldClientHeight = client_height;
2daa0ce9 2337
5b8a521e
RR
2338 if (!win->m_nativeSizeEvent)
2339 {
2340 wxSizeEvent event( win->GetSize(), win->GetId() );
2341 event.SetEventObject( win );
2342 win->GetEventHandler()->ProcessEvent( event );
2343 }
8f75cb6c
RR
2344}
2345
2346
3ed2e7ce
VZ
2347#ifdef HAVE_XIM
2348 #define WXUNUSED_UNLESS_XIM(param) param
2349#else
2350 #define WXUNUSED_UNLESS_XIM(param) WXUNUSED(param)
2351#endif
2352
b79395c5
RR
2353/* Resize XIM window */
2354
3ed2e7ce 2355static
8f75cb6c
RR
2356void gtk_wxwindow_size_callback( GtkWidget* WXUNUSED_UNLESS_XIM(widget),
2357 GtkAllocation* WXUNUSED_UNLESS_XIM(alloc),
1e6feb95 2358 wxWindowGTK* WXUNUSED_UNLESS_XIM(win) )
b79395c5
RR
2359{
2360 if (g_isIdle)
2361 wxapp_install_idle_handler();
2daa0ce9 2362
9a8c7620 2363#ifdef HAVE_XIM
b79395c5
RR
2364 if (!win->m_ic)
2365 return;
2366
b79395c5
RR
2367 if (gdk_ic_get_style (win->m_ic) & GDK_IM_PREEDIT_POSITION)
2368 {
2369 gint width, height;
2370
3ed2e7ce 2371 gdk_window_get_size (widget->window, &width, &height);
b79395c5
RR
2372 win->m_icattr->preedit_area.width = width;
2373 win->m_icattr->preedit_area.height = height;
2374 gdk_ic_set_attr (win->m_ic, win->m_icattr, GDK_IC_PREEDIT_AREA);
2375 }
9a8c7620 2376#endif // HAVE_XIM
b79395c5
RR
2377}
2378
63081513
RR
2379//-----------------------------------------------------------------------------
2380// "realize" from m_wxwindow
2381//-----------------------------------------------------------------------------
2382
2383/* Initialize XIM support */
2384
2385static gint
3ed2e7ce 2386gtk_wxwindow_realized_callback( GtkWidget * WXUNUSED_UNLESS_XIM(widget),
1e6feb95 2387 wxWindowGTK * WXUNUSED_UNLESS_XIM(win) )
63081513
RR
2388{
2389 if (g_isIdle)
2390 wxapp_install_idle_handler();
2391
d06800f1 2392#ifdef HAVE_XIM
63081513
RR
2393 if (win->m_ic) return FALSE;
2394 if (!widget) return FALSE;
2395 if (!gdk_im_ready()) return FALSE;
2396
2397 win->m_icattr = gdk_ic_attr_new();
2398 if (!win->m_icattr) return FALSE;
2daa0ce9 2399
63081513
RR
2400 gint width, height;
2401 GdkEventMask mask;
2402 GdkColormap *colormap;
2403 GdkICAttr *attr = win->m_icattr;
b58b1dfc 2404 unsigned attrmask = GDK_IC_ALL_REQ;
63081513 2405 GdkIMStyle style;
b79395c5
RR
2406 GdkIMStyle supported_style = (GdkIMStyle)
2407 (GDK_IM_PREEDIT_NONE |
1e6feb95
VZ
2408 GDK_IM_PREEDIT_NOTHING |
2409 GDK_IM_PREEDIT_POSITION |
2410 GDK_IM_STATUS_NONE |
2411 GDK_IM_STATUS_NOTHING);
63081513
RR
2412
2413 if (widget->style && widget->style->font->type != GDK_FONT_FONTSET)
1e6feb95 2414 supported_style = (GdkIMStyle)(supported_style & ~GDK_IM_PREEDIT_POSITION);
63081513
RR
2415
2416 attr->style = style = gdk_im_decide_style (supported_style);
2417 attr->client_window = widget->window;
2418
2419 if ((colormap = gtk_widget_get_colormap (widget)) !=
1e6feb95 2420 gtk_widget_get_default_colormap ())
63081513 2421 {
5cd09f0b
RR
2422 attrmask |= GDK_IC_PREEDIT_COLORMAP;
2423 attr->preedit_colormap = colormap;
63081513 2424 }
2daa0ce9 2425
63081513
RR
2426 attrmask |= GDK_IC_PREEDIT_FOREGROUND;
2427 attrmask |= GDK_IC_PREEDIT_BACKGROUND;
2428 attr->preedit_foreground = widget->style->fg[GTK_STATE_NORMAL];
2429 attr->preedit_background = widget->style->base[GTK_STATE_NORMAL];
2430
2431 switch (style & GDK_IM_PREEDIT_MASK)
2432 {
1e6feb95
VZ
2433 case GDK_IM_PREEDIT_POSITION:
2434 if (widget->style && widget->style->font->type != GDK_FONT_FONTSET)
2435 {
2436 g_warning ("over-the-spot style requires fontset");
2437 break;
2438 }
63081513 2439
1e6feb95 2440 gdk_window_get_size (widget->window, &width, &height);
63081513 2441
1e6feb95
VZ
2442 attrmask |= GDK_IC_PREEDIT_POSITION_REQ;
2443 attr->spot_location.x = 0;
2444 attr->spot_location.y = height;
2445 attr->preedit_area.x = 0;
2446 attr->preedit_area.y = 0;
2447 attr->preedit_area.width = width;
2448 attr->preedit_area.height = height;
2449 attr->preedit_fontset = widget->style->font;
63081513 2450
1e6feb95 2451 break;
b79395c5 2452 }
2daa0ce9 2453
b58b1dfc 2454 win->m_ic = gdk_ic_new (attr, (GdkICAttributesType)attrmask);
2daa0ce9 2455
63081513 2456 if (win->m_ic == NULL)
1e6feb95 2457 g_warning ("Can't create input context.");
63081513 2458 else
1e6feb95
VZ
2459 {
2460 mask = gdk_window_get_events (widget->window);
2461 mask = (GdkEventMask)(mask | gdk_ic_get_events (win->m_ic));
2462 gdk_window_set_events (widget->window, mask);
2463
2464 if (GTK_WIDGET_HAS_FOCUS(widget))
2465 gdk_im_begin (win->m_ic, widget->window);
2466 }
2467#endif // HAVE_XIM
63081513
RR
2468
2469 return FALSE;
2470}
2471
6ca41e57 2472//-----------------------------------------------------------------------------
1e6feb95 2473// InsertChild for wxWindowGTK.
6ca41e57
RR
2474//-----------------------------------------------------------------------------
2475
1e6feb95 2476/* Callback for wxWindowGTK. This very strange beast has to be used because
b1170810
RR
2477 * C++ has no virtual methods in a constructor. We have to emulate a
2478 * virtual function here as wxNotebook requires a different way to insert
2479 * a child in it. I had opted for creating a wxNotebookPage window class
2480 * which would have made this superfluous (such in the MDI window system),
2481 * but no-one was listening to me... */
6ca41e57 2482
1e6feb95 2483static void wxInsertChildInWindow( wxWindowGTK* parent, wxWindowGTK* child )
6ca41e57 2484{
bf0c00c6
RR
2485 /* the window might have been scrolled already, do we
2486 have to adapt the position */
da048e3d
RR
2487 GtkPizza *pizza = GTK_PIZZA(parent->m_wxwindow);
2488 child->m_x += pizza->xoffset;
2489 child->m_y += pizza->yoffset;
148cd9b6 2490
da048e3d 2491 gtk_pizza_put( GTK_PIZZA(parent->m_wxwindow),
a2053b27
RR
2492 GTK_WIDGET(child->m_widget),
2493 child->m_x,
2494 child->m_y,
2495 child->m_width,
2496 child->m_height );
6ca41e57
RR
2497}
2498
bbe0af5b
RR
2499//-----------------------------------------------------------------------------
2500// global functions
2501//-----------------------------------------------------------------------------
2502
1e6feb95 2503wxWindow *wxGetActiveWindow()
bbe0af5b 2504{
6cad4f1b 2505 return wxWindow::FindFocus();
bbe0af5b
RR
2506}
2507
c801d85f 2508//-----------------------------------------------------------------------------
1e6feb95 2509// wxWindowGTK
c801d85f
KB
2510//-----------------------------------------------------------------------------
2511
6522713c
VZ
2512// in wxUniv/MSW this class is abstract because it doesn't have DoPopupMenu()
2513// method
1e6feb95 2514#ifdef __WXUNIVERSAL__
6522713c
VZ
2515 IMPLEMENT_ABSTRACT_CLASS(wxWindowGTK, wxWindowBase)
2516#else // __WXGTK__
1e6feb95 2517 IMPLEMENT_DYNAMIC_CLASS(wxWindow, wxWindowBase)
6522713c 2518#endif // __WXUNIVERSAL__/__WXGTK__
c801d85f 2519
1e6feb95 2520void wxWindowGTK::Init()
c801d85f 2521{
f03fc89f
VZ
2522 // common init
2523 InitBase();
68995f26 2524
f03fc89f 2525 // GTK specific
a2053b27 2526 m_widget = (GtkWidget *) NULL;
e380f72b 2527 m_wxwindow = (GtkWidget *) NULL;
76fcf0f2 2528 m_focusWidget = (GtkWidget *) NULL;
8bbe427f 2529
f03fc89f 2530 // position/size
a2053b27
RR
2531 m_x = 0;
2532 m_y = 0;
2533 m_width = 0;
e380f72b 2534 m_height = 0;
8bbe427f 2535
e380f72b
RR
2536 m_sizeSet = FALSE;
2537 m_hasVMT = FALSE;
2538 m_needParent = TRUE;
31c6b4fc 2539 m_isBeingDeleted = FALSE;
148cd9b6 2540
147bc491 2541 m_noExpose = FALSE;
30760ce7 2542 m_nativeSizeEvent = FALSE;
94633ad9 2543
a2053b27 2544 m_hasScrolling = FALSE;
846e1424 2545 m_isScrolling = FALSE;
f03fc89f 2546
a2053b27 2547 m_hAdjust = (GtkAdjustment*) NULL;
e380f72b 2548 m_vAdjust = (GtkAdjustment*) NULL;
a2053b27 2549 m_oldHorizontalPos = 0.0;
e380f72b 2550 m_oldVerticalPos = 0.0;
8bbe427f 2551
e380f72b 2552 m_resizing = FALSE;
e380f72b 2553 m_widgetStyle = (GtkStyle*) NULL;
8bbe427f 2554
ddb6bc71 2555 m_insertCallback = (wxInsertChildFunction) NULL;
8bbe427f 2556
b292e2f5 2557 m_acceptsFocus = FALSE;
6cad4f1b 2558 m_hasFocus = FALSE;
148cd9b6 2559
b6fa52db 2560 m_clipPaintRegion = FALSE;
b6fa52db 2561
5e014a0c 2562 m_cursor = *wxSTANDARD_CURSOR;
2daa0ce9 2563
f6bcfd97
BP
2564 m_delayedForegroundColour = FALSE;
2565 m_delayedBackgroundColour = FALSE;
1e6feb95 2566
2b5f62a0
VZ
2567#ifdef __WXGTK20__
2568 m_imContext = NULL;
2569 m_x11Context = NULL;
2570#else
63081513
RR
2571#ifdef HAVE_XIM
2572 m_ic = (GdkIC*) NULL;
2573 m_icattr = (GdkICAttr*) NULL;
2574#endif
2b5f62a0 2575#endif
362c6693 2576}
c801d85f 2577
1e6feb95 2578wxWindowGTK::wxWindowGTK()
68995f26
VZ
2579{
2580 Init();
2581}
2582
1e6feb95
VZ
2583wxWindowGTK::wxWindowGTK( wxWindow *parent,
2584 wxWindowID id,
2585 const wxPoint &pos,
2586 const wxSize &size,
2587 long style,
2588 const wxString &name )
6ca41e57 2589{
68995f26
VZ
2590 Init();
2591
e380f72b 2592 Create( parent, id, pos, size, style, name );
6ca41e57 2593}
8bbe427f 2594
1e6feb95
VZ
2595bool wxWindowGTK::Create( wxWindow *parent,
2596 wxWindowID id,
2597 const wxPoint &pos,
2598 const wxSize &size,
2599 long style,
2600 const wxString &name )
c801d85f 2601{
4dcaf11a
RR
2602 if (!PreCreation( parent, pos, size ) ||
2603 !CreateBase( parent, id, pos, size, style, wxDefaultValidator, name ))
2604 {
1e6feb95 2605 wxFAIL_MSG( wxT("wxWindowGTK creation failed") );
b02da6b1 2606 return FALSE;
4dcaf11a 2607 }
47d67540 2608
ddb6bc71 2609 m_insertCallback = wxInsertChildInWindow;
2b5f62a0 2610
994bc575 2611 // always needed for background clearing
2b5f62a0 2612 m_delayedBackgroundColour = TRUE;
1e6feb95 2613
e380f72b 2614 m_widget = gtk_scrolled_window_new( (GtkAdjustment *) NULL, (GtkAdjustment *) NULL );
38c7b3d3 2615 GTK_WIDGET_UNSET_FLAGS( m_widget, GTK_CAN_FOCUS );
ff8bfdbb 2616
f03fc89f 2617 GtkScrolledWindow *scrolledWindow = GTK_SCROLLED_WINDOW(m_widget);
47d67540 2618
dd00f3f6 2619 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
e380f72b 2620 scroll_class->scrollbar_spacing = 0;
47d67540 2621
f03fc89f 2622 gtk_scrolled_window_set_policy( scrolledWindow, GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC );
47d67540 2623
f03fc89f
VZ
2624 m_hAdjust = gtk_range_get_adjustment( GTK_RANGE(scrolledWindow->hscrollbar) );
2625 m_vAdjust = gtk_range_get_adjustment( GTK_RANGE(scrolledWindow->vscrollbar) );
47d67540 2626
da048e3d 2627 m_wxwindow = gtk_pizza_new();
0fc5dbf5 2628
1e6feb95 2629#ifndef __WXUNIVERSAL__
da048e3d 2630 GtkPizza *pizza = GTK_PIZZA(m_wxwindow);
b292e2f5 2631
f03fc89f 2632 if (HasFlag(wxRAISED_BORDER))
034be888 2633 {
da048e3d 2634 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_OUT );
034be888 2635 }
f03fc89f 2636 else if (HasFlag(wxSUNKEN_BORDER))
034be888 2637 {
da048e3d 2638 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_IN );
5e014a0c
RR
2639 }
2640 else if (HasFlag(wxSIMPLE_BORDER))
2641 {
da048e3d 2642 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_THIN );
034be888
RR
2643 }
2644 else
2645 {
da048e3d 2646 gtk_pizza_set_shadow_type( pizza, GTK_MYSHADOW_NONE );
034be888 2647 }
1e6feb95 2648#endif // __WXUNIVERSAL__
47d67540 2649
4e5a4c69
RR
2650 gtk_container_add( GTK_CONTAINER(m_widget), m_wxwindow );
2651
3da17724
RR
2652 GTK_WIDGET_SET_FLAGS( m_wxwindow, GTK_CAN_FOCUS );
2653 m_acceptsFocus = TRUE;
ca298c88 2654
e380f72b 2655 // I _really_ don't want scrollbars in the beginning
a2053b27
RR
2656 m_vAdjust->lower = 0.0;
2657 m_vAdjust->upper = 1.0;
2658 m_vAdjust->value = 0.0;
2659 m_vAdjust->step_increment = 1.0;
2660 m_vAdjust->page_increment = 1.0;
2661 m_vAdjust->page_size = 5.0;
2662 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust), "changed" );
2663 m_hAdjust->lower = 0.0;
2664 m_hAdjust->upper = 1.0;
2665 m_hAdjust->value = 0.0;
2666 m_hAdjust->step_increment = 1.0;
2667 m_hAdjust->page_increment = 1.0;
2668 m_hAdjust->page_size = 5.0;
2669 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust), "changed" );
f03fc89f
VZ
2670
2671 // these handlers block mouse events to any window during scrolling such as
2672 // motion events and prevent GTK and wxWindows from fighting over where the
2673 // slider should be
2674
2675 gtk_signal_connect( GTK_OBJECT(scrolledWindow->vscrollbar), "button_press_event",
76ed8f8d
RR
2676 (GtkSignalFunc)gtk_scrollbar_button_press_callback, (gpointer) this );
2677
f03fc89f 2678 gtk_signal_connect( GTK_OBJECT(scrolledWindow->hscrollbar), "button_press_event",
76ed8f8d
RR
2679 (GtkSignalFunc)gtk_scrollbar_button_press_callback, (gpointer) this );
2680
f03fc89f 2681 gtk_signal_connect( GTK_OBJECT(scrolledWindow->vscrollbar), "button_release_event",
76ed8f8d
RR
2682 (GtkSignalFunc)gtk_scrollbar_button_release_callback, (gpointer) this );
2683
f03fc89f 2684 gtk_signal_connect( GTK_OBJECT(scrolledWindow->hscrollbar), "button_release_event",
76ed8f8d 2685 (GtkSignalFunc)gtk_scrollbar_button_release_callback, (gpointer) this );
8bbe427f 2686
034be888 2687 // these handlers get notified when screen updates are required either when
76ed8f8d
RR
2688 // scrolling or when the window size (and therefore scrollbar configuration)
2689 // has changed
2690
2691 gtk_signal_connect( GTK_OBJECT(m_hAdjust), "value_changed",
2692 (GtkSignalFunc) gtk_window_hscroll_callback, (gpointer) this );
2693 gtk_signal_connect( GTK_OBJECT(m_vAdjust), "value_changed",
2694 (GtkSignalFunc) gtk_window_vscroll_callback, (gpointer) this );
2695
2b5f62a0
VZ
2696#ifdef __WXGTK20__
2697 // Create input method handler
2698 m_imContext = (GtkIMMulticontext*) gtk_im_multicontext_new ();
2699
2700 // Cannot handle drawing preedited text yet
2701 gtk_im_context_set_use_preedit( (GtkIMContext*) m_imContext, FALSE );
2702
2703 g_signal_connect (G_OBJECT (m_imContext), "commit",
2704 G_CALLBACK (gtk_wxwindow_commit_cb), this);
2705#endif
2706
f03fc89f 2707 gtk_widget_show( m_wxwindow );
47d67540 2708
f03fc89f
VZ
2709 if (m_parent)
2710 m_parent->DoAddChild( this );
94633ad9 2711
76fcf0f2 2712 m_focusWidget = m_wxwindow;
8bbe427f 2713
e380f72b 2714 PostCreation();
8bbe427f 2715
e380f72b 2716 Show( TRUE );
c801d85f 2717
e380f72b 2718 return TRUE;
362c6693 2719}
c801d85f 2720
1e6feb95 2721wxWindowGTK::~wxWindowGTK()
c801d85f 2722{
7de59551
RD
2723 SendDestroyEvent();
2724
44cd54c2
JS
2725 if (g_focusWindow == this)
2726 g_focusWindow = NULL;
2727
e8c12a53
VS
2728 if (g_activeFrame == this)
2729 g_activeFrame = NULL;
2730
3e679f01
VZ
2731 if ( g_delayedFocus == this )
2732 g_delayedFocus = NULL;
2733
31c6b4fc 2734 m_isBeingDeleted = TRUE;
43a18898 2735 m_hasVMT = FALSE;
47d67540 2736
f03fc89f
VZ
2737 if (m_widget)
2738 Show( FALSE );
8bbe427f 2739
a2053b27
RR
2740 DestroyChildren();
2741
f03fc89f
VZ
2742 if (m_parent)
2743 m_parent->RemoveChild( this );
c801d85f 2744
63081513
RR
2745#ifdef HAVE_XIM
2746 if (m_ic)
2747 gdk_ic_destroy (m_ic);
2748 if (m_icattr)
2749 gdk_ic_attr_destroy (m_icattr);
2750#endif
2751
f03fc89f 2752 if (m_widgetStyle)
a2053b27 2753 {
a56fcaaf 2754#if DISABLE_STYLE_IF_BROKEN_THEME
bce1406b
RR
2755 // don't delete if it's a pixmap theme style
2756 if (!m_widgetStyle->engine_data)
2757 gtk_style_unref( m_widgetStyle );
1e6feb95 2758#endif
c50f1fb9 2759 m_widgetStyle = (GtkStyle*) NULL;
a2053b27 2760 }
c801d85f 2761
f03fc89f 2762 if (m_wxwindow)
a2053b27 2763 {
f03fc89f 2764 gtk_widget_destroy( m_wxwindow );
c50f1fb9 2765 m_wxwindow = (GtkWidget*) NULL;
a2053b27 2766 }
8bbe427f 2767
f03fc89f 2768 if (m_widget)
a2053b27 2769 {
f03fc89f 2770 gtk_widget_destroy( m_widget );
c50f1fb9 2771 m_widget = (GtkWidget*) NULL;
a2053b27 2772 }
362c6693 2773}
c801d85f 2774
1e6feb95 2775bool wxWindowGTK::PreCreation( wxWindowGTK *parent, const wxPoint &pos, const wxSize &size )
c801d85f 2776{
223d09f6 2777 wxCHECK_MSG( !m_needParent || parent, FALSE, wxT("Need complete parent.") );
8bbe427f 2778
3013a903
RR
2779 // This turns -1 into 30 so that a minimal window is
2780 // visible even although -1,-1 has been given as the
2781 // size of the window. the same trick is used in other
2782 // ports and should make debugging easier.
2783 m_width = WidthDefault(size.x) ;
f03fc89f 2784 m_height = HeightDefault(size.y);
8bbe427f 2785
43a18898
RR
2786 m_x = (int)pos.x;
2787 m_y = (int)pos.y;
8bbe427f 2788
3013a903 2789 // some reasonable defaults
148cd9b6 2790 if (!parent)
6ca41e57 2791 {
43a18898
RR
2792 if (m_x == -1)
2793 {
2794 m_x = (gdk_screen_width () - m_width) / 2;
2795 if (m_x < 10) m_x = 10;
2796 }
2797 if (m_y == -1)
2798 {
2799 m_y = (gdk_screen_height () - m_height) / 2;
2800 if (m_y < 10) m_y = 10;
2801 }
6ca41e57 2802 }
148cd9b6 2803
4dcaf11a 2804 return TRUE;
c801d85f
KB
2805}
2806
1e6feb95 2807void wxWindowGTK::PostCreation()
c801d85f 2808{
223d09f6 2809 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
94633ad9 2810
43a18898
RR
2811 if (m_wxwindow)
2812 {
147bc491 2813 if (!m_noExpose)
b02da6b1 2814 {
76fcf0f2 2815 // these get reported to wxWindows -> wxPaintEvent
1e6feb95 2816
b420fb6a
RR
2817 gtk_pizza_set_external( GTK_PIZZA(m_wxwindow), TRUE );
2818
147bc491
RR
2819 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "expose_event",
2820 GTK_SIGNAL_FUNC(gtk_window_expose_callback), (gpointer)this );
2821
4e5a4c69 2822#ifndef __WXGTK20__
147bc491
RR
2823 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "draw",
2824 GTK_SIGNAL_FUNC(gtk_window_draw_callback), (gpointer)this );
0fc5dbf5 2825
e22454be
RR
2826 if (HasFlag(wxNO_FULL_REPAINT_ON_RESIZE))
2827 {
2828 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "event",
2829 GTK_SIGNAL_FUNC(gtk_window_event_event_callback), (gpointer)this );
2830 }
4e5a4c69 2831#else
2b5f62a0
VZ
2832 // gtk_widget_set_redraw_on_allocate( GTK_WIDGET(m_wxwindow), HasFlag( wxNO_FULL_REPAINT_ON_RESIZE ) );
2833#endif
2834
2835#ifdef __WXGTK20__
2836 // Create input method handler
2837 m_imContext = (GtkIMMulticontext*) gtk_im_multicontext_new ();
2838
2839 // Cannot handle drawing preedited text yet
2840 gtk_im_context_set_use_preedit( (GtkIMContext*) m_imContext, FALSE );
2841
2842 g_signal_connect (G_OBJECT (m_imContext), "commit",
2843 G_CALLBACK (gtk_wxwindow_commit_cb), this);
4e5a4c69 2844#endif
b02da6b1 2845 }
148cd9b6 2846
67d78217 2847 // these are called when the "sunken" or "raised" borders are drawn
034be888
RR
2848 gtk_signal_connect( GTK_OBJECT(m_widget), "expose_event",
2849 GTK_SIGNAL_FUNC(gtk_window_own_expose_callback), (gpointer)this );
2850
4e5a4c69 2851#ifndef __WXGTK20__
034be888
RR
2852 gtk_signal_connect( GTK_OBJECT(m_widget), "draw",
2853 GTK_SIGNAL_FUNC(gtk_window_own_draw_callback), (gpointer)this );
4e5a4c69 2854#endif
43a18898 2855 }
47d67540 2856
76fcf0f2 2857 // focus handling
63081513 2858
76fcf0f2
RR
2859 if (m_focusWidget == NULL)
2860 m_focusWidget = m_widget;
2daa0ce9 2861
76fcf0f2
RR
2862 gtk_signal_connect( GTK_OBJECT(m_focusWidget), "focus_in_event",
2863 GTK_SIGNAL_FUNC(gtk_window_focus_in_callback), (gpointer)this );
2864
2865 gtk_signal_connect( GTK_OBJECT(m_focusWidget), "focus_out_event",
2866 GTK_SIGNAL_FUNC(gtk_window_focus_out_callback), (gpointer)this );
2867
2868 // connect to the various key and mouse handlers
63081513 2869
a2053b27 2870 GtkWidget *connect_widget = GetConnectWidget();
f03fc89f 2871
a2053b27 2872 ConnectWidget( connect_widget );
47d67540 2873
63081513 2874 /* We cannot set colours, fonts and cursors before the widget has
a2053b27
RR
2875 been realized, so we do this directly after realization */
2876 gtk_signal_connect( GTK_OBJECT(connect_widget), "realize",
c50f1fb9 2877 GTK_SIGNAL_FUNC(gtk_window_realized_callback), (gpointer) this );
2daa0ce9 2878
63081513
RR
2879 if (m_wxwindow)
2880 {
47c93b63 2881 // Catch native resize events
8f75cb6c
RR
2882 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "size_allocate",
2883 GTK_SIGNAL_FUNC(gtk_window_size_callback), (gpointer)this );
2daa0ce9 2884
47c93b63 2885 // Initialize XIM support
63081513
RR
2886 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "realize",
2887 GTK_SIGNAL_FUNC(gtk_wxwindow_realized_callback), (gpointer) this );
8f75cb6c 2888
47c93b63 2889 // And resize XIM window
b79395c5
RR
2890 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "size_allocate",
2891 GTK_SIGNAL_FUNC(gtk_wxwindow_size_callback), (gpointer)this );
63081513 2892 }
2daa0ce9 2893
47c93b63
RR
2894 if (!GTK_IS_COMBO(m_widget))
2895 {
2896 // This is needed if we want to add our windows into native
2897 // GTK control, such as the toolbar. With this callback, the
2898 // toolbar gets to know the correct size (the one set by the
2899 // programmer). Sadly, it misbehaves for wxComboBox. FIXME
2900 // when moving to GTK 2.0.
2901 gtk_signal_connect( GTK_OBJECT(m_widget), "size_request",
2902 GTK_SIGNAL_FUNC(gtk_window_size_request_callback), (gpointer) this );
2903 }
1e6feb95 2904
43a18898 2905 m_hasVMT = TRUE;
b4071e91
RR
2906}
2907
1e6feb95 2908void wxWindowGTK::ConnectWidget( GtkWidget *widget )
b4071e91 2909{
43a18898
RR
2910 gtk_signal_connect( GTK_OBJECT(widget), "key_press_event",
2911 GTK_SIGNAL_FUNC(gtk_window_key_press_callback), (gpointer)this );
c801d85f 2912
b666df2c
RR
2913 gtk_signal_connect( GTK_OBJECT(widget), "key_release_event",
2914 GTK_SIGNAL_FUNC(gtk_window_key_release_callback), (gpointer)this );
2915
43a18898
RR
2916 gtk_signal_connect( GTK_OBJECT(widget), "button_press_event",
2917 GTK_SIGNAL_FUNC(gtk_window_button_press_callback), (gpointer)this );
47d67540 2918
43a18898
RR
2919 gtk_signal_connect( GTK_OBJECT(widget), "button_release_event",
2920 GTK_SIGNAL_FUNC(gtk_window_button_release_callback), (gpointer)this );
47d67540 2921
43a18898
RR
2922 gtk_signal_connect( GTK_OBJECT(widget), "motion_notify_event",
2923 GTK_SIGNAL_FUNC(gtk_window_motion_notify_callback), (gpointer)this );
47d67540 2924
43a18898
RR
2925 gtk_signal_connect( GTK_OBJECT(widget), "enter_notify_event",
2926 GTK_SIGNAL_FUNC(gtk_window_enter_callback), (gpointer)this );
47d67540 2927
43a18898
RR
2928 gtk_signal_connect( GTK_OBJECT(widget), "leave_notify_event",
2929 GTK_SIGNAL_FUNC(gtk_window_leave_callback), (gpointer)this );
362c6693 2930}
c801d85f 2931
1e6feb95 2932bool wxWindowGTK::Destroy()
c801d85f 2933{
223d09f6 2934 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
47d67540 2935
43a18898 2936 m_hasVMT = FALSE;
c801d85f 2937
f03fc89f 2938 return wxWindowBase::Destroy();
362c6693 2939}
c801d85f 2940
1e6feb95 2941void wxWindowGTK::DoMoveWindow(int x, int y, int width, int height)
23efdd02
RR
2942{
2943 gtk_pizza_set_size( GTK_PIZZA(m_parent->m_wxwindow), m_widget, x, y, width, height );
2944}
2daa0ce9 2945
1e6feb95 2946void wxWindowGTK::DoSetSize( int x, int y, int width, int height, int sizeFlags )
c801d85f 2947{
223d09f6 2948 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
1e6feb95 2949 wxASSERT_MSG( (m_parent != NULL), wxT("wxWindowGTK::SetSize requires parent.\n") );
8bbe427f 2950
33611ebb 2951/*
f94fca1b 2952 printf( "DoSetSize: name %s, x,y,w,h: %d,%d,%d,%d \n", GetName().c_str(), x,y,width,height );
33611ebb
RR
2953*/
2954
e27ce4e9 2955 if (m_resizing) return; /* I don't like recursions */
fb1585ae 2956 m_resizing = TRUE;
1e6feb95 2957
b9f29261
VS
2958 int currentX, currentY;
2959 GetPosition(&currentX, &currentY);
2960 if (x == -1)
2961 x = currentX;
a200c35e 2962 if (y == -1)
b9f29261 2963 y = currentY;
a200c35e
VS
2964 AdjustForParentClientOrigin(x, y, sizeFlags);
2965
a2053b27 2966 if (m_parent->m_wxwindow == NULL) /* i.e. wxNotebook */
fb1585ae 2967 {
e27ce4e9 2968 /* don't set the size for children of wxNotebook, just take the values. */
fb1585ae
RR
2969 m_x = x;
2970 m_y = y;
2971 m_width = width;
ba4e3652 2972 m_height = height;
fb1585ae 2973 }
ba4e3652 2974 else
fb1585ae 2975 {
da048e3d 2976 GtkPizza *pizza = GTK_PIZZA(m_parent->m_wxwindow);
85ad5eb5 2977 if ((sizeFlags & wxSIZE_ALLOW_MINUS_ONE) == 0)
ba4e3652 2978 {
da048e3d
RR
2979 if (x != -1) m_x = x + pizza->xoffset;
2980 if (y != -1) m_y = y + pizza->yoffset;
ba4e3652
RR
2981 }
2982 else
2983 {
da048e3d
RR
2984 m_x = x + pizza->xoffset;
2985 m_y = y + pizza->yoffset;
ba4e3652 2986 }
d44c23ce
RR
2987 if (width != -1) m_width = width;
2988 if (height != -1) m_height = height;
47d67540 2989
ba4e3652
RR
2990 if ((sizeFlags & wxSIZE_AUTO_WIDTH) == wxSIZE_AUTO_WIDTH)
2991 {
2992 if (width == -1) m_width = 80;
2993 }
2994
2995 if ((sizeFlags & wxSIZE_AUTO_HEIGHT) == wxSIZE_AUTO_HEIGHT)
2996 {
2997 if (height == -1) m_height = 26;
2998 }
8bbe427f 2999
e7dda1ff
VS
3000 int minWidth = GetMinWidth(),
3001 minHeight = GetMinHeight(),
3002 maxWidth = GetMaxWidth(),
3003 maxHeight = GetMaxHeight();
3004
3005 if ((minWidth != -1) && (m_width < minWidth)) m_width = minWidth;
3006 if ((minHeight != -1) && (m_height < minHeight)) m_height = minHeight;
3007 if ((maxWidth != -1) && (m_width > maxWidth)) m_width = maxWidth;
3008 if ((maxHeight != -1) && (m_height > maxHeight)) m_height = maxHeight;
47d67540 3009
a2053b27 3010 int border = 0;
c50f1fb9 3011 int bottom_border = 0;
f03fc89f 3012
eb9e6a00 3013#ifndef __WXGTK20__
29f538ce 3014 if (GTK_WIDGET_CAN_DEFAULT(m_widget))
c50f1fb9
VZ
3015 {
3016 /* the default button has a border around it */
3017 border = 6;
3018 bottom_border = 5;
3019 }
67d78217 3020#endif
c50f1fb9 3021
23efdd02
RR
3022 DoMoveWindow( m_x-border,
3023 m_y-border,
3024 m_width+2*border,
3025 m_height+border+bottom_border );
54517652 3026 }
148cd9b6 3027
5b8a521e
RR
3028 if (m_hasScrolling)
3029 {
1e6feb95 3030 /* Sometimes the client area changes size without the
b6fa52db
RR
3031 whole windows's size changing, but if the whole
3032 windows's size doesn't change, no wxSizeEvent will
3033 normally be sent. Here we add an extra test if
3034 the client test has been changed and this will
3035 be used then. */
5b8a521e
RR
3036 GetClientSize( &m_oldClientWidth, &m_oldClientHeight );
3037 }
3038
54517652 3039/*
6d693bb4
RR
3040 wxPrintf( "OnSize sent from " );
3041 if (GetClassInfo() && GetClassInfo()->GetClassName())
3042 wxPrintf( GetClassInfo()->GetClassName() );
3043 wxPrintf( " %d %d %d %d\n", (int)m_x, (int)m_y, (int)m_width, (int)m_height );
3044*/
3045
30760ce7
RR
3046 if (!m_nativeSizeEvent)
3047 {
3048 wxSizeEvent event( wxSize(m_width,m_height), GetId() );
3049 event.SetEventObject( this );
3050 GetEventHandler()->ProcessEvent( event );
3051 }
6d693bb4 3052
fb1585ae 3053 m_resizing = FALSE;
362c6693 3054}
c801d85f 3055
1e6feb95 3056void wxWindowGTK::OnInternalIdle()
9390a202 3057{
beab25bd 3058 // Update invalidated regions.
010afced 3059 GtkUpdate();
0fc5dbf5 3060
beab25bd 3061 // Synthetize activate events.
148cd9b6
VZ
3062 if ( g_sendActivateEvent != -1 )
3063 {
3064 bool activate = g_sendActivateEvent != 0;
3065
3066 // do it only once
3067 g_sendActivateEvent = -1;
3068
1e6feb95 3069 wxTheApp->SetActive(activate, (wxWindow *)g_focusWindowLast);
148cd9b6
VZ
3070 }
3071
e8c12a53
VS
3072 if ( g_activeFrameLostFocus )
3073 {
3074 if ( g_activeFrame )
3075 {
45eb5249 3076 wxLogTrace(wxT("activate"), wxT("Deactivating frame %p (from idle)"), g_activeFrame);
e8c12a53
VS
3077 wxActivateEvent event(wxEVT_ACTIVATE, FALSE, g_activeFrame->GetId());
3078 event.SetEventObject(g_activeFrame);
3079 g_activeFrame->GetEventHandler()->ProcessEvent(event);
3080 g_activeFrame = NULL;
3081 }
3082 g_activeFrameLostFocus = FALSE;
3083 }
2b5f62a0 3084
9146082c
RR
3085 wxCursor cursor = m_cursor;
3086 if (g_globalCursor.Ok()) cursor = g_globalCursor;
c50f1fb9 3087
f7a11f8c 3088 if (cursor.Ok())
9146082c 3089 {
3017f78d 3090 /* I now set the cursor anew in every OnInternalIdle call
b02da6b1
VZ
3091 as setting the cursor in a parent window also effects the
3092 windows above so that checking for the current cursor is
3093 not possible. */
148cd9b6 3094
9146082c 3095 if (m_wxwindow)
6a008b33 3096 {
da048e3d 3097 GdkWindow *window = GTK_PIZZA(m_wxwindow)->bin_window;
6a008b33 3098 if (window)
c50f1fb9 3099 gdk_window_set_cursor( window, cursor.GetCursor() );
6a008b33
VZ
3100
3101 if (!g_globalCursor.Ok())
3102 cursor = *wxSTANDARD_CURSOR;
3103
3104 window = m_widget->window;
5e014a0c 3105 if ((window) && !(GTK_WIDGET_NO_WINDOW(m_widget)))
9146082c 3106 gdk_window_set_cursor( window, cursor.GetCursor() );
5e014a0c 3107
6a008b33
VZ
3108 }
3109 else
3110 {
5e014a0c 3111
9146082c 3112 GdkWindow *window = m_widget->window;
5e014a0c 3113 if ((window) && !(GTK_WIDGET_NO_WINDOW(m_widget)))
9146082c 3114 gdk_window_set_cursor( window, cursor.GetCursor() );
5e014a0c 3115
6a008b33 3116 }
9146082c 3117 }
6a008b33 3118
9390a202
RR
3119 UpdateWindowUI();
3120}
3121
1e6feb95 3122void wxWindowGTK::DoGetSize( int *width, int *height ) const
c801d85f 3123{
223d09f6 3124 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3125
fb1585ae
RR
3126 if (width) (*width) = m_width;
3127 if (height) (*height) = m_height;
362c6693 3128}
c801d85f 3129
1e6feb95 3130void wxWindowGTK::DoSetClientSize( int width, int height )
c801d85f 3131{
223d09f6 3132 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3133
1ecc4d80 3134 if (!m_wxwindow)
c801d85f 3135 {
1ecc4d80 3136 SetSize( width, height );
c801d85f
KB
3137 }
3138 else
3139 {
1ecc4d80
RR
3140 int dw = 0;
3141 int dh = 0;
3142
1e6feb95 3143#ifndef __WXUNIVERSAL__
98d3fdbe
RR
3144 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
3145 {
5e014a0c 3146 /* when using GTK 1.2 we set the shadow border size to 2 */
6a008b33 3147 dw += 2 * 2;
98d3fdbe
RR
3148 dh += 2 * 2;
3149 }
5e014a0c
RR
3150 if (HasFlag(wxSIMPLE_BORDER))
3151 {
3152 /* when using GTK 1.2 we set the simple border size to 1 */
3153 dw += 1 * 2;
3154 dh += 1 * 2;
3155 }
1e6feb95 3156#endif // __WXUNIVERSAL__
034be888 3157
5b8a521e 3158 if (m_hasScrolling)
98d3fdbe 3159 {
324dbfec 3160 GtkScrolledWindow *scroll_window = GTK_SCROLLED_WINDOW(m_widget);
2daa0ce9 3161
9000c624
RR
3162 GtkRequisition vscroll_req;
3163 vscroll_req.width = 2;
3164 vscroll_req.height = 2;
dd00f3f6 3165 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->vscrollbar) )->size_request )
9000c624 3166 (scroll_window->vscrollbar, &vscroll_req );
2daa0ce9 3167
9000c624
RR
3168 GtkRequisition hscroll_req;
3169 hscroll_req.width = 2;
3170 hscroll_req.height = 2;
dd00f3f6 3171 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->hscrollbar) )->size_request )
9000c624
RR
3172 (scroll_window->hscrollbar, &hscroll_req );
3173
dd00f3f6 3174 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
324dbfec 3175
1ecc4d80
RR
3176 if (scroll_window->vscrollbar_visible)
3177 {
9000c624 3178 dw += vscroll_req.width;
1ecc4d80
RR
3179 dw += scroll_class->scrollbar_spacing;
3180 }
3181
3182 if (scroll_window->hscrollbar_visible)
3183 {
9000c624 3184 dh += hscroll_req.height;
63cc5d9d 3185 dh += scroll_class->scrollbar_spacing;
1ecc4d80 3186 }
9000c624 3187 }
1ecc4d80 3188
034be888 3189 SetSize( width+dw, height+dh );
1ecc4d80 3190 }
362c6693 3191}
c801d85f 3192
1e6feb95 3193void wxWindowGTK::DoGetClientSize( int *width, int *height ) const
c801d85f 3194{
223d09f6 3195 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3196
1ecc4d80
RR
3197 if (!m_wxwindow)
3198 {
3199 if (width) (*width) = m_width;
3200 if (height) (*height) = m_height;
c801d85f
KB
3201 }
3202 else
3203 {
1ecc4d80
RR
3204 int dw = 0;
3205 int dh = 0;
3206
1e6feb95 3207#ifndef __WXUNIVERSAL__
98d3fdbe
RR
3208 if (HasFlag(wxRAISED_BORDER) || HasFlag(wxSUNKEN_BORDER))
3209 {
5e014a0c 3210 /* when using GTK 1.2 we set the shadow border size to 2 */
6a008b33 3211 dw += 2 * 2;
98d3fdbe
RR
3212 dh += 2 * 2;
3213 }
5e014a0c
RR
3214 if (HasFlag(wxSIMPLE_BORDER))
3215 {
3216 /* when using GTK 1.2 we set the simple border size to 1 */
3217 dw += 1 * 2;
3218 dh += 1 * 2;
3219 }
1e6feb95 3220#endif // __WXUNIVERSAL__
9000c624 3221
5b8a521e 3222 if (m_hasScrolling)
98d3fdbe 3223 {
6a008b33 3224 GtkScrolledWindow *scroll_window = GTK_SCROLLED_WINDOW(m_widget);
2daa0ce9 3225
9000c624
RR
3226 GtkRequisition vscroll_req;
3227 vscroll_req.width = 2;
3228 vscroll_req.height = 2;
dd00f3f6 3229 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->vscrollbar) )->size_request )
9000c624 3230 (scroll_window->vscrollbar, &vscroll_req );
2daa0ce9 3231
9000c624
RR
3232 GtkRequisition hscroll_req;
3233 hscroll_req.width = 2;
3234 hscroll_req.height = 2;
dd00f3f6 3235 (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(scroll_window->hscrollbar) )->size_request )
9000c624
RR
3236 (scroll_window->hscrollbar, &hscroll_req );
3237
dd00f3f6 3238 GtkScrolledWindowClass *scroll_class = GTK_SCROLLED_WINDOW_CLASS( GTK_OBJECT_GET_CLASS(m_widget) );
6a008b33 3239
1ecc4d80
RR
3240 if (scroll_window->vscrollbar_visible)
3241 {
9000c624 3242 dw += vscroll_req.width;
1ecc4d80
RR
3243 dw += scroll_class->scrollbar_spacing;
3244 }
3245
3246 if (scroll_window->hscrollbar_visible)
3247 {
9000c624 3248 dh += hscroll_req.height;
1ecc4d80
RR
3249 dh += scroll_class->scrollbar_spacing;
3250 }
6a008b33 3251 }
47d67540 3252
1ecc4d80
RR
3253 if (width) (*width) = m_width - dw;
3254 if (height) (*height) = m_height - dh;
3255 }
1e6feb95 3256
f94fca1b
RR
3257/*
3258 printf( "GetClientSize, name %s ", GetName().c_str() );
3259 if (width) printf( " width = %d", (*width) );
3260 if (height) printf( " height = %d", (*height) );
3261 printf( "\n" );
3262*/
362c6693 3263}
c801d85f 3264
1e6feb95 3265void wxWindowGTK::DoGetPosition( int *x, int *y ) const
c801d85f 3266{
223d09f6 3267 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3268
bf0c00c6
RR
3269 int dx = 0;
3270 int dy = 0;
3271 if (m_parent && m_parent->m_wxwindow)
3272 {
da048e3d 3273 GtkPizza *pizza = GTK_PIZZA(m_parent->m_wxwindow);
b02da6b1
VZ
3274 dx = pizza->xoffset;
3275 dy = pizza->yoffset;
bf0c00c6 3276 }
94633ad9 3277
496beb3f
VS
3278 if (x) (*x) = m_x - dx;
3279 if (y) (*y) = m_y - dy;
362c6693 3280}
c801d85f 3281
1e6feb95 3282void wxWindowGTK::DoClientToScreen( int *x, int *y ) const
c801d85f 3283{
223d09f6 3284 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3285
a2053b27
RR
3286 if (!m_widget->window) return;
3287
43a18898
RR
3288 GdkWindow *source = (GdkWindow *) NULL;
3289 if (m_wxwindow)
da048e3d 3290 source = GTK_PIZZA(m_wxwindow)->bin_window;
43a18898
RR
3291 else
3292 source = m_widget->window;
47d67540 3293
43a18898
RR
3294 int org_x = 0;
3295 int org_y = 0;
3296 gdk_window_get_origin( source, &org_x, &org_y );
c801d85f 3297
43a18898 3298 if (!m_wxwindow)
c801d85f 3299 {
43a18898
RR
3300 if (GTK_WIDGET_NO_WINDOW (m_widget))
3301 {
3302 org_x += m_widget->allocation.x;
3303 org_y += m_widget->allocation.y;
3304 }
362c6693 3305 }
47d67540 3306
43a18898
RR
3307 if (x) *x += org_x;
3308 if (y) *y += org_y;
362c6693 3309}
c801d85f 3310
1e6feb95 3311void wxWindowGTK::DoScreenToClient( int *x, int *y ) const
c801d85f 3312{
223d09f6 3313 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3314
a2053b27
RR
3315 if (!m_widget->window) return;
3316
1ecc4d80
RR
3317 GdkWindow *source = (GdkWindow *) NULL;
3318 if (m_wxwindow)
da048e3d 3319 source = GTK_PIZZA(m_wxwindow)->bin_window;
1ecc4d80
RR
3320 else
3321 source = m_widget->window;
47d67540 3322
1ecc4d80
RR
3323 int org_x = 0;
3324 int org_y = 0;
3325 gdk_window_get_origin( source, &org_x, &org_y );
c801d85f 3326
1ecc4d80 3327 if (!m_wxwindow)
c801d85f 3328 {
1ecc4d80
RR
3329 if (GTK_WIDGET_NO_WINDOW (m_widget))
3330 {
3331 org_x += m_widget->allocation.x;
3332 org_y += m_widget->allocation.y;
3333 }
362c6693 3334 }
47d67540 3335
1ecc4d80
RR
3336 if (x) *x -= org_x;
3337 if (y) *y -= org_y;
362c6693 3338}
c801d85f 3339
1e6feb95 3340bool wxWindowGTK::Show( bool show )
c801d85f 3341{
223d09f6 3342 wxCHECK_MSG( (m_widget != NULL), FALSE, wxT("invalid window") );
47d67540 3343
739730ca
RR
3344 if (!wxWindowBase::Show(show))
3345 {
3346 // nothing to do
f03fc89f 3347 return FALSE;
739730ca 3348 }
8bbe427f 3349
f03fc89f
VZ
3350 if (show)
3351 gtk_widget_show( m_widget );
1ecc4d80 3352 else
f03fc89f 3353 gtk_widget_hide( m_widget );
8bbe427f 3354
2b5f62a0
VZ
3355 wxShowEvent eventShow(GetId(), show);
3356 eventShow.m_eventObject = this;
3357
3358 GetEventHandler()->ProcessEvent(eventShow);
3359
f03fc89f 3360 return TRUE;
362c6693 3361}
c801d85f 3362
3379ed37 3363static void wxWindowNotifyEnable(wxWindowGTK* win, bool enable)
fdca68a6
JS
3364{
3365 win->OnParentEnable(enable);
3366
3367 // Recurse, so that children have the opportunity to Do The Right Thing
3368 // and reset colours that have been messed up by a parent's (really ancestor's)
3369 // Enable call
3370 for ( wxWindowList::Node *node = win->GetChildren().GetFirst();
3371 node;
3372 node = node->GetNext() )
3373 {
3374 wxWindow *child = node->GetData();
3375 if (!child->IsKindOf(CLASSINFO(wxDialog)) && !child->IsKindOf(CLASSINFO(wxFrame)))
3376 wxWindowNotifyEnable(child, enable);
3377 }
3378}
3379
3379ed37 3380bool wxWindowGTK::Enable( bool enable )
c801d85f 3381{
223d09f6 3382 wxCHECK_MSG( (m_widget != NULL), FALSE, wxT("invalid window") );
5e0aa05a 3383
739730ca
RR
3384 if (!wxWindowBase::Enable(enable))
3385 {
3386 // nothing to do
f03fc89f 3387 return FALSE;
739730ca 3388 }
1ecc4d80 3389
f03fc89f
VZ
3390 gtk_widget_set_sensitive( m_widget, enable );
3391 if ( m_wxwindow )
3392 gtk_widget_set_sensitive( m_wxwindow, enable );
ff8bfdbb 3393
fdca68a6 3394 wxWindowNotifyEnable(this, enable);
513903c4 3395
f03fc89f 3396 return TRUE;
362c6693 3397}
c801d85f 3398
1e6feb95 3399int wxWindowGTK::GetCharHeight() const
2f2aa628 3400{
223d09f6 3401 wxCHECK_MSG( (m_widget != NULL), 12, wxT("invalid window") );
47d67540 3402
223d09f6 3403 wxCHECK_MSG( m_font.Ok(), 12, wxT("invalid font") );
2f2aa628 3404
bbd006c0
RR
3405#ifdef __WXGTK20__
3406 PangoContext *context = NULL;
3407 if (m_widget)
3408 context = gtk_widget_get_pango_context( m_widget );
3409
3410 if (!context)
3411 return 0;
3412
3413 PangoFontDescription *desc = m_font.GetNativeFontInfo()->description;
3414 PangoLayout *layout = pango_layout_new(context);
3415 pango_layout_set_font_description(layout, desc);
3416 pango_layout_set_text(layout, "H", 1);
3417 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3418
3419 PangoRectangle rect;
3420 pango_layout_line_get_extents(line, NULL, &rect);
3421
3422 g_object_unref( G_OBJECT( layout ) );
7de59551 3423
bbd006c0
RR
3424 return (int) (rect.height / PANGO_SCALE);
3425#else
f03fc89f
VZ
3426 GdkFont *font = m_font.GetInternalFont( 1.0 );
3427
3428 return font->ascent + font->descent;
bbd006c0 3429#endif
362c6693 3430}
c801d85f 3431
1e6feb95 3432int wxWindowGTK::GetCharWidth() const
c33c4050 3433{
223d09f6 3434 wxCHECK_MSG( (m_widget != NULL), 8, wxT("invalid window") );
47d67540 3435
223d09f6 3436 wxCHECK_MSG( m_font.Ok(), 8, wxT("invalid font") );
47d67540 3437
bbd006c0
RR
3438#ifdef __WXGTK20__
3439 PangoContext *context = NULL;
3440 if (m_widget)
3441 context = gtk_widget_get_pango_context( m_widget );
3442
3443 if (!context)
3444 return 0;
3445
3446 PangoFontDescription *desc = m_font.GetNativeFontInfo()->description;
3447 PangoLayout *layout = pango_layout_new(context);
3448 pango_layout_set_font_description(layout, desc);
3449 pango_layout_set_text(layout, "H", 1);
3450 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
3451
3452 PangoRectangle rect;
3453 pango_layout_line_get_extents(line, NULL, &rect);
3454
3455 g_object_unref( G_OBJECT( layout ) );
7de59551 3456
bbd006c0
RR
3457 return (int) (rect.width / PANGO_SCALE);
3458#else
463c1fa1 3459 GdkFont *font = m_font.GetInternalFont( 1.0 );
ff8bfdbb 3460
463c1fa1 3461 return gdk_string_width( font, "H" );
bbd006c0 3462#endif
c33c4050
RR
3463}
3464
1e6feb95 3465void wxWindowGTK::GetTextExtent( const wxString& string,
f03fc89f
VZ
3466 int *x,
3467 int *y,
3468 int *descent,
3469 int *externalLeading,
3470 const wxFont *theFont ) const
c33c4050 3471{
463c1fa1
RR
3472 wxFont fontToUse = m_font;
3473 if (theFont) fontToUse = *theFont;
47d67540 3474
223d09f6 3475 wxCHECK_RET( fontToUse.Ok(), wxT("invalid font") );
2b5f62a0 3476
48d011c8
RR
3477 if (string.IsEmpty())
3478 {
b15ed747
RR
3479 if (x) (*x) = 0;
3480 if (y) (*y) = 0;
48d011c8
RR
3481 return;
3482 }
47d67540 3483
2b5f62a0 3484#ifdef __WXGTK20__
48d011c8
RR
3485 PangoContext *context = NULL;
3486 if (m_widget)
2b5f62a0
VZ
3487 context = gtk_widget_get_pango_context( m_widget );
3488
48d011c8
RR
3489 if (!context)
3490 {
b15ed747
RR
3491 if (x) (*x) = 0;
3492 if (y) (*y) = 0;
48d011c8
RR
3493 return;
3494 }
2b5f62a0 3495
48d011c8
RR
3496 PangoFontDescription *desc = fontToUse.GetNativeFontInfo()->description;
3497 PangoLayout *layout = pango_layout_new(context);
3498 pango_layout_set_font_description(layout, desc);
3499 {
fb3ed106 3500#if wxUSE_UNICODE
48d011c8
RR
3501 const wxCharBuffer data = wxConvUTF8.cWC2MB( string );
3502 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
fb3ed106
RR
3503#else
3504 const wxWCharBuffer wdata = wxConvLocal.cMB2WC( string );
3505 const wxCharBuffer data = wxConvUTF8.cWC2MB( wdata );
3506 pango_layout_set_text(layout, (const char*) data, strlen( (const char*) data ));
3507#endif
48d011c8
RR
3508 }
3509 PangoLayoutLine *line = (PangoLayoutLine *)pango_layout_get_lines(layout)->data;
2b5f62a0 3510
48d011c8
RR
3511 PangoRectangle rect;
3512 pango_layout_line_get_extents(line, NULL, &rect);
2b5f62a0 3513
b15ed747
RR
3514 if (x) (*x) = (wxCoord) (rect.width / PANGO_SCALE);
3515 if (y) (*y) = (wxCoord) (rect.height / PANGO_SCALE);
48d011c8
RR
3516 if (descent)
3517 {
3518 // Do something about metrics here
3519 (*descent) = 0;
3520 }
3521 if (externalLeading) (*externalLeading) = 0; // ??
2b5f62a0 3522
48d011c8
RR
3523 g_object_unref( G_OBJECT( layout ) );
3524#else
463c1fa1 3525 GdkFont *font = fontToUse.GetInternalFont( 1.0 );
fab591c5 3526 if (x) (*x) = gdk_string_width( font, wxGTK_CONV( string ) );
463c1fa1
RR
3527 if (y) (*y) = font->ascent + font->descent;
3528 if (descent) (*descent) = font->descent;
3529 if (externalLeading) (*externalLeading) = 0; // ??
48d011c8 3530#endif
c33c4050
RR
3531}
3532
1e6feb95 3533void wxWindowGTK::SetFocus()
c801d85f 3534{
6cad4f1b
VZ
3535 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
3536
3537 if ( m_hasFocus )
3538 {
3539 // don't do anything if we already have focus
3540 return;
3541 }
2daa0ce9 3542
354aa1e3
RR
3543 if (m_wxwindow)
3544 {
173348db 3545 if (!GTK_WIDGET_HAS_FOCUS (m_wxwindow))
b231914f 3546 {
173348db 3547 gtk_widget_grab_focus (m_wxwindow);
b231914f 3548 }
354aa1e3 3549 }
b231914f 3550 else if (m_widget)
c801d85f 3551 {
173348db 3552 if (GTK_WIDGET_CAN_FOCUS(m_widget) && !GTK_WIDGET_HAS_FOCUS (m_widget) )
463c1fa1 3553 {
d7fa7eaa 3554 if (!GTK_WIDGET_REALIZED(m_widget))
6aeb6f2a 3555 {
6cad4f1b
VZ
3556 // we can't set the focus to the widget now so we remember that
3557 // it should be focused and will do it later, during the idle
3558 // time, as soon as we can
3559 wxLogTrace(TRACE_FOCUS,
3560 _T("Delaying setting focus to %s(%s)"),
6aeb6f2a
VZ
3561 GetClassInfo()->GetClassName(), GetLabel().c_str());
3562
d7fa7eaa 3563 g_delayedFocus = this;
6aeb6f2a 3564 }
d7fa7eaa 3565 else
6aeb6f2a 3566 {
6cad4f1b
VZ
3567 wxLogTrace(TRACE_FOCUS,
3568 _T("Setting focus to %s(%s)"),
6aeb6f2a
VZ
3569 GetClassInfo()->GetClassName(), GetLabel().c_str());
3570
d7fa7eaa 3571 gtk_widget_grab_focus (m_widget);
6aeb6f2a 3572 }
463c1fa1 3573 }
354aa1e3 3574 else if (GTK_IS_CONTAINER(m_widget))
ff8bfdbb 3575 {
9e691f46 3576 SET_CONTAINER_FOCUS( m_widget, GTK_DIR_TAB_FORWARD );
ff8bfdbb
VZ
3577 }
3578 else
3579 {
6cad4f1b
VZ
3580 wxLogTrace(TRACE_FOCUS,
3581 _T("Can't set focus to %s(%s)"),
3582 GetClassInfo()->GetClassName(), GetLabel().c_str());
ff8bfdbb 3583 }
362c6693 3584 }
362c6693 3585}
c801d85f 3586
1e6feb95 3587bool wxWindowGTK::AcceptsFocus() const
b292e2f5 3588{
f03fc89f 3589 return m_acceptsFocus && wxWindowBase::AcceptsFocus();
b292e2f5
RR
3590}
3591
1e6feb95 3592bool wxWindowGTK::Reparent( wxWindowBase *newParentBase )
463c1fa1 3593{
223d09f6 3594 wxCHECK_MSG( (m_widget != NULL), FALSE, wxT("invalid window") );
c50f1fb9 3595
1e6feb95
VZ
3596 wxWindowGTK *oldParent = m_parent,
3597 *newParent = (wxWindowGTK *)newParentBase;
a2053b27 3598
5fd11f09
RR
3599 wxASSERT( GTK_IS_WIDGET(m_widget) );
3600
f03fc89f
VZ
3601 if ( !wxWindowBase::Reparent(newParent) )
3602 return FALSE;
8bbe427f 3603
5fd11f09
RR
3604 wxASSERT( GTK_IS_WIDGET(m_widget) );
3605
3606 /* prevent GTK from deleting the widget arbitrarily */
3607 gtk_widget_ref( m_widget );
3608
8ce63e9d
RR
3609 if (oldParent)
3610 {
3017f78d 3611 gtk_container_remove( GTK_CONTAINER(m_widget->parent), m_widget );
8ce63e9d 3612 }
c50f1fb9 3613
5fd11f09
RR
3614 wxASSERT( GTK_IS_WIDGET(m_widget) );
3615
8ce63e9d
RR
3616 if (newParent)
3617 {
3618 /* insert GTK representation */
3619 (*(newParent->m_insertCallback))(newParent, this);
3620 }
c50f1fb9 3621
5fd11f09
RR
3622 /* reverse: prevent GTK from deleting the widget arbitrarily */
3623 gtk_widget_unref( m_widget );
148cd9b6 3624
f03fc89f 3625 return TRUE;
362c6693 3626}
c801d85f 3627
1e6feb95 3628void wxWindowGTK::DoAddChild(wxWindowGTK *child)
ddb6bc71 3629{
223d09f6 3630 wxASSERT_MSG( (m_widget != NULL), wxT("invalid window") );
ddb6bc71 3631
223d09f6 3632 wxASSERT_MSG( (child != NULL), wxT("invalid child window") );
ddb6bc71 3633
223d09f6 3634 wxASSERT_MSG( (m_insertCallback != NULL), wxT("invalid child insertion function") );
c50f1fb9 3635
ddb6bc71
RR
3636 /* add to list */
3637 AddChild( child );
c50f1fb9 3638
ddb6bc71
RR
3639 /* insert GTK representation */
3640 (*m_insertCallback)(this, child);
3641}
3642
1e6feb95 3643void wxWindowGTK::Raise()
362c6693 3644{
223d09f6 3645 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3646
a2053b27
RR
3647 if (!m_widget->window) return;
3648
f03fc89f 3649 gdk_window_raise( m_widget->window );
362c6693
RR
3650}
3651
1e6feb95 3652void wxWindowGTK::Lower()
362c6693 3653{
223d09f6 3654 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
47d67540 3655
a2053b27
RR
3656 if (!m_widget->window) return;
3657
f03fc89f 3658 gdk_window_lower( m_widget->window );
362c6693 3659}
c801d85f 3660
1e6feb95 3661bool wxWindowGTK::SetCursor( const wxCursor &cursor )
86b29a61 3662{
223d09f6 3663 wxCHECK_MSG( (m_widget != NULL), FALSE, wxT("invalid window") );
86b29a61 3664
f6bcfd97
BP
3665 if (cursor == m_cursor)
3666 return FALSE;
3667
3668 if (g_isIdle)
3669 wxapp_install_idle_handler();
1e6feb95 3670
f6bcfd97
BP
3671 if (cursor == wxNullCursor)
3672 return wxWindowBase::SetCursor( *wxSTANDARD_CURSOR );
3673 else
3674 return wxWindowBase::SetCursor( cursor );
362c6693 3675}
c801d85f 3676
1e6feb95 3677void wxWindowGTK::WarpPointer( int x, int y )
4f22cf8d 3678{
223d09f6 3679 wxCHECK_RET( (m_widget != NULL), wxT("invalid window") );
85eb36c2 3680
3bcc8d15
RR
3681 // We provide this function ourselves as it is
3682 // missing in GDK (top of this file).
148cd9b6 3683
ed673c6a
RR
3684 GdkWindow *window = (GdkWindow*) NULL;
3685 if (m_wxwindow)
da048e3d 3686 window = GTK_PIZZA(m_wxwindow)->bin_window;
ed673c6a
RR
3687 else
3688 window = GetConnectWidget()->window;
148cd9b6 3689
ed673c6a
RR
3690 if (window)
3691 gdk_window_warp_pointer( window, x, y );
4f22cf8d
RR
3692}
3693
3013a903 3694
1e6feb95 3695void wxWindowGTK::Refresh( bool eraseBackground, const wxRect *rect )
c801d85f 3696{
f2593d0d 3697 if (!m_widget) return;
a2053b27 3698 if (!m_widget->window) return;
2b5f62a0 3699
4e5a4c69 3700#ifndef __WXGTK20__
ea323db3
RR
3701 if (g_isIdle)
3702 wxapp_install_idle_handler();
2b5f62a0 3703
139adb6a 3704 if (eraseBackground && m_wxwindow && m_wxwindow->window)
c801d85f 3705 {
139adb6a
RR
3706 if (rect)
3707 {
3bcc8d15
RR
3708 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3709 m_clearRegion.Union( rect->x, rect->y, rect->width, rect->height );
139adb6a
RR
3710 }
3711 else
3712 {
3bcc8d15
RR
3713 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3714 m_clearRegion.Clear();
3715 m_clearRegion.Union( 0, 0, m_wxwindow->allocation.width, m_wxwindow->allocation.height );
139adb6a
RR
3716 }
3717 }
ff8bfdbb 3718
3bcc8d15 3719 if (rect)
139adb6a
RR
3720 {
3721 if (m_wxwindow)
b02da6b1 3722 {
23716407 3723 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3bcc8d15 3724 m_updateRegion.Union( rect->x, rect->y, rect->width, rect->height );
b02da6b1 3725 }
ff8bfdbb 3726 else
b6fa52db 3727 {
3bcc8d15
RR
3728 GdkRectangle gdk_rect;
3729 gdk_rect.x = rect->x;
3730 gdk_rect.y = rect->y;
3731 gdk_rect.width = rect->width;
3732 gdk_rect.height = rect->height;
3733 gtk_widget_draw( m_widget, &gdk_rect );
b6fa52db 3734 }
362c6693 3735 }
c801d85f 3736 else
139adb6a 3737 {
139adb6a 3738 if (m_wxwindow)
b02da6b1 3739 {
23716407 3740 // Schedule for later Updating in ::Update() or ::OnInternalIdle().
3bcc8d15
RR
3741 m_updateRegion.Clear();
3742 m_updateRegion.Union( 0, 0, m_wxwindow->allocation.width, m_wxwindow->allocation.height );
b02da6b1 3743 }
139adb6a 3744 else
b6fa52db 3745 {
3bcc8d15 3746 gtk_widget_draw( m_widget, (GdkRectangle*) NULL );
b6fa52db 3747 }
139adb6a 3748 }
4e5a4c69
RR
3749#else
3750 if (m_wxwindow)
3751 {
3752 if (rect)
3753 {
3754 GdkRectangle gdk_rect;
3755 gdk_rect.x = rect->x;
3756 gdk_rect.y = rect->y;
3757 gdk_rect.width = rect->width;
3758 gdk_rect.height = rect->height;
3759 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow)->bin_window, &gdk_rect, TRUE );
3760 }
3761 else
3762 {
3763 gdk_window_invalidate_rect( GTK_PIZZA(m_wxwindow)->bin_window, NULL, TRUE );
3764 }
3765 }
3766#endif
362c6693 3767}
c801d85f 3768
beab25bd 3769void wxWindowGTK::Update()
010afced
RR
3770{
3771 GtkUpdate();
3772}
3773
3774void wxWindowGTK::GtkUpdate()
beab25bd 3775{
4e5a4c69
RR
3776#ifdef __WXGTK20__
3777 if (m_wxwindow && GTK_PIZZA(m_wxwindow)->bin_window)
3778 gdk_window_process_updates( GTK_PIZZA(m_wxwindow)->bin_window, FALSE );
b15ed747 3779#else
beab25bd 3780 if (!m_updateRegion.IsEmpty())
23716407 3781 GtkSendPaintEvents();
b15ed747 3782#endif
beab25bd
RR
3783}
3784
3785void wxWindowGTK::GtkSendPaintEvents()
3786{
3bcc8d15
RR
3787 if (!m_wxwindow)
3788 {
2b5f62a0 3789#ifndef __WXGTK20__
3bcc8d15 3790 m_clearRegion.Clear();
b15ed747 3791#endif
3bcc8d15
RR
3792 m_updateRegion.Clear();
3793 return;
3794 }
beab25bd 3795
f90566f5 3796 // Clip to paint region in wxClientDC
3bcc8d15 3797 m_clipPaintRegion = TRUE;
fab591c5 3798
b15ed747
RR
3799#ifndef __WXGTK20__
3800 // widget to draw on
3801 GtkPizza *pizza = GTK_PIZZA (m_wxwindow);
2b5f62a0 3802
b15ed747 3803 // later for GTK 2.0, too.
f90566f5
RR
3804 if (GetThemeEnabled())
3805 {
3806 // find ancestor from which to steal background
3807 wxWindow *parent = GetParent();
3808 while (parent && !parent->IsTopLevel())
3809 parent = parent->GetParent();
3810 if (!parent)
cc06fe74 3811 parent = (wxWindow*)this;
2b5f62a0 3812
f90566f5
RR
3813 wxRegionIterator upd( m_updateRegion );
3814 while (upd)
3815 {
3816 GdkRectangle rect;
3817 rect.x = upd.GetX();
3818 rect.y = upd.GetY();
3819 rect.width = upd.GetWidth();
3820 rect.height = upd.GetHeight();
2b5f62a0 3821
f90566f5
RR
3822 gtk_paint_flat_box( parent->m_widget->style,
3823 pizza->bin_window,
3824 GTK_STATE_NORMAL,
3825 GTK_SHADOW_NONE,
3826 &rect,
3827 parent->m_widget,
3828 (char *)"base",
3829 0, 0, -1, -1 );
2b5f62a0 3830
f90566f5
RR
3831 upd ++;
3832 }
3833 }
3834 else
fab591c5 3835#endif
b15ed747
RR
3836
3837#ifdef __WXGTK20__
3838 {
3839 wxWindowDC dc( (wxWindow*)this );
3840 dc.SetClippingRegion( m_updateRegion );
3841
3842 wxEraseEvent erase_event( GetId(), &dc );
3843 erase_event.SetEventObject( this );
3844
3845 GetEventHandler()->ProcessEvent(erase_event);
3846 }
3847#else
3848 // if (!m_clearRegion.IsEmpty()) // Always send an erase event under GTK 1.2
beab25bd 3849 {
3bcc8d15 3850 wxWindowDC dc( (wxWindow*)this );
62cb3cd8
RR
3851 if (m_clearRegion.IsEmpty())
3852 dc.SetClippingRegion( m_updateRegion );
3853 else
3854 dc.SetClippingRegion( m_clearRegion );
0fc5dbf5 3855
3bcc8d15
RR
3856 wxEraseEvent erase_event( GetId(), &dc );
3857 erase_event.SetEventObject( this );
0fc5dbf5 3858
beab25bd
RR
3859 if (!GetEventHandler()->ProcessEvent(erase_event))
3860 {
994bc575
RR
3861 if (!g_eraseGC)
3862 {
f90566f5 3863 g_eraseGC = gdk_gc_new( pizza->bin_window );
994bc575
RR
3864 gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
3865 }
4e5a4c69 3866 gdk_gc_set_foreground( g_eraseGC, m_backgroundColour.GetColor() );
0fc5dbf5 3867
3bcc8d15 3868 wxRegionIterator upd( m_clearRegion );
beab25bd
RR
3869 while (upd)
3870 {
f90566f5
RR
3871 gdk_draw_rectangle( pizza->bin_window, g_eraseGC, 1,
3872 upd.GetX(), upd.GetY(), upd.GetWidth(), upd.GetHeight() );
beab25bd
RR
3873 upd ++;
3874 }
3875 }
3bcc8d15 3876 m_clearRegion.Clear();
beab25bd 3877 }
b15ed747 3878#endif
beab25bd
RR
3879
3880 wxNcPaintEvent nc_paint_event( GetId() );
3881 nc_paint_event.SetEventObject( this );
3882 GetEventHandler()->ProcessEvent( nc_paint_event );
3883
3884 wxPaintEvent paint_event( GetId() );
3885 paint_event.SetEventObject( this );
3886 GetEventHandler()->ProcessEvent( paint_event );
3887
beab25bd 3888 m_clipPaintRegion = FALSE;
c89f5c02 3889
8f3e7ecc 3890#ifndef __WXUNIVERSAL__
4e5a4c69 3891#ifndef __WXGTK20__
8f3e7ecc
RR
3892 // The following code will result in all window-less widgets
3893 // being redrawn because the wxWindows class is allowed to
3894 // paint over the window-less widgets.
0fc5dbf5 3895
8f3e7ecc
RR
3896 GList *children = pizza->children;
3897 while (children)
c89f5c02 3898 {
8f3e7ecc
RR
3899 GtkPizzaChild *child = (GtkPizzaChild*) children->data;
3900 children = children->next;
2b5f62a0 3901
8f3e7ecc
RR
3902 if (GTK_WIDGET_NO_WINDOW (child->widget) &&
3903 GTK_WIDGET_DRAWABLE (child->widget))
3904 {
3905 // Get intersection of widget area and update region
3906 wxRegion region( m_updateRegion );
0fc5dbf5 3907
8f3e7ecc
RR
3908 GdkEventExpose gdk_event;
3909 gdk_event.type = GDK_EXPOSE;
3910 gdk_event.window = pizza->bin_window;
3911 gdk_event.count = 0;
0fc5dbf5 3912
8f3e7ecc
RR
3913 wxRegionIterator upd( m_updateRegion );
3914 while (upd)
c89f5c02 3915 {
8f3e7ecc
RR
3916 GdkRectangle rect;
3917 rect.x = upd.GetX();
3918 rect.y = upd.GetY();
3919 rect.width = upd.GetWidth();
3920 rect.height = upd.GetHeight();
0fc5dbf5 3921
8f3e7ecc
RR
3922 if (gtk_widget_intersect (child->widget, &rect, &gdk_event.area))
3923 {
3924 gtk_widget_event (child->widget, (GdkEvent*) &gdk_event);
3925 }
0fc5dbf5 3926
8f3e7ecc 3927 upd ++;
c89f5c02
RR
3928 }
3929 }
3930 }
4e5a4c69 3931#endif
8f3e7ecc 3932#endif
c89f5c02
RR
3933
3934 m_updateRegion.Clear();
beab25bd
RR
3935}
3936
1e6feb95 3937void wxWindowGTK::Clear()
c801d85f 3938{
223d09f6 3939 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
47d67540 3940
b15ed747 3941#ifndef __WXGTK20__
f234c60c
RR
3942 if (m_wxwindow && m_wxwindow->window)
3943 {
d7fa7eaa
RR
3944 m_clearRegion.Clear();
3945 wxSize size( GetClientSize() );
3946 m_clearRegion.Union( 0,0,size.x,size.y );
2b5f62a0 3947
d7fa7eaa 3948 // Better do this in idle?
010afced 3949 GtkUpdate();
f234c60c 3950 }
b15ed747 3951#endif
362c6693 3952}
c801d85f 3953
ff8bfdbb 3954#if wxUSE_TOOLTIPS
1e6feb95 3955void wxWindowGTK::DoSetToolTip( wxToolTip *tip )
b1170810 3956{
f03fc89f 3957 wxWindowBase::DoSetToolTip(tip);
ff8bfdbb 3958
f03fc89f 3959 if (m_tooltip)
3379ed37 3960 m_tooltip->Apply( (wxWindow *)this );
b1170810
RR
3961}
3962
1e6feb95 3963void wxWindowGTK::ApplyToolTip( GtkTooltips *tips, const wxChar *tip )
b1170810 3964{
dcf924a3 3965 gtk_tooltips_set_tip( tips, GetConnectWidget(), wxConvCurrent->cWX2MB(tip), (gchar*) NULL );
301cd871 3966}
ff8bfdbb 3967#endif // wxUSE_TOOLTIPS
b1170810 3968
ea323db3
RR
3969void wxWindowGTK::GtkSetBackgroundColour( const wxColour &colour )
3970{
3971 GdkWindow *window = (GdkWindow*) NULL;
3972 if (m_wxwindow)
3973 window = GTK_PIZZA(m_wxwindow)->bin_window;
3974 else
3975 window = GetConnectWidget()->window;
3976
3977 wxASSERT( window );
2b5f62a0 3978
ea323db3
RR
3979 // We need the pixel value e.g. for background clearing.
3980 m_backgroundColour.CalcPixel( gdk_window_get_colormap( window ) );
2b5f62a0 3981
f90566f5 3982 if (m_wxwindow)
ea323db3 3983 {
f90566f5 3984 // wxMSW doesn't clear the window here, either.
ea323db3
RR
3985 gdk_window_set_background( window, m_backgroundColour.GetColor() );
3986 }
3987
3988 ApplyWidgetStyle();
3989}
3990
1e6feb95 3991bool wxWindowGTK::SetBackgroundColour( const wxColour &colour )
c801d85f 3992{
223d09f6 3993 wxCHECK_MSG( m_widget != NULL, FALSE, wxT("invalid window") );
8bbe427f 3994
739730ca 3995 if (!wxWindowBase::SetBackgroundColour(colour))
ea323db3 3996 return FALSE;
c50f1fb9 3997
ed673c6a
RR
3998 GdkWindow *window = (GdkWindow*) NULL;
3999 if (m_wxwindow)
da048e3d 4000 window = GTK_PIZZA(m_wxwindow)->bin_window;
ed673c6a
RR
4001 else
4002 window = GetConnectWidget()->window;
148cd9b6 4003
ed673c6a 4004 if (!window)
739730ca
RR
4005 {
4006 // indicate that a new style has been set
c50f1fb9
VZ
4007 // but it couldn't get applied as the
4008 // widget hasn't been realized yet.
4009 m_delayedBackgroundColour = TRUE;
ea323db3 4010 return TRUE;
739730ca 4011 }
ea323db3 4012 else
994bc575 4013 {
ea323db3 4014 GtkSetBackgroundColour( colour );
994bc575 4015 }
ca298c88 4016
ea323db3
RR
4017 return TRUE;
4018}
8bbe427f 4019
ea323db3
RR
4020void wxWindowGTK::GtkSetForegroundColour( const wxColour &colour )
4021{
4022 GdkWindow *window = (GdkWindow*) NULL;
4023 if (m_wxwindow)
4024 window = GTK_PIZZA(m_wxwindow)->bin_window;
4025 else
4026 window = GetConnectWidget()->window;
c801d85f 4027
ea323db3
RR
4028 wxASSERT( window );
4029
4030 ApplyWidgetStyle();
6de97a3b
RR
4031}
4032
1e6feb95 4033bool wxWindowGTK::SetForegroundColour( const wxColour &colour )
6de97a3b 4034{
223d09f6 4035 wxCHECK_MSG( m_widget != NULL, FALSE, wxT("invalid window") );
8bbe427f 4036
739730ca
RR
4037 if (!wxWindowBase::SetForegroundColour(colour))
4038 {
4039 // don't leave if the GTK widget has just
4040 // been realized
4041 if (!m_delayedForegroundColour) return FALSE;
4042 }
c50f1fb9 4043
ed673c6a
RR
4044 GdkWindow *window = (GdkWindow*) NULL;
4045 if (m_wxwindow)
da048e3d 4046 window = GTK_PIZZA(m_wxwindow)->bin_window;
ed673c6a
RR
4047 else
4048 window = GetConnectWidget()->window;
148cd9b6 4049
ed673c6a 4050 if (!window)
739730ca
RR
4051 {
4052 // indicate that a new style has been set
c50f1fb9
VZ
4053 // but it couldn't get applied as the
4054 // widget hasn't been realized yet.
4055 m_delayedForegroundColour = TRUE;
739730ca 4056 }
ea323db3
RR
4057 else
4058 {
4059 GtkSetForegroundColour( colour );
4060 }
f03fc89f
VZ
4061
4062 return TRUE;
58614078
RR
4063}
4064
2b5f62a0
VZ
4065#ifdef __WXGTK20__
4066PangoContext *wxWindowGTK::GtkGetPangoDefaultContext()
4067{
4068 return gtk_widget_get_pango_context( m_widget );
4069}
4070
4071PangoContext *wxWindowGTK::GtkGetPangoX11Context()
4072{
4073 if (!m_x11Context)
4074 m_x11Context = pango_x_get_context( gdk_display );
4075
4076 return m_x11Context;
4077}
4078#endif
4079
1e6feb95 4080GtkStyle *wxWindowGTK::GetWidgetStyle()
58614078 4081{
103aab26
RR
4082 if (m_widgetStyle)
4083 {
4084 GtkStyle *remake = gtk_style_copy( m_widgetStyle );
9e691f46
VZ
4085
4086 // FIXME: no more klass in 2.0
4087#ifndef __WXGTK20__
103aab26 4088 remake->klass = m_widgetStyle->klass;
dd00f3f6 4089#endif
2daa0ce9 4090
103aab26
RR
4091 gtk_style_unref( m_widgetStyle );
4092 m_widgetStyle = remake;
4093 }
4094 else
4095 {
4096 GtkStyle *def = gtk_rc_get_style( m_widget );
e6527f9d 4097
103aab26
RR
4098 if (!def)
4099 def = gtk_widget_get_default_style();
e6527f9d 4100
103aab26 4101 m_widgetStyle = gtk_style_copy( def );
9e691f46
VZ
4102
4103 // FIXME: no more klass in 2.0
4104#ifndef __WXGTK20__
103aab26 4105 m_widgetStyle->klass = def->klass;
dd00f3f6 4106#endif
103aab26 4107 }
8bbe427f 4108
1ecc4d80 4109 return m_widgetStyle;
58614078
RR
4110}
4111
1e6feb95 4112void wxWindowGTK::SetWidgetStyle()
58614078 4113{
8f3e7ecc 4114#if DISABLE_STYLE_IF_BROKEN_THEME
fb65642c
RR
4115 if (m_widget->style->engine_data)
4116 {
4117 static bool s_warningPrinted = FALSE;
4118 if (!s_warningPrinted)
4119 {
4120 printf( "wxWindows warning: Widget styles disabled due to buggy GTK theme.\n" );
4121 s_warningPrinted = TRUE;
4122 }
4123 m_widgetStyle = m_widget->style;
4124 return;
4125 }
a56fcaaf 4126#endif
fb65642c 4127
1ecc4d80
RR
4128 GtkStyle *style = GetWidgetStyle();
4129
a756f210 4130 if (m_font != wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT ))
db434467 4131 {
288059b2
RR
4132#ifdef __WXGTK20__
4133 pango_font_description_free( style->font_desc );
4134 pango_font_description_copy( m_font.GetNativeFontInfo()->description );
4135#else
4136 gdk_font_unref( style->font );
f77cf4e6 4137 style->font = gdk_font_ref( m_font.GetInternalFont( 1.0 ) );
cfcc3932 4138#endif
288059b2 4139 }
1ecc4d80
RR
4140
4141 if (m_foregroundColour.Ok())
4142 {
454e2a22 4143 m_foregroundColour.CalcPixel( gtk_widget_get_colormap( m_widget ) );
a756f210 4144 if (m_foregroundColour != wxSystemSettings::GetColour(wxSYS_COLOUR_BTNTEXT))
db434467
RR
4145 {
4146 style->fg[GTK_STATE_NORMAL] = *m_foregroundColour.GetColor();
4147 style->fg[GTK_STATE_PRELIGHT] = *m_foregroundColour.GetColor();
4148 style->fg[GTK_STATE_ACTIVE] = *m_foregroundColour.GetColor();
4149 }
1e6feb95 4150 else
f6bcfd97
BP
4151 {
4152 // Try to restore the gtk default style. This is still a little
4153 // oversimplified for what is probably really needed here for controls
4154 // other than buttons, but is better than not being able to (re)set a
4155 // control's foreground colour to *wxBLACK -- RL
4156 GtkStyle *def = gtk_rc_get_style( m_widget );
4157
4158 if (!def)
4159 def = gtk_widget_get_default_style();
4160
4161 style->fg[GTK_STATE_NORMAL] = def->fg[GTK_STATE_NORMAL];
4162 style->fg[GTK_STATE_PRELIGHT] = def->fg[GTK_STATE_PRELIGHT];
4163 style->fg[GTK_STATE_ACTIVE] = def->fg[GTK_STATE_ACTIVE];
4164 }
1ecc4d80
RR
4165 }
4166
4167 if (m_backgroundColour.Ok())
4168 {
454e2a22 4169 m_backgroundColour.CalcPixel( gtk_widget_get_colormap( m_widget ) );
a756f210 4170 if (m_backgroundColour != wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE))
db434467
RR
4171 {
4172 style->bg[GTK_STATE_NORMAL] = *m_backgroundColour.GetColor();
4173 style->base[GTK_STATE_NORMAL] = *m_backgroundColour.GetColor();
4174 style->bg[GTK_STATE_PRELIGHT] = *m_backgroundColour.GetColor();
4175 style->base[GTK_STATE_PRELIGHT] = *m_backgroundColour.GetColor();
4176 style->bg[GTK_STATE_ACTIVE] = *m_backgroundColour.GetColor();
4177 style->base[GTK_STATE_ACTIVE] = *m_backgroundColour.GetColor();
4178 style->bg[GTK_STATE_INSENSITIVE] = *m_backgroundColour.GetColor();
4179 style->base[GTK_STATE_INSENSITIVE] = *m_backgroundColour.GetColor();
f6bcfd97
BP
4180 }
4181 else
4182 {
4183 // Try to restore the gtk default style. This is still a little
4184 // oversimplified for what is probably really needed here for controls
4185 // other than buttons, but is better than not being able to (re)set a
4186 // control's background colour to default grey and means resetting a
1e6feb95 4187 // button to wxSYS_COLOUR_BTNFACE will restore its usual highlighting
f6bcfd97
BP
4188 // behavior -- RL
4189 GtkStyle *def = gtk_rc_get_style( m_widget );
4190
4191 if (!def)
4192 def = gtk_widget_get_default_style();
4193
4194 style->bg[GTK_STATE_NORMAL] = def->bg[GTK_STATE_NORMAL];
4195 style->base[GTK_STATE_NORMAL] = def->base[GTK_STATE_NORMAL];
4196 style->bg[GTK_STATE_PRELIGHT] = def->bg[GTK_STATE_PRELIGHT];
4197 style->base[GTK_STATE_PRELIGHT] = def->base[GTK_STATE_PRELIGHT];
4198 style->bg[GTK_STATE_ACTIVE] = def->bg[GTK_STATE_ACTIVE];
4199 style->base[GTK_STATE_ACTIVE] = def->base[GTK_STATE_ACTIVE];
4200 style->bg[GTK_STATE_INSENSITIVE] = def->bg[GTK_STATE_INSENSITIVE];
4201 style->base[GTK_STATE_INSENSITIVE] = def->base[GTK_STATE_INSENSITIVE];
4202 }
1ecc4d80 4203 }
a81258be
RR
4204}
4205
1e6feb95 4206void wxWindowGTK::ApplyWidgetStyle()
a81258be 4207{
6de97a3b
RR
4208}
4209
2259e007
RR
4210//-----------------------------------------------------------------------------
4211// Pop-up menu stuff
4212//-----------------------------------------------------------------------------
4213
6522713c 4214#if wxUSE_MENUS_NATIVE
1e6feb95 4215
90350682
VZ
4216extern "C"
4217void gtk_pop_hide_callback( GtkWidget *WXUNUSED(widget), bool* is_waiting )
2259e007
RR
4218{
4219 *is_waiting = FALSE;
4220}
4221
1e6feb95 4222static void SetInvokingWindow( wxMenu *menu, wxWindowGTK *win )
30dea054 4223{
1ecc4d80 4224 menu->SetInvokingWindow( win );
1987af7e 4225 wxMenuItemList::Node *node = menu->GetMenuItems().GetFirst();
1ecc4d80
RR
4226 while (node)
4227 {
1987af7e 4228 wxMenuItem *menuitem = node->GetData();
1ecc4d80
RR
4229 if (menuitem->IsSubMenu())
4230 {
ff8bfdbb
VZ
4231 SetInvokingWindow( menuitem->GetSubMenu(), win );
4232 }
1987af7e
VZ
4233
4234 node = node->GetNext();
1ecc4d80 4235 }
362c6693 4236}
30dea054 4237
e3473203
VZ
4238// used to pass the coordinates from wxWindowGTK::DoPopupMenu() to
4239// wxPopupMenuPositionCallback()
4240//
4241// should be safe even in the MT case as the user can hardly popup 2 menus
4242// simultaneously, can he?
0c77152e
RR
4243static gint gs_pop_x = 0;
4244static gint gs_pop_y = 0;
4245
295272bd
VZ
4246extern "C" void wxPopupMenuPositionCallback( GtkMenu *menu,
4247 gint *x, gint *y,
9e691f46
VZ
4248#ifdef __WXGTK20__
4249 gboolean * WXUNUSED(whatever),
4250#endif
295272bd 4251 gpointer WXUNUSED(user_data) )
0c77152e 4252{
e3473203
VZ
4253 // ensure that the menu appears entirely on screen
4254 GtkRequisition req;
4255 gtk_widget_get_child_requisition(GTK_WIDGET(menu), &req);
4256
4257 wxSize sizeScreen = wxGetDisplaySize();
4258
4259 gint xmax = sizeScreen.x - req.width,
4260 ymax = sizeScreen.y - req.height;
4261
4262 *x = gs_pop_x < xmax ? gs_pop_x : xmax;
4263 *y = gs_pop_y < ymax ? gs_pop_y : ymax;
0c77152e
RR
4264}
4265
1e6feb95 4266bool wxWindowGTK::DoPopupMenu( wxMenu *menu, int x, int y )
30dea054 4267{
223d09f6 4268 wxCHECK_MSG( m_widget != NULL, FALSE, wxT("invalid window") );
47d67540 4269
223d09f6 4270 wxCHECK_MSG( menu != NULL, FALSE, wxT("invalid popup-menu") );
8bbe427f 4271
1ecc4d80 4272 SetInvokingWindow( menu, this );
ff8bfdbb 4273
631f1bfe
JS
4274 menu->UpdateUI();
4275
0c77152e
RR
4276 gs_pop_x = x;
4277 gs_pop_y = y;
e3473203 4278 ClientToScreen( &gs_pop_x, &gs_pop_y );
ff8bfdbb 4279
2259e007 4280 bool is_waiting = TRUE;
148cd9b6 4281
90350682
VZ
4282 gtk_signal_connect( GTK_OBJECT(menu->m_menu),
4283 "hide",
4284 GTK_SIGNAL_FUNC(gtk_pop_hide_callback),
4285 (gpointer)&is_waiting );
2259e007 4286
1ecc4d80 4287 gtk_menu_popup(
47d67540 4288 GTK_MENU(menu->m_menu),
e3473203
VZ
4289 (GtkWidget *) NULL, // parent menu shell
4290 (GtkWidget *) NULL, // parent menu item
4291 wxPopupMenuPositionCallback, // function to position it
4292 NULL, // client data
4293 0, // button used to activate it
4294 gs_timeLastClick // the time of activation
47d67540 4295 );
148cd9b6 4296
956dbab1
RR
4297 while (is_waiting)
4298 {
4299 while (gtk_events_pending())
4300 gtk_main_iteration();
4301 }
2259e007 4302
1ecc4d80 4303 return TRUE;
30dea054
RR
4304}
4305
6522713c 4306#endif // wxUSE_MENUS_NATIVE
1e6feb95 4307
06cfab17 4308#if wxUSE_DRAG_AND_DROP
ac57418f 4309
1e6feb95 4310void wxWindowGTK::SetDropTarget( wxDropTarget *dropTarget )
c801d85f 4311{
223d09f6 4312 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
47d67540 4313
1ecc4d80 4314 GtkWidget *dnd_widget = GetConnectWidget();
47d67540 4315
1ecc4d80 4316 if (m_dropTarget) m_dropTarget->UnregisterWidget( dnd_widget );
47d67540 4317
1ecc4d80
RR
4318 if (m_dropTarget) delete m_dropTarget;
4319 m_dropTarget = dropTarget;
47d67540 4320
1ecc4d80 4321 if (m_dropTarget) m_dropTarget->RegisterWidget( dnd_widget );
362c6693 4322}
c801d85f 4323
f03fc89f 4324#endif // wxUSE_DRAG_AND_DROP
ac57418f 4325
1e6feb95 4326GtkWidget* wxWindowGTK::GetConnectWidget()
e3e65dac 4327{
1ecc4d80
RR
4328 GtkWidget *connect_widget = m_widget;
4329 if (m_wxwindow) connect_widget = m_wxwindow;
47d67540 4330
1ecc4d80 4331 return connect_widget;
e3e65dac 4332}
47d67540 4333
1e6feb95 4334bool wxWindowGTK::IsOwnGtkWindow( GdkWindow *window )
903f689b 4335{
148cd9b6 4336 if (m_wxwindow)
da048e3d 4337 return (window == GTK_PIZZA(m_wxwindow)->bin_window);
148cd9b6 4338
1ecc4d80 4339 return (window == m_widget->window);
903f689b
RR
4340}
4341
1e6feb95 4342bool wxWindowGTK::SetFont( const wxFont &font )
c801d85f 4343{
223d09f6 4344 wxCHECK_MSG( m_widget != NULL, FALSE, wxT("invalid window") );
c801d85f 4345
739730ca
RR
4346 if (!wxWindowBase::SetFont(font))
4347 {
454e2a22 4348 return FALSE;
739730ca 4349 }
9c288e4d 4350
a756f210 4351 wxColour sysbg = wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE );
f03fc89f 4352 if ( sysbg == m_backgroundColour )
ae0bdb01
RR
4353 {
4354 m_backgroundColour = wxNullColour;
4355 ApplyWidgetStyle();
ff8bfdbb
VZ
4356 m_backgroundColour = sysbg;
4357 }
ae0bdb01
RR
4358 else
4359 {
4360 ApplyWidgetStyle();
4361 }
c801d85f 4362
f03fc89f 4363 return TRUE;
362c6693 4364}
c801d85f 4365
94633ad9 4366void wxWindowGTK::DoCaptureMouse()
c801d85f 4367{
b231914f
VZ
4368 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4369
ed673c6a 4370 GdkWindow *window = (GdkWindow*) NULL;
b231914f
VZ
4371 if (m_wxwindow)
4372 window = GTK_PIZZA(m_wxwindow)->bin_window;
ed673c6a 4373 else
b231914f 4374 window = GetConnectWidget()->window;
148cd9b6 4375
e4606ed9 4376 wxCHECK_RET( window, _T("CaptureMouse() failed") );
c50f1fb9 4377
b231914f 4378 wxCursor* cursor = & m_cursor;
cca602ac
JS
4379 if (!cursor->Ok())
4380 cursor = wxSTANDARD_CURSOR;
4381
ed673c6a 4382 gdk_pointer_grab( window, FALSE,
1ecc4d80
RR
4383 (GdkEventMask)
4384 (GDK_BUTTON_PRESS_MASK |
4385 GDK_BUTTON_RELEASE_MASK |
148cd9b6 4386 GDK_POINTER_MOTION_HINT_MASK |
1ecc4d80 4387 GDK_POINTER_MOTION_MASK),
ff8bfdbb 4388 (GdkWindow *) NULL,
cca602ac 4389 cursor->GetCursor(),
b02da6b1 4390 (guint32)GDK_CURRENT_TIME );
b231914f 4391 g_captureWindow = this;
1e6feb95 4392 g_captureWindowHasMouse = TRUE;
362c6693 4393}
c801d85f 4394
94633ad9 4395void wxWindowGTK::DoReleaseMouse()
c801d85f 4396{
b231914f
VZ
4397 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
4398
e4606ed9 4399 wxCHECK_RET( g_captureWindow, wxT("can't release mouse - not captured") );
47d67540 4400
c43430bb
VS
4401 g_captureWindow = (wxWindowGTK*) NULL;
4402
ed673c6a 4403 GdkWindow *window = (GdkWindow*) NULL;
b231914f
VZ
4404 if (m_wxwindow)
4405 window = GTK_PIZZA(m_wxwindow)->bin_window;
ed673c6a 4406 else
b231914f 4407 window = GetConnectWidget()->window;
148cd9b6 4408
b02da6b1
VZ
4409 if (!window)
4410 return;
c50f1fb9 4411
b02da6b1 4412 gdk_pointer_ungrab ( (guint32)GDK_CURRENT_TIME );
362c6693 4413}
c801d85f 4414
1e6feb95
VZ
4415/* static */
4416wxWindow *wxWindowBase::GetCapture()
4417{
4418 return (wxWindow *)g_captureWindow;
4419}
4420
4421bool wxWindowGTK::IsRetained() const
c801d85f 4422{
1ecc4d80 4423 return FALSE;
362c6693 4424}
c801d85f 4425
1e6feb95 4426void wxWindowGTK::SetScrollbar( int orient, int pos, int thumbVisible,
cb43b372 4427 int range, bool refresh )
c801d85f 4428{
223d09f6 4429 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
8bbe427f 4430
223d09f6 4431 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
c801d85f 4432
1ecc4d80 4433 m_hasScrolling = TRUE;
47d67540 4434
1ecc4d80 4435 if (orient == wxHORIZONTAL)
cb43b372 4436 {
1ecc4d80
RR
4437 float fpos = (float)pos;
4438 float frange = (float)range;
4439 float fthumb = (float)thumbVisible;
4440 if (fpos > frange-fthumb) fpos = frange-fthumb;
4441 if (fpos < 0.0) fpos = 0.0;
4442
4443 if ((fabs(frange-m_hAdjust->upper) < 0.2) &&
4444 (fabs(fthumb-m_hAdjust->page_size) < 0.2))
4445 {
4446 SetScrollPos( orient, pos, refresh );
4447 return;
4448 }
47d67540 4449
1ecc4d80 4450 m_oldHorizontalPos = fpos;
47d67540 4451
1ecc4d80
RR
4452 m_hAdjust->lower = 0.0;
4453 m_hAdjust->upper = frange;
4454 m_hAdjust->value = fpos;
4455 m_hAdjust->step_increment = 1.0;
4456 m_hAdjust->page_increment = (float)(wxMax(fthumb,0));
4457 m_hAdjust->page_size = fthumb;
cb43b372 4458 }
1ecc4d80
RR
4459 else
4460 {
4461 float fpos = (float)pos;
4462 float frange = (float)range;
4463 float fthumb = (float)thumbVisible;
4464 if (fpos > frange-fthumb) fpos = frange-fthumb;
4465 if (fpos < 0.0) fpos = 0.0;
4466
4467 if ((fabs(frange-m_vAdjust->upper) < 0.2) &&
4468 (fabs(fthumb-m_vAdjust->page_size) < 0.2))
4469 {
4470 SetScrollPos( orient, pos, refresh );
4471 return;
4472 }
47d67540 4473
1ecc4d80 4474 m_oldVerticalPos = fpos;
47d67540 4475
1ecc4d80
RR
4476 m_vAdjust->lower = 0.0;
4477 m_vAdjust->upper = frange;
4478 m_vAdjust->value = fpos;
4479 m_vAdjust->step_increment = 1.0;
4480 m_vAdjust->page_increment = (float)(wxMax(fthumb,0));
4481 m_vAdjust->page_size = fthumb;
4482 }
47d67540 4483
eb082a08
RR
4484 if (orient == wxHORIZONTAL)
4485 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust), "changed" );
4486 else
4487 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust), "changed" );
362c6693 4488}
c801d85f 4489
1e6feb95 4490void wxWindowGTK::SetScrollPos( int orient, int pos, bool WXUNUSED(refresh) )
c801d85f 4491{
223d09f6 4492 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
1ecc4d80 4493
223d09f6 4494 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
1ecc4d80
RR
4495
4496 if (orient == wxHORIZONTAL)
4497 {
4498 float fpos = (float)pos;
4499 if (fpos > m_hAdjust->upper - m_hAdjust->page_size) fpos = m_hAdjust->upper - m_hAdjust->page_size;
4500 if (fpos < 0.0) fpos = 0.0;
4501 m_oldHorizontalPos = fpos;
4502
4503 if (fabs(fpos-m_hAdjust->value) < 0.2) return;
4504 m_hAdjust->value = fpos;
4505 }
4506 else
4507 {
4508 float fpos = (float)pos;
4509 if (fpos > m_vAdjust->upper - m_vAdjust->page_size) fpos = m_vAdjust->upper - m_vAdjust->page_size;
4510 if (fpos < 0.0) fpos = 0.0;
4511 m_oldVerticalPos = fpos;
ff8bfdbb 4512
1ecc4d80
RR
4513 if (fabs(fpos-m_vAdjust->value) < 0.2) return;
4514 m_vAdjust->value = fpos;
4515 }
47d67540 4516
5b8a521e 4517 if (m_wxwindow->window)
47d67540 4518 {
5b8a521e 4519 if (orient == wxHORIZONTAL)
473d087e
RR
4520 {
4521 gtk_signal_disconnect_by_func( GTK_OBJECT(m_hAdjust),
4522 (GtkSignalFunc) gtk_window_hscroll_callback, (gpointer) this );
2daa0ce9 4523
5b8a521e 4524 gtk_signal_emit_by_name( GTK_OBJECT(m_hAdjust), "value_changed" );
2daa0ce9 4525
473d087e
RR
4526 gtk_signal_connect( GTK_OBJECT(m_hAdjust), "value_changed",
4527 (GtkSignalFunc) gtk_window_hscroll_callback, (gpointer) this );
4528 }
5b8a521e 4529 else
473d087e
RR
4530 {
4531 gtk_signal_disconnect_by_func( GTK_OBJECT(m_vAdjust),
4532 (GtkSignalFunc) gtk_window_vscroll_callback, (gpointer) this );
2daa0ce9 4533
5b8a521e 4534 gtk_signal_emit_by_name( GTK_OBJECT(m_vAdjust), "value_changed" );
473d087e
RR
4535
4536 gtk_signal_connect( GTK_OBJECT(m_vAdjust), "value_changed",
4537 (GtkSignalFunc) gtk_window_vscroll_callback, (gpointer) this );
4538 }
cb43b372 4539 }
362c6693 4540}
c801d85f 4541
1e6feb95 4542int wxWindowGTK::GetScrollThumb( int orient ) const
c801d85f 4543{
223d09f6 4544 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
47d67540 4545
223d09f6 4546 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
47d67540 4547
1ecc4d80
RR
4548 if (orient == wxHORIZONTAL)
4549 return (int)(m_hAdjust->page_size+0.5);
4550 else
4551 return (int)(m_vAdjust->page_size+0.5);
362c6693 4552}
c801d85f 4553
1e6feb95 4554int wxWindowGTK::GetScrollPos( int orient ) const
c801d85f 4555{
223d09f6 4556 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
47d67540 4557
223d09f6 4558 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
c801d85f 4559
1ecc4d80
RR
4560 if (orient == wxHORIZONTAL)
4561 return (int)(m_hAdjust->value+0.5);
4562 else
4563 return (int)(m_vAdjust->value+0.5);
362c6693 4564}
c801d85f 4565
1e6feb95 4566int wxWindowGTK::GetScrollRange( int orient ) const
c801d85f 4567{
223d09f6 4568 wxCHECK_MSG( m_widget != NULL, 0, wxT("invalid window") );
47d67540 4569
223d09f6 4570 wxCHECK_MSG( m_wxwindow != NULL, 0, wxT("window needs client area for scrolling") );
c801d85f 4571
1ecc4d80
RR
4572 if (orient == wxHORIZONTAL)
4573 return (int)(m_hAdjust->upper+0.5);
4574 else
4575 return (int)(m_vAdjust->upper+0.5);
362c6693 4576}
c801d85f 4577
1e6feb95 4578void wxWindowGTK::ScrollWindow( int dx, int dy, const wxRect* WXUNUSED(rect) )
c801d85f 4579{
223d09f6 4580 wxCHECK_RET( m_widget != NULL, wxT("invalid window") );
47d67540 4581
223d09f6 4582 wxCHECK_RET( m_wxwindow != NULL, wxT("window needs client area for scrolling") );
2b5f62a0 4583
f47ae6e7 4584 // No scrolling requested.
8e217128 4585 if ((dx == 0) && (dy == 0)) return;
0fc5dbf5 4586
4e5a4c69 4587#ifndef __WXGTK20__
35917d22
RR
4588 if (!m_updateRegion.IsEmpty())
4589 {
4590 m_updateRegion.Offset( dx, dy );
0fc5dbf5 4591
35917d22
RR
4592 int cw = 0;
4593 int ch = 0;
4594 GetClientSize( &cw, &ch );
4595 m_updateRegion.Intersect( 0, 0, cw, ch );
4596 }
0fc5dbf5 4597
3bcc8d15
RR
4598 if (!m_clearRegion.IsEmpty())
4599 {
4600 m_clearRegion.Offset( dx, dy );
0fc5dbf5 4601
3bcc8d15
RR
4602 int cw = 0;
4603 int ch = 0;
4604 GetClientSize( &cw, &ch );
4605 m_clearRegion.Intersect( 0, 0, cw, ch );
4606 }
3fc6e5fa
RR
4607#endif
4608
b6fa52db 4609 m_clipPaintRegion = TRUE;
0fc5dbf5 4610
da048e3d 4611 gtk_pizza_scroll( GTK_PIZZA(m_wxwindow), -dx, -dy );
0fc5dbf5 4612
b6fa52db 4613 m_clipPaintRegion = FALSE;
c801d85f 4614}
3723b7b1 4615
4e5a4c69 4616
3723b7b1
JS
4617// Find the wxWindow at the current mouse position, also returning the mouse
4618// position.
4619wxWindow* wxFindWindowAtPointer(wxPoint& pt)
4620{
59a12e90
JS
4621 pt = wxGetMousePosition();
4622 wxWindow* found = wxFindWindowAtPoint(pt);
4623 return found;
3723b7b1
JS
4624}
4625
4626// Get the current mouse position.
4627wxPoint wxGetMousePosition()
4628{
59a12e90
JS
4629 /* This crashes when used within wxHelpContext,
4630 so we have to use the X-specific implementation below.
4631 gint x, y;
4632 GdkModifierType *mask;
4633 (void) gdk_window_get_pointer(NULL, &x, &y, mask);
4634
4635 return wxPoint(x, y);
4636 */
4637
3723b7b1
JS
4638 int x, y;
4639 GdkWindow* windowAtPtr = gdk_window_at_pointer(& x, & y);
59a12e90 4640
37d81cc2 4641 Display *display = windowAtPtr ? GDK_WINDOW_XDISPLAY(windowAtPtr) : GDK_DISPLAY();
59a12e90
JS
4642 Window rootWindow = RootWindowOfScreen (DefaultScreenOfDisplay(display));
4643 Window rootReturn, childReturn;
4644 int rootX, rootY, winX, winY;
4645 unsigned int maskReturn;
4646
4647 XQueryPointer (display,
5cd09f0b
RR
4648 rootWindow,
4649 &rootReturn,
59a12e90
JS
4650 &childReturn,
4651 &rootX, &rootY, &winX, &winY, &maskReturn);
4652 return wxPoint(rootX, rootY);
4653
3723b7b1
JS
4654}
4655
4e5a4c69
RR
4656// ----------------------------------------------------------------------------
4657// wxDCModule
4658// ----------------------------------------------------------------------------
4659
4660class wxWinModule : public wxModule
4661{
4662public:
4663 bool OnInit();
4664 void OnExit();
4665
4666private:
4667 DECLARE_DYNAMIC_CLASS(wxWinModule)
4668};
4669
4670IMPLEMENT_DYNAMIC_CLASS(wxWinModule, wxModule)
4671
4672bool wxWinModule::OnInit()
4673{
994bc575
RR
4674 // g_eraseGC = gdk_gc_new( GDK_ROOT_PARENT() );
4675 // gdk_gc_set_fill( g_eraseGC, GDK_SOLID );
0fc5dbf5 4676
4e5a4c69
RR
4677 return TRUE;
4678}
4679
4680void wxWinModule::OnExit()
4681{
994bc575
RR
4682 if (g_eraseGC)
4683 gdk_gc_unref( g_eraseGC );
4e5a4c69
RR
4684}
4685
6728fb61 4686// vi:sts=4:sw=4:et