entry_windows.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673
  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_WINDOWS
  7. #include <bgfxplatform.h>
  8. #include <bx/uint32_t.h>
  9. #include <bx/thread.h>
  10. #include <windowsx.h>
  11. #define WM_USER_SET_WINDOW_SIZE (WM_USER+0)
  12. #define WM_USER_TOGGLE_WINDOW_FRAME (WM_USER+1)
  13. #define WM_USER_MOUSE_LOCK (WM_USER+2)
  14. namespace entry
  15. {
  16. struct TranslateKeyModifiers
  17. {
  18. int m_vk;
  19. Modifier::Enum m_modifier;
  20. };
  21. static const TranslateKeyModifiers s_translateKeyModifiers[8] =
  22. {
  23. { VK_LMENU, Modifier::LeftAlt },
  24. { VK_RMENU, Modifier::RightAlt },
  25. { VK_LCONTROL, Modifier::LeftCtrl },
  26. { VK_RCONTROL, Modifier::RightCtrl },
  27. { VK_LSHIFT, Modifier::LeftShift },
  28. { VK_RSHIFT, Modifier::RightShift },
  29. { VK_LWIN, Modifier::LeftMeta },
  30. { VK_RWIN, Modifier::RightMeta },
  31. };
  32. static uint8_t translateKeyModifiers()
  33. {
  34. uint8_t modifiers = 0;
  35. for (uint32_t ii = 0; ii < BX_COUNTOF(s_translateKeyModifiers); ++ii)
  36. {
  37. const TranslateKeyModifiers& tkm = s_translateKeyModifiers[ii];
  38. modifiers |= 0 > GetKeyState(tkm.m_vk) ? tkm.m_modifier : Modifier::None;
  39. }
  40. return modifiers;
  41. }
  42. static uint8_t s_translateKey[256];
  43. static Key::Enum translateKey(WPARAM _wparam)
  44. {
  45. return (Key::Enum)s_translateKey[_wparam&0xff];
  46. }
  47. struct MainThreadEntry
  48. {
  49. int m_argc;
  50. char** m_argv;
  51. static int32_t threadFunc(void* _userData);
  52. };
  53. struct Context
  54. {
  55. Context()
  56. : m_frame(true)
  57. , m_mouseLock(false)
  58. , m_init(false)
  59. , m_exit(false)
  60. , m_mz(0)
  61. {
  62. memset(s_translateKey, 0, sizeof(s_translateKey) );
  63. s_translateKey[VK_ESCAPE] = Key::Esc;
  64. s_translateKey[VK_RETURN] = Key::Return;
  65. s_translateKey[VK_TAB] = Key::Tab;
  66. s_translateKey[VK_BACK] = Key::Backspace;
  67. s_translateKey[VK_SPACE] = Key::Space;
  68. s_translateKey[VK_UP] = Key::Up;
  69. s_translateKey[VK_DOWN] = Key::Down;
  70. s_translateKey[VK_LEFT] = Key::Left;
  71. s_translateKey[VK_RIGHT] = Key::Right;
  72. s_translateKey[VK_PRIOR] = Key::PageUp;
  73. s_translateKey[VK_NEXT] = Key::PageUp;
  74. s_translateKey[VK_HOME] = Key::Home;
  75. s_translateKey[VK_END] = Key::End;
  76. s_translateKey[VK_SNAPSHOT] = Key::Print;
  77. s_translateKey[VK_OEM_PLUS] = Key::Plus;
  78. s_translateKey[VK_OEM_MINUS] = Key::Minus;
  79. s_translateKey[VK_F1] = Key::F1;
  80. s_translateKey[VK_F2] = Key::F2;
  81. s_translateKey[VK_F3] = Key::F3;
  82. s_translateKey[VK_F4] = Key::F4;
  83. s_translateKey[VK_F5] = Key::F5;
  84. s_translateKey[VK_F6] = Key::F6;
  85. s_translateKey[VK_F7] = Key::F7;
  86. s_translateKey[VK_F8] = Key::F8;
  87. s_translateKey[VK_F9] = Key::F9;
  88. s_translateKey[VK_F10] = Key::F10;
  89. s_translateKey[VK_F11] = Key::F11;
  90. s_translateKey[VK_F12] = Key::F12;
  91. s_translateKey[VK_NUMPAD0] = Key::NumPad0;
  92. s_translateKey[VK_NUMPAD1] = Key::NumPad1;
  93. s_translateKey[VK_NUMPAD2] = Key::NumPad2;
  94. s_translateKey[VK_NUMPAD3] = Key::NumPad3;
  95. s_translateKey[VK_NUMPAD4] = Key::NumPad4;
  96. s_translateKey[VK_NUMPAD5] = Key::NumPad5;
  97. s_translateKey[VK_NUMPAD6] = Key::NumPad6;
  98. s_translateKey[VK_NUMPAD7] = Key::NumPad7;
  99. s_translateKey[VK_NUMPAD8] = Key::NumPad8;
  100. s_translateKey[VK_NUMPAD9] = Key::NumPad9;
  101. s_translateKey['0'] = Key::Key0;
  102. s_translateKey['1'] = Key::Key1;
  103. s_translateKey['2'] = Key::Key2;
  104. s_translateKey['3'] = Key::Key3;
  105. s_translateKey['4'] = Key::Key4;
  106. s_translateKey['5'] = Key::Key5;
  107. s_translateKey['6'] = Key::Key6;
  108. s_translateKey['7'] = Key::Key7;
  109. s_translateKey['8'] = Key::Key8;
  110. s_translateKey['9'] = Key::Key9;
  111. s_translateKey['A'] = Key::KeyA;
  112. s_translateKey['B'] = Key::KeyB;
  113. s_translateKey['C'] = Key::KeyC;
  114. s_translateKey['D'] = Key::KeyD;
  115. s_translateKey['E'] = Key::KeyE;
  116. s_translateKey['F'] = Key::KeyF;
  117. s_translateKey['G'] = Key::KeyG;
  118. s_translateKey['H'] = Key::KeyH;
  119. s_translateKey['I'] = Key::KeyI;
  120. s_translateKey['J'] = Key::KeyJ;
  121. s_translateKey['K'] = Key::KeyK;
  122. s_translateKey['L'] = Key::KeyL;
  123. s_translateKey['M'] = Key::KeyM;
  124. s_translateKey['N'] = Key::KeyN;
  125. s_translateKey['O'] = Key::KeyO;
  126. s_translateKey['P'] = Key::KeyP;
  127. s_translateKey['Q'] = Key::KeyQ;
  128. s_translateKey['R'] = Key::KeyR;
  129. s_translateKey['S'] = Key::KeyS;
  130. s_translateKey['T'] = Key::KeyT;
  131. s_translateKey['U'] = Key::KeyU;
  132. s_translateKey['V'] = Key::KeyV;
  133. s_translateKey['W'] = Key::KeyW;
  134. s_translateKey['X'] = Key::KeyX;
  135. s_translateKey['Y'] = Key::KeyY;
  136. s_translateKey['Z'] = Key::KeyZ;
  137. }
  138. int32_t run(int _argc, char** _argv)
  139. {
  140. SetDllDirectory(".");
  141. HINSTANCE instance = (HINSTANCE)GetModuleHandle(NULL);
  142. WNDCLASSEX wnd;
  143. memset(&wnd, 0, sizeof(wnd) );
  144. wnd.cbSize = sizeof(wnd);
  145. wnd.lpfnWndProc = DefWindowProc;
  146. wnd.hInstance = instance;
  147. wnd.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  148. wnd.hCursor = LoadCursor(NULL, IDC_ARROW);
  149. wnd.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH);
  150. wnd.lpszClassName = "bgfx_letterbox";
  151. wnd.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
  152. RegisterClassExA(&wnd);
  153. memset(&wnd, 0, sizeof(wnd) );
  154. wnd.cbSize = sizeof(wnd);
  155. wnd.style = CS_HREDRAW | CS_VREDRAW;
  156. wnd.lpfnWndProc = wndProc;
  157. wnd.hInstance = instance;
  158. wnd.hIcon = LoadIcon(NULL, IDI_APPLICATION);
  159. wnd.hCursor = LoadCursor(NULL, IDC_ARROW);
  160. wnd.lpszClassName = "bgfx";
  161. wnd.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
  162. RegisterClassExA(&wnd);
  163. HWND hwnd = CreateWindowA("bgfx_letterbox"
  164. , "BGFX"
  165. , WS_POPUP|WS_SYSMENU
  166. , -32000
  167. , -32000
  168. , 0
  169. , 0
  170. , NULL
  171. , NULL
  172. , instance
  173. , 0
  174. );
  175. m_hwnd = CreateWindowA("bgfx"
  176. , "BGFX"
  177. , WS_OVERLAPPEDWINDOW|WS_VISIBLE
  178. , 0
  179. , 0
  180. , ENTRY_DEFAULT_WIDTH
  181. , ENTRY_DEFAULT_HEIGHT
  182. , hwnd
  183. , NULL
  184. , instance
  185. , 0
  186. );
  187. bgfx::winSetHwnd(m_hwnd);
  188. adjust(ENTRY_DEFAULT_WIDTH, ENTRY_DEFAULT_HEIGHT, true);
  189. m_width = ENTRY_DEFAULT_WIDTH;
  190. m_height = ENTRY_DEFAULT_HEIGHT;
  191. m_oldWidth = ENTRY_DEFAULT_WIDTH;
  192. m_oldHeight = ENTRY_DEFAULT_HEIGHT;
  193. MainThreadEntry mte;
  194. mte.m_argc = _argc;
  195. mte.m_argv = _argv;
  196. bx::Thread thread;
  197. thread.init(mte.threadFunc, &mte);
  198. m_init = true;
  199. m_eventQueue.postSizeEvent(m_width, m_height);
  200. MSG msg;
  201. msg.message = WM_NULL;
  202. while (!m_exit)
  203. {
  204. WaitMessage();
  205. while (0 != PeekMessage(&msg, NULL, 0U, 0U, PM_REMOVE) )
  206. {
  207. TranslateMessage(&msg);
  208. DispatchMessage(&msg);
  209. }
  210. }
  211. thread.shutdown();
  212. DestroyWindow(m_hwnd);
  213. DestroyWindow(hwnd);
  214. return 0;
  215. }
  216. LRESULT process(HWND _hwnd, UINT _id, WPARAM _wparam, LPARAM _lparam)
  217. {
  218. if (m_init)
  219. {
  220. switch (_id)
  221. {
  222. case WM_USER_SET_WINDOW_SIZE:
  223. {
  224. uint32_t width = GET_X_LPARAM(_lparam);
  225. uint32_t height = GET_Y_LPARAM(_lparam);
  226. adjust(width, height, true);
  227. }
  228. break;
  229. case WM_USER_TOGGLE_WINDOW_FRAME:
  230. {
  231. if (m_frame)
  232. {
  233. m_oldWidth = m_width;
  234. m_oldHeight = m_height;
  235. }
  236. adjust(m_oldWidth, m_oldHeight, !m_frame);
  237. }
  238. break;
  239. case WM_USER_MOUSE_LOCK:
  240. setMouseLock(!!_lparam);
  241. break;
  242. case WM_DESTROY:
  243. break;
  244. case WM_QUIT:
  245. case WM_CLOSE:
  246. m_exit = true;
  247. m_eventQueue.postExitEvent();
  248. break;
  249. case WM_SIZING:
  250. {
  251. RECT& rect = *(RECT*)_lparam;
  252. uint32_t width = rect.right - rect.left - m_frameWidth;
  253. uint32_t height = rect.bottom - rect.top - m_frameHeight;
  254. //Recalculate size according to aspect ratio
  255. switch (_wparam)
  256. {
  257. case WMSZ_LEFT:
  258. case WMSZ_RIGHT:
  259. {
  260. float aspectRatio = 1.0f/m_aspectRatio;
  261. width = bx::uint32_max(ENTRY_DEFAULT_WIDTH/4, width);
  262. height = uint32_t(float(width)*aspectRatio);
  263. }
  264. break;
  265. default:
  266. {
  267. float aspectRatio = m_aspectRatio;
  268. height = bx::uint32_max(ENTRY_DEFAULT_HEIGHT/4, height);
  269. width = uint32_t(float(height)*aspectRatio);
  270. }
  271. break;
  272. }
  273. //Recalculate position using different anchor points
  274. switch(_wparam)
  275. {
  276. case WMSZ_LEFT:
  277. case WMSZ_TOPLEFT:
  278. case WMSZ_BOTTOMLEFT:
  279. rect.left = rect.right - width - m_frameWidth;
  280. rect.bottom = rect.top + height + m_frameHeight;
  281. break;
  282. default:
  283. rect.right = rect.left + width + m_frameWidth;
  284. rect.bottom = rect.top + height + m_frameHeight;
  285. break;
  286. }
  287. m_eventQueue.postSizeEvent(m_width, m_height);
  288. }
  289. return 0;
  290. case WM_SIZE:
  291. {
  292. uint32_t width = GET_X_LPARAM(_lparam);
  293. uint32_t height = GET_Y_LPARAM(_lparam);
  294. m_width = width;
  295. m_height = height;
  296. m_eventQueue.postSizeEvent(m_width, m_height);
  297. }
  298. break;
  299. case WM_SYSCOMMAND:
  300. switch (_wparam)
  301. {
  302. case SC_MINIMIZE:
  303. case SC_RESTORE:
  304. {
  305. HWND parent = GetWindow(_hwnd, GW_OWNER);
  306. if (NULL != parent)
  307. {
  308. PostMessage(parent, _id, _wparam, _lparam);
  309. }
  310. }
  311. }
  312. break;
  313. case WM_MOUSEMOVE:
  314. {
  315. int32_t mx = GET_X_LPARAM(_lparam);
  316. int32_t my = GET_Y_LPARAM(_lparam);
  317. if (m_mouseLock)
  318. {
  319. mx -= m_mx;
  320. my -= m_my;
  321. if (0 == mx
  322. && 0 == my)
  323. {
  324. break;
  325. }
  326. setMousePos(m_mx, m_my);
  327. }
  328. m_eventQueue.postMouseEvent(mx, my, m_mz);
  329. }
  330. break;
  331. case WM_MOUSEWHEEL:
  332. {
  333. POINT pt = { GET_X_LPARAM(_lparam), GET_Y_LPARAM(_lparam) };
  334. ScreenToClient(m_hwnd, &pt);
  335. int32_t mx = pt.x;
  336. int32_t my = pt.y;
  337. m_mz += GET_WHEEL_DELTA_WPARAM(_wparam);
  338. m_eventQueue.postMouseEvent(mx, my, m_mz);
  339. }
  340. break;
  341. case WM_LBUTTONDOWN:
  342. case WM_LBUTTONUP:
  343. case WM_LBUTTONDBLCLK:
  344. {
  345. int32_t mx = GET_X_LPARAM(_lparam);
  346. int32_t my = GET_Y_LPARAM(_lparam);
  347. m_eventQueue.postMouseEvent(mx, my, m_mz, MouseButton::Left, _id == WM_LBUTTONDOWN);
  348. }
  349. break;
  350. case WM_MBUTTONDOWN:
  351. case WM_MBUTTONUP:
  352. case WM_MBUTTONDBLCLK:
  353. {
  354. int32_t mx = GET_X_LPARAM(_lparam);
  355. int32_t my = GET_Y_LPARAM(_lparam);
  356. m_eventQueue.postMouseEvent(mx, my, m_mz, MouseButton::Middle, _id == WM_MBUTTONDOWN);
  357. }
  358. break;
  359. case WM_RBUTTONUP:
  360. case WM_RBUTTONDOWN:
  361. case WM_RBUTTONDBLCLK:
  362. {
  363. int32_t mx = GET_X_LPARAM(_lparam);
  364. int32_t my = GET_Y_LPARAM(_lparam);
  365. m_eventQueue.postMouseEvent(mx, my, m_mz, MouseButton::Right, _id == WM_RBUTTONDOWN);
  366. }
  367. break;
  368. case WM_KEYDOWN:
  369. case WM_SYSKEYDOWN:
  370. case WM_KEYUP:
  371. case WM_SYSKEYUP:
  372. {
  373. uint8_t modifiers = translateKeyModifiers();
  374. Key::Enum key = translateKey(_wparam);
  375. if (Key::Print == key
  376. && 0x3 == ( (uint32_t)(_lparam)>>30) )
  377. {
  378. // VK_SNAPSHOT doesn't generate keydown event. Fire on down event when previous
  379. // key state bit is set to 1 and transition state bit is set to 1.
  380. //
  381. // http://msdn.microsoft.com/en-us/library/windows/desktop/ms646280%28v=vs.85%29.aspx
  382. m_eventQueue.postKeyEvent(key, modifiers, true);
  383. }
  384. m_eventQueue.postKeyEvent(key, modifiers, _id == WM_KEYDOWN || _id == WM_SYSKEYDOWN);
  385. }
  386. break;
  387. default:
  388. break;
  389. }
  390. }
  391. return DefWindowProc(_hwnd, _id, _wparam, _lparam);
  392. }
  393. void adjust(uint32_t _width, uint32_t _height, bool _windowFrame)
  394. {
  395. m_width = _width;
  396. m_height = _height;
  397. m_aspectRatio = float(_width)/float(_height);
  398. ShowWindow(m_hwnd, SW_SHOWNORMAL);
  399. RECT rect;
  400. RECT newrect = {0, 0, (LONG)_width, (LONG)_height};
  401. DWORD style = WS_POPUP|WS_SYSMENU;
  402. if (m_frame)
  403. {
  404. GetWindowRect(m_hwnd, &m_rect);
  405. m_style = GetWindowLong(m_hwnd, GWL_STYLE);
  406. }
  407. if (_windowFrame)
  408. {
  409. rect = m_rect;
  410. style = m_style;
  411. }
  412. else
  413. {
  414. #if defined(__MINGW32__)
  415. rect = m_rect;
  416. style = m_style;
  417. #else
  418. HMONITOR monitor = MonitorFromWindow(m_hwnd, MONITOR_DEFAULTTONEAREST);
  419. MONITORINFO mi;
  420. mi.cbSize = sizeof(mi);
  421. GetMonitorInfo(monitor, &mi);
  422. newrect = mi.rcMonitor;
  423. rect = mi.rcMonitor;
  424. #endif // !defined(__MINGW__)
  425. }
  426. SetWindowLong(m_hwnd, GWL_STYLE, style);
  427. uint32_t prewidth = newrect.right - newrect.left;
  428. uint32_t preheight = newrect.bottom - newrect.top;
  429. AdjustWindowRect(&newrect, style, FALSE);
  430. m_frameWidth = (newrect.right - newrect.left) - prewidth;
  431. m_frameHeight = (newrect.bottom - newrect.top) - preheight;
  432. UpdateWindow(m_hwnd);
  433. if (rect.left == -32000
  434. || rect.top == -32000)
  435. {
  436. rect.left = 0;
  437. rect.top = 0;
  438. }
  439. int32_t left = rect.left;
  440. int32_t top = rect.top;
  441. int32_t width = (newrect.right-newrect.left);
  442. int32_t height = (newrect.bottom-newrect.top);
  443. if (!_windowFrame)
  444. {
  445. float aspectRatio = 1.0f/m_aspectRatio;
  446. width = bx::uint32_max(ENTRY_DEFAULT_WIDTH/4, width);
  447. height = uint32_t(float(width)*aspectRatio);
  448. left = newrect.left+(newrect.right-newrect.left-width)/2;
  449. top = newrect.top+(newrect.bottom-newrect.top-height)/2;
  450. }
  451. HWND parent = GetWindow(m_hwnd, GW_OWNER);
  452. if (NULL != parent)
  453. {
  454. if (_windowFrame)
  455. {
  456. SetWindowPos(parent
  457. , HWND_TOP
  458. , -32000
  459. , -32000
  460. , 0
  461. , 0
  462. , SWP_SHOWWINDOW
  463. );
  464. }
  465. else
  466. {
  467. SetWindowPos(parent
  468. , HWND_TOP
  469. , newrect.left
  470. , newrect.top
  471. , newrect.right-newrect.left
  472. , newrect.bottom-newrect.top
  473. , SWP_SHOWWINDOW
  474. );
  475. }
  476. }
  477. SetWindowPos(m_hwnd
  478. , HWND_TOP
  479. , left
  480. , top
  481. , width
  482. , height
  483. , SWP_SHOWWINDOW
  484. );
  485. ShowWindow(m_hwnd, SW_RESTORE);
  486. m_frame = _windowFrame;
  487. }
  488. void setMousePos(int32_t _mx, int32_t _my)
  489. {
  490. POINT pt = { _mx, _my };
  491. ClientToScreen(m_hwnd, &pt);
  492. SetCursorPos(pt.x, pt.y);
  493. }
  494. void setMouseLock(bool _lock)
  495. {
  496. if (_lock != m_mouseLock)
  497. {
  498. if (_lock)
  499. {
  500. m_mx = m_width/2;
  501. m_my = m_height/2;
  502. ShowCursor(false);
  503. setMousePos(m_mx, m_my);
  504. }
  505. else
  506. {
  507. setMousePos(m_mx, m_my);
  508. ShowCursor(true);
  509. }
  510. m_mouseLock = _lock;
  511. }
  512. }
  513. void setWindowTitle(const char* _title)
  514. {
  515. SetWindowText(m_hwnd, _title);
  516. }
  517. static LRESULT CALLBACK wndProc(HWND _hwnd, UINT _id, WPARAM _wparam, LPARAM _lparam);
  518. EventQueue m_eventQueue;
  519. HWND m_hwnd;
  520. RECT m_rect;
  521. DWORD m_style;
  522. uint32_t m_width;
  523. uint32_t m_height;
  524. uint32_t m_oldWidth;
  525. uint32_t m_oldHeight;
  526. uint32_t m_frameWidth;
  527. uint32_t m_frameHeight;
  528. float m_aspectRatio;
  529. int32_t m_mx;
  530. int32_t m_my;
  531. int32_t m_mz;
  532. bool m_frame;
  533. bool m_mouseLock;
  534. bool m_init;
  535. bool m_exit;
  536. };
  537. static Context s_ctx;
  538. LRESULT CALLBACK Context::wndProc(HWND _hwnd, UINT _id, WPARAM _wparam, LPARAM _lparam)
  539. {
  540. return s_ctx.process(_hwnd, _id, _wparam, _lparam);
  541. }
  542. const Event* poll()
  543. {
  544. return s_ctx.m_eventQueue.poll();
  545. }
  546. void release(const Event* _event)
  547. {
  548. s_ctx.m_eventQueue.release(_event);
  549. }
  550. void setWindowSize(uint32_t _width, uint32_t _height)
  551. {
  552. PostMessage(s_ctx.m_hwnd, WM_USER_SET_WINDOW_SIZE, 0, (_height<<16) | (_width&0xffff) );
  553. }
  554. void setWindowTitle(const char* _title)
  555. {
  556. s_ctx.setWindowTitle(_title);
  557. }
  558. void toggleWindowFrame()
  559. {
  560. PostMessage(s_ctx.m_hwnd, WM_USER_TOGGLE_WINDOW_FRAME, 0, 0);
  561. }
  562. void setMouseLock(bool _lock)
  563. {
  564. PostMessage(s_ctx.m_hwnd, WM_USER_MOUSE_LOCK, 0, _lock);
  565. }
  566. int32_t MainThreadEntry::threadFunc(void* _userData)
  567. {
  568. MainThreadEntry* self = (MainThreadEntry*)_userData;
  569. int32_t result = main(self->m_argc, self->m_argv);
  570. PostMessage(s_ctx.m_hwnd, WM_QUIT, 0, 0);
  571. return result;
  572. }
  573. } // namespace entry
  574. int main(int _argc, char** _argv)
  575. {
  576. using namespace entry;
  577. return s_ctx.run(_argc, _argv);
  578. }
  579. #endif // BX_PLATFORM_WINDOWS