entry_linux.cpp 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  1. /*
  2. * Copyright 2011-2013 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. XMapWindow(m_display, m_window);
  149. XStoreName(m_display, m_window, "BGFX");
  150. bgfx::x11SetDisplayWindow(m_display, m_window);
  151. MainThreadEntry mte;
  152. mte.m_argc = _argc;
  153. mte.m_argv = _argv;
  154. bx::Thread thread;
  155. thread.init(mte.threadFunc, &mte);
  156. while (!m_exit)
  157. {
  158. if (XPending(m_display) )
  159. {
  160. XEvent event;
  161. XNextEvent(m_display, &event);
  162. switch (event.type)
  163. {
  164. case Expose:
  165. break;
  166. case ConfigureNotify:
  167. break;
  168. case ButtonPress:
  169. case ButtonRelease:
  170. {
  171. const XButtonEvent& xbutton = event.xbutton;
  172. MouseButton::Enum mb;
  173. switch (xbutton.button)
  174. {
  175. case Button1: mb = MouseButton::Left; break;
  176. case Button2: mb = MouseButton::Middle; break;
  177. case Button3: mb = MouseButton::Right; break;
  178. default: mb = MouseButton::None; break;
  179. }
  180. if (MouseButton::None != mb)
  181. {
  182. m_eventQueue.postMouseEvent(xbutton.x
  183. , xbutton.y
  184. , mb
  185. , event.type == ButtonPress
  186. );
  187. }
  188. }
  189. break;
  190. case MotionNotify:
  191. {
  192. const XMotionEvent& xmotion = event.xmotion;
  193. m_eventQueue.postMouseEvent(xmotion.x
  194. , xmotion.y
  195. );
  196. }
  197. break;
  198. case KeyPress:
  199. case KeyRelease:
  200. {
  201. XKeyEvent& xkey = event.xkey;
  202. KeySym keysym = XLookupKeysym(&xkey, 0);
  203. switch (keysym)
  204. {
  205. case XK_Meta_L: setModifier(Modifier::LeftMeta, KeyPress == event.type); break;
  206. case XK_Meta_R: setModifier(Modifier::RightMeta, KeyPress == event.type); break;
  207. case XK_Control_L: setModifier(Modifier::LeftCtrl, KeyPress == event.type); break;
  208. case XK_Control_R: setModifier(Modifier::RightCtrl, KeyPress == event.type); break;
  209. case XK_Shift_L: setModifier(Modifier::LeftShift, KeyPress == event.type); break;
  210. case XK_Shift_R: setModifier(Modifier::RightShift, KeyPress == event.type); break;
  211. case XK_Alt_L: setModifier(Modifier::LeftAlt, KeyPress == event.type); break;
  212. case XK_Alt_R: setModifier(Modifier::RightAlt, KeyPress == event.type); break;
  213. default:
  214. {
  215. Key::Enum key = fromXk(keysym);
  216. if (Key::None != key)
  217. {
  218. m_eventQueue.postKeyEvent(key, m_modifiers, KeyPress == event.type);
  219. }
  220. }
  221. break;
  222. }
  223. }
  224. break;
  225. case ResizeRequest:
  226. {
  227. const XResizeRequestEvent& xresize = event.xresizerequest;
  228. XResizeWindow(m_display, m_window, xresize.width, xresize.height);
  229. }
  230. break;
  231. }
  232. }
  233. }
  234. thread.shutdown();
  235. XUnmapWindow(m_display, m_window);
  236. XDestroyWindow(m_display, m_window);
  237. return EXIT_SUCCESS;
  238. }
  239. void setModifier(Modifier::Enum _modifier, bool _set)
  240. {
  241. m_modifiers &= ~_modifier;
  242. m_modifiers |= _set ? _modifier : 0;
  243. }
  244. uint8_t m_modifiers;
  245. Display* m_display;
  246. Window m_window;
  247. bool m_exit;
  248. EventQueue m_eventQueue;
  249. };
  250. static Context s_ctx;
  251. int32_t MainThreadEntry::threadFunc(void* _userData)
  252. {
  253. MainThreadEntry* self = (MainThreadEntry*)_userData;
  254. int32_t result = main(self->m_argc, self->m_argv);
  255. s_ctx.m_exit = true;
  256. return result;
  257. }
  258. const Event* poll()
  259. {
  260. return s_ctx.m_eventQueue.poll();
  261. }
  262. void release(const Event* _event)
  263. {
  264. s_ctx.m_eventQueue.release(_event);
  265. }
  266. void setWindowSize(uint32_t _width, uint32_t _height)
  267. {
  268. XResizeRequestEvent ev;
  269. ev.type = ResizeRequest;
  270. ev.serial = 0;
  271. ev.send_event = true;
  272. ev.display = s_ctx.m_display;
  273. ev.window = s_ctx.m_window;
  274. ev.width = (int)_width;
  275. ev.height = (int)_height;
  276. XSendEvent(s_ctx.m_display, s_ctx.m_window, false, ResizeRedirectMask, (XEvent*)&ev);
  277. }
  278. void toggleWindowFrame()
  279. {
  280. }
  281. void setMouseLock(bool _lock)
  282. {
  283. }
  284. } // namespace entry
  285. int main(int _argc, char** _argv)
  286. {
  287. using namespace entry;
  288. return s_ctx.run(_argc, _argv);
  289. }
  290. #endif // ENTRY_CONFIG_USE_NATIVE && BX_PLATFORM_LINUX