当我在网络浏览器窗口中使用QPixmap::GrabWindow(WId)
时,它只会返回黑屏。
我正在使用以下代码:
QScreen *screen = QGuiApplication::primaryScreen();
m_pixmap = screen->grabWindow(hW);
m_image = m_pixmap.toImage();
m_image.save("p.png");
当我打开“p.png”时,它只是黑色图片。与其他窗口一起使用,效果很好。
如何获得浏览器的正常屏幕?
最佳答案
事实是QScreen::grabWindow使用Windows GDI捕获图像。这是一个相当古老的API,程序无需硬件加速即可使用(由处理器绘制)。 chrome-该软件不是很古老,并且已经通过Windows DXGI绘制了很久。
我已经写过使用该技术的软件。已发布示例代码here。它将由Qt 5.10库上的MSVC编译器编译,似乎没有什么不同,2015或2017。我的机器是64位的,也许这也很重要。
内部有两个类:FrameBroadcast和FrameCapturer。 FrameBroadcast从FrameCapturer请求一个具有特定时间间隔的屏幕快照,并通过信号void frameCaptured (QSharedPointer <Frame> frame);
发送订户QSharedPointer一旦超出所有插槽处理程序的范围,就会自动删除为屏幕内容分配的内存。
#include <QApplication>
#include <QObject>
#include <QPixmap>
#include <QImage>
#include <QDialog>
#include <QLabel>
#include "framebroadcast.h"
/*static Frame* CopyFrame(const Frame *incomingFrame)
{
Frame *frame = new Frame();
frame->width=incomingFrame->width;
frame->height=incomingFrame->height;
frame->lenght=incomingFrame->lenght;
frame->buffer=new unsigned char[frame->lenght];
std::memcpy(frame->buffer,incomingFrame->buffer,frame->lenght);
return frame;
}
static Frame* CopyFrame(const QSharedPointer<Frame> &incomingFrame)
{
return CopyFrame(incomingFrame.data());
}*/
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
QDialog *dialog = new QDialog();
QLabel *label = new QLabel(dialog);
FrameBroadcast *cast = new FrameBroadcast();
QObject::connect(cast, &FrameBroadcast::frameCaptured, [=](const QSharedPointer<Frame> &frame) {
int w = static_cast<int>(frame.data()->width);
int h = static_cast<int>(frame.data()->height);
QImage img(frame.data()->buffer,w,h,QImage::Format_RGBA8888);
label->setPixmap(QPixmap::fromImage(img));
label->resize(w,h);
qDebug() << "Update";
});
cast->startCapture();
dialog->show();
return app.exec();
}
在main.cpp中,将创建一个简单的对话框,其中显示捕获结果。以防万一,如果不可能将所有操作都放在一个插槽中,我会附加一个代码,以使屏幕内容与QSharedPointer解除绑定(bind)。它被包含在内并被注释掉之后。
#pragma comment(lib,"dxgi.lib")
#pragma comment(lib,"D3D11.lib")
#pragma comment(lib,"Shcore.lib")
#pragma comment(lib,"winmm.lib")
#pragma comment(lib,"windowscodecs.lib")
#pragma comment (lib, "user32.lib")
#pragma comment (lib, "dxguid.lib")
详细解析代码没有任何意义。它太大了,但根据您的需求进行重新组装并不困难。值得注意的是,使用了“Auto-linking”-Microsoft编译器功能:必要的库将在编译时自动启动(请查看framecapturer.h)。