0

トラックを複数の固定サイズ (30 秒) のクリップに分割しようとしています。各クリップには、それぞれ最初と最後に (5 秒) フェードイン/フェードアウトがあります。各コマンドを ES6 Promise の配列にnode-fluent-ffmpeg接続しffmpegて保存するために使用しています。これは後で.ffmpegPromise.all()

トラックをクリップしてフェードイン フィルターを追加することはできますが、何らかの理由でフェードアウト フィルターがトラックの最初のクリップにしか適用されません。ffmpegnode-fluent-ffmpegドキュメント (ここここ)の両方で回答を探しましたが、複数回クリップされているトラックにフェードアウト フィルターを適用することから生じる問題については言及されていません。

私のコードは以下のスニペットと非常によく似ており、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
4

1 に答える 1

2

上記のコードでさまざまなトラックをクリッピングして、しばらくいじくり回した後、私は問題を理解しました。

問題

問題は、フェードの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}`
    }
])

そのため、最初のクリップのフェードアウトしか聞こえませんでした。フェードインも機能していませんでしたが、最初のいくつかのクリップに自然に聞こえるフェードインがあったため、気づきにくくなりました (トラックを上に移動していて、最終的には適用されませんでしたが)。

ソリューション

それは非常に単純であることがわかりました。フェードインの場合は常に、フェードアウトの場合は常に間隔の長さ (30) からフェード持続時間 (3) を引いた値にoptionsなるように、キーの値を置き換える必要がありました。start0start

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

結局簡単な問題でした。ただし、クリッピングとフェードの適用を同時に行っている場合は、元のトラックではなく、クリッピングされたトラックを基準にして時間を設定する必要があることに注意してください。

于 2016-06-04T16:40:17.100 に答える