2014-07-11 00:32:46 +04:00
|
|
|
/**
|
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
|
|
* X11 Server Input
|
|
|
|
*
|
|
|
|
* Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
|
|
|
|
#include <freerdp/locale/keyboard.h>
|
|
|
|
|
|
|
|
#include <winpr/crt.h>
|
|
|
|
#include <winpr/input.h>
|
|
|
|
|
|
|
|
#include "x11_shadow.h"
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
int x11_shadow_cursor_init(xfInfo* xfi)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
#ifdef WITH_XFIXES
|
|
|
|
int event;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
if (!XFixesQueryExtension(xfi->display, &event, &error))
|
|
|
|
{
|
|
|
|
fprintf(stderr, "XFixesQueryExtension failed\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
xfi->xfixes_notify_event = event + XFixesCursorNotify;
|
|
|
|
|
|
|
|
XFixesSelectCursorInput(xfi->display, DefaultRootWindow(xfi->display), XFixesDisplayCursorNotifyMask);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_synchronize_event(rdpInput* input, UINT32 flags)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Client sent a synchronize event (flags:0x%X)\n", flags);
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_keyboard_event(rdpInput* input, UINT16 flags, UINT16 code)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
#ifdef WITH_XTEST
|
|
|
|
DWORD vkcode;
|
|
|
|
DWORD keycode;
|
|
|
|
BOOL extended = FALSE;
|
|
|
|
xfPeerContext* xfp = (xfPeerContext*) input->context;
|
|
|
|
xfInfo* xfi = xfp->info;
|
|
|
|
|
|
|
|
if (flags & KBD_FLAGS_EXTENDED)
|
|
|
|
extended = TRUE;
|
|
|
|
|
|
|
|
if (extended)
|
|
|
|
code |= KBDEXT;
|
|
|
|
|
|
|
|
vkcode = GetVirtualKeyCodeFromVirtualScanCode(code, 4);
|
|
|
|
keycode = GetKeycodeFromVirtualKeyCode(vkcode, KEYCODE_TYPE_EVDEV);
|
|
|
|
|
|
|
|
if (keycode != 0)
|
|
|
|
{
|
|
|
|
XTestGrabControl(xfi->display, True);
|
|
|
|
|
|
|
|
if (flags & KBD_FLAGS_DOWN)
|
|
|
|
XTestFakeKeyEvent(xfi->display, keycode, True, 0);
|
|
|
|
else if (flags & KBD_FLAGS_RELEASE)
|
|
|
|
XTestFakeKeyEvent(xfi->display, keycode, False, 0);
|
|
|
|
|
|
|
|
XTestGrabControl(xfi->display, False);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_unicode_keyboard_event(rdpInput* input, UINT16 flags, UINT16 code)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Client sent a unicode keyboard event (flags:0x%X code:0x%X)\n", flags, code);
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_mouse_event(rdpInput* input, UINT16 flags, UINT16 x, UINT16 y)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
#ifdef WITH_XTEST
|
|
|
|
xfPeerContext* xfp = (xfPeerContext*) input->context;
|
|
|
|
int button = 0;
|
|
|
|
BOOL down = FALSE;
|
|
|
|
xfInfo* xfi = xfp->info;
|
|
|
|
|
|
|
|
XTestGrabControl(xfi->display, True);
|
|
|
|
|
|
|
|
if (flags & PTR_FLAGS_WHEEL)
|
|
|
|
{
|
|
|
|
BOOL negative = FALSE;
|
|
|
|
|
|
|
|
if (flags & PTR_FLAGS_WHEEL_NEGATIVE)
|
|
|
|
negative = TRUE;
|
|
|
|
|
|
|
|
button = (negative) ? 5 : 4;
|
|
|
|
|
|
|
|
XTestFakeButtonEvent(xfi->display, button, True, 0);
|
|
|
|
XTestFakeButtonEvent(xfi->display, button, False, 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (flags & PTR_FLAGS_MOVE)
|
|
|
|
XTestFakeMotionEvent(xfi->display, 0, x, y, 0);
|
|
|
|
|
|
|
|
if (flags & PTR_FLAGS_BUTTON1)
|
|
|
|
button = 1;
|
|
|
|
else if (flags & PTR_FLAGS_BUTTON2)
|
|
|
|
button = 3;
|
|
|
|
else if (flags & PTR_FLAGS_BUTTON3)
|
|
|
|
button = 2;
|
|
|
|
|
|
|
|
if (flags & PTR_FLAGS_DOWN)
|
|
|
|
down = TRUE;
|
|
|
|
|
|
|
|
if (button != 0)
|
|
|
|
XTestFakeButtonEvent(xfi->display, button, down, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
XTestGrabControl(xfi->display, False);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_extended_mouse_event(rdpInput* input, UINT16 flags, UINT16 x, UINT16 y)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
|
|
|
#ifdef WITH_XTEST
|
|
|
|
xfPeerContext* xfp = (xfPeerContext*) input->context;
|
|
|
|
int button = 0;
|
|
|
|
BOOL down = FALSE;
|
|
|
|
xfInfo* xfi = xfp->info;
|
|
|
|
|
|
|
|
XTestGrabControl(xfi->display, True);
|
|
|
|
XTestFakeMotionEvent(xfi->display, 0, x, y, CurrentTime);
|
|
|
|
|
|
|
|
if (flags & PTR_XFLAGS_BUTTON1)
|
|
|
|
button = 8;
|
|
|
|
else if (flags & PTR_XFLAGS_BUTTON2)
|
|
|
|
button = 9;
|
|
|
|
|
|
|
|
if (flags & PTR_XFLAGS_DOWN)
|
|
|
|
down = TRUE;
|
|
|
|
|
|
|
|
if (button != 0)
|
|
|
|
XTestFakeButtonEvent(xfi->display, button, down, 0);
|
|
|
|
|
|
|
|
XTestGrabControl(xfi->display, False);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-07-11 01:20:41 +04:00
|
|
|
void x11_shadow_input_register_callbacks(rdpInput* input)
|
2014-07-11 00:32:46 +04:00
|
|
|
{
|
2014-07-11 01:20:41 +04:00
|
|
|
input->SynchronizeEvent = x11_shadow_input_synchronize_event;
|
|
|
|
input->KeyboardEvent = x11_shadow_input_keyboard_event;
|
|
|
|
input->UnicodeKeyboardEvent = x11_shadow_input_unicode_keyboard_event;
|
|
|
|
input->MouseEvent = x11_shadow_input_mouse_event;
|
|
|
|
input->ExtendedMouseEvent = x11_shadow_input_extended_mouse_event;
|
2014-07-11 00:32:46 +04:00
|
|
|
}
|