Qt 开源音视频框架模块之QtAV播放器实践

news/2025/2/27 5:58:05

Qt 开源音视频框架模块QtAV播放器实践

1 摘要

QtAV是一个基于Qt的多媒体框架,旨在简化音视频播放和处理。它是一个跨平台的库,支持多种音视频格式,并提供了一个简单易用的API来集成音视频功能。QtAV的设计目标是为Qt应用程序提供强大的音视频处理能力,同时保持灵活性和可扩展性。可支持音频、视频播放,并提供了丰富的 API 和功能,让开发者能够轻松地处理多媒体数据。

2 QtAV的特点

•跨平台:支持Windows、Linux、macOS等多个操作系统。
•多媒体格式支持:支持多种音视频格式,包括常见的MP4、AVI、MKV、FLV、MP3、AAC等。
•高性能:基于FFmpeg,提供了高效的音视频解码和播放能力。
•简单易用的API:提供了直观的接口,方便开发者快速上手。
•集成Qt的特性:与Qt的信号和槽机制兼容,支持在Qt应用程序中使用。

3 简单播放器实践

3.1 环境配置

QtAv是基于ffmpeg开发的,因此我们需要下载相关依赖库。QtAV-depends-windows-x86+x64,下载完成后库目录结构如下
在这里插入图片描述

新建Qt项目QAVPlayer,在项目中添加QtAVWidgetsd库。

在这里插入图片描述

3.2 播放功能实现

•Qt实现简单的播放器功能,主要用到的函数:
•播放控制:play(), pause(), stop()等方法控制播放。
•音量调节:使用setVolume(int volume)方法设置音量。
•进度控制:使用seek(int position)方法跳转到特定时间。
•信号与槽:可以连接信号,例如播放结束、加载完成等事件,以便进行相应的处理。

初始化代码

 	QtAV::VideoOutput  *m_vo;
    QtAV::AVPlayer     *m_player;
 	QtAV::Widgets::registerRenderers();
    m_player = new QtAV::AVPlayer(this);
    m_vo = new QtAV::VideoOutput(this);
    m_player->setRenderer(m_vo);
    ui->verticalLayout->addWidget(m_vo->widget());
    connect(ui->playSlider, SIGNAL(sliderMoved(int)), SLOT(onSliderMoved(int)));
    connect(ui->playSlider, SIGNAL(sliderReleased()), SLOT(onSliderReleased()));
    connect(m_player, SIGNAL(positionChanged(qint64)), this, SLOT(onPositionChange(qint64)));
    connect(m_player, SIGNAL(notifyIntervalChanged()), SLOT(updateSliderUnit()));
    CommonUtils::loadStyleSheet(ui->playSlider, ":/QSS/QSS/VolumnSlider.css");
    connect(ui->volumnBtn, &VolumnButton::valueChanged, this, &MainWindow::onValueChanged);

打开文件播放功能代码

fileName= QFileDialog::getOpenFileName(this, tr("Open File"), "", tr("*.avi *.wmv *.mpg *.mpeg *.mov *.mp4 *.flv"));
           if (fileName.isEmpty())
               return;
m_player->seekBackward();
m_player->play(fileName);

暂停功能代码

	if (m_player->isPaused())
    {
        m_player->pause(false);
    }
    else
    {
        m_player->pause(true);
    }
    this->style()->unpolish(ui->abortBtn);
    this->style()->polish(ui->abortBtn);
    ui->abortBtn->update();

音量设置功能代码

void MainWindow::onValueChanged(int value)
{
    if (value == 0 && ui->volumnBtn->property("silence").toBool() == false)
    {
        ui->volumnBtn->setProperty("silence", true);
        ui->volumnBtn->setToolTip("静音");
    }
    else if (ui->volumnBtn->property("silence").toBool())
    {
        ui->volumnBtn->setProperty("silence", false);
        ui->volumnBtn->setToolTip("音量");
    }
    this->style()->unpolish(ui->volumnBtn);
    this->style()->polish(ui->volumnBtn);
    ui->volumnBtn->update();
    setVolume();
}
void MainWindow::setVolume()
{

     QtAV::AudioOutput *ao = m_player? m_player->audio() : 0;
    qreal v = qreal(ui->volumnBtn->volumnSlider()->value())*kVolumeInterval;
    if (ao) {
        if (qAbs(int(ao->volume() / kVolumeInterval) - ui->volumnBtn->volumnSlider()->value()) >= int(0.1 / kVolumeInterval)) {
            ao->setVolume(v);
        }
    }
}

音量滑竿功能代码

#include "volumnslider.h"
#include "commonutils.h"

#include <QStyleOption>
#include <QPainter>
#include <QLabel>
#include <QDebug>

VolumnSlider::VolumnSlider(QWidget *parent)
	: QSlider(parent)
{
    CommonUtils::loadStyleSheet(this, ":/QSS/QSS/VolumnSlider.css");
	setWindowFlags(Qt::FramelessWindowHint | Qt::SubWindow);
	setAttribute(Qt::WA_TranslucentBackground);

	m_label = new QLabel(nullptr);
	m_label->setStyleSheet("QLabel{color:rgb(179, 179, 179);background:rgb(69,69,69);}");
	m_label->setMargin(6);
	m_label->setFixedHeight(24);
	m_label->setWindowFlags(Qt::FramelessWindowHint | Qt::SubWindow);
	m_label->hide();
	this->setRange(0, 100);
	this->setValue(100);
}

VolumnSlider::~VolumnSlider()
{
	delete m_label;
}

void VolumnSlider::paintEvent(QPaintEvent *event)
{
	QStyleOption opt;
	opt.init(this);
	QPainter p(this);
	style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
	QSlider::paintEvent(event);
}

void VolumnSlider::showEvent(QShowEvent *event)
{
	setFocus();
	QSlider::showEvent(event);
}

void VolumnSlider::focusOutEvent(QFocusEvent *event)
{
	QPoint gloabPoint = this->mapToGlobal(QPoint(0, 0));
	QRect gloabRect(gloabPoint.x(), gloabPoint.y(), width(), height());
	if (!gloabRect.contains(QCursor::pos())){
		this->hide();
		m_label->hide();
	}
	else{
		this->setFocus();
	}
	QSlider::focusOutEvent(event);
}

void VolumnSlider::mouseMoveEvent(QMouseEvent *event)
{
	setLabelText();
	QSlider::mouseMoveEvent(event);
}

void VolumnSlider::mousePressEvent(QMouseEvent *event)
{
	setLabelText();
	m_label->show();
	QSlider::mousePressEvent(event);
}

void VolumnSlider::mouseReleaseEvent(QMouseEvent *event)
{
	m_label->hide();
	QSlider::mouseReleaseEvent(event);
}

void VolumnSlider::setLabelText()
{
	const QString&& value = QString::number(this->value());
	m_label->setFixedWidth(m_label->fontMetrics().width(value) + 12);
	m_label->setText(value);

	QPoint pos;;
	QPoint thePos = this->mapToGlobal(QPoint(0, 0));
	int posY = QCursor::pos().y();
	pos.setX(thePos.x() + 20);
	if (posY < thePos.y()){
		posY = thePos.y();
	}
	else if (posY > thePos.y() + this->height() - m_label->height()){
		posY = thePos.y() + this->height() - m_label->height();
	}
	pos.setY(posY);
	m_label->move(pos);
}

滑竿拖动快进功能核心代码

#include "playtimeslider.h"
//#include "head.h"

#include <QPainter>
#include <QVariant>
#include <QTime>
#include <QApplication>

const int pixwidth = 200;
const int pixheight = 116;

using namespace QtAV;
ImageWidget::ImageWidget(QWidget *parent)
{
    m_timestr = "00:00:00";
    m_extractor = new VideoFrameExtractor(this);
    setWindowFlags(Qt::FramelessWindowHint | Qt::SubWindow);
    setAttribute(Qt::WA_TranslucentBackground);
    hide();

    connect(m_extractor, SIGNAL(frameExtracted(QtAV::VideoFrame)), SLOT(displayFrame(QtAV::VideoFrame)));
    connect(m_extractor, SIGNAL(error(const QString &)), SLOT(displayNoFrame()));
    connect(m_extractor, SIGNAL(aborted(const QString &)), SLOT(displayNoFrame()));
}

void ImageWidget::displayFrame(const QtAV::VideoFrame &frame){
    if (!frame.isValid()) {
        displayNoFrame();
        return;
    }
    m_image = frame.toImage();
    update();
}

void ImageWidget::displayNoFrame()
{
    m_image = QImage();
    update();
}

void ImageWidget::seekVideoFrame(qint64 value)
{
    m_timestr = QTime(0, 0, 0).addMSecs(value).toString(QString::fromLatin1("HH:mm:ss"));
    m_extractor->setPosition(value);
    m_extractor->extract();
    update();
}

void ImageWidget::setPlayFile(const QString& file)
{
    m_extractor->setSource(file);
}

void ImageWidget::paintEvent(QPaintEvent *event)
{
    QWidget::paintEvent(event);
    QPainter painter(this);
    painter.setPen(Qt::NoPen);
    painter.setBrush(QColor("333333"));
    painter.drawRect(0, 0, this->width(), this->height() - 16);
    painter.setPen(Qt::white);
    painter.drawText(0, this->height() - 16, this->width(), 16, Qt::AlignCenter, m_timestr);
    painter.drawText(0, 0, this->width(), this->height() - 16, Qt::AlignCenter, "加载中...");
    if (!m_image.isNull()){
        painter.drawImage(QRect(0, 0, this->width(), this->height() - 16), m_image.scaled(this->width(), this->height() - 16, Qt::KeepAspectRatio, Qt::FastTransformation));
    }
}

PlayTimeSlider::PlayTimeSlider(QWidget *parent)
	: QSlider(parent)
{
	m_pixWidget = new ImageWidget(nullptr);
	m_pixWidget->setFixedSize(pixwidth, pixheight);
}

PlayTimeSlider::~PlayTimeSlider()
{
	delete m_pixWidget;
}

void PlayTimeSlider::stopPlayer()
{
	this->setValue(minimum());
}

void PlayTimeSlider::setPlayervalue(int value)
{
	if (!isSliderDown())
		setValue(value);
}

void PlayTimeSlider::showEvent(QShowEvent *event)
{
	setFocus();
	QSlider::showEvent(event);
}

void PlayTimeSlider::focusOutEvent(QFocusEvent *event)
{
	QPoint gloabPoint = this->mapToGlobal(QPoint(0, 0));
	QRect gloabRect(gloabPoint.x(), gloabPoint.y(), width(), height());
	if (!gloabRect.contains(QCursor::pos())){
		m_pixWidget->hide();
	}
	else{
		this->setFocus();
	}
	QSlider::focusOutEvent(event);
}

void PlayTimeSlider::mouseMoveEvent(QMouseEvent *event)
{
	updatePixWidgetPosition();
	QSlider::mouseMoveEvent(event);
}

void PlayTimeSlider::mousePressEvent(QMouseEvent *event)
{
	QSlider::mousePressEvent(event);
}

void PlayTimeSlider::mouseReleaseEvent(QMouseEvent *event)
{
	m_pixWidget->hide();
	QSlider::mouseReleaseEvent(event);
}

void PlayTimeSlider::setImageWidgetVisibe(bool visible)
{
	m_pixWidget->setVisible(visible);
}

void PlayTimeSlider::seekVideoFrame(qint64 value)
{
	m_pixWidget->seekVideoFrame(value);
}

void PlayTimeSlider::updatePixWidgetPosition()
{
	QPoint pos;
	QPoint thePos = this->mapToGlobal(QPoint(0, 0));
	int mouseX = QCursor::pos().x();
	int posX = mouseX - pixwidth / 2;
	pos.setY(thePos.y() - pixheight - 4);
	if (mouseX < thePos.x()){
		posX = thePos.x() - pixwidth / 2;
	}
	else if (mouseX > thePos.x() + this->width()){
		posX = thePos.x() + this->width() - pixwidth / 2;
	}
	pos.setX(posX);
	m_pixWidget->move(pos);
}

void PlayTimeSlider::setVideoFile(const QString& file)
{
	m_pixWidget->setPlayFile(file);
}

4 播放器效果

在这里插入图片描述

在这里插入图片描述

5 总结

本文介绍了Qt中开源音视频框架模块QtAV,通过使用该音视频播放库实现简单的播放器,了解该库的使用方法,可为后续音视频开发提供更丰富的功能,开发出更好用灵活音视频设计。后续则继续总结分享Qt应用开发中的其他应用,开发不易珍惜每一分原创和劳动成果,同时注意平时开发过程中的经验积累总结。


http://www.niftyadmin.cn/n/5869590.html

相关文章

WEBPACK打包器的使用 打包器的安全问题(源码泄露) 还原

前言 webpack是一个可以把多个前端页面进行打包的工具 他其实也是node js的延伸 可以进行打包node.js文件 基本使用 1、一键命令打包 创建一个 app目录 下设 index.html runboot.js 我们练习对Runboot.js进行打包 一键打包 命令 npx webpack app/runboot…

记一次pytorch训练loss异常的问题

记一次pytorch训练loss异常的问题 问题描述 使用mmdetection框架训练时&#xff0c;某项loss出现异常大的值&#xff0c;比如1781232349724294.000。这个问题只在多卡训练时才会出现。 解决方法 在确认target和predction没有问题后&#xff0c;发现是在dataset中的数据处理…

android keystore源码分析

架构 Android Keystore API 和底层 Keymaster HAL 提供了一套基本的但足以满足需求的加密基元&#xff0c;以便使用访问受控且由硬件支持的密钥实现相关协议。 Keymaster HAL 是由原始设备制造商 (OEM) 提供的动态加载库&#xff0c;密钥库服务使用它来提供由硬件支持的加密服…

OpenCV计算摄影学(3)CUDA 图像去噪函数fastNlMeansDenoising()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 使用非局部均值去噪算法&#xff08;Non-local Means Denoising algorithm&#xff09;执行图像去噪&#xff0c;该算法来源于 http://www.ipol.…

【错误记录】Arrays.asList 的坑

文章目录 概要原因小结 概要 最近在写一个需求的时候用到了这个方法生成一个 List&#xff0c;接着再往里面添加数据的时候就报错了&#xff0c;比如下面的例子。 public class Main {public static void main(String[] args) {List<Integer> res Arrays.asList(1, 2,…

apache-maven-3.2.1

MAVEN_HOME D:\apache-maven-3.2.1 PATH D:\apache-maven-3.2.1\bin cmd mvn -v <localRepository>d:\localRepository</localRepository> setting.xml <?xml version"1.0" encoding"UTF-8"?><!-- Licensed to the Apache Soft…

java23种设计模式-中介者模式

中介者模式&#xff08;Mediator Pattern&#xff09;学习笔记 编程相关书籍分享&#xff1a;https://blog.csdn.net/weixin_47763579/article/details/145855793 DeepSeek使用技巧pdf资料分享&#xff1a;https://blog.csdn.net/weixin_47763579/article/details/145884039 1.…

卡 bug 了?

与工作搭子的日常&#xff1a; 远光小天&#xff1a; 卡 bug&#xff1f;不存在的&#xff01; 这是三岁孩童的正确打开方式&#xff5e; Tips&#xff1a; 远光小天是基于九天研发大模型面向企业研发用户打造的一体化 AI 研发解决方案。基于通用大模型底座&#xff0c;通过模型…