1 | //======================================================================== |
2 | // GLFW 3.2 X11 - www.glfw.org |
3 | //------------------------------------------------------------------------ |
4 | // Copyright (c) 2002-2006 Marcus Geelnard |
5 | // Copyright (c) 2006-2016 Camilla Berglund <elmindreda@glfw.org> |
6 | // |
7 | // This software is provided 'as-is', without any express or implied |
8 | // warranty. In no event will the authors be held liable for any damages |
9 | // arising from the use of this software. |
10 | // |
11 | // Permission is granted to anyone to use this software for any purpose, |
12 | // including commercial applications, and to alter it and redistribute it |
13 | // freely, subject to the following restrictions: |
14 | // |
15 | // 1. The origin of this software must not be misrepresented; you must not |
16 | // claim that you wrote the original software. If you use this software |
17 | // in a product, an acknowledgment in the product documentation would |
18 | // be appreciated but is not required. |
19 | // |
20 | // 2. Altered source versions must be plainly marked as such, and must not |
21 | // be misrepresented as being the original software. |
22 | // |
23 | // 3. This notice may not be removed or altered from any source |
24 | // distribution. |
25 | // |
26 | //======================================================================== |
27 | |
28 | #include "internal.h" |
29 | |
30 | #include <X11/Xresource.h> |
31 | |
32 | #include <stdlib.h> |
33 | #include <string.h> |
34 | #include <limits.h> |
35 | #include <stdio.h> |
36 | #include <locale.h> |
37 | |
38 | |
39 | // Translate an X11 key code to a GLFW key code. |
40 | // |
41 | static int translateKeyCode(int scancode) |
42 | { |
43 | int keySym; |
44 | |
45 | // Valid key code range is [8,255], according to the Xlib manual |
46 | if (scancode < 8 || scancode > 255) |
47 | return GLFW_KEY_UNKNOWN; |
48 | |
49 | if (_glfw.x11.xkb.available) |
50 | { |
51 | // Try secondary keysym, for numeric keypad keys |
52 | // Note: This way we always force "NumLock = ON", which is intentional |
53 | // since the returned key code should correspond to a physical |
54 | // location. |
55 | keySym = XkbKeycodeToKeysym(_glfw.x11.display, scancode, 0, 1); |
56 | switch (keySym) |
57 | { |
58 | case XK_KP_0: return GLFW_KEY_KP_0; |
59 | case XK_KP_1: return GLFW_KEY_KP_1; |
60 | case XK_KP_2: return GLFW_KEY_KP_2; |
61 | case XK_KP_3: return GLFW_KEY_KP_3; |
62 | case XK_KP_4: return GLFW_KEY_KP_4; |
63 | case XK_KP_5: return GLFW_KEY_KP_5; |
64 | case XK_KP_6: return GLFW_KEY_KP_6; |
65 | case XK_KP_7: return GLFW_KEY_KP_7; |
66 | case XK_KP_8: return GLFW_KEY_KP_8; |
67 | case XK_KP_9: return GLFW_KEY_KP_9; |
68 | case XK_KP_Separator: |
69 | case XK_KP_Decimal: return GLFW_KEY_KP_DECIMAL; |
70 | case XK_KP_Equal: return GLFW_KEY_KP_EQUAL; |
71 | case XK_KP_Enter: return GLFW_KEY_KP_ENTER; |
72 | default: break; |
73 | } |
74 | |
75 | // Now try primary keysym for function keys (non-printable keys). These |
76 | // should not be layout dependent (i.e. US layout and international |
77 | // layouts should give the same result). |
78 | keySym = XkbKeycodeToKeysym(_glfw.x11.display, scancode, 0, 0); |
79 | } |
80 | else |
81 | { |
82 | int dummy; |
83 | KeySym* keySyms; |
84 | |
85 | keySyms = XGetKeyboardMapping(_glfw.x11.display, scancode, 1, &dummy); |
86 | keySym = keySyms[0]; |
87 | XFree(keySyms); |
88 | } |
89 | |
90 | switch (keySym) |
91 | { |
92 | case XK_Escape: return GLFW_KEY_ESCAPE; |
93 | case XK_Tab: return GLFW_KEY_TAB; |
94 | case XK_Shift_L: return GLFW_KEY_LEFT_SHIFT; |
95 | case XK_Shift_R: return GLFW_KEY_RIGHT_SHIFT; |
96 | case XK_Control_L: return GLFW_KEY_LEFT_CONTROL; |
97 | case XK_Control_R: return GLFW_KEY_RIGHT_CONTROL; |
98 | case XK_Meta_L: |
99 | case XK_Alt_L: return GLFW_KEY_LEFT_ALT; |
100 | case XK_Mode_switch: // Mapped to Alt_R on many keyboards |
101 | case XK_ISO_Level3_Shift: // AltGr on at least some machines |
102 | case XK_Meta_R: |
103 | case XK_Alt_R: return GLFW_KEY_RIGHT_ALT; |
104 | case XK_Super_L: return GLFW_KEY_LEFT_SUPER; |
105 | case XK_Super_R: return GLFW_KEY_RIGHT_SUPER; |
106 | case XK_Menu: return GLFW_KEY_MENU; |
107 | case XK_Num_Lock: return GLFW_KEY_NUM_LOCK; |
108 | case XK_Caps_Lock: return GLFW_KEY_CAPS_LOCK; |
109 | case XK_Print: return GLFW_KEY_PRINT_SCREEN; |
110 | case XK_Scroll_Lock: return GLFW_KEY_SCROLL_LOCK; |
111 | case XK_Pause: return GLFW_KEY_PAUSE; |
112 | case XK_Delete: return GLFW_KEY_DELETE; |
113 | case XK_BackSpace: return GLFW_KEY_BACKSPACE; |
114 | case XK_Return: return GLFW_KEY_ENTER; |
115 | case XK_Home: return GLFW_KEY_HOME; |
116 | case XK_End: return GLFW_KEY_END; |
117 | case XK_Page_Up: return GLFW_KEY_PAGE_UP; |
118 | case XK_Page_Down: return GLFW_KEY_PAGE_DOWN; |
119 | case XK_Insert: return GLFW_KEY_INSERT; |
120 | case XK_Left: return GLFW_KEY_LEFT; |
121 | case XK_Right: return GLFW_KEY_RIGHT; |
122 | case XK_Down: return GLFW_KEY_DOWN; |
123 | case XK_Up: return GLFW_KEY_UP; |
124 | case XK_F1: return GLFW_KEY_F1; |
125 | case XK_F2: return GLFW_KEY_F2; |
126 | case XK_F3: return GLFW_KEY_F3; |
127 | case XK_F4: return GLFW_KEY_F4; |
128 | case XK_F5: return GLFW_KEY_F5; |
129 | case XK_F6: return GLFW_KEY_F6; |
130 | case XK_F7: return GLFW_KEY_F7; |
131 | case XK_F8: return GLFW_KEY_F8; |
132 | case XK_F9: return GLFW_KEY_F9; |
133 | case XK_F10: return GLFW_KEY_F10; |
134 | case XK_F11: return GLFW_KEY_F11; |
135 | case XK_F12: return GLFW_KEY_F12; |
136 | case XK_F13: return GLFW_KEY_F13; |
137 | case XK_F14: return GLFW_KEY_F14; |
138 | case XK_F15: return GLFW_KEY_F15; |
139 | case XK_F16: return GLFW_KEY_F16; |
140 | case XK_F17: return GLFW_KEY_F17; |
141 | case XK_F18: return GLFW_KEY_F18; |
142 | case XK_F19: return GLFW_KEY_F19; |
143 | case XK_F20: return GLFW_KEY_F20; |
144 | case XK_F21: return GLFW_KEY_F21; |
145 | case XK_F22: return GLFW_KEY_F22; |
146 | case XK_F23: return GLFW_KEY_F23; |
147 | case XK_F24: return GLFW_KEY_F24; |
148 | case XK_F25: return GLFW_KEY_F25; |
149 | |
150 | // Numeric keypad |
151 | case XK_KP_Divide: return GLFW_KEY_KP_DIVIDE; |
152 | case XK_KP_Multiply: return GLFW_KEY_KP_MULTIPLY; |
153 | case XK_KP_Subtract: return GLFW_KEY_KP_SUBTRACT; |
154 | case XK_KP_Add: return GLFW_KEY_KP_ADD; |
155 | |
156 | // These should have been detected in secondary keysym test above! |
157 | case XK_KP_Insert: return GLFW_KEY_KP_0; |
158 | case XK_KP_End: return GLFW_KEY_KP_1; |
159 | case XK_KP_Down: return GLFW_KEY_KP_2; |
160 | case XK_KP_Page_Down: return GLFW_KEY_KP_3; |
161 | case XK_KP_Left: return GLFW_KEY_KP_4; |
162 | case XK_KP_Right: return GLFW_KEY_KP_6; |
163 | case XK_KP_Home: return GLFW_KEY_KP_7; |
164 | case XK_KP_Up: return GLFW_KEY_KP_8; |
165 | case XK_KP_Page_Up: return GLFW_KEY_KP_9; |
166 | case XK_KP_Delete: return GLFW_KEY_KP_DECIMAL; |
167 | case XK_KP_Equal: return GLFW_KEY_KP_EQUAL; |
168 | case XK_KP_Enter: return GLFW_KEY_KP_ENTER; |
169 | |
170 | // Last resort: Check for printable keys (should not happen if the XKB |
171 | // extension is available). This will give a layout dependent mapping |
172 | // (which is wrong, and we may miss some keys, especially on non-US |
173 | // keyboards), but it's better than nothing... |
174 | case XK_a: return GLFW_KEY_A; |
175 | case XK_b: return GLFW_KEY_B; |
176 | case XK_c: return GLFW_KEY_C; |
177 | case XK_d: return GLFW_KEY_D; |
178 | case XK_e: return GLFW_KEY_E; |
179 | case XK_f: return GLFW_KEY_F; |
180 | case XK_g: return GLFW_KEY_G; |
181 | case XK_h: return GLFW_KEY_H; |
182 | case XK_i: return GLFW_KEY_I; |
183 | case XK_j: return GLFW_KEY_J; |
184 | case XK_k: return GLFW_KEY_K; |
185 | case XK_l: return GLFW_KEY_L; |
186 | case XK_m: return GLFW_KEY_M; |
187 | case XK_n: return GLFW_KEY_N; |
188 | case XK_o: return GLFW_KEY_O; |
189 | case XK_p: return GLFW_KEY_P; |
190 | case XK_q: return GLFW_KEY_Q; |
191 | case XK_r: return GLFW_KEY_R; |
192 | case XK_s: return GLFW_KEY_S; |
193 | case XK_t: return GLFW_KEY_T; |
194 | case XK_u: return GLFW_KEY_U; |
195 | case XK_v: return GLFW_KEY_V; |
196 | case XK_w: return GLFW_KEY_W; |
197 | case XK_x: return GLFW_KEY_X; |
198 | case XK_y: return GLFW_KEY_Y; |
199 | case XK_z: return GLFW_KEY_Z; |
200 | case XK_1: return GLFW_KEY_1; |
201 | case XK_2: return GLFW_KEY_2; |
202 | case XK_3: return GLFW_KEY_3; |
203 | case XK_4: return GLFW_KEY_4; |
204 | case XK_5: return GLFW_KEY_5; |
205 | case XK_6: return GLFW_KEY_6; |
206 | case XK_7: return GLFW_KEY_7; |
207 | case XK_8: return GLFW_KEY_8; |
208 | case XK_9: return GLFW_KEY_9; |
209 | case XK_0: return GLFW_KEY_0; |
210 | case XK_space: return GLFW_KEY_SPACE; |
211 | case XK_minus: return GLFW_KEY_MINUS; |
212 | case XK_equal: return GLFW_KEY_EQUAL; |
213 | case XK_bracketleft: return GLFW_KEY_LEFT_BRACKET; |
214 | case XK_bracketright: return GLFW_KEY_RIGHT_BRACKET; |
215 | case XK_backslash: return GLFW_KEY_BACKSLASH; |
216 | case XK_semicolon: return GLFW_KEY_SEMICOLON; |
217 | case XK_apostrophe: return GLFW_KEY_APOSTROPHE; |
218 | case XK_grave: return GLFW_KEY_GRAVE_ACCENT; |
219 | case XK_comma: return GLFW_KEY_COMMA; |
220 | case XK_period: return GLFW_KEY_PERIOD; |
221 | case XK_slash: return GLFW_KEY_SLASH; |
222 | case XK_less: return GLFW_KEY_WORLD_1; // At least in some layouts... |
223 | default: break; |
224 | } |
225 | |
226 | // No matching translation was found |
227 | return GLFW_KEY_UNKNOWN; |
228 | } |
229 | |
230 | // Create key code translation tables |
231 | // |
232 | static void createKeyTables(void) |
233 | { |
234 | int scancode, key; |
235 | |
236 | memset(_glfw.x11.publicKeys, -1, sizeof(_glfw.x11.publicKeys)); |
237 | memset(_glfw.x11.nativeKeys, -1, sizeof(_glfw.x11.nativeKeys)); |
238 | |
239 | if (_glfw.x11.xkb.available) |
240 | { |
241 | // Use XKB to determine physical key locations independently of the current |
242 | // keyboard layout |
243 | |
244 | char name[XkbKeyNameLength + 1]; |
245 | XkbDescPtr desc = XkbGetMap(_glfw.x11.display, 0, XkbUseCoreKbd); |
246 | XkbGetNames(_glfw.x11.display, XkbKeyNamesMask, desc); |
247 | |
248 | // Find the X11 key code -> GLFW key code mapping |
249 | for (scancode = desc->min_key_code; scancode <= desc->max_key_code; scancode++) |
250 | { |
251 | memcpy(name, desc->names->keys[scancode].name, XkbKeyNameLength); |
252 | name[XkbKeyNameLength] = '\0'; |
253 | |
254 | // Map the key name to a GLFW key code. Note: We only map printable |
255 | // keys here, and we use the US keyboard layout. The rest of the |
256 | // keys (function keys) are mapped using traditional KeySym |
257 | // translations. |
258 | if (strcmp(name, "TLDE" ) == 0) key = GLFW_KEY_GRAVE_ACCENT; |
259 | else if (strcmp(name, "AE01" ) == 0) key = GLFW_KEY_1; |
260 | else if (strcmp(name, "AE02" ) == 0) key = GLFW_KEY_2; |
261 | else if (strcmp(name, "AE03" ) == 0) key = GLFW_KEY_3; |
262 | else if (strcmp(name, "AE04" ) == 0) key = GLFW_KEY_4; |
263 | else if (strcmp(name, "AE05" ) == 0) key = GLFW_KEY_5; |
264 | else if (strcmp(name, "AE06" ) == 0) key = GLFW_KEY_6; |
265 | else if (strcmp(name, "AE07" ) == 0) key = GLFW_KEY_7; |
266 | else if (strcmp(name, "AE08" ) == 0) key = GLFW_KEY_8; |
267 | else if (strcmp(name, "AE09" ) == 0) key = GLFW_KEY_9; |
268 | else if (strcmp(name, "AE10" ) == 0) key = GLFW_KEY_0; |
269 | else if (strcmp(name, "AE11" ) == 0) key = GLFW_KEY_MINUS; |
270 | else if (strcmp(name, "AE12" ) == 0) key = GLFW_KEY_EQUAL; |
271 | else if (strcmp(name, "AD01" ) == 0) key = GLFW_KEY_Q; |
272 | else if (strcmp(name, "AD02" ) == 0) key = GLFW_KEY_W; |
273 | else if (strcmp(name, "AD03" ) == 0) key = GLFW_KEY_E; |
274 | else if (strcmp(name, "AD04" ) == 0) key = GLFW_KEY_R; |
275 | else if (strcmp(name, "AD05" ) == 0) key = GLFW_KEY_T; |
276 | else if (strcmp(name, "AD06" ) == 0) key = GLFW_KEY_Y; |
277 | else if (strcmp(name, "AD07" ) == 0) key = GLFW_KEY_U; |
278 | else if (strcmp(name, "AD08" ) == 0) key = GLFW_KEY_I; |
279 | else if (strcmp(name, "AD09" ) == 0) key = GLFW_KEY_O; |
280 | else if (strcmp(name, "AD10" ) == 0) key = GLFW_KEY_P; |
281 | else if (strcmp(name, "AD11" ) == 0) key = GLFW_KEY_LEFT_BRACKET; |
282 | else if (strcmp(name, "AD12" ) == 0) key = GLFW_KEY_RIGHT_BRACKET; |
283 | else if (strcmp(name, "AC01" ) == 0) key = GLFW_KEY_A; |
284 | else if (strcmp(name, "AC02" ) == 0) key = GLFW_KEY_S; |
285 | else if (strcmp(name, "AC03" ) == 0) key = GLFW_KEY_D; |
286 | else if (strcmp(name, "AC04" ) == 0) key = GLFW_KEY_F; |
287 | else if (strcmp(name, "AC05" ) == 0) key = GLFW_KEY_G; |
288 | else if (strcmp(name, "AC06" ) == 0) key = GLFW_KEY_H; |
289 | else if (strcmp(name, "AC07" ) == 0) key = GLFW_KEY_J; |
290 | else if (strcmp(name, "AC08" ) == 0) key = GLFW_KEY_K; |
291 | else if (strcmp(name, "AC09" ) == 0) key = GLFW_KEY_L; |
292 | else if (strcmp(name, "AC10" ) == 0) key = GLFW_KEY_SEMICOLON; |
293 | else if (strcmp(name, "AC11" ) == 0) key = GLFW_KEY_APOSTROPHE; |
294 | else if (strcmp(name, "AB01" ) == 0) key = GLFW_KEY_Z; |
295 | else if (strcmp(name, "AB02" ) == 0) key = GLFW_KEY_X; |
296 | else if (strcmp(name, "AB03" ) == 0) key = GLFW_KEY_C; |
297 | else if (strcmp(name, "AB04" ) == 0) key = GLFW_KEY_V; |
298 | else if (strcmp(name, "AB05" ) == 0) key = GLFW_KEY_B; |
299 | else if (strcmp(name, "AB06" ) == 0) key = GLFW_KEY_N; |
300 | else if (strcmp(name, "AB07" ) == 0) key = GLFW_KEY_M; |
301 | else if (strcmp(name, "AB08" ) == 0) key = GLFW_KEY_COMMA; |
302 | else if (strcmp(name, "AB09" ) == 0) key = GLFW_KEY_PERIOD; |
303 | else if (strcmp(name, "AB10" ) == 0) key = GLFW_KEY_SLASH; |
304 | else if (strcmp(name, "BKSL" ) == 0) key = GLFW_KEY_BACKSLASH; |
305 | else if (strcmp(name, "LSGT" ) == 0) key = GLFW_KEY_WORLD_1; |
306 | else key = GLFW_KEY_UNKNOWN; |
307 | |
308 | if ((scancode >= 0) && (scancode < 256)) |
309 | _glfw.x11.publicKeys[scancode] = key; |
310 | } |
311 | |
312 | XkbFreeNames(desc, XkbKeyNamesMask, True); |
313 | XkbFreeKeyboard(desc, 0, True); |
314 | } |
315 | |
316 | for (scancode = 0; scancode < 256; scancode++) |
317 | { |
318 | // Translate the un-translated key codes using traditional X11 KeySym |
319 | // lookups |
320 | if (_glfw.x11.publicKeys[scancode] < 0) |
321 | _glfw.x11.publicKeys[scancode] = translateKeyCode(scancode); |
322 | |
323 | // Store the reverse translation for faster key name lookup |
324 | if (_glfw.x11.publicKeys[scancode] > 0) |
325 | _glfw.x11.nativeKeys[_glfw.x11.publicKeys[scancode]] = scancode; |
326 | } |
327 | } |
328 | |
329 | // Check whether the IM has a usable style |
330 | // |
331 | static GLFWbool hasUsableInputMethodStyle(void) |
332 | { |
333 | unsigned int i; |
334 | GLFWbool found = GLFW_FALSE; |
335 | XIMStyles* styles = NULL; |
336 | |
337 | if (XGetIMValues(_glfw.x11.im, XNQueryInputStyle, &styles, NULL) != NULL) |
338 | return GLFW_FALSE; |
339 | |
340 | for (i = 0; i < styles->count_styles; i++) |
341 | { |
342 | if (styles->supported_styles[i] == (XIMPreeditNothing | XIMStatusNothing)) |
343 | { |
344 | found = GLFW_TRUE; |
345 | break; |
346 | } |
347 | } |
348 | |
349 | XFree(styles); |
350 | return found; |
351 | } |
352 | |
353 | // Check whether the specified atom is supported |
354 | // |
355 | static Atom getSupportedAtom(Atom* supportedAtoms, |
356 | unsigned long atomCount, |
357 | const char* atomName) |
358 | { |
359 | unsigned long i; |
360 | const Atom atom = XInternAtom(_glfw.x11.display, atomName, False); |
361 | |
362 | for (i = 0; i < atomCount; i++) |
363 | { |
364 | if (supportedAtoms[i] == atom) |
365 | return atom; |
366 | } |
367 | |
368 | return None; |
369 | } |
370 | |
371 | // Check whether the running window manager is EWMH-compliant |
372 | // |
373 | static void detectEWMH(void) |
374 | { |
375 | Window* windowFromRoot = NULL; |
376 | Window* windowFromChild = NULL; |
377 | |
378 | // First we need a couple of atoms |
379 | const Atom supportingWmCheck = |
380 | XInternAtom(_glfw.x11.display, "_NET_SUPPORTING_WM_CHECK" , False); |
381 | const Atom wmSupported = |
382 | XInternAtom(_glfw.x11.display, "_NET_SUPPORTED" , False); |
383 | |
384 | // Then we look for the _NET_SUPPORTING_WM_CHECK property of the root window |
385 | if (_glfwGetWindowPropertyX11(_glfw.x11.root, |
386 | supportingWmCheck, |
387 | XA_WINDOW, |
388 | (unsigned char**) &windowFromRoot) != 1) |
389 | { |
390 | if (windowFromRoot) |
391 | XFree(windowFromRoot); |
392 | return; |
393 | } |
394 | |
395 | _glfwGrabErrorHandlerX11(); |
396 | |
397 | // It should be the ID of a child window (of the root) |
398 | // Then we look for the same property on the child window |
399 | if (_glfwGetWindowPropertyX11(*windowFromRoot, |
400 | supportingWmCheck, |
401 | XA_WINDOW, |
402 | (unsigned char**) &windowFromChild) != 1) |
403 | { |
404 | XFree(windowFromRoot); |
405 | if (windowFromChild) |
406 | XFree(windowFromChild); |
407 | return; |
408 | } |
409 | |
410 | _glfwReleaseErrorHandlerX11(); |
411 | |
412 | // It should be the ID of that same child window |
413 | if (*windowFromRoot != *windowFromChild) |
414 | { |
415 | XFree(windowFromRoot); |
416 | XFree(windowFromChild); |
417 | return; |
418 | } |
419 | |
420 | XFree(windowFromRoot); |
421 | XFree(windowFromChild); |
422 | |
423 | // We are now fairly sure that an EWMH-compliant window manager is running |
424 | |
425 | Atom* supportedAtoms; |
426 | unsigned long atomCount; |
427 | |
428 | // Now we need to check the _NET_SUPPORTED property of the root window |
429 | // It should be a list of supported WM protocol and state atoms |
430 | atomCount = _glfwGetWindowPropertyX11(_glfw.x11.root, |
431 | wmSupported, |
432 | XA_ATOM, |
433 | (unsigned char**) &supportedAtoms); |
434 | |
435 | // See which of the atoms we support that are supported by the WM |
436 | _glfw.x11.NET_WM_STATE = |
437 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_STATE" ); |
438 | _glfw.x11.NET_WM_STATE_ABOVE = |
439 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_STATE_ABOVE" ); |
440 | _glfw.x11.NET_WM_STATE_FULLSCREEN = |
441 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_STATE_FULLSCREEN" ); |
442 | _glfw.x11.NET_WM_STATE_MAXIMIZED_VERT = |
443 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_STATE_MAXIMIZED_VERT" ); |
444 | _glfw.x11.NET_WM_STATE_MAXIMIZED_HORZ = |
445 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_STATE_MAXIMIZED_HORZ" ); |
446 | _glfw.x11.NET_WM_FULLSCREEN_MONITORS = |
447 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_FULLSCREEN_MONITORS" ); |
448 | _glfw.x11.NET_WM_WINDOW_TYPE = |
449 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_WINDOW_TYPE" ); |
450 | _glfw.x11.NET_WM_WINDOW_TYPE_NORMAL = |
451 | getSupportedAtom(supportedAtoms, atomCount, "_NET_WM_WINDOW_TYPE_NORMAL" ); |
452 | _glfw.x11.NET_ACTIVE_WINDOW = |
453 | getSupportedAtom(supportedAtoms, atomCount, "_NET_ACTIVE_WINDOW" ); |
454 | _glfw.x11.NET_FRAME_EXTENTS = |
455 | getSupportedAtom(supportedAtoms, atomCount, "_NET_FRAME_EXTENTS" ); |
456 | _glfw.x11.NET_REQUEST_FRAME_EXTENTS = |
457 | getSupportedAtom(supportedAtoms, atomCount, "_NET_REQUEST_FRAME_EXTENTS" ); |
458 | |
459 | XFree(supportedAtoms); |
460 | } |
461 | |
462 | // Initialize X11 display and look for supported X11 extensions |
463 | // |
464 | static GLFWbool initExtensions(void) |
465 | { |
466 | #if defined(_GLFW_HAS_XF86VM) |
467 | // Check for XF86VidMode extension |
468 | _glfw.x11.vidmode.available = |
469 | XF86VidModeQueryExtension(_glfw.x11.display, |
470 | &_glfw.x11.vidmode.eventBase, |
471 | &_glfw.x11.vidmode.errorBase); |
472 | #endif /*_GLFW_HAS_XF86VM*/ |
473 | |
474 | // Check for RandR extension |
475 | if (XRRQueryExtension(_glfw.x11.display, |
476 | &_glfw.x11.randr.eventBase, |
477 | &_glfw.x11.randr.errorBase)) |
478 | { |
479 | if (XRRQueryVersion(_glfw.x11.display, |
480 | &_glfw.x11.randr.major, |
481 | &_glfw.x11.randr.minor)) |
482 | { |
483 | // The GLFW RandR path requires at least version 1.3 |
484 | if (_glfw.x11.randr.major > 1 || _glfw.x11.randr.minor >= 3) |
485 | _glfw.x11.randr.available = GLFW_TRUE; |
486 | } |
487 | else |
488 | { |
489 | _glfwInputError(GLFW_PLATFORM_ERROR, |
490 | "X11: Failed to query RandR version" ); |
491 | } |
492 | } |
493 | |
494 | if (_glfw.x11.randr.available) |
495 | { |
496 | XRRScreenResources* sr = XRRGetScreenResources(_glfw.x11.display, |
497 | _glfw.x11.root); |
498 | |
499 | if (!sr->ncrtc || !XRRGetCrtcGammaSize(_glfw.x11.display, sr->crtcs[0])) |
500 | { |
501 | // This is either a headless system or an older Nvidia binary driver |
502 | // with broken gamma support |
503 | // Flag it as useless and fall back to Xf86VidMode gamma, if |
504 | // available |
505 | _glfwInputError(GLFW_PLATFORM_ERROR, |
506 | "X11: RandR gamma ramp support seems broken" ); |
507 | _glfw.x11.randr.gammaBroken = GLFW_TRUE; |
508 | } |
509 | |
510 | XRRFreeScreenResources(sr); |
511 | |
512 | XRRSelectInput(_glfw.x11.display, _glfw.x11.root, |
513 | RROutputChangeNotifyMask); |
514 | } |
515 | |
516 | if (XineramaQueryExtension(_glfw.x11.display, |
517 | &_glfw.x11.xinerama.major, |
518 | &_glfw.x11.xinerama.minor)) |
519 | { |
520 | if (XineramaIsActive(_glfw.x11.display)) |
521 | _glfw.x11.xinerama.available = GLFW_TRUE; |
522 | } |
523 | |
524 | // Check if Xkb is supported on this display |
525 | _glfw.x11.xkb.major = 1; |
526 | _glfw.x11.xkb.minor = 0; |
527 | _glfw.x11.xkb.available = |
528 | XkbQueryExtension(_glfw.x11.display, |
529 | &_glfw.x11.xkb.majorOpcode, |
530 | &_glfw.x11.xkb.eventBase, |
531 | &_glfw.x11.xkb.errorBase, |
532 | &_glfw.x11.xkb.major, |
533 | &_glfw.x11.xkb.minor); |
534 | |
535 | if (_glfw.x11.xkb.available) |
536 | { |
537 | Bool supported; |
538 | |
539 | if (XkbSetDetectableAutoRepeat(_glfw.x11.display, True, &supported)) |
540 | { |
541 | if (supported) |
542 | _glfw.x11.xkb.detectable = GLFW_TRUE; |
543 | } |
544 | } |
545 | |
546 | _glfw.x11.x11xcb.handle = dlopen("libX11-xcb.so" , RTLD_LAZY | RTLD_GLOBAL); |
547 | if (_glfw.x11.x11xcb.handle) |
548 | { |
549 | _glfw.x11.x11xcb.XGetXCBConnection = (XGETXCBCONNECTION_T) |
550 | dlsym(_glfw.x11.x11xcb.handle, "XGetXCBConnection" ); |
551 | } |
552 | |
553 | // Update the key code LUT |
554 | // FIXME: We should listen to XkbMapNotify events to track changes to |
555 | // the keyboard mapping. |
556 | createKeyTables(); |
557 | |
558 | // Detect whether an EWMH-conformant window manager is running |
559 | detectEWMH(); |
560 | |
561 | // String format atoms |
562 | _glfw.x11.NULL_ = XInternAtom(_glfw.x11.display, "NULL" , False); |
563 | _glfw.x11.UTF8_STRING = |
564 | XInternAtom(_glfw.x11.display, "UTF8_STRING" , False); |
565 | _glfw.x11.COMPOUND_STRING = |
566 | XInternAtom(_glfw.x11.display, "COMPOUND_STRING" , False); |
567 | _glfw.x11.ATOM_PAIR = XInternAtom(_glfw.x11.display, "ATOM_PAIR" , False); |
568 | |
569 | // Custom selection property atom |
570 | _glfw.x11.GLFW_SELECTION = |
571 | XInternAtom(_glfw.x11.display, "GLFW_SELECTION" , False); |
572 | |
573 | // ICCCM standard clipboard atoms |
574 | _glfw.x11.TARGETS = XInternAtom(_glfw.x11.display, "TARGETS" , False); |
575 | _glfw.x11.MULTIPLE = XInternAtom(_glfw.x11.display, "MULTIPLE" , False); |
576 | _glfw.x11.CLIPBOARD = XInternAtom(_glfw.x11.display, "CLIPBOARD" , False); |
577 | |
578 | // Clipboard manager atoms |
579 | _glfw.x11.CLIPBOARD_MANAGER = |
580 | XInternAtom(_glfw.x11.display, "CLIPBOARD_MANAGER" , False); |
581 | _glfw.x11.SAVE_TARGETS = |
582 | XInternAtom(_glfw.x11.display, "SAVE_TARGETS" , False); |
583 | |
584 | // Xdnd (drag and drop) atoms |
585 | _glfw.x11.XdndAware = XInternAtom(_glfw.x11.display, "XdndAware" , False); |
586 | _glfw.x11.XdndEnter = XInternAtom(_glfw.x11.display, "XdndEnter" , False); |
587 | _glfw.x11.XdndPosition = XInternAtom(_glfw.x11.display, "XdndPosition" , False); |
588 | _glfw.x11.XdndStatus = XInternAtom(_glfw.x11.display, "XdndStatus" , False); |
589 | _glfw.x11.XdndActionCopy = XInternAtom(_glfw.x11.display, "XdndActionCopy" , False); |
590 | _glfw.x11.XdndDrop = XInternAtom(_glfw.x11.display, "XdndDrop" , False); |
591 | _glfw.x11.XdndLeave = XInternAtom(_glfw.x11.display, "XdndLeave" , False); |
592 | _glfw.x11.XdndFinished = XInternAtom(_glfw.x11.display, "XdndFinished" , False); |
593 | _glfw.x11.XdndSelection = XInternAtom(_glfw.x11.display, "XdndSelection" , False); |
594 | |
595 | // ICCCM, EWMH and Motif window property atoms |
596 | // These can be set safely even without WM support |
597 | // The EWMH atoms that require WM support are handled in detectEWMH |
598 | _glfw.x11.WM_PROTOCOLS = |
599 | XInternAtom(_glfw.x11.display, "WM_PROTOCOLS" , False); |
600 | _glfw.x11.WM_STATE = |
601 | XInternAtom(_glfw.x11.display, "WM_STATE" , False); |
602 | _glfw.x11.WM_DELETE_WINDOW = |
603 | XInternAtom(_glfw.x11.display, "WM_DELETE_WINDOW" , False); |
604 | _glfw.x11.NET_WM_ICON = |
605 | XInternAtom(_glfw.x11.display, "_NET_WM_ICON" , False); |
606 | _glfw.x11.NET_WM_PING = |
607 | XInternAtom(_glfw.x11.display, "_NET_WM_PING" , False); |
608 | _glfw.x11.NET_WM_PID = |
609 | XInternAtom(_glfw.x11.display, "_NET_WM_PID" , False); |
610 | _glfw.x11.NET_WM_NAME = |
611 | XInternAtom(_glfw.x11.display, "_NET_WM_NAME" , False); |
612 | _glfw.x11.NET_WM_ICON_NAME = |
613 | XInternAtom(_glfw.x11.display, "_NET_WM_ICON_NAME" , False); |
614 | _glfw.x11.NET_WM_BYPASS_COMPOSITOR = |
615 | XInternAtom(_glfw.x11.display, "_NET_WM_BYPASS_COMPOSITOR" , False); |
616 | _glfw.x11.MOTIF_WM_HINTS = |
617 | XInternAtom(_glfw.x11.display, "_MOTIF_WM_HINTS" , False); |
618 | |
619 | return GLFW_TRUE; |
620 | } |
621 | |
622 | // Create a blank cursor for hidden and disabled cursor modes |
623 | // |
624 | static Cursor createHiddenCursor(void) |
625 | { |
626 | unsigned char pixels[16 * 16 * 4]; |
627 | GLFWimage image = { 16, 16, pixels }; |
628 | |
629 | memset(pixels, 0, sizeof(pixels)); |
630 | |
631 | return _glfwCreateCursorX11(&image, 0, 0); |
632 | } |
633 | |
634 | // X error handler |
635 | // |
636 | static int errorHandler(Display *display, XErrorEvent* event) |
637 | { |
638 | _glfw.x11.errorCode = event->error_code; |
639 | return 0; |
640 | } |
641 | |
642 | |
643 | ////////////////////////////////////////////////////////////////////////// |
644 | ////// GLFW internal API ////// |
645 | ////////////////////////////////////////////////////////////////////////// |
646 | |
647 | // Sets the X error handler callback |
648 | // |
649 | void _glfwGrabErrorHandlerX11(void) |
650 | { |
651 | _glfw.x11.errorCode = Success; |
652 | XSetErrorHandler(errorHandler); |
653 | } |
654 | |
655 | // Clears the X error handler callback |
656 | // |
657 | void _glfwReleaseErrorHandlerX11(void) |
658 | { |
659 | // Synchronize to make sure all commands are processed |
660 | XSync(_glfw.x11.display, False); |
661 | XSetErrorHandler(NULL); |
662 | } |
663 | |
664 | // Reports the specified error, appending information about the last X error |
665 | // |
666 | void _glfwInputErrorX11(int error, const char* message) |
667 | { |
668 | char buffer[8192]; |
669 | XGetErrorText(_glfw.x11.display, _glfw.x11.errorCode, |
670 | buffer, sizeof(buffer)); |
671 | |
672 | _glfwInputError(error, "%s: %s" , message, buffer); |
673 | } |
674 | |
675 | // Creates a native cursor object from the specified image and hotspot |
676 | // |
677 | Cursor _glfwCreateCursorX11(const GLFWimage* image, int xhot, int yhot) |
678 | { |
679 | int i; |
680 | Cursor cursor; |
681 | |
682 | XcursorImage* native = XcursorImageCreate(image->width, image->height); |
683 | if (native == NULL) |
684 | return None; |
685 | |
686 | native->xhot = xhot; |
687 | native->yhot = yhot; |
688 | |
689 | unsigned char* source = (unsigned char*) image->pixels; |
690 | XcursorPixel* target = native->pixels; |
691 | |
692 | for (i = 0; i < image->width * image->height; i++, target++, source += 4) |
693 | { |
694 | unsigned int alpha = source[3]; |
695 | |
696 | *target = (alpha << 24) | |
697 | ((unsigned char) ((source[0] * alpha) / 255) << 16) | |
698 | ((unsigned char) ((source[1] * alpha) / 255) << 8) | |
699 | ((unsigned char) ((source[2] * alpha) / 255) << 0); |
700 | } |
701 | |
702 | cursor = XcursorImageLoadCursor(_glfw.x11.display, native); |
703 | XcursorImageDestroy(native); |
704 | |
705 | return cursor; |
706 | } |
707 | |
708 | |
709 | ////////////////////////////////////////////////////////////////////////// |
710 | ////// GLFW platform API ////// |
711 | ////////////////////////////////////////////////////////////////////////// |
712 | |
713 | int _glfwPlatformInit(void) |
714 | { |
715 | #if !defined(X_HAVE_UTF8_STRING) |
716 | // HACK: If the current locale is C, apply the environment's locale |
717 | // This is done because the C locale breaks wide character input |
718 | if (strcmp(setlocale(LC_CTYPE, NULL), "C" ) == 0) |
719 | setlocale(LC_CTYPE, "" ); |
720 | #endif |
721 | |
722 | XInitThreads(); |
723 | |
724 | _glfw.x11.display = XOpenDisplay(NULL); |
725 | if (!_glfw.x11.display) |
726 | { |
727 | const char* display = getenv("DISPLAY" ); |
728 | if (display) |
729 | { |
730 | _glfwInputError(GLFW_PLATFORM_ERROR, |
731 | "X11: Failed to open display %s" , display); |
732 | } |
733 | else |
734 | { |
735 | _glfwInputError(GLFW_PLATFORM_ERROR, |
736 | "X11: The DISPLAY environment variable is missing" ); |
737 | } |
738 | |
739 | return GLFW_FALSE; |
740 | } |
741 | |
742 | _glfw.x11.screen = DefaultScreen(_glfw.x11.display); |
743 | _glfw.x11.root = RootWindow(_glfw.x11.display, _glfw.x11.screen); |
744 | _glfw.x11.context = XUniqueContext(); |
745 | |
746 | if (!initExtensions()) |
747 | return GLFW_FALSE; |
748 | |
749 | _glfw.x11.cursor = createHiddenCursor(); |
750 | |
751 | if (XSupportsLocale()) |
752 | { |
753 | XSetLocaleModifiers("" ); |
754 | |
755 | _glfw.x11.im = XOpenIM(_glfw.x11.display, 0, NULL, NULL); |
756 | if (_glfw.x11.im) |
757 | { |
758 | if (!hasUsableInputMethodStyle()) |
759 | { |
760 | XCloseIM(_glfw.x11.im); |
761 | _glfw.x11.im = NULL; |
762 | } |
763 | } |
764 | } |
765 | |
766 | if (!_glfwInitThreadLocalStoragePOSIX()) |
767 | return GLFW_FALSE; |
768 | |
769 | if (!_glfwInitGLX()) |
770 | return GLFW_FALSE; |
771 | |
772 | if (!_glfwInitJoysticksLinux()) |
773 | return GLFW_FALSE; |
774 | |
775 | _glfwInitEGL(); |
776 | _glfwInitTimerPOSIX(); |
777 | |
778 | return GLFW_TRUE; |
779 | } |
780 | |
781 | void _glfwPlatformTerminate(void) |
782 | { |
783 | if (_glfw.x11.x11xcb.handle) |
784 | { |
785 | dlclose(_glfw.x11.x11xcb.handle); |
786 | _glfw.x11.x11xcb.handle = NULL; |
787 | } |
788 | |
789 | if (_glfw.x11.cursor) |
790 | { |
791 | XFreeCursor(_glfw.x11.display, _glfw.x11.cursor); |
792 | _glfw.x11.cursor = (Cursor) 0; |
793 | } |
794 | |
795 | free(_glfw.x11.clipboardString); |
796 | |
797 | if (_glfw.x11.im) |
798 | { |
799 | XCloseIM(_glfw.x11.im); |
800 | _glfw.x11.im = NULL; |
801 | } |
802 | |
803 | if (_glfw.x11.display) |
804 | { |
805 | XCloseDisplay(_glfw.x11.display); |
806 | _glfw.x11.display = NULL; |
807 | } |
808 | |
809 | // NOTE: This needs to be done after XCloseDisplay, as libGL registers |
810 | // cleanup callbacks that get called by it |
811 | _glfwTerminateEGL(); |
812 | _glfwTerminateGLX(); |
813 | |
814 | _glfwTerminateJoysticksLinux(); |
815 | _glfwTerminateThreadLocalStoragePOSIX(); |
816 | } |
817 | |
818 | const char* _glfwPlatformGetVersionString(void) |
819 | { |
820 | return _GLFW_VERSION_NUMBER " X11 GLX EGL" |
821 | #if defined(_POSIX_TIMERS) && defined(_POSIX_MONOTONIC_CLOCK) |
822 | " clock_gettime" |
823 | #else |
824 | " gettimeofday" |
825 | #endif |
826 | #if defined(__linux__) |
827 | " /dev/js" |
828 | #endif |
829 | #if defined(_GLFW_HAS_XF86VM) |
830 | " Xf86vm" |
831 | #endif |
832 | #if defined(_GLFW_BUILD_DLL) |
833 | " shared" |
834 | #endif |
835 | ; |
836 | } |
837 | |
838 | |