我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Qt音视频开发之怎么实现ffmpeg视频旋转显示

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Qt音视频开发之怎么实现ffmpeg视频旋转显示

这篇文章主要介绍了Qt音视频开发之怎么实现ffmpeg视频旋转显示的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Qt音视频开发之怎么实现ffmpeg视频旋转显示文章都会有所收获,下面我们一起来看看吧。

一、前言

用手机或者平板拍摄的视频文件,很可能是旋转的,比如分辨率是1280x720,确是垂直的,相当于分辨率变成了720x1280,如果不做旋转处理的话,那脑袋必须歪着看才行,这样看起来太难受,所以一定要想办法解析到视频的旋转角度,然后根据这个角度重新绘制。在窗体那边也需要调整对应的分辨率,一般都是宽度高度互换。其实早期的很多播放器比如vlc2版本的播放器也是不支持旋转的,从vlc3开始内置会自动给旋转,估计这种场景越来越多,毕竟现在智能手机大行其道,用手机拍摄的视频很多都是竖屏的。

在ffmpeg中旋转frame帧数据,有多种方式,方式一是直接通过运算逐行取出数据,重新组织旋转后的视频帧数据;方式二通过滤镜来实现。最开始还没学会用ffmpeg的滤镜的时候,用的就是方式一,通俗易懂,但是很傻,尤其是运算很占CPU,毕竟for循环来个很多次挨个取数据又重新组织数据。自从学会用ffmpeg滤镜以后,从滤镜大全中找到了居然也支持旋转,而且任意角度旋转都可以,甚至镜像操作,这就非常强大了,马上将这个架构的旋转部分全部换成了滤镜旋转,连之前用于旋转的中间过渡帧数据变量都不需要,代码更精简,功能更强大,拓展性更好,这其实就是一个不断精进迭代的过程,第一步解决从无到有的过程,后面才是持续不断的完善。

二、相关代码

int FFmpegFilter::initFilter(AbstractVideoThread *thread, AVStream *stream, AVCodecContext *avctx, FilterData &filterData){    int result = -1;    if (!filterData.enable) {        return result;    }    //貌似硬解码不支持滤镜    if (filterData.formatIn == AV_PIX_FMT_NV12) {        return result;    }    //先释放相关资源    freeFilter(filterData);    //获取滤镜字符串    QString filters = getFilter(filterData);    if (filters.isEmpty()) {        return result;    }    //输入帧序列的参数信息    QStringList listArg;    listArg << QString("video_size=%1x%2").arg(avctx->width).arg(avctx->height);    listArg << QString("pix_fmt=%1").arg(avctx->pix_fmt);    listArg << QString("time_base=%1/%2").arg(stream->time_base.num).arg(stream->time_base.den);    listArg << QString("pixel_aspect=%1/%2").arg(avctx->sample_aspect_ratio.num).arg(avctx->sample_aspect_ratio.den);    QString args = listArg.join(":");    //输入帧格式    enum AVPixelFormat pix_fmts[] = {filterData.formatIn, AV_PIX_FMT_NONE};    //获取要使用的滤镜    const AVFilter *filterclass="lazy" data-src = avfilter_get_by_name("buffer");    const AVFilter *filterSink = avfilter_get_by_name("buffersink");    //创建输入输出滤镜参数    AVFilterInOut *inputs = avfilter_inout_alloc();    AVFilterInOut *outputs = avfilter_inout_alloc();    //创建滤镜容器    filterData.filterGraph = avfilter_graph_alloc();    if (!inputs || !outputs || !filterData.filterGraph) {        result = AVERROR(ENOMEM);        goto end;    }    //创建输入滤镜    result = avfilter_graph_create_filter(&filterData.filterclass="lazy" data-srcCtx, filterclass="lazy" data-src, "in", args.toUtf8().constData(), NULL, filterData.filterGraph);    if (result < 0) {        thread->debug("滤镜处理", QString("创建输入滤镜失败: %1").arg(FFmpegHelper::getError(result)), "");        goto end;    }    //创建输出滤镜    result = avfilter_graph_create_filter(&filterData.filterSinkCtx, filterSink, "out", NULL, NULL, filterData.filterGraph);    if (result < 0) {        thread->debug("滤镜处理", QString("创建输出滤镜失败: %1").arg(FFmpegHelper::getError(result)), "");        goto end;    }    //设置输出滤镜格式    result = av_opt_set_int_list(filterData.filterSinkCtx, "pix_fmts", pix_fmts, filterData.formatOut, AV_OPT_SEARCH_CHILDREN);    if (result < 0) {        thread->debug("滤镜处理", QString("设置输出滤镜格式: %1").arg(FFmpegHelper::getError(result)), "");        goto end;    }    //设置滤镜的参数    outputs->name = av_strdup("in");    outputs->filter_ctx = filterData.filterclass="lazy" data-srcCtx;    outputs->pad_idx = 0;    outputs->next = NULL;    inputs->name = av_strdup("out");    inputs->filter_ctx = filterData.filterSinkCtx;    inputs->pad_idx = 0;    inputs->next = NULL;    //初始化滤镜    result = avfilter_graph_parse_ptr(filterData.filterGraph, filters.toUtf8().constData(), &inputs, &outputs, NULL);    if (result < 0) {        thread->debug("滤镜处理", QString("初始化滤镜失败: %1").arg(FFmpegHelper::getError(result)), "");        goto end;    }    //应用滤镜配置    result = avfilter_graph_config(filterData.filterGraph, NULL);    if (result < 0) {        thread->debug("滤镜处理", QString("应用滤镜配置失败: %1").arg(FFmpegHelper::getError(result)), "");        goto end;    }end:    //释放对应的输入输出    avfilter_inout_free(&inputs);    avfilter_inout_free(&outputs);    filterData.isOk = (result >= 0);    return result;}void FFmpegFilter::freeFilter(FilterData &filterData){    if (filterData.isOk) {        filterData.enable = true;        filterData.init = true;        filterData.isOk = false;        avfilter_free(filterData.filterclass="lazy" data-srcCtx);        avfilter_free(filterData.filterSinkCtx);        avfilter_graph_free(&filterData.filterGraph);        filterData.filterclass="lazy" data-srcCtx = NULL;        filterData.filterSinkCtx = NULL;        filterData.filterGraph = NULL;    }}

五、功能特点

5.1 基础功能

  • 支持各种音频视频文件格式,比如mp3、wav、mp4、asf、rm、rmvb、mkv等。

  • 支持本地摄像头设备,可指定分辨率、帧率。

  • 支持各种视频流格式,比如rtp、rtsp、rtmp、http等。

  • 本地音视频文件和网络音视频文件,自动识别文件长度、播放进度、音量大小、静音状态等。

  • 文件可以指定播放位置、调节音量大小、设置静音状态等。

  • 支持倍速播放文件,可选0.5倍、1.0倍、2.5倍、5.0倍等速度,相当于慢放和快放。

  • 支持开始播放、停止播放、暂停播放、继续播放。

  • 支持抓拍截图,可指定文件路径,可选抓拍完成是否自动显示预览。

  • 支持录像存储,手动开始录像、停止录像,部分内核支持暂停录像后继续录像,跳过不需要录像的部分。

  • 支持无感知切换循环播放、自动重连等机制。

  • 提供播放成功、播放完成、收到解码图片、收到抓拍图片、视频尺寸变化、录像状态变化等信号。

  • 多线程处理,一个解码一个线程,不卡主界面。

5.2 特色功能

  • 同时支持多种解码内核,包括qmedia内核(Qt4/Qt5/Qt6)、ffmpeg内核(ffmpeg2/ffmpeg3/ffmpeg4/ffmpeg5)、vlc内核(vlc2/vlc3)、mpv内核(mpv1/mp2)、海康sdk、easyplayer内核等。

  • 非常完善的多重基类设计,新增一种解码内核只需要实现极少的代码量,就可以应用整套机制。

  • 同时支持多种画面显示策略,自动调整(原始分辨率小于显示控件尺寸则按照原始分辨率大小显示,否则等比例缩放)、等比例缩放(永远等比例缩放)、拉伸填充(永远拉伸填充)。所有内核和所有视频显示模式下都支持三种画面显示策略。

  • 同时支持多种视频显示模式,句柄模式(传入控件句柄交给对方绘制控制)、绘制模式(回调拿到数据后转成QImage用QPainter绘制)、GPU模式(回调拿到数据后转成yuv用QOpenglWidget绘制)。

  • 支持多种硬件加速类型,ffmpeg可选dxva2、d3d11va等,mpv可选auto、dxva2、d3d11va,vlc可选any、dxva2、d3d11va。不同的系统环境有不同的类型选择,比如linux系统有vaapi、vdpau,macos系统有videotoolbox。

  • 解码线程和显示窗体分离,可指定任意解码内核挂载到任意显示窗体,动态切换。

  • 支持共享解码线程,默认开启并且自动处理,当识别到相同的视频地址,共享一个解码线程,在网络视频环境中可以大大节约网络流量以及对方设备的推流压力。国内顶尖视频厂商均采用此策略。这样只要拉一路视频流就可以共享到几十个几百个通道展示。

  • 自动识别视频旋转角度并绘制,比如手机上拍摄的视频一般是旋转了90度的,播放的时候要自动旋转处理,不然默认是倒着的。

  • 自动识别视频流播放过程中分辨率的变化,在视频控件上自动调整尺寸。比如摄像机可以在使用过程中动态配置分辨率,当分辨率改动后对应视频控件也要做出同步反应。

  • 音视频文件无感知自动切换循环播放,不会出现切换期间黑屏等肉眼可见的切换痕迹。

  • 视频控件同时支持任意解码内核、任意画面显示策略、任意视频显示模式。

  • 视频控件悬浮条同时支持句柄、绘制、GPU三种模式,非绝对坐标移来移去。

  • 本地摄像头设备支持指定设备名称、分辨率、帧率进行播放。

  • 录像文件同时支持打开的视频文件、本地摄像头、网络视频流等。

  • 瞬间响应打开和关闭,无论是打开不存在的视频或者网络流,探测设备是否存在,读取中的超时等待,收到关闭指令立即中断之前的操作并响应。

  • 支持打开各种图片文件,支持本地音视频文件拖曳播放。

  • 视频控件悬浮条自带开始和停止录像切换、声音静音切换、抓拍截图、关闭视频等功能。

  • 音频组件支持声音波形值数据解析,可以根据该值绘制波形曲线和柱状声音条,默认提供了声音振幅信号。

  • 各组件中极其详细的打印信息提示,尤其是报错信息提示,封装的统一打印格式。针对现场复杂的设备环境测试极其方便有用,相当于精确定位到具体哪个通道哪个步骤出错。

  • 代码框架和结构优化到最优,性能强悍,持续迭代更新升级。

  • 源码支持Qt4、Qt5、Qt6,兼容所有版本。

5.3 视频控件

  • 可动态添加任意多个osd标签信息,标签信息包括名字、是否可见、字号大小、文本文字、文本颜色、标签图片、标签坐标、标签格式(文本、日期、时间、日期时间、图片)、标签位置(左上角、左下角、右上角、右下角、居中、自定义坐标)。

  • 可动态添加任意多个图形信息,这个非常有用,比如人工智能算法解析后的图形区域信息直接发给视频控件即可。图形信息支持任意形状,直接绘制在原始图片上,采用绝对坐标。

  • 图形信息包括名字、边框大小、边框颜色、背景颜色、矩形区域、路径集合、点坐标集合等。

  • 每个图形信息都可指定三种区域中的一种或者多种,指定了的都会绘制。

  • 内置悬浮条控件,悬浮条位置支持顶部、底部、左侧、右侧。

  • 悬浮条控件参数包括边距、间距、背景透明度、背景颜色、文本颜色、按下颜色、位置、按钮图标代码集合、按钮名称标识集合、按钮提示信息集合。

  • 悬浮条控件一排工具按钮可自定义,通过结构体参数设置,图标可选图形字体还是自定义图片。

  • 悬浮条按钮内部实现了录像切换、抓拍截图、静音切换、关闭视频等功能,也可以自行在源码中增加自己对应的功能。

  • 悬浮条按钮对应实现了功能的按钮,有对应图标切换处理,比如录像按钮按下后会切换到正在录像中的图标,声音按钮切换后变成静音图标,再次切换还原。

  • 悬浮条按钮单击后都用名称唯一标识作为信号发出,可以自行关联响应处理。

  • 悬浮条空白区域可以显示提示信息,默认显示当前视频分辨率大小,可以增加帧率、码流大小等信息。

  • 视频控件参数包括边框大小、边框颜色、焦点颜色、背景颜色(默认透明)、文字颜色(默认全局文字颜色)、填充颜色(视频外的空白处填充黑色)、背景文字、背景图片(如果设置了图片优先取图片)、是否拷贝图片、缩放显示模式(自动调整、等比例缩放、拉伸填充)、视频显示模式(句柄、绘制、GPU)、启用悬浮条、悬浮条尺寸(横向为高度、纵向为宽度)、悬浮条位置(顶部、底部、左侧、右侧)。

关于“Qt音视频开发之怎么实现ffmpeg视频旋转显示”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“Qt音视频开发之怎么实现ffmpeg视频旋转显示”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Qt音视频开发之怎么实现ffmpeg视频旋转显示

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Qt音视频开发之实现ffmpeg视频旋转显示

这篇文章主要为大家详细介绍了在Qt音视频开发中如何利用ffmpeg实现视频旋转显示,文中的实现步骤讲讲清晰,感兴趣的小伙伴可以了解一下
2023-03-22

Qt音视频开发之怎么实现ffmpeg视频旋转显示

这篇文章主要介绍了Qt音视频开发之怎么实现ffmpeg视频旋转显示的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Qt音视频开发之怎么实现ffmpeg视频旋转显示文章都会有所收获,下面我们一起来看看吧。一、前言用
2023-07-05

Qt音视频开发之音频播放QAudioOutput的实现

这篇文章主要为大家详细介绍了如何利用Qt实现音频播放QAudioOutput功能,文中的示例代码讲解详细,对我们学习Qt开发有一定的帮助,需要的可以参考一下
2023-03-10

Qt音视频开发之音频播放QAudioOutput如何实现

这篇文章主要介绍了Qt音视频开发之音频播放QAudioOutput如何实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Qt音视频开发之音频播放QAudioOutput如何实现文章都会有所收获,下面我们一起来看
2023-07-05

Qt音视频开发之视频文件保存功能的实现

和音频存储类似,视频的存储也对应三种格式,视频最原始的数据是yuv(音频对应pcm),视频压缩后的数据是h264(音频对应aac)。本文将利用Qt实现视频文件保存功能,感兴趣的可以了解一下
2022-12-08

Qt音视频开发之怎么用ffmpeg实现解码本地摄像头

这篇文章主要介绍了Qt音视频开发之怎么用ffmpeg实现解码本地摄像头的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Qt音视频开发之怎么用ffmpeg实现解码本地摄像头文章都会有所收获,下面我们一起来看看吧。相
2023-07-05

Qt音视频开发之利用ffmpeg实现解码本地摄像头

一开始用ffmpeg做的是视频流的解析,后面增加了本地视频文件的支持,到后面发现ffmpeg也是支持本地摄像头设备的,所以本文就来用ffmpeg实现解码本地摄像头功能吧
2023-03-24

Qt+Quick实现播放音乐和视频的开发

这篇文章主要为大家详细介绍了如何利用Qt+Quick实现播放音乐和视频的开发,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
2023-03-08

Qt音视频开发之通用监控布局控件的实现

这篇文章主要为大家详细介绍了如何利用Qt开发一个通用的监控布局控件,文中的示例代码讲解详细,对我们学习Qt开发有一定的帮助,需要的可以参考一下
2023-01-13

Android音视频开发之MediaExtactor怎么使用

本文小编为大家详细介绍“Android音视频开发之MediaExtactor怎么使用”,内容详细,步骤清晰,细节处理妥当,希望这篇“Android音视频开发之MediaExtactor怎么使用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢
2023-06-30

Android NDK开发之如何实现FFmpeg视频添加水印

这篇文章主要介绍Android NDK开发之如何实现FFmpeg视频添加水印,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!前言Android也是利用FFmpeg命令行的形式(混编),进行视频转码压缩。1.FFmpeg
2023-06-22

Python怎么实现批量转换视频音频的采样率

这篇文章主要讲解了“Python怎么实现批量转换视频音频的采样率”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Python怎么实现批量转换视频音频的采样率”吧!环境依赖ffmpeg环境安装f
2023-06-21

QT在Android设备上实现FFMPEG开发: 完成拍照、MP4视频录制、rtsp推流

一、系统环境介绍 PC环境: ubuntu18.04 Android版本: 8.1 Android设备: 友善之臂 RK3399 开发板 摄像头:  罗技USB摄像头 FFMPEG版本: 4.2.2 NDK版本: R19C QT版本:  5
2022-06-06

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录