天天看点

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

最近接手一个项目,需要UOS系统上实现屏幕和窗口捕获。

由于QT只提供了屏幕捕获功能,没有提供窗口捕获,于是就找到了老朋友——OBS(毕竟MacOS的屏幕捕获也是扒的OBS代码)。

幸运的是,UOS系统商店自带OBS,直接省去了编译环节。

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示
Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示
Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

从上图可以看出,OBS自带屏幕捕获和窗口捕获,窗口捕获也可以获取当前桌面上打开的页面。

试用了一下,窗口捕获时最小化窗口,会导致程序卡死,也就是说最小化时获取不到图像。问题不大,能用就行。

看了一下obs源码,相关代码在plugins->linux-capture里。

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

用到的主要是X11库。

整理了一下obs和网上的代码,实现屏幕捕获+窗口捕获。

包含头文件

调用库,在xx.pro文件里添加

窗口列举、获取窗口名称

将窗口显示到QListWidget中

获取整个屏幕图像并显示到QListWidget中

可以看出屏幕截图跟窗口截图的区别就是获取Window参数方法不同。使用XGetImage获取图像完全相同。

再进一步观察可以发现,在获取窗口Window时,先调用 RootWindow 获取当前屏幕的Window,然后使用 XGetWindowProperty 获取当前屏幕下的所有窗口。

最终实现效果:

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

捕获到了Desktop和QT这两个程序的窗口图像,以及整个屏幕的图像。

但是Desktop为什么是黑的?

原来XGetImage获取到的是窗口的可见部分,运行程序时,QT处于最大化,把整个Desktop都挡住了,获取到的图像就是全黑的,把QT缩小以后在运行一遍试试。

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

DDE Dock是dock栏的图像,被拉伸以后就这么奇怪。

可以看出Desktop页面也显示出来了,并且Dock栏和QT遮挡住的部分是黑色。

程序目标可以说达成一半了。

顺便说一个程序运行时出现的崩溃错误

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

Debug发现程序崩溃在XImage转QImage这里。传来的XImage是一个空指针,导致程序崩溃。

 QImage image = QImage((const uchar *)(pImage->data), pImage->width, pImage->height, pImage->bytes_per_line, QImage::Format_RGB32); 

崩溃在QT Creator窗口,当时的情况是

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

当我把QT最大化或者移动到桌面最中间时,程序又能正常运行了。

经过反复试验得出了一个结论,如果窗口有一部分位于屏幕外,XGetImage就会返回空指针。

修改一下代码,只截屏显示在桌面的部分,截去在屏幕外的部分

发现还是会崩溃,调试发现attrs.x和attrs.y永远都是0,并不是窗口的实际坐标。

谷歌一下才知道,要想获取当前窗口在屏幕上的坐标,需要用 XTranslateCoordinates 转换一下

x和y就是窗口的实际坐标。

改一下代码,运行成功

Linux X11获取屏幕截图和程序窗口截图,并通过QPixmap显示

大功告成(一半)。

对比OBS,OBS可以获取窗口的完整图像,哪怕窗口有一部分在屏幕外。还需要继续研究obs代码。。。

未完待续。

继续阅读