xref: /reactos/dll/keyboard/kbdgneo/kbdgneo.c (revision 4572aad1)
1 /*
2  * ReactOS German NEO ASCII Keyboard layout
3  * Copyright (C) 2008 ReactOS
4  * License: LGPL, see: LGPL.txt
5  *
6  * Thanks to: http://www.barcodeman.com/altek/mule/scandoc.php
7  * and http://win.tue.nl/~aeb/linux/kbd/scancodes-1.html
8  */
9 
10 #define WIN32_NO_STATUS
11 #include <stdarg.h>
12 #include <windef.h>
13 #include <winuser.h>
14 #include <ndk/kbd.h>
15 
16 #ifdef _M_IA64
17 #define ROSDATA static __declspec(allocate(".data"))
18 #else
19 #ifdef _MSC_VER
20 #pragma data_seg(".data")
21 #define ROSDATA static
22 #else
23 #define ROSDATA static __attribute__((section(".data")))
24 #endif
25 #endif
26 
27 #define VK_EMPTY  0xff   /* The non-existent VK */
28 
29 #define KNUMS     KBDNUMPAD|KBDSPECIAL /* Special + number pad */
30 #define KMEXT     KBDEXT|KBDMULTIVK    /* Multi + ext */
31 
32 ROSDATA USHORT scancode_to_vk[] = {
33   /* Numbers Row */
34   /* - 00 - */
35   /* 1 ...         2 ...         3 ...         4 ... */
36   VK_EMPTY,     VK_ESCAPE,    '1',          '2',
37   '3',          '4',          '5',          '6',
38   '7',          '8',          '9',          '0',
39   VK_OEM_MINUS, VK_OEM_6,     VK_BACK,
40   /* - 0f - */
41   /* First Letters Row */
42   VK_TAB,       'Q',          'V',          'L',
43   'C',          'W',          'K',          'H',
44   'G',          'F',          'J',
45   VK_OEM_4,     VK_OEM_PLUS,  VK_RETURN,
46   /* - 1d - */
47   /* Second Letters Row */
48   VK_LCONTROL,
49   'U',          'I',          'A',          'E',
50   'O',          'S',          'N',          'R',
51   'T',          'D',          'Y',          VK_OEM_5,
52   VK_LSHIFT,    VK_OEM_2,
53   /* - 2c - */
54   /* Third letters row */
55   VK_OEM_3,     VK_OEM_1,     VK_OEM_7,     'P',
56   'Z',          'B',          'M',          VK_OEM_COMMA,
57   VK_OEM_PERIOD,'X',          VK_RSHIFT,
58   /* - 37 - */
59   /* Bottom Row */
60   VK_MULTIPLY,  VK_LMENU,     VK_SPACE,     VK_CAPITAL,
61 
62   /* - 3b - */
63   /* F-Keys */
64   VK_F1, VK_F2, VK_F3, VK_F4, VK_F5, VK_F6,
65   VK_F7, VK_F8, VK_F9, VK_F10,
66   /* - 45 - */
67   /* Locks */
68   VK_NUMLOCK | KMEXT,
69   VK_SCROLL | KBDMULTIVK,
70   /* - 47 - */
71   /* Number-Pad */
72   VK_HOME | KNUMS,      VK_UP | KNUMS,         VK_PRIOR | KNUMS, VK_SUBTRACT,
73   VK_LEFT | KNUMS,      VK_CLEAR | KNUMS,      VK_RIGHT | KNUMS, VK_ADD,
74   VK_END | KNUMS,       VK_DOWN | KNUMS,       VK_NEXT | KNUMS,
75   VK_INSERT | KNUMS,    VK_DELETE | KNUMS,
76   /* - 54 - */
77   /* Presumably PrtSc */
78   VK_SNAPSHOT,
79   /* - 55 - */
80   /* Oddities, and the remaining standard F-Keys */
81   VK_EMPTY,   VK_OEM_102,     VK_F11,       VK_F12,
82   /* - 59 - */
83   VK_CLEAR,     VK_EMPTY,     VK_EMPTY,     VK_EMPTY,     VK_EMPTY, /* EREOF */
84   VK_EMPTY,     VK_EMPTY,     VK_EMPTY,     VK_EMPTY,     VK_EMPTY, /* ZOOM */
85   VK_HELP,
86   /* - 64 - */
87   /* Even more F-Keys (for example, NCR keyboards from the early 90's) */
88   VK_F13, VK_F14, VK_F15, VK_F16, VK_F17, VK_F18, VK_F19, VK_F20,
89   VK_F21, VK_F22, VK_F23,
90   /* - 6f - */
91   /* Not sure who uses these codes */
92   VK_EMPTY, VK_EMPTY, VK_EMPTY,
93   /* - 72 - */
94   VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
95   /* - 76 - */
96   /* One more f-key */
97   VK_F24,
98   /* - 77 - */
99   VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
100   VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* PA1 */
101 };
102 
103 ROSDATA VSC_VK extcode0_to_vk[] = {
104   { 0x10, VK_MEDIA_PREV_TRACK | KBDEXT },
105   { 0x19, VK_MEDIA_NEXT_TRACK | KBDEXT },
106   { 0x1D, VK_RCONTROL | KBDEXT },
107   { 0x20, VK_VOLUME_MUTE | KBDEXT },
108   { 0x21, VK_LAUNCH_APP2 | KBDEXT },
109   { 0x22, VK_MEDIA_PLAY_PAUSE | KBDEXT },
110   { 0x24, VK_MEDIA_STOP | KBDEXT },
111   { 0x2E, VK_VOLUME_DOWN | KBDEXT },
112   { 0x30, VK_VOLUME_UP | KBDEXT },
113   { 0x32, VK_BROWSER_HOME | KBDEXT },
114   { 0x35, VK_DIVIDE | KBDEXT },
115   { 0x37, VK_SNAPSHOT | KBDEXT },
116   { 0x38, VK_RMENU | KBDEXT },
117   { 0x47, VK_HOME | KBDEXT },
118   { 0x48, VK_UP | KBDEXT },
119   { 0x49, VK_PRIOR | KBDEXT },
120   { 0x4B, VK_LEFT | KBDEXT },
121   { 0x4D, VK_RIGHT | KBDEXT },
122   { 0x4F, VK_END | KBDEXT },
123   { 0x50, VK_DOWN | KBDEXT },
124   { 0x51, VK_NEXT | KBDEXT },
125   { 0x52, VK_INSERT | KBDEXT },
126   { 0x53, VK_DELETE | KBDEXT },
127   { 0x5B, VK_LWIN | KBDEXT },
128   { 0x5C, VK_RWIN | KBDEXT },
129   { 0x5D, VK_APPS | KBDEXT },
130   { 0x5F, VK_SLEEP | KBDEXT },
131   { 0x65, VK_BROWSER_SEARCH | KBDEXT },
132   { 0x66, VK_BROWSER_FAVORITES | KBDEXT },
133   { 0x67, VK_BROWSER_REFRESH | KBDEXT },
134   { 0x68, VK_BROWSER_STOP | KBDEXT },
135   { 0x69, VK_BROWSER_FORWARD | KBDEXT },
136   { 0x6A, VK_BROWSER_BACK | KBDEXT },
137   { 0x6B, VK_LAUNCH_APP1 | KBDEXT },
138   { 0x6C, VK_LAUNCH_MAIL | KBDEXT },
139   { 0x6D, VK_LAUNCH_MEDIA_SELECT | KBDEXT },
140   { 0x1C, VK_RETURN | KBDEXT },
141   { 0x46, VK_CANCEL | KBDEXT },
142   { 0, 0 },
143 };
144 
145 ROSDATA VSC_VK extcode1_to_vk[] = {
146   { 0x1d, VK_PAUSE},
147   { 0, 0 },
148 };
149 
150 ROSDATA VK_TO_BIT modifier_keys[] = {
151   { VK_SHIFT,   KBDSHIFT },
152   { VK_CONTROL, KBDCTRL },
153   { VK_MENU,    KBDALT },
154   { 0,          0 }
155 };
156 
157 ROSDATA MODIFIERS modifier_bits = {
158   modifier_keys,
159   6,
160   { 0, 1, 3, 4, SHFT_INVALID, SHFT_INVALID, 2 } /* Modifier bit order, NONE, SHIFT, CTRL, ALT, MENU, SHIFT + MENU, CTRL + MENU */
161 };
162 
163 ROSDATA VK_TO_WCHARS2 key_to_chars_2mod[] = {
164   { VK_OEM_5,    0,      {WCH_DEAD, 0xb0} },
165   { 0xff,        0,      {'^', WCH_NONE} }, // FIXME - why doesn't this work?
166   /* Normal vs Shifted */
167   /* The numbers */
168   { '1',         0, {'1', '!'} },
169   /* Ctrl-2 generates NUL */
170   { '4',         0, {'4', '$'} },
171   { '5',         0, {'5', '%'} },
172 
173   { VK_OEM_6,    0, {WCH_DEAD, WCH_DEAD} },
174   { 0xff,        0, {0xb4, '`'} },
175 
176   /* First letter row */
177   { 'W',         CAPLOK,   {'w', 'W'} },
178   { 'R',         CAPLOK,   {'r', 'R'} },
179   { 'T',         CAPLOK,   {'t', 'T'} },
180   { 'Z',         CAPLOK,   {'z', 'Z'} },
181   { 'U',         CAPLOK,   {'u', 'U'} },
182   { 'I',         CAPLOK,   {'i', 'I'} },
183   { 'O',         CAPLOK,   {'o', 'O'} },
184   { 'P',         CAPLOK,   {'p', 'P'} },
185   /* Second letter row */
186   { 'A',         CAPLOK,   {'a', 'A'} },
187   { 'S',         CAPLOK,   {'s', 'S'} },
188   { 'D',         CAPLOK,   {'d', 'D'} },
189   { 'F',         CAPLOK,   {'f', 'F'} },
190   { 'G',         CAPLOK,   {'g', 'G'} },
191   { 'H',         CAPLOK,   {'h', 'H'} },
192   { 'J',         CAPLOK,   {'j', 'J'} },
193   { 'K',         CAPLOK,   {'k', 'K'} },
194   { 'L',         CAPLOK,   {'l', 'L'} },
195   { VK_OEM_3,    CAPLOK,   {0xf6, 0xd6} },
196   { VK_OEM_7,    CAPLOK,   {0xe4, 0xc4} },
197   /* Third letter row */
198   { 'Y',         CAPLOK,   {'y', 'Y'} },
199   { 'X',         CAPLOK,   {'x', 'X'} },
200   { 'C',         CAPLOK,   {'c', 'C'} },
201   { 'V',         CAPLOK,   {'v', 'V'} },
202   { 'B',         CAPLOK,   {'b', 'B'} },
203   { 'N',         CAPLOK,   {'n', 'N'} },
204 
205   /* Specials */
206   { VK_OEM_COMMA,  CAPLOK, {',', ';'} },
207   { VK_OEM_PERIOD, CAPLOK, {'.', ':'} },
208   { VK_DECIMAL,    0, {',',','} },
209   { VK_TAB,        0, {'\t', '\t'} },
210   { VK_ADD,        0, {'+', '+'} },
211   { VK_DIVIDE,     0, {'/', '/'} },
212   { VK_MULTIPLY,   0, {'*', '*'} },
213   { VK_SUBTRACT,   0, {'-', '-'} },
214   { 0, 0 }
215 };
216 
217 ROSDATA VK_TO_WCHARS3 key_to_chars_3mod[] = {
218   /* Normal, Shifted, Ctrl */
219   /* Legacy (telnet-style) ascii escapes */
220   { '3', CAPLOK, {'3', 0xa7, 0xb3} },
221   { '7', CAPLOK, {'7', '/', '{'} },
222   { '8', CAPLOK, {'8', '(', '['} },
223   { '9', CAPLOK, {'9', ')', ']'} },
224   { '0', CAPLOK, {'0', '=', '}'} },
225   { VK_OEM_4, CAPLOK, {0xdf, '?', '\\'} },
226   { 'Q', CAPLOK, {'q', 'Q', '@'} },
227   { 'E', CAPLOK, {'e', 'E', 0x20ac} },
228   { 'M', CAPLOK, {'m', 'M', 0xb5} },
229   { VK_OEM_102, 0, {'<', '>', '|'} }, // FIXME - why doesn't this work?
230   { 0,0 }
231 };
232 
233 ROSDATA VK_TO_WCHARS4 key_to_chars_4mod[] = {
234   /* Normal, Shifted, Ctrl, C-S-x */
235   /* Legacy Ascii generators */
236   { VK_OEM_1, CAPLOK, {0xfc, 0xdc, WCH_NONE, 0x1b} },
237   { VK_OEM_PLUS, CAPLOK, {'+', '*', '~', 0x1d} },
238   { VK_OEM_2, CAPLOK, {'#', '\'', WCH_NONE, 0x1c} },
239   { VK_BACK, 0, {'\b', '\b', WCH_NONE, 0x7f} },
240   { VK_ESCAPE, 0, {0x1b, 0x1b, WCH_NONE, 0x1b} },
241   { VK_RETURN, 0, {'\r', '\r', WCH_NONE, '\n'} },
242   { VK_SPACE, 0, {' ', ' ', WCH_NONE, ' '} },
243   { VK_CANCEL, 0, {0x03, 0x03, WCH_NONE, 0x03} },
244   { 0, 0 }
245 };
246 
247 ROSDATA VK_TO_WCHARS5 key_to_chars_5mod[] = {
248   /* Normal, Shifted, Ctrl, C-S-x */
249   { '2', CAPLOK, {'2', '\"', 0xb2, WCH_NONE, 0x00} },
250   { '6', CAPLOK, {'6', '&', WCH_NONE, WCH_NONE, 0x1e} },
251   { VK_OEM_MINUS, 0, {'-', '_', WCH_NONE, WCH_NONE, 0x1f} },
252   { 0, 0 }
253 };
254 
255 ROSDATA VK_TO_WCHARS1 keypad_numbers[] = {
256   { VK_DECIMAL, 0, {'.'} },
257   { VK_NUMPAD0, 0, {'0'} },
258   { VK_NUMPAD1, 0, {'1'} },
259   { VK_NUMPAD2, 0, {'2'} },
260   { VK_NUMPAD3, 0, {'3'} },
261   { VK_NUMPAD4, 0, {'4'} },
262   { VK_NUMPAD5, 0, {'5'} },
263   { VK_NUMPAD6, 0, {'6'} },
264   { VK_NUMPAD7, 0, {'7'} },
265   { VK_NUMPAD8, 0, {'8'} },
266   { VK_NUMPAD9, 0, {'9'} },
267 //  { VK_BACK,    0, '\010' },
268   { 0,0 }
269 };
270 
271 #define vk_master(n,x) { (PVK_TO_WCHARS1)x, n, sizeof(x[0]) }
272 
273 ROSDATA VK_TO_WCHAR_TABLE vk_to_wchar_master_table[] = {
274   vk_master(3,key_to_chars_3mod),
275   vk_master(4,key_to_chars_4mod),
276   vk_master(5,key_to_chars_5mod),
277   vk_master(2,key_to_chars_2mod),
278   vk_master(1,keypad_numbers),
279   { 0,0,0 }
280 };
281 
282 ROSDATA VSC_LPWSTR key_names[] = {
283   { 0x00, L"" },
284   { 0x01, L"Esc" },
285   { 0x0e, L"R\x00fc" L"ck" },
286   { 0x0f, L"Tabulator" },
287   { 0x1c, L"Eingabe" },
288   { 0x1d, L"Strg" },
289   { 0x2a, L"Umschalt" },
290   { 0x36, L"Umschalt Rechts" },
291   { 0x37, L" (Zehnertastatur)" },
292   { 0x38, L"Alt" },
293   { 0x39, L"Leer" },
294   { 0x3a, L"Feststell" },
295   { 0x3b, L"F1" },
296   { 0x3c, L"F2" },
297   { 0x3d, L"F3" },
298   { 0x3e, L"F4" },
299   { 0x3f, L"F5" },
300   { 0x40, L"F6" },
301   { 0x41, L"F7" },
302   { 0x42, L"F8" },
303   { 0x43, L"F9" },
304   { 0x44, L"F10" },
305   { 0x45, L"Pause" },
306   { 0x46, L"Rollen-Feststell" },
307   { 0x47, L"7 (Zehnertastatur)" },
308   { 0x48, L"8 (Zehnertastatur)" },
309   { 0x49, L"9 (Zehnertastatur)" },
310   { 0x4a, L"- (Zehnertastatur)" },
311   { 0x4b, L"4 (Zehnertastatur)" },
312   { 0x4c, L"5 (Zehnertastatur)" },
313   { 0x4d, L"6 (Zehnertastatur)" },
314   { 0x4e, L"+ (Zehnertastatur)" },
315   { 0x4f, L"1 (Zehnertastatur)" },
316   { 0x50, L"2 (Zehnertastatur)" },
317   { 0x51, L"3 (Zehnertastatur)" },
318   { 0x52, L"0 (Zehnertastatur)" },
319   { 0x53, L"Komma (Zehnertastatur)" },
320   { 0x54, L"Sys Req" },
321   { 0x57, L"F11" },
322   { 0x58, L"F12" },
323   { 0x7c, L"F13" },
324   { 0x7d, L"F14" },
325   { 0x7e, L"F15" },
326   { 0x7f, L"F16" },
327   { 0x80, L"F17" },
328   { 0x81, L"F18" },
329   { 0x82, L"F19" },
330   { 0x83, L"F20" },
331   { 0x84, L"F21" },
332   { 0x85, L"F22" },
333   { 0x86, L"F23" },
334   { 0x87, L"F24" },
335   { 0, NULL },
336 };
337 
338 ROSDATA VSC_LPWSTR extended_key_names[] = {
339   { 0x1c, L"Eingabe (Zehnertastatur" },
340   { 0x1d, L"Strg-Rechts" },
341   { 0x35, L" (Zehnertastatur)" },
342   { 0x37, L"Druck" },
343   { 0x38, L"Alt Gr" },
344   { 0x45, L"Num-Feststell" },
345   { 0x46, L"Untbr" },
346   { 0x47, L"Pos1" },
347   { 0x48, L"Nach-Oben" },
348   { 0x49, L"Bild-Nach-Oben" },
349   { 0x4b, L"Nach-Links" },
350 //{ 0x4c, L"Center" },
351   { 0x4d, L"Nach-Rechts" },
352   { 0x4f, L"Ende" },
353   { 0x50, L"Nach-Unten" },
354   { 0x51, L"Bild-Nach-Unten" },
355   { 0x52, L"Einfg" },
356   { 0x53, L"Entf" },
357   { 0x54, L"<ReactOS>" },
358   { 0x55, L"Hilfe" },
359   { 0x56, L"Linke <ReactOS>" },
360   { 0x5b, L"Rechte <ReactOS>" },
361   { 0, NULL },
362 };
363 
364 ROSDATA DEADKEY_LPWSTR dead_key_names[] = {
365     L"\x00b4"	L"Akut",
366     L"`"	L"Gravis",
367     L"^"	L"Zirkumflex",
368     NULL
369 };
370 
371 #define DEADTRANS(ch, accent, comp, flags) MAKELONG(ch, accent), comp, flags
372 
373 ROSDATA DEADKEY dead_key[] = {
374   { DEADTRANS(L'a', L'^', 0xe2, 0x00) },
375   { DEADTRANS(L'e', L'^', 0xea, 0x00) },
376   { DEADTRANS(L'i', L'^', 0xee, 0x00) },
377   { DEADTRANS(L'o', L'^', 0xf4, 0x00) },
378   { DEADTRANS(L'u', L'^', 0xfb, 0x00) },
379   { DEADTRANS(L'A', L'^', 0xc2, 0x00) },
380   { DEADTRANS(L'E', L'^', 0xca, 0x00) },
381   { DEADTRANS(L'I', L'^', 0xce, 0x00) },
382   { DEADTRANS(L'O', L'^', 0xd4, 0x00) },
383   { DEADTRANS(L'U', L'^', 0xdb, 0x00) },
384   { DEADTRANS(L' ', L'^', L'^', 0x00) },
385   { DEADTRANS(L'a', 0xb4, 0xe1, 0x00) },
386   { DEADTRANS(L'e', 0xb4, 0xe9, 0x00) },
387   { DEADTRANS(L'i', 0xb4, 0xed, 0x00) },
388   { DEADTRANS(L'o', 0xb4, 0xf3, 0x00) },
389   { DEADTRANS(L'u', 0xb4, 0xfa, 0x00) },
390   { DEADTRANS(L'y', 0xb4, 0xfd, 0x00) },
391   { DEADTRANS(L'A', 0xb4, 0xc1, 0x00) },
392   { DEADTRANS(L'E', 0xb4, 0xc9, 0x00) },
393   { DEADTRANS(L'I', 0xb4, 0xcd, 0x00) },
394   { DEADTRANS(L'O', 0xb4, 0xd3, 0x00) },
395   { DEADTRANS(L'U', 0xb4, 0xda, 0x00) },
396   { DEADTRANS(L'Y', 0xb4, 0xdd, 0x00) },
397   { DEADTRANS(L' ', 0xb4, 0xb4, 0x00) },
398   { DEADTRANS(L'a', L'`', 0xe0, 0x00) },
399   { DEADTRANS(L'e', L'`', 0xe8, 0x00) },
400   { DEADTRANS(L'i', L'`', 0xec, 0x00) },
401   { DEADTRANS(L'o', L'`', 0xf2, 0x00) },
402   { DEADTRANS(L'u', L'`', 0xf9, 0x00) },
403   { DEADTRANS(L'A', L'`', 0xc0, 0x00) },
404   { DEADTRANS(L'E', L'`', 0xc8, 0x00) },
405   { DEADTRANS(L'I', L'`', 0xcc, 0x00) },
406   { DEADTRANS(L'O', L'`', 0xd2, 0x00) },
407   { DEADTRANS(L'U', L'`', 0xd9, 0x00) },
408   { DEADTRANS(L' ', L'`', L'`', 0x00) },
409   { 0, 0 }
410 };
411 
412 /* Finally, the master table */
413 ROSDATA KBDTABLES keyboard_layout_table = {
414   /* modifier assignments */
415   &modifier_bits,
416 
417   /* character from vk tables */
418   vk_to_wchar_master_table,
419 
420   /* diacritical marks -- currently implemented by wine code */
421   dead_key,
422 
423   /* Key names */
424   (VSC_LPWSTR *)key_names,
425   (VSC_LPWSTR *)extended_key_names,
426   dead_key_names, /* Dead key names */
427 
428   /* scan code to virtual key maps */
429   scancode_to_vk,
430   RTL_NUMBER_OF(scancode_to_vk),
431   extcode0_to_vk,
432   extcode1_to_vk,
433 
434   MAKELONG(KLLF_ALTGR, 1), /* Version 1.0 */
435 
436   /* Ligatures -- German doesn't have any */
437   0,
438   0,
439   NULL
440 };
441 
442 PKBDTABLES WINAPI KbdLayerDescriptor(VOID) {
443   return &keyboard_layout_table;
444 }
445 
446