Отображение видео с альфа-каналом с помощью 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();
}