#ifndef BOO_WIN32COMMON_HPP #define BOO_WIN32COMMON_HPP #ifndef _CRT_SECURE_NO_WARNINGS #define _CRT_SECURE_NO_WARNINGS 1 /* STFU MSVC */ #endif #ifndef WIN32_LEAN_AND_MEAN #define WIN32_LEAN_AND_MEAN 1 #endif #include #include extern DWORD g_mainThreadId; #include "boo/IWindow.hpp" namespace boo {class IWindow;} #if _WIN32_WINNT_WIN10 #include #include #include #include #include struct D3D12Context { ComPtr m_dxFactory; ComPtr m_dev; ComPtr m_qalloc[2]; ComPtr m_q; ComPtr m_loadqalloc; ComPtr m_loadq; ComPtr m_loadfence; UINT64 m_loadfenceval = 0; HANDLE m_loadfencehandle; ComPtr m_loadlist; ComPtr m_rs; struct Window { ComPtr m_swapChain; UINT m_backBuf = 0; bool m_needsResize = false; size_t width, height; }; std::unordered_map m_windows; }; #elif _WIN32_WINNT_WIN7 #include #include #include #include #else #error Unsupported Windows target #endif struct D3D11Context { ComPtr m_dxFactory; ComPtr m_dev; ComPtr m_devCtx; ComPtr m_ss; struct Window { ComPtr m_swapChain; bool m_needsResize = false; size_t width, height; bool m_needsFSTransition = false; bool m_fs = false; DXGI_MODE_DESC m_fsdesc = {}; }; std::unordered_map m_windows; }; struct OGLContext { ComPtr m_dxFactory; HGLRC m_lastContext = 0; struct Window { HWND m_hwnd; HDC m_deviceContext; HGLRC m_mainContext; HGLRC m_renderContext; bool m_needsResize = false; size_t width, height; bool m_fs = false; LONG m_fsStyle; LONG m_fsExStyle; RECT m_fsRect; int m_fsCountDown = 0; }; std::unordered_map m_windows; }; struct Boo3DAppContext { D3D11Context m_ctx11; #if _WIN32_WINNT_WIN10 D3D12Context m_ctx12; #endif OGLContext m_ctxOgl; void resize(boo::IWindow* window, size_t width, size_t height) { #if _WIN32_WINNT_WIN10 if (m_ctx12.m_dev) { D3D12Context::Window& win = m_ctx12.m_windows[window]; win.width = width; win.height = height; win.m_needsResize = true; } else #endif { D3D11Context::Window& win = m_ctx11.m_windows[window]; win.width = width; win.height = height; win.m_needsResize = true; } } bool isFullscreen(const boo::IWindow* window) { #if _WIN32_WINNT_WIN10 if (m_ctx12.m_dev) { D3D12Context::Window& win = m_ctx12.m_windows[window]; BOOL isFScr; win.m_swapChain->GetFullscreenState(&isFScr, nullptr); return isFScr != 0; } #endif if (m_ctx11.m_dev) { D3D11Context::Window& win = m_ctx11.m_windows[window]; BOOL isFScr; win.m_swapChain->GetFullscreenState(&isFScr, nullptr); return isFScr != 0; } OGLContext::Window& win = m_ctxOgl.m_windows[window]; return win.m_fs; } bool setFullscreen(boo::IWindow* window, bool fs) { #if _WIN32_WINNT_WIN10 if (m_ctx12.m_dev) { D3D12Context::Window& win = m_ctx12.m_windows[window]; BOOL isFScr; win.m_swapChain->GetFullscreenState(&isFScr, nullptr); if (fs && isFScr) return false; else if (!fs && !isFScr) return false; if (fs) { ComPtr out; win.m_swapChain->GetContainingOutput(&out); DXGI_OUTPUT_DESC outDesc; out->GetDesc(&outDesc); win.m_swapChain->SetFullscreenState(true, nullptr); DXGI_MODE_DESC mdesc = {UINT(outDesc.DesktopCoordinates.right - outDesc.DesktopCoordinates.left), UINT(outDesc.DesktopCoordinates.bottom - outDesc.DesktopCoordinates.top)}; win.m_swapChain->ResizeTarget(&mdesc); } else win.m_swapChain->SetFullscreenState(false, nullptr); return true; } #endif if (m_ctx11.m_dev) { D3D11Context::Window& win = m_ctx11.m_windows[window]; BOOL isFScr; win.m_swapChain->GetFullscreenState(&isFScr, nullptr); if (fs && isFScr) return false; else if (!fs && !isFScr) return false; if (fs) { ComPtr out; win.m_swapChain->GetContainingOutput(&out); DXGI_OUTPUT_DESC outDesc; out->GetDesc(&outDesc); win.m_fsdesc.Width = outDesc.DesktopCoordinates.right; win.m_fsdesc.Height = outDesc.DesktopCoordinates.bottom; } win.m_fs = fs; win.m_needsFSTransition = true; return true; } OGLContext::Window& win = m_ctxOgl.m_windows[window]; if (fs && win.m_fs) return false; else if (!fs && !win.m_fs) return false; if (fs) { win.m_fsStyle = GetWindowLong(win.m_hwnd, GWL_STYLE); win.m_fsExStyle = GetWindowLong(win.m_hwnd, GWL_EXSTYLE); GetWindowRect(win.m_hwnd, &win.m_fsRect); SetWindowLong(win.m_hwnd, GWL_STYLE, win.m_fsStyle & ~(WS_CAPTION | WS_THICKFRAME)); SetWindowLong(win.m_hwnd, GWL_EXSTYLE, win.m_fsExStyle & ~(WS_EX_DLGMODALFRAME | WS_EX_WINDOWEDGE | WS_EX_CLIENTEDGE | WS_EX_STATICEDGE)); MONITORINFO monitor_info; monitor_info.cbSize = sizeof(monitor_info); GetMonitorInfo(MonitorFromWindow(win.m_hwnd, MONITOR_DEFAULTTONEAREST), &monitor_info); SetWindowPos(win.m_hwnd, NULL, monitor_info.rcMonitor.left, monitor_info.rcMonitor.top, monitor_info.rcMonitor.right - monitor_info.rcMonitor.left, monitor_info.rcMonitor.bottom - monitor_info.rcMonitor.top, SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED | SWP_ASYNCWINDOWPOS); win.m_fs = true; } else { SetWindowLong(win.m_hwnd, GWL_STYLE, win.m_fsStyle); SetWindowLong(win.m_hwnd, GWL_EXSTYLE, win.m_fsExStyle); SetWindowPos(win.m_hwnd, NULL, win.m_fsRect.left, win.m_fsRect.top, win.m_fsRect.right - win.m_fsRect.left, win.m_fsRect.bottom - win.m_fsRect.top, SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED | SWP_ASYNCWINDOWPOS); win.m_fs = false; } return true; } }; struct HWNDEvent { UINT uMsg; WPARAM wParam; LPARAM lParam; HWNDEvent(UINT m, WPARAM w, LPARAM l) : uMsg(m), wParam(w), lParam(l) {} }; struct Win32Cursors { HCURSOR m_arrow; HCURSOR m_hResize; HCURSOR m_vResize; HCURSOR m_wait; }; namespace boo { extern Win32Cursors WIN32_CURSORS; } #endif // BOO_WIN32COMMON_HPP