#if defined(Hiro_Window) namespace hiro { static auto CALLBACK Window_windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> LRESULT { if(Application::state().quit) return DefWindowProc(hwnd, msg, wparam, lparam); if(auto window = (mWindow*)GetWindowLongPtr(hwnd, GWLP_USERDATA)) { if(auto self = window->self()) { if(auto result = self->windowProc(hwnd, msg, wparam, lparam)) { return result(); } } } return Shared_windowProc(DefWindowProc, hwnd, msg, wparam, lparam); } //warning: do not add WS_CLIPCHILDREN; this will break painting of Frame ("BUTTON" BS_GROUPBOX) controls static const uint PopupStyle = WS_POPUP; static const uint FixedStyle = WS_SYSMENU | WS_CAPTION | WS_MINIMIZEBOX | WS_BORDER; static const uint ResizableStyle = WS_SYSMENU | WS_CAPTION | WS_MINIMIZEBOX | WS_MAXIMIZEBOX | WS_THICKFRAME; uint pWindow::minimumStatusHeight = 0; auto pWindow::initialize() -> void { pApplication::state().modalTimer.setInterval(1); pApplication::state().modalTimer.onActivate([] { Application::doMain(); }); HWND hwnd = CreateWindow(L"hiroWindow", L"", ResizableStyle, 128, 128, 256, 256, 0, 0, GetModuleHandle(0), 0); HWND hstatus = CreateWindow(STATUSCLASSNAME, L"", WS_CHILD, 0, 0, 0, 0, hwnd, nullptr, GetModuleHandle(0), 0); SetWindowPos(hstatus, nullptr, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED); RECT rc; GetWindowRect(hstatus, &rc); minimumStatusHeight = rc.bottom - rc.top; DestroyWindow(hstatus); DestroyWindow(hwnd); } auto pWindow::construct() -> void { hwnd = CreateWindow(L"hiroWindow", L"", ResizableStyle, 128, 128, 256, 256, 0, 0, GetModuleHandle(0), 0); SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)&reference); setDroppable(state().droppable); setGeometry({128, 128, 256, 256}); windows.append(self().instance); } auto pWindow::destruct() -> void { if(auto position = windows.find(self().instance)) windows.remove(*position); if(hbrush) { DeleteObject(hbrush); hbrush = nullptr; } DestroyWindow(hwnd); } auto pWindow::append(sMenuBar menuBar) -> void { } auto pWindow::append(sSizable sizable) -> void { } auto pWindow::append(sStatusBar statusBar) -> void { } auto pWindow::focused() const -> bool { return (GetForegroundWindow() == hwnd); } auto pWindow::frameMargin() const -> Geometry { RECT rc{0, 0, 640, 480}; uint style = state().fullScreen ? 0 : state().resizable ? ResizableStyle : FixedStyle; bool menuVisible = state().menuBar && state().menuBar->visible(); AdjustWindowRect(&rc, style, menuVisible); auto& efb = state().fullScreen ? settings.efbPopup : !state().resizable ? settings.efbFixed : settings.efbResizable; return { abs(rc.left) - efb.x, abs(rc.top) - efb.y, (rc.right - rc.left) - 640 - efb.width, (rc.bottom - rc.top) + _statusHeight() - 480 - efb.height }; } auto pWindow::handle() const -> uintptr_t { return (uintptr_t)hwnd; } auto pWindow::monitor() const -> uint { //TODO return 0; } auto pWindow::remove(sMenuBar menuBar) -> void { } auto pWindow::remove(sSizable sizable) -> void { } auto pWindow::remove(sStatusBar statusBar) -> void { } auto pWindow::setBackgroundColor(Color color) -> void { hbrushColor = CreateRGB(color); if(hbrush) { DeleteObject(hbrush); hbrush = nullptr; } if(color) hbrush = CreateSolidBrush(hbrushColor); } auto pWindow::setDismissable(bool dismissable) -> void { } auto pWindow::setDroppable(bool droppable) -> void { DragAcceptFiles(hwnd, droppable); } auto pWindow::setEnabled(bool enabled) -> void { if(auto& sizable = state().sizable) { if(auto self = sizable->self()) self->setEnabled(sizable->enabled(true)); } } auto pWindow::setFocused() -> void { if(!self().visible()) self().setVisible(true); SetFocus(hwnd); } auto pWindow::setFont(const Font& font) -> void { if(auto& sizable = state().sizable) { if(auto self = sizable->self()) self->setFont(sizable->font(true)); } } auto pWindow::setFullScreen(bool fullScreen) -> void { auto lock = acquire(); auto style = GetWindowLongPtr(hwnd, GWL_STYLE) & WS_VISIBLE; if(fullScreen) { windowedGeometry = self().geometry(); HMONITOR monitor = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST); MONITORINFOEX info; memset(&info, 0, sizeof(MONITORINFOEX)); info.cbSize = sizeof(MONITORINFOEX); GetMonitorInfo(monitor, &info); RECT rc = info.rcMonitor; Geometry geometry = {rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top}; SetWindowLongPtr(hwnd, GWL_STYLE, style | WS_POPUP); Geometry margin = frameMargin(); self().setGeometry({ geometry.x() + margin.x(), geometry.y() + margin.y(), geometry.width() - margin.width(), geometry.height() - margin.height() }); } else { SetWindowLongPtr(hwnd, GWL_STYLE, style | (state().resizable ? ResizableStyle : FixedStyle)); self().setGeometry(windowedGeometry); } } auto pWindow::setGeometry(Geometry geometry) -> void { auto lock = acquire(); Geometry margin = frameMargin(); auto& efb = state().fullScreen ? settings.efbPopup : !state().resizable ? settings.efbFixed : settings.efbResizable; SetWindowPos( hwnd, nullptr, geometry.x() - margin.x() - efb.x, geometry.y() - margin.y() - efb.y, geometry.width() + margin.width() + efb.width, geometry.height() + margin.height() + efb.height, SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED ); if(auto& statusBar = state().statusBar) { if(auto self = statusBar->self()) { SetWindowPos(self->hwnd, nullptr, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED); } } if(auto& sizable = state().sizable) { sizable->setGeometry(geometry.setPosition()); } } auto pWindow::setMaximized(bool maximized) -> void { if(state().minimized) return; auto lock = acquire(); ShowWindow(hwnd, maximized ? SW_MAXIMIZE : SW_SHOWNOACTIVATE); } auto pWindow::setMaximumSize(Size size) -> void { } auto pWindow::setMinimized(bool minimized) -> void { auto lock = acquire(); ShowWindow(hwnd, minimized ? SW_MINIMIZE : state().maximized ? SW_MAXIMIZE : SW_SHOWNOACTIVATE); } auto pWindow::setMinimumSize(Size size) -> void { } //never call this directly: use Window::setModal() instead //this function does not confirm the modality has actually changed before adjusting modalCount auto pWindow::setModal(bool modality) -> void { if(modality) { modalIncrement(); _modalityUpdate(); while(!Application::state().quit && state().modal) { if(Application::state().onMain) { Application::doMain(); } else { usleep(20 * 1000); } Application::processEvents(); } _modalityUpdate(); } else { modalDecrement(); } } auto pWindow::setResizable(bool resizable) -> void { auto style = GetWindowLongPtr(hwnd, GWL_STYLE) & WS_VISIBLE; SetWindowLongPtr(hwnd, GWL_STYLE, style | (state().resizable ? ResizableStyle : FixedStyle)); setGeometry(state().geometry); } auto pWindow::setTitle(string text) -> void { SetWindowText(hwnd, utf16_t(text)); } auto pWindow::setVisible(bool visible) -> void { auto lock = acquire(); ShowWindow(hwnd, visible ? SW_SHOWNORMAL : SW_HIDE); if(auto& sizable = state().sizable) { sizable->setGeometry(self().geometry().setPosition()); } if(!visible) self().setModal(false); //calculate window extended frame bounds: DwmGetWindowAttributes is only valid after the window is visible //by then, it's too late to position the window correctly, but we can cache the results here for next time //because GetWindowRect and DwmGetWindowAttribute returns different unit types, the hiro application *must* be DPI aware if(visible) { //in logical units RECT rc; GetWindowRect(hwnd, &rc); //in physical units RECT fc; DwmGetWindowAttribute(hwnd, DWMWA_EXTENDED_FRAME_BOUNDS, &fc, sizeof(RECT)); //convert to offsets useful to hiro auto& efb = state().fullScreen ? settings.efbPopup : !state().resizable ? settings.efbFixed : settings.efbResizable; efb.x = fc.left - rc.left; efb.y = fc.top - rc.top; efb.width = efb.x + (rc.right - fc.right); efb.height = efb.y + (rc.bottom - fc.bottom); //sanitize inputs: if the bounds are obviously nonsense, give up on trying to compensate for them if(efb.x > 100 || efb.y > 100 || efb.width > 100 || efb.height > 100) efb = {}; } } // auto pWindow::modalIncrement() -> void { if(pApplication::state().modalCount++ == 0) { pApplication::state().modalTimer.setEnabled(true); } } auto pWindow::modalDecrement() -> void { if(--pApplication::state().modalCount == 0) { pApplication::state().modalTimer.setEnabled(false); } } auto pWindow::windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> maybe { if(msg == WM_CLOSE || (msg == WM_KEYDOWN && wparam == VK_ESCAPE && state().dismissable)) { if(state().onClose) { self().doClose(); //doClose() may end up destroying the window when terminating the application ... //forcefully return early in said case, so that the modal check below doesn't access the destroyed pWindow object if(Application::state().quit) return true; } else { self().setVisible(false); } if(state().modal && !self().visible()) self().setModal(false); return true; } if(msg == WM_MOVE && !locked()) { state().geometry.setPosition(_geometry().position()); self().doMove(); } if(msg == WM_SIZE && !locked()) { if(auto statusBar = state().statusBar) { if(auto self = statusBar->self()) { SetWindowPos(self->hwnd, nullptr, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED); } } state().geometry.setSize(_geometry().size()); if(auto& sizable = state().sizable) { sizable->setGeometry(_geometry().setPosition()); } self().doSize(); } if(msg == WM_DROPFILES) { if(auto paths = DropPaths(wparam)) self().doDrop(paths); return false; } if(msg == WM_ERASEBKGND && hbrush) { RECT rc; GetClientRect(hwnd, &rc); PAINTSTRUCT ps; BeginPaint(hwnd, &ps); FillRect(ps.hdc, &rc, hbrush); EndPaint(hwnd, &ps); return true; } if(msg == WM_ENTERMENULOOP || msg == WM_ENTERSIZEMOVE) { modalIncrement(); return false; } if(msg == WM_EXITMENULOOP || msg == WM_EXITSIZEMOVE) { modalDecrement(); return false; } if(msg == WM_SYSCOMMAND) { if(wparam == SC_SCREENSAVE || wparam == SC_MONITORPOWER) { if(!Application::screenSaver()) return 0; } } return {}; } // auto pWindow::_geometry() -> Geometry { Geometry margin = frameMargin(); RECT rc; if(IsIconic(hwnd)) { //GetWindowRect returns x=-32000,y=-32000 when window is minimized WINDOWPLACEMENT wp; GetWindowPlacement(hwnd, &wp); rc = wp.rcNormalPosition; } else { GetWindowRect(hwnd, &rc); } auto& efb = state().fullScreen ? settings.efbPopup : !state().resizable ? settings.efbFixed : settings.efbResizable; auto x = rc.left + margin.x() + efb.x; auto y = rc.top + margin.y() + efb.y; auto width = (rc.right - rc.left) - margin.width() - efb.width; auto height = (rc.bottom - rc.top) - margin.height() - efb.height; return {x, y, width, height}; } auto pWindow::_modalityCount() -> unsigned { unsigned modalWindows = 0; for(auto& weak : windows) { if(auto object = weak.acquire()) { if(auto window = dynamic_cast(object.data())) { if(window->modal()) modalWindows++; } } } return modalWindows; } auto pWindow::_modalityDisabled() -> bool { if(_modalityCount() == 0) return false; return !state().modal; } auto pWindow::_modalityUpdate() -> void { unsigned modalWindows = _modalityCount(); for(auto& weak : windows) { if(auto object = weak.acquire()) { if(auto window = dynamic_cast(object.data())) { if(auto self = window->self()) { bool enabled = !modalWindows || window->modal(); if(IsWindowEnabled(self->hwnd) != enabled) { EnableWindow(self->hwnd, enabled); } } } } } } auto pWindow::_statusHeight() const -> int { int height = 0; if(auto& statusBar = state().statusBar) { if(statusBar->visible()) { auto& text = statusBar->state.text; height = statusBar->font(true).size(text ? text : " ").height(); height = max(height, minimumStatusHeight); } } return height; } } #endif