FFmpeg 淡出音频过滤器(afade)未应用于迭代剪辑的轨道

FFmpeg Fade Out Audio Filter (afade) not being Applied to Track Clipped Iteratively

我正在尝试将曲目拆分为多个固定大小(30 秒)的片段,每个片段在开始和结束时分别有一个(5 秒)淡入/淡出。我正在使用 node-fluent-ffmpegffmpeg 交互,并将每个 ffmpeg 命令保存在我稍后使用 Promise.all() 执行的 ES6 Promise 数组中。

我能够剪辑轨道并成功添加淡入滤镜,但由于某种原因,淡出滤镜仅应用于轨道的第一个剪辑。我在 ffmpegnode-fluent-ffmpeg 文档 (here and here) 中四处寻找答案,但没有提及将淡出滤镜应用于正在剪辑的轨道所引起的问题多次。

我的代码与下面的代码片段非常相似,使用 audioFilters 方法按顺序应用音频过滤器。请注意,我曾尝试只保留淡出滤镜,但问题仍然存在。任何指针将不胜感激。

var promises = [];
const duration = track.duration;
const interval = 30;
const fade = 5;             
const bitrate = 128;        

for (var i = 0; i <= Math.floor(duration) - interval; ++i) {
    const start = i;            // Start second.      
    const end = start + interval;
    const mp3 = `${new ObjectId().toHexString()}.mp3`;

    var command = new Promise((resolve, reject) => {
        ffmpeg(path).setStartTime(start)
                    .audioBitrate(bitrate)
                    .audioFilters([
                        {
                            filter: 'afade',
                            options: `t=in:ss=${start}:d=${fade}`
                        },
                        {
                            filter: 'afade',
                            options: `t=out:st=${end - fade}:d=${fade}`
                        }
                    ])
                    .duration(interval)
                    .on('error', (err) => {
                        reject("An error occurred while clipping.");
                    })
                    .on('end', () => {
                        resolve(`Finished processing ${output}.`);
                    })
                    .save(mp3);
    });
    promises.push(command);     
}

这是我的 ffmpeg 版本信息:

ffmpeg version 2.8.6 Copyright (c) 2000-2016 the FFmpeg developers
built with Apple LLVM version 7.0.2 (clang-700.1.81)
configuration: --prefix=/usr/local/Cellar/ffmpeg/2.8.6 --enable-shared --enable-pthreads --enable-gpl --enable-version3 --enable-hardcoded-tables --enable-avresample --cc=clang --host-cflags= --host-ldflags= --enable-opencl --enable-libx264 --enable-libmp3lame --enable-libvo-aacenc --enable-libxvid --enable-vda
libavutil      54. 31.100 / 54. 31.100
libavcodec     56. 60.100 / 56. 60.100
libavformat    56. 40.101 / 56. 40.101
libavdevice    56.  4.100 / 56.  4.100
libavfilter     5. 40.101 /  5. 40.101
libavresample   2.  1.  0 /  2.  1.  0
libswscale      3.  1.101 /  3.  1.101
libswresample   1.  2.101 /  1.  2.101
libpostproc    53.  3.100 / 53.  3.100

经过一段时间的修改,用上面的代码剪辑不同的轨道,我找到了问题。

问题

问题在于我如何跟踪渐变的 startend 次。事实证明,淡入淡出的时间是针对 30 秒剪辑而不是原始曲目本身进行测量的。另一方面,我使用下面的循环不断增加 startend 时间,认为淡入淡出是在与原始轨道相关的时间步应用的。

for (var i = 0; i <= Math.floor(duration) - interval; ++i) {
    const start = i;            // Start second.      
    const end = start + interval;
    ...

因此,通过使用循环更新 startend 值,我在应用音频过滤器时超越了 30 秒剪辑的结尾。

.audioFilters([
    {
        filter: 'afade',
        options: `t=in:ss=${start}:d=${fade}`
    },
    {
        filter: 'afade',
        options: `t=out:st=${end - fade}:d=${fade}`
    }
])

这就是为什么我只能听到第一个剪辑的淡出。淡入也不起作用,但很难注意到,因为前几个剪辑会出现听起来自然的淡入(即使它也在轨道上移动并最终没有被应用)。

解决方案

事实证明这非常简单。我只需要替换 options 键中的值,以便淡入的 start 始终是 0 并且淡出的 start 始终是间隔长度(30) 减去淡入淡出持续时间 (3).

.audioFilters([
    {
        filter: 'afade',
        options: `t=in:ss=0:d=${fade}`
    },
    {
        filter: 'afade',
        options: `t=out:st=${interval - fade}:d=${fade}`
    }
])

最后是一个简单的问题。不过,值得指出的是,如果您同时剪辑和应用淡入淡出,则需要确保您的时间是关于剪辑的轨道,而不是原始轨道。