main_windows.cpp 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  1. /*
  2. * Copyright (c) 2012-2014 Daniele Bartolini and individual contributors.
  3. * License: https://github.com/taylor001/crown/blob/master/LICENSE
  4. */
  5. #include "config.h"
  6. #if CROWN_PLATFORM_WINDOWS
  7. #include "os_event_queue.h"
  8. #include "os_window_windows.h"
  9. #include "thread.h"
  10. #include "crown.h"
  11. #include "command_line.h"
  12. #include "keyboard.h"
  13. #include "console_server.h"
  14. #include "bundle_compiler.h"
  15. #include "disk_filesystem.h"
  16. #include <bgfxplatform.h>
  17. #include <winsock2.h>
  18. #ifndef WIN32_LEAN_AND_MEAN
  19. #define WIN32_LEAN_AND_MEAN
  20. #endif
  21. #include <windowsx.h>
  22. namespace crown
  23. {
  24. static bool s_exit = false;
  25. struct MainThreadArgs
  26. {
  27. Filesystem* fs;
  28. ConfigSettings* cs;
  29. };
  30. int32_t func(void* data)
  31. {
  32. MainThreadArgs* args = (MainThreadArgs*)data;
  33. crown::init(*args->fs, *args->cs);
  34. crown::update();
  35. crown::shutdown();
  36. s_exit = true;
  37. return EXIT_SUCCESS;
  38. }
  39. static KeyboardButton::Enum win_translate_key(int32_t winkey)
  40. {
  41. switch (winkey)
  42. {
  43. case VK_BACK: return KeyboardButton::BACKSPACE;
  44. case VK_TAB: return KeyboardButton::TAB;
  45. case VK_SPACE: return KeyboardButton::SPACE;
  46. case VK_ESCAPE: return KeyboardButton::ESCAPE;
  47. case VK_RETURN: return KeyboardButton::ENTER;
  48. case VK_F1: return KeyboardButton::F1;
  49. case VK_F2: return KeyboardButton::F2;
  50. case VK_F3: return KeyboardButton::F3;
  51. case VK_F4: return KeyboardButton::F4;
  52. case VK_F5: return KeyboardButton::F5;
  53. case VK_F6: return KeyboardButton::F6;
  54. case VK_F7: return KeyboardButton::F7;
  55. case VK_F8: return KeyboardButton::F8;
  56. case VK_F9: return KeyboardButton::F9;
  57. case VK_F10: return KeyboardButton::F10;
  58. case VK_F11: return KeyboardButton::F11;
  59. case VK_F12: return KeyboardButton::F12;
  60. case VK_HOME: return KeyboardButton::HOME;
  61. case VK_LEFT: return KeyboardButton::LEFT;
  62. case VK_UP: return KeyboardButton::UP;
  63. case VK_RIGHT: return KeyboardButton::RIGHT;
  64. case VK_DOWN: return KeyboardButton::DOWN;
  65. case VK_PRIOR: return KeyboardButton::PAGE_UP;
  66. case VK_NEXT: return KeyboardButton::PAGE_DOWN;
  67. case VK_LSHIFT: return KeyboardButton::LSHIFT;
  68. case VK_RSHIFT: return KeyboardButton::RSHIFT;
  69. case VK_LCONTROL: return KeyboardButton::LCONTROL;
  70. case VK_RCONTROL: return KeyboardButton::RCONTROL;
  71. case VK_CAPITAL: return KeyboardButton::CAPS_LOCK;
  72. case VK_LMENU: return KeyboardButton::LALT;
  73. case VK_RMENU: return KeyboardButton::RALT;
  74. case VK_LWIN: return KeyboardButton::LSUPER;
  75. case VK_RWIN: return KeyboardButton::RSUPER;
  76. case VK_NUMPAD0: return KeyboardButton::KP_0;
  77. case VK_NUMPAD1: return KeyboardButton::KP_1;
  78. case VK_NUMPAD2: return KeyboardButton::KP_2;
  79. case VK_NUMPAD3: return KeyboardButton::KP_3;
  80. case VK_NUMPAD4: return KeyboardButton::KP_4;
  81. case VK_NUMPAD5: return KeyboardButton::KP_5;
  82. case VK_NUMPAD6: return KeyboardButton::KP_6;
  83. case VK_NUMPAD7: return KeyboardButton::KP_7;
  84. case VK_NUMPAD8: return KeyboardButton::KP_8;
  85. case VK_NUMPAD9: return KeyboardButton::KP_9;
  86. case '0': return KeyboardButton::NUM_0;
  87. case '1': return KeyboardButton::NUM_1;
  88. case '2': return KeyboardButton::NUM_2;
  89. case '3': return KeyboardButton::NUM_3;
  90. case '4': return KeyboardButton::NUM_4;
  91. case '5': return KeyboardButton::NUM_5;
  92. case '6': return KeyboardButton::NUM_6;
  93. case '7': return KeyboardButton::NUM_7;
  94. case '8': return KeyboardButton::NUM_8;
  95. case '9': return KeyboardButton::NUM_9;
  96. case 'A': return KeyboardButton::A;
  97. case 'B': return KeyboardButton::B;
  98. case 'C': return KeyboardButton::C;
  99. case 'D': return KeyboardButton::D;
  100. case 'E': return KeyboardButton::E;
  101. case 'F': return KeyboardButton::F;
  102. case 'G': return KeyboardButton::G;
  103. case 'H': return KeyboardButton::H;
  104. case 'I': return KeyboardButton::I;
  105. case 'J': return KeyboardButton::J;
  106. case 'K': return KeyboardButton::K;
  107. case 'L': return KeyboardButton::L;
  108. case 'M': return KeyboardButton::M;
  109. case 'N': return KeyboardButton::N;
  110. case 'O': return KeyboardButton::O;
  111. case 'P': return KeyboardButton::P;
  112. case 'Q': return KeyboardButton::Q;
  113. case 'R': return KeyboardButton::R;
  114. case 'S': return KeyboardButton::S;
  115. case 'T': return KeyboardButton::T;
  116. case 'U': return KeyboardButton::U;
  117. case 'V': return KeyboardButton::V;
  118. case 'W': return KeyboardButton::W;
  119. case 'X': return KeyboardButton::X;
  120. case 'Y': return KeyboardButton::Y;
  121. case 'Z': return KeyboardButton::Z;
  122. default: return KeyboardButton::NONE;
  123. }
  124. }
  125. struct WindowsDevice
  126. {
  127. WindowsDevice()
  128. : _hwnd(NULL)
  129. , _hdc(NULL)
  130. {
  131. }
  132. int32_t run(Filesystem* fs, ConfigSettings* cs)
  133. {
  134. HINSTANCE instance = (HINSTANCE)GetModuleHandle(NULL);
  135. WNDCLASSEX wnd;
  136. memset(&wnd, 0, sizeof(wnd));
  137. wnd.cbSize = sizeof(wnd);
  138. wnd.style = CS_HREDRAW | CS_VREDRAW;
  139. wnd.lpfnWndProc = window_proc;
  140. wnd.hInstance = instance;
  141. wnd.hIcon = LoadIcon(instance, IDI_APPLICATION);
  142. wnd.hCursor = LoadCursor(instance, IDC_ARROW);
  143. wnd.lpszClassName = "crown";
  144. wnd.hIconSm = LoadIcon(instance, IDI_APPLICATION);
  145. RegisterClassExA(&wnd);
  146. _hwnd = CreateWindowA(
  147. "crown",
  148. "Crown",
  149. WS_OVERLAPPEDWINDOW|WS_VISIBLE,
  150. 0,
  151. 0,
  152. cs->window_width,
  153. cs->window_height,
  154. 0,
  155. NULL,
  156. instance,
  157. 0);
  158. bgfx::winSetHwnd(_hwnd);
  159. // Start main thread
  160. MainThreadArgs mta;
  161. mta.fs = fs;
  162. mta.cs = cs;
  163. Thread main_thread;
  164. main_thread.start(func, &mta);
  165. MSG msg;
  166. msg.message = WM_NULL;
  167. while (!s_exit)
  168. {
  169. while (PeekMessage(&msg, NULL, 0U, 0U, PM_REMOVE) != 0)
  170. {
  171. TranslateMessage(&msg);
  172. DispatchMessage(&msg);
  173. }
  174. }
  175. main_thread.stop();
  176. DestroyWindow(_hwnd);
  177. return EXIT_SUCCESS;
  178. }
  179. LRESULT pump_events(HWND hwnd, UINT id, WPARAM wparam, LPARAM lparam)
  180. {
  181. switch (id)
  182. {
  183. case WM_DESTROY:
  184. {
  185. break;
  186. }
  187. case WM_QUIT:
  188. case WM_CLOSE:
  189. {
  190. _queue.push_exit_event(0);
  191. break;
  192. }
  193. case WM_SIZE:
  194. {
  195. uint32_t width = GET_X_LPARAM(lparam);
  196. uint32_t height = GET_Y_LPARAM(lparam);
  197. _queue.push_metrics_event(0, 0, width, height);
  198. break;
  199. }
  200. case WM_SYSCOMMAND:
  201. {
  202. switch (wparam)
  203. {
  204. case SC_MINIMIZE:
  205. case SC_RESTORE:
  206. {
  207. HWND parent = GetWindow(hwnd, GW_OWNER);
  208. if (NULL != parent)
  209. {
  210. PostMessage(parent, id, wparam, lparam);
  211. }
  212. }
  213. }
  214. break;
  215. }
  216. case WM_MOUSEWHEEL:
  217. {
  218. int32_t mx = GET_X_LPARAM(lparam);
  219. int32_t my = GET_Y_LPARAM(lparam);
  220. short delta = GET_WHEEL_DELTA_WPARAM(wparam);
  221. _queue.push_mouse_event(mx, my, (float)(delta/WHEEL_DELTA));
  222. break;
  223. }
  224. case WM_MOUSEMOVE:
  225. {
  226. int32_t mx = GET_X_LPARAM(lparam);
  227. int32_t my = GET_Y_LPARAM(lparam);
  228. _queue.push_mouse_event(mx, my);
  229. break;
  230. }
  231. case WM_LBUTTONDOWN:
  232. case WM_LBUTTONUP:
  233. {
  234. int32_t mx = GET_X_LPARAM(lparam);
  235. int32_t my = GET_Y_LPARAM(lparam);
  236. _queue.push_mouse_event(mx, my, MouseButton::LEFT, id == WM_LBUTTONDOWN);
  237. break;
  238. }
  239. case WM_RBUTTONUP:
  240. case WM_RBUTTONDOWN:
  241. {
  242. int32_t mx = GET_X_LPARAM(lparam);
  243. int32_t my = GET_Y_LPARAM(lparam);
  244. _queue.push_mouse_event(mx, my, MouseButton::RIGHT, id == WM_RBUTTONDOWN);
  245. break;
  246. }
  247. case WM_MBUTTONDOWN:
  248. case WM_MBUTTONUP:
  249. {
  250. int32_t mx = GET_X_LPARAM(lparam);
  251. int32_t my = GET_Y_LPARAM(lparam);
  252. _queue.push_mouse_event(mx, my, MouseButton::MIDDLE, id == WM_MBUTTONDOWN);
  253. break;
  254. }
  255. case WM_KEYDOWN:
  256. case WM_SYSKEYDOWN:
  257. case WM_KEYUP:
  258. case WM_SYSKEYUP:
  259. {
  260. KeyboardButton::Enum kb = win_translate_key(wparam & 0xff);
  261. int32_t modifier_mask = 0;
  262. if (kb == KeyboardButton::LSHIFT || kb == KeyboardButton::RSHIFT)
  263. {
  264. (id == WM_KEYDOWN || id == WM_SYSKEYDOWN) ? modifier_mask |= ModifierButton::SHIFT : modifier_mask &= ~ModifierButton::SHIFT;
  265. }
  266. else if (kb == KeyboardButton::LCONTROL || kb == KeyboardButton::RCONTROL)
  267. {
  268. (id == WM_KEYDOWN || id == WM_SYSKEYDOWN) ? modifier_mask |= ModifierButton::CTRL : modifier_mask &= ~ModifierButton::CTRL;
  269. }
  270. else if (kb == KeyboardButton::LALT || kb == KeyboardButton::RALT)
  271. {
  272. (id == WM_KEYDOWN || id == WM_SYSKEYDOWN) ? modifier_mask |= ModifierButton::ALT : modifier_mask &= ~ModifierButton::ALT;
  273. }
  274. _queue.push_keyboard_event(modifier_mask, kb, (id == WM_KEYDOWN || id == WM_SYSKEYDOWN));
  275. break;
  276. }
  277. default:
  278. break;
  279. }
  280. return DefWindowProc(hwnd, id, wparam, lparam);
  281. }
  282. private:
  283. static LRESULT CALLBACK window_proc(HWND hwnd, UINT id, WPARAM wparam, LPARAM lparam);
  284. public:
  285. HWND _hwnd;
  286. HDC _hdc;
  287. OsEventQueue _queue;
  288. };
  289. static WindowsDevice s_wdvc;
  290. LRESULT CALLBACK WindowsDevice::window_proc(HWND hwnd, UINT id, WPARAM wparam, LPARAM lparam)
  291. {
  292. return s_wdvc.pump_events(hwnd, id, wparam, lparam);
  293. }
  294. bool next_event(OsEvent& ev)
  295. {
  296. return s_wdvc._queue.pop_event(ev);
  297. }
  298. } // namespace crown
  299. int main(int argc, char** argv)
  300. {
  301. using namespace crown;
  302. WSADATA WsaData;
  303. int res = WSAStartup(MAKEWORD(2, 2), &WsaData);
  304. CE_ASSERT(res == 0, "Unable to initialize socket");
  305. CE_UNUSED(WsaData);
  306. CE_UNUSED(res);
  307. ConfigSettings cs;
  308. parse_command_line(argc, argv, cs);
  309. memory_globals::init();
  310. DiskFilesystem src_fs(cs.source_dir);
  311. parse_config_file(src_fs, cs);
  312. console_server_globals::init(cs.console_port, cs.wait_console);
  313. bundle_compiler_globals::init(cs.source_dir, cs.bundle_dir);
  314. bool do_continue = true;
  315. int exitcode = EXIT_SUCCESS;
  316. do_continue = bundle_compiler::main(cs.do_compile, cs.do_continue, cs.platform);
  317. if (do_continue)
  318. {
  319. DiskFilesystem dst_fs(cs.bundle_dir);
  320. exitcode = crown::s_wdvc.run(&dst_fs, &cs);
  321. }
  322. bundle_compiler_globals::shutdown();
  323. console_server_globals::shutdown();
  324. memory_globals::shutdown();
  325. WSACleanup();
  326. return exitcode;
  327. }
  328. #endif // CROWN_PLATFORM_WINDOWS