Blame client/Wayland/wlf_input.c

Packit Service fa4841
/**
Packit Service fa4841
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service fa4841
 * Wayland Input
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2014 Manuel Bachmann <tarnyko@tarnyko.net>
Packit Service fa4841
 * Copyright 2015 David Fort <contact@hardening-consulting.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
#include <stdlib.h>
Packit Service fa4841
#include <linux/input.h>
Packit Service fa4841
Packit Service fa4841
#include <freerdp/locale/keyboard.h>
Packit Service b1ea74
#include <freerdp/client/rdpei.h>
Packit Service b1ea74
#include <uwac/uwac.h>
Packit Service fa4841
Packit Service b1ea74
#include "wlfreerdp.h"
Packit Service fa4841
#include "wlf_input.h"
Packit Service fa4841
Packit Service b1ea74
#define TAG CLIENT_TAG("wayland.input")
Packit Service b1ea74
Packit Service b1ea74
#define MAX_CONTACTS 20
Packit Service b1ea74
Packit Service b1ea74
typedef struct touch_contact
Packit Service fa4841
{
Packit Service b1ea74
	int id;
Packit Service b1ea74
	double pos_x;
Packit Service b1ea74
	double pos_y;
Packit Service b1ea74
	BOOL emulate_mouse;
Packit Service b1ea74
} touchContact;
Packit Service b1ea74
Packit Service b1ea74
static touchContact contacts[MAX_CONTACTS];
Packit Service b1ea74
Packit Service b1ea74
BOOL wlf_handle_pointer_enter(freerdp* instance, const UwacPointerEnterLeaveEvent* ev)
Packit Service b1ea74
{
Packit Service b1ea74
	uint32_t x, y;
Packit Service b1ea74
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	return freerdp_input_send_mouse_event(instance->input, PTR_FLAGS_MOVE, x, y);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL wlf_handle_pointer_motion(freerdp* instance, const UwacPointerMotionEvent* ev)
Packit Service fa4841
{
Packit Service b1ea74
	uint32_t x, y;
Packit Service b1ea74
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	return freerdp_input_send_mouse_event(instance->input, PTR_FLAGS_MOVE, x, y);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL wlf_handle_pointer_buttons(freerdp* instance, const UwacPointerButtonEvent* ev)
Packit Service fa4841
{
Packit Service fa4841
	rdpInput* input;
Packit Service b1ea74
	UINT16 flags = 0;
Packit Service b1ea74
	UINT16 xflags = 0;
Packit Service b1ea74
	uint32_t x, y;
Packit Service fa4841
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service fa4841
	input = instance->input;
Packit Service fa4841
Packit Service fa4841
	if (ev->state == WL_POINTER_BUTTON_STATE_PRESSED)
Packit Service b1ea74
	{
Packit Service b1ea74
		flags |= PTR_FLAGS_DOWN;
Packit Service b1ea74
		xflags |= PTR_XFLAGS_DOWN;
Packit Service b1ea74
	}
Packit Service fa4841
Packit Service fa4841
	switch (ev->button)
Packit Service fa4841
	{
Packit Service fa4841
		case BTN_LEFT:
Packit Service fa4841
			flags |= PTR_FLAGS_BUTTON1;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case BTN_RIGHT:
Packit Service fa4841
			flags |= PTR_FLAGS_BUTTON2;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service fa4841
		case BTN_MIDDLE:
Packit Service fa4841
			flags |= PTR_FLAGS_BUTTON3;
Packit Service fa4841
			break;
Packit Service fa4841
Packit Service b1ea74
		case BTN_SIDE:
Packit Service b1ea74
			xflags |= PTR_XFLAGS_BUTTON1;
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service b1ea74
		case BTN_EXTRA:
Packit Service b1ea74
			xflags |= PTR_XFLAGS_BUTTON2;
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service fa4841
		default:
Packit Service fa4841
			return TRUE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	if ((flags & ~PTR_FLAGS_DOWN) != 0)
Packit Service b1ea74
		return freerdp_input_send_mouse_event(input, flags, x, y);
Packit Service fa4841
Packit Service b1ea74
	if ((xflags & ~PTR_XFLAGS_DOWN) != 0)
Packit Service b1ea74
		return freerdp_input_send_extended_mouse_event(input, xflags, x, y);
Packit Service b1ea74
Packit Service b1ea74
	return FALSE;
Packit Service b1ea74
}
Packit Service bb5c11
Packit Service b1ea74
BOOL wlf_handle_pointer_axis(freerdp* instance, const UwacPointerAxisEvent* ev)
Packit Service fa4841
{
Packit Service fa4841
	rdpInput* input;
Packit Service b1ea74
	UINT16 flags = 0;
Packit Service fa4841
	int direction;
Packit Service b1ea74
	uint32_t step;
Packit Service b1ea74
	uint32_t x, y;
Packit Service fa4841
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service fa4841
	input = instance->input;
Packit Service fa4841
Packit Service b1ea74
	direction = wl_fixed_to_int(ev->value);
Packit Service b1ea74
	switch (ev->axis)
Packit Service fa4841
	{
Packit Service b1ea74
		case WL_POINTER_AXIS_VERTICAL_SCROLL:
Packit Service b1ea74
			flags |= PTR_FLAGS_WHEEL;
Packit Service b1ea74
			if (direction > 0)
Packit Service b1ea74
				flags |= PTR_FLAGS_WHEEL_NEGATIVE;
Packit Service b1ea74
			break;
Packit Service b1ea74
Packit Service b1ea74
		case WL_POINTER_AXIS_HORIZONTAL_SCROLL:
Packit Service b1ea74
			flags |= PTR_FLAGS_HWHEEL;
Packit Service b1ea74
			if (direction < 0)
Packit Service b1ea74
				flags |= PTR_FLAGS_WHEEL_NEGATIVE;
Packit Service b1ea74
			break;
Packit Service fa4841
Packit Service b1ea74
		default:
Packit Service b1ea74
			return FALSE;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service b1ea74
	/* Wheel rotation steps:
Packit Service b1ea74
	 *
Packit Service b1ea74
	 * positive: 0 ... 0xFF  -> slow ... fast
Packit Service b1ea74
	 * negative: 0 ... 0xFF  -> fast ... slow
Packit Service b1ea74
	 */
Packit Service b1ea74
	step = abs(direction);
Packit Service b1ea74
	if (step > 0xFF)
Packit Service b1ea74
		step = 0xFF;
Packit Service b1ea74
Packit Service b1ea74
	/* Negative rotation, so count down steps from top */
Packit Service b1ea74
	if (flags & PTR_FLAGS_WHEEL_NEGATIVE)
Packit Service b1ea74
		step = 0xFF - step;
Packit Service b1ea74
Packit Service b1ea74
	flags |= step;
Packit Service b1ea74
Packit Service b1ea74
	return freerdp_input_send_mouse_event(input, flags, (UINT16)x, (UINT16)y);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL wlf_handle_key(freerdp* instance, const UwacKeyEvent* ev)
Packit Service fa4841
{
Packit Service fa4841
	rdpInput* input;
Packit Service fa4841
	DWORD rdp_scancode;
Packit Service fa4841
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	input = instance->input;
Packit Service fa4841
	rdp_scancode = freerdp_keyboard_get_rdp_scancode_from_x11_keycode(ev->raw_key + 8);
Packit Service fa4841
Packit Service fa4841
	if (rdp_scancode == RDP_SCANCODE_UNKNOWN)
Packit Service fa4841
		return TRUE;
Packit Service fa4841
Packit Service fa4841
	return freerdp_input_send_keyboard_event_ex(input, ev->pressed, rdp_scancode);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
BOOL wlf_keyboard_enter(freerdp* instance, const UwacKeyboardEnterLeaveEvent* ev)
Packit Service fa4841
{
Packit Service fa4841
	rdpInput* input;
Packit Service fa4841
Packit Service fa4841
	if (!instance || !ev || !instance->input)
Packit Service fa4841
		return FALSE;
Packit Service fa4841
Packit Service fa4841
	input = instance->input;
Packit Service fa4841
	return freerdp_input_send_focus_in_event(input, 0) &&
Packit Service b1ea74
	       freerdp_input_send_mouse_event(input, PTR_FLAGS_MOVE, 0, 0);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
BOOL wlf_handle_touch_up(freerdp* instance, const UwacTouchUp* ev)
Packit Service b1ea74
{
Packit Service b1ea74
	uint32_t x, y;
Packit Service b1ea74
	int i;
Packit Service b1ea74
	int touchId;
Packit Service b1ea74
	int contactId;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance || !ev || !instance->context)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	touchId = ev->id;
Packit Service b1ea74
Packit Service b1ea74
	for (i = 0; i < MAX_CONTACTS; i++)
Packit Service b1ea74
	{
Packit Service b1ea74
		if (contacts[i].id == touchId)
Packit Service b1ea74
		{
Packit Service b1ea74
			contacts[i].id = 0;
Packit Service b1ea74
			x = contacts[i].pos_x;
Packit Service b1ea74
			y = contacts[i].pos_y;
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (i == MAX_CONTACTS)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
Packit Service b1ea74
Packit Service b1ea74
	if (contacts[i].emulate_mouse == TRUE)
Packit Service b1ea74
	{
Packit Service b1ea74
		UINT16 flags = 0;
Packit Service b1ea74
		flags |= PTR_FLAGS_BUTTON1;
Packit Service b1ea74
Packit Service b1ea74
		if ((flags & ~PTR_FLAGS_DOWN) != 0)
Packit Service b1ea74
			return freerdp_input_send_mouse_event(instance->input, flags, x, y);
Packit Service b1ea74
Packit Service b1ea74
		return TRUE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!rdpei)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	rdpei->TouchEnd(rdpei, touchId, x, y, &contactId);
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
BOOL wlf_handle_touch_down(freerdp* instance, const UwacTouchDown* ev)
Packit Service b1ea74
{
Packit Service b1ea74
	uint32_t x, y;
Packit Service b1ea74
	int i;
Packit Service b1ea74
	int touchId;
Packit Service b1ea74
	int contactId;
Packit Service b1ea74
	wlfContext* context;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance || !ev || !instance->context)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
	touchId = ev->id;
Packit Service b1ea74
Packit Service b1ea74
	for (i = 0; i < MAX_CONTACTS; i++)
Packit Service b1ea74
	{
Packit Service b1ea74
		if (contacts[i].id == 0)
Packit Service b1ea74
		{
Packit Service b1ea74
			contacts[i].id = touchId;
Packit Service b1ea74
			contacts[i].pos_x = x;
Packit Service b1ea74
			contacts[i].pos_y = y;
Packit Service b1ea74
			contacts[i].emulate_mouse = FALSE;
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (i == MAX_CONTACTS)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	context = (wlfContext*)instance->context;
Packit Service b1ea74
	RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
Packit Service b1ea74
Packit Service b1ea74
	// Emulate mouse click if touch is not possible, like in login screen
Packit Service b1ea74
	if (!rdpei)
Packit Service b1ea74
	{
Packit Service b1ea74
		contacts[i].emulate_mouse = TRUE;
Packit Service b1ea74
Packit Service b1ea74
		UINT16 flags = 0;
Packit Service b1ea74
		flags |= PTR_FLAGS_DOWN;
Packit Service b1ea74
		flags |= PTR_FLAGS_BUTTON1;
Packit Service b1ea74
Packit Service b1ea74
		if ((flags & ~PTR_FLAGS_DOWN) != 0)
Packit Service b1ea74
			return freerdp_input_send_mouse_event(instance->input, flags, x, y);
Packit Service b1ea74
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	rdpei->TouchBegin(rdpei, touchId, x, y, &contactId);
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
BOOL wlf_handle_touch_motion(freerdp* instance, const UwacTouchMotion* ev)
Packit Service b1ea74
{
Packit Service b1ea74
	uint32_t x, y;
Packit Service b1ea74
	int i;
Packit Service b1ea74
	int touchId;
Packit Service b1ea74
	int contactId;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance || !ev || !instance->context)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	x = ev->x;
Packit Service b1ea74
	y = ev->y;
Packit Service b1ea74
	touchId = ev->id;
Packit Service b1ea74
Packit Service b1ea74
	for (i = 0; i < MAX_CONTACTS; i++)
Packit Service b1ea74
	{
Packit Service b1ea74
		if (contacts[i].id == touchId)
Packit Service b1ea74
		{
Packit Service b1ea74
			if (contacts[i].pos_x == x && contacts[i].pos_y == y)
Packit Service b1ea74
			{
Packit Service b1ea74
				return TRUE;
Packit Service b1ea74
			}
Packit Service b1ea74
			contacts[i].pos_x = x;
Packit Service b1ea74
			contacts[i].pos_y = y;
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (i == MAX_CONTACTS)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
Packit Service b1ea74
Packit Service b1ea74
	if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
Packit Service b1ea74
Packit Service b1ea74
	if (contacts[i].emulate_mouse == TRUE)
Packit Service b1ea74
	{
Packit Service b1ea74
		return TRUE;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!rdpei)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	rdpei->TouchUpdate(rdpei, touchId, x, y, &contactId);
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service fa4841
}