FFmpeg 淡出音频过滤器(afade)未应用于迭代剪辑的轨道
FFmpeg Fade Out Audio Filter (afade) not being Applied to Track Clipped Iteratively
我正在尝试将曲目拆分为多个固定大小(30 秒)的片段,每个片段在开始和结束时分别有一个(5 秒)淡入/淡出。我正在使用 node-fluent-ffmpeg
与 ffmpeg
交互,并将每个 ffmpeg
命令保存在我稍后使用 Promise.all()
执行的 ES6 Promise 数组中。
我能够剪辑轨道并成功添加淡入滤镜,但由于某种原因,淡出滤镜仅应用于轨道的第一个剪辑。我在 ffmpeg
和 node-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
经过一段时间的修改,用上面的代码剪辑不同的轨道,我找到了问题。
问题
问题在于我如何跟踪渐变的 start
和 end
次。事实证明,淡入淡出的时间是针对 30 秒剪辑而不是原始曲目本身进行测量的。另一方面,我使用下面的循环不断增加 start
和 end
时间,认为淡入淡出是在与原始轨道相关的时间步应用的。
for (var i = 0; i <= Math.floor(duration) - interval; ++i) {
const start = i; // Start second.
const end = start + interval;
...
因此,通过使用循环更新 start
和 end
值,我在应用音频过滤器时超越了 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}`
}
])
最后是一个简单的问题。不过,值得指出的是,如果您同时剪辑和应用淡入淡出,则需要确保您的时间是关于剪辑的轨道,而不是原始轨道。
我正在尝试将曲目拆分为多个固定大小(30 秒)的片段,每个片段在开始和结束时分别有一个(5 秒)淡入/淡出。我正在使用 node-fluent-ffmpeg
与 ffmpeg
交互,并将每个 ffmpeg
命令保存在我稍后使用 Promise.all()
执行的 ES6 Promise 数组中。
我能够剪辑轨道并成功添加淡入滤镜,但由于某种原因,淡出滤镜仅应用于轨道的第一个剪辑。我在 ffmpeg
和 node-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
经过一段时间的修改,用上面的代码剪辑不同的轨道,我找到了问题。
问题
问题在于我如何跟踪渐变的 start
和 end
次。事实证明,淡入淡出的时间是针对 30 秒剪辑而不是原始曲目本身进行测量的。另一方面,我使用下面的循环不断增加 start
和 end
时间,认为淡入淡出是在与原始轨道相关的时间步应用的。
for (var i = 0; i <= Math.floor(duration) - interval; ++i) {
const start = i; // Start second.
const end = start + interval;
...
因此,通过使用循环更新 start
和 end
值,我在应用音频过滤器时超越了 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}`
}
])
最后是一个简单的问题。不过,值得指出的是,如果您同时剪辑和应用淡入淡出,则需要确保您的时间是关于剪辑的轨道,而不是原始轨道。