如何从相机读取缓冲信息

How to read the buffer info from a camera?

本文关键字:缓冲 信息 读取 相机      更新时间:2023-10-16

我有QT应用程序的一个特定部分,该部分应该从相机拍摄照片,并使用QZXING库解码QR码。但是,我无法将照片从缓冲区中取出!我知道解码的运行良好,因为如果我将照片保存到文件中,然后立即将其重新加载为Qimage,一切都有效(当然,除了该程序完全保存之前,该程序不会阻塞,因此'将尝试解码半图像,然后在第二次尝试中解码第一个图像。)!

那里有很多人问这个问题,但似乎没有人有一个完整的答案。在过去的六个小时中,我一直在研究和猜测。QT文档存在QCAMERAIMAGECAPTURE,但根本没有谈论通过Qimage进行缓冲或铸造。有几个答案,因此有几个猜测将图像投入Qimage,还有几个谈论找到缓冲区的东西,但没有人有完整的答案。QT示例文档甚至还不完整,他们的示例项目(整个网络中的链接都有很多)不讨论它在做什么。

这是一些测试代码的示例,该测试代码使用mainwindow.ui运行和编译,并带有称为" PushButton"的按钮和称为" Verticallayout"的垂直布局。我在做什么错?

main.cpp

#include "mainwindow.h"
#include <QApplication>
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
    return a.exec();
}

mainWindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
#include <QImage>
#include <QCamera>
#include <QCameraInfo>
#include <QCameraViewfinder>
#include <QCameraImageCapture>
#include <qzxing/QZXing.h>
namespace Ui {
class MainWindow;
}
class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
private slots:
    void cameraReceiver(int f,QVideoFrame u);
    void on_pushButton_clicked();
private:
    Ui::MainWindow *ui;
    QCameraViewfinder   *viewfinder;
    QCamera             *invCam;
    QCameraImageCapture *rawImage;
    QZXing              *decoder;
};
#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    //build camera
    QList<QCameraInfo> cameras = QCameraInfo::availableCameras();
    foreach (const QCameraInfo &cameraInfo, cameras) {
        if (cameraInfo.deviceName() == "/dev/video0")
            invCam = new QCamera(cameraInfo);
    }
    //build decoder
    decoder = new QZXing;
    decoder->setDecoder(QZXing::DecoderFormat_QR_CODE);
    //build viewfinder and link to camera
    viewfinder = new QCameraViewfinder(this);
    viewfinder->setSizePolicy(QSizePolicy::Expanding,QSizePolicy::Expanding);
    ui->verticalLayout->addWidget(viewfinder);
    invCam->setViewfinder(viewfinder);
    //build image buffer, set camera mode to capture
    rawImage = new QCameraImageCapture(invCam);
    invCam->setCaptureMode(QCamera::CaptureStillImage);
    invCam->start();
    viewfinder->show();
    connect(rawImage,SIGNAL(imageAvailable(int,QVideoFrame)),this,SLOT(cameraReceiver(int,QVideoFrame)));
}
MainWindow::~MainWindow()
{
    delete ui;
}
void MainWindow::cameraReceiver(int f,QVideoFrame u) {
    QImage currentImage(u.bits(), u.width(), u.height(), u.bytesPerLine(), QVideoFrame::imageFormatFromPixelFormat(u.pixelFormat()));
    QString output = decoder->decodeImage(currentImage);
    qDebug() << output;
}
void MainWindow::on_pushButton_clicked()
{
    rawImage->setCaptureDestination(QCameraImageCapture::CaptureToBuffer);
    rawImage->capture();
}

编辑:当前在Linux环境上,相机已嵌入。取景器显示的很好,就像我说的那样,我可以捕获图像没有任何问题 - 相机正常工作

编辑2:一旦发布,我就找到了答案:

"连接"调用应使用"想象中的"信号:

connect(rawImage,SIGNAL(imageCaptured(int,QImage)),this,SLOT(cameraReceiver(int,QImage)));

然后,缓冲区的Qimage坐在其中。

void MainWindow::cameraReceiver(int f,QImage u) {
    QString output = decoder->decodeImage(u);
    qDebug() << output;
}

我看到您为自己找到了一个解决方案,但是如果有人需要使用QVideoFrame

,这是一个重要的提示。

问题是QVideoFrame的错误用法。必须在访问它之前映射框架:

void MainWindow::cameraReceiver(int f,QVideoFrame u) {
    if (u.isValid()) {
        if(u.map(QAbstractVideoBuffer::ReadOnly)) {//map the frame
            uchar * data = new uchar[u.mappedBytes()];//copy the buffer for QImage
            memcpy(data, u.bits(), u.mappedBytes());
            QImage image(data,
                         u.width(),
                         u.height(),
                         u.bytesPerLine(),
                         QVideoFrame::imageFormatFromPixelFormat(u.pixelFormat()),
                         simpleCleanupHandler,//handles the buffer cleanup
                         data);//required for the cleanup
            u.unmap();//unmap the frame
            QString output = decoder->decodeImage(currentImage);
            qDebug() << output;
        }
    }
}

由于QImage不会复制缓冲区,因此您必须自己进行。simpleImageCleanupHandler看起来像这样:

static void simpleCleanupHandler(void *info)
{
    delete[] (uchar*)info;
}