window_win32.odin 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. #+build windows
  2. #+private file
  3. package karl2d
  4. @(private="package")
  5. WINDOW_INTERFACE_WIN32 :: Window_Interface {
  6. state_size = win32_state_size,
  7. init = win32_init,
  8. shutdown = win32_shutdown,
  9. window_handle = win32_window_handle,
  10. process_events = win32_process_events,
  11. get_events = win32_get_events,
  12. get_width = win32_get_width,
  13. get_height = win32_get_height,
  14. clear_events = win32_clear_events,
  15. set_position = win32_set_position,
  16. set_size = win32_set_size,
  17. get_window_scale = win32_get_window_scale,
  18. set_flags = win32_set_flags,
  19. is_gamepad_active = win32_is_gamepad_active,
  20. get_gamepad_axis = win32_get_gamepad_axis,
  21. set_gamepad_vibration = win32_set_gamepad_vibration,
  22. set_internal_state = win32_set_internal_state,
  23. }
  24. import win32 "core:sys/windows"
  25. import "base:runtime"
  26. win32_state_size :: proc() -> int {
  27. return size_of(Win32_State)
  28. }
  29. win32_init :: proc(window_state: rawptr, window_width: int, window_height: int, window_title: string,
  30. flags: Window_Flags, allocator: runtime.Allocator) {
  31. assert(window_state != nil)
  32. s = (^Win32_State)(window_state)
  33. s.allocator = allocator
  34. s.events = make([dynamic]Window_Event, allocator)
  35. s.width = window_width
  36. s.height = window_height
  37. s.custom_context = context
  38. win32.SetProcessDpiAwarenessContext(win32.DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2)
  39. win32.SetProcessDPIAware()
  40. CLASS_NAME :: "karl2d"
  41. instance := win32.HINSTANCE(win32.GetModuleHandleW(nil))
  42. cls := win32.WNDCLASSW {
  43. style = win32.CS_OWNDC,
  44. lpfnWndProc = window_proc,
  45. lpszClassName = CLASS_NAME,
  46. hInstance = instance,
  47. hCursor = win32.LoadCursorA(nil, win32.IDC_ARROW),
  48. }
  49. win32.RegisterClassW(&cls)
  50. r: win32.RECT
  51. r.right = i32(window_width)
  52. r.bottom = i32(window_height)
  53. s.flags = flags
  54. style := style_from_flags(flags)
  55. win32.AdjustWindowRect(&r, style, false)
  56. hwnd := win32.CreateWindowW(CLASS_NAME,
  57. win32.utf8_to_wstring(window_title),
  58. style,
  59. win32.CW_USEDEFAULT, win32.CW_USEDEFAULT,
  60. r.right - r.left, r.bottom - r.top,
  61. nil, nil, instance, nil,
  62. )
  63. win32.XInputEnable(true)
  64. assert(hwnd != nil, "Failed creating window")
  65. s.hwnd = hwnd
  66. }
  67. win32_shutdown :: proc() {
  68. delete(s.events)
  69. win32.DestroyWindow(s.hwnd)
  70. }
  71. win32_window_handle :: proc() -> Window_Handle {
  72. return Window_Handle(s.hwnd)
  73. }
  74. win32_process_events :: proc() {
  75. msg: win32.MSG
  76. for win32.PeekMessageW(&msg, nil, 0, 0, win32.PM_REMOVE) {
  77. win32.TranslateMessage(&msg)
  78. win32.DispatchMessageW(&msg)
  79. }
  80. for gamepad in 0..<4 {
  81. gp_event: win32.XINPUT_KEYSTROKE
  82. for win32.XInputGetKeystroke(win32.XUSER(gamepad), 0, &gp_event) == .SUCCESS {
  83. button: Maybe(Gamepad_Button)
  84. #partial switch gp_event.VirtualKey {
  85. case .DPAD_UP: button = .Left_Face_Up
  86. case .DPAD_DOWN: button = .Left_Face_Down
  87. case .DPAD_LEFT: button = .Left_Face_Left
  88. case .DPAD_RIGHT: button = .Left_Face_Right
  89. case .Y: button = .Right_Face_Up
  90. case .A: button = .Right_Face_Down
  91. case .X: button = .Right_Face_Left
  92. case .B: button = .Right_Face_Right
  93. case .LSHOULDER: button = .Left_Shoulder
  94. case .LTRIGGER: button = .Left_Trigger
  95. case .RSHOULDER: button = .Right_Shoulder
  96. case .RTRIGGER: button = .Right_Trigger
  97. case .BACK: button = .Middle_Face_Left
  98. // Not sure you can get the "middle button" with XInput (the one that goe to dashboard)
  99. case .START: button = .Middle_Face_Right
  100. case .LTHUMB_PRESS: button = .Left_Stick_Press
  101. case .RTHUMB_PRESS: button = .Right_Stick_Press
  102. }
  103. b := button.? or_continue
  104. evt: Window_Event
  105. if .KEYDOWN in gp_event.Flags {
  106. evt = Window_Event_Gamepad_Button_Went_Down {
  107. gamepad = gamepad,
  108. button = b,
  109. }
  110. } else if .KEYUP in gp_event.Flags {
  111. evt = Window_Event_Gamepad_Button_Went_Up {
  112. gamepad = gamepad,
  113. button = b,
  114. }
  115. }
  116. if evt != nil {
  117. append(&s.events, evt)
  118. }
  119. }
  120. }
  121. }
  122. win32_get_events :: proc() -> []Window_Event {
  123. return s.events[:]
  124. }
  125. win32_get_width :: proc() -> int {
  126. return s.width
  127. }
  128. win32_get_height :: proc() -> int {
  129. return s.height
  130. }
  131. win32_clear_events :: proc() {
  132. runtime.clear(&s.events)
  133. }
  134. win32_set_position :: proc(x: int, y: int) {
  135. // TODO: Does x, y respect monitor DPI?
  136. win32.SetWindowPos(
  137. s.hwnd,
  138. {},
  139. i32(x),
  140. i32(y),
  141. 0,
  142. 0,
  143. win32.SWP_NOACTIVATE | win32.SWP_NOZORDER | win32.SWP_NOSIZE,
  144. )
  145. }
  146. win32_set_size :: proc(w, h: int) {
  147. win32.SetWindowPos(
  148. s.hwnd,
  149. {},
  150. 0,
  151. 0,
  152. i32(w),
  153. i32(h),
  154. win32.SWP_NOACTIVATE | win32.SWP_NOZORDER | win32.SWP_NOMOVE,
  155. )
  156. }
  157. win32_get_window_scale :: proc() -> f32 {
  158. return f32(win32.GetDpiForWindow(s.hwnd))/96.0
  159. }
  160. win32_set_flags :: proc(flags: Window_Flags) {
  161. s.flags = flags
  162. style := style_from_flags(flags)
  163. win32.SetWindowLongW(s.hwnd, win32.GWL_STYLE, i32(style))
  164. }
  165. win32_is_gamepad_active :: proc(gamepad: int) -> bool {
  166. if gamepad < 0 || gamepad >= MAX_GAMEPADS {
  167. return false
  168. }
  169. gp_state: win32.XINPUT_STATE
  170. return win32.XInputGetState(win32.XUSER(gamepad), &gp_state) == .SUCCESS
  171. }
  172. win32_get_gamepad_axis :: proc(gamepad: int, axis: Gamepad_Axis) -> f32 {
  173. if gamepad < 0 || gamepad >= MAX_GAMEPADS {
  174. return 0
  175. }
  176. gp_state: win32.XINPUT_STATE
  177. if win32.XInputGetState(win32.XUSER(gamepad), &gp_state) == .SUCCESS {
  178. gp := gp_state.Gamepad
  179. // Numbers from https://learn.microsoft.com/en-us/windows/win32/api/xinput/ns-xinput-xinput_gamepad
  180. STICK_MAX :: 32767
  181. TRIGGER_MAX :: 255
  182. switch axis {
  183. case .Left_Stick_X: return f32(gp.sThumbLX) / STICK_MAX
  184. case .Left_Stick_Y: return -f32(gp.sThumbLY) / STICK_MAX
  185. case .Right_Stick_X: return f32(gp.sThumbRX) / STICK_MAX
  186. case .Right_Stick_Y: return -f32(gp.sThumbRY) / STICK_MAX
  187. case .Left_Trigger: return f32(gp.bLeftTrigger) / TRIGGER_MAX
  188. case .Right_Trigger: return f32(gp.bRightTrigger) / TRIGGER_MAX
  189. }
  190. }
  191. return 0
  192. }
  193. win32_set_gamepad_vibration :: proc(gamepad: int, left: f32, right: f32) {
  194. if gamepad < 0 || gamepad >= MAX_GAMEPADS {
  195. return
  196. }
  197. vib := win32.XINPUT_VIBRATION {
  198. wLeftMotorSpeed = win32.WORD(left * 65535),
  199. wRightMotorSpeed = win32.WORD(right * 65535),
  200. }
  201. win32.XInputSetState(win32.XUSER(gamepad), &vib)
  202. }
  203. win32_set_internal_state :: proc(state: rawptr) {
  204. assert(state != nil)
  205. s = (^Win32_State)(state)
  206. }
  207. Win32_State :: struct {
  208. allocator: runtime.Allocator,
  209. custom_context: runtime.Context,
  210. hwnd: win32.HWND,
  211. flags: Window_Flags,
  212. width: int,
  213. height: int,
  214. events: [dynamic]Window_Event,
  215. }
  216. style_from_flags :: proc(flags: Window_Flags) -> win32.DWORD {
  217. style := win32.WS_OVERLAPPED | win32.WS_CAPTION | win32.WS_SYSMENU |
  218. win32.WS_MINIMIZEBOX | win32.WS_MAXIMIZEBOX | win32.WS_VISIBLE |
  219. win32.CS_OWNDC
  220. if .Resizable in flags {
  221. style |= win32.WS_THICKFRAME
  222. }
  223. return style
  224. }
  225. s: ^Win32_State
  226. window_proc :: proc "stdcall" (hwnd: win32.HWND, msg: win32.UINT, wparam: win32.WPARAM, lparam: win32.LPARAM) -> win32.LRESULT {
  227. context = s.custom_context
  228. switch msg {
  229. case win32.WM_DESTROY:
  230. win32.PostQuitMessage(0)
  231. case win32.WM_CLOSE:
  232. append(&s.events, Window_Event_Close_Wanted{})
  233. case win32.WM_KEYDOWN:
  234. key := key_from_event_params(wparam, lparam)
  235. append(&s.events, Window_Event_Key_Went_Down {
  236. key = key,
  237. })
  238. return 0
  239. case win32.WM_KEYUP:
  240. key := key_from_event_params(wparam, lparam)
  241. append(&s.events, Window_Event_Key_Went_Up {
  242. key = key,
  243. })
  244. return 0
  245. case win32.WM_MOUSEMOVE:
  246. x := win32.GET_X_LPARAM(lparam)
  247. y := win32.GET_Y_LPARAM(lparam)
  248. append(&s.events, Window_Event_Mouse_Move {
  249. position = {f32(x), f32(y)},
  250. })
  251. return 0
  252. case win32.WM_MOUSEWHEEL:
  253. delta := f32(win32.GET_WHEEL_DELTA_WPARAM(wparam))/win32.WHEEL_DELTA
  254. append(&s.events, Window_Event_Mouse_Wheel {
  255. delta = delta,
  256. })
  257. case win32.WM_LBUTTONDOWN:
  258. append(&s.events, Window_Event_Mouse_Button_Went_Down {
  259. button = .Left,
  260. })
  261. case win32.WM_LBUTTONUP:
  262. append(&s.events, Window_Event_Mouse_Button_Went_Up {
  263. button = .Left,
  264. })
  265. case win32.WM_MBUTTONDOWN:
  266. append(&s.events, Window_Event_Mouse_Button_Went_Down {
  267. button = .Middle,
  268. })
  269. case win32.WM_MBUTTONUP:
  270. append(&s.events, Window_Event_Mouse_Button_Went_Up {
  271. button = .Middle,
  272. })
  273. case win32.WM_RBUTTONDOWN:
  274. append(&s.events, Window_Event_Mouse_Button_Went_Down {
  275. button = .Right,
  276. })
  277. case win32.WM_RBUTTONUP:
  278. append(&s.events, Window_Event_Mouse_Button_Went_Up {
  279. button = .Right,
  280. })
  281. case win32.WM_SIZE:
  282. width := win32.LOWORD(lparam)
  283. height := win32.HIWORD(lparam)
  284. s.width = int(width)
  285. s.height = int(height)
  286. append(&s.events, Window_Event_Resize {
  287. width = int(width),
  288. height = int(height),
  289. })
  290. }
  291. return win32.DefWindowProcW(hwnd, msg, wparam, lparam)
  292. }
  293. key_from_event_params :: proc(wparam: win32.WPARAM, lparam: win32.LPARAM) -> Keyboard_Key{
  294. if wparam == win32.VK_RETURN && win32.HIWORD(lparam) & win32.KF_EXTENDED != 0 {
  295. return .NP_Enter
  296. }
  297. return WIN32_VK_MAP[wparam]
  298. }
  299. WIN32_VK_MAP := [255]Keyboard_Key {
  300. win32.VK_0 = .N0,
  301. win32.VK_1 = .N1,
  302. win32.VK_2 = .N2,
  303. win32.VK_3 = .N3,
  304. win32.VK_4 = .N4,
  305. win32.VK_5 = .N5,
  306. win32.VK_6 = .N6,
  307. win32.VK_7 = .N7,
  308. win32.VK_8 = .N8,
  309. win32.VK_9 = .N9,
  310. win32.VK_A = .A,
  311. win32.VK_B = .B,
  312. win32.VK_C = .C,
  313. win32.VK_D = .D,
  314. win32.VK_E = .E,
  315. win32.VK_F = .F,
  316. win32.VK_G = .G,
  317. win32.VK_H = .H,
  318. win32.VK_I = .I,
  319. win32.VK_J = .J,
  320. win32.VK_K = .K,
  321. win32.VK_L = .L,
  322. win32.VK_M = .M,
  323. win32.VK_N = .N,
  324. win32.VK_O = .O,
  325. win32.VK_P = .P,
  326. win32.VK_Q = .Q,
  327. win32.VK_R = .R,
  328. win32.VK_S = .S,
  329. win32.VK_T = .T,
  330. win32.VK_U = .U,
  331. win32.VK_V = .V,
  332. win32.VK_W = .W,
  333. win32.VK_X = .X,
  334. win32.VK_Y = .Y,
  335. win32.VK_Z = .Z,
  336. win32.VK_OEM_7 = .Apostrophe,
  337. win32.VK_OEM_COMMA = .Comma,
  338. win32.VK_OEM_MINUS = .Minus,
  339. win32.VK_OEM_PERIOD = .Period,
  340. win32.VK_OEM_2 = .Slash,
  341. win32.VK_OEM_1 = .Semicolon,
  342. win32.VK_OEM_PLUS = .Equal,
  343. win32.VK_OEM_4 = .Left_Bracket,
  344. win32.VK_OEM_5 = .Backslash,
  345. win32.VK_OEM_6 = .Right_Bracket,
  346. win32.VK_OEM_3 = .Backtick,
  347. win32.VK_SPACE = .Space,
  348. win32.VK_ESCAPE = .Escape,
  349. win32.VK_RETURN = .Enter,
  350. win32.VK_TAB = .Tab,
  351. win32.VK_BACK = .Backspace,
  352. win32.VK_INSERT = .Insert,
  353. win32.VK_DELETE = .Delete,
  354. win32.VK_RIGHT = .Right,
  355. win32.VK_LEFT = .Left,
  356. win32.VK_DOWN = .Down,
  357. win32.VK_UP = .Up,
  358. win32.VK_PRIOR = .Page_Up,
  359. win32.VK_NEXT = .Page_Down,
  360. win32.VK_HOME = .Home,
  361. win32.VK_END = .End,
  362. win32.VK_CAPITAL = .Caps_Lock,
  363. win32.VK_SCROLL = .Scroll_Lock,
  364. win32.VK_NUMLOCK = .Num_Lock,
  365. win32.VK_PRINT = .Print_Screen,
  366. win32.VK_PAUSE = .Pause,
  367. win32.VK_F1 = .F1,
  368. win32.VK_F2 = .F2,
  369. win32.VK_F3 = .F3,
  370. win32.VK_F4 = .F4,
  371. win32.VK_F5 = .F5,
  372. win32.VK_F6 = .F6,
  373. win32.VK_F7 = .F7,
  374. win32.VK_F8 = .F8,
  375. win32.VK_F9 = .F9,
  376. win32.VK_F10 = .F10,
  377. win32.VK_F11 = .F11,
  378. win32.VK_F12 = .F12,
  379. win32.VK_LSHIFT = .Left_Shift,
  380. win32.VK_LCONTROL = .Left_Control,
  381. win32.VK_LMENU = .Left_Alt,
  382. win32.VK_MENU = .Left_Alt,
  383. win32.VK_LWIN = .Left_Super,
  384. win32.VK_RSHIFT = .Right_Shift,
  385. win32.VK_RCONTROL = .Right_Control,
  386. win32.VK_RMENU = .Right_Alt,
  387. win32.VK_RWIN = .Right_Super,
  388. win32.VK_APPS = .Menu,
  389. win32.VK_NUMPAD0 = .NP_0,
  390. win32.VK_NUMPAD1 = .NP_1,
  391. win32.VK_NUMPAD2 = .NP_2,
  392. win32.VK_NUMPAD3 = .NP_3,
  393. win32.VK_NUMPAD4 = .NP_4,
  394. win32.VK_NUMPAD5 = .NP_5,
  395. win32.VK_NUMPAD6 = .NP_6,
  396. win32.VK_NUMPAD7 = .NP_7,
  397. win32.VK_NUMPAD8 = .NP_8,
  398. win32.VK_NUMPAD9 = .NP_9,
  399. win32.VK_DECIMAL = .NP_Decimal,
  400. win32.VK_DIVIDE = .NP_Divide,
  401. win32.VK_MULTIPLY = .NP_Multiply,
  402. win32.VK_SUBTRACT = .NP_Subtract,
  403. win32.VK_ADD = .NP_Add,
  404. // NP_Enter is handled separately
  405. win32.VK_OEM_NEC_EQUAL = .NP_Equal,
  406. }