Отображение видео с альфа-каналом с помощью qt

Мне нужно отображать видео в кодировке RGBA (то есть видео с прозрачным фоном) с помощью qt. Идея состоит в том, чтобы объединять видео в реальном времени. Обычно я использую libmpv, но, похоже, он не позволяет отображать прозрачное фоновое видео. Я пытаюсь использовать QMediaPlayer со следующим кодом:

 QMainWindow w;
 w.resize(1920,1080);
 QVideoWidget videoWidget(&w);
 videoWidget.move(0,100);
 videoWidget.resize(1920,1080);
 QMediaPlayer *player = new QMediaPlayer(&w);
 w.resize(w.size());
 player->setMedia( QUrl::fromLocalFile(PATH+"video2.mov") );
 player->setVideoOutput(&videoWidget);
 w.show();
 player->play();

Это успешно загружает видео (которое является видео в формате RGBA mov), но заполняет виджет видео черным фоном, где он должен быть прозрачным, таким образом закрывая любой элемент позади видеоплеера.

Есть ли способ загрузить прозрачное видео с помощью QVideoPlayer/QVideoWidget? Если нет, есть ли эффективная альтернатива (я бы предпочел не использовать решение более низкого уровня, такое как opencv).

Большое спасибо,

Фред

1 ответ

Решение

Вот решение, которое я наконец нашел:

Создайте подкласс QAbstractVideoSurface:

class alphaVideoDrawer : public QAbstractVideoSurface
{
    Q_OBJECT
public:
    alphaVideoDrawer(QLabel *displayLbl);
private:
    QLabel *displayLbl;

protected:
    bool present(const QVideoFrame &frame);

    QList<QVideoFrame::PixelFormat> supportedPixelFormats(
            QAbstractVideoBuffer::HandleType handleType = QAbstractVideoBuffer::NoHandle) const
    {
        Q_UNUSED(handleType);
        return QList<QVideoFrame::PixelFormat>() << QVideoFrame::Format_ARGB32;
    }
};

Поверхность получит преобразованный кадр и отправит его в qlabel, который отображает видео.

alphaVideoDrawer::alphaVideoDrawer(QLabel *displayLbl):displayLbl(displayLbl)
{
}

extern QImage qt_imageFromVideoFrame(const QVideoFrame &f);


bool alphaVideoDrawer::present(const QVideoFrame &frame)
{
    QImage image = qt_imageFromVideoFrame(frame);
    displayLbl->setPixmap(QPixmap::fromImage(image));
    return true;
}

Затем мы создаем подкласс QLabel, который будет нашим видеовыходом:

class alphaVideo : public QLabel
{
    Q_OBJECT
public:
    alphaVideo(QLabel *parent = nullptr);

private:
    alphaVideoDrawer *videoDrawer;
    QMediaPlayer *videoPlayer;
    QMediaPlaylist *playlist;
};

Он загружает ящик и проигрыватель и начинает воспроизведение / рендеринг видео:

    alphaVideo::alphaVideo(QLabel *parent): QLabel(parent)
{
    setStyleSheet("QLabel { background-color : transparent; }");
    videoDrawer = new alphaVideoDrawer(this);
    videoPlayer = new QMediaPlayer(this);
    playlist = new QmediaPlaylist();
    videoPlayer->setPlaylist(playlist);
    videoPlayer->setVideoOutput(videoDrawer);
    playlist->addMedia( Qurl::fromLocalFile(“your RGBA video file.mp4”) );
    videoPlayer->play();
}
Другие вопросы по тегам