/** * FreeRDP: A Remote Desktop Protocol Implementation * Wayland Input * * Copyright 2014 Manuel Bachmann * Copyright 2015 David Fort * * 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 #include #include #include #include #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; }