1 /* GDK - The GIMP Drawing Kit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
3 * Copyright (C) 1998-1999 Tor Lillqvist
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
22 * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
23 * file for a list of people on the GTK+ Team. See the ChangeLog
24 * files for a list of changes. These files are distributed with
25 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
28 /* Cannot use TrackMouseEvent, as the stupid WM_MOUSELEAVE message
29 * doesn't tell us where the mouse has gone. Thus we cannot use it to
30 * generate a correct GdkNotifyType. Pity, as using TrackMouseEvent
31 * otherwise would make it possible to reliably generate
32 * GDK_LEAVE_NOTIFY events, which would help get rid of those pesky
33 * tooltips sometimes popping up in the wrong place.
35 /* define USE_TRACKMOUSEEVENT */
37 /* Do use SetCapture, it works now. Thanks to jpe@archaeopteryx.com */
38 #define USE_SETCAPTURE 1
43 #include "gdkprivate-win32.h"
44 #include "gdkinput-win32.h"
45 #include "gdkkeysyms.h"
49 #if defined (__GNUC__) && defined (HAVE_DIMM_H)
50 /* The w32api imm.h clashes a bit with the IE5.5 dimm.h */
51 # define IMEMENUITEMINFOA hidden_IMEMENUITEMINFOA
52 # define IMEMENUITEMINFOW hidden_IMEMENUITEMINFOW
57 #if defined (__GNUC__) && defined (HAVE_DIMM_H)
58 # undef IMEMENUITEMINFOA
59 # undef IMEMENUITEMINFOW
67 typedef struct _GdkEventPrivate GdkEventPrivate;
71 /* Following flag is set for events on the event queue during
72 * translation and cleared afterwards.
74 GDK_EVENT_PENDING = 1 << 0
77 struct _GdkEventPrivate
84 * Private function declarations
87 static GdkFilterReturn
88 gdk_event_apply_filters(MSG *msg,
91 static gboolean gdk_event_translate (GdkEvent *event,
93 gboolean *ret_val_flagp,
95 gboolean return_exposes);
97 static gboolean gdk_event_prepare (GSource *source,
99 static gboolean gdk_event_check (GSource *source);
100 static gboolean gdk_event_dispatch (GSource *source,
101 GSourceFunc callback,
104 /* Private variable declarations
107 static GdkWindow *p_grab_window = NULL; /* Window that currently
108 * holds the pointer grab
111 static GdkWindow *k_grab_window = NULL; /* Window the holds the
115 static GList *client_filters; /* Filters for client messages */
117 static gboolean p_grab_automatic;
118 static GdkEventMask p_grab_mask;
119 static gboolean p_grab_owner_events, k_grab_owner_events;
120 static HCURSOR p_grab_cursor;
122 static GSourceFuncs event_funcs = {
129 GPollFD event_poll_fd;
131 static GdkWindow *current_window = NULL;
132 static gint current_x, current_y;
133 static gdouble current_x_root, current_y_root;
134 static UINT gdk_ping_msg;
135 static UINT msh_mousewheel_msg;
136 static gboolean ignore_wm_char = FALSE;
137 static gboolean is_altgr_key = FALSE;
140 static IActiveIMMApp *active_imm_app = NULL;
141 static IActiveIMMMessagePumpOwner *active_imm_msgpump_owner = NULL;
144 typedef BOOL (WINAPI *PFN_TrackMouseEvent) (LPTRACKMOUSEEVENT);
145 static PFN_TrackMouseEvent track_mouse_event = NULL;
147 static gboolean use_ime_composition = FALSE;
150 real_window_procedure (HWND hwnd,
155 GdkEventPrivate event;
163 gboolean ret_val_flag;
166 msg.message = message;
169 msg.time = GetTickCount ();
170 pos = GetMessagePos ();
171 msg.pt.x = LOWORD (pos);
172 msg.pt.y = HIWORD (pos);
174 event.flags = GDK_EVENT_PENDING;
175 if (gdk_event_translate (&event.event, &msg, &ret_val_flag, &ret_val, FALSE))
177 event.flags &= ~GDK_EVENT_PENDING;
179 if (event.event.any.type == GDK_CONFIGURE)
181 /* Compress configure events */
182 GList *list = _gdk_queued_events;
185 && (((GdkEvent *)list->data)->any.type != GDK_CONFIGURE
186 || ((GdkEvent *)list->data)->any.window != event.event.any.window))
190 GDK_NOTE (EVENTS, g_print ("... compressing an CONFIGURE event\n"));
192 *((GdkEvent *)list->data) = event.event;
193 gdk_drawable_unref (event.event.any.window);
194 /* Wake up WaitMessage */
195 PostMessage (NULL, gdk_ping_msg, 0, 0);
199 else if (event.event.any.type == GDK_EXPOSE)
201 /* Compress expose events */
202 GList *list = _gdk_queued_events;
205 && (((GdkEvent *)list->data)->any.type != GDK_EXPOSE
206 || ((GdkEvent *)list->data)->any.window != event.event.any.window))
212 GDK_NOTE (EVENTS, g_print ("... compressing an EXPOSE event\n"));
213 gdk_rectangle_union (&event.event.expose.area,
214 &((GdkEvent *)list->data)->expose.area,
216 ((GdkEvent *)list->data)->expose.area = u;
217 gdk_drawable_unref (event.event.any.window);
219 /* Wake up WaitMessage */
220 PostMessage (NULL, gdk_ping_msg, 0, 0);
226 eventp = _gdk_event_new ();
227 *((GdkEventPrivate *) eventp) = event;
229 /* Philippe Colantoni <colanton@aris.ss.uci.edu> suggests this
230 * in order to handle events while opaque resizing neatly. I
231 * don't want it as default. Set the
232 * GDK_EVENT_FUNC_FROM_WINDOW_PROC env var to get this
235 if (gdk_event_func_from_window_proc && _gdk_event_func)
237 GDK_THREADS_ENTER ();
239 (*_gdk_event_func) (eventp, _gdk_event_data);
240 gdk_event_free (eventp);
242 GDK_THREADS_LEAVE ();
246 _gdk_event_queue_append (eventp);
248 /* Wake up WaitMessage */
249 PostMessage (NULL, gdk_ping_msg, 0, 0);
264 return DefWindowProc (hwnd, message, wparam, lparam);
266 if (active_imm_app == NULL
267 || (*active_imm_app->lpVtbl->OnDefWindowProc) (active_imm_app, hwnd, message, wparam, lparam, &lres) == S_FALSE)
268 return DefWindowProc (hwnd, message, wparam, lparam);
276 _gdk_win32_window_procedure (HWND hwnd,
283 GDK_NOTE (MISC, g_print ("_gdk_win32_window_procedure: %#lx %s\n",
284 (gulong) hwnd, gdk_win32_message_name (message)));
286 retval = real_window_procedure (hwnd, message, wparam, lparam);
288 GDK_NOTE (MISC, g_print ("_gdk_win32_window_procedure: %#lx returns %ld\n",
289 (gulong) hwnd, retval));
295 _gdk_events_init (void)
301 #ifdef USE_TRACKMOUSEEVENT
302 HMODULE user32, imm32;
303 HINSTANCE commctrl32;
306 gdk_ping_msg = RegisterWindowMessage ("gdk-ping");
307 GDK_NOTE (EVENTS, g_print ("gdk-ping = %#x\n", gdk_ping_msg));
309 /* This is the string MSH_MOUSEWHEEL from zmouse.h,
310 * http://www.microsoft.com/mouse/intellimouse/sdk/zmouse.h
311 * This message is used by mouse drivers than cannot generate WM_MOUSEWHEEL
314 msh_mousewheel_msg = RegisterWindowMessage ("MSWHEEL_ROLLMSG");
315 GDK_NOTE (EVENTS, g_print ("MSH_MOUSEWHEEL = %#x\n", msh_mousewheel_msg));
317 source = g_source_new (&event_funcs, sizeof (GSource));
318 g_source_set_priority (source, GDK_PRIORITY_EVENTS);
320 event_poll_fd.fd = G_WIN32_MSG_HANDLE;
321 event_poll_fd.events = G_IO_IN;
323 g_source_add_poll (source, &event_poll_fd);
324 g_source_set_can_recurse (source, TRUE);
325 g_source_attach (source, NULL);
328 hres = CoCreateInstance (&CLSID_CActiveIMM,
332 (LPVOID *) &active_imm_app);
336 GDK_NOTE (EVENTS, g_print ("IActiveIMMApp created %p\n",
338 (*active_imm_app->lpVtbl->Activate) (active_imm_app, TRUE);
340 hres = (*active_imm_app->lpVtbl->QueryInterface) (active_imm_app, &IID_IActiveIMMMessagePumpOwner, &active_imm_msgpump_owner);
341 GDK_NOTE (EVENTS, g_print ("IActiveIMMMessagePumpOwner created %p\n",
342 active_imm_msgpump_owner));
343 (active_imm_msgpump_owner->lpVtbl->Start) (active_imm_msgpump_owner);
347 #ifdef USE_TRACKMOUSEEVENT
348 user32 = GetModuleHandle ("user32.dll");
349 if ((track_mouse_event = GetProcAddress (user32, "TrackMouseEvent")) == NULL)
351 if ((commctrl32 = LoadLibrary ("commctrl32.dll")) != NULL)
352 track_mouse_event = (PFN_TrackMouseEvent)
353 GetProcAddress (commctrl32, "_TrackMouseEvent");
355 if (track_mouse_event != NULL)
356 GDK_NOTE (EVENTS, g_print ("Using TrackMouseEvent to detect leave events\n"));
358 if (IS_WIN_NT () && (windows_version & 0xFF) == 5)
360 /* On Win2k (Beta 3, at least) WM_IME_CHAR doesn't seem to work
361 * correctly for non-Unicode applications. Handle
362 * WM_IME_COMPOSITION with GCS_RESULTSTR instead, fetch the
363 * Unicode char from the IME with ImmGetCompositionStringW().
365 use_ime_composition = TRUE;
370 *--------------------------------------------------------------
373 * Returns if events are pending on the queue.
378 * Returns TRUE if events are pending
382 *--------------------------------------------------------------
386 gdk_events_pending (void)
390 return (_gdk_event_queue_find_first() ||
391 PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE));
395 *--------------------------------------------------------------
396 * gdk_event_get_graphics_expose
398 * Waits for a GraphicsExpose or NoExpose event
403 * For GraphicsExpose events, returns a pointer to the event
404 * converted into a GdkEvent Otherwise, returns NULL.
408 *-------------------------------------------------------------- */
411 gdk_event_get_graphics_expose (GdkWindow *window)
416 g_return_val_if_fail (window != NULL, NULL);
418 GDK_NOTE (EVENTS, g_print ("gdk_event_get_graphics_expose\n"));
421 /* Some nasty bugs here, just return NULL for now. */
424 if (PeekMessage (&msg, GDK_WINDOW_HWND (window), WM_PAINT, WM_PAINT, PM_REMOVE))
426 event = _gdk_event_new ();
428 if (gdk_event_translate (event, &msg, NULL, NULL, TRUE))
431 gdk_event_free (event);
439 event_mask_string (GdkEventMask mask)
441 static char bfr[500];
446 if (mask & GDK_##x##_MASK) \
447 p += sprintf (p, "%s" #x, (p > bfr ? " " : ""))
450 BIT(POINTER_MOTION_HINT);
463 BIT(PROPERTY_CHANGE);
464 BIT(VISIBILITY_NOTIFY);
475 *--------------------------------------------------------------
478 * Grabs the pointer to a specific window
481 * "window" is the window which will receive the grab
482 * "owner_events" specifies whether events will be reported as is,
483 * or relative to "window"
484 * "event_mask" masks only interesting events
485 * "confine_to" limits the cursor movement to the specified window
486 * "cursor" changes the cursor for the duration of the grab
487 * "time" specifies the time
492 * requires a corresponding call to gdk_pointer_ungrab
494 *--------------------------------------------------------------
498 gdk_pointer_grab (GdkWindow *window,
499 gboolean owner_events,
500 GdkEventMask event_mask,
501 GdkWindow *confine_to,
505 HWND hwnd_confined_to;
507 GdkCursorPrivate *cursor_private;
510 g_return_val_if_fail (window != NULL, 0);
511 g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
512 g_return_val_if_fail (confine_to == NULL || GDK_IS_WINDOW (confine_to), 0);
514 cursor_private = (GdkCursorPrivate*) cursor;
516 if (!confine_to || GDK_WINDOW_DESTROYED (confine_to))
517 hwnd_confined_to = NULL;
519 hwnd_confined_to = GDK_WINDOW_HWND (confine_to);
524 hcursor = cursor_private->hcursor;
526 return_val = _gdk_input_grab_pointer (window,
532 if (return_val == GDK_GRAB_SUCCESS)
534 if (!GDK_WINDOW_DESTROYED (window))
536 GDK_NOTE (EVENTS, g_print ("gdk_pointer_grab: %#lx %s %#lx %s\n",
537 (gulong) GDK_WINDOW_HWND (window),
538 (owner_events ? "TRUE" : "FALSE"),
540 event_mask_string (event_mask)));
541 p_grab_mask = event_mask;
542 p_grab_owner_events = (owner_events != 0);
543 p_grab_automatic = FALSE;
546 SetCapture (GDK_WINDOW_HWND (window));
548 return_val = GDK_GRAB_SUCCESS;
551 return_val = GDK_GRAB_ALREADY_GRABBED;
554 if (return_val == GDK_GRAB_SUCCESS)
556 p_grab_window = window;
557 p_grab_cursor = hcursor;
564 *--------------------------------------------------------------
567 * Releases any pointer grab
575 *--------------------------------------------------------------
579 gdk_pointer_ungrab (guint32 time)
581 GDK_NOTE (EVENTS, g_print ("gdk_pointer_ungrab\n"));
583 _gdk_input_ungrab_pointer (time);
586 if (GetCapture () != NULL)
590 p_grab_window = NULL;
594 *--------------------------------------------------------------
595 * find_window_for_pointer_event
597 * Find the window a pointer event (mouse up, down, move) should
598 * be reported to. If the return value != reported_window then
599 * the ref count of reported_window will be decremented and the
600 * ref count of the return value will be incremented.
604 * "reported_window" is the gdk window the xevent was reported relative to
605 * "xevent" is the win32 message
611 *--------------------------------------------------------------
615 find_window_for_pointer_event (GdkWindow* reported_window,
621 GdkWindow* other_window;
623 if (p_grab_window == NULL || !p_grab_owner_events)
624 return reported_window;
626 points = MAKEPOINTS (msg->lParam);
629 ClientToScreen (msg->hwnd, &pt);
631 GDK_NOTE (EVENTS, g_print ("Finding window for grabbed pointer event at (%ld, %ld)\n",
634 hwnd = WindowFromPoint (pt);
636 return reported_window;
637 other_window = gdk_win32_handle_table_lookup ((GdkNativeWindow) hwnd);
638 if (other_window == NULL)
639 return reported_window;
641 GDK_NOTE (EVENTS, g_print ("Found window %#x for point (%ld, %ld)\n",
642 (guint) hwnd, pt.x, pt.y));
644 gdk_window_unref (reported_window);
645 gdk_window_ref (other_window);
651 *--------------------------------------------------------------
652 * gdk_pointer_is_grabbed
654 * Tell wether there is an active x pointer grab in effect
662 *--------------------------------------------------------------
666 gdk_pointer_is_grabbed (void)
668 return p_grab_window != NULL;
672 *--------------------------------------------------------------
675 * Grabs the keyboard to a specific window
678 * "window" is the window which will receive the grab
679 * "owner_events" specifies whether events will be reported as is,
680 * or relative to "window"
681 * "time" specifies the time
686 * requires a corresponding call to gdk_keyboard_ungrab
688 *--------------------------------------------------------------
692 gdk_keyboard_grab (GdkWindow *window,
693 gboolean owner_events,
698 g_return_val_if_fail (window != NULL, 0);
699 g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
701 GDK_NOTE (EVENTS, g_print ("gdk_keyboard_grab %#lx\n",
702 (gulong) GDK_WINDOW_HWND (window)));
704 if (!GDK_WINDOW_DESTROYED (window))
706 k_grab_owner_events = owner_events != 0;
707 return_val = GDK_GRAB_SUCCESS;
710 return_val = GDK_GRAB_ALREADY_GRABBED;
712 if (return_val == GDK_GRAB_SUCCESS)
713 k_grab_window = window;
719 *--------------------------------------------------------------
720 * gdk_keyboard_ungrab
722 * Releases any keyboard grab
730 *--------------------------------------------------------------
734 gdk_keyboard_ungrab (guint32 time)
736 GDK_NOTE (EVENTS, g_print ("gdk_keyboard_ungrab\n"));
738 k_grab_window = NULL;
741 static GdkFilterReturn
742 gdk_event_apply_filters (MSG *msg,
746 GdkEventFilter *filter;
748 GdkFilterReturn result;
754 filter = (GdkEventFilter *) tmp_list->data;
756 result = (*filter->function) (msg, event, filter->data);
757 if (result != GDK_FILTER_CONTINUE)
760 tmp_list = tmp_list->next;
763 return GDK_FILTER_CONTINUE;
767 gdk_add_client_message_filter (GdkAtom message_type,
771 GdkClientFilter *filter = g_new (GdkClientFilter, 1);
773 filter->type = message_type;
774 filter->function = func;
777 client_filters = g_list_prepend (client_filters, filter);
780 /* Thanks to Markus G. Kuhn <mkuhn@acm.org> for the ksysym<->Unicode
781 * mapping functions, from the xterm sources.
785 build_key_event_state (GdkEvent *event)
787 if (GetKeyState (VK_SHIFT) < 0)
788 event->key.state |= GDK_SHIFT_MASK;
789 if (GetKeyState (VK_CAPITAL) & 0x1)
790 event->key.state |= GDK_LOCK_MASK;
793 if (GetKeyState (VK_CONTROL) < 0)
795 event->key.state |= GDK_CONTROL_MASK;
797 if (event->key.keyval < ' ')
798 event->key.keyval += '@';
802 else if (event->key.keyval < ' ')
804 event->key.state |= GDK_CONTROL_MASK;
805 event->key.keyval += '@';
808 if (GetKeyState (VK_MENU) < 0)
809 event->key.state |= GDK_MOD1_MASK;
814 build_pointer_event_state (MSG *msg)
819 if (msg->wParam & MK_CONTROL)
820 state |= GDK_CONTROL_MASK;
821 if (msg->wParam & MK_LBUTTON)
822 state |= GDK_BUTTON1_MASK;
823 if (msg->wParam & MK_MBUTTON)
824 state |= GDK_BUTTON2_MASK;
825 if (msg->wParam & MK_RBUTTON)
826 state |= GDK_BUTTON3_MASK;
827 if (msg->wParam & MK_SHIFT)
828 state |= GDK_SHIFT_MASK;
829 if (GetKeyState (VK_MENU) < 0)
830 state |= GDK_MOD1_MASK;
831 if (GetKeyState (VK_CAPITAL) & 0x1)
832 state |= GDK_LOCK_MASK;
838 build_keypress_event (GdkWindowImplWin32 *impl,
843 gint i, bytecount, ucount, ucleft, len;
844 guchar buf[100], *bp;
845 wchar_t wbuf[100], *wcp;
847 event->key.type = GDK_KEY_PRESS;
848 event->key.time = msg->time;
849 event->key.state = 0;
851 if (msg->message == WM_IME_COMPOSITION)
853 himc = ImmGetContext (msg->hwnd);
855 bytecount = ImmGetCompositionStringW (himc, GCS_RESULTSTR,
856 wbuf, sizeof (wbuf));
857 ucount = bytecount / 2;
861 if (msg->message == WM_CHAR || msg->message == WM_SYSCHAR)
863 bytecount = MIN ((msg->lParam & 0xFFFF), sizeof (buf));
864 for (i = 0; i < bytecount; i++)
865 buf[i] = msg->wParam;
867 else /* WM_IME_CHAR */
869 event->key.keyval = GDK_VoidSymbol;
870 if (msg->wParam & 0xFF00)
872 /* Contrary to some versions of the documentation,
873 * the lead byte is the most significant byte.
875 buf[0] = ((msg->wParam >> 8) & 0xFF);
876 buf[1] = (msg->wParam & 0xFF);
881 buf[0] = (msg->wParam & 0xFF);
886 /* Convert from the window's current code page
887 * to Unicode. Then convert to UTF-8.
888 * We don't handle the surrogate stuff. Should we?
890 ucount = MultiByteToWideChar (impl->charset_info.ciACP,
892 wbuf, sizeof (wbuf) / sizeof (wbuf[0]));
896 event->key.keyval = GDK_VoidSymbol;
897 else if (msg->message == WM_CHAR || msg->message == WM_SYSCHAR)
899 if (msg->wParam < ' ')
901 event->key.keyval = msg->wParam + '@';
902 /* This is needed in case of Alt+nnn or Alt+0nnn (on the numpad)
905 event->key.state |= GDK_CONTROL_MASK;
908 event->key.keyval = gdk_unicode_to_keyval (wbuf[0]);
911 build_key_event_state (event);
913 /* Build UTF-8 string */
929 event->key.string = g_malloc (len + 1);
930 event->key.length = len;
934 bp = event->key.string;
960 case 3: bp[2] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
961 case 2: bp[1] = (c & 0x3f) | 0x80; c >>= 6; /* Fall through */
962 case 1: bp[0] = c | first;
965 for (i = len - 1; i > 0; --i)
967 bp[i] = (c & 0x3f) | 0x80;
979 build_keyrelease_event (GdkWindowImplWin32 *impl,
986 event->key.type = GDK_KEY_RELEASE;
987 event->key.time = msg->time;
988 event->key.state = 0;
990 if (msg->message == WM_CHAR || msg->message == WM_SYSCHAR)
991 if (msg->wParam < ' ')
992 event->key.keyval = msg->wParam + '@';
996 MultiByteToWideChar (impl->charset_info.ciACP,
997 0, &buf, 1, &wbuf, 1);
999 event->key.keyval = gdk_unicode_to_keyval (wbuf);
1002 event->key.keyval = GDK_VoidSymbol;
1003 build_key_event_state (event);
1004 event->key.string = NULL;
1005 event->key.length = 0;
1009 print_event_state (gint state)
1011 #define CASE(bit) if (state & GDK_ ## bit ## _MASK) g_print (#bit " ");
1023 print_window_state (GdkWindowState state)
1025 #define CASE(bit) if (state & GDK_WINDOW_STATE_ ## bit ) g_print (#bit " ");
1034 print_event (GdkEvent *event)
1036 gchar *escaped, *kvname;
1038 switch (event->any.type)
1040 #define CASE(x) case x: g_print ( #x " "); break;
1045 CASE (GDK_MOTION_NOTIFY);
1046 CASE (GDK_BUTTON_PRESS);
1047 CASE (GDK_2BUTTON_PRESS);
1048 CASE (GDK_3BUTTON_PRESS);
1049 CASE (GDK_BUTTON_RELEASE);
1050 CASE (GDK_KEY_PRESS);
1051 CASE (GDK_KEY_RELEASE);
1052 CASE (GDK_ENTER_NOTIFY);
1053 CASE (GDK_LEAVE_NOTIFY);
1054 CASE (GDK_FOCUS_CHANGE);
1055 CASE (GDK_CONFIGURE);
1058 CASE (GDK_PROPERTY_NOTIFY);
1059 CASE (GDK_SELECTION_CLEAR);
1060 CASE (GDK_SELECTION_REQUEST);
1061 CASE (GDK_SELECTION_NOTIFY);
1062 CASE (GDK_PROXIMITY_IN);
1063 CASE (GDK_PROXIMITY_OUT);
1064 CASE (GDK_DRAG_ENTER);
1065 CASE (GDK_DRAG_LEAVE);
1066 CASE (GDK_DRAG_MOTION);
1067 CASE (GDK_DRAG_STATUS);
1068 CASE (GDK_DROP_START);
1069 CASE (GDK_DROP_FINISHED);
1070 CASE (GDK_CLIENT_EVENT);
1071 CASE (GDK_VISIBILITY_NOTIFY);
1072 CASE (GDK_NO_EXPOSE);
1074 CASE (GDK_WINDOW_STATE);
1078 g_print ("%#lx ", (gulong) GDK_WINDOW_HWND (event->any.window));
1080 switch (event->any.type)
1083 g_print ("%dx%d@+%d+%d %d",
1084 event->expose.area.width,
1085 event->expose.area.height,
1086 event->expose.area.x,
1087 event->expose.area.y,
1088 event->expose.count);
1090 case GDK_MOTION_NOTIFY:
1091 g_print ("(%.4g,%.4g) %s",
1092 event->motion.x, event->motion.y,
1093 event->motion.is_hint ? "HINT " : "");
1094 print_event_state (event->motion.state);
1096 case GDK_BUTTON_PRESS:
1097 case GDK_2BUTTON_PRESS:
1098 case GDK_3BUTTON_PRESS:
1099 case GDK_BUTTON_RELEASE:
1100 g_print ("%d (%.4g,%.4g) ",
1101 event->button.button,
1102 event->button.x, event->button.y);
1103 print_event_state (event->button.state);
1106 case GDK_KEY_RELEASE:
1107 if (event->key.length == 0)
1108 escaped = g_strdup ("");
1110 escaped = g_strescape (event->key.string, NULL);
1111 kvname = gdk_keyval_name (event->key.keyval);
1112 g_print ("%s %d:\"%s\" ",
1113 (kvname ? kvname : "??"),
1117 print_event_state (event->key.state);
1119 case GDK_ENTER_NOTIFY:
1120 case GDK_LEAVE_NOTIFY:
1122 (event->crossing.detail == GDK_NOTIFY_INFERIOR ? "INFERIOR" :
1123 (event->crossing.detail == GDK_NOTIFY_ANCESTOR ? "ANCESTOR" :
1124 (event->crossing.detail == GDK_NOTIFY_NONLINEAR ? "NONLINEAR" :
1129 (event->scroll.direction == GDK_SCROLL_UP ? "UP" :
1130 (event->scroll.direction == GDK_SCROLL_DOWN ? "DOWN" :
1131 (event->scroll.direction == GDK_SCROLL_LEFT ? "LEFT" :
1132 (event->scroll.direction == GDK_SCROLL_RIGHT ? "RIGHT" :
1134 print_event_state (event->scroll.state);
1136 case GDK_WINDOW_STATE:
1137 print_window_state (event->window_state.changed_mask);
1138 print_window_state (event->window_state.new_window_state);
1147 gdk_window_is_child (GdkWindow *parent,
1150 if (parent == NULL || window == NULL)
1153 return (gdk_window_get_parent (window) == parent ||
1154 gdk_window_is_child (parent, gdk_window_get_parent (window)));
1158 synthesize_enter_or_leave_event (GdkWindow *window,
1161 GdkNotifyType detail,
1167 event = _gdk_event_new ();
1168 event->crossing.type = type;
1169 event->crossing.window = window;
1170 event->crossing.send_event = FALSE;
1171 gdk_window_ref (event->crossing.window);
1172 event->crossing.subwindow = NULL;
1173 event->crossing.time = msg->time;
1174 event->crossing.x = x;
1175 event->crossing.y = y;
1176 event->crossing.x_root = msg->pt.x;
1177 event->crossing.y_root = msg->pt.y;
1178 event->crossing.mode = GDK_CROSSING_NORMAL;
1179 event->crossing.detail = detail;
1180 event->crossing.focus = TRUE; /* ??? */
1181 event->crossing.state = 0; /* ??? */
1183 _gdk_event_queue_append (event);
1185 if (type == GDK_ENTER_NOTIFY
1186 && GDK_WINDOW_OBJECT (window)->extension_events != 0)
1187 _gdk_input_enter_event (&event->crossing, window);
1189 GDK_NOTE (EVENTS, print_event (event));
1193 synthesize_leave_event (GdkWindow *window,
1195 GdkNotifyType detail)
1199 if (!(GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->event_mask & GDK_LEAVE_NOTIFY_MASK))
1202 /* Leave events are at (current_x,current_y) in current_window */
1204 if (current_window != window)
1208 ClientToScreen (GDK_WINDOW_HWND (current_window), &pt);
1209 ScreenToClient (GDK_WINDOW_HWND (window), &pt);
1210 synthesize_enter_or_leave_event (window, msg, GDK_LEAVE_NOTIFY, detail, pt.x, pt.y);
1213 synthesize_enter_or_leave_event (window, msg, GDK_LEAVE_NOTIFY, detail, current_x, current_y);
1218 synthesize_enter_event (GdkWindow *window,
1220 GdkNotifyType detail)
1224 if (!(GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->event_mask & GDK_ENTER_NOTIFY_MASK))
1227 /* Enter events are at LOWORD (msg->lParam), HIWORD
1228 * (msg->lParam) in msg->hwnd */
1230 pt.x = LOWORD (msg->lParam);
1231 pt.y = HIWORD (msg->lParam);
1232 if (msg->hwnd != GDK_WINDOW_HWND (window))
1234 ClientToScreen (msg->hwnd, &pt);
1235 ScreenToClient (GDK_WINDOW_HWND (window), &pt);
1237 synthesize_enter_or_leave_event (window, msg, GDK_ENTER_NOTIFY, detail, pt.x, pt.y);
1241 synthesize_enter_events (GdkWindow *from,
1244 GdkNotifyType detail)
1246 GdkWindow *prev = gdk_window_get_parent (to);
1249 synthesize_enter_events (from, prev, msg, detail);
1250 synthesize_enter_event (to, msg, detail);
1254 synthesize_leave_events (GdkWindow *from,
1257 GdkNotifyType detail)
1259 GdkWindow *next = gdk_window_get_parent (from);
1261 synthesize_leave_event (from, msg, detail);
1263 synthesize_leave_events (next, to, msg, detail);
1267 synthesize_crossing_events (GdkWindow *window,
1270 GdkWindow *intermediate, *tem, *common_ancestor;
1272 if (gdk_window_is_child (current_window, window))
1274 /* Pointer has moved to an inferior window. */
1275 synthesize_leave_event (current_window, msg, GDK_NOTIFY_INFERIOR);
1277 /* If there are intermediate windows, generate ENTER_NOTIFY
1280 intermediate = gdk_window_get_parent (window);
1281 if (intermediate != current_window)
1283 synthesize_enter_events (current_window, intermediate, msg, GDK_NOTIFY_VIRTUAL);
1286 synthesize_enter_event (window, msg, GDK_NOTIFY_ANCESTOR);
1288 else if (gdk_window_is_child (window, current_window))
1290 /* Pointer has moved to an ancestor window. */
1291 synthesize_leave_event (current_window, msg, GDK_NOTIFY_ANCESTOR);
1293 /* If there are intermediate windows, generate LEAVE_NOTIFY
1296 intermediate = gdk_window_get_parent (current_window);
1297 if (intermediate != window)
1299 synthesize_leave_events (intermediate, window, msg, GDK_NOTIFY_VIRTUAL);
1302 else if (current_window)
1304 /* Find least common ancestor of current_window and window */
1305 tem = current_window;
1307 common_ancestor = gdk_window_get_parent (tem);
1308 tem = common_ancestor;
1309 } while (common_ancestor &&
1310 !gdk_window_is_child (common_ancestor, window));
1311 if (common_ancestor)
1313 synthesize_leave_event (current_window, msg, GDK_NOTIFY_NONLINEAR);
1314 intermediate = gdk_window_get_parent (current_window);
1315 if (intermediate != common_ancestor)
1317 synthesize_leave_events (intermediate, common_ancestor,
1318 msg, GDK_NOTIFY_NONLINEAR_VIRTUAL);
1320 intermediate = gdk_window_get_parent (window);
1321 if (intermediate != common_ancestor)
1323 synthesize_enter_events (common_ancestor, intermediate,
1324 msg, GDK_NOTIFY_NONLINEAR_VIRTUAL);
1326 synthesize_enter_event (window, msg, GDK_NOTIFY_NONLINEAR);
1331 /* Dunno where we are coming from */
1332 synthesize_enter_event (window, msg, GDK_NOTIFY_UNKNOWN);
1336 gdk_window_unref (current_window);
1337 current_window = window;
1338 gdk_window_ref (current_window);
1342 translate_mouse_coords (GdkWindow *window1,
1348 pt.x = LOWORD (msg->lParam);
1349 pt.y = HIWORD (msg->lParam);
1350 ClientToScreen (GDK_WINDOW_HWND (window1), &pt);
1351 ScreenToClient (GDK_WINDOW_HWND (window2), &pt);
1352 msg->lParam = MAKELPARAM (pt.x, pt.y);
1353 GDK_NOTE (EVENTS, g_print ("...new coords are (%ld,%ld)\n", pt.x, pt.y));
1357 propagate (GdkWindow **window,
1359 GdkWindow *grab_window,
1360 gboolean grab_owner_events,
1362 gboolean (*doesnt_want_it) (gint mask,
1365 gboolean in_propagation = FALSE;
1367 if (grab_window != NULL && !grab_owner_events)
1369 /* Event source is grabbed with owner_events FALSE */
1370 GDK_NOTE (EVENTS, g_print ("...grabbed, owner_events FALSE, "));
1371 if ((*doesnt_want_it) (grab_mask, msg))
1373 GDK_NOTE (EVENTS, g_print ("...grabber doesn't want it\n"));
1378 GDK_NOTE (EVENTS, g_print ("...sending to grabber %#lx\n",
1379 (gulong) GDK_WINDOW_HWND (grab_window)));
1380 gdk_drawable_unref (*window);
1381 *window = grab_window;
1382 gdk_drawable_ref (*window);
1388 if ((*doesnt_want_it) (GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (*window)->impl)->event_mask, msg))
1390 /* Owner doesn't want it, propagate to parent. */
1391 if (GDK_WINDOW (GDK_WINDOW_OBJECT (*window)->parent) == _gdk_parent_root)
1393 /* No parent; check if grabbed */
1394 if (grab_window != NULL)
1396 /* Event source is grabbed with owner_events TRUE */
1397 GDK_NOTE (EVENTS, g_print ("...undelivered, but grabbed\n"));
1398 if ((*doesnt_want_it) (grab_mask, msg))
1400 /* Grabber doesn't want it either */
1401 GDK_NOTE (EVENTS, g_print ("...grabber doesn't want it\n"));
1408 g_print ("...sending to grabber %#lx\n",
1409 (gulong) GDK_WINDOW_HWND (grab_window)));
1410 gdk_drawable_unref (*window);
1411 *window = grab_window;
1412 gdk_drawable_ref (*window);
1418 GDK_NOTE (EVENTS, g_print ("...undelivered\n"));
1424 gdk_drawable_unref (*window);
1425 *window = GDK_WINDOW (GDK_WINDOW_OBJECT (*window)->parent);
1426 gdk_drawable_ref (*window);
1427 GDK_NOTE (EVENTS, g_print ("%s %#lx",
1428 (in_propagation ? "," : " ...propagating to"),
1429 (gulong) GDK_WINDOW_HWND (*window)));
1430 /* The only branch where we actually continue the loop */
1431 in_propagation = TRUE;
1440 doesnt_want_key (gint mask,
1443 return (((msg->message == WM_KEYUP || msg->message == WM_SYSKEYUP)
1444 && !(mask & GDK_KEY_RELEASE_MASK))
1446 ((msg->message == WM_KEYDOWN || msg->message == WM_SYSKEYDOWN)
1447 && !(mask & GDK_KEY_PRESS_MASK)));
1451 doesnt_want_char (gint mask,
1454 return !(mask & (GDK_KEY_PRESS_MASK | GDK_KEY_RELEASE_MASK));
1458 doesnt_want_button_press (gint mask,
1461 return !(mask & GDK_BUTTON_PRESS_MASK);
1465 doesnt_want_button_release (gint mask,
1468 return !(mask & GDK_BUTTON_RELEASE_MASK);
1472 doesnt_want_button_motion (gint mask,
1475 return !((mask & GDK_POINTER_MOTION_MASK)
1476 || ((msg->wParam & (MK_LBUTTON|MK_MBUTTON|MK_RBUTTON))
1477 && (mask & GDK_BUTTON_MOTION_MASK))
1478 || ((msg->wParam & MK_LBUTTON)
1479 && (mask & GDK_BUTTON1_MOTION_MASK))
1480 || ((msg->wParam & MK_MBUTTON)
1481 && (mask & GDK_BUTTON2_MOTION_MASK))
1482 || ((msg->wParam & MK_RBUTTON)
1483 && (mask & GDK_BUTTON3_MOTION_MASK)));
1487 doesnt_want_scroll (gint mask,
1491 return !(mask & GDK_SCROLL_MASK);
1493 return !(mask & GDK_BUTTON_PRESS_MASK);
1498 decode_key_lparam (LPARAM lParam)
1500 static char buf[100];
1503 if (HIWORD (lParam) & KF_UP)
1504 p += sprintf (p, "KF_UP ");
1505 if (HIWORD (lParam) & KF_REPEAT)
1506 p += sprintf (p, "KF_REPEAT ");
1507 if (HIWORD (lParam) & KF_ALTDOWN)
1508 p += sprintf (p, "KF_ALTDOWN ");
1509 if (HIWORD (lParam) & KF_EXTENDED)
1510 p += sprintf (p, "KF_EXTENDED ");
1511 p += sprintf (p, "sc%d rep%d", LOBYTE (HIWORD (lParam)), LOWORD (lParam));
1517 gdk_event_translate (GdkEvent *event,
1519 gboolean *ret_val_flagp,
1521 gboolean return_exposes)
1525 PAINTSTRUCT paintstruct;
1538 * window_impl == GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)
1541 GdkWindowImplWin32 *window_impl;
1542 #define ASSIGN_WINDOW(rhs) \
1544 window_impl = (window ? GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl) : NULL))
1546 GdkWindow *orig_window, *new_window;
1547 GdkColormap *colormap;
1548 GdkColormapPrivateWin32 *colormap_private;
1550 GdkPixmapImplWin32 *pixmap_impl;
1551 gint xoffset, yoffset;
1557 gboolean return_val;
1562 *ret_val_flagp = FALSE;
1564 ASSIGN_WINDOW (gdk_win32_handle_table_lookup ((GdkNativeWindow) msg->hwnd));
1565 orig_window = window;
1567 event->any.window = window;
1569 /* InSendMessage() does not really mean the same as X11's send_event flag,
1570 * but it is close enough, says jpe@archaeopteryx.com.
1572 event->any.send_event = InSendMessage ();
1576 /* Handle WM_QUIT here ? */
1577 if (msg->message == WM_QUIT)
1579 GDK_NOTE (EVENTS, g_print ("WM_QUIT: %d\n", msg->wParam));
1582 else if (msg->message == WM_MOVE
1583 || msg->message == WM_SIZE)
1585 /* It's quite normal to get these messages before we have
1586 * had time to register the window in our lookup table, or
1587 * when the window is being destroyed and we already have
1588 * removed it. Repost the same message to our queue so that
1589 * we will get it later when we are prepared.
1591 GDK_NOTE(MISC, g_print("gdk_event_translate: %#lx %s posted.\n",
1593 msg->message == WM_MOVE ?
1594 "WM_MOVE" : "WM_SIZE"));
1596 PostMessage (msg->hwnd, msg->message,
1597 msg->wParam, msg->lParam);
1599 #ifndef WITHOUT_WM_CREATE
1600 else if (WM_CREATE == msg->message)
1602 window = (UNALIGNED GdkWindow*) (((LPCREATESTRUCT) msg->lParam)->lpCreateParams);
1603 GDK_WINDOW_HWND (window) = msg->hwnd;
1604 GDK_NOTE (EVENTS, g_print ("gdk_event_translate: created %#x\n",
1605 (guint) msg->hwnd));
1607 /* This should handle allmost all the other window==NULL cases.
1608 * This code is executed while gdk_window_new is in it's
1609 * CreateWindowEx call.
1610 * Don't insert xid there a second time, if it's done here.
1612 gdk_drawable_ref (window);
1613 gdk_win32_handle_table_insert (&GDK_WINDOW_HWND(window), window);
1618 GDK_NOTE (EVENTS, g_print ("gdk_event_translate: %s for %#x (NULL)\n",
1619 gdk_win32_message_name(msg->message),
1620 (guint) msg->hwnd));
1626 gdk_drawable_ref (window);
1628 if (!GDK_WINDOW_DESTROYED (window))
1630 /* Check for filters for this window */
1631 GdkFilterReturn result;
1633 result = gdk_event_apply_filters
1634 (msg, event, GDK_WINDOW_OBJECT (window)->filters);
1636 if (result != GDK_FILTER_CONTINUE)
1638 return_val = (result == GDK_FILTER_TRANSLATE) ? TRUE : FALSE;
1640 *ret_val_flagp = TRUE;
1642 *ret_valp = return_val;
1647 /* to translate coordinates to the internal > 16 bit system */
1648 _gdk_windowing_window_get_offsets (window, &xoffset, &yoffset);
1650 if (msg->message == gdk_selection_notify_msg)
1652 GDK_NOTE (EVENTS, g_print ("gdk_selection_notify_msg: %#lx\n",
1653 (gulong) msg->hwnd));
1655 event->selection.type = GDK_SELECTION_NOTIFY;
1656 event->selection.window = window;
1657 event->selection.selection = GDK_POINTER_TO_ATOM (msg->wParam);
1658 event->selection.target = GDK_POINTER_TO_ATOM (msg->lParam);
1659 event->selection.property = _gdk_selection_property;
1660 event->selection.time = msg->time;
1662 return_val = !GDK_WINDOW_DESTROYED (window);
1666 else if (msg->message == gdk_selection_request_msg)
1668 GDK_NOTE (EVENTS, g_print ("gdk_selection_request_msg: %#lx\n",
1669 (gulong) msg->hwnd));
1671 event->selection.type = GDK_SELECTION_REQUEST;
1672 event->selection.window = window;
1673 event->selection.selection = gdk_clipboard_atom;
1674 event->selection.target = GDK_TARGET_STRING;
1675 event->selection.property = _gdk_selection_property;
1676 event->selection.requestor = (guint32) msg->hwnd;
1677 event->selection.time = msg->time;
1679 return_val = !GDK_WINDOW_DESTROYED (window);
1683 else if (msg->message == gdk_selection_clear_msg)
1685 GDK_NOTE (EVENTS, g_print ("gdk_selection_clear_msg: %#lx\n",
1686 (gulong) msg->hwnd));
1688 event->selection.type = GDK_SELECTION_CLEAR;
1689 event->selection.window = window;
1690 event->selection.selection = GDK_POINTER_TO_ATOM (msg->wParam);
1691 event->selection.target = GDK_POINTER_TO_ATOM (msg->lParam);
1692 event->selection.time = msg->time;
1694 return_val = !GDK_WINDOW_DESTROYED (window);
1698 else if (msg->message == msh_mousewheel_msg)
1700 GDK_NOTE (EVENTS, g_print ("MSH_MOUSEWHEEL: %#lx %d\n",
1701 (gulong) msg->hwnd, msg->wParam));
1703 event->scroll.type = GDK_SCROLL;
1705 /* MSG_MOUSEWHEEL is delivered to the foreground window. Work
1706 * around that. Also, the position is in screen coordinates, not
1707 * client coordinates as with the button messages.
1709 pt.x = LOWORD (msg->lParam);
1710 pt.y = HIWORD (msg->lParam);
1711 if ((hwnd = WindowFromPoint (pt)) == NULL)
1715 if ((new_window = gdk_win32_handle_table_lookup ((GdkNativeWindow) msg->hwnd)) == NULL)
1718 if (new_window != window)
1720 gdk_drawable_unref (window);
1721 ASSIGN_WINDOW (new_window);
1722 gdk_drawable_ref (window);
1725 if (GDK_WINDOW_OBJECT (window)->extension_events != 0
1726 && _gdk_input_ignore_core)
1728 GDK_NOTE (EVENTS, g_print ("...ignored\n"));
1732 if (!propagate (&window, msg,
1733 p_grab_window, p_grab_owner_events, p_grab_mask,
1734 doesnt_want_scroll))
1737 ASSIGN_WINDOW (window);
1739 ScreenToClient (msg->hwnd, &pt);
1740 event->button.window = window;
1741 event->scroll.direction = ((int) msg->wParam > 0) ?
1742 GDK_SCROLL_UP : GDK_SCROLL_DOWN;
1743 event->scroll.window = window;
1744 event->scroll.time = msg->time;
1745 event->scroll.x = (gint16) pt.x + xoffset;
1746 event->scroll.y = (gint16) pt.y + yoffset;
1747 event->scroll.x_root = (gint16) LOWORD (msg->lParam);
1748 event->scroll.y_root = (gint16) HIWORD (msg->lParam);
1749 event->scroll.state = 0; /* No state information with MSH_MOUSEWHEEL */
1750 event->scroll.device = _gdk_core_pointer;
1751 return_val = !GDK_WINDOW_DESTROYED (window);
1758 GdkFilterReturn result = GDK_FILTER_CONTINUE;
1760 tmp_list = client_filters;
1763 GdkClientFilter *filter = tmp_list->data;
1764 /* FIXME: under win32 messages are not really atoms
1765 * as the following cast suggest, but the appears to be right
1766 * Haven't found a use case though ...
1768 if (filter->type == GDK_POINTER_TO_ATOM (msg->message))
1770 GDK_NOTE (EVENTS, g_print ("client filter matched\n"));
1771 event->any.window = window;
1772 result = (*filter->function) (msg, event, filter->data);
1775 case GDK_FILTER_REMOVE:
1776 *ret_val_flagp = TRUE;
1781 case GDK_FILTER_TRANSLATE:
1785 case GDK_FILTER_CONTINUE:
1786 *ret_val_flagp = TRUE;
1789 event->client.type = GDK_CLIENT_EVENT;
1790 event->client.window = window;
1791 /* FIXME: check if the cast is correct, see above */
1792 event->client.message_type = GDK_POINTER_TO_ATOM (msg->message);
1793 event->client.data_format = 0;
1794 event->client.data.l[0] = msg->wParam;
1795 event->client.data.l[1] = msg->lParam;
1800 tmp_list = tmp_list->next;
1804 switch (msg->message)
1806 case WM_INPUTLANGCHANGE:
1808 g_print ("WM_INPUTLANGCHANGE: %#lx charset %lu locale %lx\n",
1809 (gulong) msg->hwnd, (gulong) msg->wParam, msg->lParam));
1810 window_impl->input_locale = (HKL) msg->lParam;
1811 TranslateCharsetInfo ((DWORD FAR *) msg->wParam,
1812 &window_impl->charset_info,
1819 g_print ("WM_SYSKEY%s: %#lx %s %#x %s\n",
1820 (msg->message == WM_SYSKEYUP ? "UP" : "DOWN"),
1822 (GetKeyNameText (msg->lParam, buf,
1826 decode_key_lparam (msg->lParam)));
1828 /* Let the system handle Alt-Tab and Alt-Enter */
1829 if (msg->wParam == VK_TAB
1830 || msg->wParam == VK_RETURN
1831 || msg->wParam == VK_F4)
1833 /* If posted without us having keyboard focus, ignore */
1834 if (!(msg->lParam & 0x20000000))
1837 /* don't generate events for just the Alt key */
1838 if (msg->wParam == VK_MENU)
1841 /* Jump to code in common with WM_KEYUP and WM_KEYDOWN */
1847 g_print ("WM_KEY%s: %#lx %s %#x %s\n",
1848 (msg->message == WM_KEYUP ? "UP" : "DOWN"),
1850 (GetKeyNameText (msg->lParam, buf,
1854 decode_key_lparam (msg->lParam)));
1856 ignore_wm_char = TRUE;
1860 event->key.window = window;
1861 event->key.hardware_keycode = msg->wParam;
1863 switch (msg->wParam)
1866 event->key.keyval = GDK_Pointer_Button1; break;
1868 event->key.keyval = GDK_Pointer_Button3; break;
1870 event->key.keyval = GDK_Pointer_Button2; break;
1872 event->key.keyval = GDK_Cancel; break;
1874 event->key.keyval = GDK_BackSpace; break;
1876 event->key.keyval = (GetKeyState(VK_SHIFT) < 0 ?
1877 GDK_ISO_Left_Tab : GDK_Tab);
1880 event->key.keyval = GDK_Clear; break;
1882 event->key.keyval = GDK_Return; break;
1884 /* Don't let Shift auto-repeat */
1885 if (msg->message == WM_KEYDOWN
1886 && (HIWORD (msg->lParam) & KF_REPEAT))
1887 ignore_wm_char = FALSE;
1889 event->key.keyval = GDK_Shift_L;
1892 /* And not Control either */
1893 if (msg->message == WM_KEYDOWN
1894 && (HIWORD (msg->lParam) & KF_REPEAT))
1895 ignore_wm_char = FALSE;
1896 else if (HIWORD (msg->lParam) & KF_EXTENDED)
1897 event->key.keyval = GDK_Control_R;
1899 event->key.keyval = GDK_Control_L;
1903 if (msg->message == WM_KEYDOWN
1904 && (HIWORD (msg->lParam) & KF_REPEAT))
1905 ignore_wm_char = FALSE;
1906 else if (HIWORD (msg->lParam) & KF_EXTENDED)
1908 /* AltGr key comes in as Control+Right Alt */
1909 if (GetKeyState (VK_CONTROL) < 0)
1911 ignore_wm_char = FALSE;
1912 is_altgr_key = TRUE;
1914 event->key.keyval = GDK_Alt_R;
1918 event->key.keyval = GDK_Alt_L;
1919 /* This needed in case she types Alt+nnn (on the numpad) */
1920 ignore_wm_char = FALSE;
1924 event->key.keyval = GDK_Pause; break;
1926 event->key.keyval = GDK_Caps_Lock; break;
1928 event->key.keyval = GDK_Escape; break;
1930 event->key.keyval = GDK_Prior; break;
1932 event->key.keyval = GDK_Next; break;
1934 event->key.keyval = GDK_End; break;
1936 event->key.keyval = GDK_Home; break;
1938 event->key.keyval = GDK_Left; break;
1940 event->key.keyval = GDK_Up; break;
1942 event->key.keyval = GDK_Right; break;
1944 event->key.keyval = GDK_Down; break;
1946 event->key.keyval = GDK_Select; break;
1948 event->key.keyval = GDK_Print; break;
1950 event->key.keyval = GDK_Execute; break;
1952 event->key.keyval = GDK_Insert; break;
1954 event->key.keyval = GDK_Delete; break;
1956 event->key.keyval = GDK_Help; break;
1967 /* Apparently applications work better if we just pass numpad digits
1968 * on as real digits? So wait for the WM_CHAR instead.
1970 ignore_wm_char = FALSE;
1973 event->key.keyval = GDK_KP_Multiply; break;
1975 /* Pass it on as an ASCII plus in WM_CHAR. */
1976 ignore_wm_char = FALSE;
1979 event->key.keyval = GDK_KP_Separator; break;
1981 /* Pass it on as an ASCII minus in WM_CHAR. */
1982 ignore_wm_char = FALSE;
1985 /* The keypad decimal key should also be passed on as the decimal
1986 * sign ('.' or ',' depending on the Windows locale settings,
1987 * apparently). So wait for the WM_CHAR here, also.
1989 ignore_wm_char = FALSE;
1992 event->key.keyval = GDK_KP_Divide; break;
1994 event->key.keyval = GDK_F1; break;
1996 event->key.keyval = GDK_F2; break;
1998 event->key.keyval = GDK_F3; break;
2000 event->key.keyval = GDK_F4; break;
2002 event->key.keyval = GDK_F5; break;
2004 event->key.keyval = GDK_F6; break;
2006 event->key.keyval = GDK_F7; break;
2008 event->key.keyval = GDK_F8; break;
2010 event->key.keyval = GDK_F9; break;
2012 event->key.keyval = GDK_F10; break;
2014 event->key.keyval = GDK_F11; break;
2016 event->key.keyval = GDK_F12; break;
2018 event->key.keyval = GDK_F13; break;
2020 event->key.keyval = GDK_F14; break;
2022 event->key.keyval = GDK_F15; break;
2024 event->key.keyval = GDK_F16; break;
2035 if (!is_altgr_key && (GetKeyState (VK_CONTROL) < 0
2036 || GetKeyState (VK_MENU) < 0))
2037 /* Control- or Alt-digits won't come in as a WM_CHAR,
2038 * but beware of AltGr-digits, which are used for instance
2039 * on Finnish keyboards.
2041 event->key.keyval = GDK_0 + (msg->wParam - '0');
2043 ignore_wm_char = FALSE;
2045 case VK_OEM_PLUS: /* On my Win98, the '+' key comes in
2046 * as VK_OEM_PLUS, etc.
2055 if (!is_altgr_key && (GetKeyState (VK_CONTROL) < 0
2056 || GetKeyState (VK_MENU) < 0))
2057 /* Control- or Alt-plus won't come in as WM_CHAR,
2058 * but beware of AltGr-plus which is backslash on
2061 /* All these VK_OEM keycodes happen to be the corresponding ASCII
2064 event->key.keyval = msg->wParam - 0x90;
2066 ignore_wm_char = FALSE;
2069 if (!is_altgr_key && (GetKeyState (VK_CONTROL) < 0
2070 || GetKeyState (VK_MENU) < 0))
2071 /* ;: on US keyboard */
2072 event->key.keyval = ';';
2074 ignore_wm_char = FALSE;
2077 if (!is_altgr_key && (GetKeyState (VK_CONTROL) < 0
2078 || GetKeyState (VK_MENU) < 0))
2079 /* `~ on US keyboard */
2080 event->key.keyval = '`';
2082 ignore_wm_char = FALSE;
2085 if (!is_altgr_key && (GetKeyState (VK_CONTROL) < 0
2086 || GetKeyState (VK_MENU) < 0))
2087 /* '" on US keyboard */
2088 event->key.keyval = '\'';
2090 ignore_wm_char = FALSE;
2093 if (msg->message == WM_SYSKEYDOWN || msg->message == WM_SYSKEYUP)
2094 event->key.keyval = msg->wParam;
2096 ignore_wm_char = FALSE;
2100 if (!ignore_wm_char)
2103 if (!propagate (&window, msg,
2104 k_grab_window, k_grab_owner_events, GDK_ALL_EVENTS_MASK,
2107 ASSIGN_WINDOW (window);
2109 is_altgr_key = FALSE;
2110 event->key.type = ((msg->message == WM_KEYDOWN
2111 || msg->message == WM_SYSKEYDOWN) ?
2112 GDK_KEY_PRESS : GDK_KEY_RELEASE);
2113 event->key.time = msg->time;
2114 event->key.state = 0;
2115 if (GetKeyState (VK_SHIFT) < 0)
2116 event->key.state |= GDK_SHIFT_MASK;
2117 if (GetKeyState (VK_CAPITAL) & 0x1)
2118 event->key.state |= GDK_LOCK_MASK;
2119 if (GetKeyState (VK_CONTROL) < 0)
2120 event->key.state |= GDK_CONTROL_MASK;
2121 if (msg->wParam != VK_MENU && GetKeyState (VK_MENU) < 0)
2122 event->key.state |= GDK_MOD1_MASK;
2123 event->key.string = NULL;
2124 event->key.length = 0;
2125 return_val = !GDK_WINDOW_DESTROYED (window);
2128 case WM_IME_COMPOSITION:
2129 if (!use_ime_composition)
2132 GDK_NOTE (EVENTS, g_print ("WM_IME_COMPOSITION: %#lx %#lx\n",
2133 (gulong) msg->hwnd, msg->lParam));
2134 if (msg->lParam & GCS_RESULTSTR)
2140 g_print ("WM_IME_CHAR: %#lx bytes: %#.04x\n",
2141 (gulong) msg->hwnd, msg->wParam));
2147 g_print ("WM_%sCHAR: %#lx %#x %s %s\n",
2148 (msg->message == WM_CHAR ? "" : "SYS"),
2149 (gulong) msg->hwnd, msg->wParam,
2150 decode_key_lparam (msg->lParam),
2151 (ignore_wm_char ? "ignored" : "")));
2155 ignore_wm_char = FALSE;
2160 if (!propagate (&window, msg,
2161 k_grab_window, k_grab_owner_events, GDK_ALL_EVENTS_MASK,
2164 ASSIGN_WINDOW (window);
2166 event->key.window = window;
2167 return_val = !GDK_WINDOW_DESTROYED (window);
2169 if (return_val && (event->key.window == k_grab_window
2170 || (window_impl->event_mask & GDK_KEY_RELEASE_MASK)))
2172 if (window == k_grab_window
2173 || (window_impl->event_mask & GDK_KEY_PRESS_MASK))
2175 /* Append a GDK_KEY_PRESS event to the pushback list
2176 * (from which it will be fetched before the release
2179 GdkEvent *event2 = _gdk_event_new ();
2180 build_keypress_event (window_impl, event2, msg);
2181 event2->key.window = window;
2182 gdk_drawable_ref (window);
2183 _gdk_event_queue_append (event2);
2184 GDK_NOTE (EVENTS, print_event (event2));
2186 /* Return the key release event. */
2187 build_keyrelease_event (window_impl, event, msg);
2190 && (window_impl->event_mask & GDK_KEY_PRESS_MASK))
2192 /* Return just the key press event. */
2193 build_keypress_event (window_impl, event, msg);
2198 #if 0 /* Don't reset is_AltGr_key here. Othewise we can't type several
2199 * AltGr-accessed chars while keeping the AltGr pressed down
2202 is_AltGr_key = FALSE;
2206 case WM_LBUTTONDOWN:
2209 case WM_MBUTTONDOWN:
2212 case WM_RBUTTONDOWN:
2217 g_print ("WM_%cBUTTONDOWN: %#lx (%d,%d)\n",
2220 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2222 if (GDK_WINDOW_OBJECT (window)->extension_events != 0
2223 && _gdk_input_ignore_core)
2225 GDK_NOTE (EVENTS, g_print ("...ignored\n"));
2229 ASSIGN_WINDOW (find_window_for_pointer_event (window, msg));
2231 if (window != current_window)
2232 synthesize_crossing_events (window, msg);
2234 event->button.type = GDK_BUTTON_PRESS;
2235 if (!propagate (&window, msg,
2236 p_grab_window, p_grab_owner_events, p_grab_mask,
2237 doesnt_want_button_press))
2239 ASSIGN_WINDOW (window);
2241 event->button.window = window;
2243 /* Emulate X11's automatic active grab */
2246 /* No explicit active grab, let's start one automatically */
2247 gint owner_events = window_impl->event_mask
2248 & (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK);
2250 GDK_NOTE (EVENTS, g_print ("...automatic grab started\n"));
2251 gdk_pointer_grab (window,
2253 window_impl->event_mask,
2255 p_grab_automatic = TRUE;
2258 event->button.time = msg->time;
2259 if (window != orig_window)
2260 translate_mouse_coords (orig_window, window, msg);
2261 event->button.x = current_x = (gint16) LOWORD (msg->lParam);
2262 event->button.y = current_y = (gint16) HIWORD (msg->lParam);
2263 event->button.x += xoffset; /* XXX translate current_x, y too? */
2264 event->button.y += yoffset;
2265 event->button.x_root = msg->pt.x;
2266 event->button.y_root = msg->pt.y;
2267 event->button.axes = NULL;
2268 event->button.state = build_pointer_event_state (msg);
2269 event->button.button = button;
2270 event->button.device = _gdk_core_pointer;
2272 _gdk_event_button_generate (event);
2274 return_val = !GDK_WINDOW_DESTROYED (window);
2288 g_print ("WM_%cBUTTONUP: %#lx (%d,%d)\n",
2291 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2293 ASSIGN_WINDOW (find_window_for_pointer_event (window, msg));
2295 if (GDK_WINDOW_OBJECT (window)->extension_events != 0
2296 && _gdk_input_ignore_core)
2298 GDK_NOTE (EVENTS, g_print ("...ignored\n"));
2302 if (window != current_window)
2303 synthesize_crossing_events (window, msg);
2305 event->button.type = GDK_BUTTON_RELEASE;
2306 if (!propagate (&window, msg,
2307 p_grab_window, p_grab_owner_events, p_grab_mask,
2308 doesnt_want_button_release))
2313 ASSIGN_WINDOW (window);
2315 event->button.window = window;
2316 event->button.time = msg->time;
2317 if (window != orig_window)
2318 translate_mouse_coords (orig_window, window, msg);
2319 event->button.x = (gint16) LOWORD (msg->lParam) + xoffset;
2320 event->button.y = (gint16) HIWORD (msg->lParam) + yoffset;
2321 event->button.x_root = msg->pt.x;
2322 event->button.y_root = msg->pt.y;
2323 event->button.axes = NULL;
2324 event->button.state = build_pointer_event_state (msg);
2325 event->button.button = button;
2326 event->button.device = _gdk_core_pointer;
2328 return_val = !GDK_WINDOW_DESTROYED (window);
2331 if (p_grab_window != NULL
2333 && (msg->wParam & (MK_LBUTTON | MK_MBUTTON | MK_RBUTTON)) == 0)
2334 gdk_pointer_ungrab (0);
2339 g_print ("WM_MOUSEMOVE: %#lx %#x (%d,%d)\n",
2340 (gulong) msg->hwnd, msg->wParam,
2341 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2343 ASSIGN_WINDOW (find_window_for_pointer_event (window, msg));
2345 /* If we haven't moved, don't create any event.
2346 * Windows sends WM_MOUSEMOVE messages after button presses
2347 * even if the mouse doesn't move. This disturbs gtk.
2349 if (window == current_window
2350 && LOWORD (msg->lParam) == current_x
2351 && HIWORD (msg->lParam) == current_y)
2354 /* HB: only process mouse move messages if we own the active window. */
2355 GetWindowThreadProcessId(GetActiveWindow(), &pidActWin);
2356 GetWindowThreadProcessId(msg->hwnd, &pidThis);
2357 if (pidActWin != pidThis)
2360 if (window != current_window)
2361 synthesize_crossing_events (window, msg);
2363 if (GDK_WINDOW_OBJECT (window)->extension_events != 0
2364 && _gdk_input_ignore_core)
2366 GDK_NOTE (EVENTS, g_print ("...ignored\n"));
2370 event->motion.type = GDK_MOTION_NOTIFY;
2371 if (!propagate (&window, msg,
2372 p_grab_window, p_grab_owner_events, p_grab_mask,
2373 doesnt_want_button_motion))
2375 ASSIGN_WINDOW (window);
2377 event->motion.window = window;
2378 event->motion.time = msg->time;
2379 if (window != orig_window)
2380 translate_mouse_coords (orig_window, window, msg);
2381 event->motion.x = current_x = (gint16) LOWORD (msg->lParam);
2382 event->motion.y = current_y = (gint16) HIWORD (msg->lParam);
2383 event->motion.x += xoffset;
2384 event->motion.y += yoffset;
2385 event->motion.x_root = current_x_root = msg->pt.x;
2386 event->motion.y_root = current_y_root = msg->pt.y;
2387 event->motion.axes = NULL;
2388 event->motion.state = build_pointer_event_state (msg);
2389 event->motion.is_hint = FALSE;
2390 event->motion.device = _gdk_core_pointer;
2392 return_val = !GDK_WINDOW_DESTROYED (window);
2395 case WM_NCMOUSEMOVE:
2397 g_print ("WM_NCMOUSEMOVE: %#lx x,y: %d %d\n",
2399 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2400 if (track_mouse_event == NULL
2401 && current_window != NULL
2402 && (GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (current_window)->impl)->event_mask & GDK_LEAVE_NOTIFY_MASK))
2404 GDK_NOTE (EVENTS, g_print ("...synthesizing LEAVE_NOTIFY event\n"));
2406 event->crossing.type = GDK_LEAVE_NOTIFY;
2407 event->crossing.window = current_window;
2408 event->crossing.subwindow = NULL;
2409 event->crossing.time = msg->time;
2410 event->crossing.x = current_x + xoffset; /* XXX translated current_x */
2411 event->crossing.y = current_y + yoffset;
2412 event->crossing.x_root = current_x_root;
2413 event->crossing.y_root = current_y_root;
2414 event->crossing.mode = GDK_CROSSING_NORMAL;
2415 event->crossing.detail = GDK_NOTIFY_NONLINEAR;
2417 event->crossing.focus = TRUE; /* ??? */
2418 event->crossing.state = 0; /* ??? */
2424 gdk_drawable_unref (current_window);
2425 current_window = NULL;
2431 GDK_NOTE (EVENTS, g_print ("WM_MOUSEWHEEL: %#lx %d\n",
2432 (gulong) msg->hwnd, HIWORD (msg->wParam)));
2434 event->scroll.type = GDK_SCROLL;
2436 /* WM_MOUSEWHEEL is delivered to the focus window Work around
2437 * that. Also, the position is in screen coordinates, not client
2438 * coordinates as with the button messages. I love the
2439 * consistency of Windows.
2441 pt.x = LOWORD (msg->lParam);
2442 pt.y = HIWORD (msg->lParam);
2443 if ((hwnd = WindowFromPoint (pt)) == NULL)
2447 if ((new_window = gdk_win32_handle_table_lookup ((GdkNativeWindow) msg->hwnd)) == NULL)
2450 if (new_window != window)
2452 gdk_drawable_unref (window);
2453 ASSIGN_WINDOW (new_window);
2454 gdk_drawable_ref (window);
2457 if (GDK_WINDOW_OBJECT (window)->extension_events != 0
2458 && _gdk_input_ignore_core)
2460 GDK_NOTE (EVENTS, g_print ("...ignored\n"));
2464 if (!propagate (&window, msg,
2465 p_grab_window, p_grab_owner_events, p_grab_mask,
2466 doesnt_want_scroll))
2469 ASSIGN_WINDOW (window);
2471 ScreenToClient (msg->hwnd, &pt);
2472 event->button.window = window;
2473 event->scroll.direction = (((short) HIWORD (msg->wParam)) > 0) ?
2474 GDK_SCROLL_UP : GDK_SCROLL_DOWN;
2475 event->scroll.window = window;
2476 event->scroll.time = msg->time;
2477 event->scroll.x = (gint16) pt.x + xoffset;
2478 event->scroll.y = (gint16) pt.y + yoffset;
2479 event->scroll.x_root = (gint16) LOWORD (msg->lParam);
2480 event->scroll.y_root = (gint16) HIWORD (msg->lParam);
2481 event->scroll.state = build_pointer_event_state (msg);
2482 event->scroll.device = _gdk_core_pointer;
2483 return_val = !GDK_WINDOW_DESTROYED (window);
2487 #ifdef USE_TRACKMOUSEEVENT
2489 GDK_NOTE (EVENTS, g_print ("WM_MOUSELEAVE: %#lx\n", (gulong) msg->hwnd));
2491 if (!(window_impl->event_mask & GDK_LEAVE_NOTIFY_MASK))
2494 event->crossing.type = GDK_LEAVE_NOTIFY;
2495 event->crossing.window = window;
2496 event->crossing.subwindow = NULL;
2497 event->crossing.time = msg->time;
2498 event->crossing.x = current_x + xoffset; /* XXX translated current_x */
2499 event->crossing.y = current_y + yoffset;
2500 event->crossing.x_root = current_xroot;
2501 event->crossing.y_root = current_yroot;
2502 event->crossing.mode = GDK_CROSSING_NORMAL;
2504 && IsChild (GDK_WINDOW_HWND (current_window), GDK_WINDOW_HWND (window)))
2505 event->crossing.detail = GDK_NOTIFY_INFERIOR;
2506 else if (current_window
2507 && IsChild (GDK_WINDOW_HWND (window), GDK_WINDOW_HWND (current_window)))
2508 event->crossing.detail = GDK_NOTIFY_ANCESTOR;
2510 event->crossing.detail = GDK_NOTIFY_NONLINEAR;
2512 event->crossing.focus = TRUE; /* ??? */
2513 event->crossing.state = 0; /* ??? */
2517 gdk_drawable_unref (current_window);
2518 current_window = NULL;
2521 return_val = !GDK_WINDOW_DESTROYED (window);
2527 GDK_NOTE (EVENTS, g_print ("WM_%sFOCUS: %#lx\n",
2528 (msg->message == WM_SETFOCUS ?
2530 (gulong) msg->hwnd));
2532 if (!(window_impl->event_mask & GDK_FOCUS_CHANGE_MASK))
2535 event->focus_change.type = GDK_FOCUS_CHANGE;
2536 event->focus_change.window = window;
2537 event->focus_change.in = (msg->message == WM_SETFOCUS);
2538 return_val = !GDK_WINDOW_DESTROYED (window);
2542 GDK_NOTE (EVENTS, g_print ("WM_ERASEBKGND: %#lx dc %#x\n",
2543 (gulong) msg->hwnd, msg->wParam));
2545 if (GDK_WINDOW_DESTROYED (window))
2548 *ret_val_flagp = TRUE; /* always claim as handled */
2551 if (GDK_WINDOW_OBJECT (window)->input_only)
2554 if (GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->position_info.no_bg)
2556 /* improves scolling effect, e.g. main buttons of testgtk */
2557 *ret_val_flagp = TRUE;
2562 colormap = gdk_drawable_get_colormap (window);
2564 colormap_private = GDK_COLORMAP_PRIVATE_DATA (colormap);
2565 hdc = (HDC) msg->wParam;
2566 if (colormap && colormap_private->xcolormap->rc_palette)
2570 if (SelectPalette (hdc, colormap_private->xcolormap->palette,
2572 WIN32_GDI_FAILED ("SelectPalette");
2573 if ((k = RealizePalette (hdc)) == GDI_ERROR)
2574 WIN32_GDI_FAILED ("RealizePalette");
2576 g_print ("WM_ERASEBKGND: selected %#x, realized %d colors\n",
2577 colormap_private->xcolormap->palette, k);
2581 if (GDK_WINDOW_OBJECT (window)->bg_pixmap == GDK_PARENT_RELATIVE_BG)
2583 /* If this window should have the same background as the
2584 * parent, fetch the parent. (And if the same goes for
2585 * the parent, fetch the grandparent, etc.)
2587 while (window && GDK_WINDOW_OBJECT (window)->bg_pixmap == GDK_PARENT_RELATIVE_BG)
2589 gdk_drawable_unref (window);
2590 ASSIGN_WINDOW (GDK_WINDOW (GDK_WINDOW_OBJECT (window)->parent));
2591 gdk_drawable_ref (window);
2595 if (GDK_WINDOW_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->position_info.no_bg)
2597 /* improves scolling effect, e.g. main buttons of testgtk */
2598 *ret_val_flagp = TRUE;
2603 if (GDK_WINDOW_OBJECT (window)->bg_pixmap == NULL)
2605 bg = _gdk_win32_colormap_color (GDK_DRAWABLE_IMPL_WIN32 (GDK_WINDOW_OBJECT (window)->impl)->colormap,
2606 GDK_WINDOW_OBJECT (window)->bg_color.pixel);
2608 GetClipBox (hdc, &rect);
2610 g_print ("...%ldx%ld@+%ld+%ld BG_PIXEL %.06lx\n",
2611 rect.right - rect.left,
2612 rect.bottom - rect.top,
2613 rect.left, rect.top,
2615 hbr = CreateSolidBrush (bg);
2617 g_print ("...CreateSolidBrush (%.08x) = %.08x\n", bg, hbr);
2619 if (!FillRect (hdc, &rect, hbr))
2620 WIN32_GDI_FAILED ("FillRect");
2623 else if (GDK_WINDOW_OBJECT (window)->bg_pixmap != NULL &&
2624 GDK_WINDOW_OBJECT (window)->bg_pixmap != GDK_NO_BG)
2626 pixmap = GDK_WINDOW_OBJECT (window)->bg_pixmap;
2627 pixmap_impl = GDK_PIXMAP_IMPL_WIN32 (GDK_PIXMAP_OBJECT (pixmap)->impl);
2628 GetClipBox (hdc, &rect);
2630 if (pixmap_impl->width <= 8 && pixmap_impl->height <= 8)
2632 GDK_NOTE (EVENTS, g_print ("...small pixmap, using brush\n"));
2633 hbr = CreatePatternBrush (GDK_PIXMAP_HBITMAP (pixmap));
2634 if (!FillRect (hdc, &rect, hbr))
2635 WIN32_GDI_FAILED ("FillRect");
2641 g_print ("...blitting pixmap %#lx (%dx%d) "
2642 "all over the place,\n"
2643 "...clip box = %ldx%ld@+%ld+%ld\n",
2644 (gulong) GDK_PIXMAP_HBITMAP (pixmap),
2645 pixmap_impl->width, pixmap_impl->height,
2646 rect.right - rect.left, rect.bottom - rect.top,
2647 rect.left, rect.top));
2649 if (!(bgdc = CreateCompatibleDC (hdc)))
2651 WIN32_GDI_FAILED ("CreateCompatibleDC");
2654 if (!(oldbitmap = SelectObject (bgdc, GDK_PIXMAP_HBITMAP (pixmap))))
2656 WIN32_GDI_FAILED ("SelectObject");
2661 while (i < rect.right)
2664 while (j < rect.bottom)
2666 if (i + pixmap_impl->width >= rect.left
2667 && j + pixmap_impl->height >= rect.top)
2669 if (!BitBlt (hdc, i, j,
2670 pixmap_impl->width, pixmap_impl->height,
2671 bgdc, 0, 0, SRCCOPY))
2673 WIN32_GDI_FAILED ("BitBlt");
2677 j += pixmap_impl->height;
2679 i += pixmap_impl->width;
2682 SelectObject (bgdc, oldbitmap);
2688 GDK_NOTE (EVENTS, g_print ("...BLACK_BRUSH (?)\n"));
2689 hbr = GetStockObject (BLACK_BRUSH);
2690 GetClipBox (hdc, &rect);
2691 if (!FillRect (hdc, &rect, hbr))
2692 WIN32_GDI_FAILED ("FillRect");
2697 if (!GetUpdateRect(msg->hwnd, NULL, FALSE))
2699 GDK_NOTE (EVENTS, g_print ("WM_PAINT: %#lx no update rect\n",
2700 (gulong) msg->hwnd));
2704 hdc = BeginPaint (msg->hwnd, &paintstruct);
2707 g_print ("WM_PAINT: %#lx %ldx%ld@+%ld+%ld %s dc %#lx\n",
2709 paintstruct.rcPaint.right - paintstruct.rcPaint.left,
2710 paintstruct.rcPaint.bottom - paintstruct.rcPaint.top,
2711 paintstruct.rcPaint.left, paintstruct.rcPaint.top,
2712 (paintstruct.fErase ? "erase" : ""),
2715 EndPaint (msg->hwnd, &paintstruct);
2717 /* HB: don't generate GDK_EXPOSE events for InputOnly
2718 * windows -> backing store now works!
2720 if (GDK_WINDOW_OBJECT (window)->input_only)
2723 if (!(window_impl->event_mask & GDK_EXPOSURE_MASK))
2726 if (GDK_WINDOW_OBJECT (window)->bg_pixmap == GDK_NO_BG)
2729 if ((paintstruct.rcPaint.right == paintstruct.rcPaint.left)
2730 || (paintstruct.rcPaint.bottom == paintstruct.rcPaint.top))
2735 event->expose.type = GDK_EXPOSE;
2736 event->expose.window = window;
2737 event->expose.area.x = paintstruct.rcPaint.left;
2738 event->expose.area.y = paintstruct.rcPaint.top;
2739 event->expose.area.width = paintstruct.rcPaint.right - paintstruct.rcPaint.left;
2740 event->expose.area.height = paintstruct.rcPaint.bottom - paintstruct.rcPaint.top;
2741 event->expose.region = gdk_region_rectangle (&(event->expose.area));
2742 event->expose.count = 0;
2744 return_val = !GDK_WINDOW_DESTROYED (window);
2747 GList *list = _gdk_queued_events;
2748 while (list != NULL )
2750 if ((((GdkEvent *)list->data)->any.type == GDK_EXPOSE) &&
2751 (((GdkEvent *)list->data)->any.window == window) &&
2752 !(((GdkEventPrivate *)list->data)->flags & GDK_EVENT_PENDING))
2753 ((GdkEvent *)list->data)->expose.count++;
2761 GdkRectangle expose_rect;
2763 expose_rect.x = paintstruct.rcPaint.left + xoffset;
2764 expose_rect.y = paintstruct.rcPaint.top + yoffset;
2765 expose_rect.width = paintstruct.rcPaint.right - paintstruct.rcPaint.left;
2766 expose_rect.height = paintstruct.rcPaint.bottom - paintstruct.rcPaint.top;
2768 _gdk_window_process_expose (window, msg->time, &expose_rect);
2775 GDK_NOTE (EVENTS, g_print ("WM_GETICON: %#lx %s\n",
2777 (ICON_BIG == msg->wParam ? "big" : "small")));
2781 GDK_NOTE (EVENTS, g_print ("WM_SETCURSOR: %#lx %#x %#x\n",
2783 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2785 if (LOWORD (msg->lParam) != HTCLIENT)
2788 if (p_grab_window != NULL && p_grab_cursor != NULL)
2789 hcursor = p_grab_cursor;
2790 else if (!GDK_WINDOW_DESTROYED (window))
2791 hcursor = window_impl->hcursor;
2795 if (hcursor != NULL)
2797 GDK_NOTE (EVENTS, g_print ("...SetCursor(%#lx)\n", (gulong) hcursor));
2798 SetCursor (hcursor);
2799 *ret_val_flagp = TRUE;
2805 GDK_NOTE (EVENTS, g_print ("WM_SHOWWINDOW: %#lx %d\n",
2809 if (!(window_impl->event_mask & GDK_STRUCTURE_MASK))
2812 event->any.type = (msg->wParam ? GDK_MAP : GDK_UNMAP);
2813 event->any.window = window;
2815 if (event->any.type == GDK_UNMAP
2816 && p_grab_window == window)
2817 gdk_pointer_ungrab (msg->time);
2819 if (event->any.type == GDK_UNMAP
2820 && k_grab_window == window)
2821 gdk_keyboard_ungrab (msg->time);
2823 return_val = !GDK_WINDOW_DESTROYED (window);
2828 g_print ("WM_SIZE: %#lx %s %dx%d\n",
2830 (msg->wParam == SIZE_MAXHIDE ? "MAXHIDE" :
2831 (msg->wParam == SIZE_MAXIMIZED ? "MAXIMIZED" :
2832 (msg->wParam == SIZE_MAXSHOW ? "MAXSHOW" :
2833 (msg->wParam == SIZE_MINIMIZED ? "MINIMIZED" :
2834 (msg->wParam == SIZE_RESTORED ? "RESTORED" : "?"))))),
2835 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2837 if (!(window_impl->event_mask & GDK_STRUCTURE_MASK))
2840 if (msg->wParam == SIZE_MINIMIZED)
2842 event->any.type = GDK_UNMAP;
2843 event->any.window = window;
2845 if (p_grab_window == window)
2846 gdk_pointer_ungrab (msg->time);
2848 if (k_grab_window == window)
2849 gdk_keyboard_ungrab (msg->time);
2851 return_val = !GDK_WINDOW_DESTROYED (window);
2853 else if ((msg->wParam == SIZE_RESTORED
2854 || msg->wParam == SIZE_MAXIMIZED)
2856 && GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD
2860 if (LOWORD (msg->lParam) == 0)
2863 event->configure.type = GDK_CONFIGURE;
2864 event->configure.window = window;
2867 ClientToScreen (msg->hwnd, &pt);
2868 event->configure.x = pt.x;
2869 event->configure.y = pt.y;
2870 event->configure.width = LOWORD (msg->lParam);
2871 event->configure.height = HIWORD (msg->lParam);
2872 GDK_WINDOW_OBJECT (window)->x = event->configure.x;
2873 GDK_WINDOW_OBJECT (window)->y = event->configure.y;
2874 window_impl->width = event->configure.width;
2875 window_impl->height = event->configure.height;
2877 if (GDK_WINDOW_OBJECT (window)->resize_count > 1)
2878 GDK_WINDOW_OBJECT (window)->resize_count -= 1;
2880 return_val = !GDK_WINDOW_DESTROYED (window);
2882 && GDK_WINDOW_OBJECT (window)->extension_events != 0)
2883 _gdk_input_configure_event (&event->configure, window);
2889 LPRECT lpr = (LPRECT) msg->lParam;
2890 NONCLIENTMETRICS ncm;
2891 ncm.cbSize = sizeof (NONCLIENTMETRICS);
2893 SystemParametersInfo (SPI_GETNONCLIENTMETRICS, ncm.cbSize, &ncm, 0);
2895 g_print ("WM_SIZING borderWidth %d captionHeight %d\n",
2896 ncm.iBorderWidth, ncm.iCaptionHeight);
2897 event->configure.type = GDK_CONFIGURE;
2898 event->configure.window = window;
2900 event->configure.x = lpr->left + ncm.iBorderWidth;
2901 event->configure.y = lpr->top + ncm.iCaptionHeight;
2902 event->configure.width = lpr->right - lpr->left - 2 * ncm.iBorderWidth;
2903 event->configure.height = lpr->bottom - lpr->top - ncm.iCaptionHeight;
2904 GDK_WINDOW_OBJECT (window)->x = event->configure.x;
2905 GDK_WINDOW_OBJECT (window)->y = event->configure.y;
2906 window_impl->width = event->configure.width;
2907 window_impl->height = event->configure.height;
2909 if (GDK_WINDOW_OBJECT (window)->resize_count > 1)
2910 GDK_WINDOW_OBJECT (window)->resize_count -= 1;
2912 return_val = !GDK_WINDOW_DESTROYED (window);
2914 && GDK_WINDOW_OBJECT (window)->extension_events != 0)
2915 _gdk_input_configure_event (&event->configure, window);
2919 case WM_GETMINMAXINFO:
2920 GDK_NOTE (EVENTS, g_print ("WM_GETMINMAXINFO: %#lx\n", (gulong) msg->hwnd));
2922 mmi = (MINMAXINFO*) msg->lParam;
2923 if (window_impl->hint_flags & GDK_HINT_MIN_SIZE)
2925 mmi->ptMinTrackSize.x = window_impl->hint_min_width;
2926 mmi->ptMinTrackSize.y = window_impl->hint_min_height;
2928 if (window_impl->hint_flags & GDK_HINT_MAX_SIZE)
2930 mmi->ptMaxTrackSize.x = window_impl->hint_max_width;
2931 mmi->ptMaxTrackSize.y = window_impl->hint_max_height;
2933 /* kind of WM functionality, limit maximized size to screen */
2934 mmi->ptMaxPosition.x = 0; mmi->ptMaxPosition.y = 0;
2935 mmi->ptMaxSize.x = MIN(window_impl->hint_max_width, gdk_screen_width ());
2936 mmi->ptMaxSize.y = MIN(window_impl->hint_max_height, gdk_screen_height ());
2938 else if (window_impl->hint_flags & GDK_HINT_MIN_SIZE)
2940 /* need to initialize */
2941 mmi->ptMaxSize.x = gdk_screen_width ();
2942 mmi->ptMaxSize.y = gdk_screen_height ();
2944 /* lovely API inconsistence: return FALSE when handled */
2946 *ret_val_flagp = !(window_impl->hint_flags &
2947 (GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE));
2951 GDK_NOTE (EVENTS, g_print ("WM_MOVE: %#lx (%d,%d)\n",
2953 LOWORD (msg->lParam), HIWORD (msg->lParam)));
2955 if (!(window_impl->event_mask & GDK_STRUCTURE_MASK))
2958 if (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD
2959 && !IsIconic(msg->hwnd)
2960 && IsWindowVisible(msg->hwnd))
2962 event->configure.type = GDK_CONFIGURE;
2963 event->configure.window = window;
2964 event->configure.x = LOWORD (msg->lParam);
2965 event->configure.y = HIWORD (msg->lParam);
2966 GetClientRect (msg->hwnd, &rect);
2967 event->configure.width = rect.right;
2968 event->configure.height = rect.bottom;
2969 GDK_WINDOW_OBJECT (window)->x = event->configure.x;
2970 GDK_WINDOW_OBJECT (window)->y = event->configure.y;
2971 window_impl->width = event->configure.width;
2972 window_impl->height = event->configure.height;
2974 return_val = !GDK_WINDOW_DESTROYED (window);
2977 #if 0 /* not quite right, otherwise it may be faster/better than WM_(MOVE|SIZE)
2978 * remove decoration (frame) sizes ?
2980 case WM_WINDOWPOSCHANGED :
2982 if (!(window_impl->event_mask & GDK_STRUCTURE_MASK))
2985 if (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD
2986 && !IsIconic(msg->hwnd)
2987 && IsWindowVisible(msg->hwnd))
2989 LPWINDOWPOS lpwp = (LPWINDOWPOS) (msg->lParam);
2991 event->configure.type = GDK_CONFIGURE;
2992 event->configure.window = window;
2993 event->configure.x = lpwp->x;
2994 event->configure.y = lpwp->y;
2995 event->configure.width = lpwp->cx;
2996 event->configure.height = lpwp->cy;
2997 GDK_WINDOW_OBJECT (window)->x = event->configure.x;
2998 GDK_WINDOW_OBJECT (window)->y = event->configure.y;
2999 window_impl->width = event->configure.width;
3000 window_impl->height = event->configure.height;
3002 return_val = !GDK_WINDOW_DESTROYED (window);
3004 GDK_NOTE (EVENTS, g_print ("WM_WINDOWPOSCHANGED: %#lx %ldx%ld@+%ld+%ld\n",
3006 lpwp->cx, lpwp->cy, lpwp->x, lpwp->y));
3009 *ret_val_flagp = TRUE;
3016 GDK_NOTE (EVENTS, g_print ("WM_CLOSE: %#lx\n", (gulong) msg->hwnd));
3018 event->any.type = GDK_DELETE;
3019 event->any.window = window;
3021 return_val = !GDK_WINDOW_DESTROYED (window);
3025 /* No, don't use delayed rendering after all. It works only if the
3026 * delayed SetClipboardData is called from the WindowProc, it
3027 * seems. (The #else part below is test code for that. It succeeds
3028 * in setting the clipboard data. But if I call SetClipboardData
3029 * in gdk_property_change (as a consequence of the
3030 * GDK_SELECTION_REQUEST event), it fails. I deduce that this is
3031 * because delayed rendering requires that SetClipboardData is
3032 * called in the window procedure.)
3034 case WM_RENDERFORMAT:
3035 case WM_RENDERALLFORMATS:
3037 GDK_NOTE (EVENTS, flag = TRUE);
3039 g_print ("WM_%s: %#lx %#x (%s)\n",
3040 (msg->message == WM_RENDERFORMAT ? "RENDERFORMAT" :
3041 "RENDERALLFORMATS"),
3044 (msg->wParam == CF_TEXT ? "CF_TEXT" :
3045 (msg->wParam == CF_DIB ? "CF_DIB" :
3046 (msg->wParam == CF_UNICODETEXT ? "CF_UNICODETEXT" :
3047 (GetClipboardFormatName (msg->wParam, buf, sizeof (buf)), buf)))));
3050 event->selection.type = GDK_SELECTION_REQUEST;
3051 event->selection.window = window;
3052 event->selection.selection = gdk_clipboard_atom;
3053 if (msg->wParam == CF_TEXT)
3054 event->selection.target = GDK_TARGET_STRING;
3057 GetClipboardFormatName (msg->wParam, buf, sizeof (buf));
3058 event->selection.target = gdk_atom_intern (buf, FALSE);
3060 event->selection.property = _gdk_selection_property;
3061 event->selection.requestor = (guint32) msg->hwnd;
3062 event->selection.time = msg->time;
3063 return_val = !GDK_WINDOW_DESTROYED (window);
3065 /* Test code, to see if SetClipboardData works when called from
3066 * the window procedure.
3069 HGLOBAL hdata = GlobalAlloc (GMEM_MOVEABLE|GMEM_DDESHARE, 10);
3070 char *ptr = GlobalLock (hdata);
3071 strcpy (ptr, "Huhhaa");
3072 GlobalUnlock (hdata);
3073 if (!SetClipboardData (CF_TEXT, hdata))
3074 WIN32_API_FAILED ("SetClipboardData");
3077 *ret_val_flagp = TRUE;
3081 #endif /* No delayed rendering */
3084 GDK_NOTE (EVENTS, g_print ("WM_DESTROY: %#lx\n", (gulong) msg->hwnd));
3086 event->any.type = GDK_DESTROY;
3087 event->any.window = window;
3088 if (window != NULL && window == current_window)
3090 gdk_drawable_unref (current_window);
3091 current_window = NULL;
3094 if (p_grab_window == window)
3095 gdk_pointer_ungrab (msg->time);
3097 if (k_grab_window == window)
3098 gdk_keyboard_ungrab (msg->time);
3100 return_val = window != NULL && !GDK_WINDOW_DESTROYED (window);
3102 if ((window != NULL) && (gdk_root_window != msg->hwnd))
3103 gdk_window_destroy_notify (window);
3108 /* Handle WINTAB events here, as we know that gdkinput.c will
3109 * use the fixed WT_DEFBASE as lcMsgBase, and we thus can use the
3110 * constants as case labels.
3113 GDK_NOTE (EVENTS, g_print ("WT_PACKET: %#lx %d %#lx\n",
3115 msg->wParam, msg->lParam));
3119 GDK_NOTE (EVENTS, g_print ("WT_CSRCHANGE: %#lx %d %#lx\n",
3121 msg->wParam, msg->lParam));
3125 GDK_NOTE (EVENTS, g_print ("WT_PROXIMITY: %#lx %#x %d %d\n",
3126 (gulong) msg->hwnd, msg->wParam,
3127 LOWORD (msg->lParam),
3128 HIWORD (msg->lParam)));
3131 event->any.window = window;
3132 return_val = _gdk_input_other_event(event, msg, window);
3137 GDK_NOTE (EVENTS, g_print ("%s: %#lx %#x %#lx\n",
3138 gdk_win32_message_name (msg->message),
3140 msg->wParam, msg->lParam));
3147 if (event->any.window)
3148 gdk_drawable_ref (event->any.window);
3149 if (((event->any.type == GDK_ENTER_NOTIFY) ||
3150 (event->any.type == GDK_LEAVE_NOTIFY)) &&
3151 (event->crossing.subwindow != NULL))
3152 gdk_drawable_ref (event->crossing.subwindow);
3154 GDK_NOTE (EVENTS, print_event (event));
3158 /* Mark this event as having no resources to be freed */
3159 event->any.window = NULL;
3160 event->any.type = GDK_NOTHING;
3164 gdk_drawable_unref (window);
3170 _gdk_events_queue (void)
3176 while (!_gdk_event_queue_find_first ()
3177 && PeekMessage (&msg, NULL, 0, 0, PM_REMOVE))
3179 GDK_NOTE (EVENTS, g_print ("PeekMessage: %#lx %s\n",
3180 (gulong) msg.hwnd, gdk_win32_message_name (msg.message)));
3182 TranslateMessage (&msg);
3184 if (active_imm_msgpump_owner == NULL
3185 || (active_imm_msgpump_owner->lpVtbl->OnTranslateMessage) (active_imm_msgpump_owner, &msg) != S_OK)
3186 TranslateMessage (&msg);
3189 #if 1 /* It was like this all the time */
3190 DispatchMessage (&msg);
3191 #else /* but this one is more similar to the X implementation. Any effect ? */
3192 event = _gdk_event_new ();
3194 event->any.type = GDK_NOTHING;
3195 event->any.window = NULL;
3196 event->any.send_event = InSendMessage ();
3198 ((GdkEventPrivate *)event)->flags |= GDK_EVENT_PENDING;
3200 _gdk_event_queue_append (event);
3201 node = _gdk_queued_tail;
3203 if (gdk_event_translate (event, &msg, NULL, NULL, FALSE))
3205 ((GdkEventPrivate *)event)->flags &= ~GDK_EVENT_PENDING;
3209 _gdk_event_queue_remove_link (node);
3210 g_list_free_1 (node);
3211 gdk_event_free (event);
3212 DispatchMessage (&msg);
3220 gdk_event_prepare (GSource *source,
3226 GDK_THREADS_ENTER ();
3230 retval = (_gdk_event_queue_find_first () != NULL)
3231 || PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE);
3233 GDK_THREADS_LEAVE ();
3239 gdk_event_check (GSource *source)
3244 GDK_THREADS_ENTER ();
3246 if (event_poll_fd.revents & G_IO_IN)
3247 retval = (_gdk_event_queue_find_first () != NULL)
3248 || PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE);
3252 GDK_THREADS_LEAVE ();
3258 gdk_event_dispatch (GSource *source,
3259 GSourceFunc callback,
3264 GDK_THREADS_ENTER ();
3266 _gdk_events_queue();
3267 event = _gdk_event_unqueue();
3271 if (_gdk_event_func)
3272 (*_gdk_event_func) (event, _gdk_event_data);
3274 gdk_event_free (event);
3277 GDK_THREADS_LEAVE ();
3282 /* Sends a ClientMessage to all toplevel client windows */
3284 gdk_event_send_client_message (GdkEvent *event, guint32 xid)
3291 gdk_event_send_clientmessage_toall (GdkEvent *event)
3302 /* Process all messages currently available */
3303 while (PeekMessage (&msg, NULL, 0, 0, PM_REMOVE))
3305 TranslateMessage (&msg); /* Translate virt. key codes */
3306 DispatchMessage (&msg); /* Dispatch msg. to window */
3313 #ifdef G_ENABLE_DEBUG
3316 gdk_win32_message_name (UINT msg)
3318 static gchar bfr[100];
3322 #define CASE(x) case x: return #x
3330 CASE (WM_KILLFOCUS);
3332 CASE (WM_SETREDRAW);
3335 CASE (WM_GETTEXTLENGTH);
3338 CASE (WM_QUERYENDSESSION);
3339 CASE (WM_QUERYOPEN);
3340 CASE (WM_ENDSESSION);
3342 CASE (WM_ERASEBKGND);
3343 CASE (WM_SYSCOLORCHANGE);
3344 CASE (WM_SHOWWINDOW);
3345 CASE (WM_WININICHANGE);
3346 CASE (WM_DEVMODECHANGE);
3347 CASE (WM_ACTIVATEAPP);
3348 CASE (WM_FONTCHANGE);
3349 CASE (WM_TIMECHANGE);
3350 CASE (WM_CANCELMODE);
3351 CASE (WM_SETCURSOR);
3352 CASE (WM_MOUSEACTIVATE);
3353 CASE (WM_CHILDACTIVATE);
3354 CASE (WM_QUEUESYNC);
3355 CASE (WM_GETMINMAXINFO);
3356 CASE (WM_PAINTICON);
3357 CASE (WM_ICONERASEBKGND);
3358 CASE (WM_NEXTDLGCTL);
3359 CASE (WM_SPOOLERSTATUS);
3361 CASE (WM_MEASUREITEM);
3362 CASE (WM_DELETEITEM);
3363 CASE (WM_VKEYTOITEM);
3364 CASE (WM_CHARTOITEM);
3367 CASE (WM_SETHOTKEY);
3368 CASE (WM_GETHOTKEY);
3369 CASE (WM_QUERYDRAGICON);
3370 CASE (WM_COMPAREITEM);
3371 CASE (WM_GETOBJECT);
3372 CASE (WM_COMPACTING);
3373 CASE (WM_WINDOWPOSCHANGING);
3374 CASE (WM_WINDOWPOSCHANGED);
3377 CASE (WM_CANCELJOURNAL);
3379 CASE (WM_INPUTLANGCHANGEREQUEST);
3380 CASE (WM_INPUTLANGCHANGE);
3383 CASE (WM_USERCHANGED);
3384 CASE (WM_NOTIFYFORMAT);
3385 CASE (WM_CONTEXTMENU);
3386 CASE (WM_STYLECHANGING);
3387 CASE (WM_STYLECHANGED);
3388 CASE (WM_DISPLAYCHANGE);
3392 CASE (WM_NCDESTROY);
3393 CASE (WM_NCCALCSIZE);
3394 CASE (WM_NCHITTEST);
3396 CASE (WM_NCACTIVATE);
3397 CASE (WM_GETDLGCODE);
3398 CASE (WM_SYNCPAINT);
3399 CASE (WM_NCMOUSEMOVE);
3400 CASE (WM_NCLBUTTONDOWN);
3401 CASE (WM_NCLBUTTONUP);
3402 CASE (WM_NCLBUTTONDBLCLK);
3403 CASE (WM_NCRBUTTONDOWN);
3404 CASE (WM_NCRBUTTONUP);
3405 CASE (WM_NCRBUTTONDBLCLK);
3406 CASE (WM_NCMBUTTONDOWN);
3407 CASE (WM_NCMBUTTONUP);
3408 CASE (WM_NCMBUTTONDBLCLK);
3409 CASE (WM_NCXBUTTONDOWN);
3410 CASE (WM_NCXBUTTONUP);
3411 CASE (WM_NCXBUTTONDBLCLK);
3416 CASE (WM_SYSKEYDOWN);
3419 CASE (WM_SYSDEADCHAR);
3421 CASE (WM_IME_STARTCOMPOSITION);
3422 CASE (WM_IME_ENDCOMPOSITION);
3423 CASE (WM_IME_COMPOSITION);
3424 CASE (WM_INITDIALOG);
3426 CASE (WM_SYSCOMMAND);
3431 CASE (WM_INITMENUPOPUP);
3432 CASE (WM_MENUSELECT);
3434 CASE (WM_ENTERIDLE);
3435 CASE (WM_MENURBUTTONUP);
3437 CASE (WM_MENUGETOBJECT);
3438 CASE (WM_UNINITMENUPOPUP);
3439 CASE (WM_MENUCOMMAND);
3440 CASE (WM_CHANGEUISTATE);
3441 CASE (WM_UPDATEUISTATE);
3442 CASE (WM_QUERYUISTATE);
3443 CASE (WM_CTLCOLORMSGBOX);
3444 CASE (WM_CTLCOLOREDIT);
3445 CASE (WM_CTLCOLORLISTBOX);
3446 CASE (WM_CTLCOLORBTN);
3447 CASE (WM_CTLCOLORDLG);
3448 CASE (WM_CTLCOLORSCROLLBAR);
3449 CASE (WM_CTLCOLORSTATIC);
3450 CASE (WM_MOUSEMOVE);
3451 CASE (WM_LBUTTONDOWN);
3452 CASE (WM_LBUTTONUP);
3453 CASE (WM_LBUTTONDBLCLK);
3454 CASE (WM_RBUTTONDOWN);
3455 CASE (WM_RBUTTONUP);
3456 CASE (WM_RBUTTONDBLCLK);
3457 CASE (WM_MBUTTONDOWN);
3458 CASE (WM_MBUTTONUP);
3459 CASE (WM_MBUTTONDBLCLK);
3460 CASE (WM_MOUSEWHEEL);
3461 CASE (WM_XBUTTONDOWN);
3462 CASE (WM_XBUTTONUP);
3463 CASE (WM_XBUTTONDBLCLK);
3464 CASE (WM_PARENTNOTIFY);
3465 CASE (WM_ENTERMENULOOP);
3466 CASE (WM_EXITMENULOOP);
3469 CASE (WM_CAPTURECHANGED);
3471 CASE (WM_POWERBROADCAST);
3472 CASE (WM_DEVICECHANGE);
3473 CASE (WM_MDICREATE);
3474 CASE (WM_MDIDESTROY);
3475 CASE (WM_MDIACTIVATE);
3476 CASE (WM_MDIRESTORE);
3478 CASE (WM_MDIMAXIMIZE);
3480 CASE (WM_MDICASCADE);
3481 CASE (WM_MDIICONARRANGE);
3482 CASE (WM_MDIGETACTIVE);
3483 CASE (WM_MDISETMENU);
3484 CASE (WM_ENTERSIZEMOVE);
3485 CASE (WM_EXITSIZEMOVE);
3486 CASE (WM_DROPFILES);
3487 CASE (WM_MDIREFRESHMENU);
3488 CASE (WM_IME_SETCONTEXT);
3489 CASE (WM_IME_NOTIFY);
3490 CASE (WM_IME_CONTROL);
3491 CASE (WM_IME_COMPOSITIONFULL);
3492 CASE (WM_IME_SELECT);
3494 CASE (WM_IME_REQUEST);
3495 CASE (WM_IME_KEYDOWN);
3496 CASE (WM_IME_KEYUP);
3497 CASE (WM_MOUSEHOVER);
3498 CASE (WM_MOUSELEAVE);
3499 CASE (WM_NCMOUSEHOVER);
3500 CASE (WM_NCMOUSELEAVE);
3506 CASE (WM_RENDERFORMAT);
3507 CASE (WM_RENDERALLFORMATS);
3508 CASE (WM_DESTROYCLIPBOARD);
3509 CASE (WM_DRAWCLIPBOARD);
3510 CASE (WM_PAINTCLIPBOARD);
3511 CASE (WM_VSCROLLCLIPBOARD);
3512 CASE (WM_SIZECLIPBOARD);
3513 CASE (WM_ASKCBFORMATNAME);
3514 CASE (WM_CHANGECBCHAIN);
3515 CASE (WM_HSCROLLCLIPBOARD);
3516 CASE (WM_QUERYNEWPALETTE);
3517 CASE (WM_PALETTEISCHANGING);
3518 CASE (WM_PALETTECHANGED);
3521 CASE (WM_PRINTCLIENT);
3522 CASE (WM_APPCOMMAND);
3523 CASE (WM_HANDHELDFIRST);
3524 CASE (WM_HANDHELDLAST);
3527 CASE (WM_PENWINFIRST);
3528 CASE (WM_PENWINLAST);
3532 if (msg >= WM_HANDHELDFIRST && msg <= WM_HANDHELDLAST)
3533 sprintf (bfr, "WM_HANDHELDFIRST+%d", msg - WM_HANDHELDFIRST);
3534 else if (msg >= WM_AFXFIRST && msg <= WM_AFXLAST)
3535 sprintf (bfr, "WM_AFXFIRST+%d", msg - WM_AFXFIRST);
3536 else if (msg >= WM_PENWINFIRST && msg <= WM_PENWINLAST)
3537 sprintf (bfr, "WM_PENWINFIRST+%d", msg - WM_PENWINFIRST);
3538 else if (msg >= WM_USER && msg <= 0x7FFF)
3539 sprintf (bfr, "WM_USER+%d", msg - WM_USER);
3540 else if (msg >= 0xC000 && msg <= 0xFFFF)
3541 sprintf (bfr, "reg-%#x", msg);
3543 sprintf (bfr, "unk-%#x", msg);
3546 g_assert_not_reached ();
3549 #endif /* G_ENABLE_DEBUG */