402 lines
8.3 KiB
C
402 lines
8.3 KiB
C
|
/**
|
||
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
||
|
* Wayland Input
|
||
|
*
|
||
|
* Copyright 2014 Manuel Bachmann <tarnyko@tarnyko.net>
|
||
|
* Copyright 2015 David Fort <contact@hardening-consulting.com>
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
#include <stdlib.h>
|
||
|
#include <linux/input.h>
|
||
|
|
||
|
#include <freerdp/locale/keyboard.h>
|
||
|
#include <freerdp/client/rdpei.h>
|
||
|
#include <uwac/uwac.h>
|
||
|
|
||
|
#include "wlfreerdp.h"
|
||
|
#include "wlf_input.h"
|
||
|
|
||
|
#define TAG CLIENT_TAG("wayland.input")
|
||
|
|
||
|
#define MAX_CONTACTS 20
|
||
|
|
||
|
typedef struct touch_contact
|
||
|
{
|
||
|
int id;
|
||
|
double pos_x;
|
||
|
double pos_y;
|
||
|
BOOL emulate_mouse;
|
||
|
} touchContact;
|
||
|
|
||
|
static touchContact contacts[MAX_CONTACTS];
|
||
|
|
||
|
BOOL wlf_handle_pointer_enter(freerdp* instance, const UwacPointerEnterLeaveEvent* ev)
|
||
|
{
|
||
|
uint32_t x, y;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
return freerdp_input_send_mouse_event(instance->input, PTR_FLAGS_MOVE, x, y);
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_pointer_motion(freerdp* instance, const UwacPointerMotionEvent* ev)
|
||
|
{
|
||
|
uint32_t x, y;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
return freerdp_input_send_mouse_event(instance->input, PTR_FLAGS_MOVE, x, y);
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_pointer_buttons(freerdp* instance, const UwacPointerButtonEvent* ev)
|
||
|
{
|
||
|
rdpInput* input;
|
||
|
UINT16 flags = 0;
|
||
|
UINT16 xflags = 0;
|
||
|
uint32_t x, y;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
input = instance->input;
|
||
|
|
||
|
if (ev->state == WL_POINTER_BUTTON_STATE_PRESSED)
|
||
|
{
|
||
|
flags |= PTR_FLAGS_DOWN;
|
||
|
xflags |= PTR_XFLAGS_DOWN;
|
||
|
}
|
||
|
|
||
|
switch (ev->button)
|
||
|
{
|
||
|
case BTN_LEFT:
|
||
|
flags |= PTR_FLAGS_BUTTON1;
|
||
|
break;
|
||
|
|
||
|
case BTN_RIGHT:
|
||
|
flags |= PTR_FLAGS_BUTTON2;
|
||
|
break;
|
||
|
|
||
|
case BTN_MIDDLE:
|
||
|
flags |= PTR_FLAGS_BUTTON3;
|
||
|
break;
|
||
|
|
||
|
case BTN_SIDE:
|
||
|
xflags |= PTR_XFLAGS_BUTTON1;
|
||
|
break;
|
||
|
|
||
|
case BTN_EXTRA:
|
||
|
xflags |= PTR_XFLAGS_BUTTON2;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
if ((flags & ~PTR_FLAGS_DOWN) != 0)
|
||
|
return freerdp_input_send_mouse_event(input, flags, x, y);
|
||
|
|
||
|
if ((xflags & ~PTR_XFLAGS_DOWN) != 0)
|
||
|
return freerdp_input_send_extended_mouse_event(input, xflags, x, y);
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_pointer_axis(freerdp* instance, const UwacPointerAxisEvent* ev)
|
||
|
{
|
||
|
rdpInput* input;
|
||
|
UINT16 flags = 0;
|
||
|
int32_t direction;
|
||
|
uint32_t x, y;
|
||
|
uint32_t i;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
input = instance->input;
|
||
|
|
||
|
direction = ev->value;
|
||
|
switch (ev->axis)
|
||
|
{
|
||
|
case WL_POINTER_AXIS_VERTICAL_SCROLL:
|
||
|
flags |= PTR_FLAGS_WHEEL;
|
||
|
if (direction > 0)
|
||
|
flags |= PTR_FLAGS_WHEEL_NEGATIVE;
|
||
|
break;
|
||
|
|
||
|
case WL_POINTER_AXIS_HORIZONTAL_SCROLL:
|
||
|
flags |= PTR_FLAGS_HWHEEL;
|
||
|
if (direction < 0)
|
||
|
flags |= PTR_FLAGS_WHEEL_NEGATIVE;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
/* Wheel rotation steps:
|
||
|
*
|
||
|
* positive: 0 ... 0xFF -> slow ... fast
|
||
|
* negative: 0 ... 0xFF -> fast ... slow
|
||
|
*/
|
||
|
for (i = 0; i < abs(direction); i++)
|
||
|
{
|
||
|
uint32_t cflags = flags | 0x78;
|
||
|
/* Convert negative values to 9bit twos complement */
|
||
|
if (flags & PTR_FLAGS_WHEEL_NEGATIVE)
|
||
|
cflags = (flags & 0xFF00) | (0x100 - (cflags & 0xFF));
|
||
|
if (!freerdp_input_send_mouse_event(input, cflags, (UINT16)x, (UINT16)y))
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_key(freerdp* instance, const UwacKeyEvent* ev)
|
||
|
{
|
||
|
rdpInput* input;
|
||
|
DWORD rdp_scancode;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
input = instance->input;
|
||
|
rdp_scancode = freerdp_keyboard_get_rdp_scancode_from_x11_keycode(ev->raw_key + 8);
|
||
|
|
||
|
if (rdp_scancode == RDP_SCANCODE_UNKNOWN)
|
||
|
return TRUE;
|
||
|
|
||
|
return freerdp_input_send_keyboard_event_ex(input, ev->pressed, rdp_scancode);
|
||
|
}
|
||
|
|
||
|
BOOL wlf_keyboard_enter(freerdp* instance, const UwacKeyboardEnterLeaveEvent* ev)
|
||
|
{
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
((wlfContext*)instance->context)->focusing = TRUE;
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
BOOL wlf_keyboard_modifiers(freerdp* instance, const UwacKeyboardModifiersEvent* ev)
|
||
|
{
|
||
|
rdpInput* input;
|
||
|
uint32_t syncFlags;
|
||
|
|
||
|
if (!instance || !ev || !instance->input)
|
||
|
return FALSE;
|
||
|
|
||
|
input = instance->input;
|
||
|
syncFlags = 0;
|
||
|
|
||
|
if (ev->modifiers & UWAC_MOD_CAPS_MASK)
|
||
|
syncFlags |= KBD_SYNC_CAPS_LOCK;
|
||
|
if (ev->modifiers & UWAC_MOD_NUM_MASK)
|
||
|
syncFlags |= KBD_SYNC_NUM_LOCK;
|
||
|
|
||
|
if (!((wlfContext*)instance->context)->focusing)
|
||
|
return TRUE;
|
||
|
|
||
|
((wlfContext*)instance->context)->focusing = FALSE;
|
||
|
return freerdp_input_send_focus_in_event(input, syncFlags) &&
|
||
|
freerdp_input_send_mouse_event(input, PTR_FLAGS_MOVE, 0, 0);
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_touch_up(freerdp* instance, const UwacTouchUp* ev)
|
||
|
{
|
||
|
uint32_t x, y;
|
||
|
int i;
|
||
|
int touchId;
|
||
|
int contactId;
|
||
|
|
||
|
if (!instance || !ev || !instance->context)
|
||
|
return FALSE;
|
||
|
|
||
|
touchId = ev->id;
|
||
|
|
||
|
for (i = 0; i < MAX_CONTACTS; i++)
|
||
|
{
|
||
|
if (contacts[i].id == touchId)
|
||
|
{
|
||
|
contacts[i].id = 0;
|
||
|
x = contacts[i].pos_x;
|
||
|
y = contacts[i].pos_y;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (i == MAX_CONTACTS)
|
||
|
return FALSE;
|
||
|
|
||
|
WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
|
||
|
|
||
|
if (contacts[i].emulate_mouse == TRUE)
|
||
|
{
|
||
|
UINT16 flags = 0;
|
||
|
flags |= PTR_FLAGS_BUTTON1;
|
||
|
|
||
|
if ((flags & ~PTR_FLAGS_DOWN) != 0)
|
||
|
return freerdp_input_send_mouse_event(instance->input, flags, x, y);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
if (!rdpei)
|
||
|
return FALSE;
|
||
|
|
||
|
rdpei->TouchEnd(rdpei, touchId, x, y, &contactId);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_touch_down(freerdp* instance, const UwacTouchDown* ev)
|
||
|
{
|
||
|
uint32_t x, y;
|
||
|
int i;
|
||
|
int touchId;
|
||
|
int contactId;
|
||
|
wlfContext* context;
|
||
|
|
||
|
if (!instance || !ev || !instance->context)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
touchId = ev->id;
|
||
|
|
||
|
for (i = 0; i < MAX_CONTACTS; i++)
|
||
|
{
|
||
|
if (contacts[i].id == 0)
|
||
|
{
|
||
|
contacts[i].id = touchId;
|
||
|
contacts[i].pos_x = x;
|
||
|
contacts[i].pos_y = y;
|
||
|
contacts[i].emulate_mouse = FALSE;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (i == MAX_CONTACTS)
|
||
|
return FALSE;
|
||
|
|
||
|
WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
context = (wlfContext*)instance->context;
|
||
|
RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
|
||
|
|
||
|
// Emulate mouse click if touch is not possible, like in login screen
|
||
|
if (!rdpei)
|
||
|
{
|
||
|
contacts[i].emulate_mouse = TRUE;
|
||
|
|
||
|
UINT16 flags = 0;
|
||
|
flags |= PTR_FLAGS_DOWN;
|
||
|
flags |= PTR_FLAGS_BUTTON1;
|
||
|
|
||
|
if ((flags & ~PTR_FLAGS_DOWN) != 0)
|
||
|
return freerdp_input_send_mouse_event(instance->input, flags, x, y);
|
||
|
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
rdpei->TouchBegin(rdpei, touchId, x, y, &contactId);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
BOOL wlf_handle_touch_motion(freerdp* instance, const UwacTouchMotion* ev)
|
||
|
{
|
||
|
uint32_t x, y;
|
||
|
int i;
|
||
|
int touchId;
|
||
|
int contactId;
|
||
|
|
||
|
if (!instance || !ev || !instance->context)
|
||
|
return FALSE;
|
||
|
|
||
|
x = ev->x;
|
||
|
y = ev->y;
|
||
|
touchId = ev->id;
|
||
|
|
||
|
for (i = 0; i < MAX_CONTACTS; i++)
|
||
|
{
|
||
|
if (contacts[i].id == touchId)
|
||
|
{
|
||
|
if (contacts[i].pos_x == x && contacts[i].pos_y == y)
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
contacts[i].pos_x = x;
|
||
|
contacts[i].pos_y = y;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (i == MAX_CONTACTS)
|
||
|
return FALSE;
|
||
|
|
||
|
WLog_DBG(TAG, "%s called | event_id: %u | x: %u / y: %u", __FUNCTION__, touchId, x, y);
|
||
|
|
||
|
if (!wlf_scale_coordinates(instance->context, &x, &y, TRUE))
|
||
|
return FALSE;
|
||
|
|
||
|
RdpeiClientContext* rdpei = ((wlfContext*)instance->context)->rdpei;
|
||
|
|
||
|
if (contacts[i].emulate_mouse == TRUE)
|
||
|
{
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
if (!rdpei)
|
||
|
return FALSE;
|
||
|
|
||
|
rdpei->TouchUpdate(rdpei, touchId, x, y, &contactId);
|
||
|
|
||
|
return TRUE;
|
||
|
}
|