main_windows.cpp 9.3 KB

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