forked from AuroraMiddleware/gtk
995f00d23f
Use native Windows API for converting keystrokes to characters Closes #2944 See merge request GNOME/gtk!4986
3403 lines
100 KiB
C
3403 lines
100 KiB
C
/* GDK - The GIMP Drawing Kit
|
||
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
||
* Copyright (C) 1998-2002 Tor Lillqvist
|
||
* Copyright (C) 2001,2009 Hans Breuer
|
||
* Copyright (C) 2007-2009 Cody Russell
|
||
*
|
||
* This library is free software; you can redistribute it and/or
|
||
* modify it under the terms of the GNU Lesser General Public
|
||
* License as published by the Free Software Foundation; either
|
||
* version 2 of the License, or (at your option) any later version.
|
||
*
|
||
* This library is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
* Lesser General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Lesser General Public
|
||
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2020. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
/* Cannot use TrackMouseEvent, as the stupid WM_MOUSELEAVE message
|
||
* doesn’t tell us where the mouse has gone. Thus we cannot use it to
|
||
* generate a correct GdkNotifyType. Pity, as using TrackMouseEvent
|
||
* otherwise would make it possible to reliably generate
|
||
* GDK_LEAVE_NOTIFY events, which would help get rid of those pesky
|
||
* tooltips sometimes popping up in the wrong place.
|
||
*
|
||
* Update: a combination of TrackMouseEvent, GetCursorPos and
|
||
* GetWindowPos can and is actually used to get rid of those
|
||
* pesky tooltips. It should be possible to use this for the
|
||
* whole ENTER/LEAVE NOTIFY handling but some platforms may
|
||
* not have TrackMouseEvent at all (?) --hb
|
||
*/
|
||
|
||
#include "config.h"
|
||
|
||
#include "gdkprivate-win32.h"
|
||
|
||
#include <glib/gprintf.h>
|
||
#include <cairo-win32.h>
|
||
|
||
#include "gdk.h"
|
||
#include "gdkdevicemanager-win32.h"
|
||
#include "gdkdisplayprivate.h"
|
||
#include "gdkeventsprivate.h"
|
||
#include "gdkglcontext-win32.h"
|
||
#include "gdkkeysyms.h"
|
||
#include "gdkmonitorprivate.h"
|
||
#include "gdkwin32.h"
|
||
#include "gdkdisplay-win32.h"
|
||
#include "gdkdeviceprivate.h"
|
||
#include "gdkdevice-virtual.h"
|
||
#include "gdkdevice-wintab.h"
|
||
#include "gdkinput-dmanipulation.h"
|
||
#include "gdkinput-winpointer.h"
|
||
#include "gdkwin32dnd.h"
|
||
#include "gdkwin32dnd-private.h"
|
||
#include "gdkdisplay-win32.h"
|
||
//#include "gdkselection-win32.h"
|
||
#include "gdkdragprivate.h"
|
||
#include "gdkprivate.h"
|
||
|
||
#include <windowsx.h>
|
||
|
||
#ifdef G_WITH_CYGWIN
|
||
#include <fcntl.h>
|
||
#include <errno.h>
|
||
#endif
|
||
|
||
#include <objbase.h>
|
||
#include <imm.h>
|
||
#include <tchar.h>
|
||
#include <tpcshrd.h>
|
||
|
||
#include <stdint.h>
|
||
|
||
#define GDK_MOD2_MASK (1 << 4)
|
||
|
||
#ifndef XBUTTON1
|
||
#define XBUTTON1 1
|
||
#define XBUTTON2 2
|
||
#endif
|
||
|
||
#ifndef VK_XBUTTON1
|
||
#define VK_XBUTTON1 5
|
||
#define VK_XBUTTON2 6
|
||
#endif
|
||
|
||
#ifndef MK_XBUTTON1
|
||
#define MK_XBUTTON1 32
|
||
#define MK_XBUTTON2 64
|
||
#endif
|
||
|
||
/* Undefined flags: */
|
||
#define SWP_NOCLIENTSIZE 0x0800
|
||
#define SWP_NOCLIENTMOVE 0x1000
|
||
#define SWP_STATECHANGED 0x8000
|
||
/*
|
||
* Private function declarations
|
||
*/
|
||
|
||
#define SYNAPSIS_ICON_WINDOW_CLASS "SynTrackCursorWindowClass"
|
||
|
||
static gboolean gdk_event_translate (MSG *msg,
|
||
int *ret_valp);
|
||
static gboolean gdk_event_prepare (GSource *source,
|
||
int *timeout);
|
||
static gboolean gdk_event_check (GSource *source);
|
||
static gboolean gdk_event_dispatch (GSource *source,
|
||
GSourceFunc callback,
|
||
gpointer user_data);
|
||
|
||
/* Private variable declarations
|
||
*/
|
||
|
||
extern int _gdk_input_ignore_core;
|
||
|
||
typedef struct
|
||
{
|
||
GSource source;
|
||
|
||
GdkDisplay *display;
|
||
GPollFD event_poll_fd;
|
||
} GdkWin32EventSource;
|
||
|
||
static GSourceFuncs event_funcs = {
|
||
gdk_event_prepare,
|
||
gdk_event_check,
|
||
gdk_event_dispatch,
|
||
NULL
|
||
};
|
||
|
||
static GdkSurface *mouse_window = NULL;
|
||
static GdkSurface *mouse_window_ignored_leave = NULL;
|
||
static int current_x, current_y;
|
||
static int current_root_x, current_root_y;
|
||
|
||
static UINT got_gdk_events_message;
|
||
static HWND modal_win32_dialog = NULL;
|
||
|
||
#if 0
|
||
static HKL latin_locale = NULL;
|
||
#endif
|
||
|
||
static gboolean in_ime_composition = FALSE;
|
||
static UINT modal_timer;
|
||
|
||
static int debug_indent = 0;
|
||
|
||
static int both_shift_pressed[2]; /* to store keycodes for shift keys */
|
||
|
||
/* low-level keyboard hook handle */
|
||
static HHOOK keyboard_hook = NULL;
|
||
static UINT aerosnap_message;
|
||
|
||
static gboolean pen_touch_input;
|
||
static POINT pen_touch_cursor_position;
|
||
static LONG last_digitizer_time;
|
||
|
||
static void
|
||
track_mouse_event (DWORD dwFlags,
|
||
HWND hwnd)
|
||
{
|
||
TRACKMOUSEEVENT tme;
|
||
|
||
tme.cbSize = sizeof(TRACKMOUSEEVENT);
|
||
tme.dwFlags = dwFlags;
|
||
tme.hwndTrack = hwnd;
|
||
tme.dwHoverTime = HOVER_DEFAULT; /* not used */
|
||
|
||
if (!TrackMouseEvent (&tme))
|
||
WIN32_API_FAILED ("TrackMouseEvent");
|
||
else if (dwFlags == TME_LEAVE)
|
||
GDK_NOTE (EVENTS, g_print(" (TrackMouseEvent %p)", hwnd));
|
||
else if (dwFlags == TME_CANCEL)
|
||
GDK_NOTE (EVENTS, g_print(" (cancel TrackMouseEvent %p)", hwnd));
|
||
}
|
||
|
||
gulong
|
||
_gdk_win32_get_next_tick (gulong suggested_tick)
|
||
{
|
||
static gulong cur_tick = 0;
|
||
|
||
if (suggested_tick == 0)
|
||
suggested_tick = GetTickCount ();
|
||
/* Ticks eventually wrap around.
|
||
* This works as long as the interval between ticks is < 2147483648ms */
|
||
if (suggested_tick <= cur_tick && ((cur_tick - suggested_tick) < 0x7FFFFFFF))
|
||
return cur_tick;
|
||
else
|
||
return cur_tick = suggested_tick;
|
||
}
|
||
|
||
BOOL
|
||
_gdk_win32_get_cursor_pos (LPPOINT lpPoint)
|
||
{
|
||
if (pen_touch_input)
|
||
{
|
||
*lpPoint = pen_touch_cursor_position;
|
||
return TRUE;
|
||
}
|
||
else
|
||
return GetCursorPos (lpPoint);
|
||
}
|
||
|
||
static void
|
||
generate_focus_event (GdkDeviceManagerWin32 *device_manager,
|
||
GdkSurface *window,
|
||
gboolean in)
|
||
{
|
||
GdkDevice *device;
|
||
GdkEvent *event;
|
||
|
||
device = GDK_DEVICE_MANAGER_WIN32 (device_manager)->core_keyboard;
|
||
|
||
event = gdk_focus_event_new (window, device, in);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
static void
|
||
generate_grab_broken_event (GdkDeviceManagerWin32 *device_manager,
|
||
GdkSurface *window,
|
||
gboolean keyboard,
|
||
GdkSurface *grab_window)
|
||
{
|
||
GdkEvent *event;
|
||
GdkDevice *device;
|
||
|
||
if (keyboard)
|
||
device = device_manager->core_keyboard;
|
||
else
|
||
device = device_manager->core_pointer;
|
||
|
||
event = gdk_grab_broken_event_new (window,
|
||
device,
|
||
grab_window,
|
||
FALSE);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
static LRESULT
|
||
inner_window_procedure (HWND hwnd,
|
||
UINT message,
|
||
WPARAM wparam,
|
||
LPARAM lparam)
|
||
{
|
||
MSG msg;
|
||
DWORD pos;
|
||
int ret_val = 0;
|
||
|
||
msg.hwnd = hwnd;
|
||
msg.message = message;
|
||
msg.wParam = wparam;
|
||
msg.lParam = lparam;
|
||
msg.time = _gdk_win32_get_next_tick (0);
|
||
pos = GetMessagePos ();
|
||
msg.pt.x = GET_X_LPARAM (pos);
|
||
msg.pt.y = GET_Y_LPARAM (pos);
|
||
|
||
if (gdk_event_translate (&msg, &ret_val))
|
||
{
|
||
/* If gdk_event_translate() returns TRUE, we return ret_val from
|
||
* the window procedure.
|
||
*/
|
||
if (modal_win32_dialog)
|
||
PostMessageW (modal_win32_dialog, got_gdk_events_message,
|
||
(WPARAM) 1, 0);
|
||
return ret_val;
|
||
}
|
||
else
|
||
{
|
||
/* Otherwise call DefWindowProcW(). */
|
||
GDK_NOTE (EVENTS, g_print (" DefWindowProcW"));
|
||
return DefWindowProcW (hwnd, message, wparam, lparam);
|
||
}
|
||
}
|
||
|
||
LRESULT CALLBACK
|
||
_gdk_win32_surface_procedure (HWND hwnd,
|
||
UINT message,
|
||
WPARAM wparam,
|
||
LPARAM lparam)
|
||
{
|
||
LRESULT retval;
|
||
|
||
GDK_NOTE (EVENTS, g_print ("%s%*s%s %p %#" G_GINTPTR_MODIFIER "x %#" G_GINTPTR_MODIFIER "x",
|
||
(debug_indent > 0 ? "\n" : ""),
|
||
debug_indent, "",
|
||
_gdk_win32_message_to_string (message), hwnd,
|
||
wparam, lparam));
|
||
debug_indent += 2;
|
||
retval = inner_window_procedure (hwnd, message, wparam, lparam);
|
||
debug_indent -= 2;
|
||
|
||
GDK_NOTE (EVENTS, g_print (" => %" G_GINT64_FORMAT "%s", (gint64) retval, (debug_indent == 0 ? "\n" : "")));
|
||
|
||
return retval;
|
||
}
|
||
|
||
static LRESULT
|
||
low_level_keystroke_handler (WPARAM message,
|
||
KBDLLHOOKSTRUCT *kbdhook,
|
||
GdkSurface *window)
|
||
{
|
||
GdkSurface *toplevel = window;
|
||
static DWORD last_keydown = 0;
|
||
|
||
if (message == WM_KEYDOWN &&
|
||
!GDK_SURFACE_DESTROYED (toplevel) &&
|
||
_gdk_win32_surface_lacks_wm_decorations (toplevel) && /* For CSD only */
|
||
last_keydown != kbdhook->vkCode &&
|
||
((GetKeyState (VK_LWIN) & 0x8000) ||
|
||
(GetKeyState (VK_RWIN) & 0x8000)))
|
||
{
|
||
GdkWin32AeroSnapCombo combo = GDK_WIN32_AEROSNAP_COMBO_NOTHING;
|
||
gboolean lshiftdown = GetKeyState (VK_LSHIFT) & 0x8000;
|
||
gboolean rshiftdown = GetKeyState (VK_RSHIFT) & 0x8000;
|
||
gboolean oneshiftdown = (lshiftdown || rshiftdown) && !(lshiftdown && rshiftdown);
|
||
gboolean maximized = gdk_toplevel_get_state (GDK_TOPLEVEL (toplevel)) & GDK_TOPLEVEL_STATE_MAXIMIZED;
|
||
|
||
switch (kbdhook->vkCode)
|
||
{
|
||
case VK_UP:
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_UP;
|
||
break;
|
||
case VK_DOWN:
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_DOWN;
|
||
break;
|
||
case VK_LEFT:
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_LEFT;
|
||
break;
|
||
case VK_RIGHT:
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_RIGHT;
|
||
break;
|
||
}
|
||
|
||
if (oneshiftdown && combo != GDK_WIN32_AEROSNAP_COMBO_NOTHING)
|
||
combo += 4;
|
||
|
||
/* These are the only combos that Windows WM does handle for us */
|
||
if (combo == GDK_WIN32_AEROSNAP_COMBO_SHIFTLEFT ||
|
||
combo == GDK_WIN32_AEROSNAP_COMBO_SHIFTRIGHT)
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_NOTHING;
|
||
|
||
/* On Windows 10 the WM will handle this specific combo */
|
||
if (combo == GDK_WIN32_AEROSNAP_COMBO_DOWN && maximized &&
|
||
g_win32_check_windows_version (6, 4, 0, G_WIN32_OS_ANY))
|
||
combo = GDK_WIN32_AEROSNAP_COMBO_NOTHING;
|
||
|
||
if (combo != GDK_WIN32_AEROSNAP_COMBO_NOTHING)
|
||
PostMessage (GDK_SURFACE_HWND (toplevel), aerosnap_message, (WPARAM) combo, 0);
|
||
}
|
||
|
||
if (message == WM_KEYDOWN)
|
||
last_keydown = kbdhook->vkCode;
|
||
else if (message == WM_KEYUP && last_keydown == kbdhook->vkCode)
|
||
last_keydown = 0;
|
||
|
||
return 0;
|
||
}
|
||
|
||
static LRESULT CALLBACK
|
||
low_level_keyboard_proc (int code,
|
||
WPARAM wParam,
|
||
LPARAM lParam)
|
||
{
|
||
KBDLLHOOKSTRUCT *kbdhook;
|
||
HWND kbd_focus_owner;
|
||
GdkSurface *gdk_kbd_focus_owner;
|
||
LRESULT chain;
|
||
|
||
do
|
||
{
|
||
if (code < 0)
|
||
break;
|
||
|
||
kbd_focus_owner = GetFocus ();
|
||
|
||
if (kbd_focus_owner == NULL)
|
||
break;
|
||
|
||
gdk_kbd_focus_owner = gdk_win32_handle_table_lookup_ (kbd_focus_owner);
|
||
|
||
if (gdk_kbd_focus_owner == NULL)
|
||
break;
|
||
|
||
kbdhook = (KBDLLHOOKSTRUCT *) lParam;
|
||
chain = low_level_keystroke_handler (wParam, kbdhook, gdk_kbd_focus_owner);
|
||
|
||
if (chain != 0)
|
||
return chain;
|
||
} while (FALSE);
|
||
|
||
return CallNextHookEx (0, code, wParam, lParam);
|
||
}
|
||
|
||
static void
|
||
set_up_low_level_keyboard_hook (void)
|
||
{
|
||
HHOOK hook_handle;
|
||
|
||
if (keyboard_hook != NULL)
|
||
return;
|
||
|
||
hook_handle = SetWindowsHookEx (WH_KEYBOARD_LL,
|
||
(HOOKPROC) low_level_keyboard_proc,
|
||
_gdk_dll_hinstance,
|
||
0);
|
||
|
||
if (hook_handle != NULL)
|
||
keyboard_hook = hook_handle;
|
||
else
|
||
WIN32_API_FAILED ("SetWindowsHookEx");
|
||
|
||
aerosnap_message = RegisterWindowMessage ("GDK_WIN32_AEROSNAP_MESSAGE");
|
||
}
|
||
|
||
void
|
||
_gdk_events_init (GdkDisplay *display)
|
||
{
|
||
GSource *source;
|
||
GdkWin32EventSource *event_source;
|
||
|
||
#if 0
|
||
int i, j, n;
|
||
|
||
/* List of languages that use a latin keyboard. Somewhat sorted in
|
||
* "order of least surprise", in case we have to load one of them if
|
||
* the user only has arabic loaded, for instance.
|
||
*/
|
||
static int latin_languages[] = {
|
||
LANG_ENGLISH,
|
||
LANG_SPANISH,
|
||
LANG_PORTUGUESE,
|
||
LANG_FRENCH,
|
||
LANG_GERMAN,
|
||
/* Rest in numeric order */
|
||
LANG_CZECH,
|
||
LANG_DANISH,
|
||
LANG_FINNISH,
|
||
LANG_HUNGARIAN,
|
||
LANG_ICELANDIC,
|
||
LANG_ITALIAN,
|
||
LANG_DUTCH,
|
||
LANG_NORWEGIAN,
|
||
LANG_POLISH,
|
||
LANG_ROMANIAN,
|
||
LANG_SLOVAK,
|
||
LANG_ALBANIAN,
|
||
LANG_SWEDISH,
|
||
LANG_TURKISH,
|
||
LANG_INDONESIAN,
|
||
LANG_SLOVENIAN,
|
||
LANG_ESTONIAN,
|
||
LANG_LATVIAN,
|
||
LANG_LITHUANIAN,
|
||
LANG_VIETNAMESE,
|
||
LANG_AFRIKAANS,
|
||
LANG_FAEROESE
|
||
#ifdef LANG_SWAHILI
|
||
,LANG_SWAHILI
|
||
#endif
|
||
};
|
||
#endif
|
||
|
||
got_gdk_events_message = RegisterWindowMessage ("GDK_WIN32_GOT_EVENTS");
|
||
|
||
#if 0
|
||
/* Check if we have some input locale identifier loaded that uses a
|
||
* latin keyboard, to be able to get the virtual-key code for the
|
||
* latin characters corresponding to ASCII control characters.
|
||
*/
|
||
if ((n = GetKeyboardLayoutList (0, NULL)) == 0)
|
||
WIN32_API_FAILED ("GetKeyboardLayoutList");
|
||
else
|
||
{
|
||
HKL *hkl_list = g_new (HKL, n);
|
||
if (GetKeyboardLayoutList (n, hkl_list) == 0)
|
||
WIN32_API_FAILED ("GetKeyboardLayoutList");
|
||
else
|
||
{
|
||
for (i = 0; latin_locale == NULL && i < n; i++)
|
||
for (j = 0; j < G_N_ELEMENTS (latin_languages); j++)
|
||
if (PRIMARYLANGID (LOWORD (hkl_list[i])) == latin_languages[j])
|
||
{
|
||
latin_locale = hkl_list [i];
|
||
break;
|
||
}
|
||
}
|
||
g_free (hkl_list);
|
||
}
|
||
|
||
if (latin_locale == NULL)
|
||
{
|
||
/* Try to load a keyboard layout with latin characters then.
|
||
*/
|
||
i = 0;
|
||
while (latin_locale == NULL && i < G_N_ELEMENTS (latin_languages))
|
||
{
|
||
char id[9];
|
||
g_sprintf (id, "%08x", MAKELANGID (latin_languages[i++], SUBLANG_DEFAULT));
|
||
latin_locale = LoadKeyboardLayout (id, KLF_NOTELLSHELL|KLF_SUBSTITUTE_OK);
|
||
}
|
||
}
|
||
|
||
GDK_NOTE (EVENTS, g_print ("latin_locale = %08x\n", (guint) latin_locale));
|
||
#endif
|
||
|
||
source = g_source_new (&event_funcs, sizeof (GdkWin32EventSource));
|
||
g_source_set_static_name (source, "GDK Win32 event source");
|
||
g_source_set_priority (source, GDK_PRIORITY_EVENTS);
|
||
|
||
event_source = (GdkWin32EventSource *)source;
|
||
event_source->display = display;
|
||
#ifdef G_WITH_CYGWIN
|
||
event_source->event_poll_fd.fd = open ("/dev/windows", O_RDONLY);
|
||
if (event_source->event_poll_fd.fd == -1)
|
||
g_error ("can't open \"/dev/windows\": %s", g_strerror (errno));
|
||
#else
|
||
event_source->event_poll_fd.fd = G_WIN32_MSG_HANDLE;
|
||
#endif
|
||
event_source->event_poll_fd.events = G_IO_IN;
|
||
|
||
g_source_add_poll (source, &event_source->event_poll_fd);
|
||
g_source_set_can_recurse (source, TRUE);
|
||
g_source_attach (source, NULL);
|
||
|
||
set_up_low_level_keyboard_hook ();
|
||
}
|
||
|
||
gboolean
|
||
_gdk_win32_display_has_pending (GdkDisplay *display)
|
||
{
|
||
return (_gdk_event_queue_find_first (display) ||
|
||
(modal_win32_dialog == NULL &&
|
||
GetQueueStatus (QS_ALLINPUT) != 0));
|
||
}
|
||
|
||
#if 0 /* Unused, but might be useful to re-introduce in some debugging output? */
|
||
|
||
static char *
|
||
event_mask_string (GdkEventMask mask)
|
||
{
|
||
static char bfr[500];
|
||
char *p = bfr;
|
||
|
||
*p = '\0';
|
||
#define BIT(x) \
|
||
if (mask & GDK_##x##_MASK) \
|
||
p += g_sprintf (p, "%s" #x, (p > bfr ? " " : ""))
|
||
BIT (EXPOSURE);
|
||
BIT (POINTER_MOTION);
|
||
BIT (POINTER_MOTION_HINT);
|
||
BIT (BUTTON_MOTION);
|
||
BIT (BUTTON1_MOTION);
|
||
BIT (BUTTON2_MOTION);
|
||
BIT (BUTTON3_MOTION);
|
||
BIT (BUTTON_PRESS);
|
||
BIT (BUTTON_RELEASE);
|
||
BIT (KEY_PRESS);
|
||
BIT (KEY_RELEASE);
|
||
BIT (ENTER_NOTIFY);
|
||
BIT (LEAVE_NOTIFY);
|
||
BIT (FOCUS_CHANGE);
|
||
BIT (STRUCTURE);
|
||
BIT (PROPERTY_CHANGE);
|
||
BIT (VISIBILITY_NOTIFY);
|
||
BIT (PROXIMITY_IN);
|
||
BIT (PROXIMITY_OUT);
|
||
BIT (SCROLL);
|
||
#undef BIT
|
||
|
||
return bfr;
|
||
}
|
||
|
||
#endif
|
||
|
||
static GdkSurface *
|
||
find_window_for_mouse_event (GdkSurface* reported_window,
|
||
MSG* msg)
|
||
{
|
||
POINT pt;
|
||
GdkDisplay *display;
|
||
GdkDeviceManagerWin32 *device_manager;
|
||
GdkSurface *event_surface;
|
||
HWND hwnd;
|
||
RECT rect;
|
||
GdkDeviceGrabInfo *grab;
|
||
|
||
display = gdk_display_get_default ();
|
||
device_manager = GDK_DEVICE_MANAGER_WIN32 (_gdk_device_manager);
|
||
|
||
grab = _gdk_display_get_last_device_grab (display, device_manager->core_pointer);
|
||
if (grab == NULL)
|
||
return reported_window;
|
||
|
||
pt = msg->pt;
|
||
|
||
if (!grab->owner_events)
|
||
event_surface = grab->surface;
|
||
else
|
||
{
|
||
event_surface = NULL;
|
||
hwnd = WindowFromPoint (pt);
|
||
if (hwnd != NULL)
|
||
{
|
||
POINT client_pt = pt;
|
||
|
||
ScreenToClient (hwnd, &client_pt);
|
||
GetClientRect (hwnd, &rect);
|
||
if (PtInRect (&rect, client_pt))
|
||
event_surface = gdk_win32_handle_table_lookup_ (hwnd);
|
||
}
|
||
if (event_surface == NULL)
|
||
event_surface = grab->surface;
|
||
}
|
||
|
||
/* need to also adjust the coordinates to the new window */
|
||
ScreenToClient (GDK_SURFACE_HWND (event_surface), &pt);
|
||
|
||
/* ATTENTION: need to update client coords */
|
||
msg->lParam = MAKELPARAM (pt.x, pt.y);
|
||
|
||
return event_surface;
|
||
}
|
||
|
||
static GdkModifierType
|
||
build_key_event_state (BYTE *key_state)
|
||
{
|
||
GdkModifierType state;
|
||
GdkWin32Keymap *keymap;
|
||
keymap = GDK_WIN32_KEYMAP (_gdk_win32_display_get_keymap (_gdk_display));
|
||
|
||
state = _gdk_win32_keymap_get_mod_mask (keymap);
|
||
|
||
if (key_state[VK_CAPITAL] & 0x01)
|
||
state |= GDK_LOCK_MASK;
|
||
|
||
if (key_state[VK_LBUTTON] & 0x80)
|
||
state |= GDK_BUTTON1_MASK;
|
||
if (key_state[VK_MBUTTON] & 0x80)
|
||
state |= GDK_BUTTON2_MASK;
|
||
if (key_state[VK_RBUTTON] & 0x80)
|
||
state |= GDK_BUTTON3_MASK;
|
||
if (key_state[VK_XBUTTON1] & 0x80)
|
||
state |= GDK_BUTTON4_MASK;
|
||
if (key_state[VK_XBUTTON2] & 0x80)
|
||
state |= GDK_BUTTON5_MASK;
|
||
|
||
return state;
|
||
}
|
||
|
||
static guint8
|
||
get_active_group (void)
|
||
{
|
||
GdkWin32Keymap *keymap;
|
||
|
||
keymap = GDK_WIN32_KEYMAP (_gdk_win32_display_get_keymap (_gdk_display));
|
||
|
||
return _gdk_win32_keymap_get_active_group (keymap);
|
||
}
|
||
|
||
static int
|
||
build_pointer_event_state (MSG *msg)
|
||
{
|
||
int state;
|
||
|
||
state = 0;
|
||
|
||
if (msg->wParam & MK_CONTROL)
|
||
state |= GDK_CONTROL_MASK;
|
||
|
||
if ((msg->message != WM_LBUTTONDOWN &&
|
||
(msg->wParam & MK_LBUTTON)) ||
|
||
msg->message == WM_LBUTTONUP)
|
||
state |= GDK_BUTTON1_MASK;
|
||
|
||
if ((msg->message != WM_MBUTTONDOWN &&
|
||
(msg->wParam & MK_MBUTTON)) ||
|
||
msg->message == WM_MBUTTONUP)
|
||
state |= GDK_BUTTON2_MASK;
|
||
|
||
if ((msg->message != WM_RBUTTONDOWN &&
|
||
(msg->wParam & MK_RBUTTON)) ||
|
||
msg->message == WM_RBUTTONUP)
|
||
state |= GDK_BUTTON3_MASK;
|
||
|
||
if (((msg->message != WM_XBUTTONDOWN || HIWORD (msg->wParam) != XBUTTON1) &&
|
||
(msg->wParam & MK_XBUTTON1)) ||
|
||
(msg->message == WM_XBUTTONUP && HIWORD (msg->wParam) == XBUTTON1))
|
||
state |= GDK_BUTTON4_MASK;
|
||
|
||
if (((msg->message != WM_XBUTTONDOWN || HIWORD (msg->wParam) != XBUTTON2) &&
|
||
(msg->wParam & MK_XBUTTON2)) ||
|
||
(msg->message == WM_XBUTTONUP && HIWORD (msg->wParam) == XBUTTON2))
|
||
state |= GDK_BUTTON5_MASK;
|
||
|
||
if (msg->wParam & MK_SHIFT)
|
||
state |= GDK_SHIFT_MASK;
|
||
|
||
if (GetKeyState (VK_MENU) < 0)
|
||
state |= GDK_ALT_MASK;
|
||
|
||
if (GetKeyState (VK_CAPITAL) & 0x1)
|
||
state |= GDK_LOCK_MASK;
|
||
|
||
return state;
|
||
}
|
||
|
||
#ifdef G_ENABLE_DEBUG
|
||
|
||
static void
|
||
print_event_state (guint state)
|
||
{
|
||
#define CASE(bit) if (state & GDK_ ## bit ## _MASK) g_print (#bit " ");
|
||
CASE (SHIFT);
|
||
CASE (LOCK);
|
||
CASE (CONTROL);
|
||
CASE (ALT);
|
||
CASE (MOD2);
|
||
CASE (BUTTON1);
|
||
CASE (BUTTON2);
|
||
CASE (BUTTON3);
|
||
CASE (BUTTON4);
|
||
CASE (BUTTON5);
|
||
#undef CASE
|
||
}
|
||
|
||
void
|
||
_gdk_win32_print_event (GdkEvent *event)
|
||
{
|
||
const char *kvname;
|
||
double x, y;
|
||
GdkCrossingMode mode;
|
||
GdkNotifyType detail;
|
||
GdkScrollDirection direction;
|
||
|
||
g_print ("%s%*s===> ", (debug_indent > 0 ? "\n" : ""), debug_indent, "");
|
||
switch (gdk_event_get_event_type (event))
|
||
{
|
||
#define CASE(x) case x: g_print (#x); break;
|
||
CASE (GDK_DELETE);
|
||
CASE (GDK_MOTION_NOTIFY);
|
||
CASE (GDK_BUTTON_PRESS);
|
||
CASE (GDK_BUTTON_RELEASE);
|
||
CASE (GDK_KEY_PRESS);
|
||
CASE (GDK_KEY_RELEASE);
|
||
CASE (GDK_ENTER_NOTIFY);
|
||
CASE (GDK_LEAVE_NOTIFY);
|
||
CASE (GDK_FOCUS_CHANGE);
|
||
CASE (GDK_PROXIMITY_IN);
|
||
CASE (GDK_PROXIMITY_OUT);
|
||
CASE (GDK_DRAG_ENTER);
|
||
CASE (GDK_DRAG_LEAVE);
|
||
CASE (GDK_DRAG_MOTION);
|
||
CASE (GDK_DROP_START);
|
||
CASE (GDK_SCROLL);
|
||
CASE (GDK_GRAB_BROKEN);
|
||
#undef CASE
|
||
default: g_assert_not_reached ();
|
||
}
|
||
|
||
g_print (" %p @ %ums ",
|
||
GDK_SURFACE_HWND (gdk_event_get_surface (event)),
|
||
gdk_event_get_time (event));
|
||
|
||
switch (gdk_event_get_event_type (event))
|
||
{
|
||
case GDK_MOTION_NOTIFY:
|
||
gdk_event_get_position (event, &x, &y);
|
||
g_print ("(%.4g,%.4g) ", x, y);
|
||
print_event_state (gdk_event_get_modifier_state (event));
|
||
break;
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
gdk_event_get_position (event, &x, &y);
|
||
g_print ("%d (%.4g,%.4g) ", gdk_button_event_get_button (event), x, y);
|
||
print_event_state (gdk_event_get_modifier_state (event));
|
||
break;
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
kvname = gdk_keyval_name (gdk_key_event_get_keyval (event));
|
||
g_print ("%#.02x group:%d %s",
|
||
gdk_key_event_get_keycode (event),
|
||
gdk_key_event_get_layout (event),
|
||
(kvname ? kvname : "??"));
|
||
print_event_state (gdk_event_get_modifier_state (event));
|
||
break;
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_LEAVE_NOTIFY:
|
||
gdk_event_get_position (event, &x, &y);
|
||
mode = gdk_crossing_event_get_mode (event);
|
||
detail = gdk_crossing_event_get_detail (event);
|
||
g_print ("(%.4g,%.4g) %s %s",
|
||
x, y,
|
||
(mode == GDK_CROSSING_NORMAL ? "NORMAL" :
|
||
(mode == GDK_CROSSING_GRAB ? "GRAB" :
|
||
(mode == GDK_CROSSING_UNGRAB ? "UNGRAB" :
|
||
"???"))),
|
||
(detail == GDK_NOTIFY_ANCESTOR ? "ANCESTOR" :
|
||
(detail == GDK_NOTIFY_VIRTUAL ? "VIRTUAL" :
|
||
(detail == GDK_NOTIFY_INFERIOR ? "INFERIOR" :
|
||
(detail == GDK_NOTIFY_NONLINEAR ? "NONLINEAR" :
|
||
(detail == GDK_NOTIFY_NONLINEAR_VIRTUAL ? "NONLINEAR_VIRTUAL" :
|
||
(detail == GDK_NOTIFY_UNKNOWN ? "UNKNOWN" :
|
||
"???")))))));
|
||
print_event_state (gdk_event_get_modifier_state (event));
|
||
break;
|
||
case GDK_FOCUS_CHANGE:
|
||
g_print ("%s", (gdk_focus_event_get_in (event) ? "IN" : "OUT"));
|
||
break;
|
||
case GDK_DRAG_ENTER:
|
||
case GDK_DRAG_LEAVE:
|
||
case GDK_DRAG_MOTION:
|
||
case GDK_DROP_START:
|
||
g_print ("DND");
|
||
break;
|
||
case GDK_SCROLL:
|
||
direction = gdk_scroll_event_get_direction (event);
|
||
g_print (" %s ",
|
||
(direction == GDK_SCROLL_UP ? "UP" :
|
||
(direction == GDK_SCROLL_DOWN ? "DOWN" :
|
||
(direction == GDK_SCROLL_LEFT ? "LEFT" :
|
||
(direction == GDK_SCROLL_RIGHT ? "RIGHT" :
|
||
"???")))));
|
||
print_event_state (gdk_event_get_modifier_state (event));
|
||
break;
|
||
case GDK_GRAB_BROKEN:
|
||
g_print ("Grab broken");
|
||
default:
|
||
/* Nothing */
|
||
break;
|
||
}
|
||
g_print ("%s", (debug_indent == 0 ? "\n" : ""));
|
||
}
|
||
|
||
static char *
|
||
decode_key_lparam (LPARAM lParam)
|
||
{
|
||
static char buf[100];
|
||
char *p = buf;
|
||
|
||
if (HIWORD (lParam) & KF_UP)
|
||
p += g_sprintf (p, "KF_UP ");
|
||
if (HIWORD (lParam) & KF_REPEAT)
|
||
p += g_sprintf (p, "KF_REPEAT ");
|
||
if (HIWORD (lParam) & KF_ALTDOWN)
|
||
p += g_sprintf (p, "KF_ALTDOWN ");
|
||
if (HIWORD (lParam) & KF_EXTENDED)
|
||
p += g_sprintf (p, "KF_EXTENDED ");
|
||
p += g_sprintf (p, "sc:%d rep:%d", LOBYTE (HIWORD (lParam)), LOWORD (lParam));
|
||
|
||
return buf;
|
||
}
|
||
|
||
#endif
|
||
|
||
static void
|
||
fixup_event (GdkEvent *event)
|
||
{
|
||
if (event->surface)
|
||
g_object_ref (event->surface);
|
||
}
|
||
|
||
void
|
||
_gdk_win32_append_event (GdkEvent *event)
|
||
{
|
||
GdkDisplay *display;
|
||
GList *link;
|
||
gulong serial;
|
||
|
||
display = gdk_display_get_default ();
|
||
|
||
fixup_event (event);
|
||
#if 1
|
||
link = _gdk_event_queue_append (display, event);
|
||
GDK_NOTE (EVENTS, _gdk_win32_print_event (event));
|
||
serial = _gdk_display_get_next_serial (display);
|
||
/* event morphing, the passed in may not be valid afterwards */
|
||
_gdk_windowing_got_event (display, link, event, serial);
|
||
#else
|
||
_gdk_event_queue_append (display, event);
|
||
GDK_NOTE (EVENTS, _gdk_win32_print_event (event));
|
||
#endif
|
||
}
|
||
|
||
static GdkWin32MessageFilterReturn
|
||
apply_message_filters (GdkDisplay *display,
|
||
MSG *msg,
|
||
int *ret_valp,
|
||
GList **filters)
|
||
{
|
||
GdkWin32MessageFilterReturn result = GDK_WIN32_MESSAGE_FILTER_CONTINUE;
|
||
GList *tmp_list;
|
||
|
||
tmp_list = *filters;
|
||
while (tmp_list)
|
||
{
|
||
GdkWin32MessageFilter *filter = (GdkWin32MessageFilter *) tmp_list->data;
|
||
GList *node;
|
||
|
||
if (filter->removed)
|
||
{
|
||
tmp_list = tmp_list->next;
|
||
continue;
|
||
}
|
||
|
||
filter->ref_count++;
|
||
result = filter->function (GDK_WIN32_DISPLAY (display), msg, ret_valp, filter->data);
|
||
|
||
/* get the next node after running the function since the
|
||
function may add or remove a next node */
|
||
node = tmp_list;
|
||
tmp_list = tmp_list->next;
|
||
|
||
filter->ref_count--;
|
||
if (filter->ref_count == 0)
|
||
{
|
||
*filters = g_list_remove_link (*filters, node);
|
||
g_list_free_1 (node);
|
||
g_free (filter);
|
||
}
|
||
|
||
if (result != GDK_WIN32_MESSAGE_FILTER_CONTINUE)
|
||
break;
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
/*
|
||
* On Windows, transient windows will not have their own taskbar entries.
|
||
* Because of this, we must hide and restore groups of transients in both
|
||
* directions. That is, all transient children must be hidden or restored
|
||
* with this window, but if this window’s transient owner also has a
|
||
* transient owner then this window’s transient owner must be hidden/restored
|
||
* with this one. And etc, up the chain until we hit an ancestor that has no
|
||
* transient owner.
|
||
*
|
||
* It would be a good idea if applications don’t chain transient windows
|
||
* together. There’s a limit to how much evil GTK can try to shield you
|
||
* from.
|
||
*/
|
||
static void
|
||
show_window_recurse (GdkSurface *window, gboolean hide_window)
|
||
{
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
GSList *children = impl->transient_children;
|
||
GdkSurface *child = NULL;
|
||
|
||
if (!impl->changing_state)
|
||
{
|
||
impl->changing_state = TRUE;
|
||
|
||
if (children != NULL)
|
||
{
|
||
while (children != NULL)
|
||
{
|
||
child = children->data;
|
||
show_window_recurse (child, hide_window);
|
||
|
||
children = children->next;
|
||
}
|
||
}
|
||
|
||
if (GDK_SURFACE_IS_MAPPED (window))
|
||
{
|
||
if (!hide_window)
|
||
{
|
||
if (gdk_toplevel_get_state (GDK_TOPLEVEL (window)) & GDK_TOPLEVEL_STATE_MINIMIZED)
|
||
{
|
||
if (gdk_toplevel_get_state (GDK_TOPLEVEL (window)) & GDK_TOPLEVEL_STATE_MAXIMIZED)
|
||
{
|
||
GtkShowWindow (window, SW_SHOWMAXIMIZED);
|
||
}
|
||
else
|
||
{
|
||
GtkShowWindow (window, SW_RESTORE);
|
||
}
|
||
}
|
||
}
|
||
else
|
||
{
|
||
GtkShowWindow (window, SW_MINIMIZE);
|
||
}
|
||
}
|
||
|
||
impl->changing_state = FALSE;
|
||
}
|
||
}
|
||
|
||
static void
|
||
do_show_window (GdkSurface *window, gboolean hide_window)
|
||
{
|
||
GdkSurface *tmp_window = NULL;
|
||
GdkWin32Surface *tmp_impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (!tmp_impl->changing_state)
|
||
{
|
||
/* Find the top-level window in our transient chain. */
|
||
while (tmp_impl->transient_owner != NULL)
|
||
{
|
||
tmp_window = tmp_impl->transient_owner;
|
||
tmp_impl = GDK_WIN32_SURFACE (tmp_window);
|
||
}
|
||
|
||
/* If we couldn't find one, use the window provided. */
|
||
if (tmp_window == NULL)
|
||
{
|
||
tmp_window = window;
|
||
}
|
||
|
||
/* Recursively show/hide every window in the chain. */
|
||
if (tmp_window != window)
|
||
{
|
||
show_window_recurse (tmp_window, hide_window);
|
||
}
|
||
}
|
||
}
|
||
|
||
static void
|
||
send_crossing_event (GdkDisplay *display,
|
||
GdkDevice *physical_device,
|
||
GdkSurface *window,
|
||
GdkEventType type,
|
||
GdkCrossingMode mode,
|
||
GdkNotifyType notify_type,
|
||
GdkSurface *subwindow,
|
||
POINT *screen_pt,
|
||
GdkModifierType mask,
|
||
guint32 time_)
|
||
{
|
||
GdkEvent *event;
|
||
GdkDeviceGrabInfo *grab;
|
||
GdkDeviceManagerWin32 *device_manager;
|
||
POINT pt;
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
|
||
device_manager = _gdk_device_manager;
|
||
|
||
grab = _gdk_display_has_device_grab (display, device_manager->core_pointer, 0);
|
||
|
||
if (grab != NULL &&
|
||
!grab->owner_events &&
|
||
mode != GDK_CROSSING_UNGRAB)
|
||
{
|
||
/* !owner_event => only report events wrt grab window, ignore rest */
|
||
if ((GdkSurface *)window != grab->surface)
|
||
return;
|
||
}
|
||
|
||
pt = *screen_pt;
|
||
ScreenToClient (GDK_SURFACE_HWND (window), &pt);
|
||
|
||
_gdk_device_virtual_set_active (_gdk_device_manager->core_pointer, physical_device);
|
||
|
||
event = gdk_crossing_event_new (type,
|
||
window,
|
||
device_manager->core_pointer,
|
||
time_,
|
||
mask,
|
||
pt.x / impl->surface_scale,
|
||
pt.y / impl->surface_scale,
|
||
mode,
|
||
notify_type);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
static GdkSurface *
|
||
find_common_ancestor (GdkSurface *win1,
|
||
GdkSurface *win2)
|
||
{
|
||
GdkSurface *tmp;
|
||
GList *path1 = NULL, *path2 = NULL;
|
||
GList *list1, *list2;
|
||
|
||
tmp = win1;
|
||
while (tmp != NULL)
|
||
{
|
||
path1 = g_list_prepend (path1, tmp);
|
||
tmp = tmp->parent;
|
||
}
|
||
|
||
tmp = win2;
|
||
while (tmp != NULL)
|
||
{
|
||
path2 = g_list_prepend (path2, tmp);
|
||
tmp = tmp->parent;
|
||
}
|
||
|
||
list1 = path1;
|
||
list2 = path2;
|
||
tmp = NULL;
|
||
while (list1 && list2 && (list1->data == list2->data))
|
||
{
|
||
tmp = (GdkSurface *)list1->data;
|
||
list1 = list1->next;
|
||
list2 = list2->next;
|
||
}
|
||
g_list_free (path1);
|
||
g_list_free (path2);
|
||
|
||
return tmp;
|
||
}
|
||
|
||
void
|
||
synthesize_crossing_events (GdkDisplay *display,
|
||
GdkDevice *physical_device,
|
||
GdkSurface *src,
|
||
GdkSurface *dest,
|
||
GdkCrossingMode mode,
|
||
POINT *screen_pt,
|
||
GdkModifierType mask,
|
||
guint32 time_,
|
||
gboolean non_linear)
|
||
{
|
||
GdkSurface *c;
|
||
GdkSurface *win, *last, *next;
|
||
GList *path, *list;
|
||
GdkSurface *a;
|
||
GdkSurface *b;
|
||
GdkNotifyType notify_type;
|
||
|
||
a = src;
|
||
b = dest;
|
||
if (a == b)
|
||
return; /* No crossings generated between src and dest */
|
||
|
||
c = find_common_ancestor (a, b);
|
||
|
||
non_linear |= (c != a) && (c != b);
|
||
|
||
if (a) /* There might not be a source (i.e. if no previous pointer_in_window) */
|
||
{
|
||
/* Traverse up from a to (excluding) c sending leave events */
|
||
if (non_linear)
|
||
notify_type = GDK_NOTIFY_NONLINEAR;
|
||
else if (c == a)
|
||
notify_type = GDK_NOTIFY_INFERIOR;
|
||
else
|
||
notify_type = GDK_NOTIFY_ANCESTOR;
|
||
send_crossing_event (display,
|
||
physical_device,
|
||
a, GDK_LEAVE_NOTIFY,
|
||
mode,
|
||
notify_type,
|
||
NULL,
|
||
screen_pt,
|
||
mask, time_);
|
||
|
||
if (c != a)
|
||
{
|
||
if (non_linear)
|
||
notify_type = GDK_NOTIFY_NONLINEAR_VIRTUAL;
|
||
else
|
||
notify_type = GDK_NOTIFY_VIRTUAL;
|
||
|
||
last = a;
|
||
win = a->parent;
|
||
while (win != c && win != NULL)
|
||
{
|
||
send_crossing_event (display,
|
||
physical_device,
|
||
win, GDK_LEAVE_NOTIFY,
|
||
mode,
|
||
notify_type,
|
||
(GdkSurface *)last,
|
||
screen_pt,
|
||
mask, time_);
|
||
|
||
last = win;
|
||
win = win->parent;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (b) /* Might not be a dest, e.g. if we're moving out of the window */
|
||
{
|
||
/* Traverse down from c to b */
|
||
if (c != b)
|
||
{
|
||
path = NULL;
|
||
win = b->parent;
|
||
while (win != c && win != NULL)
|
||
{
|
||
path = g_list_prepend (path, win);
|
||
win = win->parent;
|
||
}
|
||
|
||
if (non_linear)
|
||
notify_type = GDK_NOTIFY_NONLINEAR_VIRTUAL;
|
||
else
|
||
notify_type = GDK_NOTIFY_VIRTUAL;
|
||
|
||
list = path;
|
||
while (list)
|
||
{
|
||
win = (GdkSurface *)list->data;
|
||
list = list->next;
|
||
if (list)
|
||
next = (GdkSurface *)list->data;
|
||
else
|
||
next = b;
|
||
|
||
send_crossing_event (display,
|
||
physical_device,
|
||
win, GDK_ENTER_NOTIFY,
|
||
mode,
|
||
notify_type,
|
||
next,
|
||
screen_pt,
|
||
mask, time_);
|
||
}
|
||
g_list_free (path);
|
||
}
|
||
|
||
|
||
if (non_linear)
|
||
notify_type = GDK_NOTIFY_NONLINEAR;
|
||
else if (c == a)
|
||
notify_type = GDK_NOTIFY_ANCESTOR;
|
||
else
|
||
notify_type = GDK_NOTIFY_INFERIOR;
|
||
|
||
send_crossing_event (display,
|
||
physical_device,
|
||
b, GDK_ENTER_NOTIFY,
|
||
mode,
|
||
notify_type,
|
||
NULL,
|
||
screen_pt,
|
||
mask, time_);
|
||
}
|
||
}
|
||
|
||
static void
|
||
make_crossing_event (GdkDevice *physical_device,
|
||
GdkSurface *surface,
|
||
POINT *screen_pt,
|
||
guint32 time_)
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" mouse_window %p -> %p",
|
||
mouse_window ? GDK_SURFACE_HWND (mouse_window) : NULL,
|
||
surface ? GDK_SURFACE_HWND (surface) : NULL));
|
||
synthesize_crossing_events (_gdk_display,
|
||
physical_device,
|
||
mouse_window, surface,
|
||
GDK_CROSSING_NORMAL,
|
||
screen_pt,
|
||
0, /* TODO: Set right mask */
|
||
time_,
|
||
FALSE);
|
||
g_set_object (&mouse_window, surface);
|
||
}
|
||
|
||
/* Acquires actual client area size of the underlying native window.
|
||
* Returns FALSE if configure events should be inhibited,
|
||
* TRUE otherwise.
|
||
*/
|
||
gboolean
|
||
_gdk_win32_get_window_rect (GdkSurface *window,
|
||
RECT *rect)
|
||
{
|
||
RECT client_rect;
|
||
POINT point;
|
||
HWND hwnd;
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
|
||
hwnd = GDK_SURFACE_HWND (window);
|
||
|
||
GetClientRect (hwnd, &client_rect);
|
||
point.x = client_rect.left; /* always 0 */
|
||
point.y = client_rect.top;
|
||
|
||
/* top level windows need screen coords */
|
||
if (GDK_IS_TOPLEVEL (window))
|
||
ClientToScreen (hwnd, &point);
|
||
|
||
rect->left = point.x;
|
||
rect->top = point.y;
|
||
rect->right = point.x + client_rect.right - client_rect.left;
|
||
rect->bottom = point.y + client_rect.bottom - client_rect.top;
|
||
|
||
return !impl->inhibit_configure;
|
||
}
|
||
|
||
cairo_region_t *
|
||
_gdk_win32_hrgn_to_region (HRGN hrgn,
|
||
guint scale)
|
||
{
|
||
RGNDATA *rgndata;
|
||
RECT *rects;
|
||
cairo_region_t *result;
|
||
int nbytes;
|
||
guint i;
|
||
|
||
if ((nbytes = GetRegionData (hrgn, 0, NULL)) == 0)
|
||
{
|
||
WIN32_GDI_FAILED ("GetRegionData");
|
||
return NULL;
|
||
}
|
||
|
||
rgndata = (RGNDATA *) g_malloc (nbytes);
|
||
|
||
if (GetRegionData (hrgn, nbytes, rgndata) == 0)
|
||
{
|
||
WIN32_GDI_FAILED ("GetRegionData");
|
||
g_free (rgndata);
|
||
return NULL;
|
||
}
|
||
|
||
result = cairo_region_create ();
|
||
rects = (RECT *) rgndata->Buffer;
|
||
for (i = 0; i < rgndata->rdh.nCount; i++)
|
||
{
|
||
GdkRectangle r;
|
||
|
||
r.x = rects[i].left;
|
||
r.y = rects[i].top;
|
||
r.width = (rects[i].right - r.x) / scale;
|
||
r.height = (rects[i].bottom - r.y) / scale;
|
||
|
||
cairo_region_union_rectangle (result, &r);
|
||
}
|
||
|
||
g_free (rgndata);
|
||
|
||
return result;
|
||
}
|
||
|
||
static void
|
||
handle_wm_paint (MSG *msg,
|
||
GdkSurface *window)
|
||
{
|
||
HRGN hrgn = CreateRectRgn (0, 0, 0, 0);
|
||
HDC hdc;
|
||
PAINTSTRUCT paintstruct;
|
||
cairo_region_t *update_region;
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (GetUpdateRgn (msg->hwnd, hrgn, FALSE) == ERROR)
|
||
{
|
||
WIN32_GDI_FAILED ("GetUpdateRgn");
|
||
DeleteObject (hrgn);
|
||
return;
|
||
}
|
||
|
||
hdc = BeginPaint (msg->hwnd, &paintstruct);
|
||
|
||
GDK_NOTE (EVENTS, g_print (" %s %s dc %p",
|
||
_gdk_win32_rect_to_string (&paintstruct.rcPaint),
|
||
(paintstruct.fErase ? "erase" : ""),
|
||
hdc));
|
||
|
||
EndPaint (msg->hwnd, &paintstruct);
|
||
|
||
if ((paintstruct.rcPaint.right == paintstruct.rcPaint.left) ||
|
||
(paintstruct.rcPaint.bottom == paintstruct.rcPaint.top))
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" (empty paintstruct, ignored)"));
|
||
DeleteObject (hrgn);
|
||
return;
|
||
}
|
||
|
||
update_region = _gdk_win32_hrgn_to_region (hrgn, impl->surface_scale);
|
||
if (!cairo_region_is_empty (update_region))
|
||
gdk_surface_invalidate_region (window, update_region);
|
||
cairo_region_destroy (update_region);
|
||
|
||
DeleteObject (hrgn);
|
||
}
|
||
|
||
static VOID CALLBACK
|
||
modal_timer_proc (HWND hwnd,
|
||
UINT msg,
|
||
UINT_PTR id,
|
||
DWORD time)
|
||
{
|
||
int arbitrary_limit = 10;
|
||
|
||
while (_modal_operation_in_progress != GDK_WIN32_MODAL_OP_NONE &&
|
||
g_main_context_pending (NULL) &&
|
||
arbitrary_limit--)
|
||
g_main_context_iteration (NULL, FALSE);
|
||
}
|
||
|
||
void
|
||
_gdk_win32_begin_modal_call (GdkWin32ModalOpKind kind)
|
||
{
|
||
GdkWin32ModalOpKind was = _modal_operation_in_progress;
|
||
g_assert (!(_modal_operation_in_progress & kind));
|
||
|
||
_modal_operation_in_progress |= kind;
|
||
|
||
if (was == GDK_WIN32_MODAL_OP_NONE)
|
||
{
|
||
modal_timer = SetTimer (NULL, 0, 10, modal_timer_proc);
|
||
|
||
if (modal_timer == 0)
|
||
WIN32_API_FAILED ("SetTimer");
|
||
}
|
||
}
|
||
|
||
void
|
||
_gdk_win32_end_modal_call (GdkWin32ModalOpKind kind)
|
||
{
|
||
g_assert (_modal_operation_in_progress & kind);
|
||
|
||
_modal_operation_in_progress &= ~kind;
|
||
|
||
if (_modal_operation_in_progress == GDK_WIN32_MODAL_OP_NONE &&
|
||
modal_timer != 0)
|
||
{
|
||
API_CALL (KillTimer, (NULL, modal_timer));
|
||
modal_timer = 0;
|
||
}
|
||
}
|
||
|
||
static gboolean
|
||
handle_nchittest (HWND hwnd,
|
||
GdkSurface *window,
|
||
gint16 screen_x,
|
||
gint16 screen_y,
|
||
int *ret_valp)
|
||
{
|
||
GdkWin32Surface *impl;
|
||
RECT client_rect;
|
||
POINT client_pt;
|
||
|
||
if (window == NULL)
|
||
return FALSE;
|
||
|
||
/* If the window has no particular input pass-through region,
|
||
* then we can simply let DefWindowProc() handle the message */
|
||
if (window->input_region == NULL)
|
||
return FALSE;
|
||
|
||
if (!GetClientRect (hwnd, &client_rect))
|
||
return FALSE;
|
||
|
||
client_pt.x = screen_x;
|
||
client_pt.y = screen_y;
|
||
if (!ScreenToClient (hwnd, &client_pt))
|
||
return FALSE;
|
||
|
||
/* Check whether the point lies within the client area */
|
||
if (!PtInRect (&client_rect, client_pt))
|
||
return FALSE;
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
/* If the point lies inside the input region, return HTCLIENT,
|
||
* otherwise return HTTRANSPARENT. */
|
||
if (cairo_region_contains_point (window->input_region,
|
||
client_pt.x / impl->surface_scale,
|
||
client_pt.y / impl->surface_scale))
|
||
*ret_valp = HTCLIENT;
|
||
else
|
||
*ret_valp = HTTRANSPARENT;
|
||
|
||
/* We handled the message, no need to call DefWindowProc() */
|
||
return TRUE;
|
||
}
|
||
|
||
static void
|
||
handle_dpi_changed (GdkSurface *window,
|
||
MSG *msg)
|
||
{
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
GdkDisplay *display = gdk_display_get_default ();
|
||
GdkWin32Display *win32_display = GDK_WIN32_DISPLAY (display);
|
||
RECT *rect = (RECT *)msg->lParam;
|
||
guint old_scale = impl->surface_scale;
|
||
|
||
/* MSDN for WM_DPICHANGED: dpi_x == dpi_y here, so LOWORD (msg->wParam) == HIWORD (msg->wParam) */
|
||
guint dpi = LOWORD (msg->wParam);
|
||
|
||
/* Don't bother if we use a fixed scale */
|
||
if (win32_display->has_fixed_scale)
|
||
return;
|
||
|
||
impl->surface_scale = dpi / USER_DEFAULT_SCREEN_DPI;
|
||
|
||
/* Don't bother if scales did not change in the end */
|
||
if (old_scale == impl->surface_scale)
|
||
return;
|
||
|
||
if (!IsIconic (msg->hwnd) &&
|
||
!GDK_SURFACE_DESTROYED (window))
|
||
{
|
||
GdkMonitor *monitor;
|
||
|
||
monitor = gdk_display_get_monitor_at_surface (display, window);
|
||
gdk_monitor_set_scale_factor (monitor, impl->surface_scale);
|
||
}
|
||
|
||
_gdk_win32_adjust_client_rect (window, rect);
|
||
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE)
|
||
gdk_win32_surface_move_resize (window,
|
||
window->x, window->y,
|
||
window->width, window->height);
|
||
else
|
||
gdk_win32_surface_resize (window, window->width, window->height);
|
||
}
|
||
|
||
static void
|
||
generate_button_event (GdkEventType type,
|
||
int button,
|
||
GdkSurface *window,
|
||
MSG *msg)
|
||
{
|
||
GdkEvent *event;
|
||
GdkDeviceManagerWin32 *device_manager;
|
||
GdkWin32Surface *impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (_gdk_input_ignore_core > 0)
|
||
return;
|
||
|
||
device_manager = GDK_DEVICE_MANAGER_WIN32 (_gdk_device_manager);
|
||
|
||
current_x = (gint16) GET_X_LPARAM (msg->lParam) / impl->surface_scale;
|
||
current_y = (gint16) GET_Y_LPARAM (msg->lParam) / impl->surface_scale;
|
||
|
||
_gdk_device_virtual_set_active (_gdk_device_manager->core_pointer,
|
||
_gdk_device_manager->system_pointer);
|
||
|
||
event = gdk_button_event_new (type,
|
||
window,
|
||
device_manager->core_pointer,
|
||
NULL,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
build_pointer_event_state (msg),
|
||
button,
|
||
current_x,
|
||
current_y,
|
||
NULL);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
static gboolean
|
||
handle_wm_sysmenu (GdkSurface *window, MSG *msg, int *ret_valp)
|
||
{
|
||
GdkWin32Surface *impl;
|
||
LONG_PTR style, tmp_style;
|
||
LONG_PTR additional_styles;
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
style = GetWindowLongPtr (msg->hwnd, GWL_STYLE);
|
||
|
||
additional_styles = 0;
|
||
|
||
if (!(style & WS_SYSMENU))
|
||
additional_styles |= WS_SYSMENU;
|
||
|
||
if (!(style & WS_MAXIMIZEBOX))
|
||
additional_styles |= WS_MAXIMIZEBOX;
|
||
|
||
if (!(style & WS_MINIMIZEBOX))
|
||
additional_styles |= WS_MINIMIZEBOX;
|
||
|
||
if (!(style & WS_SIZEBOX))
|
||
additional_styles |= WS_SIZEBOX;
|
||
|
||
if (!(style & WS_DLGFRAME))
|
||
additional_styles |= WS_DLGFRAME;
|
||
|
||
if (!(style & WS_BORDER))
|
||
additional_styles |= WS_BORDER;
|
||
|
||
if (additional_styles == 0)
|
||
/* The caller will eventually pass this to DefWindowProc (),
|
||
* only without the style dance, which isn't needed, as it turns out.
|
||
*/
|
||
return FALSE;
|
||
|
||
/* Note: This code will enable resizing, maximizing and minimizing windows
|
||
* via window menu even if these are non-CSD windows that were explicitly
|
||
* forbidden from doing this by removing the appropriate styles,
|
||
* or if these are CSD windows that were explicitly forbidden from doing
|
||
* this by removing appropriate decorations from the headerbar and/or
|
||
* changing hints or properties.
|
||
*
|
||
* If doing this for non-CSD windows is not desired,
|
||
* do a _gdk_win32_surface_lacks_wm_decorations() check and return FALSE
|
||
* if it doesn't pass.
|
||
*
|
||
* If doing this for CSD windows with disabled decorations is not desired,
|
||
* tough luck - GDK can't know which CSD decorations are enabled, and which
|
||
* are not.
|
||
*
|
||
* If doing this for CSD windows with particular hints is not desired,
|
||
* check window hints here and return FALSE (DefWindowProc() will return
|
||
* FALSE later) or set *ret_valp to 0 and return TRUE.
|
||
*/
|
||
tmp_style = style | additional_styles;
|
||
GDK_NOTE (EVENTS, g_print (" Handling WM_SYSMENU: style 0x%" G_GINTPTR_MODIFIER "x -> 0x%" G_GINTPTR_MODIFIER "x\n", style, tmp_style));
|
||
impl->have_temp_styles = TRUE;
|
||
impl->temp_styles = additional_styles;
|
||
SetWindowLongPtr (msg->hwnd, GWL_STYLE, tmp_style);
|
||
|
||
*ret_valp = DefWindowProc (msg->hwnd, msg->message, msg->wParam, msg->lParam);
|
||
|
||
tmp_style = GetWindowLongPtr (msg->hwnd, GWL_STYLE);
|
||
style = tmp_style & ~additional_styles;
|
||
|
||
GDK_NOTE (EVENTS, g_print (" Handling WM_SYSMENU: style 0x%" G_GINTPTR_MODIFIER "x <- 0x%" G_GINTPTR_MODIFIER "x\n", style, tmp_style));
|
||
SetWindowLongPtr (msg->hwnd, GWL_STYLE, style);
|
||
impl->have_temp_styles = FALSE;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
gboolean
|
||
_gdk_win32_surface_fill_min_max_info (GdkSurface *window,
|
||
MINMAXINFO *mmi)
|
||
{
|
||
GdkWin32Surface *impl;
|
||
RECT rect;
|
||
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
return FALSE;
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->hint_flags & GDK_HINT_MIN_SIZE)
|
||
{
|
||
rect.left = rect.top = 0;
|
||
rect.right = impl->hints.min_width * impl->surface_scale;
|
||
rect.bottom = impl->hints.min_height * impl->surface_scale;
|
||
|
||
_gdk_win32_adjust_client_rect (window, &rect);
|
||
|
||
mmi->ptMinTrackSize.x = rect.right - rect.left;
|
||
mmi->ptMinTrackSize.y = rect.bottom - rect.top;
|
||
}
|
||
|
||
if (impl->hint_flags & GDK_HINT_MAX_SIZE)
|
||
{
|
||
int maxw, maxh;
|
||
|
||
rect.left = rect.top = 0;
|
||
rect.right = impl->hints.max_width * impl->surface_scale;
|
||
rect.bottom = impl->hints.max_height * impl->surface_scale;
|
||
|
||
_gdk_win32_adjust_client_rect (window, &rect);
|
||
|
||
/* at least on win9x we have the 16 bit trouble */
|
||
maxw = rect.right - rect.left;
|
||
maxh = rect.bottom - rect.top;
|
||
mmi->ptMaxTrackSize.x = maxw > 0 && maxw < G_MAXSHORT ? maxw : G_MAXSHORT;
|
||
mmi->ptMaxTrackSize.y = maxh > 0 && maxh < G_MAXSHORT ? maxh : G_MAXSHORT;
|
||
}
|
||
else
|
||
{
|
||
/* According to "How does the window manager adjust ptMaxSize and
|
||
* ptMaxPosition for multiple monitors?" article
|
||
* https://blogs.msdn.microsoft.com/oldnewthing/20150501-00/?p=44964
|
||
* if ptMaxSize >= primary_monitor_size, then it will be adjusted by
|
||
* WM to account for the monitor size differences if the window gets
|
||
* maximized on a non-primary monitor, by simply adding the size
|
||
* difference (i.e. if non-primary monitor is larger by 100px, then
|
||
* window will be made larger exactly by 100px).
|
||
* If ptMaxSize < primary_monitor_size at least in one direction,
|
||
* nothing is adjusted.
|
||
* Therefore, if primary monitor is smaller than the actual monitor,
|
||
* then it is not possible to give window a size that is larger than
|
||
* the primary monitor and smaller than the non-primary monitor,
|
||
* because WM will always enlarge the window.
|
||
* Therefore, it is impossible to account for taskbar size.
|
||
* So we don't try at all. Instead we just remember that we're trying
|
||
* to maximize the window, catch WM_WINDOWPOSCHANGING and
|
||
* adjust the size then.
|
||
*/
|
||
HMONITOR nearest_monitor;
|
||
MONITORINFO nearest_info;
|
||
|
||
nearest_monitor = MonitorFromWindow (GDK_SURFACE_HWND (window), MONITOR_DEFAULTTONEAREST);
|
||
nearest_info.cbSize = sizeof (nearest_info);
|
||
|
||
if (GetMonitorInfoA (nearest_monitor, &nearest_info))
|
||
{
|
||
/* MSDN says that we must specify maximized window
|
||
* size as if it was located on the primary monitor.
|
||
* However, we still need to account for a taskbar
|
||
* that might or might not be on the nearest monitor where
|
||
* window will actually end up.
|
||
* "0" here is the top-left corner of the primary monitor.
|
||
*/
|
||
/* An investigation into bug 765161 turned up a weird Windows WM behaviour
|
||
* where it would interpret "0:0" as "top-left of the workea" (accounting for a taskbar
|
||
* possibly being along the left/top edge of the screen) when window has styles
|
||
* (i.e. not CSD), and interpret the same "0:0" as "top-left of the screen" (not
|
||
* accounting for a taskbar) when window has no styles (i.e. a CSD window).
|
||
* This doesn't seem to be documented anywhere.
|
||
* The following code uses a simple CSD/non-CSD test, but it could be that
|
||
* this behaviour hinges on just one particular window style.
|
||
* Finding exactly which style that could be is not very useful for GTK, however.
|
||
*/
|
||
mmi->ptMaxPosition.x = 0;
|
||
mmi->ptMaxPosition.y = 0;
|
||
|
||
if (_gdk_win32_surface_lacks_wm_decorations (window))
|
||
{
|
||
mmi->ptMaxPosition.x += (nearest_info.rcWork.left - nearest_info.rcMonitor.left);
|
||
mmi->ptMaxPosition.y += (nearest_info.rcWork.top - nearest_info.rcMonitor.top);
|
||
}
|
||
|
||
mmi->ptMaxSize.x = nearest_info.rcWork.right - nearest_info.rcWork.left;
|
||
mmi->ptMaxSize.y = nearest_info.rcWork.bottom - nearest_info.rcWork.top;
|
||
}
|
||
|
||
mmi->ptMaxTrackSize.x = GetSystemMetrics (SM_CXVIRTUALSCREEN) + impl->shadow_x * impl->surface_scale;
|
||
mmi->ptMaxTrackSize.y = GetSystemMetrics (SM_CYVIRTUALSCREEN) + impl->shadow_y * impl->surface_scale;
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
#define GDK_ANY_BUTTON_MASK (GDK_BUTTON1_MASK | \
|
||
GDK_BUTTON2_MASK | \
|
||
GDK_BUTTON3_MASK | \
|
||
GDK_BUTTON4_MASK | \
|
||
GDK_BUTTON5_MASK)
|
||
|
||
static gboolean
|
||
gdk_event_translate (MSG *msg,
|
||
int *ret_valp)
|
||
{
|
||
RECT rect;
|
||
POINT point;
|
||
MINMAXINFO *mmi;
|
||
HWND hwnd;
|
||
HIMC himc;
|
||
WINDOWPOS *windowpos;
|
||
gboolean ignore_leave;
|
||
|
||
GdkEvent *event;
|
||
|
||
GdkDisplay *display;
|
||
GdkSurface *window = NULL;
|
||
GdkWin32Surface *impl;
|
||
GdkWin32Display *win32_display;
|
||
|
||
GdkSurface *new_window;
|
||
|
||
GdkDeviceManagerWin32 *device_manager_win32;
|
||
|
||
GdkDeviceGrabInfo *keyboard_grab = NULL;
|
||
GdkDeviceGrabInfo *pointer_grab = NULL;
|
||
GdkSurface *grab_window = NULL;
|
||
|
||
crossing_cb_t crossing_cb = NULL;
|
||
|
||
int button;
|
||
|
||
gboolean return_val = FALSE;
|
||
|
||
int i;
|
||
|
||
display = gdk_display_get_default ();
|
||
win32_display = GDK_WIN32_DISPLAY (display);
|
||
|
||
if (win32_display->filters)
|
||
{
|
||
/* Apply display filters */
|
||
GdkWin32MessageFilterReturn result = apply_message_filters (display, msg, ret_valp, &win32_display->filters);
|
||
|
||
if (result == GDK_WIN32_MESSAGE_FILTER_REMOVE)
|
||
return TRUE;
|
||
}
|
||
|
||
window = gdk_win32_handle_table_lookup_ (msg->hwnd);
|
||
|
||
if (window == NULL)
|
||
{
|
||
/* XXX Handle WM_QUIT here ? */
|
||
if (msg->message == WM_QUIT)
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" %d", (int) msg->wParam));
|
||
exit (msg->wParam);
|
||
}
|
||
else if (msg->message == WM_CREATE)
|
||
{
|
||
window = (UNALIGNED GdkSurface*) (((LPCREATESTRUCTW) msg->lParam)->lpCreateParams);
|
||
GDK_SURFACE_HWND (window) = msg->hwnd;
|
||
}
|
||
else
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" (no GdkSurface)"));
|
||
}
|
||
return FALSE;
|
||
}
|
||
|
||
device_manager_win32 = GDK_DEVICE_MANAGER_WIN32 (_gdk_device_manager);
|
||
|
||
keyboard_grab = _gdk_display_get_last_device_grab (display,
|
||
device_manager_win32->core_keyboard);
|
||
pointer_grab = _gdk_display_get_last_device_grab (display,
|
||
device_manager_win32->core_pointer);
|
||
|
||
g_object_ref (window);
|
||
|
||
/* window's refcount has now been increased, so code below should
|
||
* not just return from this function, but instead goto done (or
|
||
* break out of the big switch). To protect against forgetting this,
|
||
* #define return to a syntax error...
|
||
*/
|
||
#define return GOTO_DONE_INSTEAD
|
||
|
||
if (msg->message == aerosnap_message)
|
||
_gdk_win32_surface_handle_aerosnap (window,
|
||
(GdkWin32AeroSnapCombo) msg->wParam);
|
||
|
||
switch (msg->message)
|
||
{
|
||
case WM_INPUTLANGCHANGE:
|
||
{
|
||
GdkWin32Keymap *win32_keymap;
|
||
GdkTranslatedKey translated;
|
||
|
||
win32_keymap = GDK_WIN32_KEYMAP (_gdk_win32_display_get_keymap (_gdk_display));
|
||
|
||
_gdk_input_locale = (HKL) msg->lParam;
|
||
_gdk_win32_keymap_set_active_layout (win32_keymap, _gdk_input_locale);
|
||
_gdk_keymap_serial++;
|
||
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" cs:%lu hkl:%p%s",
|
||
(gulong) msg->wParam,
|
||
(gpointer) msg->lParam,
|
||
_gdk_input_locale_is_ime ? " (IME)" : ""));
|
||
gdk_display_setting_changed (display, "gtk-im-module");
|
||
|
||
/* Generate a dummy key event to "nudge" IMContext */
|
||
translated.keyval = GDK_KEY_VoidSymbol;
|
||
translated.consumed = 0;
|
||
translated.layout = 0;
|
||
translated.level = 0;
|
||
event = gdk_key_event_new (GDK_KEY_PRESS,
|
||
window,
|
||
device_manager_win32->core_keyboard,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
0,
|
||
0,
|
||
FALSE,
|
||
&translated,
|
||
&translated,
|
||
NULL);
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
break;
|
||
|
||
case WM_SYSKEYUP:
|
||
case WM_SYSKEYDOWN:
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" %s ch:%.02x %s",
|
||
_gdk_win32_key_to_string (msg->lParam),
|
||
(int) msg->wParam,
|
||
decode_key_lparam (msg->lParam)));
|
||
|
||
/* If posted without us having keyboard focus, ignore */
|
||
if ((msg->wParam != VK_F10 && msg->wParam != VK_MENU) &&
|
||
!(HIWORD (msg->lParam) & KF_ALTDOWN))
|
||
break;
|
||
|
||
/* Let the system handle Alt-Tab, Alt-Space and Alt-F4 unless
|
||
* the keyboard is grabbed.
|
||
*/
|
||
if (!keyboard_grab &&
|
||
(msg->wParam == VK_TAB ||
|
||
msg->wParam == VK_SPACE ||
|
||
msg->wParam == VK_F4))
|
||
break;
|
||
|
||
/* Jump to code in common with WM_KEYUP and WM_KEYDOWN */
|
||
goto keyup_or_down;
|
||
|
||
case WM_KEYUP:
|
||
case WM_KEYDOWN:
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" %s ch:%.02x %s",
|
||
_gdk_win32_key_to_string (msg->lParam),
|
||
(int) msg->wParam,
|
||
decode_key_lparam (msg->lParam)));
|
||
|
||
keyup_or_down:
|
||
{
|
||
GdkWin32Keymap *win32_keymap;
|
||
GdkModifierType state;
|
||
guint keyval;
|
||
guint16 keycode;
|
||
guint8 group;
|
||
gboolean is_modifier;
|
||
GdkTranslatedKey translated;
|
||
GdkTranslatedKey no_lock;
|
||
BYTE key_state[256];
|
||
GArray *translation;
|
||
MSG msg2;
|
||
int level = 0;
|
||
int effective_group = 0;
|
||
GdkModifierType consumed = 0;
|
||
char *composed = NULL;
|
||
|
||
/* Ignore key messages intended for the IME */
|
||
if (msg->wParam == VK_PROCESSKEY || in_ime_composition)
|
||
break;
|
||
|
||
/* Ignore autorepeats on modifiers */
|
||
if (msg->message == WM_KEYDOWN &&
|
||
(msg->wParam == VK_MENU ||
|
||
msg->wParam == VK_CONTROL ||
|
||
msg->wParam == VK_SHIFT) &&
|
||
((HIWORD(msg->lParam) & KF_REPEAT) >= 1))
|
||
break;
|
||
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
win32_keymap = GDK_WIN32_KEYMAP (_gdk_win32_display_get_keymap (display));
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
API_CALL (GetKeyboardState, (key_state));
|
||
|
||
keyval = GDK_KEY_VoidSymbol;
|
||
keycode = msg->wParam;
|
||
|
||
/* Get the WinAPI translation of the WM_KEY messages to characters.
|
||
|
||
The WM_CHAR messages are generated by a previous call to TranslateMessage() and always
|
||
follow directly after the corresponding WM_KEY* messages.
|
||
There could be 0 or more WM_CHAR messages following (for example dead keys don't generate
|
||
WM_CHAR messages - they generate WM_DEAD_CHAR instead, but we are not interested in those
|
||
messages). */
|
||
|
||
translation = g_array_sized_new (FALSE, FALSE, sizeof (gunichar2), 2);
|
||
while (PeekMessageW (&msg2, msg->hwnd, 0, 0, 0) && (msg2.message == WM_CHAR || msg2.message == WM_SYSCHAR))
|
||
{
|
||
/* The character is encoded in WPARAM as UTF-16. */
|
||
gunichar2 c = msg2.wParam;
|
||
|
||
/* Append character to translation string. */
|
||
g_array_append_val (translation, c);
|
||
|
||
/* Remove message from queue */
|
||
GetMessageW (&msg2, msg->hwnd, 0, 0);
|
||
}
|
||
|
||
if (translation->len > 0)
|
||
composed = g_utf16_to_utf8 ((gunichar2*)translation->data,
|
||
translation->len, NULL, NULL, NULL);
|
||
|
||
g_array_unref (translation);
|
||
translation = NULL;
|
||
|
||
/* Ignore control sequences like Backspace */
|
||
if (composed && g_unichar_iscntrl (g_utf8_get_char (composed)))
|
||
g_clear_pointer (&composed, g_free);
|
||
|
||
if (HIWORD (msg->lParam) & KF_EXTENDED)
|
||
{
|
||
switch (msg->wParam)
|
||
{
|
||
case VK_CONTROL:
|
||
keycode = VK_RCONTROL;
|
||
break;
|
||
case VK_SHIFT: /* Actually, KF_EXTENDED is not set
|
||
* for the right shift key.
|
||
*/
|
||
keycode = VK_RSHIFT;
|
||
break;
|
||
case VK_MENU:
|
||
keycode = VK_RMENU;
|
||
break;
|
||
}
|
||
}
|
||
else if (msg->wParam == VK_SHIFT &&
|
||
LOBYTE (HIWORD (msg->lParam)) == _gdk_win32_keymap_get_rshift_scancode (win32_keymap))
|
||
keycode = VK_RSHIFT;
|
||
|
||
is_modifier = (msg->wParam == VK_CONTROL ||
|
||
msg->wParam == VK_SHIFT ||
|
||
msg->wParam == VK_MENU);
|
||
|
||
state = build_key_event_state (key_state);
|
||
group = get_active_group ();
|
||
|
||
gdk_keymap_translate_keyboard_state ((GdkKeymap*) win32_keymap, keycode, state, group,
|
||
&keyval, &effective_group, &level, &consumed);
|
||
translated.keyval = keyval;
|
||
translated.consumed = consumed;
|
||
translated.layout = effective_group;
|
||
translated.level = level;
|
||
|
||
gdk_keymap_translate_keyboard_state ((GdkKeymap*) win32_keymap, keycode,
|
||
state & ~GDK_LOCK_MASK, group, &keyval,
|
||
&effective_group, &level, &consumed);
|
||
no_lock.keyval = keyval;
|
||
no_lock.consumed = consumed;
|
||
no_lock.layout = effective_group;
|
||
no_lock.level = level;
|
||
|
||
/* Only one release key event is fired when both shift keys are pressed together
|
||
and then released. In order to send the missing event, press events for shift
|
||
keys are recorded and sent together when the release event occurs.
|
||
Other modifiers (e.g. ctrl, alt) don't have this problem. */
|
||
if (msg->message == WM_KEYDOWN && msg->wParam == VK_SHIFT)
|
||
{
|
||
int pressed_shift = msg->lParam & 0xffffff; /* mask shift modifier */
|
||
if (both_shift_pressed[0] == 0)
|
||
both_shift_pressed[0] = pressed_shift;
|
||
else if (both_shift_pressed[0] != pressed_shift)
|
||
both_shift_pressed[1] = pressed_shift;
|
||
}
|
||
|
||
if (msg->message == WM_KEYUP && msg->wParam == VK_SHIFT)
|
||
{
|
||
if (both_shift_pressed[0] != 0 && both_shift_pressed[1] != 0)
|
||
{
|
||
int tmp_retval;
|
||
MSG fake_release = *msg;
|
||
int pressed_shift = msg->lParam & 0xffffff;
|
||
|
||
if (both_shift_pressed[0] == pressed_shift)
|
||
fake_release.lParam = both_shift_pressed[1];
|
||
else
|
||
fake_release.lParam = both_shift_pressed[0];
|
||
|
||
both_shift_pressed[0] = both_shift_pressed[1] = 0;
|
||
gdk_event_translate (&fake_release, &tmp_retval);
|
||
}
|
||
both_shift_pressed[0] = both_shift_pressed[1] = 0;
|
||
}
|
||
|
||
/* Reset ALT_MASK if it is the Alt key itself */
|
||
if (msg->wParam == VK_MENU)
|
||
state &= ~GDK_ALT_MASK;
|
||
|
||
event = gdk_key_event_new ((msg->message == WM_KEYDOWN || msg->message == WM_SYSKEYDOWN)
|
||
? GDK_KEY_PRESS
|
||
: GDK_KEY_RELEASE,
|
||
window,
|
||
device_manager_win32->core_keyboard,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
keycode,
|
||
state,
|
||
is_modifier,
|
||
&translated,
|
||
&no_lock,
|
||
composed);
|
||
|
||
_gdk_win32_append_event (event);
|
||
|
||
g_free (composed);
|
||
return_val = TRUE;
|
||
}
|
||
break;
|
||
|
||
case WM_SYSCHAR:
|
||
if (msg->wParam != VK_SPACE)
|
||
{
|
||
/* To prevent beeps, don't let DefWindowProcW() be called */
|
||
return_val = TRUE;
|
||
goto done;
|
||
}
|
||
break;
|
||
|
||
case WM_IME_STARTCOMPOSITION:
|
||
in_ime_composition = TRUE;
|
||
break;
|
||
|
||
case WM_IME_ENDCOMPOSITION:
|
||
in_ime_composition = FALSE;
|
||
break;
|
||
|
||
case WM_IME_COMPOSITION:
|
||
{
|
||
BYTE key_state[256];
|
||
wchar_t wbuf[100];
|
||
int ccount = 0;
|
||
|
||
/* On Win2k WM_IME_CHAR doesn't work correctly for non-Unicode
|
||
* applications. Thus, handle WM_IME_COMPOSITION with
|
||
* GCS_RESULTSTR instead, fetch the Unicode chars from the IME
|
||
* with ImmGetCompositionStringW().
|
||
*
|
||
* See for instance
|
||
* http://groups.google.com/groups?selm=natX5.57%24g77.19788%40nntp2.onemain.com
|
||
* and
|
||
* http://groups.google.com/groups?selm=u2XfrXw5BHA.1628%40tkmsftngp02
|
||
* for comments by other people that seems to have the same
|
||
* experience. WM_IME_CHAR just gives question marks, apparently
|
||
* because of going through some conversion to the current code
|
||
* page.
|
||
*
|
||
* WM_IME_CHAR might work on NT4 or Win9x with ActiveIMM, but
|
||
* use WM_IME_COMPOSITION there, too, to simplify the code.
|
||
*/
|
||
GDK_NOTE (EVENTS, g_print (" %#lx", (long) msg->lParam));
|
||
|
||
if (!(msg->lParam & GCS_RESULTSTR))
|
||
break;
|
||
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
himc = ImmGetContext (msg->hwnd);
|
||
ccount = ImmGetCompositionStringW (himc, GCS_RESULTSTR,
|
||
wbuf, sizeof (wbuf));
|
||
ImmReleaseContext (msg->hwnd, himc);
|
||
|
||
ccount /= 2;
|
||
|
||
API_CALL (GetKeyboardState, (key_state));
|
||
|
||
for (i = 0; i < ccount; i++)
|
||
{
|
||
GdkTranslatedKey translated;
|
||
|
||
/* Build a key press event */
|
||
translated.keyval = gdk_unicode_to_keyval (wbuf[i]);
|
||
translated.consumed = 0;
|
||
translated.layout = get_active_group ();
|
||
translated.level = 0;
|
||
event = gdk_key_event_new (GDK_KEY_PRESS,
|
||
window,
|
||
device_manager_win32->core_keyboard,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
0,
|
||
build_key_event_state (key_state),
|
||
FALSE,
|
||
&translated,
|
||
&translated,
|
||
NULL);
|
||
|
||
_gdk_win32_append_event (event);
|
||
|
||
/* Build a key release event. */
|
||
event = gdk_key_event_new (GDK_KEY_RELEASE,
|
||
window,
|
||
device_manager_win32->core_keyboard,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
0,
|
||
build_key_event_state (key_state),
|
||
FALSE,
|
||
&translated,
|
||
&translated,
|
||
NULL);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
return_val = TRUE;
|
||
}
|
||
break;
|
||
|
||
case WM_LBUTTONDOWN:
|
||
button = 1;
|
||
goto buttondown0;
|
||
|
||
case WM_MBUTTONDOWN:
|
||
button = 2;
|
||
goto buttondown0;
|
||
|
||
case WM_RBUTTONDOWN:
|
||
button = 3;
|
||
goto buttondown0;
|
||
|
||
case WM_XBUTTONDOWN:
|
||
if (HIWORD (msg->wParam) == XBUTTON1)
|
||
button = 4;
|
||
else
|
||
button = 5;
|
||
|
||
buttondown0:
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" (%d,%d)",
|
||
GET_X_LPARAM (msg->lParam), GET_Y_LPARAM (msg->lParam)));
|
||
|
||
pen_touch_input = FALSE;
|
||
|
||
g_set_object (&window, find_window_for_mouse_event (window, msg));
|
||
/* TODO_CSW?: there used to some synthesize and propagate */
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
if (pointer_grab == NULL)
|
||
{
|
||
SetCapture (GDK_SURFACE_HWND (window));
|
||
}
|
||
|
||
generate_button_event (GDK_BUTTON_PRESS, button,
|
||
window, msg);
|
||
|
||
*ret_valp = (msg->message == WM_XBUTTONDOWN ? TRUE : 0);
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_LBUTTONUP:
|
||
button = 1;
|
||
goto buttonup0;
|
||
|
||
case WM_MBUTTONUP:
|
||
button = 2;
|
||
goto buttonup0;
|
||
|
||
case WM_RBUTTONUP:
|
||
button = 3;
|
||
goto buttonup0;
|
||
|
||
case WM_XBUTTONUP:
|
||
if (HIWORD (msg->wParam) == XBUTTON1)
|
||
button = 4;
|
||
else
|
||
button = 5;
|
||
|
||
buttonup0:
|
||
{
|
||
gboolean release_implicit_grab = FALSE;
|
||
GdkSurface *prev_surface = NULL;
|
||
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" (%d,%d)",
|
||
GET_X_LPARAM (msg->lParam), GET_Y_LPARAM (msg->lParam)));
|
||
|
||
pen_touch_input = FALSE;
|
||
|
||
g_set_object (&window, find_window_for_mouse_event (window, msg));
|
||
|
||
if (pointer_grab != NULL && pointer_grab->implicit)
|
||
{
|
||
int state = build_pointer_event_state (msg);
|
||
|
||
/* We keep the implicit grab until no buttons at all are held down */
|
||
if ((state & GDK_ANY_BUTTON_MASK & ~(GDK_BUTTON1_MASK << (button - 1))) == 0)
|
||
{
|
||
release_implicit_grab = TRUE;
|
||
prev_surface = pointer_grab->surface;
|
||
}
|
||
}
|
||
|
||
generate_button_event (GDK_BUTTON_RELEASE, button, window, msg);
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
/* End a drag op when the same button that started it is released */
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE &&
|
||
impl->drag_move_resize_context.button == button)
|
||
gdk_win32_surface_end_move_resize_drag (window);
|
||
|
||
if (release_implicit_grab)
|
||
{
|
||
ReleaseCapture ();
|
||
|
||
new_window = NULL;
|
||
hwnd = WindowFromPoint (msg->pt);
|
||
if (hwnd != NULL)
|
||
{
|
||
POINT client_pt = msg->pt;
|
||
|
||
ScreenToClient (hwnd, &client_pt);
|
||
GetClientRect (hwnd, &rect);
|
||
if (PtInRect (&rect, client_pt))
|
||
new_window = gdk_win32_handle_table_lookup_ (hwnd);
|
||
}
|
||
|
||
synthesize_crossing_events (display,
|
||
_gdk_device_manager->system_pointer,
|
||
prev_surface, new_window,
|
||
GDK_CROSSING_UNGRAB,
|
||
&msg->pt,
|
||
0, /* TODO: Set right mask */
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
FALSE);
|
||
g_set_object (&mouse_window, new_window);
|
||
mouse_window_ignored_leave = NULL;
|
||
}
|
||
|
||
*ret_valp = (msg->message == WM_XBUTTONUP ? TRUE : 0);
|
||
return_val = TRUE;
|
||
break;
|
||
}
|
||
|
||
case WM_MOUSEMOVE:
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" %p (%d,%d)",
|
||
(gpointer) msg->wParam,
|
||
GET_X_LPARAM (msg->lParam), GET_Y_LPARAM (msg->lParam)));
|
||
|
||
/* Even if we handle WM_POINTER messages, synthetic WM_MOUSEMOVE messages
|
||
* are still sent occasionally by the OS, e.g. when a surface is hidden
|
||
* or shown. Discard spurious WM_MOUSEMOVE messages while handling pen or
|
||
* touch input
|
||
*
|
||
* See the article
|
||
* "Why do I get spurious WM_MOUSEMOVE messages?" by Raymond Chen:
|
||
* https://devblogs.microsoft.com/oldnewthing/20031001-00/?p=42343
|
||
*
|
||
*/
|
||
if (win32_display->tablet_input_api == GDK_WIN32_TABLET_INPUT_API_WINPOINTER &&
|
||
( (msg->time - last_digitizer_time) < 200 ||
|
||
-(msg->time - last_digitizer_time) < 200 ))
|
||
break;
|
||
|
||
pen_touch_input = FALSE;
|
||
|
||
g_set_object (&window, find_window_for_mouse_event (window, msg));
|
||
|
||
if (mouse_window != window)
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" mouse_window %p -> %p",
|
||
mouse_window ? GDK_SURFACE_HWND (mouse_window) : NULL,
|
||
window ? GDK_SURFACE_HWND (window) : NULL));
|
||
synthesize_crossing_events (display,
|
||
_gdk_device_manager->system_pointer,
|
||
mouse_window, window,
|
||
GDK_CROSSING_NORMAL,
|
||
&msg->pt,
|
||
0, /* TODO: Set right mask */
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
FALSE);
|
||
g_set_object (&mouse_window, window);
|
||
mouse_window_ignored_leave = NULL;
|
||
if (window != NULL)
|
||
track_mouse_event (TME_LEAVE, GDK_SURFACE_HWND (window));
|
||
}
|
||
else if (window != NULL && window == mouse_window_ignored_leave)
|
||
{
|
||
/* If we ignored a leave event for this window and we're now getting
|
||
input again we need to re-arm the mouse tracking, as that was
|
||
cancelled by the mouseleave. */
|
||
mouse_window_ignored_leave = NULL;
|
||
track_mouse_event (TME_LEAVE, GDK_SURFACE_HWND (window));
|
||
}
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
/* If we haven't moved, don't create any GDK event. Windows
|
||
* sends WM_MOUSEMOVE messages after a new window is shown under
|
||
* the mouse, even if the mouse hasn't moved. This disturbs gtk.
|
||
*/
|
||
if (msg->pt.x / impl->surface_scale == current_root_x &&
|
||
msg->pt.y / impl->surface_scale == current_root_y)
|
||
break;
|
||
|
||
current_root_x = msg->pt.x / impl->surface_scale;
|
||
current_root_y = msg->pt.y / impl->surface_scale;
|
||
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE)
|
||
gdk_win32_surface_do_move_resize_drag (window, current_root_x, current_root_y);
|
||
else if (_gdk_input_ignore_core == 0)
|
||
{
|
||
current_x = (gint16) GET_X_LPARAM (msg->lParam) / impl->surface_scale;
|
||
current_y = (gint16) GET_Y_LPARAM (msg->lParam) / impl->surface_scale;
|
||
|
||
_gdk_device_virtual_set_active (_gdk_device_manager->core_pointer,
|
||
_gdk_device_manager->system_pointer);
|
||
|
||
event = gdk_motion_event_new (window,
|
||
device_manager_win32->core_pointer,
|
||
NULL,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
build_pointer_event_state (msg),
|
||
current_x,
|
||
current_y,
|
||
NULL);
|
||
|
||
_gdk_win32_append_event (event);
|
||
}
|
||
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_NCMOUSEMOVE:
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" (%d,%d)",
|
||
GET_X_LPARAM (msg->lParam), GET_Y_LPARAM (msg->lParam)));
|
||
break;
|
||
|
||
case WM_MOUSELEAVE:
|
||
GDK_NOTE (EVENTS, g_print (" %d (%ld,%ld)",
|
||
HIWORD (msg->wParam), msg->pt.x, msg->pt.y));
|
||
|
||
pen_touch_input = FALSE;
|
||
|
||
new_window = NULL;
|
||
hwnd = WindowFromPoint (msg->pt);
|
||
ignore_leave = FALSE;
|
||
if (hwnd != NULL)
|
||
{
|
||
char classname[64];
|
||
|
||
POINT client_pt = msg->pt;
|
||
|
||
/* The synapitics trackpad drivers have this irritating
|
||
feature where it pops up a window right under the pointer
|
||
when you scroll. We ignore the leave and enter events for
|
||
this window */
|
||
if (GetClassNameA (hwnd, classname, sizeof(classname)) &&
|
||
strcmp (classname, SYNAPSIS_ICON_WINDOW_CLASS) == 0)
|
||
ignore_leave = TRUE;
|
||
|
||
ScreenToClient (hwnd, &client_pt);
|
||
GetClientRect (hwnd, &rect);
|
||
if (PtInRect (&rect, client_pt))
|
||
new_window = gdk_win32_handle_table_lookup_ (hwnd);
|
||
}
|
||
|
||
if (!ignore_leave)
|
||
synthesize_crossing_events (display,
|
||
_gdk_device_manager->system_pointer,
|
||
mouse_window, new_window,
|
||
GDK_CROSSING_NORMAL,
|
||
&msg->pt,
|
||
0, /* TODO: Set right mask */
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
FALSE);
|
||
g_set_object (&mouse_window, new_window);
|
||
mouse_window_ignored_leave = ignore_leave ? new_window : NULL;
|
||
|
||
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_POINTERDOWN:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (pointer_grab != NULL &&
|
||
!pointer_grab->implicit &&
|
||
!pointer_grab->owner_events)
|
||
g_set_object (&window, pointer_grab->surface);
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam) && mouse_window != window)
|
||
crossing_cb = make_crossing_event;
|
||
|
||
gdk_winpointer_input_events (window, crossing_cb, msg);
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_POINTERUP:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (pointer_grab != NULL &&
|
||
!pointer_grab->implicit &&
|
||
!pointer_grab->owner_events)
|
||
g_set_object (&window, pointer_grab->surface);
|
||
|
||
gdk_winpointer_input_events (window, NULL, msg);
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE)
|
||
{
|
||
gdk_win32_surface_end_move_resize_drag (window);
|
||
}
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_POINTERUPDATE:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (pointer_grab != NULL &&
|
||
!pointer_grab->implicit &&
|
||
!pointer_grab->owner_events)
|
||
g_set_object (&window, pointer_grab->surface);
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam) && mouse_window != window)
|
||
crossing_cb = make_crossing_event;
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE)
|
||
{
|
||
gdk_win32_surface_do_move_resize_drag (window, current_root_x, current_root_y);
|
||
}
|
||
else
|
||
{
|
||
gdk_winpointer_input_events (window, crossing_cb, msg);
|
||
}
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_NCPOINTERUPDATE:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam) &&
|
||
!IS_POINTER_INCONTACT_WPARAM (msg->wParam) &&
|
||
mouse_window != NULL)
|
||
{
|
||
GdkDevice *event_device = NULL;
|
||
guint32 event_time = 0;
|
||
|
||
if (gdk_winpointer_get_message_info (msg, &event_device, &event_time))
|
||
{
|
||
make_crossing_event(event_device,
|
||
NULL,
|
||
&pen_touch_cursor_position,
|
||
event_time);
|
||
}
|
||
}
|
||
|
||
return_val = FALSE; /* forward to DefWindowProc */
|
||
break;
|
||
|
||
case WM_POINTERENTER:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (pointer_grab != NULL &&
|
||
!pointer_grab->implicit &&
|
||
!pointer_grab->owner_events)
|
||
g_set_object (&window, pointer_grab->surface);
|
||
|
||
if (IS_POINTER_NEW_WPARAM (msg->wParam))
|
||
{
|
||
gdk_winpointer_input_events (window, NULL, msg);
|
||
}
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_POINTERLEAVE:
|
||
if (win32_display->tablet_input_api != GDK_WIN32_TABLET_INPUT_API_WINPOINTER ||
|
||
gdk_winpointer_should_forward_message (msg))
|
||
{
|
||
return_val = FALSE;
|
||
break;
|
||
}
|
||
|
||
if (IS_POINTER_PRIMARY_WPARAM (msg->wParam))
|
||
{
|
||
current_root_x = pen_touch_cursor_position.x = GET_X_LPARAM (msg->lParam);
|
||
current_root_y = pen_touch_cursor_position.y = GET_Y_LPARAM (msg->lParam);
|
||
pen_touch_input = TRUE;
|
||
last_digitizer_time = msg->time;
|
||
}
|
||
|
||
if (!IS_POINTER_INRANGE_WPARAM (msg->wParam))
|
||
{
|
||
gdk_winpointer_input_events (window, NULL, msg);
|
||
}
|
||
else if (IS_POINTER_PRIMARY_WPARAM (msg->wParam) && mouse_window != NULL)
|
||
{
|
||
GdkDevice *event_device = NULL;
|
||
guint32 event_time = 0;
|
||
|
||
if (gdk_winpointer_get_message_info (msg, &event_device, &event_time))
|
||
{
|
||
make_crossing_event(event_device,
|
||
NULL,
|
||
&pen_touch_cursor_position,
|
||
event_time);
|
||
}
|
||
}
|
||
|
||
gdk_winpointer_interaction_ended (msg);
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case DM_POINTERHITTEST:
|
||
gdk_dmanipulation_maybe_add_contact (window, msg);
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_MOUSEWHEEL:
|
||
case WM_MOUSEHWHEEL:
|
||
{
|
||
int16_t scroll_x = 0;
|
||
int16_t scroll_y = 0;
|
||
GdkScrollDirection direction;
|
||
|
||
char classname[64];
|
||
|
||
GDK_NOTE (EVENTS, g_print (" %d", (short) HIWORD (msg->wParam)));
|
||
|
||
/* On versions of Windows before Windows 10, the WM_MOUSEWHEEL
|
||
* is delivered to the window that has keyboard focus, not the
|
||
* window under the pointer. Work around that.
|
||
* Also, the position is in screen coordinates, not client
|
||
* coordinates as with the button messages. */
|
||
point.x = GET_X_LPARAM (msg->lParam);
|
||
point.y = GET_Y_LPARAM (msg->lParam);
|
||
|
||
hwnd = WindowFromPoint (point);
|
||
if (!hwnd)
|
||
break;
|
||
|
||
/* The synapitics trackpad drivers have this irritating
|
||
feature where it pops up a window right under the pointer
|
||
when you scroll. We backtrack and to the toplevel and
|
||
find the innermost child instead. */
|
||
if (GetClassNameA (hwnd, classname, sizeof(classname)) &&
|
||
strcmp (classname, SYNAPSIS_ICON_WINDOW_CLASS) == 0)
|
||
{
|
||
HWND hwndc;
|
||
|
||
/* Find our toplevel window */
|
||
hwnd = GetAncestor (msg->hwnd, GA_ROOT);
|
||
|
||
/* Walk back up to the outermost child at the desired point */
|
||
do {
|
||
ScreenToClient (hwnd, &point);
|
||
hwndc = ChildWindowFromPoint (hwnd, point);
|
||
ClientToScreen (hwnd, &point);
|
||
} while (hwndc != hwnd && (hwnd = hwndc, 1));
|
||
}
|
||
|
||
msg->hwnd = hwnd;
|
||
|
||
g_set_object (&window, gdk_win32_handle_table_lookup_ (hwnd));
|
||
if (!window)
|
||
break;
|
||
|
||
if (msg->message == WM_MOUSEWHEEL)
|
||
scroll_y = GET_WHEEL_DELTA_WPARAM (msg->wParam);
|
||
else if (msg->message == WM_MOUSEHWHEEL)
|
||
scroll_x = GET_WHEEL_DELTA_WPARAM (msg->wParam);
|
||
|
||
_gdk_device_virtual_set_active (_gdk_device_manager->core_pointer,
|
||
_gdk_device_manager->system_pointer);
|
||
|
||
direction = 0;
|
||
if (msg->message == WM_MOUSEWHEEL)
|
||
direction = (((short) HIWORD (msg->wParam)) > 0)
|
||
? GDK_SCROLL_UP
|
||
: GDK_SCROLL_DOWN;
|
||
else if (msg->message == WM_MOUSEHWHEEL)
|
||
direction = (((short) HIWORD (msg->wParam)) > 0)
|
||
? GDK_SCROLL_RIGHT
|
||
: GDK_SCROLL_LEFT;
|
||
|
||
event = gdk_scroll_event_new_value120 (window,
|
||
device_manager_win32->core_pointer,
|
||
NULL,
|
||
_gdk_win32_get_next_tick (msg->time),
|
||
build_pointer_event_state (msg),
|
||
direction,
|
||
(double) scroll_x,
|
||
(double) -scroll_y);
|
||
|
||
_gdk_win32_append_event (event);
|
||
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
}
|
||
break;
|
||
|
||
case WM_MOUSEACTIVATE:
|
||
{
|
||
if (GDK_IS_DRAG_SURFACE (window))
|
||
{
|
||
*ret_valp = MA_NOACTIVATE;
|
||
return_val = TRUE;
|
||
}
|
||
|
||
if (_gdk_modal_blocked (window))
|
||
{
|
||
*ret_valp = MA_NOACTIVATEANDEAT;
|
||
return_val = TRUE;
|
||
}
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_POINTERACTIVATE:
|
||
if (GDK_IS_DRAG_SURFACE (window) ||
|
||
_gdk_modal_blocked (window))
|
||
{
|
||
*ret_valp = PA_NOACTIVATE;
|
||
return_val = TRUE;
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_KILLFOCUS:
|
||
if (keyboard_grab != NULL &&
|
||
!GDK_SURFACE_DESTROYED (keyboard_grab->surface) &&
|
||
(_modal_operation_in_progress & GDK_WIN32_MODAL_OP_DND) == 0)
|
||
{
|
||
generate_grab_broken_event (_gdk_device_manager, keyboard_grab->surface, TRUE, NULL);
|
||
}
|
||
G_GNUC_FALLTHROUGH;
|
||
|
||
case WM_SETFOCUS:
|
||
if (keyboard_grab != NULL &&
|
||
!keyboard_grab->owner_events)
|
||
break;
|
||
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
generate_focus_event (_gdk_device_manager, window, (msg->message == WM_SETFOCUS));
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_ERASEBKGND:
|
||
GDK_NOTE (EVENTS, g_print (" %p", (HANDLE) msg->wParam));
|
||
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
return_val = TRUE;
|
||
*ret_valp = 1;
|
||
break;
|
||
|
||
case WM_PAINT:
|
||
handle_wm_paint (msg, window);
|
||
break;
|
||
|
||
case WM_SETCURSOR:
|
||
GDK_NOTE (EVENTS, g_print (" %#x %#x",
|
||
LOWORD (msg->lParam), HIWORD (msg->lParam)));
|
||
|
||
if (pointer_grab != NULL)
|
||
grab_window = pointer_grab->surface;
|
||
|
||
if (grab_window == NULL && LOWORD (msg->lParam) != HTCLIENT)
|
||
break;
|
||
|
||
return_val = FALSE;
|
||
|
||
if (grab_window != NULL &&
|
||
!GDK_SURFACE_DESTROYED (grab_window))
|
||
{
|
||
win32_display = GDK_WIN32_DISPLAY (gdk_surface_get_display (grab_window));
|
||
|
||
if (win32_display->grab_cursor != NULL)
|
||
{
|
||
GDK_NOTE (EVENTS, g_print (" (grab SetCursor(%p)", gdk_win32_hcursor_get_handle (win32_display->grab_cursor)));
|
||
SetCursor (gdk_win32_hcursor_get_handle (win32_display->grab_cursor));
|
||
return_val = TRUE;
|
||
*ret_valp = TRUE;
|
||
}
|
||
}
|
||
|
||
if (!return_val &&
|
||
!GDK_SURFACE_DESTROYED (window) &&
|
||
GDK_WIN32_SURFACE (window)->cursor != NULL)
|
||
{
|
||
win32_display = GDK_WIN32_DISPLAY (gdk_surface_get_display (window));
|
||
GDK_NOTE (EVENTS, g_print (" (window SetCursor(%p)", gdk_win32_hcursor_get_handle (GDK_WIN32_SURFACE (window)->cursor)));
|
||
SetCursor (gdk_win32_hcursor_get_handle (GDK_WIN32_SURFACE (window)->cursor));
|
||
return_val = TRUE;
|
||
*ret_valp = TRUE;
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_SYSMENU:
|
||
return_val = handle_wm_sysmenu (window, msg, ret_valp);
|
||
break;
|
||
|
||
case WM_INITMENU:
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->have_temp_styles)
|
||
{
|
||
LONG_PTR window_style;
|
||
|
||
window_style = GetWindowLongPtr (GDK_SURFACE_HWND (window),
|
||
GWL_STYLE);
|
||
/* Handling WM_SYSMENU added extra styles to this window,
|
||
* remove them now.
|
||
*/
|
||
window_style &= ~impl->temp_styles;
|
||
SetWindowLongPtr (GDK_SURFACE_HWND (window),
|
||
GWL_STYLE,
|
||
window_style);
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_SYSCOMMAND:
|
||
switch (msg->wParam)
|
||
{
|
||
case SC_MINIMIZE:
|
||
case SC_RESTORE:
|
||
do_show_window (window, msg->wParam == SC_MINIMIZE ? TRUE : FALSE);
|
||
|
||
if (msg->wParam == SC_RESTORE)
|
||
_gdk_win32_surface_invalidate_egl_framebuffer (window);
|
||
|
||
break;
|
||
case SC_MAXIMIZE:
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
impl->maximizing = TRUE;
|
||
break;
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_ENTERSIZEMOVE:
|
||
_modal_move_resize_window = msg->hwnd;
|
||
_gdk_win32_begin_modal_call (GDK_WIN32_MODAL_OP_SIZEMOVE_MASK);
|
||
break;
|
||
|
||
case WM_EXITSIZEMOVE:
|
||
if (_modal_operation_in_progress & GDK_WIN32_MODAL_OP_SIZEMOVE_MASK)
|
||
{
|
||
_modal_move_resize_window = NULL;
|
||
_gdk_win32_end_modal_call (GDK_WIN32_MODAL_OP_SIZEMOVE_MASK);
|
||
}
|
||
break;
|
||
|
||
case WM_ENTERMENULOOP:
|
||
_gdk_win32_begin_modal_call (GDK_WIN32_MODAL_OP_MENU);
|
||
break;
|
||
|
||
case WM_EXITMENULOOP:
|
||
if (_modal_operation_in_progress & GDK_WIN32_MODAL_OP_MENU)
|
||
_gdk_win32_end_modal_call (GDK_WIN32_MODAL_OP_MENU);
|
||
break;
|
||
|
||
break;
|
||
|
||
/*
|
||
* Handle WM_CANCELMODE and do nothing in response to it when DnD is
|
||
* active. Otherwise pass it to DefWindowProc, which will call ReleaseCapture()
|
||
* on our behalf.
|
||
* This prevents us from losing mouse capture when alt-tabbing during DnD
|
||
* (this includes the feature of Windows Explorer where dragging stuff over
|
||
* a window button in the taskbar causes that window to receive focus, i.e.
|
||
* keyboardless alt-tabbing).
|
||
*/
|
||
case WM_CANCELMODE:
|
||
if (_modal_operation_in_progress & GDK_WIN32_MODAL_OP_DND)
|
||
{
|
||
return_val = TRUE;
|
||
*ret_valp = 0;
|
||
}
|
||
break;
|
||
|
||
case WM_CAPTURECHANGED:
|
||
/* Sometimes we don't get WM_EXITSIZEMOVE, for instance when you
|
||
select move/size in the menu and then click somewhere without
|
||
moving/resizing. We work around this using WM_CAPTURECHANGED. */
|
||
if (_modal_operation_in_progress & GDK_WIN32_MODAL_OP_SIZEMOVE_MASK)
|
||
{
|
||
_modal_move_resize_window = NULL;
|
||
_gdk_win32_end_modal_call (GDK_WIN32_MODAL_OP_SIZEMOVE_MASK);
|
||
}
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->drag_move_resize_context.op != GDK_WIN32_DRAGOP_NONE)
|
||
gdk_win32_surface_end_move_resize_drag (window);
|
||
break;
|
||
|
||
case WM_WINDOWPOSCHANGING:
|
||
#ifdef G_ENABLE_DEBUG
|
||
{
|
||
char buf[256];
|
||
GDK_NOTE (EVENTS, (windowpos = (WINDOWPOS *) msg->lParam,
|
||
g_print (" %s %s %dx%d@%+d%+d now below %p",
|
||
_gdk_win32_surface_pos_bits_to_string (windowpos->flags),
|
||
(windowpos->hwndInsertAfter == HWND_BOTTOM ? "BOTTOM" :
|
||
(windowpos->hwndInsertAfter == HWND_NOTOPMOST ? "NOTOPMOST" :
|
||
(windowpos->hwndInsertAfter == HWND_TOP ? "TOP" :
|
||
(windowpos->hwndInsertAfter == HWND_TOPMOST ? "TOPMOST" :
|
||
(sprintf (buf, "%p", windowpos->hwndInsertAfter),
|
||
buf))))),
|
||
windowpos->cx, windowpos->cy, windowpos->x, windowpos->y,
|
||
GetNextWindow (msg->hwnd, GW_HWNDPREV))));
|
||
}
|
||
#endif
|
||
|
||
if (GDK_SURFACE_IS_MAPPED (window))
|
||
{
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->maximizing)
|
||
{
|
||
MINMAXINFO our_mmi;
|
||
|
||
if (_gdk_win32_surface_fill_min_max_info (window, &our_mmi))
|
||
{
|
||
windowpos = (WINDOWPOS *) msg->lParam;
|
||
windowpos->cx = our_mmi.ptMaxSize.x;
|
||
windowpos->cy = our_mmi.ptMaxSize.y;
|
||
}
|
||
|
||
impl->maximizing = FALSE;
|
||
}
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_WINDOWPOSCHANGED:
|
||
windowpos = (WINDOWPOS *) msg->lParam;
|
||
|
||
#ifdef G_ENABLE_DEBUG
|
||
{
|
||
char buf[256];
|
||
GDK_NOTE (EVENTS, g_print (" %s %s %dx%d@%+d%+d",
|
||
_gdk_win32_surface_pos_bits_to_string (windowpos->flags),
|
||
(windowpos->hwndInsertAfter == HWND_BOTTOM ? "BOTTOM" :
|
||
(windowpos->hwndInsertAfter == HWND_NOTOPMOST ? "NOTOPMOST" :
|
||
(windowpos->hwndInsertAfter == HWND_TOP ? "TOP" :
|
||
(windowpos->hwndInsertAfter == HWND_TOPMOST ? "TOPMOST" :
|
||
(sprintf (buf, "%p", windowpos->hwndInsertAfter),
|
||
buf))))),
|
||
windowpos->cx, windowpos->cy, windowpos->x, windowpos->y));
|
||
}
|
||
#endif
|
||
|
||
/* Break grabs on unmap or minimize */
|
||
if (windowpos->flags & SWP_HIDEWINDOW ||
|
||
((windowpos->flags & SWP_STATECHANGED) && IsIconic (msg->hwnd)))
|
||
{
|
||
GdkDevice *device = gdk_seat_get_pointer (gdk_display_get_default_seat (display));
|
||
|
||
if ((pointer_grab != NULL && pointer_grab->surface == window) ||
|
||
(keyboard_grab != NULL && keyboard_grab->surface == window))
|
||
gdk_device_ungrab (device, msg -> time);
|
||
}
|
||
|
||
/* Update window state */
|
||
if (windowpos->flags & (SWP_STATECHANGED | SWP_SHOWWINDOW | SWP_HIDEWINDOW))
|
||
{
|
||
GdkToplevelState set_bits, unset_bits, old_state, new_state;
|
||
|
||
old_state = window->state;
|
||
|
||
set_bits = 0;
|
||
unset_bits = 0;
|
||
|
||
if (IsIconic (msg->hwnd))
|
||
set_bits |= GDK_TOPLEVEL_STATE_MINIMIZED;
|
||
else
|
||
unset_bits |= GDK_TOPLEVEL_STATE_MINIMIZED;
|
||
|
||
if (IsZoomed (msg->hwnd))
|
||
set_bits |= GDK_TOPLEVEL_STATE_MAXIMIZED;
|
||
else
|
||
unset_bits |= GDK_TOPLEVEL_STATE_MAXIMIZED;
|
||
|
||
/*
|
||
* If we are minizing, pause all surface layout computations, and re-start the
|
||
* computation once we are coming out of a minimized state
|
||
*/
|
||
if (!(old_state & GDK_TOPLEVEL_STATE_MINIMIZED) && set_bits & GDK_TOPLEVEL_STATE_MINIMIZED)
|
||
gdk_surface_freeze_updates (window);
|
||
|
||
if (old_state & GDK_TOPLEVEL_STATE_MINIMIZED && unset_bits & GDK_TOPLEVEL_STATE_MINIMIZED)
|
||
gdk_surface_thaw_updates (window);
|
||
|
||
gdk_surface_set_is_mapped (window, !!IsWindowVisible (msg->hwnd));
|
||
gdk_synthesize_surface_state (window, unset_bits, set_bits);
|
||
|
||
new_state = window->state;
|
||
|
||
/* Whenever one window changes iconified state we need to also
|
||
* change the iconified state in all transient related windows,
|
||
* as windows doesn't give icons for transient childrens.
|
||
*/
|
||
if ((old_state & GDK_TOPLEVEL_STATE_MINIMIZED) !=
|
||
(new_state & GDK_TOPLEVEL_STATE_MINIMIZED))
|
||
do_show_window (window, (new_state & GDK_TOPLEVEL_STATE_MINIMIZED));
|
||
}
|
||
|
||
/* Show, New size or position => configure event */
|
||
if (!(windowpos->flags & SWP_NOCLIENTMOVE) ||
|
||
!(windowpos->flags & SWP_NOCLIENTSIZE) ||
|
||
(windowpos->flags & SWP_SHOWWINDOW))
|
||
{
|
||
if (!IsIconic (msg->hwnd) &&
|
||
!GDK_SURFACE_DESTROYED (window))
|
||
gdk_surface_request_layout (window);
|
||
}
|
||
|
||
if ((windowpos->flags & SWP_HIDEWINDOW) &&
|
||
!GDK_SURFACE_DESTROYED (window))
|
||
{
|
||
/* Make transient parent the foreground window when window unmaps */
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->transient_owner &&
|
||
GetForegroundWindow () == GDK_SURFACE_HWND (window))
|
||
{
|
||
SetForegroundWindow (GDK_SURFACE_HWND (impl->transient_owner));
|
||
SetCapture (GDK_SURFACE_HWND (impl->transient_owner));
|
||
}
|
||
}
|
||
|
||
if (!(windowpos->flags & SWP_NOCLIENTSIZE))
|
||
{
|
||
if (window->resize_count > 1)
|
||
window->resize_count -= 1;
|
||
}
|
||
|
||
/* Call modal timer immediate so that we repaint faster after a resize. */
|
||
if (_modal_operation_in_progress & GDK_WIN32_MODAL_OP_SIZEMOVE_MASK)
|
||
modal_timer_proc (0,0,0,0);
|
||
|
||
/* Claim as handled, so that WM_SIZE and WM_MOVE are avoided */
|
||
return_val = TRUE;
|
||
*ret_valp = 0;
|
||
break;
|
||
|
||
case WM_SIZING:
|
||
GetWindowRect (GDK_SURFACE_HWND (window), &rect);
|
||
GDK_NOTE (EVENTS, g_print (" %s curr:%s drag:%s",
|
||
(msg->wParam == WMSZ_BOTTOM ? "BOTTOM" :
|
||
(msg->wParam == WMSZ_BOTTOMLEFT ? "BOTTOMLEFT" :
|
||
(msg->wParam == WMSZ_LEFT ? "LEFT" :
|
||
(msg->wParam == WMSZ_TOPLEFT ? "TOPLEFT" :
|
||
(msg->wParam == WMSZ_TOP ? "TOP" :
|
||
(msg->wParam == WMSZ_TOPRIGHT ? "TOPRIGHT" :
|
||
(msg->wParam == WMSZ_RIGHT ? "RIGHT" :
|
||
|
||
(msg->wParam == WMSZ_BOTTOMRIGHT ? "BOTTOMRIGHT" :
|
||
"???")))))))),
|
||
_gdk_win32_rect_to_string (&rect),
|
||
_gdk_win32_rect_to_string ((RECT *) msg->lParam)));
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
break;
|
||
|
||
case WM_GETMINMAXINFO:
|
||
mmi = (MINMAXINFO*) msg->lParam;
|
||
|
||
GDK_NOTE (EVENTS, g_print (" (mintrack:%ldx%ld maxtrack:%ldx%ld "
|
||
"maxpos:%+ld%+ld maxsize:%ldx%ld)",
|
||
mmi->ptMinTrackSize.x, mmi->ptMinTrackSize.y,
|
||
mmi->ptMaxTrackSize.x, mmi->ptMaxTrackSize.y,
|
||
mmi->ptMaxPosition.x, mmi->ptMaxPosition.y,
|
||
mmi->ptMaxSize.x, mmi->ptMaxSize.y));
|
||
|
||
if (_gdk_win32_surface_fill_min_max_info (window, mmi))
|
||
{
|
||
/* Don't call DefWindowProcW() */
|
||
GDK_NOTE (EVENTS,
|
||
g_print (" (handled, mintrack:%ldx%ld maxtrack:%ldx%ld "
|
||
"maxpos:%+ld%+ld maxsize:%ldx%ld)",
|
||
mmi->ptMinTrackSize.x, mmi->ptMinTrackSize.y,
|
||
mmi->ptMaxTrackSize.x, mmi->ptMaxTrackSize.y,
|
||
mmi->ptMaxPosition.x, mmi->ptMaxPosition.y,
|
||
mmi->ptMaxSize.x, mmi->ptMaxSize.y));
|
||
|
||
return_val = TRUE;
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_CLOSE:
|
||
if (GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
event = gdk_delete_event_new (window);
|
||
|
||
_gdk_win32_append_event (event);
|
||
|
||
impl = GDK_WIN32_SURFACE (window);
|
||
|
||
if (impl->transient_owner && GetForegroundWindow() == GDK_SURFACE_HWND (window))
|
||
{
|
||
SetForegroundWindow (GDK_SURFACE_HWND (impl->transient_owner));
|
||
}
|
||
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_DPICHANGED:
|
||
handle_dpi_changed (window, msg);
|
||
return_val = FALSE;
|
||
*ret_valp = 0;
|
||
break;
|
||
|
||
case WM_DESTROY:
|
||
if (win32_display->tablet_input_api == GDK_WIN32_TABLET_INPUT_API_WINPOINTER)
|
||
gdk_winpointer_finalize_surface (window);
|
||
|
||
gdk_dmanipulation_finalize_surface (window);
|
||
|
||
return_val = FALSE;
|
||
break;
|
||
|
||
case WM_NCDESTROY:
|
||
if ((pointer_grab != NULL && pointer_grab->surface == window) ||
|
||
(keyboard_grab && keyboard_grab->surface == window))
|
||
{
|
||
GdkDevice *device = gdk_seat_get_pointer (gdk_display_get_default_seat (display));
|
||
gdk_device_ungrab (device, msg -> time);
|
||
}
|
||
|
||
if ((window != NULL) && (msg->hwnd != GetDesktopWindow ()))
|
||
gdk_surface_destroy_notify (window);
|
||
|
||
if (window == NULL || GDK_SURFACE_DESTROYED (window))
|
||
break;
|
||
|
||
event = gdk_delete_event_new (window);
|
||
|
||
_gdk_win32_append_event (event);
|
||
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
case WM_DWMCOMPOSITIONCHANGED:
|
||
gdk_win32_display_check_composited (GDK_WIN32_DISPLAY (display));
|
||
_gdk_win32_surface_enable_transparency (window);
|
||
break;
|
||
|
||
case WM_ACTIVATE:
|
||
GDK_NOTE (EVENTS, g_print (" %s%s %p",
|
||
(LOWORD (msg->wParam) == WA_ACTIVE ? "ACTIVE" :
|
||
(LOWORD (msg->wParam) == WA_CLICKACTIVE ? "CLICKACTIVE" :
|
||
(LOWORD (msg->wParam) == WA_INACTIVE ? "INACTIVE" : "???"))),
|
||
HIWORD (msg->wParam) ? " minimized" : "",
|
||
(HWND) msg->lParam));
|
||
if (GDK_IS_POPUP (window) || GDK_IS_DRAG_SURFACE (window))
|
||
{
|
||
/* Popups cannot be activated or de-activated -
|
||
* they only support keyboard focus, which GTK
|
||
* will handle for us.
|
||
*/
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
}
|
||
/* We handle mouse clicks for modally-blocked windows under WM_MOUSEACTIVATE,
|
||
* but we still need to deal with alt-tab, or with SetActiveWindow() type
|
||
* situations.
|
||
*/
|
||
if (_gdk_modal_blocked (window) && LOWORD (msg->wParam) == WA_ACTIVE)
|
||
{
|
||
GdkSurface *modal_current = _gdk_modal_current ();
|
||
SetActiveWindow (GDK_SURFACE_HWND (modal_current));
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
}
|
||
|
||
if (LOWORD (msg->wParam) == WA_INACTIVE)
|
||
{
|
||
if (msg->lParam != 0)
|
||
{
|
||
GdkSurface *other_surface = gdk_win32_handle_table_lookup_ ((HWND) msg->lParam);
|
||
if (other_surface != NULL &&
|
||
(GDK_IS_POPUP (other_surface) || GDK_IS_DRAG_SURFACE (other_surface)))
|
||
{
|
||
/* We're being deactivated in favour of some popup or temp window.
|
||
* Since only toplevels can have visual focus, pretend that
|
||
* nothing happened.
|
||
*/
|
||
*ret_valp = 0;
|
||
return_val = TRUE;
|
||
break;
|
||
}
|
||
}
|
||
|
||
gdk_synthesize_surface_state (window, GDK_TOPLEVEL_STATE_FOCUSED, 0);
|
||
}
|
||
else
|
||
{
|
||
gdk_synthesize_surface_state (window, 0, GDK_TOPLEVEL_STATE_FOCUSED);
|
||
|
||
if (win32_display->tablet_input_api == GDK_WIN32_TABLET_INPUT_API_WINTAB)
|
||
{
|
||
/* Bring any tablet contexts to the top of the overlap order when
|
||
* one of our windows is activated.
|
||
* NOTE: It doesn't seem to work well if it is done in WM_ACTIVATEAPP
|
||
* instead
|
||
*/
|
||
_gdk_wintab_set_tablet_active ();
|
||
}
|
||
}
|
||
|
||
break;
|
||
|
||
case WM_ACTIVATEAPP:
|
||
GDK_NOTE (EVENTS, g_print (" %s thread: %" G_GINT64_FORMAT,
|
||
msg->wParam ? "YES" : "NO",
|
||
(gint64) msg->lParam));
|
||
break;
|
||
case WM_NCHITTEST:
|
||
/* TODO: pass all messages to DwmDefWindowProc() first! */
|
||
return_val = handle_nchittest (msg->hwnd, window,
|
||
GET_X_LPARAM (msg->lParam),
|
||
GET_Y_LPARAM (msg->lParam), ret_valp);
|
||
break;
|
||
|
||
case WM_TABLET_QUERYSYSTEMGESTURESTATUS:
|
||
*ret_valp = TABLET_DISABLE_PRESSANDHOLD |
|
||
TABLET_DISABLE_PENTAPFEEDBACK |
|
||
TABLET_DISABLE_PENBARRELFEEDBACK |
|
||
TABLET_DISABLE_FLICKS |
|
||
TABLET_DISABLE_FLICKFALLBACKKEYS;
|
||
return_val = TRUE;
|
||
break;
|
||
|
||
|
||
/* Handle WINTAB events here, as we know that the device manager will
|
||
* use the fixed WT_DEFBASE as lcMsgBase, and we thus can use the
|
||
* constants as case labels.
|
||
*/
|
||
case WT_PACKET:
|
||
GDK_NOTE (EVENTS, g_print (" %d %p",
|
||
(int) msg->wParam, (gpointer) msg->lParam));
|
||
goto wintab;
|
||
|
||
case WT_CSRCHANGE:
|
||
GDK_NOTE (EVENTS, g_print (" %d %p",
|
||
(int) msg->wParam, (gpointer) msg->lParam));
|
||
goto wintab;
|
||
|
||
case WT_PROXIMITY:
|
||
GDK_NOTE (EVENTS, g_print (" %p %d %d",
|
||
(gpointer) msg->wParam,
|
||
LOWORD (msg->lParam),
|
||
HIWORD (msg->lParam)));
|
||
/* Fall through */
|
||
wintab:
|
||
|
||
if (win32_display->tablet_input_api == GDK_WIN32_TABLET_INPUT_API_WINTAB)
|
||
{
|
||
event = gdk_wintab_make_event (display, msg, window);
|
||
if (event)
|
||
{
|
||
_gdk_win32_append_event (event);
|
||
gdk_event_unref (event);
|
||
}
|
||
}
|
||
|
||
break;
|
||
}
|
||
|
||
done:
|
||
|
||
if (window)
|
||
g_object_unref (window);
|
||
|
||
#undef return
|
||
return return_val;
|
||
}
|
||
|
||
void
|
||
_gdk_win32_display_queue_events (GdkDisplay *display)
|
||
{
|
||
MSG msg;
|
||
|
||
if (modal_win32_dialog != NULL)
|
||
return;
|
||
|
||
while (PeekMessageW (&msg, NULL, 0, 0, PM_REMOVE))
|
||
{
|
||
TranslateMessage (&msg);
|
||
DispatchMessageW (&msg);
|
||
}
|
||
}
|
||
|
||
static gboolean
|
||
gdk_event_prepare (GSource *source,
|
||
int *timeout)
|
||
{
|
||
GdkWin32EventSource *event_source = (GdkWin32EventSource *)source;
|
||
gboolean retval;
|
||
|
||
*timeout = -1;
|
||
|
||
if (event_source->display->event_pause_count > 0)
|
||
retval =_gdk_event_queue_find_first (event_source->display) != NULL;
|
||
else
|
||
retval = (_gdk_event_queue_find_first (event_source->display) != NULL ||
|
||
(modal_win32_dialog == NULL &&
|
||
GetQueueStatus (QS_ALLINPUT) != 0));
|
||
|
||
return retval;
|
||
}
|
||
|
||
static gboolean
|
||
gdk_event_check (GSource *source)
|
||
{
|
||
GdkWin32EventSource *event_source = (GdkWin32EventSource *)source;
|
||
gboolean retval;
|
||
|
||
if (event_source->display->event_pause_count > 0)
|
||
retval = _gdk_event_queue_find_first (event_source->display) != NULL;
|
||
else if (event_source->event_poll_fd.revents & G_IO_IN)
|
||
retval = (_gdk_event_queue_find_first (event_source->display) != NULL ||
|
||
(modal_win32_dialog == NULL &&
|
||
GetQueueStatus (QS_ALLINPUT) != 0));
|
||
else
|
||
retval = FALSE;
|
||
|
||
return retval;
|
||
}
|
||
|
||
static gboolean
|
||
gdk_event_dispatch (GSource *source,
|
||
GSourceFunc callback,
|
||
gpointer user_data)
|
||
{
|
||
GdkWin32EventSource *event_source = (GdkWin32EventSource *)source;
|
||
GdkEvent *event;
|
||
|
||
_gdk_win32_display_queue_events (event_source->display);
|
||
event = _gdk_event_unqueue (event_source->display);
|
||
|
||
if (event)
|
||
{
|
||
_gdk_event_emit (event);
|
||
|
||
gdk_event_unref (event);
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
void
|
||
gdk_win32_set_modal_dialog_libgtk_only (HWND window)
|
||
{
|
||
modal_win32_dialog = window;
|
||
}
|