【问题标题】:How to create semi transparent white window in XLib如何在 XLib 中创建半透明的白色窗口
【发布时间】:2017-02-15 19:07:41
【问题描述】:

我想在 XLib 中创建一个半透明的白色窗口,但窗口不是半透明的,它仍然是完全不透明的。我用的是compton合成器,系统有透明窗口,所以问题出在代码上:

#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;
}

【问题讨论】:

  • 请注意,OP 提到必须在您的系统上运行一个合成器程序才能使这些程序正常工作。 Lubuntu 不需要,需要 xcompmgr 或类似的。

标签: c x11 transparent alpha xlib


【解决方案1】:

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

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

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

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

【讨论】:

  • 不,不是那样。检查此图像:oscomp.hu/depot/white_transparency_qt5_vs_xlib.png 两个窗口的背景颜色相同:0x80ffffff; Qt5 就像我需要的那样,而 XLib 不是。它一定是代码中的某些东西,我做错了,但是什么?
  • @jsr 你试过这样做吗?看图像。你认为我是用 gimp 编辑的还是什么?
  • Qt 不是 XLib。它是一个可移植的工具包,适用于 X11、Windows GDI、原始 Linux 帧缓冲区等等。我不知道 Qt 在内部做了什么来取悦用户并保持一致的 API。 X11 完全符合您在我发布的图片中看到的效果。
  • Qt 使用 XRender 扩展。这是我的实际问题,而不是透明窗口:Qt 使用 XRenderComposite 将 alpha 图像正确地混合到 X11 窗口,而如果我使用 XRenderComposite,则会发生相同的效果:一个像素越接近白色,像素的透明度就越少.如果一个像素是完全白色的,那么 XRenderComposite 会在此处绘制一个白色像素,而不管该像素具有什么 alpha。我敢肯定,这不是 XRender 混合的假定行为......如果你能告诉我如何正确使用 XRenderComposite,那么我会很好。
  • 当然我尝试改变颜色的亮度:颜色越接近白色,alpha 的工作量越少,颜色越接近黑色,alpha 的工作量越大,但那是错误!如果我将 alpha 设置为零,那么无论颜色如何,颜色都应该是完全透明的。检查我的康普顿主题中的示例图片:github.com/chjj/compton/issues/382
【解决方案2】:

您需要设置_NET_WM_WINDOW_OPACITY。这是在映射窗口之前要添加的 sn-p:

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 &lt;X11/Xatom.h&gt; 以获得 XA_CARDINAL 的声明。

我不完全确定这个界面有多稳定。这似乎是对Extended Window Manager Hints 规范的提议扩展,但据我所知,它还没有进入任何最终修订版。我知道这就是 Tk implements transparency support 在 unix 上的方式。

结果如下:

【讨论】:

  • 不幸的是它没有奏效。我在 Xfce、TDE 和 MotifWM 中使用 compton 进行了尝试,它在所有环境中仍然完全是白色的。我有一个 Qt5 代码,它可以通过设置背景颜色的前 8 位来设置它的背景透明度。它之前唯一做的就是设置窗口的“WA_TranslucentBackground”属性。我仍然试图通过查看 Qt5 源来弄清楚它的作用,但没有运气。
  • @jsr 适用于最近的 KDE。我猜你可以把 Xfce 和 TDE 中的窗口管理器改成支持这个 atom 的东西。
  • 我需要一个通用的解决方案,而不是依赖于 WM 的解决方案。另外Qt5可以做半透明的白色窗口:oscomp.hu/depot/white_transparency_qt5_vs_xlib.png两个程序的背景颜色是相同的:0x80ffffff,而纯XLib的显示完全不透明的窗口。
猜你喜欢
  • 1970-01-01
  • 2018-06-18
  • 2021-09-17
  • 1970-01-01
  • 2016-03-03
  • 1970-01-01
  • 1970-01-01
  • 2015-04-25
  • 1970-01-01
相关资源
最近更新 更多