Blame client/X11/xf_window.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * X11 Windows
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 * Copyright 2012 HP Development Company, LLC
Packit Service fa4841
 * Copyright 2016 Thincast Technologies GmbH
Packit Service fa4841
 * Copyright 2016 Armin Novak <armin.novak@thincast.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 <stdarg.h>
Packit Service fa4841
#include <unistd.h>
Packit Service fa4841
#include <sys/types.h>
Packit Service fa4841
Packit Service fa4841
#include <X11/Xlib.h>
Packit Service fa4841
#include <X11/Xutil.h>
Packit Service fa4841
#include <X11/Xatom.h>
Packit Service fa4841
Packit Service fa4841
#include <sys/mman.h>
Packit Service fa4841
#include <sys/stat.h>
Packit Service fa4841
#include <fcntl.h>
Packit Service fa4841
Packit Service fa4841
#include <winpr/thread.h>
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
#include <winpr/string.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/rail.h>
Packit Service fa4841
#include <freerdp/log.h>
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XEXT
Packit Service fa4841
#include <X11/extensions/shape.h>
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XI
Packit Service fa4841
#include <X11/extensions/XInput2.h>
Packit Service fa4841
#include "xf_input.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include "xf_rail.h"
Packit Service fa4841
#include "xf_input.h"
Packit Service fa4841
Packit Service fa4841
#define TAG CLIENT_TAG("x11")
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_DEBUG_X11
Packit Service fa4841
#define DEBUG_X11(...) WLog_DBG(TAG, __VA_ARGS__)
Packit Service fa4841
#else
Packit Service b1ea74
#define DEBUG_X11(...) \
Packit Service b1ea74
	do                 \
Packit Service b1ea74
	{                  \
Packit Service b1ea74
	} while (0)
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include "FreeRDP_Icon_256px.h"
Packit Service fa4841
#define xf_icon_prop FreeRDP_Icon_256px_prop
Packit Service fa4841
Packit Service fa4841
#include "xf_window.h"
Packit Service fa4841
Packit Service fa4841
/* Extended Window Manager Hints: http://standards.freedesktop.org/wm-spec/wm-spec-1.3.html */
Packit Service fa4841
Packit Service fa4841
/* bit definitions for MwmHints.flags */
Packit Service b1ea74
#define MWM_HINTS_FUNCTIONS (1L << 0)
Packit Service b1ea74
#define MWM_HINTS_DECORATIONS (1L << 1)
Packit Service b1ea74
#define MWM_HINTS_INPUT_MODE (1L << 2)
Packit Service b1ea74
#define MWM_HINTS_STATUS (1L << 3)
Packit Service fa4841
Packit Service fa4841
/* bit definitions for MwmHints.functions */
Packit Service b1ea74
#define MWM_FUNC_ALL (1L << 0)
Packit Service b1ea74
#define MWM_FUNC_RESIZE (1L << 1)
Packit Service b1ea74
#define MWM_FUNC_MOVE (1L << 2)
Packit Service b1ea74
#define MWM_FUNC_MINIMIZE (1L << 3)
Packit Service b1ea74
#define MWM_FUNC_MAXIMIZE (1L << 4)
Packit Service b1ea74
#define MWM_FUNC_CLOSE (1L << 5)
Packit Service fa4841
Packit Service fa4841
/* bit definitions for MwmHints.decorations */
Packit Service b1ea74
#define MWM_DECOR_ALL (1L << 0)
Packit Service b1ea74
#define MWM_DECOR_BORDER (1L << 1)
Packit Service b1ea74
#define MWM_DECOR_RESIZEH (1L << 2)
Packit Service b1ea74
#define MWM_DECOR_TITLE (1L << 3)
Packit Service b1ea74
#define MWM_DECOR_MENU (1L << 4)
Packit Service b1ea74
#define MWM_DECOR_MINIMIZE (1L << 5)
Packit Service b1ea74
#define MWM_DECOR_MAXIMIZE (1L << 6)
Packit Service fa4841
Packit Service b1ea74
#define PROP_MOTIF_WM_HINTS_ELEMENTS 5
Packit Service fa4841
Packit Service fa4841
struct _PropMotifWmHints
Packit Service fa4841
{
Packit Service fa4841
	unsigned long flags;
Packit Service fa4841
	unsigned long functions;
Packit Service fa4841
	unsigned long decorations;
Packit Service fa4841
	long inputMode;
Packit Service fa4841
	unsigned long status;
Packit Service fa4841
};
Packit Service fa4841
typedef struct _PropMotifWmHints PropMotifWmHints;
Packit Service fa4841
Packit Service fa4841
static void xf_SetWindowTitleText(xfContext* xfc, Window window, const char* name)
Packit Service fa4841
{
Packit Service b1ea74
	const size_t i = strnlen(name, MAX_PATH);
Packit Service fa4841
	XStoreName(xfc->display, window, name);
Packit Service fa4841
	Atom wm_Name = xfc->_NET_WM_NAME;
Packit Service fa4841
	Atom utf8Str = xfc->UTF8_STRING;
Packit Service b1ea74
	XChangeProperty(xfc->display, window, wm_Name, utf8Str, 8, PropModeReplace,
Packit Service b1ea74
	                (const unsigned char*)name, (int)i);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Post an event from the client to the X server
Packit Service fa4841
 */
Packit Service b1ea74
void xf_SendClientEvent(xfContext* xfc, Window window, Atom atom, unsigned int numArgs, ...)
Packit Service fa4841
{
Packit Service fa4841
	XEvent xevent;
Packit Service fa4841
	unsigned int i;
Packit Service fa4841
	va_list argp;
Packit Service fa4841
	va_start(argp, numArgs);
Packit Service fa4841
	ZeroMemory(&xevent, sizeof(XEvent));
Packit Service fa4841
	xevent.xclient.type = ClientMessage;
Packit Service fa4841
	xevent.xclient.serial = 0;
Packit Service fa4841
	xevent.xclient.send_event = False;
Packit Service fa4841
	xevent.xclient.display = xfc->display;
Packit Service fa4841
	xevent.xclient.window = window;
Packit Service fa4841
	xevent.xclient.message_type = atom;
Packit Service fa4841
	xevent.xclient.format = 32;
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < numArgs; i++)
Packit Service fa4841
	{
Packit Service fa4841
		xevent.xclient.data.l[i] = va_arg(argp, int);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	DEBUG_X11("Send ClientMessage Event: wnd=0x%04lX", (unsigned long)xevent.xclient.window);
Packit Service fa4841
	XSendEvent(xfc->display, RootWindowOfScreen(xfc->screen), False,
Packit Service fa4841
	           SubstructureRedirectMask | SubstructureNotifyMask, &xevent);
Packit Service fa4841
	XSync(xfc->display, False);
Packit Service fa4841
	va_end(argp);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_SetWindowMinimized(xfContext* xfc, xfWindow* window)
Packit Service fa4841
{
Packit Service fa4841
	XIconifyWindow(xfc->display, window->handle, xfc->screen_number);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_SetWindowFullscreen(xfContext* xfc, xfWindow* window, BOOL fullscreen)
Packit Service fa4841
{
Packit Service fa4841
	UINT32 i;
Packit Service fa4841
	rdpSettings* settings = xfc->context.settings;
Packit Service fa4841
	int startX, startY;
Packit Service fa4841
	UINT32 width = window->width;
Packit Service fa4841
	UINT32 height = window->height;
Packit Service b1ea74
	/* xfc->decorations is set by caller depending on settings and whether it is fullscreen or not
Packit Service b1ea74
	 */
Packit Service fa4841
	window->decorations = xfc->decorations;
Packit Service fa4841
	/* show/hide decorations (e.g. title bar) as guided by xfc->decorations */
Packit Service fa4841
	xf_SetWindowDecorations(xfc, window->handle, window->decorations);
Packit Service fa4841
	DEBUG_X11(TAG, "X window decoration set to %d", (int)window->decorations);
Packit Service b1ea74
	xf_floatbar_toggle_fullscreen(xfc->window->floatbar, fullscreen);
Packit Service fa4841
Packit Service fa4841
	if (fullscreen)
Packit Service fa4841
	{
Packit Service fa4841
		xfc->savedWidth = xfc->window->width;
Packit Service fa4841
		xfc->savedHeight = xfc->window->height;
Packit Service fa4841
		xfc->savedPosX = xfc->window->left;
Packit Service fa4841
		xfc->savedPosY = xfc->window->top;
Packit Service fa4841
		startX = (settings->DesktopPosX != UINT32_MAX) ? settings->DesktopPosX : 0;
Packit Service fa4841
		startY = (settings->DesktopPosY != UINT32_MAX) ? settings->DesktopPosY : 0;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		width = xfc->savedWidth;
Packit Service fa4841
		height = xfc->savedHeight;
Packit Service fa4841
		startX = xfc->savedPosX;
Packit Service fa4841
		startY = xfc->savedPosY;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* Determine the x,y starting location for the fullscreen window */
Packit Service fa4841
	if (fullscreen)
Packit Service fa4841
	{
Packit Service fa4841
		/* Initialize startX and startY with reasonable values */
Packit Service fa4841
		startX = xfc->context.settings->MonitorDefArray[0].x;
Packit Service fa4841
		startY = xfc->context.settings->MonitorDefArray[0].y;
Packit Service fa4841
Packit Service fa4841
		/* Search all monitors to find the lowest startX and startY values */
Packit Service fa4841
		for (i = 0; i < xfc->context.settings->MonitorCount; i++)
Packit Service fa4841
		{
Packit Service fa4841
			startX = MIN(startX, xfc->context.settings->MonitorDefArray[i].x);
Packit Service fa4841
			startY = MIN(startY, xfc->context.settings->MonitorDefArray[i].y);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		/* Lastly apply any monitor shift(translation from remote to local coordinate system)
Packit Service fa4841
		 *  to startX and startY values
Packit Service fa4841
		 */
Packit Service fa4841
		startX += xfc->context.settings->MonitorLocalShiftX;
Packit Service fa4841
		startY += xfc->context.settings->MonitorLocalShiftY;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service b1ea74
	  It is safe to proceed with simply toogling _NET_WM_STATE_FULLSCREEN window state on the
Packit Service b1ea74
	  following conditions:
Packit Service fa4841
	       - The window manager supports multiple monitor full screen
Packit Service fa4841
	       - The user requested to use a single monitor to render the remote desktop
Packit Service fa4841
	 */
Packit Service fa4841
	if (xfc->_NET_WM_FULLSCREEN_MONITORS != None || settings->MonitorCount == 1)
Packit Service fa4841
	{
Packit Service fa4841
		xf_ResizeDesktopWindow(xfc, window, width, height);
Packit Service fa4841
Packit Service fa4841
		if (fullscreen)
Packit Service fa4841
		{
Packit Service fa4841
			/* enter full screen: move the window before adding NET_WM_STATE_FULLSCREEN */
Packit Service fa4841
			XMoveWindow(xfc->display, window->handle, startX, startY);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		/* Set the fullscreen state */
Packit Service fa4841
		xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4,
Packit Service fa4841
		                   fullscreen ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE,
Packit Service fa4841
		                   xfc->_NET_WM_STATE_FULLSCREEN, 0, 0);
Packit Service fa4841
Packit Service fa4841
		if (!fullscreen)
Packit Service fa4841
		{
Packit Service fa4841
			/* leave full screen: move the window after removing NET_WM_STATE_FULLSCREEN */
Packit Service fa4841
			XMoveWindow(xfc->display, window->handle, startX, startY);
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		/* Set monitor bounds */
Packit Service fa4841
		if (settings->MonitorCount > 1)
Packit Service fa4841
		{
Packit Service fa4841
			xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_FULLSCREEN_MONITORS, 5,
Packit Service b1ea74
			                   xfc->fullscreenMonitors.top, xfc->fullscreenMonitors.bottom,
Packit Service b1ea74
			                   xfc->fullscreenMonitors.left, xfc->fullscreenMonitors.right, 1);
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		if (fullscreen)
Packit Service fa4841
		{
Packit Service fa4841
			xf_SetWindowDecorations(xfc, window->handle, FALSE);
Packit Service fa4841
Packit Service fa4841
			if (xfc->fullscreenMonitors.top)
Packit Service fa4841
			{
Packit Service b1ea74
				xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_ADD,
Packit Service fa4841
				                   xfc->fullscreenMonitors.top, 0, 0);
Packit Service fa4841
			}
Packit Service fa4841
			else
Packit Service fa4841
			{
Packit Service fa4841
				XSetWindowAttributes xswa;
Packit Service fa4841
				xswa.override_redirect = True;
Packit Service fa4841
				XChangeWindowAttributes(xfc->display, window->handle, CWOverrideRedirect, &xswa);
Packit Service fa4841
				XRaiseWindow(xfc->display, window->handle);
Packit Service fa4841
				xswa.override_redirect = False;
Packit Service fa4841
				XChangeWindowAttributes(xfc->display, window->handle, CWOverrideRedirect, &xswa);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			/* if window is in maximized state, save and remove */
Packit Service fa4841
			if (xfc->_NET_WM_STATE_MAXIMIZED_VERT != None)
Packit Service fa4841
			{
Packit Service fa4841
				BYTE state;
Packit Service fa4841
				unsigned long nitems;
Packit Service fa4841
				unsigned long bytes;
Packit Service fa4841
				BYTE* prop;
Packit Service fa4841
Packit Service b1ea74
				if (xf_GetWindowProperty(xfc, window->handle, xfc->_NET_WM_STATE, 255, &nitems,
Packit Service b1ea74
				                         &bytes, &prop))
Packit Service fa4841
				{
Packit Service fa4841
					state = 0;
Packit Service fa4841
Packit Service fa4841
					while (nitems-- > 0)
Packit Service fa4841
					{
Packit Service b1ea74
						if (((Atom*)prop)[nitems] == xfc->_NET_WM_STATE_MAXIMIZED_VERT)
Packit Service fa4841
							state |= 0x01;
Packit Service fa4841
Packit Service b1ea74
						if (((Atom*)prop)[nitems] == xfc->_NET_WM_STATE_MAXIMIZED_HORZ)
Packit Service fa4841
							state |= 0x02;
Packit Service fa4841
					}
Packit Service fa4841
Packit Service fa4841
					if (state)
Packit Service fa4841
					{
Packit Service fa4841
						xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4,
Packit Service fa4841
						                   _NET_WM_STATE_REMOVE, xfc->_NET_WM_STATE_MAXIMIZED_VERT,
Packit Service fa4841
						                   0, 0);
Packit Service fa4841
						xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4,
Packit Service fa4841
						                   _NET_WM_STATE_REMOVE, xfc->_NET_WM_STATE_MAXIMIZED_HORZ,
Packit Service fa4841
						                   0, 0);
Packit Service fa4841
						xfc->savedMaximizedState = state;
Packit Service fa4841
					}
Packit Service fa4841
Packit Service fa4841
					XFree(prop);
Packit Service fa4841
				}
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			width = xfc->vscreen.area.right - xfc->vscreen.area.left + 1;
Packit Service fa4841
			height = xfc->vscreen.area.bottom - xfc->vscreen.area.top + 1;
Packit Service fa4841
			DEBUG_X11("X window move and resize %dx%d@%dx%d", startX, startY, width, height);
Packit Service fa4841
			xf_ResizeDesktopWindow(xfc, window, width, height);
Packit Service fa4841
			XMoveWindow(xfc->display, window->handle, startX, startY);
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			xf_SetWindowDecorations(xfc, window->handle, window->decorations);
Packit Service fa4841
			xf_ResizeDesktopWindow(xfc, window, width, height);
Packit Service fa4841
			XMoveWindow(xfc->display, window->handle, startX, startY);
Packit Service fa4841
Packit Service fa4841
			if (xfc->fullscreenMonitors.top)
Packit Service fa4841
			{
Packit Service b1ea74
				xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_REMOVE,
Packit Service fa4841
				                   xfc->fullscreenMonitors.top, 0, 0);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			/* restore maximized state, if the window was maximized before setting fullscreen */
Packit Service fa4841
			if (xfc->savedMaximizedState & 0x01)
Packit Service fa4841
			{
Packit Service b1ea74
				xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_ADD,
Packit Service b1ea74
				                   xfc->_NET_WM_STATE_MAXIMIZED_VERT, 0, 0);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			if (xfc->savedMaximizedState & 0x02)
Packit Service fa4841
			{
Packit Service b1ea74
				xf_SendClientEvent(xfc, window->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_ADD,
Packit Service b1ea74
				                   xfc->_NET_WM_STATE_MAXIMIZED_HORZ, 0, 0);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			xfc->savedMaximizedState = 0;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/* http://tronche.com/gui/x/xlib/window-information/XGetWindowProperty.html */
Packit Service fa4841
Packit Service b1ea74
BOOL xf_GetWindowProperty(xfContext* xfc, Window window, Atom property, int length,
Packit Service fa4841
                          unsigned long* nitems, unsigned long* bytes, BYTE** prop)
Packit Service fa4841
{
Packit Service fa4841
	int status;
Packit Service fa4841
	Atom actual_type;
Packit Service fa4841
	int actual_format;
Packit Service fa4841
Packit Service fa4841
	if (property == None)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	status = XGetWindowProperty(xfc->display, window, property, 0, length, False, AnyPropertyType,
Packit Service fa4841
	                            &actual_type, &actual_format, nitems, bytes, prop);
Packit Service fa4841
Packit Service fa4841
	if (status != Success)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	if (actual_type == None)
Packit Service fa4841
	{
Packit Service b1ea74
		WLog_INFO(TAG, "Property %lu does not exist", (unsigned long)property);
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL xf_GetCurrentDesktop(xfContext* xfc)
Packit Service fa4841
{
Packit Service fa4841
	BOOL status;
Packit Service fa4841
	unsigned long nitems;
Packit Service fa4841
	unsigned long bytes;
Packit Service fa4841
	unsigned char* prop;
Packit Service b1ea74
	status = xf_GetWindowProperty(xfc, DefaultRootWindow(xfc->display), xfc->_NET_CURRENT_DESKTOP,
Packit Service b1ea74
	                              1, &nitems, &bytes, &prop);
Packit Service fa4841
Packit Service fa4841
	if (!status)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	xfc->current_desktop = (int)*prop;
Packit Service fa4841
	free(prop);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
BOOL xf_GetWorkArea(xfContext* xfc)
Packit Service fa4841
{
Packit Service fa4841
	long* plong;
Packit Service fa4841
	BOOL status;
Packit Service fa4841
	unsigned long nitems;
Packit Service fa4841
	unsigned long bytes;
Packit Service fa4841
	unsigned char* prop;
Packit Service fa4841
	status = xf_GetCurrentDesktop(xfc);
Packit Service fa4841
Packit Service fa4841
	if (!status)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	status = xf_GetWindowProperty(xfc, DefaultRootWindow(xfc->display), xfc->_NET_WORKAREA, 32 * 4,
Packit Service b1ea74
	                              &nitems, &bytes, &prop);
Packit Service fa4841
Packit Service fa4841
	if (!status)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	if ((xfc->current_desktop * 4 + 3) >= (INT64)nitems)
Packit Service fa4841
	{
Packit Service fa4841
		free(prop);
Packit Service fa4841
		return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	plong = (long*)prop;
Packit Service fa4841
	xfc->workArea.x = plong[xfc->current_desktop * 4 + 0];
Packit Service fa4841
	xfc->workArea.y = plong[xfc->current_desktop * 4 + 1];
Packit Service fa4841
	xfc->workArea.width = plong[xfc->current_desktop * 4 + 2];
Packit Service fa4841
	xfc->workArea.height = plong[xfc->current_desktop * 4 + 3];
Packit Service fa4841
	free(prop);
Packit Service fa4841
	return TRUE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_SetWindowDecorations(xfContext* xfc, Window window, BOOL show)
Packit Service fa4841
{
Packit Service fa4841
	PropMotifWmHints hints;
Packit Service fa4841
	hints.decorations = (show) ? MWM_DECOR_ALL : 0;
Packit Service b1ea74
	hints.functions = MWM_FUNC_ALL;
Packit Service fa4841
	hints.flags = MWM_HINTS_DECORATIONS | MWM_HINTS_FUNCTIONS;
Packit Service fa4841
	hints.inputMode = 0;
Packit Service fa4841
	hints.status = 0;
Packit Service b1ea74
	XChangeProperty(xfc->display, window, xfc->_MOTIF_WM_HINTS, xfc->_MOTIF_WM_HINTS, 32,
Packit Service b1ea74
	                PropModeReplace, (BYTE*)&hints, PROP_MOTIF_WM_HINTS_ELEMENTS);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_SetWindowUnlisted(xfContext* xfc, Window window)
Packit Service fa4841
{
Packit Service fa4841
	Atom window_state[2];
Packit Service fa4841
	window_state[0] = xfc->_NET_WM_STATE_SKIP_PAGER;
Packit Service fa4841
	window_state[1] = xfc->_NET_WM_STATE_SKIP_TASKBAR;
Packit Service b1ea74
	XChangeProperty(xfc->display, window, xfc->_NET_WM_STATE, XA_ATOM, 32, PropModeReplace,
Packit Service b1ea74
	                (BYTE*)&window_state, 2);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static void xf_SetWindowPID(xfContext* xfc, Window window, pid_t pid)
Packit Service fa4841
{
Packit Service fa4841
	Atom am_wm_pid;
Packit Service fa4841
Packit Service fa4841
	if (!pid)
Packit Service fa4841
		pid = getpid();
Packit Service fa4841
Packit Service fa4841
	am_wm_pid = xfc->_NET_WM_PID;
Packit Service b1ea74
	XChangeProperty(xfc->display, window, am_wm_pid, XA_CARDINAL, 32, PropModeReplace, (BYTE*)&pid,
Packit Service b1ea74
	                1);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
static const char* get_shm_id(void)
Packit Service fa4841
{
Packit Service fa4841
	static char shm_id[64];
Packit Service b1ea74
	sprintf_s(shm_id, sizeof(shm_id), "/com.freerdp.xfreerdp.tsmf_%016X", GetCurrentProcessId());
Packit Service fa4841
	return shm_id;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
Window xf_CreateDummyWindow(xfContext* xfc)
Packit Service fa4841
{
Packit Service b1ea74
	return XCreateSimpleWindow(xfc->display, DefaultRootWindow(xfc->display), 0, 0, 1, 1, 0, 0, 0);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_DestroyDummyWindow(xfContext* xfc, Window window)
Packit Service fa4841
{
Packit Service fa4841
	if (window)
Packit Service fa4841
		XDestroyWindow(xfc->display, window);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
xfWindow* xf_CreateDesktopWindow(xfContext* xfc, char* name, int width, int height)
Packit Service fa4841
{
Packit Service fa4841
	XEvent xevent;
Packit Service fa4841
	int input_mask;
Packit Service fa4841
	xfWindow* window;
Packit Service fa4841
	Window parentWindow;
Packit Service fa4841
	XClassHint* classHints;
Packit Service fa4841
	rdpSettings* settings;
Packit Service b1ea74
	window = (xfWindow*)calloc(1, sizeof(xfWindow));
Packit Service fa4841
Packit Service fa4841
	if (!window)
Packit Service fa4841
		return NULL;
Packit Service fa4841
Packit Service fa4841
	settings = xfc->context.settings;
Packit Service b1ea74
	parentWindow = (Window)xfc->context.settings->ParentWindowId;
Packit Service fa4841
	window->width = width;
Packit Service fa4841
	window->height = height;
Packit Service fa4841
	window->decorations = xfc->decorations;
Packit Service fa4841
	window->is_mapped = FALSE;
Packit Service fa4841
	window->is_transient = FALSE;
Packit Service b1ea74
	window->handle = XCreateWindow(xfc->display, RootWindowOfScreen(xfc->screen), xfc->workArea.x,
Packit Service b1ea74
	                               xfc->workArea.y, xfc->workArea.width, xfc->workArea.height, 0,
Packit Service b1ea74
	                               xfc->depth, InputOutput, xfc->visual,
Packit Service fa4841
	                               CWBackPixel | CWBackingStore | CWOverrideRedirect | CWColormap |
Packit Service b1ea74
	                                   CWBorderPixel | CWWinGravity | CWBitGravity,
Packit Service b1ea74
	                               &xfc->attribs);
Packit Service b1ea74
	window->shmid = shm_open(get_shm_id(), (O_CREAT | O_RDWR), (S_IREAD | S_IWRITE));
Packit Service fa4841
Packit Service fa4841
	if (window->shmid < 0)
Packit Service fa4841
	{
Packit Service fa4841
		DEBUG_X11("xf_CreateDesktopWindow: failed to get access to shared memory - shmget()\n");
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		void* mem;
Packit Service fa4841
		ftruncate(window->shmid, sizeof(window->handle));
Packit Service b1ea74
		mem = mmap(0, sizeof(window->handle), PROT_READ | PROT_WRITE, MAP_SHARED, window->shmid, 0);
Packit Service fa4841
Packit Service fa4841
		if (mem == MAP_FAILED)
Packit Service fa4841
		{
Packit Service b1ea74
			DEBUG_X11("xf_CreateDesktopWindow: failed to assign pointer to the memory address - "
Packit Service b1ea74
			          "shmat()\n");
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			window->xfwin = mem;
Packit Service fa4841
			*window->xfwin = window->handle;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	classHints = XAllocClassHint();
Packit Service fa4841
Packit Service fa4841
	if (classHints)
Packit Service fa4841
	{
Packit Service fa4841
		classHints->res_name = "xfreerdp";
Packit Service fa4841
Packit Service fa4841
		if (xfc->context.settings->WmClass)
Packit Service fa4841
			classHints->res_class = xfc->context.settings->WmClass;
Packit Service fa4841
		else
Packit Service fa4841
			classHints->res_class = "xfreerdp";
Packit Service fa4841
Packit Service fa4841
		XSetClassHint(xfc->display, window->handle, classHints);
Packit Service fa4841
		XFree(classHints);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	xf_ResizeDesktopWindow(xfc, window, width, height);
Packit Service fa4841
	xf_SetWindowDecorations(xfc, window->handle, window->decorations);
Packit Service fa4841
	xf_SetWindowPID(xfc, window->handle, 0);
Packit Service b1ea74
	input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
Packit Service b1ea74
	             VisibilityChangeMask | FocusChangeMask | StructureNotifyMask | PointerMotionMask |
Packit Service b1ea74
	             ExposureMask | PropertyChangeMask;
Packit Service fa4841
Packit Service fa4841
	if (xfc->grab_keyboard)
Packit Service fa4841
		input_mask |= EnterWindowMask | LeaveWindowMask;
Packit Service fa4841
Packit Service b1ea74
	XChangeProperty(xfc->display, window->handle, xfc->_NET_WM_ICON, XA_CARDINAL, 32,
Packit Service b1ea74
	                PropModeReplace, (BYTE*)xf_icon_prop, ARRAYSIZE(xf_icon_prop));
Packit Service fa4841
Packit Service fa4841
	if (parentWindow)
Packit Service fa4841
		XReparentWindow(xfc->display, window->handle, parentWindow, 0, 0);
Packit Service fa4841
Packit Service fa4841
	XSelectInput(xfc->display, window->handle, input_mask);
Packit Service fa4841
	XClearWindow(xfc->display, window->handle);
Packit Service fa4841
	xf_SetWindowTitleText(xfc, window->handle, name);
Packit Service fa4841
	XMapWindow(xfc->display, window->handle);
Packit Service fa4841
	xf_input_init(xfc, window->handle);
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service fa4841
	 * NOTE: This must be done here to handle reparenting the window,
Packit Service fa4841
	 * so that we don't miss the event and hang waiting for the next one
Packit Service fa4841
	 */
Packit Service fa4841
	do
Packit Service fa4841
	{
Packit Service fa4841
		XMaskEvent(xfc->display, VisibilityChangeMask, &xevent);
Packit Service b1ea74
	} while (xevent.type != VisibilityNotify);
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service fa4841
	 * The XCreateWindow call will start the window in the upper-left corner of our current
Packit Service fa4841
	 * monitor instead of the upper-left monitor for remote app mode (which uses all monitors).
Packit Service fa4841
	 * This extra call after the window is mapped will position the login window correctly
Packit Service fa4841
	 */
Packit Service fa4841
	if (xfc->context.settings->RemoteApplicationMode)
Packit Service fa4841
	{
Packit Service fa4841
		XMoveWindow(xfc->display, window->handle, 0, 0);
Packit Service fa4841
	}
Packit Service fa4841
	else if (settings->DesktopPosX != UINT32_MAX && settings->DesktopPosY != UINT32_MAX)
Packit Service fa4841
	{
Packit Service b1ea74
		XMoveWindow(xfc->display, window->handle, settings->DesktopPosX, settings->DesktopPosY);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	window->floatbar = xf_floatbar_new(xfc, window->handle, name, settings->Floatbar);
Packit Service fa4841
	return window;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_ResizeDesktopWindow(xfContext* xfc, xfWindow* window, int width, int height)
Packit Service fa4841
{
Packit Service fa4841
	XSizeHints* size_hints;
Packit Service fa4841
	rdpSettings* settings = NULL;
Packit Service fa4841
Packit Service fa4841
	if (!xfc || !window)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	settings = xfc->context.settings;
Packit Service fa4841
Packit Service fa4841
	if (!(size_hints = XAllocSizeHints()))
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	size_hints->flags = PMinSize | PMaxSize | PWinGravity;
Packit Service fa4841
	size_hints->win_gravity = NorthWestGravity;
Packit Service fa4841
	size_hints->min_width = size_hints->min_height = 1;
Packit Service fa4841
	size_hints->max_width = size_hints->max_height = 16384;
Packit Service fa4841
	XResizeWindow(xfc->display, window->handle, width, height);
Packit Service fa4841
#ifdef WITH_XRENDER
Packit Service fa4841
Packit Service fa4841
	if (!settings->SmartSizing && !settings->DynamicResolutionUpdate)
Packit Service fa4841
#endif
Packit Service fa4841
	{
Packit Service fa4841
		if (!xfc->fullscreen)
Packit Service fa4841
		{
Packit Service fa4841
			/* min == max is an hint for the WM to indicate that the window should
Packit Service fa4841
			 * not be resizable */
Packit Service fa4841
			size_hints->min_width = size_hints->max_width = width;
Packit Service fa4841
			size_hints->min_height = size_hints->max_height = height;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	XSetWMNormalHints(xfc->display, window->handle, size_hints);
Packit Service fa4841
	XFree(size_hints);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_DestroyDesktopWindow(xfContext* xfc, xfWindow* window)
Packit Service fa4841
{
Packit Service fa4841
	if (!window)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (xfc->window == window)
Packit Service fa4841
		xfc->window = NULL;
Packit Service fa4841
Packit Service b1ea74
	xf_floatbar_free(window->floatbar);
Packit Service fa4841
Packit Service fa4841
	if (window->gc)
Packit Service fa4841
		XFreeGC(xfc->display, window->gc);
Packit Service fa4841
Packit Service fa4841
	if (window->handle)
Packit Service fa4841
	{
Packit Service fa4841
		XUnmapWindow(xfc->display, window->handle);
Packit Service fa4841
		XDestroyWindow(xfc->display, window->handle);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (window->xfwin)
Packit Service fa4841
		munmap(0, sizeof(*window->xfwin));
Packit Service fa4841
Packit Service fa4841
	if (window->shmid >= 0)
Packit Service fa4841
		close(window->shmid);
Packit Service fa4841
Packit Service fa4841
	shm_unlink(get_shm_id());
Packit Service b1ea74
	window->xfwin = (Window*)-1;
Packit Service fa4841
	window->shmid = -1;
Packit Service fa4841
	free(window);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_SetWindowStyle(xfContext* xfc, xfAppWindow* appWindow, UINT32 style, UINT32 ex_style)
Packit Service fa4841
{
Packit Service fa4841
	Atom window_type;
Packit Service fa4841
	BOOL redirect = FALSE;
Packit Service fa4841
Packit Service fa4841
	if ((ex_style & WS_EX_NOACTIVATE) || (ex_style & WS_EX_TOOLWINDOW))
Packit Service fa4841
	{
Packit Service fa4841
		redirect = TRUE;
Packit Service fa4841
		appWindow->is_transient = TRUE;
Packit Service fa4841
		xf_SetWindowUnlisted(xfc, appWindow->handle);
Packit Service fa4841
		window_type = xfc->_NET_WM_WINDOW_TYPE_DROPDOWN_MENU;
Packit Service fa4841
	}
Packit Service fa4841
	/*
Packit Service b1ea74
	 * TOPMOST window that is not a tool window is treated like a regular window (i.e. task
Packit Service b1ea74
	 * manager). Want to do this here, since the window may have type WS_POPUP
Packit Service fa4841
	 */
Packit Service fa4841
	else if (ex_style & WS_EX_TOPMOST)
Packit Service fa4841
	{
Packit Service fa4841
		window_type = xfc->_NET_WM_WINDOW_TYPE_NORMAL;
Packit Service fa4841
	}
Packit Service fa4841
	else if (style & WS_POPUP)
Packit Service fa4841
	{
Packit Service fa4841
		/* this includes dialogs, popups, etc, that need to be full-fledged windows */
Packit Service fa4841
		appWindow->is_transient = TRUE;
Packit Service fa4841
		window_type = xfc->_NET_WM_WINDOW_TYPE_DIALOG;
Packit Service fa4841
		xf_SetWindowUnlisted(xfc, appWindow->handle);
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		window_type = xfc->_NET_WM_WINDOW_TYPE_NORMAL;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	{
Packit Service fa4841
		/*
Packit Service fa4841
		 * Tooltips and menu items should be unmanaged windows
Packit Service fa4841
		 * (called "override redirect" in X windows parlance)
Packit Service fa4841
		 * If they are managed, there are issues with window focus that
Packit Service fa4841
		 * cause the windows to behave improperly.  For example, a mouse
Packit Service fa4841
		 * press will dismiss a drop-down menu because the RDP server
Packit Service fa4841
		 * sees that as a focus out event from the window owning the
Packit Service fa4841
		 * dropdown.
Packit Service fa4841
		 */
Packit Service fa4841
		XSetWindowAttributes attrs;
Packit Service fa4841
		attrs.override_redirect = redirect ? True : False;
Packit Service b1ea74
		XChangeWindowAttributes(xfc->display, appWindow->handle, CWOverrideRedirect, &attrs);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	XChangeProperty(xfc->display, appWindow->handle, xfc->_NET_WM_WINDOW_TYPE, XA_ATOM, 32,
Packit Service b1ea74
	                PropModeReplace, (BYTE*)&window_type, 1);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_SetWindowText(xfContext* xfc, xfAppWindow* appWindow, const char* name)
Packit Service fa4841
{
Packit Service fa4841
	xf_SetWindowTitleText(xfc, appWindow->handle, name);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static void xf_FixWindowCoordinates(xfContext* xfc, int* x, int* y, int* width, int* height)
Packit Service fa4841
{
Packit Service fa4841
	int vscreen_width;
Packit Service fa4841
	int vscreen_height;
Packit Service fa4841
	vscreen_width = xfc->vscreen.area.right - xfc->vscreen.area.left + 1;
Packit Service fa4841
	vscreen_height = xfc->vscreen.area.bottom - xfc->vscreen.area.top + 1;
Packit Service fa4841
Packit Service fa4841
	if (*x < xfc->vscreen.area.left)
Packit Service fa4841
	{
Packit Service fa4841
		*width += *x;
Packit Service fa4841
		*x = xfc->vscreen.area.left;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*y < xfc->vscreen.area.top)
Packit Service fa4841
	{
Packit Service fa4841
		*height += *y;
Packit Service fa4841
		*y = xfc->vscreen.area.top;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*width > vscreen_width)
Packit Service fa4841
	{
Packit Service fa4841
		*width = vscreen_width;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*height > vscreen_height)
Packit Service fa4841
	{
Packit Service fa4841
		*height = vscreen_height;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*width < 1)
Packit Service fa4841
	{
Packit Service fa4841
		*width = 1;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (*height < 1)
Packit Service fa4841
	{
Packit Service fa4841
		*height = 1;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
int xf_AppWindowInit(xfContext* xfc, xfAppWindow* appWindow)
Packit Service fa4841
{
Packit Service b1ea74
	if (!xfc || !appWindow)
Packit Service b1ea74
		return -1;
Packit Service b1ea74
Packit Service b1ea74
	xf_SetWindowDecorations(xfc, appWindow->handle, appWindow->decorations);
Packit Service b1ea74
	xf_SetWindowStyle(xfc, appWindow, appWindow->dwStyle, appWindow->dwExStyle);
Packit Service b1ea74
	xf_SetWindowPID(xfc, appWindow->handle, 0);
Packit Service b1ea74
	xf_ShowWindow(xfc, appWindow, WINDOW_SHOW);
Packit Service b1ea74
	XClearWindow(xfc->display, appWindow->handle);
Packit Service b1ea74
	XMapWindow(xfc->display, appWindow->handle);
Packit Service b1ea74
	/* Move doesn't seem to work until window is mapped. */
Packit Service b1ea74
	xf_MoveWindow(xfc, appWindow, appWindow->x, appWindow->y, appWindow->width, appWindow->height);
Packit Service b1ea74
	xf_SetWindowText(xfc, appWindow, appWindow->title);
Packit Service b1ea74
	return 1;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
int xf_AppWindowCreate(xfContext* xfc, xfAppWindow* appWindow)
Packit Service b1ea74
{
Packit Service fa4841
	XGCValues gcv;
Packit Service fa4841
	int input_mask;
Packit Service fa4841
	XWMHints* InputModeHint;
Packit Service fa4841
	XClassHint* class_hints;
Packit Service fa4841
	xf_FixWindowCoordinates(xfc, &appWindow->x, &appWindow->y, &appWindow->width,
Packit Service fa4841
	                        &appWindow->height);
Packit Service fa4841
	appWindow->decorations = FALSE;
Packit Service fa4841
	appWindow->fullscreen = FALSE;
Packit Service fa4841
	appWindow->local_move.state = LMS_NOT_ACTIVE;
Packit Service fa4841
	appWindow->is_mapped = FALSE;
Packit Service fa4841
	appWindow->is_transient = FALSE;
Packit Service fa4841
	appWindow->rail_state = 0;
Packit Service fa4841
	appWindow->rail_ignore_configure = FALSE;
Packit Service b1ea74
	appWindow->handle = XCreateWindow(xfc->display, RootWindowOfScreen(xfc->screen), appWindow->x,
Packit Service b1ea74
	                                  appWindow->y, appWindow->width, appWindow->height, 0,
Packit Service b1ea74
	                                  xfc->depth, InputOutput, xfc->visual, 0, &xfc->attribs);
Packit Service fa4841
Packit Service fa4841
	if (!appWindow->handle)
Packit Service fa4841
		return -1;
Packit Service fa4841
Packit Service fa4841
	ZeroMemory(&gcv, sizeof(gcv));
Packit Service b1ea74
	appWindow->gc = XCreateGC(xfc->display, appWindow->handle, GCGraphicsExposures, &gcv;;
Packit Service fa4841
	class_hints = XAllocClassHint();
Packit Service fa4841
Packit Service fa4841
	if (class_hints)
Packit Service fa4841
	{
Packit Service fa4841
		char* class = NULL;
Packit Service fa4841
Packit Service fa4841
		if (xfc->context.settings->WmClass)
Packit Service fa4841
		{
Packit Service fa4841
			class_hints->res_class = xfc->context.settings->WmClass;
Packit Service fa4841
		}
Packit Service fa4841
		else
Packit Service fa4841
		{
Packit Service fa4841
			class = malloc(sizeof("RAIL:00000000"));
Packit Service b1ea74
			sprintf_s(class, sizeof("RAIL:00000000"), "RAIL:%08" PRIX64 "", appWindow->windowId);
Packit Service fa4841
			class_hints->res_class = class;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		class_hints->res_name = "RAIL";
Packit Service fa4841
		XSetClassHint(xfc->display, appWindow->handle, class_hints);
Packit Service fa4841
		XFree(class_hints);
Packit Service fa4841
		free(class);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* Set the input mode hint for the WM */
Packit Service fa4841
	InputModeHint = XAllocWMHints();
Packit Service fa4841
	InputModeHint->flags = (1L << 0);
Packit Service fa4841
	InputModeHint->input = True;
Packit Service fa4841
	XSetWMHints(xfc->display, appWindow->handle, InputModeHint);
Packit Service fa4841
	XFree(InputModeHint);
Packit Service fa4841
	XSetWMProtocols(xfc->display, appWindow->handle, &(xfc->WM_DELETE_WINDOW), 1);
Packit Service b1ea74
	input_mask = KeyPressMask | KeyReleaseMask | ButtonPressMask | ButtonReleaseMask |
Packit Service b1ea74
	             EnterWindowMask | LeaveWindowMask | PointerMotionMask | Button1MotionMask |
Packit Service b1ea74
	             Button2MotionMask | Button3MotionMask | Button4MotionMask | Button5MotionMask |
Packit Service b1ea74
	             ButtonMotionMask | KeymapStateMask | ExposureMask | VisibilityChangeMask |
Packit Service b1ea74
	             StructureNotifyMask | SubstructureNotifyMask | SubstructureRedirectMask |
Packit Service b1ea74
	             FocusChangeMask | PropertyChangeMask | ColormapChangeMask | OwnerGrabButtonMask;
Packit Service fa4841
	XSelectInput(xfc->display, appWindow->handle, input_mask);
Packit Service b1ea74
Packit Service fa4841
	return 1;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_SetWindowMinMaxInfo(xfContext* xfc, xfAppWindow* appWindow, int maxWidth, int maxHeight,
Packit Service b1ea74
                            int maxPosX, int maxPosY, int minTrackWidth, int minTrackHeight,
Packit Service b1ea74
                            int maxTrackWidth, int maxTrackHeight)
Packit Service fa4841
{
Packit Service fa4841
	XSizeHints* size_hints;
Packit Service fa4841
	size_hints = XAllocSizeHints();
Packit Service fa4841
Packit Service fa4841
	if (size_hints)
Packit Service fa4841
	{
Packit Service fa4841
		size_hints->flags = PMinSize | PMaxSize | PResizeInc;
Packit Service b1ea74
		size_hints->min_width = minTrackWidth;
Packit Service fa4841
		size_hints->min_height = minTrackHeight;
Packit Service b1ea74
		size_hints->max_width = maxTrackWidth;
Packit Service fa4841
		size_hints->max_height = maxTrackHeight;
Packit Service fa4841
		/* to speedup window drawing we need to select optimal value for sizing step. */
Packit Service fa4841
		size_hints->width_inc = size_hints->height_inc = 1;
Packit Service fa4841
		XSetWMNormalHints(xfc->display, appWindow->handle, size_hints);
Packit Service fa4841
		XFree(size_hints);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_StartLocalMoveSize(xfContext* xfc, xfAppWindow* appWindow, int direction, int x, int y)
Packit Service fa4841
{
Packit Service fa4841
	if (appWindow->local_move.state != LMS_NOT_ACTIVE)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	/*
Packit Service b1ea74
	 * Save original mouse location relative to root.  This will be needed
Packit Service b1ea74
	 * to end local move to RDP server and/or X server
Packit Service b1ea74
	 */
Packit Service fa4841
	appWindow->local_move.root_x = x;
Packit Service fa4841
	appWindow->local_move.root_y = y;
Packit Service fa4841
	appWindow->local_move.state = LMS_STARTING;
Packit Service fa4841
	appWindow->local_move.direction = direction;
Packit Service fa4841
	XUngrabPointer(xfc->display, CurrentTime);
Packit Service b1ea74
	xf_SendClientEvent(
Packit Service b1ea74
	    xfc, appWindow->handle,
Packit Service b1ea74
	    xfc->_NET_WM_MOVERESIZE, /* request X window manager to initiate a local move */
Packit Service b1ea74
	    5,                       /* 5 arguments to follow */
Packit Service b1ea74
	    x,                       /* x relative to root window */
Packit Service b1ea74
	    y,                       /* y relative to root window */
Packit Service b1ea74
	    direction,               /* extended ICCM direction flag */
Packit Service b1ea74
	    1,                       /* simulated mouse button 1 */
Packit Service b1ea74
	    1);                      /* 1 == application request per extended ICCM */
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_EndLocalMoveSize(xfContext* xfc, xfAppWindow* appWindow)
Packit Service fa4841
{
Packit Service fa4841
	if (appWindow->local_move.state == LMS_NOT_ACTIVE)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (appWindow->local_move.state == LMS_STARTING)
Packit Service fa4841
	{
Packit Service fa4841
		/*
Packit Service fa4841
		 * The move never was property started. This can happen due to race
Packit Service fa4841
		 * conditions between the mouse button up and the communications to the
Packit Service fa4841
		 * RDP server for local moves. We must cancel the X window manager move.
Packit Service fa4841
		 * Per ICCM, the X client can ask to cancel an active move.
Packit Service fa4841
		 */
Packit Service b1ea74
		xf_SendClientEvent(
Packit Service b1ea74
		    xfc, appWindow->handle,
Packit Service b1ea74
		    xfc->_NET_WM_MOVERESIZE,      /* request X window manager to abort a local move */
Packit Service b1ea74
		    5,                            /* 5 arguments to follow */
Packit Service b1ea74
		    appWindow->local_move.root_x, /* x relative to root window */
Packit Service b1ea74
		    appWindow->local_move.root_y, /* y relative to root window */
Packit Service b1ea74
		    _NET_WM_MOVERESIZE_CANCEL,    /* extended ICCM direction flag */
Packit Service b1ea74
		    1,                            /* simulated mouse button 1 */
Packit Service b1ea74
		    1);                           /* 1 == application request per extended ICCM */
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	appWindow->local_move.state = LMS_NOT_ACTIVE;
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_MoveWindow(xfContext* xfc, xfAppWindow* appWindow, int x, int y, int width, int height)
Packit Service fa4841
{
Packit Service fa4841
	BOOL resize = FALSE;
Packit Service fa4841
Packit Service fa4841
	if ((width * height) < 1)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if ((appWindow->width != width) || (appWindow->height != height))
Packit Service fa4841
		resize = TRUE;
Packit Service fa4841
Packit Service b1ea74
	if (appWindow->local_move.state == LMS_STARTING || appWindow->local_move.state == LMS_ACTIVE)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	appWindow->x = x;
Packit Service fa4841
	appWindow->y = y;
Packit Service fa4841
	appWindow->width = width;
Packit Service fa4841
	appWindow->height = height;
Packit Service fa4841
Packit Service fa4841
	if (resize)
Packit Service fa4841
		XMoveResizeWindow(xfc->display, appWindow->handle, x, y, width, height);
Packit Service fa4841
	else
Packit Service fa4841
		XMoveWindow(xfc->display, appWindow->handle, x, y);
Packit Service fa4841
Packit Service fa4841
	xf_UpdateWindowArea(xfc, appWindow, 0, 0, width, height);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_ShowWindow(xfContext* xfc, xfAppWindow* appWindow, BYTE state)
Packit Service fa4841
{
Packit Service fa4841
	switch (state)
Packit Service fa4841
	{
Packit Service fa4841
		case WINDOW_HIDE:
Packit Service fa4841
			XWithdrawWindow(xfc->display, appWindow->handle, xfc->screen_number);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WINDOW_SHOW_MINIMIZED:
Packit Service fa4841
			XIconifyWindow(xfc->display, appWindow->handle, xfc->screen_number);
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WINDOW_SHOW_MAXIMIZED:
Packit Service fa4841
			/* Set the window as maximized */
Packit Service b1ea74
			xf_SendClientEvent(xfc, appWindow->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_ADD,
Packit Service b1ea74
			                   xfc->_NET_WM_STATE_MAXIMIZED_VERT, xfc->_NET_WM_STATE_MAXIMIZED_HORZ,
Packit Service b1ea74
			                   0);
Packit Service fa4841
Packit Service fa4841
			/*
Packit Service b1ea74
			 * This is a workaround for the case where the window is maximized locally before the
Packit Service b1ea74
			 * rail server is told to maximize the window, this appears to be a race condition where
Packit Service b1ea74
			 * the local window with incomplete data and once the window is actually maximized on
Packit Service b1ea74
			 * the server
Packit Service b1ea74
			 * - an update of the new areas may not happen. So, we simply to do a full update of the
Packit Service b1ea74
			 * entire window once the rail server notifies us that the window is now maximized.
Packit Service fa4841
			 */
Packit Service fa4841
			if (appWindow->rail_state == WINDOW_SHOW_MAXIMIZED)
Packit Service fa4841
			{
Packit Service fa4841
				xf_UpdateWindowArea(xfc, appWindow, 0, 0, appWindow->windowWidth,
Packit Service fa4841
				                    appWindow->windowHeight);
Packit Service fa4841
			}
Packit Service fa4841
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case WINDOW_SHOW:
Packit Service fa4841
			/* Ensure the window is not maximized */
Packit Service b1ea74
			xf_SendClientEvent(xfc, appWindow->handle, xfc->_NET_WM_STATE, 4, _NET_WM_STATE_REMOVE,
Packit Service b1ea74
			                   xfc->_NET_WM_STATE_MAXIMIZED_VERT, xfc->_NET_WM_STATE_MAXIMIZED_HORZ,
Packit Service b1ea74
			                   0);
Packit Service fa4841
Packit Service fa4841
			/*
Packit Service fa4841
			 * Ignore configure requests until both the Maximized properties have been processed
Packit Service b1ea74
			 * to prevent condition where WM overrides size of request due to one or both of these
Packit Service b1ea74
			 * properties still being set - which causes a position adjustment to be sent back to
Packit Service b1ea74
			 * the server thus causing the window to not return to its original size
Packit Service fa4841
			 */
Packit Service fa4841
			if (appWindow->rail_state == WINDOW_SHOW_MAXIMIZED)
Packit Service fa4841
				appWindow->rail_ignore_configure = TRUE;
Packit Service fa4841
Packit Service fa4841
			if (appWindow->is_transient)
Packit Service fa4841
				xf_SetWindowUnlisted(xfc, appWindow->handle);
Packit Service fa4841
Packit Service b1ea74
			XMapWindow(xfc->display, appWindow->handle);
Packit Service fa4841
			break;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	/* Save the current rail state of this window */
Packit Service fa4841
	appWindow->rail_state = state;
Packit Service fa4841
	XFlush(xfc->display);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_SetWindowRects(xfContext* xfc, xfAppWindow* appWindow, RECTANGLE_16* rects, int nrects)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
	XRectangle* xrects;
Packit Service fa4841
Packit Service fa4841
	if (nrects < 1)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XEXT
Packit Service b1ea74
	xrects = (XRectangle*)calloc(nrects, sizeof(XRectangle));
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < nrects; i++)
Packit Service fa4841
	{
Packit Service fa4841
		xrects[i].x = rects[i].left;
Packit Service fa4841
		xrects[i].y = rects[i].top;
Packit Service fa4841
		xrects[i].width = rects[i].right - rects[i].left;
Packit Service fa4841
		xrects[i].height = rects[i].bottom - rects[i].top;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	XShapeCombineRectangles(xfc->display, appWindow->handle, ShapeBounding, 0, 0, xrects, nrects,
Packit Service b1ea74
	                        ShapeSet, 0);
Packit Service fa4841
	free(xrects);
Packit Service fa4841
#endif
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_SetWindowVisibilityRects(xfContext* xfc, xfAppWindow* appWindow, UINT32 rectsOffsetX,
Packit Service b1ea74
                                 UINT32 rectsOffsetY, RECTANGLE_16* rects, int nrects)
Packit Service fa4841
{
Packit Service fa4841
	int i;
Packit Service fa4841
	XRectangle* xrects;
Packit Service fa4841
Packit Service fa4841
	if (nrects < 1)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
#ifdef WITH_XEXT
Packit Service b1ea74
	xrects = (XRectangle*)calloc(nrects, sizeof(XRectangle));
Packit Service fa4841
Packit Service fa4841
	for (i = 0; i < nrects; i++)
Packit Service fa4841
	{
Packit Service fa4841
		xrects[i].x = rects[i].left;
Packit Service fa4841
		xrects[i].y = rects[i].top;
Packit Service fa4841
		xrects[i].width = rects[i].right - rects[i].left;
Packit Service fa4841
		xrects[i].height = rects[i].bottom - rects[i].top;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	XShapeCombineRectangles(xfc->display, appWindow->handle, ShapeBounding, rectsOffsetX,
Packit Service b1ea74
	                        rectsOffsetY, xrects, nrects, ShapeSet, 0);
Packit Service fa4841
	free(xrects);
Packit Service fa4841
#endif
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
void xf_UpdateWindowArea(xfContext* xfc, xfAppWindow* appWindow, int x, int y, int width,
Packit Service b1ea74
                         int height)
Packit Service fa4841
{
Packit Service fa4841
	int ax, ay;
Packit Service fa4841
Packit Service fa4841
	if (appWindow == NULL)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service b1ea74
	if (appWindow->surfaceId < UINT16_MAX)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service fa4841
	ax = x + appWindow->windowOffsetX;
Packit Service fa4841
	ay = y + appWindow->windowOffsetY;
Packit Service fa4841
Packit Service fa4841
	if (ax + width > appWindow->windowOffsetX + appWindow->width)
Packit Service fa4841
		width = (appWindow->windowOffsetX + appWindow->width - 1) - ax;
Packit Service fa4841
Packit Service fa4841
	if (ay + height > appWindow->windowOffsetY + appWindow->height)
Packit Service fa4841
		height = (appWindow->windowOffsetY + appWindow->height - 1) - ay;
Packit Service fa4841
Packit Service b1ea74
	xf_lock_x11(xfc);
Packit Service fa4841
Packit Service fa4841
	if (xfc->context.settings->SoftwareGdi)
Packit Service fa4841
	{
Packit Service b1ea74
		XPutImage(xfc->display, xfc->primary, appWindow->gc, xfc->image, ax, ay, ax, ay, width,
Packit Service b1ea74
		          height);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	XCopyArea(xfc->display, xfc->primary, appWindow->handle, appWindow->gc, ax, ay, width, height,
Packit Service b1ea74
	          x, y);
Packit Service fa4841
	XFlush(xfc->display);
Packit Service b1ea74
	xf_unlock_x11(xfc);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void xf_DestroyWindow(xfContext* xfc, xfAppWindow* appWindow)
Packit Service fa4841
{
Packit Service fa4841
	if (!appWindow)
Packit Service fa4841
		return;
Packit Service fa4841
Packit Service fa4841
	if (appWindow->gc)
Packit Service fa4841
		XFreeGC(xfc->display, appWindow->gc);
Packit Service fa4841
Packit Service fa4841
	if (appWindow->handle)
Packit Service fa4841
	{
Packit Service fa4841
		XUnmapWindow(xfc->display, appWindow->handle);
Packit Service fa4841
		XDestroyWindow(xfc->display, appWindow->handle);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (appWindow->xfwin)
Packit Service fa4841
		munmap(0, sizeof(*appWindow->xfwin));
Packit Service fa4841
Packit Service fa4841
	if (appWindow->shmid >= 0)
Packit Service fa4841
		close(appWindow->shmid);
Packit Service fa4841
Packit Service fa4841
	shm_unlink(get_shm_id());
Packit Service b1ea74
	appWindow->xfwin = (Window*)-1;
Packit Service fa4841
	appWindow->shmid = -1;
Packit Service fa4841
	free(appWindow->title);
Packit Service fa4841
	free(appWindow->windowRects);
Packit Service fa4841
	free(appWindow->visibilityRects);
Packit Service fa4841
	free(appWindow);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
xfAppWindow* xf_AppWindowFromX11Window(xfContext* xfc, Window wnd)
Packit Service fa4841
{
Packit Service fa4841
	int index;
Packit Service fa4841
	int count;
Packit Service fa4841
	ULONG_PTR* pKeys = NULL;
Packit Service fa4841
	xfAppWindow* appWindow;
Packit Service fa4841
	count = HashTable_GetKeys(xfc->railWindows, &pKeys);
Packit Service fa4841
Packit Service fa4841
	for (index = 0; index < count; index++)
Packit Service fa4841
	{
Packit Service b1ea74
		appWindow = xf_rail_get_window(xfc, *(UINT64*)pKeys[index]);
Packit Service b1ea74
Packit Service b1ea74
		if (!appWindow)
Packit Service b1ea74
			return NULL;
Packit Service fa4841
Packit Service fa4841
		if (appWindow->handle == wnd)
Packit Service fa4841
		{
Packit Service fa4841
			free(pKeys);
Packit Service fa4841
			return appWindow;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	free(pKeys);
Packit Service fa4841
	return NULL;
Packit Service fa4841
}