1 //
2 // Copyright (c) 2015 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6
7 // X11Window.cpp: Implementation of OSWindow for X11
8
9 #include "x11/X11Window.h"
10
11 #include "aemu/base/system/System.h"
12
13 #include <assert.h>
14
15 namespace {
16
WaitForMapNotify(Display * dpy,XEvent * event,XPointer window)17 Bool WaitForMapNotify(Display *dpy, XEvent *event, XPointer window)
18 {
19 return event->type == MapNotify && event->xmap.window == reinterpret_cast<Window>(window);
20 }
21
X11CodeToKey(Display * display,unsigned int scancode)22 static Key X11CodeToKey(Display *display, unsigned int scancode)
23 {
24 int temp;
25 KeySym *keySymbols;
26 keySymbols = XGetKeyboardMapping(display, scancode, 1, &temp);
27
28 unsigned int keySymbol = keySymbols[0];
29 XFree(keySymbols);
30
31 switch (keySymbol)
32 {
33 case XK_Shift_L: return KEY_LSHIFT;
34 case XK_Shift_R: return KEY_RSHIFT;
35 case XK_Alt_L: return KEY_LALT;
36 case XK_Alt_R: return KEY_RALT;
37 case XK_Control_L: return KEY_LCONTROL;
38 case XK_Control_R: return KEY_RCONTROL;
39 case XK_Super_L: return KEY_LSYSTEM;
40 case XK_Super_R: return KEY_RSYSTEM;
41 case XK_Menu: return KEY_MENU;
42
43 case XK_semicolon: return KEY_SEMICOLON;
44 case XK_slash: return KEY_SLASH;
45 case XK_equal: return KEY_EQUAL;
46 case XK_minus: return KEY_DASH;
47 case XK_bracketleft: return KEY_LBRACKET;
48 case XK_bracketright:return KEY_RBRACKET;
49 case XK_comma: return KEY_COMMA;
50 case XK_period: return KEY_PERIOD;
51 case XK_backslash: return KEY_BACKSLASH;
52 case XK_asciitilde: return KEY_TILDE;
53 case XK_Escape: return KEY_ESCAPE;
54 case XK_space: return KEY_SPACE;
55 case XK_Return: return KEY_RETURN;
56 case XK_BackSpace: return KEY_BACK;
57 case XK_Tab: return KEY_TAB;
58 case XK_Page_Up: return KEY_PAGEUP;
59 case XK_Page_Down: return KEY_PAGEDOWN;
60 case XK_End: return KEY_END;
61 case XK_Home: return KEY_HOME;
62 case XK_Insert: return KEY_INSERT;
63 case XK_Delete: return KEY_DELETE;
64 case XK_KP_Add: return KEY_ADD;
65 case XK_KP_Subtract: return KEY_SUBTRACT;
66 case XK_KP_Multiply: return KEY_MULTIPLY;
67 case XK_KP_Divide: return KEY_DIVIDE;
68 case XK_Pause: return KEY_PAUSE;
69
70 case XK_F1: return KEY_F1;
71 case XK_F2: return KEY_F2;
72 case XK_F3: return KEY_F3;
73 case XK_F4: return KEY_F4;
74 case XK_F5: return KEY_F5;
75 case XK_F6: return KEY_F6;
76 case XK_F7: return KEY_F7;
77 case XK_F8: return KEY_F8;
78 case XK_F9: return KEY_F9;
79 case XK_F10: return KEY_F10;
80 case XK_F11: return KEY_F11;
81 case XK_F12: return KEY_F12;
82 case XK_F13: return KEY_F13;
83 case XK_F14: return KEY_F14;
84 case XK_F15: return KEY_F15;
85
86 case XK_Left: return KEY_LEFT;
87 case XK_Right: return KEY_RIGHT;
88 case XK_Down: return KEY_DOWN;
89 case XK_Up: return KEY_UP;
90
91 case XK_KP_Insert: return KEY_NUMPAD0;
92 case XK_KP_End: return KEY_NUMPAD1;
93 case XK_KP_Down: return KEY_NUMPAD2;
94 case XK_KP_Page_Down:return KEY_NUMPAD3;
95 case XK_KP_Left: return KEY_NUMPAD4;
96 case XK_KP_5: return KEY_NUMPAD5;
97 case XK_KP_Right: return KEY_NUMPAD6;
98 case XK_KP_Home: return KEY_NUMPAD7;
99 case XK_KP_Up: return KEY_NUMPAD8;
100 case XK_KP_Page_Up: return KEY_NUMPAD9;
101
102 case XK_a: return KEY_A;
103 case XK_b: return KEY_B;
104 case XK_c: return KEY_C;
105 case XK_d: return KEY_D;
106 case XK_e: return KEY_E;
107 case XK_f: return KEY_F;
108 case XK_g: return KEY_G;
109 case XK_h: return KEY_H;
110 case XK_i: return KEY_I;
111 case XK_j: return KEY_J;
112 case XK_k: return KEY_K;
113 case XK_l: return KEY_L;
114 case XK_m: return KEY_M;
115 case XK_n: return KEY_N;
116 case XK_o: return KEY_O;
117 case XK_p: return KEY_P;
118 case XK_q: return KEY_Q;
119 case XK_r: return KEY_R;
120 case XK_s: return KEY_S;
121 case XK_t: return KEY_T;
122 case XK_u: return KEY_U;
123 case XK_v: return KEY_V;
124 case XK_w: return KEY_W;
125 case XK_x: return KEY_X;
126 case XK_y: return KEY_Y;
127 case XK_z: return KEY_Z;
128
129 case XK_1: return KEY_NUM1;
130 case XK_2: return KEY_NUM2;
131 case XK_3: return KEY_NUM3;
132 case XK_4: return KEY_NUM4;
133 case XK_5: return KEY_NUM5;
134 case XK_6: return KEY_NUM6;
135 case XK_7: return KEY_NUM7;
136 case XK_8: return KEY_NUM8;
137 case XK_9: return KEY_NUM9;
138 case XK_0: return KEY_NUM0;
139 }
140
141 return Key(0);
142 }
143
AddX11KeyStateToEvent(Event * event,unsigned int state)144 static void AddX11KeyStateToEvent(Event *event, unsigned int state)
145 {
146 event->Key.Shift = state & ShiftMask;
147 event->Key.Control = state & ControlMask;
148 event->Key.Alt = state & Mod1Mask;
149 event->Key.System = state & Mod4Mask;
150 }
151
152 }
153
X11Window()154 X11Window::X11Window()
155 : WM_DELETE_WINDOW(None),
156 WM_PROTOCOLS(None),
157 TEST_EVENT(None),
158 mDisplay(nullptr),
159 mWindow(0),
160 mRequestedVisualId(-1),
161 mVisible(false)
162 {
163 }
164
X11Window(int visualId)165 X11Window::X11Window(int visualId)
166 : WM_DELETE_WINDOW(None),
167 WM_PROTOCOLS(None),
168 TEST_EVENT(None),
169 mDisplay(nullptr),
170 mWindow(0),
171 mRequestedVisualId(visualId),
172 mVisible(false)
173 {
174 }
175
~X11Window()176 X11Window::~X11Window()
177 {
178 destroy();
179 }
180
initialize(const std::string & name,size_t width,size_t height)181 bool X11Window::initialize(const std::string &name, size_t width, size_t height)
182 {
183 destroy();
184
185 mDisplay = XOpenDisplay(nullptr);
186 if (!mDisplay)
187 {
188 return false;
189 }
190
191 {
192 int screen = DefaultScreen(mDisplay);
193 Window root = RootWindow(mDisplay, screen);
194
195 Visual *visual;
196 if (mRequestedVisualId == -1)
197 {
198 visual = DefaultVisual(mDisplay, screen);
199 }
200 else
201 {
202 XVisualInfo visualTemplate;
203 visualTemplate.visualid = mRequestedVisualId;
204
205 int numVisuals = 0;
206 XVisualInfo *visuals = XGetVisualInfo(mDisplay, VisualIDMask, &visualTemplate, &numVisuals);
207 if (numVisuals <= 0)
208 {
209 return false;
210 }
211 assert(numVisuals == 1);
212
213 visual = visuals[0].visual;
214 XFree(visuals);
215 }
216
217 int depth = DefaultDepth(mDisplay, screen);
218 Colormap colormap = XCreateColormap(mDisplay, root, visual, AllocNone);
219
220 XSetWindowAttributes attributes;
221 unsigned long attributeMask = CWBorderPixel | CWColormap | CWEventMask;
222
223 attributes.event_mask = StructureNotifyMask | PointerMotionMask | ButtonPressMask |
224 ButtonReleaseMask | FocusChangeMask | EnterWindowMask |
225 LeaveWindowMask | KeyPressMask | KeyReleaseMask;
226 attributes.border_pixel = 0;
227 attributes.colormap = colormap;
228
229 mWindow = XCreateWindow(mDisplay, root, 0, 0, width, height, 0, depth, InputOutput,
230 visual, attributeMask, &attributes);
231 XFreeColormap(mDisplay, colormap);
232 }
233
234 if (!mWindow)
235 {
236 destroy();
237 return false;
238 }
239
240 // Tell the window manager to notify us when the user wants to close the
241 // window so we can do it ourselves.
242 WM_DELETE_WINDOW = XInternAtom(mDisplay, "WM_DELETE_WINDOW", False);
243 WM_PROTOCOLS = XInternAtom(mDisplay, "WM_PROTOCOLS", False);
244 if (WM_DELETE_WINDOW == None || WM_PROTOCOLS == None)
245 {
246 destroy();
247 return false;
248 }
249
250 if(XSetWMProtocols(mDisplay, mWindow, &WM_DELETE_WINDOW, 1) == 0)
251 {
252 destroy();
253 return false;
254 }
255
256 // Create an atom to identify our test event
257 TEST_EVENT = XInternAtom(mDisplay, "ANGLE_TEST_EVENT", False);
258 if (TEST_EVENT == None)
259 {
260 destroy();
261 return false;
262 }
263
264 XFlush(mDisplay);
265
266 mX = 0;
267 mY = 0;
268 mWidth = width;
269 mHeight = height;
270
271 return true;
272 }
273
destroy()274 void X11Window::destroy()
275 {
276 if (mWindow)
277 {
278 XDestroyWindow(mDisplay, mWindow);
279 mWindow = 0;
280 }
281 if (mDisplay)
282 {
283 XCloseDisplay(mDisplay);
284 mDisplay = nullptr;
285 }
286 WM_DELETE_WINDOW = None;
287 WM_PROTOCOLS = None;
288 }
289
getNativeWindow() const290 EGLNativeWindowType X11Window::getNativeWindow() const
291 {
292 return mWindow;
293 }
294
getNativeDisplay() const295 EGLNativeDisplayType X11Window::getNativeDisplay() const
296 {
297 return mDisplay;
298 }
299
getFramebufferNativeWindow() const300 void* X11Window::getFramebufferNativeWindow() const
301 {
302 int screen = DefaultScreen(mDisplay);
303 Window root = RootWindow(mDisplay, screen);
304 return (void*)(uintptr_t)root;
305 }
306
messageLoop()307 void X11Window::messageLoop()
308 {
309 int eventCount = XPending(mDisplay);
310 while (eventCount--)
311 {
312 XEvent event;
313 XNextEvent(mDisplay, &event);
314 processEvent(event);
315 }
316 }
317
setMousePosition(int x,int y)318 void X11Window::setMousePosition(int x, int y)
319 {
320 XWarpPointer(mDisplay, None, mWindow, 0, 0, 0, 0, x, y);
321 }
322
CreateOSWindow()323 OSWindow *CreateOSWindow()
324 {
325 return new X11Window();
326 }
327
setPosition(int x,int y)328 bool X11Window::setPosition(int x, int y)
329 {
330 XMoveWindow(mDisplay, mWindow, x, y);
331 XFlush(mDisplay);
332 return true;
333 }
334
resize(int width,int height)335 bool X11Window::resize(int width, int height)
336 {
337 XResizeWindow(mDisplay, mWindow, width, height);
338 XFlush(mDisplay);
339
340 // Wait until the window as actually been resized so that the code calling resize
341 // can assume the window has been resized.
342 const double kResizeWaitDelay = 0.2;
343 while (mHeight != height && mWidth != width)
344 {
345 messageLoop();
346 android::base::sleepMs(100);
347 }
348
349 return true;
350 }
351
setVisible(bool isVisible)352 void X11Window::setVisible(bool isVisible)
353 {
354 if (mVisible == isVisible)
355 {
356 return;
357 }
358
359 if (isVisible)
360 {
361 XMapWindow(mDisplay, mWindow);
362
363 // Wait until we get an event saying this window is mapped so that the
364 // code calling setVisible can assume the window is visible.
365 // This is important when creating a framebuffer as the framebuffer content
366 // is undefined when the window is not visible.
367 XEvent dummyEvent;
368 XIfEvent(mDisplay, &dummyEvent, WaitForMapNotify, reinterpret_cast<XPointer>(mWindow));
369 }
370 else
371 {
372 XUnmapWindow(mDisplay, mWindow);
373 XFlush(mDisplay);
374 }
375 mVisible = isVisible;
376 }
377
signalTestEvent()378 void X11Window::signalTestEvent()
379 {
380 XEvent event;
381 event.type = ClientMessage;
382 event.xclient.message_type = TEST_EVENT;
383 // Format needs to be valid or a BadValue is generated
384 event.xclient.format = 32;
385
386 // Hijack StructureNotifyMask as we know we will be listening for it.
387 XSendEvent(mDisplay, mWindow, False, StructureNotifyMask, &event);
388 }
389
processEvent(const XEvent & xEvent)390 void X11Window::processEvent(const XEvent &xEvent)
391 {
392 // TODO(cwallez) text events
393 switch (xEvent.type)
394 {
395 case ButtonPress:
396 {
397 Event event;
398 MouseButton button = MOUSEBUTTON_UNKNOWN;
399 int wheelY = 0;
400
401 // The mouse wheel updates are sent via button events.
402 switch (xEvent.xbutton.button)
403 {
404 case Button4:
405 wheelY = 1;
406 break;
407 case Button5:
408 wheelY = -1;
409 break;
410 case 6:
411 break;
412 case 7:
413 break;
414
415 case Button1:
416 button = MOUSEBUTTON_LEFT;
417 break;
418 case Button2:
419 button = MOUSEBUTTON_MIDDLE;
420 break;
421 case Button3:
422 button = MOUSEBUTTON_RIGHT;
423 break;
424 case 8:
425 button = MOUSEBUTTON_BUTTON4;
426 break;
427 case 9:
428 button = MOUSEBUTTON_BUTTON5;
429 break;
430
431 default:
432 break;
433 }
434
435 if (wheelY != 0)
436 {
437 event.Type = Event::EVENT_MOUSE_WHEEL_MOVED;
438 event.MouseWheel.Delta = wheelY;
439 pushEvent(event);
440 }
441
442 if (button != MOUSEBUTTON_UNKNOWN)
443 {
444 event.Type = Event::EVENT_MOUSE_BUTTON_RELEASED;
445 event.MouseButton.Button = button;
446 event.MouseButton.X = xEvent.xbutton.x;
447 event.MouseButton.Y = xEvent.xbutton.y;
448 pushEvent(event);
449 }
450 }
451 break;
452
453 case ButtonRelease:
454 {
455 Event event;
456 MouseButton button = MOUSEBUTTON_UNKNOWN;
457
458 switch (xEvent.xbutton.button)
459 {
460 case Button1:
461 button = MOUSEBUTTON_LEFT;
462 break;
463 case Button2:
464 button = MOUSEBUTTON_MIDDLE;
465 break;
466 case Button3:
467 button = MOUSEBUTTON_RIGHT;
468 break;
469 case 8:
470 button = MOUSEBUTTON_BUTTON4;
471 break;
472 case 9:
473 button = MOUSEBUTTON_BUTTON5;
474 break;
475
476 default:
477 break;
478 }
479
480 if (button != MOUSEBUTTON_UNKNOWN)
481 {
482 event.Type = Event::EVENT_MOUSE_BUTTON_RELEASED;
483 event.MouseButton.Button = button;
484 event.MouseButton.X = xEvent.xbutton.x;
485 event.MouseButton.Y = xEvent.xbutton.y;
486 pushEvent(event);
487 }
488 }
489 break;
490
491 case KeyPress:
492 {
493 Event event;
494 event.Type = Event::EVENT_KEY_PRESSED;
495 event.Key.Code = X11CodeToKey(mDisplay, xEvent.xkey.keycode);
496 AddX11KeyStateToEvent(&event, xEvent.xkey.state);
497 pushEvent(event);
498 }
499 break;
500
501 case KeyRelease:
502 {
503 Event event;
504 event.Type = Event::EVENT_KEY_RELEASED;
505 event.Key.Code = X11CodeToKey(mDisplay, xEvent.xkey.keycode);
506 AddX11KeyStateToEvent(&event, xEvent.xkey.state);
507 pushEvent(event);
508 }
509 break;
510
511 case EnterNotify:
512 {
513 Event event;
514 event.Type = Event::EVENT_MOUSE_ENTERED;
515 pushEvent(event);
516 }
517 break;
518
519 case LeaveNotify:
520 {
521 Event event;
522 event.Type = Event::EVENT_MOUSE_LEFT;
523 pushEvent(event);
524 }
525 break;
526
527 case MotionNotify:
528 {
529 Event event;
530 event.Type = Event::EVENT_MOUSE_MOVED;
531 event.MouseMove.X = xEvent.xmotion.x;
532 event.MouseMove.Y = xEvent.xmotion.y;
533 pushEvent(event);
534 }
535 break;
536
537 case ConfigureNotify:
538 {
539 if (xEvent.xconfigure.width != mWidth || xEvent.xconfigure.height != mHeight)
540 {
541 Event event;
542 event.Type = Event::EVENT_RESIZED;
543 event.Size.Width = xEvent.xconfigure.width;
544 event.Size.Height = xEvent.xconfigure.height;
545 pushEvent(event);
546 }
547 if (xEvent.xconfigure.x != mX || xEvent.xconfigure.y != mY)
548 {
549 // Sometimes, the window manager reparents our window (for example
550 // when resizing) then the X and Y coordinates will be with respect to
551 // the new parent and not what the user wants to know. Use
552 // XTranslateCoordinates to get the coordinates on the screen.
553 int screen = DefaultScreen(mDisplay);
554 Window root = RootWindow(mDisplay, screen);
555
556 int x, y;
557 Window child;
558 XTranslateCoordinates(mDisplay, mWindow, root, 0, 0, &x, &y, &child);
559
560 if (x != mX || y != mY)
561 {
562 Event event;
563 event.Type = Event::EVENT_MOVED;
564 event.Move.X = x;
565 event.Move.Y = y;
566 pushEvent(event);
567 }
568 }
569 }
570 break;
571
572 case FocusIn:
573 if (xEvent.xfocus.mode == NotifyNormal || xEvent.xfocus.mode == NotifyWhileGrabbed)
574 {
575 Event event;
576 event.Type = Event::EVENT_GAINED_FOCUS;
577 pushEvent(event);
578 }
579 break;
580
581 case FocusOut:
582 if (xEvent.xfocus.mode == NotifyNormal || xEvent.xfocus.mode == NotifyWhileGrabbed)
583 {
584 Event event;
585 event.Type = Event::EVENT_LOST_FOCUS;
586 pushEvent(event);
587 }
588 break;
589
590 case DestroyNotify:
591 // We already received WM_DELETE_WINDOW
592 break;
593
594 case ClientMessage:
595 if (xEvent.xclient.message_type == WM_PROTOCOLS &&
596 static_cast<Atom>(xEvent.xclient.data.l[0]) == WM_DELETE_WINDOW)
597 {
598 Event event;
599 event.Type = Event::EVENT_CLOSED;
600 pushEvent(event);
601 }
602 else if (xEvent.xclient.message_type == TEST_EVENT)
603 {
604 Event event;
605 event.Type = Event::EVENT_TEST;
606 pushEvent(event);
607 }
608 break;
609 }
610 }
611