如何在 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 进行必要的颜色混合)。
我想在 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 进行必要的颜色混合)。