如何使一个窗口始终在顶部?

时间:2013-12-22 19:33:00

标签: ubuntu x11 qt5 window-managers compiz

我已经创建了一个无框架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也可以解决问题。

3 个答案:

答案 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