entry_linux.cpp 9.4 KB

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