Blame client/Windows/wf_event.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * Event Handling
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2009-2011 Jay Sorg
Packit Service fa4841
 * Copyright 2010-2011 Vic Lee
Packit Service fa4841
 * Copyright 2010-2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <stdio.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/freerdp.h>
Packit Service fa4841
Packit Service fa4841
#include "wf_client.h"
Packit Service fa4841
Packit Service fa4841
#include "wf_gdi.h"
Packit Service fa4841
#include "wf_event.h"
Packit Service fa4841
Packit Service fa4841
#include <freerdp/event.h>
Packit Service fa4841
Packit Service fa4841
static HWND g_focus_hWnd;
Packit Service fa4841
Packit Service b1ea74
#define X_POS(lParam) ((UINT16)(lParam & 0xFFFF))
Packit Service b1ea74
#define Y_POS(lParam) ((UINT16)((lParam >> 16) & 0xFFFF))
Packit Service fa4841
Packit Service b1ea74
static BOOL wf_scale_blt(wfContext* wfc, HDC hdc, int x, int y, int w, int h, HDC hdcSrc, int x1,
Packit Service b1ea74
                         int y1, DWORD rop);
Packit Service b1ea74
static BOOL wf_scale_mouse_event(wfContext* wfc, rdpInput* input, UINT16 flags, UINT16 x, UINT16 y);
Packit Service fa4841
#if (_WIN32_WINNT >= 0x0500)
Packit Service fa4841
static BOOL wf_scale_mouse_event_ex(wfContext* wfc, rdpInput* input, UINT16 flags,
Packit Service fa4841
                                    UINT16 buttonMask, UINT16 x, UINT16 y);
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
static BOOL g_flipping_in;
Packit Service fa4841
static BOOL g_flipping_out;
Packit Service fa4841
Packit Service fa4841
static BOOL alt_ctrl_down()
Packit Service fa4841
{
Packit Service b1ea74
	return ((GetAsyncKeyState(VK_CONTROL) & 0x8000) || (GetAsyncKeyState(VK_MENU) & 0x8000));
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
LRESULT CALLBACK wf_ll_kbd_proc(int nCode, WPARAM wParam, LPARAM lParam)
Packit Service fa4841
{
Packit Service fa4841
	wfContext* wfc;
Packit Service fa4841
	DWORD rdp_scancode;
Packit Service fa4841
	rdpInput* input;
Packit Service fa4841
	PKBDLLHOOKSTRUCT p;
Packit Service b1ea74
	DEBUG_KBD("Low-level keyboard hook, hWnd %X nCode %X wParam %X", g_focus_hWnd, nCode, wParam);
Packit Service fa4841
Packit Service fa4841
	if (g_flipping_in)
Packit Service fa4841
	{
Packit Service fa4841
		if (!alt_ctrl_down())
Packit Service fa4841
			g_flipping_in = FALSE;
Packit Service fa4841
Packit Service fa4841
		return CallNextHookEx(NULL, nCode, wParam, lParam);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (g_focus_hWnd && (nCode == HC_ACTION))
Packit Service fa4841
	{
Packit Service fa4841
		switch (wParam)
Packit Service fa4841
		{
Packit Service fa4841
			case WM_KEYDOWN:
Packit Service fa4841
			case WM_SYSKEYDOWN:
Packit Service fa4841
			case WM_KEYUP:
Packit Service fa4841
			case WM_SYSKEYUP:
Packit Service b1ea74
				wfc = (wfContext*)GetWindowLongPtr(g_focus_hWnd, GWLP_USERDATA);
Packit Service b1ea74
				p = (PKBDLLHOOKSTRUCT)lParam;
Packit Service fa4841
Packit Service fa4841
				if (!wfc || !p)
Packit Service fa4841
					return 1;
Packit Service fa4841
Packit Service fa4841
				input = wfc->context.input;
Packit Service b1ea74
				rdp_scancode = MAKE_RDP_SCANCODE((BYTE)p->scanCode, p->flags & LLKHF_EXTENDED);
Packit Service fa4841
				DEBUG_KBD("keydown %d scanCode 0x%08lX flags 0x%08lX vkCode 0x%08lX",
Packit Service fa4841
				          (wParam == WM_KEYDOWN), p->scanCode, p->flags, p->vkCode);
Packit Service fa4841
Packit Service b1ea74
				if (wfc->fullscreen_toggle &&
Packit Service fa4841
				    ((p->vkCode == VK_RETURN) || (p->vkCode == VK_CANCEL)) &&
Packit Service fa4841
				    (GetAsyncKeyState(VK_CONTROL) & 0x8000) &&
Packit Service fa4841
				    (GetAsyncKeyState(VK_MENU) & 0x8000)) /* could also use flags & LLKHF_ALTDOWN */
Packit Service fa4841
				{
Packit Service fa4841
					if (wParam == WM_KEYDOWN)
Packit Service fa4841
					{
Packit Service fa4841
						wf_toggle_fullscreen(wfc);
Packit Service fa4841
						return 1;
Packit Service fa4841
					}
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				if (rdp_scancode == RDP_SCANCODE_NUMLOCK_EXTENDED)
Packit Service fa4841
				{
Packit Service fa4841
					/* Windows sends NumLock as extended - rdp doesn't */
Packit Service fa4841
					DEBUG_KBD("hack: NumLock (x45) should not be extended");
Packit Service fa4841
					rdp_scancode = RDP_SCANCODE_NUMLOCK;
Packit Service fa4841
				}
Packit Service fa4841
				else if (rdp_scancode == RDP_SCANCODE_NUMLOCK)
Packit Service fa4841
				{
Packit Service fa4841
					/* Windows sends Pause as if it was a RDP NumLock (handled above).
Packit Service b1ea74
					 * It must however be sent as a one-shot Ctrl+NumLock */
Packit Service fa4841
					if (wParam == WM_KEYDOWN)
Packit Service fa4841
					{
Packit Service fa4841
						DEBUG_KBD("Pause, sent as Ctrl+NumLock");
Packit Service fa4841
						freerdp_input_send_keyboard_event_ex(input, TRUE, RDP_SCANCODE_LCONTROL);
Packit Service fa4841
						freerdp_input_send_keyboard_event_ex(input, TRUE, RDP_SCANCODE_NUMLOCK);
Packit Service fa4841
						freerdp_input_send_keyboard_event_ex(input, FALSE, RDP_SCANCODE_LCONTROL);
Packit Service fa4841
						freerdp_input_send_keyboard_event_ex(input, FALSE, RDP_SCANCODE_NUMLOCK);
Packit Service fa4841
					}
Packit Service fa4841
					else
Packit Service fa4841
					{
Packit Service fa4841
						DEBUG_KBD("Pause up");
Packit Service fa4841
					}
Packit Service fa4841
Packit Service fa4841
					return 1;
Packit Service fa4841
				}
Packit Service fa4841
				else if (rdp_scancode == RDP_SCANCODE_RSHIFT_EXTENDED)
Packit Service fa4841
				{
Packit Service fa4841
					DEBUG_KBD("right shift (x36) should not be extended");
Packit Service fa4841
					rdp_scancode = RDP_SCANCODE_RSHIFT;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service b1ea74
				freerdp_input_send_keyboard_event_ex(input, !(p->flags & LLKHF_UP), rdp_scancode);
Packit Service fa4841
Packit Service b1ea74
				if (p->vkCode == VK_NUMLOCK || p->vkCode == VK_CAPITAL || p->vkCode == VK_SCROLL ||
Packit Service b1ea74
				    p->vkCode == VK_KANA)
Packit Service b1ea74
					DEBUG_KBD(
Packit Service b1ea74
					    "lock keys are processed on client side too to toggle their indicators");
Packit Service fa4841
				else
Packit Service fa4841
					return 1;
Packit Service fa4841
Packit Service fa4841
				break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (g_flipping_out)
Packit Service fa4841
	{
Packit Service fa4841
		if (!alt_ctrl_down())
Packit Service fa4841
		{
Packit Service fa4841
			g_flipping_out = FALSE;
Packit Service fa4841
			g_focus_hWnd = NULL;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return CallNextHookEx(NULL, nCode, wParam, lParam);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void wf_event_focus_in(wfContext* wfc)
Packit Service fa4841
{
Packit Service fa4841
	UINT16 syncFlags;
Packit Service fa4841
	rdpInput* input;
Packit Service fa4841
	POINT pt;
Packit Service fa4841
	RECT rc;
Packit Service fa4841
	input = wfc->context.input;
Packit Service fa4841
	syncFlags = 0;
Packit Service fa4841
Packit Service fa4841
	if (GetKeyState(VK_NUMLOCK))
Packit Service fa4841
		syncFlags |= KBD_SYNC_NUM_LOCK;
Packit Service fa4841
Packit Service fa4841
	if (GetKeyState(VK_CAPITAL))
Packit Service fa4841
		syncFlags |= KBD_SYNC_CAPS_LOCK;
Packit Service fa4841
Packit Service fa4841
	if (GetKeyState(VK_SCROLL))
Packit Service fa4841
		syncFlags |= KBD_SYNC_SCROLL_LOCK;
Packit Service fa4841
Packit Service fa4841
	if (GetKeyState(VK_KANA))
Packit Service fa4841
		syncFlags |= KBD_SYNC_KANA_LOCK;
Packit Service fa4841
Packit Service fa4841
	input->FocusInEvent(input, syncFlags);
Packit Service fa4841
	/* send pointer position if the cursor is currently inside our client area */
Packit Service fa4841
	GetCursorPos(&pt;;
Packit Service fa4841
	ScreenToClient(wfc->hwnd, &pt;;
Packit Service fa4841
	GetClientRect(wfc->hwnd, &rc);
Packit Service fa4841
Packit Service fa4841
	if (pt.x >= rc.left && pt.x < rc.right && pt.y >= rc.top && pt.y < rc.bottom)
Packit Service fa4841
		input->MouseEvent(input, PTR_FLAGS_MOVE, (UINT16)pt.x, (UINT16)pt.y);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static BOOL wf_event_process_WM_MOUSEWHEEL(wfContext* wfc, HWND hWnd, UINT Msg, WPARAM wParam,
Packit Service b1ea74
                                           LPARAM lParam, BOOL horizontal, UINT16 x, UINT16 y)
Packit Service fa4841
{
Packit Service fa4841
	int delta;
Packit Service fa4841
	UINT16 flags = 0;
Packit Service fa4841
	rdpInput* input;
Packit Service fa4841
	DefWindowProc(hWnd, Msg, wParam, lParam);
Packit Service fa4841
	input = wfc->context.input;
Packit Service b1ea74
	delta = ((signed short)HIWORD(wParam)); /* GET_WHEEL_DELTA_WPARAM(wParam); */
Packit Service fa4841
Packit Service fa4841
	if (horizontal)
Packit Service fa4841
		flags |= PTR_FLAGS_HWHEEL;
Packit Service fa4841
	else
Packit Service fa4841
		flags |= PTR_FLAGS_WHEEL;
Packit Service fa4841
Packit Service fa4841
	if (delta < 0)
Packit Service fa4841
	{
Packit Service fa4841
		flags |= PTR_FLAGS_WHEEL_NEGATIVE;
Packit Service fa4841
		delta = -delta;
Packit Service fa4841
	}
Packit Service bb5c11
Packit Service b1ea74
	flags |= delta;
Packit Service fa4841
	return wf_scale_mouse_event(wfc, input, flags, x, y);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void wf_sizing(wfContext* wfc, WPARAM wParam, LPARAM lParam)
Packit Service fa4841
{
Packit Service fa4841
	rdpSettings* settings = wfc->context.settings;
Packit Service fa4841
	// Holding the CTRL key down while resizing the window will force the desktop aspect ratio.
Packit Service fa4841
	LPRECT rect;
Packit Service fa4841
Packit Service fa4841
	if (settings->SmartSizing && (GetAsyncKeyState(VK_CONTROL) & 0x8000))
Packit Service fa4841
	{
Packit Service b1ea74
		rect = (LPRECT)wParam;
Packit Service fa4841
Packit Service fa4841
		switch (lParam)
Packit Service fa4841
		{
Packit Service fa4841
			case WMSZ_LEFT:
Packit Service fa4841
			case WMSZ_RIGHT:
Packit Service fa4841
			case WMSZ_BOTTOMRIGHT:
Packit Service fa4841
				// Adjust height
Packit Service b1ea74
				rect->bottom = rect->top + settings->DesktopHeight * (rect->right - rect->left) /
Packit Service b1ea74
				                               settings->DesktopWidth;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WMSZ_TOP:
Packit Service fa4841
			case WMSZ_BOTTOM:
Packit Service fa4841
			case WMSZ_TOPRIGHT:
Packit Service fa4841
				// Adjust width
Packit Service fa4841
				rect->right = rect->left + settings->DesktopWidth * (rect->bottom - rect->top) /
Packit Service b1ea74
				                               settings->DesktopHeight;
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WMSZ_BOTTOMLEFT:
Packit Service fa4841
			case WMSZ_TOPLEFT:
Packit Service fa4841
				// adjust width
Packit Service b1ea74
				rect->left = rect->right - (settings->DesktopWidth * (rect->bottom - rect->top) /
Packit Service b1ea74
				                            settings->DesktopHeight);
Packit Service fa4841
				break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
LRESULT CALLBACK wf_event_proc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
Packit Service fa4841
{
Packit Service fa4841
	HDC hdc;
Packit Service fa4841
	LONG_PTR ptr;
Packit Service fa4841
	wfContext* wfc;
Packit Service fa4841
	int x, y, w, h;
Packit Service fa4841
	PAINTSTRUCT ps;
Packit Service fa4841
	BOOL processed;
Packit Service fa4841
	RECT windowRect;
Packit Service fa4841
	MINMAXINFO* minmax;
Packit Service fa4841
	SCROLLINFO si;
Packit Service fa4841
	processed = TRUE;
Packit Service fa4841
	ptr = GetWindowLongPtr(hWnd, GWLP_USERDATA);
Packit Service b1ea74
	wfc = (wfContext*)ptr;
Packit Service fa4841
Packit Service fa4841
	if (wfc != NULL)
Packit Service fa4841
	{
Packit Service fa4841
		rdpInput* input = wfc->context.input;
Packit Service fa4841
		rdpSettings* settings = wfc->context.settings;
Packit Service fa4841
Packit Service fa4841
		switch (Msg)
Packit Service fa4841
		{
Packit Service fa4841
			case WM_MOVE:
Packit Service fa4841
				if (!wfc->disablewindowtracking)
Packit Service fa4841
				{
Packit Service b1ea74
					int x = (int)(short)LOWORD(lParam);
Packit Service b1ea74
					int y = (int)(short)HIWORD(lParam);
Packit Service fa4841
					wfc->client_x = x;
Packit Service fa4841
					wfc->client_y = y;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_GETMINMAXINFO:
Packit Service fa4841
				if (wfc->context.settings->SmartSizing)
Packit Service fa4841
				{
Packit Service fa4841
					processed = FALSE;
Packit Service fa4841
				}
Packit Service fa4841
				else
Packit Service fa4841
				{
Packit Service fa4841
					// Set maximum window size for resizing
Packit Service b1ea74
					minmax = (MINMAXINFO*)lParam;
Packit Service fa4841
Packit Service b1ea74
					// always use the last determined canvas diff, because it could be
Packit Service b1ea74
					// that the window is minimized when this gets called
Packit Service b1ea74
					// wf_update_canvas_diff(wfc);
Packit Service fa4841
Packit Service fa4841
					if (!wfc->fullscreen)
Packit Service fa4841
					{
Packit Service fa4841
						// add window decoration
Packit Service fa4841
						minmax->ptMaxTrackSize.x = settings->DesktopWidth + wfc->diff.x;
Packit Service fa4841
						minmax->ptMaxTrackSize.y = settings->DesktopHeight + wfc->diff.y;
Packit Service fa4841
					}
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_SIZING:
Packit Service fa4841
				wf_sizing(wfc, lParam, wParam);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_SIZE:
Packit Service fa4841
				GetWindowRect(wfc->hwnd, &windowRect);
Packit Service fa4841
Packit Service fa4841
				if (!wfc->fullscreen)
Packit Service fa4841
				{
Packit Service fa4841
					wfc->client_width = LOWORD(lParam);
Packit Service fa4841
					wfc->client_height = HIWORD(lParam);
Packit Service fa4841
					wfc->client_x = windowRect.left;
Packit Service fa4841
					wfc->client_y = windowRect.top;
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				if (wfc->client_width && wfc->client_height)
Packit Service fa4841
				{
Packit Service fa4841
					wf_size_scrollbars(wfc, LOWORD(lParam), HIWORD(lParam));
Packit Service fa4841
Packit Service b1ea74
					// Workaround: when the window is maximized, the call to "ShowScrollBars"
Packit Service b1ea74
					// returns TRUE but has no effect.
Packit Service fa4841
					if (wParam == SIZE_MAXIMIZED && !wfc->fullscreen)
Packit Service fa4841
						SetWindowPos(wfc->hwnd, HWND_TOP, 0, 0, windowRect.right - windowRect.left,
Packit Service b1ea74
						             windowRect.bottom - windowRect.top,
Packit Service b1ea74
						             SWP_NOMOVE | SWP_FRAMECHANGED);
Packit Service fa4841
				}
Packit Service fa4841
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_EXITSIZEMOVE:
Packit Service fa4841
				wf_size_scrollbars(wfc, wfc->client_width, wfc->client_height);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_ERASEBKGND:
Packit Service fa4841
				/* Say we handled it - prevents flickering */
Packit Service b1ea74
				return (LRESULT)1;
Packit Service fa4841
Packit Service fa4841
			case WM_PAINT:
Packit Service fa4841
				hdc = BeginPaint(hWnd, &ps);
Packit Service fa4841
				x = ps.rcPaint.left;
Packit Service fa4841
				y = ps.rcPaint.top;
Packit Service fa4841
				w = ps.rcPaint.right - ps.rcPaint.left + 1;
Packit Service fa4841
				h = ps.rcPaint.bottom - ps.rcPaint.top + 1;
Packit Service fa4841
				wf_scale_blt(wfc, hdc, x, y, w, h, wfc->primary->hdc,
Packit Service fa4841
				             x - wfc->offset_x + wfc->xCurrentScroll,
Packit Service fa4841
				             y - wfc->offset_y + wfc->yCurrentScroll, SRCCOPY);
Packit Service fa4841
				EndPaint(hWnd, &ps);
Packit Service fa4841
				break;
Packit Service bb5c11
#if (_WIN32_WINNT >= 0x0500)
Packit Service b1ea74
Packit Service fa4841
			case WM_XBUTTONDOWN:
Packit Service fa4841
				wf_scale_mouse_event_ex(wfc, input, PTR_XFLAGS_DOWN, GET_XBUTTON_WPARAM(wParam),
Packit Service b1ea74
				                        X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                        Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_XBUTTONUP:
Packit Service fa4841
				wf_scale_mouse_event_ex(wfc, input, 0, GET_XBUTTON_WPARAM(wParam),
Packit Service b1ea74
				                        X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                        Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
			case WM_MBUTTONDOWN:
Packit Service fa4841
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_DOWN | PTR_FLAGS_BUTTON3,
Packit Service fa4841
				                     X_POS(lParam) - wfc->offset_x, Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_MBUTTONUP:
Packit Service b1ea74
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_BUTTON3, X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                     Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_LBUTTONDOWN:
Packit Service fa4841
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_DOWN | PTR_FLAGS_BUTTON1,
Packit Service fa4841
				                     X_POS(lParam) - wfc->offset_x, Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_LBUTTONUP:
Packit Service b1ea74
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_BUTTON1, X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                     Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_RBUTTONDOWN:
Packit Service fa4841
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_DOWN | PTR_FLAGS_BUTTON2,
Packit Service fa4841
				                     X_POS(lParam) - wfc->offset_x, Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_RBUTTONUP:
Packit Service b1ea74
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_BUTTON2, X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                     Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_MOUSEMOVE:
Packit Service fa4841
				wf_scale_mouse_event(wfc, input, PTR_FLAGS_MOVE, X_POS(lParam) - wfc->offset_x,
Packit Service fa4841
				                     Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service bb5c11
#if (_WIN32_WINNT >= 0x0400) || (_WIN32_WINDOWS > 0x0400)
Packit Service b1ea74
Packit Service fa4841
			case WM_MOUSEWHEEL:
Packit Service fa4841
				wf_event_process_WM_MOUSEWHEEL(wfc, hWnd, Msg, wParam, lParam, FALSE,
Packit Service b1ea74
				                               X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                               Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
#endif
Packit Service bb5c11
#if (_WIN32_WINNT >= 0x0600)
Packit Service b1ea74
Packit Service fa4841
			case WM_MOUSEHWHEEL:
Packit Service fa4841
				wf_event_process_WM_MOUSEWHEEL(wfc, hWnd, Msg, wParam, lParam, TRUE,
Packit Service b1ea74
				                               X_POS(lParam) - wfc->offset_x,
Packit Service b1ea74
				                               Y_POS(lParam) - wfc->offset_y);
Packit Service fa4841
				break;
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
			case WM_SETCURSOR:
Packit Service fa4841
				if (LOWORD(lParam) == HTCLIENT)
Packit Service fa4841
					SetCursor(wfc->cursor);
Packit Service fa4841
				else
Packit Service fa4841
					DefWindowProc(hWnd, Msg, wParam, lParam);
Packit Service fa4841
Packit Service fa4841
				break;
Packit Service fa4841
Packit Service fa4841
			case WM_HSCROLL:
Packit Service b1ea74
			{
Packit Service b1ea74
				int xDelta;  // xDelta = new_pos - current_pos
Packit Service b1ea74
				int xNewPos; // new position
Packit Service b1ea74
				int yDelta = 0;
Packit Service b1ea74
Packit Service b1ea74
				switch (LOWORD(wParam))
Packit Service fa4841
				{
Packit Service b1ea74
					// User clicked the scroll bar shaft left of the scroll box.
Packit Service b1ea74
					case SB_PAGEUP:
Packit Service b1ea74
						xNewPos = wfc->xCurrentScroll - 50;
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// User clicked the scroll bar shaft right of the scroll box.
Packit Service b1ea74
					case SB_PAGEDOWN:
Packit Service b1ea74
						xNewPos = wfc->xCurrentScroll + 50;
Packit Service b1ea74
						break;
Packit Service b1ea74
Packit Service b1ea74
					// User clicked the left arrow.
Packit Service b1ea74
					case SB_LINEUP:
Packit Service b1ea74
						xNewPos = wfc->xCurrentScroll - 5;
Packit Service b1ea74
						break;
Packit Service b1ea74
Packit Service b1ea74
					// User clicked the right arrow.
Packit Service b1ea74
					case SB_LINEDOWN:
Packit Service b1ea74
						xNewPos = wfc->xCurrentScroll + 5;
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// User dragged the scroll box.
Packit Service b1ea74
					case SB_THUMBPOSITION:
Packit Service b1ea74
						xNewPos = HIWORD(wParam);
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// user is dragging the scrollbar
Packit Service b1ea74
					case SB_THUMBTRACK:
Packit Service b1ea74
						xNewPos = HIWORD(wParam);
Packit Service fa4841
						break;
Packit Service fa4841
Packit Service b1ea74
					default:
Packit Service b1ea74
						xNewPos = wfc->xCurrentScroll;
Packit Service fa4841
				}
Packit Service b1ea74
Packit Service b1ea74
				// New position must be between 0 and the screen width.
Packit Service b1ea74
				xNewPos = MAX(0, xNewPos);
Packit Service b1ea74
				xNewPos = MIN(wfc->xMaxScroll, xNewPos);
Packit Service b1ea74
Packit Service b1ea74
				// If the current position does not change, do not scroll.
Packit Service b1ea74
				if (xNewPos == wfc->xCurrentScroll)
Packit Service b1ea74
					break;
Packit Service b1ea74
Packit Service b1ea74
				// Determine the amount scrolled (in pixels).
Packit Service b1ea74
				xDelta = xNewPos - wfc->xCurrentScroll;
Packit Service b1ea74
				// Reset the current scroll position.
Packit Service b1ea74
				wfc->xCurrentScroll = xNewPos;
Packit Service b1ea74
				// Scroll the window. (The system repaints most of the
Packit Service b1ea74
				// client area when ScrollWindowEx is called; however, it is
Packit Service b1ea74
				// necessary to call UpdateWindow in order to repaint the
Packit Service b1ea74
				// rectangle of pixels that were invalidated.)
Packit Service b1ea74
				ScrollWindowEx(wfc->hwnd, -xDelta, -yDelta, (CONST RECT*)NULL, (CONST RECT*)NULL,
Packit Service b1ea74
				               (HRGN)NULL, (PRECT)NULL, SW_INVALIDATE);
Packit Service b1ea74
				UpdateWindow(wfc->hwnd);
Packit Service b1ea74
				// Reset the scroll bar.
Packit Service b1ea74
				si.cbSize = sizeof(si);
Packit Service b1ea74
				si.fMask = SIF_POS;
Packit Service b1ea74
				si.nPos = wfc->xCurrentScroll;
Packit Service b1ea74
				SetScrollInfo(wfc->hwnd, SB_HORZ, &si, TRUE);
Packit Service b1ea74
			}
Packit Service b1ea74
			break;
Packit Service fa4841
Packit Service fa4841
			case WM_VSCROLL:
Packit Service b1ea74
			{
Packit Service b1ea74
				int xDelta = 0;
Packit Service b1ea74
				int yDelta;  // yDelta = new_pos - current_pos
Packit Service b1ea74
				int yNewPos; // new position
Packit Service b1ea74
Packit Service b1ea74
				switch (LOWORD(wParam))
Packit Service fa4841
				{
Packit Service b1ea74
					// User clicked the scroll bar shaft above the scroll box.
Packit Service b1ea74
					case SB_PAGEUP:
Packit Service b1ea74
						yNewPos = wfc->yCurrentScroll - 50;
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// User clicked the scroll bar shaft below the scroll box.
Packit Service b1ea74
					case SB_PAGEDOWN:
Packit Service b1ea74
						yNewPos = wfc->yCurrentScroll + 50;
Packit Service b1ea74
						break;
Packit Service b1ea74
Packit Service b1ea74
					// User clicked the top arrow.
Packit Service b1ea74
					case SB_LINEUP:
Packit Service b1ea74
						yNewPos = wfc->yCurrentScroll - 5;
Packit Service b1ea74
						break;
Packit Service b1ea74
Packit Service b1ea74
					// User clicked the bottom arrow.
Packit Service b1ea74
					case SB_LINEDOWN:
Packit Service b1ea74
						yNewPos = wfc->yCurrentScroll + 5;
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// User dragged the scroll box.
Packit Service b1ea74
					case SB_THUMBPOSITION:
Packit Service b1ea74
						yNewPos = HIWORD(wParam);
Packit Service b1ea74
						break;
Packit Service fa4841
Packit Service b1ea74
					// user is dragging the scrollbar
Packit Service b1ea74
					case SB_THUMBTRACK:
Packit Service b1ea74
						yNewPos = HIWORD(wParam);
Packit Service fa4841
						break;
Packit Service fa4841
Packit Service b1ea74
					default:
Packit Service b1ea74
						yNewPos = wfc->yCurrentScroll;
Packit Service fa4841
				}
Packit Service b1ea74
Packit Service b1ea74
				// New position must be between 0 and the screen height.
Packit Service b1ea74
				yNewPos = MAX(0, yNewPos);
Packit Service b1ea74
				yNewPos = MIN(wfc->yMaxScroll, yNewPos);
Packit Service b1ea74
Packit Service b1ea74
				// If the current position does not change, do not scroll.
Packit Service b1ea74
				if (yNewPos == wfc->yCurrentScroll)
Packit Service b1ea74
					break;
Packit Service b1ea74
Packit Service b1ea74
				// Determine the amount scrolled (in pixels).
Packit Service b1ea74
				yDelta = yNewPos - wfc->yCurrentScroll;
Packit Service b1ea74
				// Reset the current scroll position.
Packit Service b1ea74
				wfc->yCurrentScroll = yNewPos;
Packit Service b1ea74
				// Scroll the window. (The system repaints most of the
Packit Service b1ea74
				// client area when ScrollWindowEx is called; however, it is
Packit Service b1ea74
				// necessary to call UpdateWindow in order to repaint the
Packit Service b1ea74
				// rectangle of pixels that were invalidated.)
Packit Service b1ea74
				ScrollWindowEx(wfc->hwnd, -xDelta, -yDelta, (CONST RECT*)NULL, (CONST RECT*)NULL,
Packit Service b1ea74
				               (HRGN)NULL, (PRECT)NULL, SW_INVALIDATE);
Packit Service b1ea74
				UpdateWindow(wfc->hwnd);
Packit Service b1ea74
				// Reset the scroll bar.
Packit Service b1ea74
				si.cbSize = sizeof(si);
Packit Service b1ea74
				si.fMask = SIF_POS;
Packit Service b1ea74
				si.nPos = wfc->yCurrentScroll;
Packit Service b1ea74
				SetScrollInfo(wfc->hwnd, SB_VERT, &si, TRUE);
Packit Service b1ea74
			}
Packit Service b1ea74
			break;
Packit Service fa4841
Packit Service fa4841
			case WM_SYSCOMMAND:
Packit Service b1ea74
			{
Packit Service b1ea74
				if (wParam == SYSCOMMAND_ID_SMARTSIZING)
Packit Service fa4841
				{
Packit Service b1ea74
					HMENU hMenu = GetSystemMenu(wfc->hwnd, FALSE);
Packit Service b1ea74
					freerdp_set_param_bool(wfc->context.settings, FreeRDP_SmartSizing,
Packit Service b1ea74
					                       !wfc->context.settings->SmartSizing);
Packit Service b1ea74
					CheckMenuItem(hMenu, SYSCOMMAND_ID_SMARTSIZING,
Packit Service b1ea74
					              wfc->context.settings->SmartSizing ? MF_CHECKED : MF_UNCHECKED);
Packit Service fa4841
				}
Packit Service b1ea74
				else
Packit Service b1ea74
				{
Packit Service b1ea74
					processed = FALSE;
Packit Service b1ea74
				}
Packit Service b1ea74
			}
Packit Service b1ea74
			break;
Packit Service fa4841
Packit Service fa4841
			default:
Packit Service fa4841
				processed = FALSE;
Packit Service fa4841
				break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		processed = FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (processed)
Packit Service fa4841
		return 0;
Packit Service fa4841
Packit Service fa4841
	switch (Msg)
Packit Service fa4841
	{
Packit Service fa4841
		case WM_DESTROY:
Packit Service fa4841
			PostQuitMessage(WM_QUIT);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WM_SETFOCUS:
Packit Service fa4841
			DEBUG_KBD("getting focus %X", hWnd);
Packit Service fa4841
Packit Service fa4841
			if (alt_ctrl_down())
Packit Service fa4841
				g_flipping_in = TRUE;
Packit Service fa4841
Packit Service fa4841
			g_focus_hWnd = hWnd;
Packit Service fa4841
			freerdp_set_focus(wfc->context.instance);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WM_KILLFOCUS:
Packit Service fa4841
			if (g_focus_hWnd == hWnd && wfc && !wfc->fullscreen)
Packit Service fa4841
			{
Packit Service fa4841
				DEBUG_KBD("loosing focus %X", hWnd);
Packit Service fa4841
Packit Service fa4841
				if (alt_ctrl_down())
Packit Service fa4841
					g_flipping_out = TRUE;
Packit Service fa4841
				else
Packit Service fa4841
					g_focus_hWnd = NULL;
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WM_ACTIVATE:
Packit Service b1ea74
		{
Packit Service b1ea74
			int activate = (int)(short)LOWORD(wParam);
Packit Service fa4841
Packit Service b1ea74
			if (activate != WA_INACTIVE)
Packit Service b1ea74
			{
Packit Service b1ea74
				if (alt_ctrl_down())
Packit Service b1ea74
					g_flipping_in = TRUE;
Packit Service bb5c11
Packit Service b1ea74
				g_focus_hWnd = hWnd;
Packit Service b1ea74
			}
Packit Service b1ea74
			else
Packit Service b1ea74
			{
Packit Service b1ea74
				if (alt_ctrl_down())
Packit Service b1ea74
					g_flipping_out = TRUE;
Packit Service fa4841
				else
Packit Service b1ea74
					g_focus_hWnd = NULL;
Packit Service fa4841
			}
Packit Service b1ea74
		}
Packit Service fa4841
Packit Service fa4841
		default:
Packit Service fa4841
			return DefWindowProc(hWnd, Msg, wParam, lParam);
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return 0;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL wf_scale_blt(wfContext* wfc, HDC hdc, int x, int y, int w, int h, HDC hdcSrc, int x1, int y1,
Packit Service b1ea74
                  DWORD rop)
Packit Service fa4841
{
Packit Service fa4841
	rdpSettings* settings;
Packit Service fa4841
	UINT32 ww, wh, dw, dh;
Packit Service fa4841
	settings = wfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (!wfc->client_width)
Packit Service fa4841
		wfc->client_width = settings->DesktopWidth;
Packit Service fa4841
Packit Service fa4841
	if (!wfc->client_height)
Packit Service fa4841
		wfc->client_height = settings->DesktopHeight;
Packit Service fa4841
Packit Service fa4841
	ww = wfc->client_width;
Packit Service fa4841
	wh = wfc->client_height;
Packit Service fa4841
	dw = settings->DesktopWidth;
Packit Service fa4841
	dh = settings->DesktopHeight;
Packit Service fa4841
Packit Service fa4841
	if (!ww)
Packit Service fa4841
		ww = dw;
Packit Service fa4841
Packit Service fa4841
	if (!wh)
Packit Service fa4841
		wh = dh;
Packit Service fa4841
Packit Service b1ea74
	if (wfc->fullscreen || !wfc->context.settings->SmartSizing || (ww == dw && wh == dh))
Packit Service fa4841
	{
Packit Service fa4841
		return BitBlt(hdc, x, y, w, h, wfc->primary->hdc, x1, y1, SRCCOPY);
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		SetStretchBltMode(hdc, HALFTONE);
Packit Service fa4841
		SetBrushOrgEx(hdc, 0, 0, NULL);
Packit Service fa4841
		return StretchBlt(hdc, 0, 0, ww, wh, wfc->primary->hdc, 0, 0, dw, dh, SRCCOPY);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static BOOL wf_scale_mouse_pos(wfContext* wfc, UINT16* x, UINT16* y)
Packit Service fa4841
{
Packit Service fa4841
	int ww, wh, dw, dh;
Packit Service fa4841
	rdpContext* context;
Packit Service fa4841
	rdpSettings* settings;
Packit Service b1ea74
Packit Service fa4841
	if (!wfc || !x || !y)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	settings = wfc->context.settings;
Packit Service b1ea74
Packit Service fa4841
	if (!settings)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (!wfc->client_width)
Packit Service fa4841
		wfc->client_width = settings->DesktopWidth;
Packit Service fa4841
Packit Service fa4841
	if (!wfc->client_height)
Packit Service fa4841
		wfc->client_height = settings->DesktopHeight;
Packit Service fa4841
Packit Service fa4841
	ww = wfc->client_width;
Packit Service fa4841
	wh = wfc->client_height;
Packit Service fa4841
	dw = settings->DesktopWidth;
Packit Service fa4841
	dh = settings->DesktopHeight;
Packit Service fa4841
Packit Service fa4841
	if (!settings->SmartSizing || ((ww == dw) && (wh == dh)))
Packit Service fa4841
	{
Packit Service fa4841
		*x += wfc->xCurrentScroll;
Packit Service fa4841
		*y += wfc->yCurrentScroll;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		*x = *x * dw / ww + wfc->xCurrentScroll;
Packit Service fa4841
		*y = *y * dh / wh + wfc->yCurrentScroll;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static BOOL wf_scale_mouse_event(wfContext* wfc, rdpInput* input, UINT16 flags, UINT16 x, UINT16 y)
Packit Service fa4841
{
Packit Service fa4841
	MouseEventEventArgs eventArgs;
Packit Service fa4841
Packit Service fa4841
	if (!wf_scale_mouse_pos(wfc, &x, &y))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (freerdp_input_send_mouse_event(input, flags, x, y))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	eventArgs.flags = flags;
Packit Service fa4841
	eventArgs.x = x;
Packit Service fa4841
	eventArgs.y = y;
Packit Service fa4841
	PubSub_OnMouseEvent(wfc->context.pubSub, &wfc->context, &eventArgs);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
#if (_WIN32_WINNT >= 0x0500)
Packit Service b1ea74
static BOOL wf_scale_mouse_event_ex(wfContext* wfc, rdpInput* input, UINT16 flags,
Packit Service b1ea74
                                    UINT16 buttonMask, UINT16 x, UINT16 y)
Packit Service fa4841
{
Packit Service fa4841
	MouseEventExEventArgs eventArgs;
Packit Service fa4841
Packit Service fa4841
	if (buttonMask & XBUTTON1)
Packit Service fa4841
		flags |= PTR_XFLAGS_BUTTON1;
Packit Service b1ea74
Packit Service fa4841
	if (buttonMask & XBUTTON2)
Packit Service fa4841
		flags |= PTR_XFLAGS_BUTTON2;
Packit Service fa4841
Packit Service fa4841
	if (!wf_scale_mouse_pos(wfc, &x, &y))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (freerdp_input_send_extended_mouse_event(input, flags, x, y))
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	eventArgs.flags = flags;
Packit Service fa4841
	eventArgs.x = x;
Packit Service fa4841
	eventArgs.y = y;
Packit Service fa4841
	PubSub_OnMouseEventEx(wfc->context.pubSub, &wfc->context, &eventArgs);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
#endif