entry_linux.cpp 9.5 KB

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