#ifdef BLAH_PLATFORM_WIN32 // Note: // This backend implementation is unfinished! // It's missing a few things, namely: // - Controller Support // (And error testing) #include "platform.h" #include "input.h" #include "graphics.h" #include #include #include #include #include #define WIN32_LEAN_AND_MEAN #include #include // for SetProcessDPIAware #include // for File Reading/Writing #include // for file explore #include // for known folder #include // for ticks method namespace Blah { using Duration = std::chrono::system_clock::duration; // Win32 Platform State struct Win32Platform { // Main State HWND hwnd; FilePath working_directory; FilePath user_directory; Duration start_time; RECT windowed_position; bool fullscreen = false; InputState* input_state = nullptr; // OpenGL Methods // These are only loaded if built using the OpenGL Backend HMODULE opengl_dll; void* (WINAPI* wglGetProcAddress) (const char* proc); HGLRC (WINAPI* wglCreateContext) (HDC hdc); BOOL (WINAPI* wglDeleteContext) (HGLRC hglrc); BOOL (WINAPI* wglMakeCurrent) (HDC hdc, HGLRC hglrc); }; // Win32 File Class class Win32File : public File { private: HANDLE m_handle; public: Win32File(HANDLE handle) { m_handle = handle; } ~Win32File() { CloseHandle(m_handle); } size_t length() override { // Todo: cache this value? not sure how performant it is LARGE_INTEGER file_size; if (GetFileSizeEx(m_handle, &file_size)) return file_size.QuadPart; return 0; } size_t position() override { LARGE_INTEGER move; LARGE_INTEGER result; move.QuadPart = 0; result.QuadPart = 0; SetFilePointerEx(m_handle, move, &result, FILE_CURRENT); return result.QuadPart; } size_t seek(size_t position) override { LARGE_INTEGER move; LARGE_INTEGER result; move.QuadPart = position; result.QuadPart = 0; SetFilePointerEx(m_handle, move, &result, FILE_BEGIN); return result.QuadPart; } size_t read(unsigned char* buffer, size_t length) override { static const DWORD read_step = 65536; size_t read = 0; while (read < length) { DWORD to_read = read_step; if (to_read > length - read) to_read = (DWORD)(length - read); DWORD moved = 0; if (ReadFile(m_handle, buffer + read, to_read, &moved, NULL)) read += moved; if (moved < to_read) break; } return read; } size_t write(const unsigned char* buffer, size_t length) override { static const DWORD write_step = 65536; size_t written = 0; while (written < length) { DWORD to_write = write_step; if (to_write > length - written) to_write = (DWORD)(length - written); DWORD moved = 0; if (WriteFile(m_handle, buffer + written, to_write, &moved, NULL)) written += moved; if (moved < to_write) break; } return written; } }; // Converts Windows scancode to Blah key Key win32_scancode_to_key(WPARAM wParam, LPARAM lParam); // Main Windows Procedure callback LRESULT CALLBACK win32_window_procedure(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam); } using namespace Blah; namespace fs = std::filesystem; static Blah::Win32Platform g_platform; bool Platform::init(const Config& config) { // clear platform g_platform = Win32Platform(); // Required to call this for Windows SetProcessDPIAware(); // Get the hInstance HINSTANCE hInstance = GetModuleHandle(NULL); // Create the Window Class WNDCLASS wc = {}; wc.lpfnWndProc = DefWindowProc; wc.lpszClassName = "BLAH WINDOW"; wc.hInstance = hInstance; wc.lpfnWndProc = win32_window_procedure; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hIcon = NULL; wc.lpszMenuName = NULL; wc.hbrBackground = (HBRUSH)COLOR_WINDOW; wc.cbClsExtra = 0; wc.cbWndExtra = 0; // Register the Window class RegisterClass(&wc); // Create the Window Instance g_platform.hwnd = CreateWindow("BLAH WINDOW", config.name, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, 0, 640, 480, NULL, NULL, hInstance, NULL); // Failed to create the Window if (g_platform.hwnd == NULL) { Log::error("Window Creation Failed"); return false; } // Create the OpenGL device info if (App::renderer() == Renderer::OpenGL) { // Load the DLL g_platform.opengl_dll = LoadLibraryA("opengl32.dll"); if (g_platform.opengl_dll == NULL) { Log::error("OpenGL Instantiation Failed - unable to fine opengl32.dll"); return false; } // Get the Windows GL functions we need g_platform.wglGetProcAddress = (void* (WINAPI*)(const char*))GetProcAddress(g_platform.opengl_dll, "wglGetProcAddress"); g_platform.wglCreateContext = (HGLRC(WINAPI*) (HDC))GetProcAddress(g_platform.opengl_dll, "wglCreateContext"); g_platform.wglDeleteContext = (BOOL(WINAPI*) (HGLRC))GetProcAddress(g_platform.opengl_dll, "wglDeleteContext"); g_platform.wglMakeCurrent = (BOOL(WINAPI*) (HDC, HGLRC))GetProcAddress(g_platform.opengl_dll, "wglMakeCurrent"); // TODO: // Allow the user to apply (some of) these values before instantiation. // Also applies to the SDL2 Backend PIXELFORMATDESCRIPTOR pfd = { sizeof(PIXELFORMATDESCRIPTOR), // size of this pfd 1, // version number PFD_DRAW_TO_WINDOW | // support window PFD_SUPPORT_OPENGL | // support OpenGL PFD_DOUBLEBUFFER, // double buffered PFD_TYPE_RGBA, // RGBA type 32, // 32-bit color depth 0, 0, 0, 0, 0, 0, // color bits ignored 0, // no alpha buffer 0, // shift bit ignored 0, // no accumulation buffer 0, 0, 0, 0, // accum bits ignored 24, // 24-bit z-buffer 8, // 8-bit stencil buffer 0, // no auxiliary buffer PFD_MAIN_PLANE, // main layer 0, // reserved 0, 0, 0 // layer masks ignored }; HDC hdc = GetDC(g_platform.hwnd); // get the best available match of pixel format for the device context int pixel_format = ChoosePixelFormat(hdc, &pfd); // make that the pixel format of the device context SetPixelFormat(hdc, pixel_format, &pfd); } // Reset our game timer g_platform.start_time = std::chrono::system_clock::now().time_since_epoch(); // Get Working Directory { TCHAR buffer[MAX_PATH]; GetModuleFileName(NULL, buffer, MAX_PATH); auto normalized = Path::normalize(buffer); auto end = normalized.last_index_of('/');; if (end >= 0) g_platform.working_directory = FilePath(normalized.begin(), normalized.begin() + end); else g_platform.working_directory = normalized; g_platform.working_directory.append("/"); } // Get Application User Directory { PWSTR path = NULL; if (SUCCEEDED(SHGetKnownFolderPath(FOLDERID_RoamingAppData, KF_FLAG_CREATE, NULL, &path))) { auto end = path; while (*end != 0) end++; FilePath result; result.append_utf16((u16*)path, (u16*)end); g_platform.user_directory = Path::join(Path::normalize(result), config.name) + "/"; } CoTaskMemFree(path); } // Not currently fullscreen g_platform.fullscreen = false; // Finished Platform Setup return true; } void Platform::ready() { // Setup Window Size { auto scale = get_content_scale(); int sw = (int)(App::config().width * scale); int sh = (int)(App::config().height * scale); set_size(sw, sh); } // Display the game window ShowWindow(g_platform.hwnd, SW_SHOW); } void Platform::shutdown() { DestroyWindow(g_platform.hwnd); } u64 Platform::ticks() { // Todo: // This should account for whatever Time::ticks_per_second is set to auto now = std::chrono::system_clock::now().time_since_epoch(); return std::chrono::duration_cast(now - g_platform.start_time).count(); } LRESULT CALLBACK Blah::win32_window_procedure(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) { switch (msg) { case WM_CLOSE: { auto& config = App::config(); if (config.on_exit_request != nullptr) config.on_exit_request(); return 0; } case WM_DESTROY: PostQuitMessage(0); return 0; // Mouse Input case WM_LBUTTONDOWN: g_platform.input_state->mouse.on_press(MouseButton::Left); return 0; case WM_LBUTTONUP: g_platform.input_state->mouse.on_release(MouseButton::Left); return 0; case WM_RBUTTONDOWN: g_platform.input_state->mouse.on_press(MouseButton::Right); return 0; case WM_RBUTTONUP: g_platform.input_state->mouse.on_release(MouseButton::Right); return 0; case WM_MBUTTONDOWN: g_platform.input_state->mouse.on_press(MouseButton::Middle); return 0; case WM_MBUTTONUP: g_platform.input_state->mouse.on_release(MouseButton::Middle); return 0; case WM_MOUSEMOVE: g_platform.input_state->mouse.on_move(Vec2f((float)((u16)lParam), (float)(lParam >> 16)), Vec2f::zero); return 0; case WM_MOUSEWHEEL: g_platform.input_state->mouse.wheel = Point(0, GET_WHEEL_DELTA_WPARAM(wParam) / WHEEL_DELTA); return 0; // Text Input case WM_UNICHAR: if (wParam == UNICODE_NOCHAR) return 1; case WM_CHAR: { String result; result.append((u32)wParam); if (result.length() > 0) g_platform.input_state->keyboard.text += result.cstr(); return 0; } // Keyboard Input case WM_KEYDOWN: case WM_SYSKEYDOWN: { auto is_repeat = ((lParam & (1 << 30)) >> 30) == 1; if (!is_repeat) { auto key = Blah::win32_scancode_to_key(wParam, lParam); if (key != Key::Unknown) g_platform.input_state->keyboard.on_press(key); } return 0; } case WM_KEYUP: case WM_SYSKEYUP: { auto key = Blah::win32_scancode_to_key(wParam, lParam); if (key != Key::Unknown) g_platform.input_state->keyboard.on_release(key); return 0; } } return DefWindowProc(hwnd, msg, wParam, lParam); } void Platform::update(InputState& state) { // store reference to input state g_platform.input_state = &state; // Catch & Dispatch Window Messages MSG msg; while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { TranslateMessage(&msg); DispatchMessage(&msg); } } void Platform::sleep(int milliseconds) { if (milliseconds > 0) Sleep(milliseconds); } void Platform::present() { if (App::renderer() == Renderer::OpenGL) { HDC hdc = GetDC(g_platform.hwnd); SwapBuffers(hdc); } } const char* Platform::get_title() { return nullptr; } void Platform::set_title(const char* title) { SetWindowText(g_platform.hwnd, title); } void Platform::get_position(int* x, int* y) { RECT rect; if (GetWindowRect(g_platform.hwnd, &rect)) { *x = rect.left; *y = rect.top; } } void Platform::set_position(int x, int y) { int w, h; get_size(&w, &h); SetWindowPos(g_platform.hwnd, NULL, x, y, w, h, 0); } void Platform::set_fullscreen(bool enabled) { if (g_platform.fullscreen == enabled) return; g_platform.fullscreen = enabled; if (g_platform.fullscreen) { GetWindowRect(g_platform.hwnd, &g_platform.windowed_position); int w = GetSystemMetrics(SM_CXSCREEN); int h = GetSystemMetrics(SM_CYSCREEN); SetWindowLongPtr(g_platform.hwnd, GWL_STYLE, WS_VISIBLE | WS_POPUP); SetWindowPos(g_platform.hwnd, HWND_TOP, 0, 0, w, h, 0); ShowWindow(g_platform.hwnd, SW_SHOW); } else { SetWindowLongPtr(g_platform.hwnd, GWL_STYLE, WS_OVERLAPPEDWINDOW); SetWindowPos(g_platform.hwnd, HWND_TOP, g_platform.windowed_position.left, g_platform.windowed_position.top, g_platform.windowed_position.right - g_platform.windowed_position.left, g_platform.windowed_position.bottom - g_platform.windowed_position.top, 0); ShowWindow(g_platform.hwnd, SW_SHOW); } } void Platform::get_size(int* width, int* height) { RECT rect; if (GetClientRect(g_platform.hwnd, &rect)) { *width = rect.right - rect.left; *height = rect.bottom - rect.top; } } void Platform::set_size(int width, int height) { RECT client_rect; RECT border_rect; GetClientRect(g_platform.hwnd, &client_rect); GetWindowRect(g_platform.hwnd, &border_rect); int border_width = (border_rect.right - border_rect.left) - (client_rect.right - client_rect.left); int border_height = (border_rect.bottom - border_rect.top) - (client_rect.bottom - client_rect.top); SetWindowPos(g_platform.hwnd, NULL, border_rect.left, border_rect.top, width + border_width, height + border_height, 0); } void Platform::get_draw_size(int* width, int* height) { RECT rect; if (GetClientRect(g_platform.hwnd, &rect)) { *width = rect.right - rect.left; *height = rect.bottom - rect.top; } } float Platform::get_content_scale() { // base value of Windows DPI // as seen here: https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-getdpiforwindow constexpr float base_raw_value = 96.0f; UINT raw_value = GetDpiForWindow(g_platform.hwnd); return (raw_value / base_raw_value); } const char* Platform::app_path() { return g_platform.working_directory.cstr(); } const char* Platform::user_path() { return g_platform.user_directory.cstr(); } bool Platform::file_exists(const char* path) { return fs::is_regular_file(path); } bool Platform::file_delete(const char* path) { return fs::remove(path); } bool Platform::dir_create(const char* path) { std::error_code error; return fs::create_directories(path, error); } bool Platform::dir_exists(const char* path) { return fs::is_directory(path); } bool Platform::dir_delete(const char* path) { return fs::remove_all(path) > 0; } void Platform::dir_enumerate(Vector& list, const char* path, bool recursive) { if (fs::is_directory(path)) { if (recursive) { for (auto& p : fs::recursive_directory_iterator(path)) list.emplace_back(p.path().string().c_str()); } else { for (auto& p : fs::directory_iterator(path)) list.emplace_back(p.path().string().c_str()); } } } void Platform::dir_explore(const char* path) { ShellExecute(NULL, "open", path, NULL, NULL, SW_SHOWDEFAULT); } FileRef Platform::file_open(const char* path, FileMode mode) { int access = 0; int creation = 0; switch (mode) { case FileMode::OpenRead: access = GENERIC_READ; creation = OPEN_EXISTING; break; case FileMode::Open: access = GENERIC_READ | GENERIC_WRITE; creation = OPEN_EXISTING; break; case FileMode::CreateWrite: access = GENERIC_WRITE; creation = CREATE_ALWAYS; break; case FileMode::Create: access = GENERIC_READ | GENERIC_WRITE; creation = CREATE_ALWAYS; break; } auto result = CreateFile(path, access, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, creation, FILE_ATTRIBUTE_NORMAL, NULL); if (result == INVALID_HANDLE_VALUE) return FileRef(); return FileRef(new Win32File(result)); } // clipboard void Platform::set_clipboard(const char* text) { BLAH_ASSERT(false, "Not Implemented Yet"); } const char* Platform::get_clipboard() { BLAH_ASSERT(false, "Not Implemented Yet"); } void* Platform::gl_get_func(const char* name) { // this check is taken from https://www.khronos.org/opengl/wiki/Load_OpenGL_Functions // wglGetProcAddress doesn't always return valid pointers for some specific methods? void* p = (void*)g_platform.wglGetProcAddress(name); if ((p == 0) || (p == (void*)0x1) || (p == (void*)0x2) || (p == (void*)0x3) || (p == (void*)-1)) { p = (void*)GetProcAddress(g_platform.opengl_dll, name); } return p; } void* Platform::gl_context_create() { HDC hdc = GetDC(g_platform.hwnd); return g_platform.wglCreateContext(hdc); } void Platform::gl_context_make_current(void* context) { if (context != nullptr) { HDC hdc = GetDC(g_platform.hwnd); g_platform.wglMakeCurrent(hdc, (HGLRC)context); } else g_platform.wglMakeCurrent(NULL, NULL); } void Platform::gl_context_destroy(void* context) { g_platform.wglDeleteContext((HGLRC)context); } void* Platform::d3d11_get_hwnd() { return g_platform.hwnd; } Key Blah::win32_scancode_to_key(WPARAM wParam, LPARAM lParam) { // scancodes switch ((lParam >> 16) & 0xFF) { case 1: return Key::Escape; case 2: return Key::D1; case 3: return Key::D2; case 4: return Key::D3; case 5: return Key::D4; case 6: return Key::D5; case 7: return Key::D6; case 8: return Key::D7; case 9: return Key::D8; case 10: return Key::D9; case 11: return Key::D0; case 12: return Key::Minus; case 13: return Key::Equals; case 14: return Key::Backspace; case 15: return Key::Tab; case 16: return Key::Q; case 17: return Key::W; case 18: return Key::E; case 19: return Key::R; case 20: return Key::T; case 21: return Key::Y; case 22: return Key::U; case 23: return Key::I; case 24: return Key::O; case 25: return Key::P; case 26: return Key::LeftBracket; case 27: return Key::RightBracket; case 28: return Key::Enter; case 29: return Key::LeftControl; case 30: return Key::A; case 31: return Key::S; case 32: return Key::D; case 33: return Key::F; case 34: return Key::G; case 35: return Key::H; case 36: return Key::J; case 37: return Key::K; case 38: return Key::L; case 39: return Key::Semicolon; case 40: return Key::Apostrophe; case 41: return Key::Tilde; case 42: return Key::LeftShift; case 43: return Key::Backslash; case 44: return Key::Z; case 45: return Key::X; case 46: return Key::C; case 47: return Key::V; case 48: return Key::B; case 49: return Key::N; case 50: return Key::M; case 51: return Key::Comma; case 52: return Key::Period; case 53: return Key::Slash; case 54: return Key::RightShift; case 55: return Key::PrintScreen; case 56: return Key::LeftAlt; case 57: return Key::Space; case 58: return Key::Capslock; case 59: return Key::F1; case 60: return Key::F2; case 61: return Key::F3; case 62: return Key::F4; case 63: return Key::F5; case 64: return Key::F6; case 65: return Key::F7; case 66: return Key::F8; case 67: return Key::F9; case 68: return Key::F10; case 71: return Key::Home; case 72: return Key::Up; case 73: return Key::PageUp; case 74: return Key::KeypadMinus; case 75: return Key::Left; case 76: return Key::Keypad5; case 77: return Key::Right; case 78: return Key::KeypadPlus; case 79: return Key::End; case 80: return Key::Down; case 81: return Key::PageDown; case 82: return Key::Insert; case 83: return Key::Delete; case 87: return Key::F11; case 88: return Key::F12; case 89: return Key::Pause; case 91: return Key::LeftOS; case 92: return Key::RightOS; case 93: return Key::Application; case 100: return Key::F13; case 101: return Key::F14; case 102: return Key::F15; case 103: return Key::F16; case 104: return Key::F17; case 105: return Key::F18; case 106: return Key::F19; } // virtual keys switch (wParam) { case VK_CANCEL: return Key::Cancel; case VK_BACK: return Key::Backspace; case VK_TAB: return Key::Tab; case VK_CLEAR: return Key::Clear; case VK_RETURN: return Key::Enter; case VK_SHIFT: return Key::LeftShift; case VK_CONTROL: return Key::LeftControl; case VK_PAUSE: return Key::Pause; case VK_CAPITAL: return Key::Capslock; case VK_ESCAPE: return Key::Escape; case VK_SPACE: return Key::Space; case VK_PRIOR: return Key::Prior; case VK_END: return Key::End; case VK_HOME: return Key::Home; case VK_LEFT: return Key::Left; case VK_UP: return Key::Up; case VK_RIGHT: return Key::Right; case VK_DOWN: return Key::Down; case VK_SELECT: return Key::Select; case VK_PRINT: return Key::PrintScreen; case VK_EXECUTE: return Key::Execute; case VK_SNAPSHOT: return Key::PrintScreen; case VK_INSERT: return Key::Insert; case VK_DELETE: return Key::Delete; case VK_HELP: return Key::Help; case VK_LWIN: return Key::LeftOS; case VK_RWIN: return Key::RightOS; case VK_APPS: return Key::Application; case VK_SLEEP: return Key::Unknown; case VK_NUMPAD0: return Key::Keypad0; case VK_NUMPAD1: return Key::Keypad1; case VK_NUMPAD2: return Key::Keypad2; case VK_NUMPAD3: return Key::Keypad3; case VK_NUMPAD4: return Key::Keypad4; case VK_NUMPAD5: return Key::Keypad5; case VK_NUMPAD6: return Key::Keypad6; case VK_NUMPAD7: return Key::Keypad7; case VK_NUMPAD8: return Key::Keypad8; case VK_NUMPAD9: return Key::Keypad9; case VK_F1: return Key::F1; case VK_F2: return Key::F2; case VK_F3: return Key::F3; case VK_F4: return Key::F4; case VK_F5: return Key::F5; case VK_F6: return Key::F6; case VK_F7: return Key::F7; case VK_F8: return Key::F8; case VK_F9: return Key::F9; case VK_F10: return Key::F10; case VK_F11: return Key::F11; case VK_F12: return Key::F12; case VK_F13: return Key::F13; case VK_F14: return Key::F14; case VK_F15: return Key::F15; case VK_F16: return Key::F16; case VK_F17: return Key::F17; case VK_F18: return Key::F18; case VK_F19: return Key::F19; case VK_F20: return Key::F20; case VK_F21: return Key::F21; case VK_F22: return Key::F22; case VK_F23: return Key::F23; case VK_F24: return Key::F24; case VK_NUMLOCK: return Key::Numlock; case VK_LSHIFT: return Key::LeftShift; case VK_RSHIFT: return Key::RightShift; case VK_LCONTROL: return Key::LeftControl; case VK_RCONTROL: return Key::RightControl; case VK_VOLUME_MUTE: return Key::Mute; case VK_VOLUME_DOWN: return Key::VolumeDown; case VK_VOLUME_UP: return Key::VolumeUp; } return Key::Unknown; } #endif // BLAH_PLATFORM_WIN32