How to use QVideoSink in QML in Qt6
Asked Answered
L

1

5

I created my own video sink inherited from QVideoSink in Qt 6. And I want to show content of this sink in QML side. How can I do it?

VideoOutput QML type has videoSink property, but it's read only..

Localism answered 4/10, 2021 at 7:36 Comment(0)
B
8

The output elements such as the VideoOutput and the QVideoWidget have a QVideoSink so you should not create one but write over that QVideoSink:

#ifndef PRODUCER_H
#define PRODUCER_H

#include <QObject>
#include <QPointer>
#include <QVideoSink>
#include <QQmlEngine>
#include <QTimer>

class Producer : public QObject
{
    Q_OBJECT
    QML_ELEMENT
    Q_PROPERTY(QVideoSink* videoSink READ videoSink WRITE setVideoSink NOTIFY videoSinkChanged)
public:
    Producer(QObject *parent=nullptr);
    QVideoSink *videoSink() const;
    void setVideoSink(QVideoSink *newVideoSink);
    Q_INVOKABLE void start();
signals:
    void videoSinkChanged();
private:
    QPointer<QVideoSink> m_videoSink;
    void handleTimeout();
    QTimer m_timer;
};

#endif // PRODUCER_H
#include "producer.h"

#include <QImage>
#include <QPainter>
#include <QSize>
#include <QVideoFrame>

#include <QRandomGenerator>
#include <QDateTime>

Producer::Producer(QObject *parent):QObject(parent)
{
    m_timer.setInterval(500);
    connect(&m_timer, &QTimer::timeout, this, &Producer::handleTimeout);
}

QVideoSink *Producer::videoSink() const
{
    return m_videoSink.get();
}

void Producer::setVideoSink(QVideoSink *newVideoSink)
{
    if (m_videoSink == newVideoSink)
        return;
    m_videoSink = newVideoSink;
    emit videoSinkChanged();
}

void Producer::start()
{
    m_timer.start();
    handleTimeout();
}

void Producer::handleTimeout()
{
    if(!m_videoSink)
        return;
    QVideoFrame video_frame(QVideoFrameFormat(QSize(640, 480),QVideoFrameFormat::Format_BGRA8888));
    if(!video_frame.isValid() || !video_frame.map(QVideoFrame::WriteOnly)){
        qWarning() << "QVideoFrame is not valid or not writable";
        return;
    }
    QImage::Format image_format = QVideoFrameFormat::imageFormatFromPixelFormat(video_frame.pixelFormat());
    if(image_format == QImage::Format_Invalid){
        qWarning() << "It is not possible to obtain image format from the pixel format of the videoframe";
        return;
    }
    int plane = 0;
    QImage image(video_frame.bits(plane), video_frame.width(),video_frame.height(), image_format);
    image.fill(QColor::fromRgb(QRandomGenerator::global()->generate()));
    QPainter painter(&image);
    painter.drawText(image.rect(), Qt::AlignCenter, QDateTime::currentDateTime().toString());
    painter.end();

    video_frame.unmap();
    m_videoSink->setVideoFrame(video_frame);
}
#include <QGuiApplication>
#include <QQmlApplicationEngine>


int main(int argc, char *argv[])
{
    QGuiApplication app(argc, argv);

    QQmlApplicationEngine engine;
    const QUrl url(QStringLiteral("qrc:/main.qml"));
    QObject::connect(&engine, &QQmlApplicationEngine::objectCreated,
                     &app, [url](QObject *obj, const QUrl &objUrl) {
        if (!obj && url == objUrl)
            QCoreApplication::exit(-1);
    }, Qt::QueuedConnection);
    engine.load(url);

    return app.exec();
}
import QtQuick
import QtQuick.Window
import QtMultimedia

import com.eyllanesc.multimedia

Window {
    width: 640
    height: 480
    visible: true
    title: qsTr("Hello World")

    Producer{
        id: producer
        videoSink: videoOutput.videoSink
    }
    VideoOutput{
        id: videoOutput
        anchors.fill: parent
    }
    Component.onCompleted: producer.start()
}

enter image description here

The complete example can be found here.

Barbiturate answered 4/10, 2021 at 8:21 Comment(5)
How to set VideoFrame from QImage ?Humph
@Humph If there is a QVideoFrameFormat::PixelFormat associated with the image format (use doc-snapshots.qt.io/qt6-dev/… to verify that) then build the QVideoFrame with that pixelFormat and the size of the image, then copy the bits of the image to the QVideoFrame using std::memcpy. If there is no associated PixelFormat then it converts the image format to one that does.Barbiturate
@Humph Another option is to use the code of my example and paint the QImage that you have to the QImage of my example using QPainterBarbiturate
I Already used this QPainter painter(&image); painter.drawImage(0,0,frame);Humph
@Humph That is what I pointed out in my previous commentBarbiturate

© 2022 - 2025 — McMap. All rights reserved.