0e7cc1610749c4c0b510816d199e7f93a0562983
[reactos.git] / reactos / dll / keyboard / kbdgr / kbdgr.c
1 /*
2 * ReactOS German ASCII Keyboard layout
3 * Copyright (C) 2003 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 #include <windows.h>
11 #include <ndk/kbd.h>
12
13 #ifdef _M_IA64
14 #define ROSDATA static __declspec(allocate(".data"))
15 #else
16 #ifdef _MSC_VER
17 #pragma data_seg(".data")
18 #define ROSDATA static
19 #else
20 #define ROSDATA static __attribute__((section(".data")))
21 #endif
22 #endif
23
24 #define VK_EMPTY 0xff /* The non-existent VK */
25
26 #define KNUMS KBDNUMPAD|KBDSPECIAL /* Special + number pad */
27 #define KMEXT KBDEXT|KBDMULTIVK /* Multi + ext */
28
29 ROSDATA USHORT scancode_to_vk[] = {
30 /* Numbers Row */
31 /* - 00 - */
32 /* 1 ... 2 ... 3 ... 4 ... */
33 VK_EMPTY, VK_ESCAPE, '1', '2',
34 '3', '4', '5', '6',
35 '7', '8', '9', '0',
36 VK_OEM_4, VK_OEM_6, VK_BACK,
37 /* - 0f - */
38 /* First Letters Row */
39 VK_TAB, 'Q', 'W', 'E',
40 'R', 'T', 'Z', 'U',
41 'I', 'O', 'P',
42 VK_OEM_1, VK_OEM_PLUS, VK_RETURN,
43 /* - 1d - */
44 /* Second Letters Row */
45 VK_LCONTROL,
46 'A', 'S', 'D', 'F',
47 'G', 'H', 'J', 'K',
48 'L', VK_OEM_3, VK_OEM_7, VK_OEM_5,
49 VK_LSHIFT, VK_OEM_2,
50 /* - 2c - */
51 /* Third letters row */
52 'Y', 'X', 'C', 'V',
53 'B', 'N', 'M', VK_OEM_COMMA,
54 VK_OEM_PERIOD,VK_OEM_MINUS, VK_RSHIFT,
55 /* - 37 - */
56 /* Bottom Row */
57 VK_MULTIPLY, VK_LMENU, VK_SPACE, VK_CAPITAL,
58
59 /* - 3b - */
60 /* F-Keys */
61 VK_F1, VK_F2, VK_F3, VK_F4, VK_F5, VK_F6,
62 VK_F7, VK_F8, VK_F9, VK_F10,
63 /* - 45 - */
64 /* Locks */
65 VK_NUMLOCK | KMEXT,
66 VK_SCROLL | KBDMULTIVK,
67 /* - 47 - */
68 /* Number-Pad */
69 VK_HOME | KNUMS, VK_UP | KNUMS, VK_PRIOR | KNUMS, VK_SUBTRACT,
70 VK_LEFT | KNUMS, VK_CLEAR | KNUMS, VK_RIGHT | KNUMS, VK_ADD,
71 VK_END | KNUMS, VK_DOWN | KNUMS, VK_NEXT | KNUMS,
72 VK_INSERT | KNUMS, VK_DELETE | KNUMS,
73 /* - 54 - */
74 /* Presumably PrtSc */
75 VK_SNAPSHOT,
76 /* - 55 - */
77 /* Oddities, and the remaining standard F-Keys */
78 VK_EMPTY, VK_OEM_102, VK_F11, VK_F12,
79 /* - 59 - */
80 VK_CLEAR, VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* EREOF */
81 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* ZOOM */
82 VK_HELP,
83 /* - 64 - */
84 /* Even more F-Keys (for example, NCR keyboards from the early 90's) */
85 VK_F13, VK_F14, VK_F15, VK_F16, VK_F17, VK_F18, VK_F19, VK_F20,
86 VK_F21, VK_F22, VK_F23,
87 /* - 6f - */
88 /* Not sure who uses these codes */
89 VK_EMPTY, VK_EMPTY, VK_EMPTY,
90 /* - 72 - */
91 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
92 /* - 76 - */
93 /* One more f-key */
94 VK_F24,
95 /* - 77 - */
96 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY,
97 VK_EMPTY, VK_EMPTY, VK_EMPTY, VK_EMPTY, /* PA1 */
98 VK_EMPTY,
99 /* - 80 - */
100 0
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 sizeof(scancode_to_vk) / sizeof(scancode_to_vk[0]),
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