当我在网页浏览器窗口中使用QPixmap::GrabWindow(WId)
时,它只返回黑屏。
我使用以下代码:
QScreen *screen = QGuiApplication::primaryScreen();
m_pixmap = screen->grabWindow(hW);
m_image = m_pixmap.toImage();
m_image.save("p.png");
当我打开" p.png"它只是黑色的图片。使用其他窗口,这很有效。
如何使用普通的浏览器屏幕?
答案 0 :(得分:1)
事实是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解除绑定。列入内容后立即注释掉。
#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)