如何在 XLib 中创建半透明白色 window

How to create semi transparent white window in XLib

我想在 XLib 中创建一个半透明的白色 window,但是 window 不是半透明的,它仍然是完全不透明的。我用的是compton compositor,系统里面有透明的windows,所以问题出在代码上:

#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <stdio.h>

int main(int argc, char* argv[])
{
    Display* display = XOpenDisplay(NULL);

    XVisualInfo vinfo;

    XMatchVisualInfo(display, DefaultScreen(display), 32, TrueColor, &vinfo);

    XSetWindowAttributes attr;
    attr.colormap = XCreateColormap(display, DefaultRootWindow(display), vinfo.visual, AllocNone);
    attr.border_pixel = 0;
    attr.background_pixel = 0x80ffffff;

    Window win = XCreateWindow(display, DefaultRootWindow(display), 0, 0, 300, 200, 0, vinfo.depth, InputOutput, vinfo.visual, CWColormap | CWBorderPixel | CWBackPixel, &attr);
    XSelectInput(display, win, StructureNotifyMask);
    GC gc = XCreateGC(display, win, 0, 0);

    Atom wm_delete_window = XInternAtom(display, "WM_DELETE_WINDOW", 0);
    XSetWMProtocols(display, win, &wm_delete_window, 1);

    XMapWindow(display, win);

    int keep_running = 1;
    XEvent event;

    while (keep_running) {
        XNextEvent(display, &event);

        switch(event.type) {
            case ClientMessage:
                if (event.xclient.message_type == XInternAtom(display, "WM_PROTOCOLS", 1) && (Atom)event.xclient.data.l[0] == XInternAtom(display, "WM_DELETE_WINDOW", 1))
                    keep_running = 0;

                break;

            default:
                break;
        }
    }

    XDestroyWindow(display, win);
    XCloseDisplay(display);
    return 0;
}

您需要设置_NET_WM_WINDOW_OPACITY。这是在映射 window:

之前要添加的片段
double alpha = 0.8;
unsigned long opacity = (unsigned long)(0xFFFFFFFFul * alpha);
Atom XA_NET_WM_WINDOW_OPACITY = XInternAtom(display, "_NET_WM_WINDOW_OPACITY", False);
XChangeProperty(display, win, XA_NET_WM_WINDOW_OPACITY, XA_CARDINAL, 32,
                PropModeReplace, (unsigned char *)&opacity, 1L);

请注意,您应该添加 #include <X11/Xatom.h> 以获得 XA_CARDINAL 的声明。

我不完全确定这个界面有多稳定。不过,这似乎是对 unix 上 Extended Window Manager Hints specification but has not made it into any final revision from what I can see. I know that this is how Tk implements transparency support 的提议扩展。

结果如下:

X11 期望 预乘 颜色,即真正的不透明颜色需要乘以 alpha 值(并相应缩放,即当通道宽度为 8 位时除以 256) .当您需要组合多个级别时,这种格式更容易使用。请参阅公式 here。当一切都预乘时,计算量更少。

因此您需要将每个 R、G 和 B 通道乘以 alpha 值 (0x80),然后除以 256。

将背景设置为 0x80808080 可得到所需的结果:

注意结果与@patthoyts 的建议不同:这里只有 window 本身是半透明的,WM 装饰保持不透明; window 本身和装饰都由 WM 透明化(并且 WM 进行必要的颜色混合)。