1

我有一个用 C 语言编写的小示例程序,它使用 XCB API 打开一个窗口。

严格来说,在我创建并显示了窗口之后,我会(稍后)想要隐藏窗口。

(很明显,在这个特定的例子中,我可以删除对 xcb_map_window 的调用,并且窗口将被隐藏,但我想稍后在我的较大应用程序中执行此操作,例如显示/隐藏窗口的切换,注意:我不想最小化它)。

这是示例代码(注意:由于答案,此代码现在可以工作):

#include <unistd.h>
#include <stdio.h>
#include <stdbool.h>
#include <xcb/xcb.h>

void set_window_visible(xcb_connection_t* c, xcb_window_t win, bool visible) {
    xcb_generic_event_t *event;

    if(visible) {
        // Map the window on the screen
        xcb_map_window (c, win);

        // Make sure the map window command is sent
        xcb_flush(c);

        // Wait for EXPOSE event.
        //
        // TODO: add timeout in-case X server does not ever send the expose event.
        while(event = xcb_wait_for_event(c)) {
            bool gotExpose = false;

            switch(event->response_type & ~0x80) {
            case XCB_EXPOSE:
                gotExpose = true;
                break;

            default:
                break; // We don't know the event type, then.
            }
            free(event);

            if(gotExpose) {
                break;
            }
        }

    } else {
        // Hide the window
        xcb_unmap_window(c, win);

        // Make sure the unmap window command is sent
        xcb_flush(c);
    }
}

int main() {
    xcb_connection_t *c;
    xcb_screen_t     *screen;
    xcb_window_t      win;
    xcb_generic_event_t *event;

    // Open the connection to the X server
    c = xcb_connect (NULL, NULL);

    // Get the first screen
    screen = xcb_setup_roots_iterator (xcb_get_setup (c)).data;

    // Ask for our window's Id
    win = xcb_generate_id(c);

    // Create the window
    uint32_t mask = XCB_CW_EVENT_MASK;
    uint32_t valwin[] = {XCB_EVENT_MASK_EXPOSURE | XCB_BUTTON_PRESS};
    xcb_create_window(
        c,                             // Connection
        XCB_COPY_FROM_PARENT,          // depth (same as root)
        win,                           // window Id
        screen->root,                  // parent window
        0, 0,                          // x, y
        150, 150,                      // width, height
        10,                            // border_width
        XCB_WINDOW_CLASS_INPUT_OUTPUT, // class
        screen->root_visual,           // visual
        mask, valwin                   // masks
    );

    bool visible = true;
    set_window_visible(c, win, true);

    while(1) {
        sleep(2);

        // Toggle visibility
        visible = !visible;
        set_window_visible(c, win, visible);

        printf("Window visible: ");
        if(visible) {
            printf("true.\n");
        } else {
            printf("false.\n");
        }
    }

    // pause until Ctrl-C
    pause();
    return 0;
}

我编译和运行的是:

gcc xcbwindow.c -o xcbwindow -lxcb
./xcbwindow

从我在谷歌或这里可以找到的任何内容中,我做的一切都是正确的。因此,为了澄清起见,我使用的是 Unity 和 Ubuntu 12.04 LTS:

unity --version 报告:

unity 5.20.0

uname -a 报告:

Linux [redacted] 3.2.0-32-generic #51-Ubuntu SMP Wed Sep 26 21:33:09 UTC 2012 x86_64 x86_64 x86_64 GNU/Linux

谁能解释我在这段代码中哪里出错了?

编辑:在 xcb_unmap_window() 之后使用 flush() 更新代码;还是不行。

EDIT2:尝试使用肉桂 WM 的代码;仍然不起作用(这不是 Unity 错误)。

EDIT3:这篇文章中更新的代码现在可以工作了。

4

1 回答 1

2

您的程序运行得太快了。

它映射窗口,然后立即取消映射。窗口是顶级的,这意味着请求被重定向到窗口管理器。但是窗口管理器在窗口尚未映射时收到取消映射请求,因此它只是丢弃该请求。sleep(3)在 map 和 unmap 调用之间插入并观察。

在实际代码中,您的窗口需要在发出取消映射请求之前至少获得一个公开事件。这保证了它实际上是由窗口管理器映射的。

于 2013-10-17T16:18:24.033 回答