Qt使用ffmpeg获取视频文件封面图

Qt使用ffmpeg获取视频文件封面图

#ifndef __THUM_HELPER_H_
#define __THUM_HELPER_H_extern "C" {
#include "libavformat/avformat.h"
#include "libavutil/imgutils.h"
#include "libswscale/swscale.h"
}
#include <QObject>
#include <QImage>
namespace Media {class ThumHelper :public QObject{Q_OBJECTpublic:ThumHelper();~ThumHelper();void initGlobal();QImage thumnail(const QString& videoPath);};
}#endif  //__THUM_HELPER_H_
#include "ThumHelper.h"Media::ThumHelper::ThumHelper()
{initGlobal();
}void Media::ThumHelper::initGlobal()
{av_register_all();
}QImage Media::ThumHelper::thumnail(const QString& videoPath)
{QImage image;AVFormatContext * fmtContext = nullptr;if (avformat_open_input(&fmtContext, videoPath.toStdString().c_str(), nullptr, nullptr) < 0) {return image;}if (avformat_find_stream_info(fmtContext, nullptr) < 0) {avformat_close_input(&fmtContext);return image;}int nStreamIndex = -1;AVCodecParameters *codecParameters = nullptr;for (int i = 0; i < fmtContext->nb_streams; i++) {if (fmtContext->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO) {nStreamIndex = i;codecParameters = fmtContext->streams[i]->codecpar;break;}}if (nStreamIndex == -1) {avformat_close_input(&fmtContext);return image;}AVCodec* codec = avcodec_find_decoder(codecParameters->codec_id);if (!codec) {avformat_close_input(&fmtContext);return image;}AVCodecContext* codecContext = avcodec_alloc_context3(codec);if (!codecContext) {// 分配解码器上下文失败avformat_close_input(&fmtContext);return image;}if (avcodec_parameters_to_context(codecContext, codecParameters) < 0) {// 复制解码器参数到解码器上下文失败avcodec_free_context(&codecContext);avformat_close_input(&fmtContext);return image;}if (avcodec_open2(codecContext, codec, nullptr) < 0) {// 打开解码器失败avcodec_free_context(&codecContext);avformat_close_input(&fmtContext);return image;}AVPacket packet;av_init_packet(&packet);packet.data = nullptr;packet.size = 0;while (av_read_frame(fmtContext, &packet) >= 0) {if (packet.stream_index == nStreamIndex) {AVFrame* frame = av_frame_alloc();if (frame) {int ret = avcodec_send_packet(codecContext, &packet);if (ret >= 0) {ret = avcodec_receive_frame(codecContext, frame);if (ret >= 0) {// 将第一帧保存为封面图像if (frame->key_frame) {AVFrame* rgbFrame = av_frame_alloc();if (rgbFrame) {rgbFrame->format = AV_PIX_FMT_RGB24;rgbFrame->width = frame->width;rgbFrame->height = frame->height;int bufferSize = av_image_get_buffer_size(AV_PIX_FMT_RGB24, frame->width, frame->height, 1);uint8_t* buffer = new uint8_t[bufferSize];av_image_fill_arrays(rgbFrame->data, rgbFrame->linesize, buffer, AV_PIX_FMT_RGB24, frame->width, frame->height, 1);SwsContext* swsContext = sws_getContext(frame->width, frame->height, codecContext->pix_fmt,frame->width, frame->height, AV_PIX_FMT_RGB24, SWS_BICUBIC, nullptr, nullptr, nullptr);if (swsContext) {sws_scale(swsContext, frame->data, frame->linesize, 0, frame->height, rgbFrame->data, rgbFrame->linesize);sws_freeContext(swsContext);// 保存封面图像到文件int outputBufferSize = rgbFrame->width * rgbFrame->height * 3;uchar* outputBuffer = new uchar[outputBufferSize];for (int i = 0; i < rgbFrame->height; i++) {memcpy(outputBuffer + i * rgbFrame->width * 3, rgbFrame->data[0] + i * rgbFrame->linesize[0], rgbFrame->width * 3);}image = QImage(outputBuffer, rgbFrame->width, rgbFrame->height, QImage::Format_RGB888).copy();if (outputBuffer) {delete [] outputBuffer;outputBuffer = nullptr;}}if (buffer) {delete[] buffer;buffer = nullptr;}av_frame_free(&rgbFrame);}}}}av_frame_free(&frame);}break;}av_packet_unref(&packet);}avcodec_free_context(&codecContext);avformat_close_input(&fmtContext);return image;}Media::ThumHelper::~ThumHelper()
{}

调用demo

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QPushButton>
#include <QListWidget>class Widget : public QWidget
{Q_OBJECTpublic:Widget(QWidget *parent = 0);~Widget();private:QPushButton* m_loadVideo = nullptr;QListWidget *m_imageList;
};#endif // WIDGET_H
#include "widget.h"
#include "ThumHelper.h"
#include <QVBoxLayout>
#include <QFileDialog>
#include <QPixmap>
#include <QLabel>
#include <QDebug>
#include <QDateTime>
Widget::Widget(QWidget *parent): QWidget(parent)
{QVBoxLayout  *layout = new QVBoxLayout;m_loadVideo = new QPushButton(this);m_loadVideo->setText("Load Image");m_imageList = new QListWidget;layout->addWidget(m_loadVideo);layout->addWidget(m_imageList);this->setLayout(layout);connect(m_loadVideo, &QPushButton::clicked, [&]() {QStringList paths = QFileDialog::getOpenFileNames(nullptr, QString("Open File"), "",tr("Videos(*.mp4 *.irgd)"));Media::ThumHelper  helper;QDateTime dateTime = QDateTime::currentDateTime();qDebug() << "Begin Load:" << QDateTime::currentDateTime().toString("yyyy-mm-DD-hh-MM-ss");for (int i = 0; i < paths.count(); i++) {QString videoPath = paths.at(i);QListWidgetItem*  item = new QListWidgetItem(m_imageList);QImage image = helper.thumnail(videoPath);item->setSizeHint(QSize(image.width() / 4, image.height() / 4));QLabel* label = new QLabel(m_imageList);label->setPixmap(QPixmap::fromImage(image).scaled(image.width()/4, image.height()/4));m_imageList->addItem(item);m_imageList->setItemWidget(item, label);}qDebug() << "finish Load:" << QDateTime::currentDateTime().toString("yyyy-mm-DD-hh-MM-ss");});
}Widget::~Widget()
{}
#-------------------------------------------------
#
# Project created by QtCreator 2023-12-21T13:13:56
#
#-------------------------------------------------QT       += core guigreaterThan(QT_MAJOR_VERSION, 4): QT += widgetsTARGET = Media
TEMPLATE = app# The following define makes your compiler emit warnings if you use
# any feature of Qt which as been marked as deprecated (the exact warnings
# depend on your compiler). Please consult the documentation of the
# deprecated API in order to know how to port your code away from it.
DEFINES += QT_DEPRECATED_WARNINGS# You can also make your code fail to compile if you use deprecated APIs.
# In order to do so, uncomment the following line.
# You can also select to disable deprecated APIs only up to a certain version of Qt.
#DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000    # disables all the APIs deprecated before Qt 6.0.0SOURCES += main.cpp\widget.cppHEADERS  += widget.hINCLUDEPATH +=$$PWD/ffmpeg/4.2.1/includemessage($$PWD/ffmpeg/4.2.1/include)win32{CONFIG(debug,debug|release){LIBS+= -L$$PWD/ffmpeg/4.2.1/x64/libmessage($$PWD/ffmpeg/4.2.1/x64/lib)}
}LIBS+= -lavcodec \-lavfilter \-lavformat \-lswresample \-lswscale \-lpostproc \-lavutil \-lavdevice

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/237583.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【SpringMVC】SpringMVC的请求与响应

文章目录 0. Tomcat环境的配置1. PostMan工具介绍创建WorkSpace建立新的请求 2. 请求映射路径案例结构与代码案例结构案例代码 案例存在问题解决方案方法方法升级版——配置请求路径前缀注解总结 3. Get请求与Post请求案例结构与案例代码案例结构案例代码 Get请求Post请求接收中…

3. BlazorSignalRApp 结合使用 ASP.NET Core SignalR 和 Blazor

参考&#xff1a;https://learn.microsoft.com/zh-cn/aspnet/core/blazor/tutorials/signalr-blazor?viewaspnetcore-8.0&tabsvisual-studio 1.创建新项目 BlazorSignalRApp 2.添加项目依赖项 依赖项&#xff1a;Microsoft.AspNetCore.SignalR.Client 方式1 管理解决方案…

SOLIDWORKS Flow Simulation升力仿真分析

仿真飞车起飞和飞机起飞的原理相同,当等质量的空气同时通过机翼上表面和下表面时,会在机翼上下方形成不同流速,空气通过机翼上表面时流速大&#xff0c;压强较小;通过下表面时流速较小,压强大。此时飞车会受一个向上的合力,即向上的升力,空气速度越快,升力越大,当升力大于飞车重…

力扣每日一题day36[112.路径总和]

给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径&#xff0c;这条路径上所有节点值相加等于目标和 targetSum 。如果存在&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 叶子节点 是指没有子节点…

SQL基础:查询的基本使用

上一节我们讲述了记录的基本操作&#xff0c;这一节我们来单独讲一下查询。 查询基本结构 首先我们来看下查询的基本结构 SELECTcolumn1,column2,... FROMtable_name [WHEREcondition] [GROUP BYcolumn1, column2, ...] [HAVINGaggregate_function(column) condition] [ORDE…

【算法】算法题-20231222

这里写目录标题 一、1002. 查找共用字符二、1047. 删除字符串中的所有相邻重复项三、面试题 01.04. 回文排列 一、1002. 查找共用字符 给你一个字符串数组 words &#xff0c;请你找出所有在 words 的每个字符串中都出现的共用字符&#xff08; 包括重复字符&#xff09;&…

Linux cat命令教程:如何连接文件并打印到标准输出设备上(附实例详解和注意事项)

Linux cat命令介绍 cat命令&#xff0c;全称为concatenate&#xff0c;用于连接文件并打印到标准输出设备上。 Linux cat命令适用的Linux版本 cat命令在所有Linux发行版中都是可用的&#xff0c;包括但不限于Ubuntu, Debian, Fedora, RHEL, CentOS等。 Linux cat命令的基本…

SpringIOC之MethodBasedEvaluationContext

博主介绍&#xff1a;✌全网粉丝5W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面有丰富的经验…

bootstrap基础

&#xff08;一&#xff09;bootstrap初识 bootstrap由美国数学家Efron于20世界70年代创建。 bootstrap方法用于样本数较少时的数学统计和参数估计。其数学原理基于格里汶科定理。 格里汶科&#xff08; G l i v e n k o &#xff09;定理&#xff1a; \color{red}格里汶科&a…

React和umi搭建项目的操作步骤

​​​​​​一、react脚手架新建项目 (1.1)、命令行 前提&#xff1a;react ES2015,nodejs v8 npx create-react-app myReactName //2022年v16以下版本 myReactName(自定义项目名) react中文官网&#xff0c;快速上手&#xff1a;react中文官网 react框架&#xff0c;…

基于Linphone android sdk开发Android软话机

1.Linphone简介 1.1 简介 LinPhone是一个遵循GPL协议的开源网络电话或者IP语音电话&#xff08;VOIP&#xff09;系统&#xff0c;其主要如下。使用linphone&#xff0c;开发者可以在互联网上随意的通信&#xff0c;包括语音、视频、即时文本消息。linphone使用SIP协议&#…

CGAL的3D Alpha Shapes

假设我们给定一个二维或三维的点集S&#xff0c;我们希望得到类似“这些点形成的形状”的东西。这是一个相当模糊的概念&#xff0c;可能有许多可能的解释&#xff0c;阿尔法形状就是其中之一。阿尔法形状可用于从密集的无组织数据点集进行形状重建。事实上&#xff0c;阿尔法形…

Go和Java实现命令模式

Go和Java实现命令模式 下面通过一个烧烤的例子来说明命令模式的使用。 1、命令模式 命令模式是一种数据驱动的设计模式&#xff0c;它属于行为型模式。请求以命令的形式包裹在对象中&#xff0c;并传给调用对象。调 用对象寻找可以处理该命令的合适的对象&#xff0c;并把该…

在 MyBatis 中<应该怎么写

在 MyBatis 中&#xff0c;< 符号在 XML 配置文件中是一个特殊字符&#xff0c;用于标记 XML 标签的开始。因此&#xff0c;如果你在 MyBatis 的 if 标签中直接使用 < 符号&#xff0c;它会被解析为 XML 标签的开始&#xff0c;从而导致解析错误。 为了避免这个问题&…

用户管理第2节课--idea 2023.2 后端--实现基本数据库操作(操作user表) -- 自动生成

一、插件 Settings... 1.1 File -- Settings 1.2 Settings -- Plugins 1.2.1 搜索框&#xff0c;也可以直接搜索 1.3 Plugins -- 【输入 & 搜索】mybatis 1.3.1 插件不同功能介绍 1.3.2 翻译如下 1.4 选中 Update&#xff0c;更新下 1.4.1 更新中 1.4.2 Restart IDE 1…

C++ 类的析构函数和构造函数

构造函数 类的构造函数是类的一种特殊的成员函数&#xff0c;它会在每次创建类的新对象时执行。主要用来在创建对象时初始化对象即为对象成员变量赋初始值。 构造函数的名称与类的名称是完全相同的&#xff0c;并且不会返回任何类型&#xff0c;也不会返回 void。构造函数可用…

ARM GIC(一) cortex-A 处理器中断简介

对于ARM的处理器&#xff0c;中断给处理器提供了触觉&#xff0c;使处理器能够感知到外界的变化&#xff0c;从而实时的处理。本系列博文&#xff0c;是以ARM cortex-A系列处理器&#xff0c;来介绍ARM的soc中&#xff0c;中断的处理。 ARM cortex-A系列处理器&#xff0c;提供…

[node] Node.js的路由

[node] Node.js的路由 路由 & 路由解析路由信息的整合URL信息路由处理逻辑路由逻辑与URL信息的整合路由的使用 路由 & 路由解析 路由需要提供请求的 URL 和其他需要的 GET/POST 参数&#xff0c;随后路由需要根据这些数据来执行相应的代码。 因此&#xff0c;根据 HT…

Android 13 - Media框架(25)- OMXNodeInstance(二)

上一节我们了解了 OMXNodeInstance 的创建过程&#xff0c;以及 IOmx 服务和 OMXNodeInstance、OMX组件之间的联系。这一节我们将一起了解 ACodec 是如何通过 OMXNodeInstance 这个中间层进行端口定义设置&#xff0c;以及端口Buffer分配的。 OMXNodeInstance 的代码还是比较长…

Python之Django项目的功能配置

1.创建Django项目 进入项目管理目录&#xff0c;比如&#xff1a;D盘 执行命令&#xff1a;diango-admin startproject demo1 创建项目 如果提示diango命令不存在&#xff0c;搜索diango-admin程序的位置&#xff0c;然后加入到环境变量path中。 进入项目&#xff0c;cd demo…