我已经创建了一个无框架Qt / QML窗口,我真的想知道任何设置其“Always On Top”系统菜单标志的程序化方法。单击 ALT + SPACE 我可以打开无框窗口的系统菜单,单击“Always On Top”选项,窗口始终保持在顶部,但我没有找到编程方式做同样的事。 Qt.WindowStaysOnTopHint
不起作用,尝试wmctrl -r "window name" -b add,above
也不起作用,即使wmctrl
适用于其他窗口也是如此。 wmctrl
不适用于我感兴趣的窗口显然与N/A
上的计算机名称列的wmctrl -l
有关:
francisco@Ubuntu:~$ wmctrl -l
0x02600006 0 Ubuntu Área de trabalho
0x03c00002 0 Ubuntu XdndCollectionWindowImp
0x03c00005 0 Ubuntu unity-launcher
0x03c00008 0 Ubuntu unity-panel
0x03c0000b 0 Ubuntu unity-dash
0x03c0000c 0 Ubuntu Hud
0x046000b3 0 Ubuntu How to make a window aways on top? - Stack Overflow - Mozilla Firefox
0x0520000b 0 N/A Qt Creator
0x05002396 0 Ubuntu francisco@Ubuntu: ~
0x0540000b 0 N/A backlight
我也经历了this procedure但是对于用户提出的要求,它也不适合我,同样的行为。 _NET_WM_STATE_ABOVE
已设置,但是聚焦窗口然后再次检查标记它不再存在,只有在单击系统菜单时它才是粘性的。
这是QML:https://gist.github.com/oblitum/8050586
相关问题:https://askubuntu.com/questions/394998
在相关的askubuntu问题中,发现wmctrl上应该存在一个错误,用于通过名称定位某些窗口。使用wmctrl -i -r <window id> -b add,above
也可以解决问题。
答案 0 :(得分:6)
EWMH规范明确指出:
_NET_WM_STATE_ABOVE和_NET_WM_STATE_BELOW主要用于用户首选项,不应由应用程序使用,例如引起对其对话框的注意(在这种情况下应使用紧急提示,请参阅the section called “Urgency”)。
因此,窗口管理员没有责任尊重直接设置此属性的应用程序(即由XChangeProperty)自行设置。只能通过向根窗口发送客户端消息来更改此属性 哪个窗口管理员会监听。
我不知道如何在像Qt这样的高级gui工具包中做到这一点,但这里是如何在普通的X11中做到这一点。(参见EWMH规范,或者_wnck_change_state进行示例实现)。 / p>
//file: test.c
//to build it, run
//shell> gcc test.c -lX11
#include <X11/Xlib.h>
#define _NET_WM_STATE_REMOVE 0 /* remove/unset property */
#define _NET_WM_STATE_ADD 1 /* add/set property */
#define _NET_WM_STATE_TOGGLE 2 /* toggle property */
// change a window's _NET_WM_STATE property so that it can be kept on top.
// @display: x11 display singleton.
// @xid : the window to set on top.
Status x11_window_set_on_top (Display* display, Window xid)
{
XEvent event;
event.xclient.type = ClientMessage;
event.xclient.serial = 0;
event.xclient.send_event = True;
event.xclient.display = display;
event.xclient.window = xid;
event.xclient.message_type = XInternAtom (display, "_NET_WM_STATE", False);
event.xclient.format = 32;
event.xclient.data.l[0] = _NET_WM_STATE_ADD;
event.xclient.data.l[1] = XInternAtom (display, "_NET_WM_STATE_ABOVE", False);
event.xclient.data.l[2] = 0; //unused.
event.xclient.data.l[3] = 0;
event.xclient.data.l[4] = 0;
return XSendEvent (display, DefaultRootWindow(display), False,
SubstructureRedirectMask|SubstructureNotifyMask, &event);
}
// a sample main function for testing.
// shell> ./a.out window_xid
int main (int argc, char** argv)
{
Window xid = strtol (argv[1], NULL, 0);
Display* display = XOpenDisplay (NULL);
x11_window_set_on_top (display, xid);
XFlush (display); //for simplicity, no event loops here.
XCloseDisplay (display);
}
另请注意,在某些x11环境(例如compiz)中,系统菜单由单独的装饰器程序而不是合成窗口管理器提供。
答案 1 :(得分:0)
在Go中,您就是这样做的:
import (
"github.com/BurntSushi/xgb"
"github.com/BurntSushi/xgb/xproto"
)
func (window *Window) AlwaysOnTop() {
xid := xproto.Window(window.WinId())
X, err := xgb.NewConn()
if err != nil {
log.Println(err)
return
}
defer X.Close()
state, err := xproto.InternAtom(X, false, uint16(len("_NET_WM_STATE")),
"_NET_WM_STATE").Reply()
if err != nil {
log.Println(err)
return
}
stateAbove, err := xproto.InternAtom(X, false,
uint16(len("_NET_WM_STATE_ABOVE")), "_NET_WM_STATE_ABOVE").Reply()
if err != nil {
log.Println(err)
return
}
evt := xproto.ClientMessageEvent{
Window: xid,
Format: 32,
Type: state.Atom,
Data: xproto.ClientMessageDataUnionData32New([]uint32{
_NET_WM_STATE_ADD,
uint32(stateAbove.Atom),
0,
0,
0,
}),
}
err = xproto.SendEventChecked(X, false, xproto.Setup(X).DefaultScreen(X).Root,
xproto.EventMaskSubstructureRedirect|xproto.EventMaskSubstructureNotify,
string(evt.Bytes())).Check()
if err != nil {
log.Println(err)
}
}
其中window.WinId()
是窗口的本机X11句柄。
答案 2 :(得分:-1)
我希望我能正确理解问题,因为您尝试通过菜单选项将QML视图启动到“Always On Top”模式。
我在Windows上尝试了下面的代码到我的主程序中,它对我来说总是在顶部显示窗口,所以我相信视图对象也可以从菜单选项中更改。
QApplication app(argc, argv);
QDeclarativeView viewer;
**viewer.setWindowFlags(Qt::WindowStaysOnTopHint);**
viewer.setSource(QUrl::fromLocalFile("TestView.qml"));
viewer.showNormal();
return app.exec();
谢谢, Zeeshan