entry_linux.cpp 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346
  1. /*
  2. * Copyright 2011-2014 Branimir Karadzic. All rights reserved.
  3. * License: http://www.opensource.org/licenses/BSD-2-Clause
  4. */
  5. #include "entry_p.h"
  6. #if ENTRY_CONFIG_USE_NATIVE && BX_PLATFORM_LINUX
  7. #define XK_MISCELLANY
  8. #define XK_LATIN1
  9. #include <X11/keysymdef.h>
  10. #include <bgfxplatform.h> // will include X11 which #defines None... Don't mess with order of includes.
  11. #undef None
  12. #include <bx/thread.h>
  13. #include <bx/os.h>
  14. #include <string.h> // memset
  15. namespace entry
  16. {
  17. static uint8_t s_translateKey[512];
  18. static void initTranslateKey(uint16_t _xk, Key::Enum _key)
  19. {
  20. _xk += 256;
  21. BX_CHECK(_xk < BX_COUNTOF(s_translateKey), "Out of bounds %d.", _xk);
  22. s_translateKey[_xk&0x1ff] = (uint8_t)_key;
  23. }
  24. Key::Enum fromXk(uint16_t _xk)
  25. {
  26. _xk += 256;
  27. return 512 > _xk ? (Key::Enum)s_translateKey[_xk] : Key::None;
  28. }
  29. struct MainThreadEntry
  30. {
  31. int m_argc;
  32. char** m_argv;
  33. static int32_t threadFunc(void* _userData);
  34. };
  35. struct Context
  36. {
  37. Context()
  38. : m_modifiers(Modifier::None)
  39. , m_exit(false)
  40. {
  41. memset(s_translateKey, 0, sizeof(s_translateKey) );
  42. initTranslateKey(XK_Escape, Key::Esc);
  43. initTranslateKey(XK_Return, Key::Return);
  44. initTranslateKey(XK_Tab, Key::Tab);
  45. initTranslateKey(XK_BackSpace, Key::Backspace);
  46. initTranslateKey(XK_space, Key::Space);
  47. initTranslateKey(XK_Up, Key::Up);
  48. initTranslateKey(XK_Down, Key::Down);
  49. initTranslateKey(XK_Left, Key::Left);
  50. initTranslateKey(XK_Right, Key::Right);
  51. initTranslateKey(XK_Page_Up, Key::PageUp);
  52. initTranslateKey(XK_Page_Down, Key::PageUp);
  53. initTranslateKey(XK_Home, Key::Home);
  54. initTranslateKey(XK_KP_End, Key::End);
  55. initTranslateKey(XK_Print, Key::Print);
  56. initTranslateKey(XK_equal, Key::Plus);
  57. initTranslateKey(XK_minus, Key::Minus);
  58. initTranslateKey(XK_F1, Key::F1);
  59. initTranslateKey(XK_F2, Key::F2);
  60. initTranslateKey(XK_F3, Key::F3);
  61. initTranslateKey(XK_F4, Key::F4);
  62. initTranslateKey(XK_F5, Key::F5);
  63. initTranslateKey(XK_F6, Key::F6);
  64. initTranslateKey(XK_F7, Key::F7);
  65. initTranslateKey(XK_F8, Key::F8);
  66. initTranslateKey(XK_F9, Key::F9);
  67. initTranslateKey(XK_F10, Key::F10);
  68. initTranslateKey(XK_F11, Key::F11);
  69. initTranslateKey(XK_F12, Key::F12);
  70. initTranslateKey(XK_KP_Insert, Key::NumPad0);
  71. initTranslateKey(XK_KP_End, Key::NumPad1);
  72. initTranslateKey(XK_KP_Down, Key::NumPad2);
  73. initTranslateKey(XK_KP_Page_Down, Key::NumPad3);
  74. initTranslateKey(XK_KP_Left, Key::NumPad4);
  75. initTranslateKey(XK_KP_Begin, Key::NumPad5);
  76. initTranslateKey(XK_KP_Right, Key::NumPad6);
  77. initTranslateKey(XK_KP_Home, Key::NumPad7);
  78. initTranslateKey(XK_KP_Up, Key::NumPad8);
  79. initTranslateKey(XK_KP_Page_Up, Key::NumPad9);
  80. initTranslateKey('0', Key::Key0);
  81. initTranslateKey('1', Key::Key1);
  82. initTranslateKey('2', Key::Key2);
  83. initTranslateKey('3', Key::Key3);
  84. initTranslateKey('4', Key::Key4);
  85. initTranslateKey('5', Key::Key5);
  86. initTranslateKey('6', Key::Key6);
  87. initTranslateKey('7', Key::Key7);
  88. initTranslateKey('8', Key::Key8);
  89. initTranslateKey('9', Key::Key9);
  90. initTranslateKey('a', Key::KeyA);
  91. initTranslateKey('b', Key::KeyB);
  92. initTranslateKey('c', Key::KeyC);
  93. initTranslateKey('d', Key::KeyD);
  94. initTranslateKey('e', Key::KeyE);
  95. initTranslateKey('f', Key::KeyF);
  96. initTranslateKey('g', Key::KeyG);
  97. initTranslateKey('h', Key::KeyH);
  98. initTranslateKey('i', Key::KeyI);
  99. initTranslateKey('j', Key::KeyJ);
  100. initTranslateKey('k', Key::KeyK);
  101. initTranslateKey('l', Key::KeyL);
  102. initTranslateKey('m', Key::KeyM);
  103. initTranslateKey('n', Key::KeyN);
  104. initTranslateKey('o', Key::KeyO);
  105. initTranslateKey('p', Key::KeyP);
  106. initTranslateKey('q', Key::KeyQ);
  107. initTranslateKey('r', Key::KeyR);
  108. initTranslateKey('s', Key::KeyS);
  109. initTranslateKey('t', Key::KeyT);
  110. initTranslateKey('u', Key::KeyU);
  111. initTranslateKey('v', Key::KeyV);
  112. initTranslateKey('w', Key::KeyW);
  113. initTranslateKey('x', Key::KeyX);
  114. initTranslateKey('y', Key::KeyY);
  115. initTranslateKey('z', Key::KeyZ);
  116. }
  117. int32_t run(int _argc, char** _argv)
  118. {
  119. XInitThreads();
  120. m_display = XOpenDisplay(0);
  121. int32_t screen = DefaultScreen(m_display);
  122. int32_t depth = DefaultDepth(m_display, screen);
  123. Visual* visual = DefaultVisual(m_display, screen);
  124. Window root = RootWindow(m_display, screen);
  125. XSetWindowAttributes windowAttrs;
  126. memset(&windowAttrs, 0, sizeof(windowAttrs) );
  127. windowAttrs.background_pixmap = 0;
  128. windowAttrs.border_pixel = 0;
  129. windowAttrs.event_mask = 0
  130. | ButtonPressMask
  131. | ButtonReleaseMask
  132. | ExposureMask
  133. | KeyPressMask
  134. | KeyReleaseMask
  135. | PointerMotionMask
  136. | ResizeRedirectMask
  137. | StructureNotifyMask
  138. ;
  139. m_window = XCreateWindow(m_display
  140. , root
  141. , 0, 0
  142. , ENTRY_DEFAULT_WIDTH, ENTRY_DEFAULT_HEIGHT, 0, depth
  143. , InputOutput
  144. , visual
  145. , CWBorderPixel|CWEventMask
  146. , &windowAttrs
  147. );
  148. const char *wmDeleteWindowName = "WM_DELETE_WINDOW";
  149. Atom wmDeleteWindow;
  150. XInternAtoms(m_display, (char **)&wmDeleteWindowName, 1, False, &wmDeleteWindow);
  151. XSetWMProtocols(m_display, m_window, &wmDeleteWindow, 1);
  152. XMapWindow(m_display, m_window);
  153. XStoreName(m_display, m_window, "BGFX");
  154. bgfx::x11SetDisplayWindow(m_display, m_window);
  155. MainThreadEntry mte;
  156. mte.m_argc = _argc;
  157. mte.m_argv = _argv;
  158. bx::Thread thread;
  159. thread.init(mte.threadFunc, &mte);
  160. while (!m_exit)
  161. {
  162. if (XPending(m_display) )
  163. {
  164. XEvent event;
  165. XNextEvent(m_display, &event);
  166. switch (event.type)
  167. {
  168. case Expose:
  169. break;
  170. case ConfigureNotify:
  171. break;
  172. case ClientMessage:
  173. if((Atom)event.xclient.data.l[0] == wmDeleteWindow)
  174. {
  175. m_eventQueue.postExitEvent();
  176. }
  177. break;
  178. case ButtonPress:
  179. case ButtonRelease:
  180. {
  181. const XButtonEvent& xbutton = event.xbutton;
  182. MouseButton::Enum mb;
  183. switch (xbutton.button)
  184. {
  185. case Button1: mb = MouseButton::Left; break;
  186. case Button2: mb = MouseButton::Middle; break;
  187. case Button3: mb = MouseButton::Right; break;
  188. default: mb = MouseButton::None; break;
  189. }
  190. if (MouseButton::None != mb)
  191. {
  192. m_eventQueue.postMouseEvent(xbutton.x
  193. , xbutton.y
  194. , mb
  195. , event.type == ButtonPress
  196. );
  197. }
  198. }
  199. break;
  200. case MotionNotify:
  201. {
  202. const XMotionEvent& xmotion = event.xmotion;
  203. m_eventQueue.postMouseEvent(xmotion.x
  204. , xmotion.y
  205. );
  206. }
  207. break;
  208. case KeyPress:
  209. case KeyRelease:
  210. {
  211. XKeyEvent& xkey = event.xkey;
  212. KeySym keysym = XLookupKeysym(&xkey, 0);
  213. switch (keysym)
  214. {
  215. case XK_Meta_L: setModifier(Modifier::LeftMeta, KeyPress == event.type); break;
  216. case XK_Meta_R: setModifier(Modifier::RightMeta, KeyPress == event.type); break;
  217. case XK_Control_L: setModifier(Modifier::LeftCtrl, KeyPress == event.type); break;
  218. case XK_Control_R: setModifier(Modifier::RightCtrl, KeyPress == event.type); break;
  219. case XK_Shift_L: setModifier(Modifier::LeftShift, KeyPress == event.type); break;
  220. case XK_Shift_R: setModifier(Modifier::RightShift, KeyPress == event.type); break;
  221. case XK_Alt_L: setModifier(Modifier::LeftAlt, KeyPress == event.type); break;
  222. case XK_Alt_R: setModifier(Modifier::RightAlt, KeyPress == event.type); break;
  223. default:
  224. {
  225. Key::Enum key = fromXk(keysym);
  226. if (Key::None != key)
  227. {
  228. m_eventQueue.postKeyEvent(key, m_modifiers, KeyPress == event.type);
  229. }
  230. }
  231. break;
  232. }
  233. }
  234. break;
  235. case ResizeRequest:
  236. {
  237. const XResizeRequestEvent& xresize = event.xresizerequest;
  238. XResizeWindow(m_display, m_window, xresize.width, xresize.height);
  239. }
  240. break;
  241. }
  242. }
  243. }
  244. thread.shutdown();
  245. XUnmapWindow(m_display, m_window);
  246. XDestroyWindow(m_display, m_window);
  247. return EXIT_SUCCESS;
  248. }
  249. void setModifier(Modifier::Enum _modifier, bool _set)
  250. {
  251. m_modifiers &= ~_modifier;
  252. m_modifiers |= _set ? _modifier : 0;
  253. }
  254. uint8_t m_modifiers;
  255. Display* m_display;
  256. Window m_window;
  257. bool m_exit;
  258. EventQueue m_eventQueue;
  259. };
  260. static Context s_ctx;
  261. int32_t MainThreadEntry::threadFunc(void* _userData)
  262. {
  263. MainThreadEntry* self = (MainThreadEntry*)_userData;
  264. int32_t result = main(self->m_argc, self->m_argv);
  265. s_ctx.m_exit = true;
  266. return result;
  267. }
  268. const Event* poll()
  269. {
  270. return s_ctx.m_eventQueue.poll();
  271. }
  272. void release(const Event* _event)
  273. {
  274. s_ctx.m_eventQueue.release(_event);
  275. }
  276. void setWindowSize(uint32_t _width, uint32_t _height)
  277. {
  278. XResizeRequestEvent ev;
  279. ev.type = ResizeRequest;
  280. ev.serial = 0;
  281. ev.send_event = true;
  282. ev.display = s_ctx.m_display;
  283. ev.window = s_ctx.m_window;
  284. ev.width = (int)_width;
  285. ev.height = (int)_height;
  286. XSendEvent(s_ctx.m_display, s_ctx.m_window, false, ResizeRedirectMask, (XEvent*)&ev);
  287. }
  288. void toggleWindowFrame()
  289. {
  290. }
  291. void setMouseLock(bool _lock)
  292. {
  293. BX_UNUSED(_lock);
  294. }
  295. } // namespace entry
  296. int main(int _argc, char** _argv)
  297. {
  298. using namespace entry;
  299. return s_ctx.run(_argc, _argv);
  300. }
  301. #endif // ENTRY_CONFIG_USE_NATIVE && BX_PLATFORM_LINUX