3

我已经使用 QCamera 和 QAbstractVideoSurface 实现了相机捕捉。我将 QAbstractVideoSurface 扩展为派生类,以将捕获编组到缓冲区中以供将来处理。一切正常,但我在更改输入捕获的捕获分辨率时遇到问题。

使用 setNativeResolution() 似乎不起作用。

下面是代码的简要说明。

#ifndef _CAPTURE_BUFFER_H_
#define _CAPTURE_BUFFER_H_

#include <QMutex>
#include <QWidget>
#include <QImage>
#include <QVideoFrame>
#include <QAbstractVideoSurface>
#include <QVideoSurfaceFormat>
#include <control/qcircularbuffer.h>

class CaptureBuffer: public QAbstractVideoSurface
{
    Q_OBJECT

public:
    CaptureBuffer(int size = 30);
    QList<QVideoFrame::PixelFormat> supportedPixelFormats(QAbstractVideoBuffer::HandleType handleType) const;
    bool start(const QVideoSurfaceFormat& format);
    void stop();
    bool present(const QVideoFrame& frame);
    bool isEmpty() const;
    void pushBack(const QVideoFrame& new_frame);    
    void popFront();
    bool top(QVideoFrame& frame);
    bool back(QVideoFrame& frame);

    const QImage::Format& image_format() const {return m_image_format;}
    const QSize& image_size() const {return m_image_size;}

protected:
    void setNativeResolution(const QSize & resolution);

private:
    QSize                        m_image_size;
    QImage::Format               m_image_format;
    QCircularBuffer<QVideoFrame> m_buffer;
    QMutex                       m_buffer_mutex;
};

#endif



   CaptureBuffer::CaptureBuffer(int size) :
    m_buffer(QCircularBuffer<QVideoFrame>(size))
{
}

QList<QVideoFrame::PixelFormat> CaptureBuffer::supportedPixelFormats(
        QAbstractVideoBuffer::HandleType handleType) const
{
    if (handleType == QAbstractVideoBuffer::NoHandle) {
        return QList<QVideoFrame::PixelFormat>()
                << QVideoFrame::Format_RGB24
                << QVideoFrame::Format_RGB32
                << QVideoFrame::Format_ARGB32
                << QVideoFrame::Format_ARGB32_Premultiplied
                << QVideoFrame::Format_RGB565
                << QVideoFrame::Format_RGB555;
    } else {
        return QList<QVideoFrame::PixelFormat>();
    }
}

bool CaptureBuffer::start(const QVideoSurfaceFormat& format)
{
    const QImage::Format image_format = QVideoFrame::imageFormatFromPixelFormat(format.pixelFormat());
    const QSize size = format.frameSize();

    if (image_format != QImage::Format_Invalid && !size.isEmpty()) {
        m_image_format = image_format;
        m_image_size = size;

        QAbstractVideoSurface::start(format);

        return true;
    } else {
        return false;
    }
}

void CaptureBuffer::stop()
{
    QAbstractVideoSurface::stop();
}

bool CaptureBuffer::present(const QVideoFrame& frame)
{   
    pushBack(frame);
    return true;
}

bool CaptureBuffer::isEmpty() const
{
    return m_buffer.empty();
}

void CaptureBuffer::pushBack(const QVideoFrame& frame)
{
    m_buffer_mutex.lock();
    m_buffer.push_back(frame);
    m_buffer_mutex.unlock();
}

void CaptureBuffer::popFront()
{
    m_buffer_mutex.lock();
    m_buffer.pop_front();
    m_buffer_mutex.unlock();
}

bool CaptureBuffer::top(QVideoFrame& frame)
{
    if(m_buffer.empty()) 
        return false;

    m_buffer_mutex.lock();
    frame = m_buffer.front();
    m_buffer_mutex.unlock();
    return true;
}

bool CaptureBuffer::back(QVideoFrame& frame)
{
    if(m_buffer.empty())
        return false;

    m_buffer_mutex.lock();
    frame = m_buffer.back();
    m_buffer_mutex.unlock();
    return true;
}

void CaptureBuffer::setNativeResolution( const QSize & resolution )
{
    QAbstractVideoSurface::setNativeResolution(resolution);
}

以下是 QCamera 的使用方式和附加到捕获缓冲区的方式:

m_camera = camera;
m_camera->setCaptureMode(QCamera::CaptureVideo);
m_camera->setViewfinder(m_capture_buffer);
m_camera->start(); 

鉴于网络摄像头支持此分辨率,如何将输入捕获分辨率从 640 x 480 调整为 1280 x 720 等。

4

1 回答 1

3

从 Qt5.2.1(来自 git)开始,Digia 似乎没有完全完成 Windows 的 QCamera。然而,有更多的选择如何克服分辨率设置问题。

如果便携性是必须的:

你可以试试 gstreamer。正如我所见,gstreamer 插件的必要部分已实现。

我在 Windows 上工作,gstreamer 在 Windows 上使用 DirectShow,所以我决定直接使用 DirectShow 插件。

Qt5.2.1 有一个功能性的 DirectShow 核心插件,但它并没有完全连接到 Qt 框架本身,因为 DirectShow 插件中不存在 DSImageEncoderControl(将来自 QImageEncoderControl)。

存在 QAndroidImageEncoderControl 和其他一些移动性实现。Digia 似乎决定首先推动 Qt Mobility 业务。

无论如何,他们在 Qt 文档中说:

QImageEncoderSettings imageSettings;
imageSettings.setCodec("image/jpeg");
imageSettings.setResolution(1600, 1200);
imageCapture->setEncodingSettings(imageSettings);

但是,当您调用 QCameraImageCapture::setEncodingSettings Qt 尝试通过 DSImageEncoderControl 设置分辨率时,由于缺少它,该部分代码将无法运行。大多数控件尚未在 DirectShow 插件中实现。

对我来说,另一个问题是捕获静止图像也需要设置表面,但我只想将图像数据用于 OpenCV 的进一步处理。

可能的解决方案:

如果你不需要跨平台的东西,你可以使用我从 Qt 的 DirectShow 插件中借来的代码,你也可以设置分辨率和像素格式。在我的示例中,我尝试遵循 Qt 的命名约定。

另一个问题是目前 QImage 只知道 RGB 格式,但是一些相机设备以 YUYV 格式输出捕获的数据。因此需要转换。我还在我的代码中添加了一个简单的 YUYV 到 RGB24 转换器(感谢 FourCC.org)来测试我的笔记本电脑的摄像头,但我主要使用 Logitech C920 Pro 高清摄像头,它也输出 RGB24 并且不需要转换。

从这里下载代码

于 2014-01-15T14:34:47.970 回答