背景
需要在屏幕在居中位置显示一个对话框,由用户来进行决策;且此对话框是非模态对话框。
实现方式
1、顶层窗口是一个Window,此窗口设置屏幕居中,透明。
2、对话框设计为Dialog,再将此Dialog挂载在Window上。
这样,只要Windows可能居中、置顶即可。
结果发现
此对话框并不会置顶显示,会被其他窗口挡住。
顶层窗口
Window {
id:topWindow
property
int
__hintdlgWidth: 480
property
int
__hintdlgHeight: 320
visible:
true
x: (Screen.width - __hintdlgWidth) / 2
y: (Screen.height - __hintdlgHeight) / 2
width: __hintdlgWidth+20
height: __hintdlgHeight+20
color:
"#00000000"
flags: Qt.FramelessWindowHint | Qt.Window | Qt.WindowStaysOnTopHint
}
|
Qt.WindowStaysOnTopHint
按照官方说法,该属性仅将最小化的窗口,再次置顶显示;而不是将一个新创建的窗口直接置顶显示。
解决方案
方式一:在C++中重置窗口位置
windows:
setWindowFlags(windowFlags() | Qt::WindowStaysOnTopHint);
::SetWindowPos((
HWND
)winId(), HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
Linux:
setWindowFlags(windowFlags() | Qt::BypassWindowManagerHint);
或:
setWindowFlags(windowFlags() | Qt::X11BypassWindowManagerHint);
|
方式二:将创建最小化窗口,后将窗口显示
activateWindow();
setWindowState((windowState() & ~Qt::WindowMinimized) | Qt::WindowActive);
raise
();
//必须加,不然X11会不起作用
#ifdef Q_OS_WIN32 //windows必须加这个,不然windows10 会不起作用,具体参看activateWindow 函数的文档
HWND
hForgroundWnd = GetForegroundWindow();
DWORD
dwForeID = ::GetWindowThreadProcessId(hForgroundWnd, NULL);
DWORD
dwCurID = ::GetCurrentThreadId();
::AttachThreadInput(dwCurID, dwForeID, TRUE);
::SetForegroundWindow((
HWND
)winId());
::AttachThreadInput(dwCurID, dwForeID, FALSE);
#endif // MAC_OS
|
方式三:在QML中,在必要时再设置Window的风格
//激活窗口
topWindow.requestActivate();
topWindow.
raise
();
//置顶窗口
topWindow.flags |= Qt.WindowStaysOnTopHint
topWindow.show()
topWindow.flags &= ~Qt.WindowStaysOnTopHint
|