20#include <freerdp/config.h>
29#include <winpr/path.h>
30#include <winpr/assert.h>
31#include <winpr/cast.h>
32#include <winpr/collections.h>
34#include <freerdp/utils/string.h>
38#include <X11/keysym.h>
39#include <X11/XKBlib.h>
41#include <freerdp/locale/keyboard.h>
42#include <freerdp/locale/locale.h>
46#include "xf_keyboard.h"
49#include "keyboard_x11.h"
51#include <freerdp/log.h>
52#define TAG CLIENT_TAG("x11")
70struct x11_key_scancode_t
76static const struct x11_key_scancode_t XKB_KEY_NAME_SCANCODE_TABLE[] = {
77 {
"AB01", RDP_SCANCODE_KEY_Z },
78 {
"AB02", RDP_SCANCODE_KEY_X },
79 {
"AB03", RDP_SCANCODE_KEY_C },
80 {
"AB04", RDP_SCANCODE_KEY_V },
81 {
"AB05", RDP_SCANCODE_KEY_B },
82 {
"AB06", RDP_SCANCODE_KEY_N },
83 {
"AB07", RDP_SCANCODE_KEY_M },
84 {
"AB08", RDP_SCANCODE_OEM_COMMA },
85 {
"AB09", RDP_SCANCODE_OEM_PERIOD },
86 {
"AB10", RDP_SCANCODE_OEM_2 },
87 {
"AB11", RDP_SCANCODE_ABNT_C1 },
88 {
"AC01", RDP_SCANCODE_KEY_A },
89 {
"AC02", RDP_SCANCODE_KEY_S },
90 {
"AC03", RDP_SCANCODE_KEY_D },
91 {
"AC04", RDP_SCANCODE_KEY_F },
92 {
"AC05", RDP_SCANCODE_KEY_G },
93 {
"AC06", RDP_SCANCODE_KEY_H },
94 {
"AC07", RDP_SCANCODE_KEY_J },
95 {
"AC08", RDP_SCANCODE_KEY_K },
96 {
"AC09", RDP_SCANCODE_KEY_L },
97 {
"AC10", RDP_SCANCODE_OEM_1 },
98 {
"AC11", RDP_SCANCODE_OEM_7 },
99 {
"AD01", RDP_SCANCODE_KEY_Q },
100 {
"AD02", RDP_SCANCODE_KEY_W },
101 {
"AD03", RDP_SCANCODE_KEY_E },
102 {
"AD04", RDP_SCANCODE_KEY_R },
103 {
"AD05", RDP_SCANCODE_KEY_T },
104 {
"AD06", RDP_SCANCODE_KEY_Y },
105 {
"AD07", RDP_SCANCODE_KEY_U },
106 {
"AD08", RDP_SCANCODE_KEY_I },
107 {
"AD09", RDP_SCANCODE_KEY_O },
108 {
"AD10", RDP_SCANCODE_KEY_P },
109 {
"AD11", RDP_SCANCODE_OEM_4 },
110 {
"AD12", RDP_SCANCODE_OEM_6 },
111 {
"AE01", RDP_SCANCODE_KEY_1 },
112 {
"AE02", RDP_SCANCODE_KEY_2 },
113 {
"AE03", RDP_SCANCODE_KEY_3 },
114 {
"AE04", RDP_SCANCODE_KEY_4 },
115 {
"AE05", RDP_SCANCODE_KEY_5 },
116 {
"AE06", RDP_SCANCODE_KEY_6 },
117 {
"AE07", RDP_SCANCODE_KEY_7 },
118 {
"AE08", RDP_SCANCODE_KEY_8 },
119 {
"AE09", RDP_SCANCODE_KEY_9 },
120 {
"AE10", RDP_SCANCODE_KEY_0 },
121 {
"AE11", RDP_SCANCODE_OEM_MINUS },
122 {
"AE12", RDP_SCANCODE_OEM_PLUS },
123 {
"AE13", RDP_SCANCODE_BACKSLASH_JP },
124 {
"AGAI" , RDP_SCANCODE_UNKNOWN },
125 {
"ALT", RDP_SCANCODE_LMENU },
126 {
"BKSL", RDP_SCANCODE_OEM_5 },
127 {
"BKSP", RDP_SCANCODE_BACKSPACE },
128 {
"CAPS", RDP_SCANCODE_CAPSLOCK },
129 {
"COMP", RDP_SCANCODE_APPS },
130 {
"COPY", RDP_SCANCODE_UNKNOWN },
131 {
"CUT", RDP_SCANCODE_UNKNOWN },
132 {
"DELE", RDP_SCANCODE_DELETE },
133 {
"DOWN", RDP_SCANCODE_DOWN },
134 {
"END", RDP_SCANCODE_END },
135 {
"ESC", RDP_SCANCODE_ESCAPE },
136 {
"FIND", RDP_SCANCODE_UNKNOWN },
137 {
"FK01", RDP_SCANCODE_F1 },
138 {
"FK02", RDP_SCANCODE_F2 },
139 {
"FK03", RDP_SCANCODE_F3 },
140 {
"FK04", RDP_SCANCODE_F4 },
141 {
"FK05", RDP_SCANCODE_F5 },
142 {
"FK06", RDP_SCANCODE_F6 },
143 {
"FK07", RDP_SCANCODE_F7 },
144 {
"FK08", RDP_SCANCODE_F8 },
145 {
"FK09", RDP_SCANCODE_F9 },
146 {
"FK10", RDP_SCANCODE_F10 },
147 {
"FK11", RDP_SCANCODE_F11 },
148 {
"FK12", RDP_SCANCODE_F12 },
149 {
"FK13", RDP_SCANCODE_F13 },
150 {
"FK14", RDP_SCANCODE_F14 },
151 {
"FK15", RDP_SCANCODE_F15 },
152 {
"FK16", RDP_SCANCODE_F16 },
153 {
"FK17", RDP_SCANCODE_F17 },
154 {
"FK18", RDP_SCANCODE_F18 },
155 {
"FK19", RDP_SCANCODE_F19 },
156 {
"FK20", RDP_SCANCODE_F20 },
157 {
"FK21", RDP_SCANCODE_F21 },
158 {
"FK22", RDP_SCANCODE_F22 },
159 {
"FK23", RDP_SCANCODE_F23 },
160 {
"FK24", RDP_SCANCODE_F24 },
161 {
"FRNT", RDP_SCANCODE_UNKNOWN },
162 {
"HELP", RDP_SCANCODE_HELP },
163 {
"HENK", RDP_SCANCODE_CONVERT_JP },
164 {
"HIRA", RDP_SCANCODE_HIRAGANA },
165 {
"HJCV", RDP_SCANCODE_HANJA },
166 {
"HKTG", RDP_SCANCODE_HIRAGANA },
167 {
"HNGL", RDP_SCANCODE_HANGUL },
168 {
"HOME", RDP_SCANCODE_HOME },
169 {
"HYPR", RDP_SCANCODE_LWIN },
170 {
"I120", RDP_SCANCODE_UNKNOWN },
171 {
"I126", RDP_SCANCODE_UNKNOWN },
172 {
"I128", RDP_SCANCODE_LAUNCH_MEDIA_SELECT },
173 {
"I129", RDP_SCANCODE_ABNT_C2 },
174 {
"I147", RDP_SCANCODE_UNKNOWN },
175 {
"I148", RDP_SCANCODE_UNKNOWN },
176 {
"I149", RDP_SCANCODE_UNKNOWN },
177 {
"I150", RDP_SCANCODE_SLEEP },
178 {
"I151", RDP_SCANCODE_UNKNOWN },
179 {
"I152", RDP_SCANCODE_UNKNOWN },
180 {
"I153", RDP_SCANCODE_UNKNOWN },
181 {
"I154", RDP_SCANCODE_UNKNOWN },
182 {
"I155", RDP_SCANCODE_UNKNOWN },
183 {
"I156", RDP_SCANCODE_LAUNCH_APP1 },
184 {
"I157", RDP_SCANCODE_LAUNCH_APP2 },
185 {
"I158", RDP_SCANCODE_BROWSER_HOME },
186 {
"I159", RDP_SCANCODE_UNKNOWN },
187 {
"I160", RDP_SCANCODE_UNKNOWN },
188 {
"I161", RDP_SCANCODE_UNKNOWN },
189 {
"I162", RDP_SCANCODE_UNKNOWN },
190 {
"I163", RDP_SCANCODE_LAUNCH_MAIL },
191 {
"I164", RDP_SCANCODE_BROWSER_FAVORITES },
192 {
"I165", RDP_SCANCODE_UNKNOWN },
193 {
"I166", RDP_SCANCODE_BROWSER_BACK },
194 {
"I167", RDP_SCANCODE_BROWSER_FORWARD },
195 {
"I168", RDP_SCANCODE_UNKNOWN },
196 {
"I169", RDP_SCANCODE_UNKNOWN },
197 {
"I170", RDP_SCANCODE_UNKNOWN },
198 {
"I171", RDP_SCANCODE_MEDIA_NEXT_TRACK },
199 {
"I172", RDP_SCANCODE_MEDIA_PLAY_PAUSE },
200 {
"I173", RDP_SCANCODE_MEDIA_PREV_TRACK },
201 {
"I174", RDP_SCANCODE_MEDIA_STOP },
202 {
"I175", RDP_SCANCODE_UNKNOWN },
203 {
"I176", RDP_SCANCODE_UNKNOWN },
204 {
"I177", RDP_SCANCODE_UNKNOWN },
205 {
"I178", RDP_SCANCODE_UNKNOWN },
206 {
"I179", RDP_SCANCODE_UNKNOWN },
207 {
"I180", RDP_SCANCODE_BROWSER_HOME },
208 {
"I181", RDP_SCANCODE_BROWSER_REFRESH },
209 {
"I182", RDP_SCANCODE_UNKNOWN },
210 {
"I183", RDP_SCANCODE_UNKNOWN },
211 {
"I184", RDP_SCANCODE_UNKNOWN },
212 {
"I185", RDP_SCANCODE_UNKNOWN },
213 {
"I186", RDP_SCANCODE_UNKNOWN },
214 {
"I187", RDP_SCANCODE_UNKNOWN },
215 {
"I188", RDP_SCANCODE_UNKNOWN },
216 {
"I189", RDP_SCANCODE_UNKNOWN },
217 {
"I190", RDP_SCANCODE_UNKNOWN },
218 {
"I208", RDP_SCANCODE_MEDIA_PLAY_PAUSE },
219 {
"I209", RDP_SCANCODE_MEDIA_PLAY_PAUSE },
220 {
"I210", RDP_SCANCODE_UNKNOWN },
221 {
"I211", RDP_SCANCODE_UNKNOWN },
222 {
"I212", RDP_SCANCODE_UNKNOWN },
223 {
"I213", RDP_SCANCODE_UNKNOWN },
224 {
"I214", RDP_SCANCODE_UNKNOWN },
225 {
"I215", RDP_SCANCODE_MEDIA_PLAY_PAUSE },
226 {
"I216", RDP_SCANCODE_MEDIA_NEXT_TRACK },
227 {
"I217", RDP_SCANCODE_UNKNOWN },
228 {
"I218", RDP_SCANCODE_UNKNOWN },
229 {
"I219", RDP_SCANCODE_UNKNOWN },
230 {
"I220", RDP_SCANCODE_UNKNOWN },
231 {
"I221", RDP_SCANCODE_UNKNOWN },
232 {
"I222", RDP_SCANCODE_UNKNOWN },
233 {
"I223", RDP_SCANCODE_LAUNCH_MAIL },
234 {
"I224", RDP_SCANCODE_UNKNOWN },
235 {
"I225", RDP_SCANCODE_BROWSER_SEARCH },
236 {
"I226", RDP_SCANCODE_UNKNOWN },
237 {
"I227", RDP_SCANCODE_UNKNOWN },
238 {
"I228", RDP_SCANCODE_UNKNOWN },
239 {
"I229", RDP_SCANCODE_UNKNOWN },
240 {
"I230", RDP_SCANCODE_UNKNOWN },
241 {
"I231", RDP_SCANCODE_UNKNOWN },
242 {
"I232", RDP_SCANCODE_UNKNOWN },
243 {
"I233", RDP_SCANCODE_UNKNOWN },
244 {
"I234", RDP_SCANCODE_LAUNCH_MEDIA_SELECT },
245 {
"I235", RDP_SCANCODE_UNKNOWN },
246 {
"I236", RDP_SCANCODE_UNKNOWN },
247 {
"I237", RDP_SCANCODE_UNKNOWN },
248 {
"I238", RDP_SCANCODE_UNKNOWN },
249 {
"I239", RDP_SCANCODE_UNKNOWN },
250 {
"I240", RDP_SCANCODE_UNKNOWN },
251 {
"I241", RDP_SCANCODE_UNKNOWN },
252 {
"I242", RDP_SCANCODE_UNKNOWN },
253 {
"I243", RDP_SCANCODE_UNKNOWN },
254 {
"I244", RDP_SCANCODE_UNKNOWN },
255 {
"I245", RDP_SCANCODE_UNKNOWN },
256 {
"I246", RDP_SCANCODE_UNKNOWN },
257 {
"I247", RDP_SCANCODE_UNKNOWN },
258 {
"I248", RDP_SCANCODE_UNKNOWN },
259 {
"I249", RDP_SCANCODE_UNKNOWN },
260 {
"I250", RDP_SCANCODE_UNKNOWN },
261 {
"I251", RDP_SCANCODE_UNKNOWN },
262 {
"I252", RDP_SCANCODE_UNKNOWN },
263 {
"I253", RDP_SCANCODE_UNKNOWN },
264 {
"I254", RDP_SCANCODE_UNKNOWN },
265 {
"I255", RDP_SCANCODE_UNKNOWN },
266 {
"INS", RDP_SCANCODE_INSERT },
267 {
"JPCM", RDP_SCANCODE_UNKNOWN },
268 {
"KATA", RDP_SCANCODE_KANA_HANGUL },
269 {
"KP0", RDP_SCANCODE_NUMPAD0 },
270 {
"KP1", RDP_SCANCODE_NUMPAD1 },
271 {
"KP2", RDP_SCANCODE_NUMPAD2 },
272 {
"KP3", RDP_SCANCODE_NUMPAD3 },
273 {
"KP4", RDP_SCANCODE_NUMPAD4 },
274 {
"KP5", RDP_SCANCODE_NUMPAD5 },
275 {
"KP6", RDP_SCANCODE_NUMPAD6 },
276 {
"KP7", RDP_SCANCODE_NUMPAD7 },
277 {
"KP8", RDP_SCANCODE_NUMPAD8 },
278 {
"KP9", RDP_SCANCODE_NUMPAD9 },
279 {
"KPAD", RDP_SCANCODE_ADD },
280 {
"KPDL", RDP_SCANCODE_DECIMAL },
281 {
"KPDV", RDP_SCANCODE_DIVIDE },
282 {
"KPEN", RDP_SCANCODE_RETURN_KP },
283 {
"KPEQ", RDP_SCANCODE_UNKNOWN },
284 {
"KPMU", RDP_SCANCODE_MULTIPLY },
285 {
"KPSU", RDP_SCANCODE_SUBTRACT },
286 {
"LALT", RDP_SCANCODE_LMENU },
287 {
"LCTL", RDP_SCANCODE_LCONTROL },
288 {
"LEFT", RDP_SCANCODE_LEFT },
289 {
"LFSH", RDP_SCANCODE_LSHIFT },
290 {
"LNFD", RDP_SCANCODE_UNKNOWN },
291 {
"LSGT", RDP_SCANCODE_OEM_102 },
292 {
"LVL3", RDP_SCANCODE_RMENU },
293 {
"LVL5", RDP_SCANCODE_UNKNOWN },
294 {
"LWIN", RDP_SCANCODE_LWIN },
295 {
"META", RDP_SCANCODE_LMENU },
296 {
"MUHE", RDP_SCANCODE_NONCONVERT_JP },
297 {
"MUTE", RDP_SCANCODE_VOLUME_MUTE },
298 {
"NMLK", RDP_SCANCODE_NUMLOCK },
299 {
"OPEN", RDP_SCANCODE_UNKNOWN },
300 {
"PAST", RDP_SCANCODE_UNKNOWN },
301 {
"PAUS", RDP_SCANCODE_PAUSE },
302 {
"PGDN", RDP_SCANCODE_NEXT },
303 {
"PGUP", RDP_SCANCODE_PRIOR },
304 {
"POWR", RDP_SCANCODE_UNKNOWN },
305 {
"PROP", RDP_SCANCODE_UNKNOWN },
306 {
"PRSC", RDP_SCANCODE_PRINTSCREEN },
307 {
"RALT", RDP_SCANCODE_RMENU },
308 {
"RCTL", RDP_SCANCODE_RCONTROL },
309 {
"RGHT", RDP_SCANCODE_RIGHT },
310 {
"RTRN", RDP_SCANCODE_RETURN },
311 {
"RTSH", RDP_SCANCODE_RSHIFT },
312 {
"RWIN", RDP_SCANCODE_RWIN },
313 {
"SCLK", RDP_SCANCODE_SCROLLLOCK },
314 {
"SPCE", RDP_SCANCODE_SPACE },
315 {
"STOP", RDP_SCANCODE_BROWSER_STOP },
316 {
"SUPR", RDP_SCANCODE_LWIN },
317 {
"TAB", RDP_SCANCODE_TAB },
318 {
"TLDE", RDP_SCANCODE_OEM_3 },
319 {
"UNDO", RDP_SCANCODE_UNKNOWN },
320 {
"UP", RDP_SCANCODE_UP },
321 {
"VOL+", RDP_SCANCODE_VOLUME_UP },
322 {
"VOL-", RDP_SCANCODE_VOLUME_DOWN }
329} x11_keysym_scancode_t;
331static const x11_keysym_scancode_t KEYSYM_SCANCODE_TABLE[] = {
332 { XK_space, RDP_SCANCODE_SPACE },
333 { XK_apostrophe, RDP_SCANCODE_OEM_7 },
334 { XK_comma, RDP_SCANCODE_OEM_COMMA },
335 { XK_minus, RDP_SCANCODE_OEM_MINUS },
336 { XK_period, RDP_SCANCODE_OEM_PERIOD },
337 { XK_slash, RDP_SCANCODE_OEM_2 },
338 { XK_0, RDP_SCANCODE_KEY_0 },
339 { XK_1, RDP_SCANCODE_KEY_1 },
340 { XK_2, RDP_SCANCODE_KEY_2 },
341 { XK_3, RDP_SCANCODE_KEY_3 },
342 { XK_4, RDP_SCANCODE_KEY_4 },
343 { XK_5, RDP_SCANCODE_KEY_5 },
344 { XK_6, RDP_SCANCODE_KEY_6 },
345 { XK_7, RDP_SCANCODE_KEY_7 },
346 { XK_8, RDP_SCANCODE_KEY_8 },
347 { XK_9, RDP_SCANCODE_KEY_9 },
348 { XK_semicolon, RDP_SCANCODE_OEM_1 },
349 { XK_less, RDP_SCANCODE_OEM_102 },
350 { XK_equal, RDP_SCANCODE_OEM_PLUS },
351 { XK_A, RDP_SCANCODE_KEY_A },
352 { XK_B, RDP_SCANCODE_KEY_B },
353 { XK_C, RDP_SCANCODE_KEY_C },
354 { XK_D, RDP_SCANCODE_KEY_D },
355 { XK_E, RDP_SCANCODE_KEY_E },
356 { XK_F, RDP_SCANCODE_KEY_F },
357 { XK_G, RDP_SCANCODE_KEY_G },
358 { XK_H, RDP_SCANCODE_KEY_H },
359 { XK_I, RDP_SCANCODE_KEY_I },
360 { XK_J, RDP_SCANCODE_KEY_J },
361 { XK_K, RDP_SCANCODE_KEY_K },
362 { XK_L, RDP_SCANCODE_KEY_L },
363 { XK_M, RDP_SCANCODE_KEY_M },
364 { XK_N, RDP_SCANCODE_KEY_N },
365 { XK_O, RDP_SCANCODE_KEY_O },
366 { XK_P, RDP_SCANCODE_KEY_P },
367 { XK_Q, RDP_SCANCODE_KEY_Q },
368 { XK_R, RDP_SCANCODE_KEY_R },
369 { XK_S, RDP_SCANCODE_KEY_S },
370 { XK_T, RDP_SCANCODE_KEY_T },
371 { XK_U, RDP_SCANCODE_KEY_U },
372 { XK_V, RDP_SCANCODE_KEY_V },
373 { XK_W, RDP_SCANCODE_KEY_W },
374 { XK_X, RDP_SCANCODE_KEY_X },
375 { XK_Y, RDP_SCANCODE_KEY_Y },
376 { XK_Z, RDP_SCANCODE_KEY_Z },
377 { XK_bracketleft, RDP_SCANCODE_OEM_4 },
378 { XK_backslash, RDP_SCANCODE_OEM_5 },
379 { XK_bracketright, RDP_SCANCODE_OEM_6 },
380 { XK_grave, RDP_SCANCODE_OEM_3 },
381 { XK_a, RDP_SCANCODE_KEY_A },
382 { XK_b, RDP_SCANCODE_KEY_B },
383 { XK_c, RDP_SCANCODE_KEY_C },
384 { XK_d, RDP_SCANCODE_KEY_D },
385 { XK_e, RDP_SCANCODE_KEY_E },
386 { XK_f, RDP_SCANCODE_KEY_F },
387 { XK_g, RDP_SCANCODE_KEY_G },
388 { XK_h, RDP_SCANCODE_KEY_H },
389 { XK_i, RDP_SCANCODE_KEY_I },
390 { XK_j, RDP_SCANCODE_KEY_J },
391 { XK_k, RDP_SCANCODE_KEY_K },
392 { XK_l, RDP_SCANCODE_KEY_L },
393 { XK_m, RDP_SCANCODE_KEY_M },
394 { XK_n, RDP_SCANCODE_KEY_N },
395 { XK_o, RDP_SCANCODE_KEY_O },
396 { XK_p, RDP_SCANCODE_KEY_P },
397 { XK_q, RDP_SCANCODE_KEY_Q },
398 { XK_r, RDP_SCANCODE_KEY_R },
399 { XK_s, RDP_SCANCODE_KEY_S },
400 { XK_t, RDP_SCANCODE_KEY_T },
401 { XK_u, RDP_SCANCODE_KEY_U },
402 { XK_v, RDP_SCANCODE_KEY_V },
403 { XK_w, RDP_SCANCODE_KEY_W },
404 { XK_x, RDP_SCANCODE_KEY_X },
405 { XK_y, RDP_SCANCODE_KEY_Y },
406 { XK_z, RDP_SCANCODE_KEY_Z },
407 { XK_ISO_Level3_Shift, RDP_SCANCODE_RMENU },
408 { XK_ISO_Left_Tab, RDP_SCANCODE_TAB },
409 { XK_BackSpace, RDP_SCANCODE_BACKSPACE },
410 { XK_Tab, RDP_SCANCODE_TAB },
411 { XK_Return, RDP_SCANCODE_RETURN },
412 { XK_Pause, RDP_SCANCODE_PAUSE },
413 { XK_Scroll_Lock, RDP_SCANCODE_SCROLLLOCK },
414 { XK_Escape, RDP_SCANCODE_ESCAPE },
415 { XK_Home, RDP_SCANCODE_HOME },
416 { XK_Left, RDP_SCANCODE_LEFT },
417 { XK_Up, RDP_SCANCODE_UP },
418 { XK_Right, RDP_SCANCODE_RIGHT },
419 { XK_Down, RDP_SCANCODE_DOWN },
420 { XK_Prior, RDP_SCANCODE_PRIOR },
421 { XK_Next, RDP_SCANCODE_NEXT },
422 { XK_End, RDP_SCANCODE_END },
423 { XK_Print, RDP_SCANCODE_PRINTSCREEN },
424 { XK_Insert, RDP_SCANCODE_INSERT },
425 { XK_Menu, RDP_SCANCODE_APPS },
426 { XK_Help, RDP_SCANCODE_HELP },
427 { XK_Mode_switch, RDP_SCANCODE_RMENU },
428 { XK_Num_Lock, RDP_SCANCODE_NUMLOCK },
429 { XK_KP_Enter, RDP_SCANCODE_RETURN_KP },
430 { XK_KP_Home, RDP_SCANCODE_NUMPAD7 },
431 { XK_KP_Left, RDP_SCANCODE_NUMPAD4 },
432 { XK_KP_Up, RDP_SCANCODE_NUMPAD8 },
433 { XK_KP_Right, RDP_SCANCODE_NUMPAD6 },
434 { XK_KP_Down, RDP_SCANCODE_NUMPAD2 },
435 { XK_KP_Prior, RDP_SCANCODE_NUMPAD9 },
436 { XK_KP_Next, RDP_SCANCODE_NUMPAD3 },
437 { XK_KP_End, RDP_SCANCODE_NUMPAD1 },
438 { XK_KP_Begin, RDP_SCANCODE_NUMPAD5 },
439 { XK_KP_Insert, RDP_SCANCODE_NUMPAD0 },
440 { XK_KP_Delete, RDP_SCANCODE_DECIMAL },
441 { XK_KP_Multiply, RDP_SCANCODE_MULTIPLY },
442 { XK_KP_Add, RDP_SCANCODE_ADD },
443 { XK_KP_Separator, RDP_SCANCODE_DECIMAL },
444 { XK_KP_Subtract, RDP_SCANCODE_SUBTRACT },
445 { XK_KP_Decimal, RDP_SCANCODE_DECIMAL },
446 { XK_KP_Divide, RDP_SCANCODE_DIVIDE },
447 { XK_KP_0, RDP_SCANCODE_NUMPAD0 },
448 { XK_KP_1, RDP_SCANCODE_NUMPAD1 },
449 { XK_KP_2, RDP_SCANCODE_NUMPAD2 },
450 { XK_KP_3, RDP_SCANCODE_NUMPAD3 },
451 { XK_KP_4, RDP_SCANCODE_NUMPAD4 },
452 { XK_KP_5, RDP_SCANCODE_NUMPAD5 },
453 { XK_KP_6, RDP_SCANCODE_NUMPAD6 },
454 { XK_KP_7, RDP_SCANCODE_NUMPAD7 },
455 { XK_KP_8, RDP_SCANCODE_NUMPAD8 },
456 { XK_KP_9, RDP_SCANCODE_NUMPAD9 },
457 { XK_F1, RDP_SCANCODE_F1 },
458 { XK_F2, RDP_SCANCODE_F2 },
459 { XK_F3, RDP_SCANCODE_F3 },
460 { XK_F4, RDP_SCANCODE_F4 },
461 { XK_F5, RDP_SCANCODE_F5 },
462 { XK_F6, RDP_SCANCODE_F6 },
463 { XK_F7, RDP_SCANCODE_F7 },
464 { XK_F8, RDP_SCANCODE_F8 },
465 { XK_F9, RDP_SCANCODE_F9 },
466 { XK_F10, RDP_SCANCODE_F10 },
467 { XK_F11, RDP_SCANCODE_F11 },
468 { XK_F12, RDP_SCANCODE_F12 },
469 { XK_F13, RDP_SCANCODE_F13 },
470 { XK_F14, RDP_SCANCODE_F14 },
471 { XK_F15, RDP_SCANCODE_F15 },
472 { XK_F16, RDP_SCANCODE_F16 },
473 { XK_F17, RDP_SCANCODE_F17 },
474 { XK_F18, RDP_SCANCODE_F18 },
475 { XK_F19, RDP_SCANCODE_F19 },
476 { XK_F20, RDP_SCANCODE_F20 },
477 { XK_F21, RDP_SCANCODE_F21 },
478 { XK_F22, RDP_SCANCODE_F22 },
479 { XK_F23, RDP_SCANCODE_F23 },
480 { XK_F24, RDP_SCANCODE_F24 },
481 { XK_Shift_L, RDP_SCANCODE_LSHIFT },
482 { XK_Shift_R, RDP_SCANCODE_RSHIFT },
483 { XK_Control_L, RDP_SCANCODE_LCONTROL },
484 { XK_Control_R, RDP_SCANCODE_RCONTROL },
485 { XK_Caps_Lock, RDP_SCANCODE_CAPSLOCK },
486 { XK_Meta_L, RDP_SCANCODE_LWIN },
487 { XK_Meta_R, RDP_SCANCODE_RWIN },
488 { XK_Alt_L, RDP_SCANCODE_LMENU },
489 { XK_Alt_R, RDP_SCANCODE_RMENU },
490 { XK_Super_L, RDP_SCANCODE_LWIN },
491 { XK_Super_R, RDP_SCANCODE_RWIN },
492 { XK_Hyper_L, RDP_SCANCODE_LWIN },
493 { XK_Hyper_R, RDP_SCANCODE_RWIN },
494 { XK_Delete, RDP_SCANCODE_DELETE }
497static UINT32 xf_keyboard_get_toggle_keys_state(xfContext* xfc);
498static BOOL xf_keyboard_handle_special_keys(xfContext* xfc, KeySym keysym);
499static void xf_keyboard_handle_special_keys_release(xfContext* xfc, KeySym keysym);
501static void xf_keyboard_modifier_map_free(xfContext* xfc)
504 if (xfc->modifierMap)
506 XFreeModifiermap(xfc->modifierMap);
507 xfc->modifierMap =
nullptr;
511BOOL xf_keyboard_update_modifier_map(xfContext* xfc)
514 xf_keyboard_modifier_map_free(xfc);
515 xfc->modifierMap = XGetModifierMapping(xfc->display);
516 return xfc->modifierMap !=
nullptr;
519static void xf_keyboard_send_key(xfContext* xfc, BOOL down, BOOL repeat,
const XKeyEvent* ev);
521static BOOL xf_sync_kbd_state(xfContext* xfc)
523 const UINT32 syncFlags = xf_keyboard_get_toggle_keys_state(xfc);
526 return freerdp_input_send_synchronize_event(xfc->common.context.input, syncFlags);
529static void xf_keyboard_clear(xfContext* xfc)
532 ZeroMemory(xfc->KeyboardState,
sizeof(xfc->KeyboardState));
535static BOOL xf_action_script_append(xfContext* xfc,
const char* buffer,
size_t size,
536 WINPR_ATTR_UNUSED
void* user,
const char* what,
const char* arg)
542 if (!buffer || (size == 0))
544 return ArrayList_Append(xfc->keyCombinations, buffer);
547static void xf_keyboard_action_script_free(xfContext* xfc)
549 xf_event_action_script_free(xfc);
551 if (xfc->keyCombinations)
553 ArrayList_Free(xfc->keyCombinations);
554 xfc->keyCombinations =
nullptr;
555 xfc->actionScriptExists = FALSE;
559BOOL xf_keyboard_action_script_init(xfContext* xfc)
563 xf_keyboard_action_script_free(xfc);
564 xfc->keyCombinations = ArrayList_New(TRUE);
566 if (!xfc->keyCombinations)
569 wObject* obj = ArrayList_Object(xfc->keyCombinations);
574 if (!run_action_script(xfc,
"key",
nullptr, xf_action_script_append,
nullptr))
577 return xf_event_action_script_init(xfc);
580static int xkb_cmp(
const void* pva,
const void* pvb)
582 const struct x11_key_scancode_t* a = pva;
583 const struct x11_key_scancode_t* b = pvb;
591 return strcmp(a->name, b->name);
594static int keysym_cmp(
const void* pva,
const void* pvb)
596 const x11_keysym_scancode_t* a = pva;
597 const x11_keysym_scancode_t* b = pvb;
605 if (a->keysym < b->keysym)
607 if (a->keysym > b->keysym)
612static BOOL try_add(xfContext* xfc,
size_t offset,
const char* xkb_keyname)
616 struct x11_key_scancode_t key = { .name = xkb_keyname,
617 .sc = WINPR_ASSERTING_INT_CAST(uint32_t, offset) };
619 struct x11_key_scancode_t* found =
620 bsearch(&key, XKB_KEY_NAME_SCANCODE_TABLE, ARRAYSIZE(XKB_KEY_NAME_SCANCODE_TABLE),
621 sizeof(
struct x11_key_scancode_t), xkb_cmp);
624 WLog_Print(xfc->log, WLOG_DEBUG,
625 "%4s: keycode: 0x%02" PRIuz
" -> rdp scancode: 0x%08" PRIx32
"", xkb_keyname,
627 xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE[offset] = found->sc;
635static DWORD xf_keysym_to_rdp_scancode(KeySym keysym)
637 x11_keysym_scancode_t key = { .keysym = keysym, .sc = 0 };
638 x11_keysym_scancode_t* found =
639 bsearch(&key, KEYSYM_SCANCODE_TABLE, ARRAYSIZE(KEYSYM_SCANCODE_TABLE),
640 sizeof(x11_keysym_scancode_t), keysym_cmp);
644 return RDP_SCANCODE_UNKNOWN;
647static int load_map_from_xkbfile(xfContext* xfc)
655 XkbDescPtr xkb = XkbGetMap(xfc->display, 0, XkbUseCoreKbd);
658 WLog_Print(xfc->log, WLOG_WARN,
"XkbGetMap() == nullptr");
662 const int rc = XkbGetNames(xfc->display, XkbKeyNamesMask, xkb);
665 char buffer[64] = WINPR_C_ARRAY_INIT;
666 WLog_Print(xfc->log, WLOG_WARN,
"XkbGetNames() != Success: [%s]",
667 x11_error_to_string(xfc, rc, buffer,
sizeof(buffer)));
671 char xkb_keyname[XkbKeyNameLength + 1] = { 42, 42, 42, 42,
674 WLog_Print(xfc->log, WLOG_TRACE,
"XkbGetNames() == Success, min=%" PRIu8
", max=%" PRIu8,
675 xkb->min_key_code, xkb->max_key_code);
676 for (
size_t i = xkb->min_key_code; i < xkb->max_key_code; i++)
679 strncpy(xkb_keyname, xkb->names->keys[i].name, XkbKeyNameLength);
681 WLog_Print(xfc->log, WLOG_TRACE,
"KeyCode %" PRIuz
" -> %s", i, xkb_keyname);
682 if (strnlen(xkb_keyname, ARRAYSIZE(xkb_keyname)) >= 1)
683 found = try_add(xfc, i, xkb_keyname);
687#if defined(__APPLE__)
689 GetVirtualKeyCodeFromKeycode((UINT32)i - 8u, WINPR_KEYCODE_TYPE_APPLE);
690 xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE[i] =
691 GetVirtualScanCodeFromVirtualKeyCode(vkcode, WINPR_KBD_TYPE_IBM_ENHANCED);
697 WLog_Print(xfc->log, WLOG_WARN,
698 "%4s: keycode: 0x%02" PRIx32
" -> no RDP scancode found", xkb_keyname,
699 WINPR_ASSERTING_INT_CAST(UINT32, i));
706 XkbFreeKeyboard(xkb, 0, 1);
712static BOOL load_map_from_keysym(xfContext* xfc)
722 XDisplayKeycodes(xfc->display, &min_kc, &max_kc);
724 for (
int i = min_kc; i <= max_kc; i++)
726 if ((i < 0) || ((
size_t)i >= ARRAYSIZE(xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE)))
729 if (xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE[i] != RDP_SCANCODE_UNKNOWN)
732 const KeySym ks = XkbKeycodeToKeysym(xfc->display, (KeyCode)i, 0, 0);
736 const DWORD sc = xf_keysym_to_rdp_scancode(ks);
737 if (sc != RDP_SCANCODE_UNKNOWN)
739 xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE[i] = sc;
741 WLog_Print(xfc->log, WLOG_DEBUG,
742 "keycode: 0x%02x -> keysym: 0x%04lx -> rdp scancode: 0x%08" PRIx32
743 " (keysym fallback)",
744 (
unsigned)i, (
unsigned long)ks, sc);
751#if defined(WITH_VERBOSE_WINPR_ASSERT)
752static BOOL compareKeySym(
const x11_keysym_scancode_t* a,
size_t counta,
753 const x11_keysym_scancode_t* b,
size_t countb)
755 WINPR_ASSERT(a || (counta == 0));
756 WINPR_ASSERT(b || (countb == 0));
758 if (counta != countb)
761 for (
size_t x = 0; x < counta; x++)
763 const x11_keysym_scancode_t* ca = &a[x];
764 const x11_keysym_scancode_t* cb = &b[x];
765 if (keysym_cmp(ca, cb) != 0)
767 WLog_ERR(TAG,
"%" PRIuz
"\ta=%lx, should be %lx", x, ca->keysym, cb->keysym);
768 WLog_ERR(TAG,
"KEYSYM_SCANCODE_TABLE is not properly sorted!");
775static BOOL compareKey(
const struct x11_key_scancode_t* a,
size_t counta,
776 const struct x11_key_scancode_t* b,
size_t countb)
778 WINPR_ASSERT(a || (counta == 0));
779 WINPR_ASSERT(b || (countb == 0));
781 if (counta != countb)
784 for (
size_t x = 0; x < counta; x++)
786 const struct x11_key_scancode_t* ca = &a[x];
787 const struct x11_key_scancode_t* cb = &b[x];
788 if (xkb_cmp(ca, cb) != 0)
790 WLog_ERR(TAG,
"%" PRIuz
"\ta=%s [%" PRIu32
"], should be %s [%" PRIu32
"]", x, a->name,
791 a->sc, b->name, b->sc);
792 WLog_ERR(TAG,
"XKB_KEY_NAME_SCANCODE_TABLE is not properly sorted!");
800BOOL xf_keyboard_init(xfContext* xfc)
802 rdpSettings* settings =
nullptr;
807#if defined(WITH_VERBOSE_WINPR_ASSERT)
809 x11_keysym_scancode_t copy[ARRAYSIZE(KEYSYM_SCANCODE_TABLE)] = WINPR_C_ARRAY_INIT;
810 memcpy(copy, KEYSYM_SCANCODE_TABLE,
sizeof(copy));
811 qsort(copy, ARRAYSIZE(copy),
sizeof(x11_keysym_scancode_t), keysym_cmp);
813 if (!compareKeySym(KEYSYM_SCANCODE_TABLE, ARRAYSIZE(KEYSYM_SCANCODE_TABLE), copy,
818 struct x11_key_scancode_t copy[ARRAYSIZE(XKB_KEY_NAME_SCANCODE_TABLE)] = WINPR_C_ARRAY_INIT;
819 memcpy(copy, XKB_KEY_NAME_SCANCODE_TABLE,
sizeof(copy));
820 qsort(copy, ARRAYSIZE(copy),
sizeof(
struct x11_key_scancode_t), xkb_cmp);
822 if (!compareKey(XKB_KEY_NAME_SCANCODE_TABLE, ARRAYSIZE(XKB_KEY_NAME_SCANCODE_TABLE), copy,
828 settings = xfc->common.context.settings;
829 WINPR_ASSERT(settings);
831 xf_keyboard_clear(xfc);
841 if (KeyboardLayout == 0)
843 xf_detect_keyboard_layout_from_xkb(xfc->log, &KeyboardLayout);
844 if (KeyboardLayout == 0)
845 freerdp_detect_keyboard_layout_from_system_locale(&KeyboardLayout);
846 if (KeyboardLayout == 0)
847 KeyboardLayout = ENGLISH_UNITED_STATES;
852 const int rc = load_map_from_xkbfile(xfc);
854 const BOOL keysym_mapped = load_map_from_keysym(xfc);
856 if (rc != 0 && !keysym_mapped)
859 return xf_keyboard_update_modifier_map(xfc);
862void xf_keyboard_free(xfContext* xfc)
864 xf_keyboard_modifier_map_free(xfc);
865 xf_keyboard_action_script_free(xfc);
868void xf_keyboard_key_press(xfContext* xfc,
const XKeyEvent* event, KeySym keysym)
874 WINPR_ASSERT(event->keycode < ARRAYSIZE(xfc->KeyboardState));
876 last = xfc->KeyboardState[
event->keycode];
877 xfc->KeyboardState[
event->keycode] = TRUE;
879 if (xf_keyboard_handle_special_keys(xfc, keysym))
882 xf_keyboard_send_key(xfc, TRUE, last, event);
885void xf_keyboard_key_release(xfContext* xfc,
const XKeyEvent* event, KeySym keysym)
889 WINPR_ASSERT(event->keycode < ARRAYSIZE(xfc->KeyboardState));
891 BOOL last = xfc->KeyboardState[
event->keycode];
892 xfc->KeyboardState[
event->keycode] = FALSE;
893 xf_keyboard_handle_special_keys_release(xfc, keysym);
894 xf_keyboard_send_key(xfc, FALSE, last, event);
897static DWORD get_rdp_scancode_from_x11_keycode(xfContext* xfc, DWORD keycode)
900 if (keycode >= ARRAYSIZE(xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE))
902 WLog_ERR(TAG,
"KeyCode %" PRIu32
" exceeds allowed value range [0,%" PRIuz
"]", keycode,
903 ARRAYSIZE(xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE));
907 const DWORD scancode = xfc->X11_KEYCODE_TO_VIRTUAL_SCANCODE[keycode];
908 const DWORD remapped = freerdp_keyboard_remap_key(xfc->remap_table, scancode);
910#if defined(WITH_DEBUG_KBD)
912 const BOOL ex = RDP_SCANCODE_EXTENDED(scancode);
913 const DWORD sc = RDP_SCANCODE_CODE(scancode);
914 WLog_DBG(TAG,
"x11 keycode: %02" PRIX32
" -> rdp code: [%04" PRIx16
"] %02" PRIX8
"%s",
915 keycode, scancode, sc, ex ?
" extended" :
"");
921#if defined(WITH_DEBUG_KBD)
923 const BOOL ex = RDP_SCANCODE_EXTENDED(remapped);
924 const DWORD sc = RDP_SCANCODE_CODE(remapped);
926 "x11 keycode: %02" PRIX32
" -> remapped rdp code: [%04" PRIx16
"] %02" PRIX8
928 keycode, remapped, sc, ex ?
" extended" :
"");
937void xf_keyboard_release_all_keypress(xfContext* xfc)
941 WINPR_STATIC_ASSERT(ARRAYSIZE(xfc->KeyboardState) <= UINT32_MAX);
942 for (
size_t keycode = 0; keycode < ARRAYSIZE(xfc->KeyboardState); keycode++)
944 if (xfc->KeyboardState[keycode])
946 const DWORD rdp_scancode =
947 get_rdp_scancode_from_x11_keycode(xfc, WINPR_ASSERTING_INT_CAST(UINT32, keycode));
951 if (rdp_scancode == RDP_SCANCODE_LWIN)
952 freerdp_input_send_keyboard_event_ex(xfc->common.context.input, FALSE, FALSE,
955 freerdp_input_send_keyboard_event_ex(xfc->common.context.input, FALSE, FALSE,
957 xfc->KeyboardState[keycode] = FALSE;
960 xf_sync_kbd_state(xfc);
963static BOOL xf_keyboard_key_pressed(xfContext* xfc, KeySym keysym)
965 KeyCode keycode = XKeysymToKeycode(xfc->display, keysym);
966 WINPR_ASSERT(keycode < ARRAYSIZE(xfc->KeyboardState));
967 return xfc->KeyboardState[keycode];
970void xf_keyboard_send_key(xfContext* xfc, BOOL down, BOOL repeat,
const XKeyEvent* event)
975 rdpInput* input = xfc->common.context.input;
978 const DWORD rdp_scancode = get_rdp_scancode_from_x11_keycode(xfc, event->keycode);
979 if (rdp_scancode == RDP_SCANCODE_PAUSE && !xf_keyboard_key_pressed(xfc, XK_Control_L) &&
980 !xf_keyboard_key_pressed(xfc, XK_Control_R))
988 freerdp_input_send_keyboard_pause_event(input);
995 wchar_t buffer[32] = WINPR_C_ARRAY_INIT;
998 switch (rdp_scancode)
1000 case RDP_SCANCODE_RETURN:
1004 XIM xim = XOpenIM(xfc->display,
nullptr,
nullptr,
nullptr);
1007 WLog_WARN(TAG,
"Failed to XOpenIM");
1011 XIC xic = XCreateIC(xim, XNInputStyle, XIMPreeditNothing | XIMStatusNothing,
1015 WLog_WARN(TAG,
"XCreateIC failed");
1019 KeySym ignore = WINPR_C_ARRAY_INIT;
1020 Status return_status = 0;
1021 XKeyEvent ev = *event;
1023 xwc = XwcLookupString(xic, &ev, buffer, ARRAYSIZE(buffer), &ignore,
1035 if (rdp_scancode == RDP_SCANCODE_UNKNOWN)
1036 WLog_ERR(TAG,
"Unknown key with X keycode 0x%02" PRIx8
"", event->keycode);
1038 freerdp_input_send_keyboard_event_ex(input, down, repeat, rdp_scancode);
1042 char str[3 * ARRAYSIZE(buffer)] = WINPR_C_ARRAY_INIT;
1044 const size_t rc = wcstombs(str, buffer, ARRAYSIZE(buffer));
1046 WCHAR wbuffer[ARRAYSIZE(buffer)] = WINPR_C_ARRAY_INIT;
1047 (void)ConvertUtf8ToWChar(str, wbuffer, rc);
1048 freerdp_input_send_unicode_keyboard_event(input, down ? 0 : KBD_FLAGS_RELEASE,
1052 else if (rdp_scancode == RDP_SCANCODE_UNKNOWN)
1053 WLog_ERR(TAG,
"Unknown key with X keycode 0x%02" PRIx8
"", event->keycode);
1055 freerdp_input_send_keyboard_event_ex(input, down, repeat, rdp_scancode);
1059static int xf_keyboard_read_keyboard_state(xfContext* xfc)
1065 if (!xfc->remote_app && xfc->window)
1067 XQueryPointer(xfc->display, xfc->window->handle, &wdummy, &wdummy, &dummy, &dummy, &dummy,
1072 XQueryPointer(xfc->display, DefaultRootWindow(xfc->display), &wdummy, &wdummy, &dummy,
1073 &dummy, &dummy, &dummy, &state);
1076 return WINPR_ASSERTING_INT_CAST(
int, state);
1079static int xf_keyboard_get_keymask(xfContext* xfc, KeySym keysym)
1082 KeyCode keycode = XKeysymToKeycode(xfc->display, keysym);
1084 if (keycode == NoSymbol)
1087 WINPR_ASSERT(xfc->modifierMap);
1088 for (
int modifierpos = 0; modifierpos < 8; modifierpos++)
1090 int offset = xfc->modifierMap->max_keypermod * modifierpos;
1092 for (
int key = 0; key < xfc->modifierMap->max_keypermod; key++)
1094 if (xfc->modifierMap->modifiermap[offset + key] == keycode)
1096 keysymMask |= 1 << modifierpos;
1104static BOOL xf_keyboard_get_key_state(xfContext* xfc,
int state, KeySym keysym)
1106 int keysymMask = xf_keyboard_get_keymask(xfc, keysym);
1111 return (state & keysymMask) != 0;
1114static BOOL xf_keyboard_set_key_state(xfContext* xfc, BOOL on, KeySym keysym)
1116 if (!xfc->xkbAvailable)
1119 const int keysymMask = xf_keyboard_get_keymask(xfc, keysym);
1126 return XkbLockModifiers(xfc->display, XkbUseCoreKbd,
1127 WINPR_ASSERTING_INT_CAST(uint32_t, keysymMask),
1128 on ? WINPR_ASSERTING_INT_CAST(uint32_t, keysymMask) : 0);
1131UINT32 xf_keyboard_get_toggle_keys_state(xfContext* xfc)
1133 UINT32 toggleKeysState = 0;
1134 const int state = xf_keyboard_read_keyboard_state(xfc);
1136 if (xf_keyboard_get_key_state(xfc, state, XK_Scroll_Lock))
1137 toggleKeysState |= KBD_SYNC_SCROLL_LOCK;
1139 if (xf_keyboard_get_key_state(xfc, state, XK_Num_Lock))
1140 toggleKeysState |= KBD_SYNC_NUM_LOCK;
1142 if (xf_keyboard_get_key_state(xfc, state, XK_Caps_Lock))
1143 toggleKeysState |= KBD_SYNC_CAPS_LOCK;
1145 if (xf_keyboard_get_key_state(xfc, state, XK_Kana_Lock))
1146 toggleKeysState |= KBD_SYNC_KANA_LOCK;
1148 return toggleKeysState;
1151static void xk_keyboard_update_modifier_keys(xfContext* xfc)
1153 const KeySym keysyms[] = { XK_Shift_L, XK_Shift_R, XK_Alt_L, XK_Alt_R,
1154 XK_Control_L, XK_Control_R, XK_Super_L, XK_Super_R };
1156 xf_keyboard_clear(xfc);
1158 const int state = xf_keyboard_read_keyboard_state(xfc);
1160 for (
size_t i = 0; i < ARRAYSIZE(keysyms); i++)
1162 if (xf_keyboard_get_key_state(xfc, state, keysyms[i]))
1164 const KeyCode keycode = XKeysymToKeycode(xfc->display, keysyms[i]);
1165 WINPR_ASSERT(keycode < ARRAYSIZE(xfc->KeyboardState));
1166 xfc->KeyboardState[keycode] = TRUE;
1171void xf_keyboard_focus_in(xfContext* xfc)
1180 if (!xfc->display || !xfc->window)
1183 rdpInput* input = xfc->common.context.input;
1184 WINPR_ASSERT(input);
1186 const UINT32 syncFlags = xf_keyboard_get_toggle_keys_state(xfc);
1187 freerdp_input_send_focus_in_event(input, WINPR_ASSERTING_INT_CAST(UINT16, syncFlags));
1188 xk_keyboard_update_modifier_keys(xfc);
1192 if (xfc->remote_app || !xfc->window)
1195 if (XQueryPointer(xfc->display, xfc->window->handle, &w, &w, &d, &d, &x, &y, &state))
1197 if ((x >= 0) && (x < xfc->window->width) && (y >= 0) && (y < xfc->window->height))
1199 xf_event_adjust_coordinates(xfc, &x, &y);
1200 freerdp_client_send_button_event(&xfc->common, FALSE, PTR_FLAGS_MOVE, x, y);
1205static BOOL action_script_run(xfContext* xfc,
const char* buffer,
size_t size,
void* user,
1206 const char* what,
const char* arg)
1212 int* pstatus = user;
1216 WLog_WARN(TAG,
"ActionScript key: script did not return data");
1220 if (strcmp(buffer,
"key-local") == 0)
1222 else if (winpr_PathFileExists(buffer))
1224 FILE* fp = popen(buffer,
"w");
1227 WLog_ERR(TAG,
"Failed to execute '%s'", buffer);
1231 *pstatus = pclose(fp);
1234 WLog_ERR(TAG,
"Command '%s' returned %d", buffer, *pstatus);
1240 WLog_WARN(TAG,
"ActionScript key: no such file '%s'", buffer);
1246static int xf_keyboard_execute_action_script(xfContext* xfc, XF_MODIFIER_KEYS* mod, KeySym keysym)
1250 char command[2048] = WINPR_C_ARRAY_INIT;
1251 char combination[1024] = WINPR_C_ARRAY_INIT;
1253 if (!xfc->actionScriptExists)
1256 if ((keysym == XK_Shift_L) || (keysym == XK_Shift_R) || (keysym == XK_Alt_L) ||
1257 (keysym == XK_Alt_R) || (keysym == XK_Control_L) || (keysym == XK_Control_R))
1262 const char* keyStr = XKeysymToString(keysym);
1264 if (keyStr ==
nullptr)
1270 winpr_str_append(
"Shift", combination,
sizeof(combination),
"+");
1273 winpr_str_append(
"Ctrl", combination,
sizeof(combination),
"+");
1276 winpr_str_append(
"Alt", combination,
sizeof(combination),
"+");
1279 winpr_str_append(
"Super", combination,
sizeof(combination),
"+");
1281 winpr_str_append(keyStr, combination,
sizeof(combination),
"+");
1283 for (
size_t i = 0; i < strnlen(combination,
sizeof(combination)); i++)
1284 combination[i] = WINPR_ASSERTING_INT_CAST(
char, tolower(combination[i]));
1286 const size_t count = ArrayList_Count(xfc->keyCombinations);
1288 for (
size_t index = 0; index < count; index++)
1290 const char* keyCombination = (
const char*)ArrayList_GetItem(xfc->keyCombinations, index);
1292 if (_stricmp(keyCombination, combination) == 0)
1302 (void)sprintf_s(command,
sizeof(command),
"key %s", combination);
1303 if (!run_action_script(xfc, command,
nullptr, action_script_run, &status))
1309static int xk_keyboard_get_modifier_keys(xfContext* xfc, XF_MODIFIER_KEYS* mod)
1311 mod->LeftShift = xf_keyboard_key_pressed(xfc, XK_Shift_L);
1312 mod->RightShift = xf_keyboard_key_pressed(xfc, XK_Shift_R);
1313 mod->Shift = mod->LeftShift || mod->RightShift;
1314 mod->LeftAlt = xf_keyboard_key_pressed(xfc, XK_Alt_L);
1315 mod->RightAlt = xf_keyboard_key_pressed(xfc, XK_Alt_R);
1316 mod->Alt = mod->LeftAlt || mod->RightAlt;
1317 mod->LeftCtrl = xf_keyboard_key_pressed(xfc, XK_Control_L);
1318 mod->RightCtrl = xf_keyboard_key_pressed(xfc, XK_Control_R);
1319 mod->Ctrl = mod->LeftCtrl || mod->RightCtrl;
1320 mod->LeftSuper = xf_keyboard_key_pressed(xfc, XK_Super_L);
1321 mod->RightSuper = xf_keyboard_key_pressed(xfc, XK_Super_R);
1322 mod->Super = mod->LeftSuper || mod->RightSuper;
1326BOOL xf_keyboard_handle_special_keys(xfContext* xfc, KeySym keysym)
1328 XF_MODIFIER_KEYS mod = WINPR_C_ARRAY_INIT;
1329 xk_keyboard_get_modifier_keys(xfc, &mod);
1333 if (keysym == XK_Control_R)
1335 if (mod.RightCtrl && !xfc->wasRightCtrlAlreadyPressed)
1338 xfc->ungrabKeyboardWithRightCtrl = TRUE;
1339 xfc->wasRightCtrlAlreadyPressed = TRUE;
1345 if (xfc->ungrabKeyboardWithRightCtrl)
1346 xfc->ungrabKeyboardWithRightCtrl = FALSE;
1349 const int rc = xf_keyboard_execute_action_script(xfc, &mod, keysym);
1355 if (!xfc->remote_app && xfc->fullscreen_toggle)
1360 if (mod.Ctrl && mod.Alt)
1363 WLog_INFO(TAG,
"<ctrl>+<alt>+<enter> pressed, toggling fullscreen state...");
1364 xf_sync_kbd_state(xfc);
1365 xf_toggle_fullscreen(xfc);
1374 if (mod.Ctrl && mod.Alt)
1380 WLog_INFO(TAG,
"<ctrl>+<alt>+m pressed, minimizing RDP session...");
1381 xf_sync_kbd_state(xfc);
1387 WLog_INFO(TAG,
"<ctrl>+<alt>+c pressed, toggle encomps control...");
1388 if (freerdp_client_encomsp_toggle_control(xfc->common.encomsp))
1394 WLog_INFO(TAG,
"<ctrl>+<alt>+d pressed, terminating RDP session...");
1395 xf_sync_kbd_state(xfc);
1396 return freerdp_abort_connect_context(&xfc->common.context);
1408 rdpContext* ctx = &xfc->common.context;
1410 if (mod.Ctrl && mod.Alt)
1423 xfc->scaledWidth = sessionWidth;
1424 xfc->scaledHeight = sessionHeight;
1428 if (!xfc->fullscreen && (sessionWidth != xfc->window->width ||
1429 sessionHeight != xfc->window->height))
1431 xf_ResizeDesktopWindow(xfc, xfc->window, sessionWidth, sessionHeight);
1434 xf_draw_screen(xfc, 0, 0, sessionWidth, sessionHeight);
1463 if (pdx != 0 || pdy != 0)
1465 PanningChangeEventArgs e;
1466 EventArgsInit(&e,
"xfreerdp");
1469 PubSub_OnPanningChange(ctx->pubSub, xfc, &e);
1473 if (zdx != 0 || zdy != 0)
1475 ZoomingChangeEventArgs e;
1476 EventArgsInit(&e,
"xfreerdp");
1479 PubSub_OnZoomingChange(ctx->pubSub, xfc, &e);
1490void xf_keyboard_handle_special_keys_release(xfContext* xfc, KeySym keysym)
1492 if (keysym != XK_Control_R)
1495 xfc->wasRightCtrlAlreadyPressed = FALSE;
1497 if (!xfc->ungrabKeyboardWithRightCtrl)
1501 XF_MODIFIER_KEYS mod = WINPR_C_ARRAY_INIT;
1502 xk_keyboard_get_modifier_keys(xfc, &mod);
1506 if (!xfc->fullscreen)
1508 xf_sync_kbd_state(xfc);
1509 freerdp_client_encomsp_toggle_control(xfc->common.encomsp);
1512 xfc->mouse_active = FALSE;
1517 xfc->ungrabKeyboardWithRightCtrl = FALSE;
1520BOOL xf_keyboard_set_indicators(rdpContext* context, UINT16 led_flags)
1522 xfContext* xfc = (xfContext*)context;
1523 xf_keyboard_set_key_state(xfc, led_flags & KBD_SYNC_SCROLL_LOCK, XK_Scroll_Lock);
1524 xf_keyboard_set_key_state(xfc, led_flags & KBD_SYNC_NUM_LOCK, XK_Num_Lock);
1525 xf_keyboard_set_key_state(xfc, led_flags & KBD_SYNC_CAPS_LOCK, XK_Caps_Lock);
1526 xf_keyboard_set_key_state(xfc, led_flags & KBD_SYNC_KANA_LOCK, XK_Kana_Lock);
1530BOOL xf_keyboard_set_ime_status(rdpContext* context, UINT16 imeId, UINT32 imeState,
1537 "KeyboardSetImeStatus(unitId=%04" PRIx16
", imeState=%08" PRIx32
1538 ", imeConvMode=%08" PRIx32
") ignored",
1539 imeId, imeState, imeConvMode);
1543BOOL xf_ungrab(xfContext* xfc)
1546 XUngrabKeyboard(xfc->display, CurrentTime);
1547 XUngrabPointer(xfc->display, CurrentTime);
1548 xfc->common.mouse_grabbed = FALSE;
FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
This struct contains function pointer to initialize/free objects.
OBJECT_FREE_FN fnObjectFree
OBJECT_NEW_FN fnObjectNew