SDL2:如何在调整窗口大小时保持纵横比

时间:2015-02-18 15:45:01

标签: c++ linux sdl sdl-2

我正在尝试创建一个SDL窗口,在调整大小事件时保持其宽高比。如果用户加宽窗口,则高度增加,反之亦然。我捕获SDL_WINDOWEVENT_RESIZED事件,计算保持宽高比的新宽度或高度,然后使用计算值调用SDL_SetWindowSize()。

问题是在事件轮询循环内调用SDL_SetWindowSize()函数不会在屏幕上执行任何操作。 SDL会更新窗口大小变量(在我的主循环中调用SDL_GetWindowSize()会返回更新的窗口尺寸)。但是,实际窗口不会更新。

我能让它工作的唯一方法是在主循环中不断调用SDL_SetWindowSize(),但我认为这是错误的做事方式。下面的代码说明了我的问题。是否有一种更好,更清洁的方法来实现这一目标?

我正在使用SDL 2.0.3和64位Ubuntu Linux和GNOME桌面。

#include <SDL2/SDL.h>

static const float ASPECT_RATIO = 16.f/9.f;

SDL_Window* window;
SDL_Renderer* renderer;

uint32_t windowID;
SDL_Rect screen;
bool done = false;
bool resizeDone = false;

void handle_events()
{
    SDL_Event e;
    while (SDL_PollEvent(&e)) {
        switch (e.type) {

        case SDL_WINDOWEVENT: 
            if(e.window.windowID == windowID) {
                switch(e.window.event) {
                    case SDL_WINDOWEVENT_RESIZED: { 
                        int width = e.window.data1;
                        int height = e.window.data2;
                        float aspectRatio = (float)width/(float)height;

                        if(aspectRatio != ASPECT_RATIO) {
                            if(aspectRatio > ASPECT_RATIO) {
                                height = (1.f / ASPECT_RATIO) * width; 
                            }
                            else {
                                width = ASPECT_RATIO * height; 
                            }

                            printf("Setting window size to %d, %d, aspect ratio: %f\n", 
                                width, height, (float)width/(float)height);
                    }
                    screen.w = width;
                    screen.h = height; 

                    SDL_SetWindowSize(window, width, height); // <-- does not work
                    resizeDone = true;
                    break;
                    }
                }
             }
             break;

        case SDL_QUIT:
            done = true;
            break;

        default:
            break;
        }
    }   
}

void run() {
    while(!done) {
        //SDL_SetWindowSize(window, screen.w, screen.h); // <-- works
        handle_events();
        SDL_RenderClear(renderer);
        SDL_RenderPresent(renderer);

        if(resizeDone) {
            int w, h;
            SDL_GetWindowSize(window, &w, &h);
            printf("SDL_GetWindowSize: %d, %d\n", w, h);
            resizeDone = false;
        }
    }
}

int main(int, char**) {
    SDL_Init(SDL_INIT_VIDEO);
    uint32_t window_flags = SDL_WINDOW_SHOWN | SDL_WINDOW_ALLOW_HIGHDPI | SDL_WINDOW_RESIZABLE;
    window = SDL_CreateWindow("Test", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, 1280, 720, window_flags);
    windowID = SDL_GetWindowID(window);
    renderer = SDL_CreateRenderer(window, -1, 0);
    SDL_SetRenderDrawColor(renderer, 255, 0, 0, 255);
    run();
    SDL_Quit();
    return 0;
}

2 个答案:

答案 0 :(得分:1)

某些窗口管理器似乎忽略了在WM自身调整窗口大小时所做的调整大小请求(例如,当鼠标按钮保持时)。相反,SDL_GetWindowSize返回缓存值,在特定情况下有时会出错。

除了在每个帧上不断调用SDL_SetWindowSize之外,我认为没有与平台无关的方法来实现这一点,以防万一。但是,它可以使用特定于平台的API实现(例如SDL_GetWindowSysWMInfo然后使用Xlib)。

答案 1 :(得分:0)

macOS 上,我已经这样解决了:

cocoa.m:

#import <Cocoa/Cocoa.h>

void SetWindowRatio(void *window) {
    NSWindow *win = (__bridge NSWindow*) window;
    win.aspectRatio = NSMakeSize( 1280, 720 );
}

main.cpp:

#include <SDL.h>
#include <SDL_syswm.h>

extern "C" void SetWindowRatio(void *window);

// and later..

SDL_SysWMinfo wmInfo;
SDL_VERSION(&wmInfo.version);
SDL_GetWindowWMInfo(sdl.window, &wmInfo);
SetWindowRatio(wmInfo.info.cocoa.window);

也许可以在Linux上完成类似的操作,仅访问wmInfo.info的不同部分。并调用本机函数?