前面文章FFmpeg像素格式转换中我们使用FFmpeg实现了一个像素格式转换工具类,现在我们就可以在Qt中利用QImage很容易的实现一个简单的YUV播放器了。

播放器功能很简单,只有播放、暂停和停止。我们定义了一个播放器类YuvPlayer,首先在yuvplayer.h文件中定义外部调用的函数,还需要一个设置播放文件的函数,既然是播放yuv文件,那么就需要额外再告诉播放器视频的宽高、像素格式以及帧率,我们定义了一个包括这些参数的结构体Yuv:

#ifndef YUVPLAYER_H
#define YUVPLAYER_H #include <QWidget>
#include <QFile> extern "C"{
#include <libavutil/avutil.h>
#include <libavutil/imgutils.h>
} typedef struct{
const char *filename;
int width;
int height;
AVPixelFormat pixelFormat;
int fps;
}Yuv; class YuvPlayer : public QWidget{
Q_OBJECT
public:
// 状态
typedef enum{
Stopped = 0,
Playing,
Paused,
Finished
} State;
explicit YuvPlayer(QWidget *parent = nullptr);
~YuvPlayer(); void play();
void pause();
void stop();
bool isPlaying();
void setYuv(Yuv &yuv);
State getState(); signals: private:
QFile _file;
int _timerId = 0;
State _state = Stopped;
Yuv _yuv;
bool _playing;
QImage *_currentImage = nullptr;
QRect _dstRect; void timerEvent(QTimerEvent *event);
void paintEvent(QPaintEvent *event);
void freeCurrentImage();
}; #endif // YUVPLAYER_H

setYuv函数用来设置我们要播放的 yuv 文件,可以放到这个函数中的操作有:

  1. 打开yuv文件;
  2. 计算刷帧的时间间隔;
  3. 计算一帧图像的大小;
  4. 计算视频目标尺寸,在播放控件中居中显示视频;
void YuvPlayer::setYuv(Yuv &yuv){
_yuv = yuv; // 关闭上一个文件
closeFile(); // 打开文件
_file = new QFile(yuv.filename);
if(!_file->open(QFile::ReadOnly)){
qDebug()<< "file open error" << yuv.filename;
} // 刷帧的时间间隔
_interval = 1000 / _yuv.fps; // 一帧图片的大小
_imgSize = av_image_get_buffer_size(_yuv.pixelFormat,
_yuv.width,
_yuv.height,
1); // 组件的尺寸
int w = width();
int h = height(); // 计算rect
int dx = 0;
int dy = 0;
int dw = yuv.width;
int dh = yuv.height; // 缩放视频,计算目标尺寸
if (dw > w || dh > h) { // 缩放
//视频的宽高比 > 播放器的宽高比,由(dstW / dstH) > (w / h) 变换而来
if (dw * h > w * dh) {
dh = w * dh / dw;
dw = w;
} else {
dw = h * dw / dh;
dh = h;
}
} // 居中
dx = (w - dw) >> 1;
dy = (h - dh) >> 1; // 计算后的视频宽高
_dstRect = QRect(dx, dy, dw, dh);
qDebug() << "视频的矩形框" << dx << dy << dw << dh;
}

在播放器中完整居中显示 YUV 视频,会遇到四种情况:

  1. 视频宽高都小于等于播放器宽高;
  2. 视频宽大于播放器宽,视频高小于播放器高;
  3. 视频高大于播放器高,视频宽小于播放器宽;
  4. 视频宽高都大于播放器宽高(等同于情况 2 或者 3);

总结下来实际有下图三种情况,第 1 种情况,我们居中显示视频就可以,第 2、3、4 种情况需要视频宽高比不变的情况下对视频进行等比例伸缩,需要伸缩到视频可以在播放器中完整显示。

play函数中开启了一个定时器,定时器执行间隔取决于帧率,执行间隔在setYuv中计算得到,startTimerQObject中的方法,只要继承QObject就可以使用这个函数:

void YuvPlayer::play(){
// 防止多次调用 play 函数开启多个定时器
if(_state == Playing) return;
// 状态可能是:暂停、停止、正常完毕
// 开启定时器
_timerId = startTimer(_interval); setState(Playing);
}

定时器开启后每隔一定间隔会调用timerEvent函数,这个函数中我们从文件读取一帧yuv数据,使用我们之前实现的像素格式转换工具将yuv420p格式数据转换成rgb24格式数据,然后将数据渲染到QImage上面,调用update`函数刷新。此处需要注意一个问题,像素格式转换后的输出视频宽高不是16的倍数会降低转码速度,建议输出视频宽高是16倍数:

void YuvPlayer::timerEvent(QTimerEvent *event){
// 图片大小
char data[_imgSize];
if(_file->read(data,_imgSize) == _imgSize){
RawVideoFrame in = {
data,
_yuv.width,_yuv.height,
_yuv.pixelFormat
}; RawVideoFrame out = {
nullptr,
_yuv.width,_yuv.height,
AV_PIX_FMT_RGB24
};
FFmpegUtils::convertRawVideo(in,out); freeCurrentImage();
_currentImage = new QImage((uchar *)out.pixels,
out.width,out.height,QImage::Format_BGR888); // 刷新
update();
}else{// 文件数据已经读取完毕
// 停止定时器
stopTimer(); // 正常播放完毕
setState(Finished);
}
}

当调用update函数的时候,就会触发paintEvent,在这个函数中将图片绘制到当前组件上。当组件想重绘的时候,也会调用这个函数:

// 当组件想重绘的时候,就会调用这个函数
// 想要绘制什么内容,在这个函数中实现
void YuvPlayer::paintEvent(QPaintEvent *event){
if (!_currentImage) return; // 将图片绘制到当前组件上
QPainter(this).drawImage(_dstRect, *_currentImage);
}

接下来继续实现暂停和停止功能:

void YuvPlayer::pause(){
if(_state != Playing) return;
// 状态可能是:正在播放 // 停止定时器
stopTimer(); setState(Paused);
} void YuvPlayer::stop(){
if(_state == Stopped) return; // 状态可能是:正在播放、暂停、正常完毕 // 释放图片
freeCurrentImage(); // 停止定时器
stopTimer(); // 刷新,会触发paintEvent方法的执行
update(); setState(Stopped);
}

QFile会记录上次读取文件的位置,当播放完毕时,要将读取指针回归到最初始的位置。作为一个播放器,需要时刻向外界发送一些消息,比如暂停或者继续播放等等需要通知外界,我们利用Qt信号和槽机制,在信号声明区下面定义了一个信号stateChange,当播放器状态发生改变时我们发送一个信号,外界与此信号关联的槽函数就会被调用:

void YuvPlayer::setState(State state){
if(state == _state) return; if(state == Stopped || state == Finished){
// 让文件读取指针回到文件首部
_file->seek(0);
} _state = state;
emit stateChanged();
}

具体示例代码

yuvplayer.h

#ifndef YUVPLAYER_H
#define YUVPLAYER_H #include <QWidget>
#include <QFile> extern "C"{
#include <libavutil/avutil.h>
#include <libavutil/imgutils.h>
} typedef struct{
const char *filename;
int width;
int height;
AVPixelFormat pixelFormat;
int fps;
}Yuv; class YuvPlayer : public QWidget{
Q_OBJECT
public:
// 状态
typedef enum{
Stopped = 0,
Playing,
Paused,
Finished
} State;
explicit YuvPlayer(QWidget *parent = nullptr);
~YuvPlayer(); void play();
void pause();
void stop();
bool isPlaying();
void setYuv(Yuv &yuv);
State getState(); signals:
void stateChanged(); private:
QFile *_file = nullptr;
int _timerId = 0;// 先写一个0,否则有可能是个垃圾值
State _state = Stopped;
Yuv _yuv;
bool _playing;
QImage *_currentImage = nullptr;
// 视频大小
QRect _dstRect;
// 刷帧的时间间隔
int _interval;
// 一帧图片的大小
int _imgSize = 0; void timerEvent(QTimerEvent *event);
void paintEvent(QPaintEvent *event);
void freeCurrentImage();
// 改变状态
void setState(State state); void stopTimer(); void closeFile();
}; #endif // YUVPLAYER_H

yuvplayer.cpp

#include "yuvplayer.h"
#include <QDebug>
#include <QPainter>
#include "ffmpegutils.h" #define RET(judge, func) \
if (judge) { \
qDebug() << #func << "error" << SDL_GetError(); \
return; \
} YuvPlayer::YuvPlayer(QWidget *parent) : QWidget(parent){
// 设置背景色
// setAttribute(Qt::WA_StyledBackground,true);
setAttribute(Qt::WA_StyledBackground);
setStyleSheet("background:black"); } YuvPlayer::~YuvPlayer(){
closeFile();
freeCurrentImage();
stopTimer();
} void YuvPlayer::play(){
// 防止多次调用 play 函数开启多个定时器
if(_state == Playing) return;
// 状态可能是:暂停、停止、正常完毕
// 开启定时器
_timerId = startTimer(_interval); setState(Playing);
} void YuvPlayer::pause(){
if(_state != Playing) return;
// 状态可能是:正在播放 // 停止定时器
stopTimer(); setState(Paused);
} void YuvPlayer::stop(){
if(_state == Stopped) return; // 状态可能是:正在播放、暂停、正常完毕 // 释放图片
freeCurrentImage(); // 停止定时器
stopTimer(); // 刷新,会触发paintEvent方法的执行
update(); setState(Stopped);
} bool YuvPlayer::isPlaying(){
return _state == YuvPlayer::Playing;
} YuvPlayer::State YuvPlayer::getState(){
return _state;
} void YuvPlayer::setState(State state){
if(state == _state) return; if(state == Stopped || state == Finished){
// 让文件读取指针回到文件首部
_file->seek(0);
} _state = state;
emit stateChanged();
} void YuvPlayer::setYuv(Yuv &yuv){
_yuv = yuv; // 关闭上一个文件
closeFile(); // 打开文件
_file = new QFile(yuv.filename);
if(!_file->open(QFile::ReadOnly)){
qDebug()<< "file open error" << yuv.filename;
} // 刷帧的时间间隔
_interval = 1000 / _yuv.fps; // 一帧图片的大小
_imgSize = av_image_get_buffer_size(_yuv.pixelFormat,
_yuv.width,
_yuv.height,
1); // 组件的尺寸
int w = width();
int h = height(); // 计算rect
int dx = 0;
int dy = 0;
int dw = yuv.width;
int dh = yuv.height; // 缩放视频,计算目标尺寸
if (dw > w || dh > h) { // 缩放
//视频的宽高比 > 播放器的宽高比,由(dstW / dstH) > (w / h) 变换而来
if (dw * h > w * dh) {
dh = w * dh / dw;
dw = w;
} else {
dw = h * dw / dh;
dh = h;
}
} // 居中
dx = (w - dw) >> 1;
dy = (h - dh) >> 1; // 计算后的视频宽高
_dstRect = QRect(dx, dy, dw, dh);
qDebug() << "视频的矩形框" << dx << dy << dw << dh;
} // 当组件想重绘的时候,就会调用这个函数
// 想要绘制什么内容,在这个函数中实现
void YuvPlayer::paintEvent(QPaintEvent *event){
if (!_currentImage) return; // 将图片绘制到当前组件上
// QPainter(this).drawImage(QPoint(0,0), *_currentImage);
QPainter(this).drawImage(_dstRect, *_currentImage);
// QPainter(this).drawImage(QRect(0,0,width(),height()), *_currentImage);
} void YuvPlayer::timerEvent(QTimerEvent *event){
// 图片大小
char data[_imgSize];
if(_file->read(data,_imgSize) == _imgSize){
RawVideoFrame in = {
data,
_yuv.width,_yuv.height,
_yuv.pixelFormat
}; RawVideoFrame out = {
nullptr,
_yuv.width,_yuv.height,
AV_PIX_FMT_RGB24
};
FFmpegUtils::convertRawVideo(in,out); freeCurrentImage();
_currentImage = new QImage((uchar *)out.pixels,
out.width,out.height,QImage::Format_BGR888); // 刷新
update();
}else{// 文件数据已经读取完毕
// 停止定时器
stopTimer(); // 正常播放完毕
setState(Finished);
}
} void YuvPlayer::freeCurrentImage() {
if(!_currentImage) return;
free(_currentImage->bits());
delete _currentImage;
_currentImage = nullptr;
} void YuvPlayer::stopTimer(){
if(_timerId == 0) return;
killTimer(_timerId);
_timerId = 0;
} void YuvPlayer::closeFile(){
if(!_file) return; _file->close();
delete _file;
_file = nullptr;
}

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QDebug> #ifdef Q_OS_WIN
#define FILENAME "../test/out_640x480.yuv"
#else
#define FILENAME "/Users/zuojie/QtProjects/audio-video-dev/test/out_640x480.yuv"
#endif MainWindow::MainWindow(QWidget *parent)
: QMainWindow(parent)
, ui(new Ui::MainWindow){
ui->setupUi(this); // 创建播放器
_player = new YuvPlayer(this);
// 设置播放器的位置和尺寸
int w = 640;
int h = 480;
int x = (width() - w) >> 1;
int y = (height() - h) >> 1;
_player->setGeometry(x, y, w, h); // 设置需要播放的文件
Yuv yuv = {
FILENAME,
640,480,
AV_PIX_FMT_YUV420P,
30
};
_player->setYuv(yuv); // 监听播放
connect(_player,&YuvPlayer::stateChanged,this,&MainWindow::onPlayerStateChanged);
} MainWindow::~MainWindow(){
delete ui;
} void MainWindow::on_playButton_clicked(){
if(_player->isPlaying()){// 正在播放
_player->pause();
}else{// 没有正在播放
_player->play();
}
} void MainWindow::on_stopButton_clicked(){
_player->stop();
} void MainWindow::onPlayerStateChanged(){
if(_player->getState() == YuvPlayer::Playing){
ui->playButton->setText("暂停");
}else{// 没有正在播放
ui->playButton->setText("播放");
}
}

代码连接

24_用Qt和FFmpeg实现简单的YUV播放器的更多相关文章

  1. 简单的音乐播放器(VS 2010 + Qt 4.8.5)

    昨天历经千辛万苦,配置好了VS 2010中的Qt环境(包括Qt for VS插件),今天决定浅浅地品味一下将两者结合进行编程的魅力. 上网查了一些资料,学习了一些基础知识,决定做一个简单的音乐播放器, ...

  2. swift3.0 简单直播和简单网络音乐播放器

    本项目采用swift3.0所写,适配iOS9.0+,所有界面均采用代码布局. 第一个tab写的是简单直播,传统MVC模式,第二个tab写的是简单网络音乐播放器.传说MVVM模式(至于血统是否纯正我就不 ...

  3. 用PHP+H5+Boostrap做简单的音乐播放器(进阶版)

    前言:之前做了一个音乐播放器(纯前端),意外的受欢迎,然后有人建议我把后台一起做了,正好也想学习后台,所以学了两天php(不要吐槽我的速度,慢工出细活嘛~)然后在之前的基础上也又完善了一些功能,所以这 ...

  4. ffmpeg+SDL2实现的音频播放器V2.0(无杂音)

    1. 前言 目前为止,学习了并记录了ffmpeg+SDL2显示视频以及事件(event)的内容. 这篇中记录ffmpeg+SDL2播放音频,没加入事件处理. 接下来加入事件处理并继续学习音视频同步,再 ...

  5. 一个基于JRTPLIB的轻量级RTSP客户端(myRTSPClient)——解码篇:(三)一个简单的rtsp播放器

    该篇内容简单的将前两篇内容组合在一起,创建了2个线程,分别播放音频和视频. int main(int argc, char * argv[]) { RtspClient Client; pthread ...

  6. 使用Service组件实现简单的音乐播放器功能 --Android基础

    1.本例利用Service实现简单的音乐播放功能,下面是效果图.(点击开始播放开启服务,音乐播放,点击“停止播放”关闭服务,音乐停止播放.) 2.核心代码: MusicService.java: pa ...

  7. Android课程---简单的音乐播放器

    第一个:用Activity实现 activity_music_play1.xml <?xml version="1.0" encoding="utf-8" ...

  8. 用H5+Boostrap做简单的音乐播放器

    前言:这个是综合一下我最近在学的东西做的小Demo,到实际使用还有距离,但是用来练手巩固知识点还是不错的,最近在二刷JS书和Boostrap.css的源码,做完这个Demo也算是暂告一段落,接下来是j ...

  9. HTML5+CSS3+jquery实现简单的音乐播放器

    ...最近天热的,感觉就像煎饼...然后别人在把妹子的时候,只有偶们这帮苦逼的程序员在那边撸代码...我日哦! 然后今天晒的是偶早年写的一个播放器...看上去是不是很有感觉的样子!一番宝物,Lisa唱 ...

  10. Android——用Activity和Service实现简单的音乐播放器

    一.只用Activity 容易出现问题 xml <?xml version="1.0" encoding="utf-8"?> <LinearL ...

随机推荐

  1. Android 相册

  2. 洛谷P3046 海底高铁 巧用差分统计经过区间次数

    洛谷P3046 海底高铁 -差分统计经过区间次数 题目贴在这里P3406 海底高铁 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 分析 本题题干很长,但是题意理解很简单.就是给定n ...

  3. 5 款轻松上手的开源项目「GitHub 热点速览」

    大家都忙一年了,所以今天来点轻松的吧!就是那种拿来直接用.免费看的开源项目. 开源真是一个充满惊喜的宝库,很多开源软件比收费软件还好用,比如这款开箱即用的电视直播软件:my-tv,它免费.无广告.启动 ...

  4. 冰点还原 deep freeze 安装,招聘机试时用到

    官方地址:https://www.faronics.com https://www.bingdiancn.com/(也可以从这个网站下载,我是从这个网站下载的) 需求:招聘时 需要机试,需要做项目,为 ...

  5. 让python程序一直在window后台进程运行

    一.让python程序后台运行 1.创建一个app.py文件,如 while 1: print(123)2.创建一个set_py.bat文件,里面写 python app.py3.创建一个start_ ...

  6. ABC 332

    ABCDF 都赛时做出来了. E \(\displaystyle\dfrac{1}{D}\sum_{i=1}^D (x_i-\overline{x})^2=\dfrac{1}{D}(\sum_{i=1 ...

  7. Python3排序sorted(key=lambda)

    Python3排序sorted(key=lambda) 简述: 假如d是一个由元组构成的列表,我们需要用到参数key,也就是关键词,看下面这句命令,lambda是一个隐函数,是固定写法,不要写成别的单 ...

  8. AIR32F103(十二) 搭载 AIR32F103CBT6 的Bluepill核心板

    目录 AIR32F103(一) 合宙AIR32F103CBT6开发板上手报告 AIR32F103(二) Linux环境和LibOpenCM3项目模板 AIR32F103(三) Linux环境基于标准外 ...

  9. 使用sqlmap执行SQL注入并获取数据库用户名

    Sqlmap介绍 sqlmap支持MySQL, Oracle,PostgreSQL, Microsoft SQL Server, Microsoft Access, IBM DB2, SQLite, ...

  10. java 打包jar文件实战

    本文只介绍实用步骤,预备知识请自查阅: 参考资料: http://docs.oracle.com/javase/tutorial/deployment/jar/appman.html http://w ...