FFMPEG的filter滤镜使用

前段时间研究了下ffmpeg的滤镜功能,效果还不错,部分滤镜的描述字符串,需要一些图像处理的的知识,如图像的伽马值、亮度、对比度等对图像的影响。

主要使用了部分滤镜:

const char *filter_mirror = "crop=iw/2:ih:0:0,split[left][tmp];[tmp]hflip[right]; \
												[left]pad=iw*2[a];[a][right]overlay=w";
const char *filter_watermark = "movie=test.jpg[wm];[in][wm]overlay=5:5[out]";
const char *filter_negate = "negate[out]";
const char *filter_edge = "edgedetect[out]";
const char *filter_split4 = "scale=iw/2:ih/2[in_tmp];[in_tmp]split=4[in_1][in_2][in_3][in_4];[in_1]pad=iw*2:ih*2[a];[a][in_2]overlay=w[b];[b][in_3]overlay=0:h[d];[d][in_4]overlay=w:h[out]";
const char *filter_vintage = "curves=vintage";//复古
const char *fileter_vignette = "vignette=PI/4";//光晕
const char *fileter_colorDown = "colorlevels=rimin=0.058:gimin=0.058:bimin=0.058";// 变暗
const char *fileter_colorUp = "fftfilt=dc_Y=0:weight_Y='exp(-4 * ((Y+X)/(W+H)))'";// 增加对比度
const char *fileter_hqdn3d = "hqdn3d=luma_spatial=15.0";//降噪
const char *fileter_strong_contrast = "curves=strong_contrast";//强对比度
const char *fileter_lighter = "curves=lighter";//变量

// "scale=iw/2:ih/2[in_tmp];[in_tmp]split=4[in_1][in_2][in_3][in_4];[in_1]pad=iw*2:ih*2[a];[a][in_2]overlay=w[b];[b][in_3]overlay=0:h[d];[d][in_4]overlay=w:h[out]" 视频组和,本条语句是4个输入视频源的组合
// "crop=iw/2:ih:0:0,split[left][tmp];[tmp]hflip[right]; [left]pad=iw*2[a];[a][right]overlay=w" 镜像
// "movie=test.jpg[wm];[in][wm]overlay=5:5[out]" 水印
// "fftfilt=dc_Y=0:weight_Y='1+squish(1-(Y+X)/100)'";锐化
// "fftfilt=dc_Y=0:weight_Y='squish((Y+X)/100-1)'"; 低通滤波
// "fftfilt=dc_Y=128:weight_Y='squish(1-(Y+X)/100)'"; 高通滤波
// "colorlevels=rimin=0.058:gimin=0.058:bimin=0.058"; 变暗
// "colorlevels=rimin=0.039:gimin=0.039:bimin=0.039:rimax=0.96:gimax=0.96:bimax=0.96"; 增加对比度
// "hqdn3d" 降噪
// "curves=strong_contrast"强对比度
// "curves=vintage" 复古
// "edgedetect[out]" 边缘检测
// "negate[out]" 底片
/*
curves = 
‘none’
‘color_negative’ // 彩色底片
‘cross_process’
‘darker’ // 变暗
‘increase_contrast’// 增加对比度
‘lighter’ // 变亮
‘linear_contrast’ // 线性对比
‘medium_contrast’
‘negative’ // 底片
‘strong_contrast’
‘vintage’ // 复古
*/

调用av_buffersrc_add_frame(buffersrc_ctx, pframe)对编码前的视频帧进行处理处理完便可编码发送。

以下为测试的效果:

1.原图

2.复古

3.光晕

4.降噪

5.强对比度

6.变亮

7.变暗

以上为效果图,可以看看,降噪效果还不错,如果你觉得图中效果不够好,可以去调节滤镜参数去调节效果。



  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
滤镜FFmpeg中的一个重要概念,它可以对音视频进行各种处理,例如裁剪、旋转、缩放、调整亮度、对比度等等。FFmpeg提供了丰富的滤镜库,可以通过API在代码中使用滤镜。下面是一个简单的使用FFmpeg API添加滤镜的例子: ```c #include <libavfilter/avfilter.h> #include <libavfilter/buffersink.h> #include <libavfilter/buffersrc.h> AVFilterContext *buffersink_ctx; AVFilterContext *buffersrc_ctx; AVFilterGraph *filter_graph; int video_stream_index = -1; // 初始化滤镜 int init_filters(const char *filters_descr) { char args[512]; int ret = 0; AVFilter *buffersrc = avfilter_get_by_name("buffer"); AVFilter *buffersink = avfilter_get_by_name("buffersink"); AVFilterInOut *outputs = avfilter_inout_alloc(); AVFilterInOut *inputs = avfilter_inout_alloc(); enum AVPixelFormat pix_fmts[] = { AV_PIX_FMT_YUV420P, AV_PIX_FMT_NONE }; filter_graph = avfilter_graph_alloc(); if (!outputs || !inputs || !filter_graph) { ret = AVERROR(ENOMEM); goto end; } // 设置输入参数 snprintf(args, sizeof(args), "video_size=%dx%d:pix_fmt=%d:time_base=%d/%d:pixel_aspect=%d/%d", buffersrc_ctx->inputs[0]->width, buffersrc_ctx->inputs[0]->height, buffersrc_ctx->inputs[0]->format, buffersrc_ctx->inputs[0]->time_base.num, buffersrc_ctx->inputs[0]->time_base.den, buffersrc_ctx->inputs[0]->sample_aspect_ratio.num, buffersrc_ctx->inputs[0]->sample_aspect_ratio.den); // 添加输入滤镜 ret = avfilter_graph_create_filter(&buffersrc_ctx, buffersrc, "in", args, NULL, filter_graph); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot create buffer source\n"); goto end; } // 添加输出滤镜 ret = avfilter_graph_create_filter(&buffersink_ctx, buffersink, "out", NULL, NULL, filter_graph); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot create buffer sink\n"); goto end; } // 设置输出参数 ret = av_opt_set_int_list(buffersink_ctx, "pix_fmts", pix_fmts, AV_PIX_FMT_NONE, AV_OPT_SEARCH_CHILDREN); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot set output pixel format\n"); goto end; } // 添加滤镜 outputs->name = av_strdup("in"); outputs->filter_ctx = buffersrc_ctx; outputs->pad_idx = 0; outputs->next = NULL; inputs->name = av_strdup("out"); inputs->filter_ctx = buffersink_ctx; inputs->pad_idx = 0; inputs->next = NULL; if ((ret = avfilter_graph_parse_ptr(filter_graph, filters_descr, &inputs, &outputs, NULL)) < 0) goto end; if ((ret = avfilter_graph_config(filter_graph, NULL)) < 0) goto end; end: avfilter_inout_free(&inputs); avfilter_inout_free(&outputs); return ret; } // 处理滤镜 int filter_encode_write_frame(AVFrame *frame, AVCodecContext *enc_ctx, AVFormatContext *fmt_ctx) { int ret; AVFrame *filt_frame = av_frame_alloc(); if (!filt_frame) { ret = AVERROR(ENOMEM); goto end; } // 添加滤镜 if (av_buffersrc_add_frame_flags(buffersrc_ctx, frame, AV_BUFFERSRC_FLAG_KEEP_REF) < 0) { av_log(NULL, AV_LOG_ERROR, "Error while feeding the filtergraph\n"); goto end; } // 获取输出帧 while (1) { ret = av_buffersink_get_frame(buffersink_ctx, filt_frame); if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) break; if (ret < 0) goto end; // 编码并写入文件 filt_frame->pict_type = AV_PICTURE_TYPE_NONE; ret = encode_write_frame(filt_frame, enc_ctx, fmt_ctx); av_frame_unref(filt_frame); if (ret < 0) goto end; } end: av_frame_free(&filt_frame); return ret; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值