ffmpeg利用crop滤镜进行视频裁剪

本文介绍使用FFmpeg进行视频裁剪的方法,包括命令行操作及C++代码实现过程。详细解析了如何通过构建filter滤镜进行视频裁剪,并演示了具体的代码实现。

首先可以用ffmpeg命令行对视频进行裁剪,原始文件是1920x1080尺寸。
ffmpeg -i e:/in-vs.mp4 -vf crop=‘1280:640:200:200’ out_crop.mp4

其中1280和640代表裁剪后的宽和高,200和200分别代表裁剪的x坐标和y坐标(基于原始图像)。

下面用代码给予说明:

ret = avfilter_graph_create_filter(&m_pFilterCtxSrcVideoA, filter_src_videoA, pad_name_videoA, args_videoA, NULL, m_pFilterGraph);
if (ret < 0)
{
   
   
	break;
}

AVFilterContext *cropFilter_ctx;
ret = avfilter_graph_create_filter(&cropFilter_ctx, filter_crop, name_crop, filter_desc, NULL, m_pFilterGraph);
if (ret < 0)
{
   
   
	break;
}

ret = avfilter_graph_create_filter(&m_pFilterCtxSink, filter_sink, "out", NULL, NULL, m_pFilterGraph);
if (ret < 0)
{
   
   
	break;
}

这里面构建了三个滤镜,buffer,crop,buffersink。
其中buffer用于接收读取到的yuv数据,crop用于裁剪, buffersink用于接收裁剪后的数据。

下面是滤镜的连接:

ret = avfilter_link(m_pFilterCtxSrcVideoA, 0, cropFilter_ctx, 0);
if (ret != 0)
{
   
   
	break;
}

ret = avfilter_link(cropFilter_ctx, 0, m_pFilterCtxSink, 0);
if (ret != 0)
{
   
   
	break;
}

ret = avfilter_graph_config(m_pFilterGraph, NULL);
if (ret < 0)
{
   
   
	break;
}

在开发过程中,出现一个很低级的错误,后面用写yuv数据,然后用相应的yuv查看工具才发现代码中的具体错误位置,CropFile.cpp文件中的SaveAvFrame用于将AVFrame写入到文件。

代码结构如下:
在这里插入图片描述

main函数所在文件FfmpegCropTest.cpp的内容如下:

#include <iostream>
#include "CropFile.h"

#ifdef	__cplusplus
extern "C"
{
   
   
#endif

#pragma comment(lib, "avcodec.lib")
#pragma comment(lib, "avformat.lib")
#pragma comment(lib, "avutil.lib")
#pragma comment(lib, "avdevice.lib")
#pragma comment(lib, "avfilter.lib")
#pragma comment(lib, "postproc.lib")
#pragma comment(lib, "swresample.lib")
#pragma comment(lib, "swscale.lib")


#ifdef __cplusplus
};
#endif





int main()
{
   
   
	CCropFile cVideoCopy;

	const char *pFileA = "E:\\learn\\ffmpeg\\FfmpegFilterTest\\x64\\Release\\in-vs.mp4";

	const char *pFileOut = "E:\\learn\\ffmpeg\\FfmpegFilterTest\\x64\\Release\\out-crop.mp4";

	cVideoCopy.StartCrop(200, 200, 1280, 640, pFileA, pFileOut);
	cVideoCopy.WaitFinish();
	return 0;
}



CropFile.h内容如下:

#pragma once

#include <Windows.h>

#ifdef	__cplusplus
extern "C"
{
   
   
#endif
#include "libavcodec/avcodec.h"
#include "libavformat/avformat.h"
#include "libswscale/swscale.h"
#include "libswresample/swresample.h"
#include "libavdevice/avdevice.h"
#include "libavutil/audio_fifo.h"
#include "libavutil/avutil.h"
#include "libavutil/fifo.h"
#include "libavutil/frame.h"
#include "libavutil/imgutils.h"

#include "libavfilter/avfilter.h"
#include "libavfilter/buffersink.h"
#include "libavfilter/buffersrc.h"


#ifdef __cplusplus
};
#endif

class CCropFile
{
   
   
public:
	CCropFile();
	~CCropFile();
public:
	int StartCrop(int x, int y, int width, int height, const char *pFileA, const char *pFileOut);
	int WaitFinish();
private:
	int OpenFileA(const char *pFileA);
	int OpenOutPut(const char *pFileOut);
	int InitFilter(const char* filter_desc);
private:
	static DWORD WINAPI VideoAReadProc(LPVOID lpParam);
	void VideoARead();


	static DWORD WINAPI VideoCropProc(LPVOID lpParam);
	void VideoCrop();
private:
	AVFormatContext *m_pFormatCtx_FileA = NULL;

	AVCodecContext *m_pReadCodecCtx_VideoA = NULL;
	AVCodec *m_pReadCodec_VideoA = NULL;


	AVCodecContext	*m_pCodecEncodeCtx_Video = NULL;
	AVFormatContext *m_pFormatCtx_Out = NULL;

	AVFifoBuffer *m_pVideoAFifo = NULL
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值