diff --git a/backends/graphics/opengl/opengl-graphics.cpp b/backends/graphics/opengl/opengl-graphics.cpp index 49ee47ab83c..50feccec4bc 100644 --- a/backends/graphics/opengl/opengl-graphics.cpp +++ b/backends/graphics/opengl/opengl-graphics.cpp @@ -955,17 +955,13 @@ void OpenGLGraphicsManager::handleResizeImpl(const int width, const int height, overlayWidth = MAX(overlayWidth, 256); overlayHeight = MAX(overlayHeight, 200); - // HACK: Reduce the size of the overlay on high DPI screens. - overlayWidth = fracToInt(overlayWidth * (intToFrac(90) / xdpi)); - overlayHeight = fracToInt(overlayHeight * (intToFrac(90) / ydpi)); - if (!_overlay || _overlay->getFormat() != _defaultFormatAlpha) { delete _overlay; _overlay = nullptr; _overlay = createSurface(_defaultFormatAlpha); assert(_overlay); - // We should NOT always filter the overlay with GL_LINEAR. + // We should NOT always filter the overlay with GL_LINEAR. // In previous versions we always did use GL_LINEAR to assure the UI // would be readable in case it needed to be scaled -- it would not affect it otherwise. // However in modern devices due to larger screen size the UI display looks blurry diff --git a/backends/graphics/openglsdl/openglsdl-graphics.cpp b/backends/graphics/openglsdl/openglsdl-graphics.cpp index f2c3fd6363c..433d51a7f91 100644 --- a/backends/graphics/openglsdl/openglsdl-graphics.cpp +++ b/backends/graphics/openglsdl/openglsdl-graphics.cpp @@ -287,7 +287,7 @@ void OpenGLSdlGraphicsManager::updateScreen() { void OpenGLSdlGraphicsManager::notifyVideoExpose() { } -void OpenGLSdlGraphicsManager::notifyResize(const int width, const int height) { +void OpenGLSdlGraphicsManager::notifyResize(const int w, const int h) { #if SDL_VERSION_ATLEAST(2, 0, 0) // We sometime get outdated resize events from SDL2. So check that the size we get // is the actual current window size. If not ignore the resize. @@ -298,9 +298,14 @@ void OpenGLSdlGraphicsManager::notifyResize(const int width, const int height) { // event is processed after recreating the window at the new resolution. int currentWidth, currentHeight; getWindowSizeFromSdl(¤tWidth, ¤tHeight); + uint scale; + getDpiScalingFactor(&scale); + int width = w * scale; + int height = h * scale; + debug(3, "req: %d x %d cur: %d x %d, scale: %d", width, height, currentWidth, currentHeight, scale); if (width != currentWidth || height != currentHeight) return; - // TODO: Implement high DPI support + handleResize(width, height, 90, 90); #else if (!_ignoreResizeEvents && _hwScreen && !(_hwScreen->flags & SDL_FULLSCREEN)) { @@ -422,7 +427,8 @@ bool OpenGLSdlGraphicsManager::setupMode(uint width, uint height) { _glContext = nullptr; } - uint32 flags = SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE; + uint32 flags = SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE | SDL_WINDOW_ALLOW_HIGHDPI; + if (_wantsFullScreen) { // On Linux/X11, when toggling to fullscreen, the window manager saves // the window size to be able to restore it when going back to windowed mode. @@ -459,8 +465,8 @@ bool OpenGLSdlGraphicsManager::setupMode(uint width, uint height) { notifyContextCreate(rgba8888, rgba8888); int actualWidth, actualHeight; getWindowSizeFromSdl(&actualWidth, &actualHeight); - // TODO: Implement high DPI support - handleResize(actualWidth, actualHeight, 90, 90); + + handleResize(actualWidth, actualHeight, 72, 72); return true; #else // WORKAROUND: Working around infamous SDL bugs when switching @@ -507,6 +513,7 @@ bool OpenGLSdlGraphicsManager::setupMode(uint width, uint height) { if (_hwScreen) { notifyContextCreate(rgba8888, rgba8888); + // TODO: hidpi handleResize(_hwScreen->w, _hwScreen->h, 90, 90); } diff --git a/backends/graphics/openglsdl/openglsdl-graphics.h b/backends/graphics/openglsdl/openglsdl-graphics.h index 3f75fa7923e..d06b64a01c1 100644 --- a/backends/graphics/openglsdl/openglsdl-graphics.h +++ b/backends/graphics/openglsdl/openglsdl-graphics.h @@ -60,7 +60,11 @@ protected: virtual bool saveScreenshot(const Common::String &filename) const override; - virtual int getGraphicsModeScale(int mode) const override { return 1; } + virtual int getGraphicsModeScale(int mode) const override { + uint scale; + getDpiScalingFactor(&scale); + return scale; + } private: bool setupMode(uint width, uint height); diff --git a/backends/graphics/sdl/sdl-graphics.cpp b/backends/graphics/sdl/sdl-graphics.cpp index 8edce253b1f..76e4f3510d0 100644 --- a/backends/graphics/sdl/sdl-graphics.cpp +++ b/backends/graphics/sdl/sdl-graphics.cpp @@ -208,6 +208,11 @@ bool SdlGraphicsManager::notifyMousePosition(Common::Point &mouse) { mouse.y = CLIP(mouse.y, 0, _windowHeight - 1); int showCursor = SDL_DISABLE; + // DPI aware scaling to mouse position + uint scale; + getDpiScalingFactor(&scale); + mouse.x *= scale; + mouse.y *= scale; bool valid = true; if (_activeArea.drawRect.contains(mouse)) { _cursorLastInActiveArea = true; @@ -267,6 +272,9 @@ bool SdlGraphicsManager::createOrUpdateWindow(int width, int height, const Uint3 return false; } + // width *=3; + // height *=3; + // We only update the actual window when flags change (which usually means // fullscreen mode is entered/exited), when updates are forced so that we // do not reset the window size whenever a game makes a call to change the diff --git a/backends/graphics/sdl/sdl-graphics.h b/backends/graphics/sdl/sdl-graphics.h index 92fc4d32bf0..4ddc015810d 100644 --- a/backends/graphics/sdl/sdl-graphics.h +++ b/backends/graphics/sdl/sdl-graphics.h @@ -163,7 +163,8 @@ protected: void getWindowSizeFromSdl(int *width, int *height) const { #if SDL_VERSION_ATLEAST(2, 0, 0) assert(_window); - SDL_GetWindowSize(_window->getSDLWindow(), width, height); + SDL_GL_GetDrawableSize(_window->getSDLWindow(), width, height); + // SDL_GetWindowSize(_window->getSDLWindow(), width, height); #else assert(_hwScreen); @@ -176,6 +177,57 @@ protected: } #endif } + /** + * Gets the display index that the ScummVM window is on + */ + void getWindowDisplayIndexFromSdl(int *index) const { +#if SDL_VERSION_ATLEAST(2, 0, 0) + assert(_window); + *index = SDL_GetWindowDisplayIndex(_window->getSDLWindow()); +#else + *index = 0; +#endif + } + + void getDisplayDpiFromSdl(float *dpi, float *defaultDpi) const { + const float systemDpi = +#ifdef __APPLE__ + 72.0f; +#elif defined(_WIN32) + 96.0f; +#else + 90.0f; // ScummVM default +#endif + if (defaultDpi) + *defaultDpi = systemDpi; +#if SDL_VERSION_ATLEAST(2, 0, 0) + int displayIndex = 0; + getWindowDisplayIndexFromSdl(&displayIndex); + + if (SDL_GetDisplayDPI(displayIndex, NULL, dpi, NULL) != 0) { + if (dpi) + *dpi = systemDpi; + } +#else + *dpi = systemDpi; +#endif + } + + /** + * Returns the scaling mode based on the display DPI + */ + void getDpiScalingFactor(uint *scale) const { + float dpi, defaultDpi, ratio; + + getDisplayDpiFromSdl(&dpi, &defaultDpi); + debug(4, "dpi: %g default: %g", dpi, defaultDpi); + ratio = dpi / defaultDpi; + if (ratio >= 2.0f) { + *scale = 2; + } else { + *scale = 1; + } + } virtual void setSystemMousePosition(const int x, const int y) override; diff --git a/dists/macosx/Info.plist b/dists/macosx/Info.plist index d041203d325..676e2ce5f49 100644 --- a/dists/macosx/Info.plist +++ b/dists/macosx/Info.plist @@ -78,5 +78,7 @@ ScummVM saves and loads savegames in the Documents folder by default. SUPublicDSAKeyFile dsa_pub.pem + NSHighResolutionCapable + diff --git a/dists/macosx/Info.plist.in b/dists/macosx/Info.plist.in index 66c0072ec52..62ffd5c6f2d 100644 --- a/dists/macosx/Info.plist.in +++ b/dists/macosx/Info.plist.in @@ -78,5 +78,7 @@ ScummVM saves and loads savegames in the Documents folder by default. SUPublicDSAKeyFile dsa_pub.pem + NSHighResolutionCapable +