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

时间:2016-05-29 20:55:01

标签: node.js ffmpeg ecmascript-6 es6-promise fluent-ffmpeg

我试图将曲目分成多个固定大小(30秒)的剪辑,每个剪辑在开头和结尾分别有一个(5秒)淡入/淡出。我使用node-fluent-ffmpegffmpeg进行交互,并将每个ffmpeg命令保存在ES6 Promises数组中,稍后我将使用Promise.all()执行这些命令。

我能够剪辑曲目并成功添加淡入滤镜,但由于某种原因,淡出滤镜仅应用于曲目的第一个剪辑。我在ffmpegnode-fluent-ffmpeg文档(herehere)中查找了答案,但没有提及将淡出过滤器应用于被多次剪裁的曲目。

我的代码与下面的代码段非常相似,使用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

1 个答案:

答案 0 :(得分:1)

经过一段时间的修补,用上面的代码剪掉不同的曲目后,我发现了问题。

问题

问题在于我如何跟踪淡出的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}`
    }
])

最后这是一个简单的问题。然而,指出如果你同时剪裁和应用淡入淡出,你需要确保你的时间与剪辑的曲目有关,而不是原始曲目。

相关问题