Xlib 應用被覆蓋遮擋或者移出屏幕時,獲取窗口截圖


https://www.cnblogs.com/chaichengxun/p/15409996.html這篇文章里已經講過,如果應用窗口位於屏幕外或者被遮擋,xlib不能獲取完整的應用窗口截屏,如下

今天就來解決這個問題。

用到的頭文件:

#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/extensions/Xcomposite.h>
#include <X11/extensions/Xrender.h>

用到的庫:

-lXlib -lXcomposite -lXrender

代碼:

static Display *xdisplay = 0;

Display *disp()
{
    if (!xdisplay)
        xdisplay = XOpenDisplay(NULL);

    return xdisplay;
}

XImage* GetDrawableThumbnail(Drawable src_drawable,
                             Visual* visual,
                             int src_width,
                             int src_height) {
    XRenderPictFormat* format = XRenderFindVisualFormat(disp(),
                                                        visual);
    if (!format) {
        cout <<"XRenderFindVisualFormat() failed";
        return NULL;
    }

    XRenderPictureAttributes pa;
    pa.subwindow_mode = IncludeInferiors;
    Picture src = XRenderCreatePicture(disp(),
                                       src_drawable,
                                       format,
                                       CPSubwindowMode,
                                       &pa);
    if (!src) {
        cout << "XRenderCreatePicture() failed";
        return NULL;
    }

    Pixmap dst_pixmap = XCreatePixmap(disp(),
                                      src_drawable,
                                      src_width,
                                      src_height,
                                      format->depth);
    if (!dst_pixmap) {
        cout << "XCreatePixmap() failed";
        XRenderFreePicture(disp(), src);
        return NULL;
    }
    Picture dst = XRenderCreatePicture(disp(), dst_pixmap, format, 0, NULL);
    if (!dst) {
        cout << "XRenderCreatePicture() failed";
        XFreePixmap(disp(), dst_pixmap);
        XRenderFreePicture(disp(), src);
        return NULL;
    }

    XRenderColor transparent = {0};
    XRenderFillRectangle(disp(),
                         PictOpSrc,
                         dst,
                         &transparent,
                         0,
                         0,
                         src_width,
                         src_height);

    XTransform xform = { {
                             { XDoubleToFixed(1), XDoubleToFixed(0), XDoubleToFixed(0) },
                             { XDoubleToFixed(0), XDoubleToFixed(1), XDoubleToFixed(0) },
                             { XDoubleToFixed(0), XDoubleToFixed(0), XDoubleToFixed(1) }
                         } };
    XRenderSetPictureTransform(disp(), src, &xform);
    XRenderSetPictureFilter(disp(), src, FilterBest, NULL, 0);
    XRenderComposite(disp(),
                     PictOpSrc,
                     src,
                     None,
                     dst,
                     0,
                     0,
                     0,
                     0,
                     0,
                     0,
                     src_width,
                     src_height);
    XImage* image = XGetImage(disp(),
                              dst_pixmap,
                              0,
                              0,
                              src_width,
                              src_height,
                              AllPlanes, ZPixmap);
    return image;
}

將XImage轉換成QPixmap:

void AppListSelDlg::GetPixmap(unsigned long win, QPixmap &pixmap)
{
    Display *display_ = disp();

    XCompositeRedirectWindow(display_, win, CompositeRedirectAutomatic);
    Pixmap src_pixmap = XCompositeNameWindowPixmap(display_, win);
    if (!src_pixmap) {
        printf("XCompositeNameWindowPixmap() failed");
        return;
    }

    XWindowAttributes attr;
    if (!XGetWindowAttributes(display_, win, &attr)) {
        printf( "XGetWindowAttributes() failed");
        XFreePixmap(display_, src_pixmap);
        return ;
    }

    XImage* pImage = XCompcap::GetDrawableThumbnail(src_pixmap,
                                          attr.visual,
                                          attr.width,
                                          attr.height);

    QImage image = QImage((const uchar *)(pImage->data), pImage->width, pImage->height, pImage->bytes_per_line, QImage::Format_RGB32);
    pixmap = QPixmap::fromImage(image);
    XDestroyImage(pImage);
    XFreePixmap(display_, src_pixmap);
}

 

看一下實際效果:

 

 

可以看出,雖然終端遮擋住了文件夾,但是獲取到的圖像仍然是完整的圖像。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM