通过单击任务栏最小化窗口
似乎Qt::FramelessWindowHint
' 的实现是有限的。当这个标志被设置时,Windows 认为这个窗口不能被最小化或最大化。我已经尝试过在纯 winapi 中实现的这个解决方案。通过单击任务栏最小化和恢复无框窗口工作正常。显然,Qt 设置了一些阻止此功能的错误标志。可能有一个很好的理由,我不知道。
我们可以同时使用winapi和Qt,但是很麻烦。首先,应该在设置窗口标志并使用 Qt 显示窗口后执行 winapi 代码。否则 Qt 将覆盖窗口标志。
另一个问题是,当我们使用 winapi 删除边框时,窗口几何形状突然改变,而 Qt 不知道这一点。渲染和事件映射(包括鼠标点击位置)变得无效。我没有找到任何记录在案的方法来更新映射。我发现我们可以告诉 Qt 屏幕方向已经改变,它会强制它重新计算窗口几何形状。但这看起来像一个肮脏的黑客。Qt 4 中也缺少该QWidget::windowHandle
功能,Qt 5 中“可能会发生变化”。所以这种方法不可靠。但无论如何,它现在有效。这是应放置在顶部窗口类构造函数中的完整代码(在 Windows 8 中测试):
#include "windows.h"
#include <QWindow>
//...
show();
HWND hwnd = reinterpret_cast<HWND>(effectiveWinId());
LONG lStyle = GetWindowLong(hwnd, GWL_STYLE);
lStyle &= ~(WS_CAPTION | WS_THICKFRAME | WS_MINIMIZE | WS_MAXIMIZE | WS_SYSMENU);
SetWindowLong(hwnd, GWL_STYLE, lStyle);
setWindowFlags(windowFlags() | Qt::FramelessWindowHint);
windowHandle()->reportContentOrientationChange(Qt::PrimaryOrientation);
真正解决这个问题的方法是修改Window Qt平台插件(见Qt源码中的QWindowsWindow类)。可能有一种方法可以从默认实现继承,修改它并在您的应用程序中使用。您也可以询问 Qt 开发人员这种行为是合理的还是一个错误。我认为这个问题可以通过补丁解决。
如果您仍打算使用此代码并且还应支持其他操作系统,请不要忘记将特定于 windows 的实现包装在#ifdef Q_OS_WIN
.
仅在单击标题栏且未最大化窗口时启用窗口拖动
其他问题可以更容易地解决。当您处理鼠标事件以实现窗口拖动时,检查窗口状态和事件位置并在不需要时禁用移动。
void MainWindow::mousePressEvent(QMouseEvent *e) {
if (!isMaximized() &&
e->button() == Qt::LeftButton &&
ui->title->geometry().contains(e->pos())) {
window_drag_start_pos = e->pos();
}
}
void MainWindow::mouseReleaseEvent(QMouseEvent *e) {
window_drag_start_pos = QPoint(0, 0);
}
void MainWindow::mouseMoveEvent(QMouseEvent *e) {
if (!window_drag_start_pos.isNull()) {
move(pos() + e->pos() - window_drag_start_pos);
}
}
void MainWindow::on_minimize_clicked() {
showMinimized();
}
void MainWindow::on_maximize_clicked() {
if (isMaximized()) {
showNormal();
} else {
showMaximized();
}
}
这ui->title
是一个用于显示假标题栏的标签,QPoint window_drag_start_pos
是一个类变量。