我正试图在Qt5.6.0中制作一个具有aero snap功能的无边界窗口。一切正常,除了当我最大化窗口时:它太大了。
我的屏幕分辨率是2560x1440
,所以窗口的大小应该是2560x1400
(任务栏为40像素),但在WM_SIZE
消息中,新的大小是2576x1416
。所以这个窗口在每个方向上都大了8个像素。这也意味着窗口没有在左上角对齐,它在两个方向上都离屏幕正好8个像素。
我找不到这个问题的解决方案,我所尝试的一切都不起作用,并导致错误。
唯一能解决这个问题的是删除WS_CAPTION
和WS_THICKFRAME
样式,但之后我就失去了areo快照功能。
我不得不告诉Qt或DWM,让窗口缩小16个像素,并将其向右和向下移动8个像素。有人知道怎么做吗?
我不得不告诉Qt或DWM,让窗口缩小16个像素并将其向右和向下移动8个像素。有人知道吗怎么做?
DWM是桌面窗口管理器吗?那么平台就是Windows。
只要它是关于Qt 5.6的,并且你很可能谈论的是设置了Qt::CustomizeWindowHint属性的小部件,那么Qt中有一个已知的错误,它还没有修复:
https://bugreports.qt.io/browse/QTBUG-4362
我偶然发现了几个错误,BiTOk在上面的链接中提出的解决方法对我很有效。
我的第一次尝试是将窗口几何体设置为可用的几何体:
QRect rect = QApplication::desktop()->availableGeometry();
setGeometry(rect.left() , rect.top(), rect.right(), rect.bottom());
唯一的问题是窗口的右侧和底部像素太小,
setGeometry(rect.left() , rect.top(), rect.right() + 1, rect.bottom() + 1);
给我一个错误:
QWindowsWindow::setGeometry: Unable to set geometry 2560x1400+0+0 on QWidgetWindow/'MainWindowWindow'. Resulting geometry: 2576x1416+-8+-8 (frame: 0, 0, 0, 0, custom margin: 0, 0, 0, 0, minimum size: 45x13, maximum size: 16777215x16777215)
然后我查看了Visual Studio 2015的矩形坐标,它们的大小与我实现的无边界窗口的大小相同,每个方向都大8个像素。
我可以给我的窗口的内容留出8的空白,这样如果窗口被最大化并设置窗口区域,它就不会从屏幕上弹出:
setContentsMargins({ 8, 8, 8, 8 });
HRGN WinRgn;
RECT winrect;
GetClientRect(hwnd, &winrect);
WinRgn = CreateRectRgn(8, 8, winrect.right - 8, winrect.bottom - 8);
SetWindowRgn(hwnd, WinRgn, true);
当窗口恢复时,我们需要重置以前的更改。结果是:
case WM_SIZE:
WINDOWPLACEMENT wp;
wp.length = sizeof(WINDOWPLACEMENT);
GetWindowPlacement(hwnd, &wp);
if (wp.showCmd == SW_MAXIMIZE) {
setContentsMargins({ 8, 8, 8, 8 });
HRGN WinRgn;
RECT winrect;
GetClientRect(hwnd, &winrect);
WinRgn = CreateRectRgn(8, 8, winrect.right - 8, winrect.bottom - 8);
SetWindowRgn(hwnd, WinRgn, true);
UpdateWindow(hwnd);
is_fullscreen = true;
} else {
if (is_fullscreen) {
setContentsMargins({ 0, 0, 0, 0 });
SetWindowRgn(hwnd, NULL, true);
is_fullscreen = false;
}
}
break;
其他帖子已经回答了这个问题,但我只想补充一点,使用GetSystemMetrics
而不是硬编码值8
可能是个好主意。
示例
#include <Windows.h>
void MyWindow::changeEvent(QEvent* ev) {
if (ev->type() == QEvent::WindowStateChange) {
const auto state = windowState();
if(state & Qt::WindowMaximized) {
const int x = GetSystemMetrics(SM_CXFRAME) + GetSystemMetrics(SM_CXPADDEDBORDER);
const int y = GetSystemMetrics(SM_CYFRAME) + GetSystemMetrics(SM_CXPADDEDBORDER);
setContentsMargins({x, y, x, y});
}
else {
setContentsMargins({0, 0, 0, 0});
}
}