FFmpeg を使用してビデオのいくつかのセクションを削除しようとしています。
たとえば、テレビ番組を録画して、コマーシャルをカットしたいとします。これは GUI ビデオ エディタを使えば簡単です。削除する各クリップの先頭と末尾をマークして、削除を選択するだけです。私は、FFmpeg を使用して、コマンド ラインから同じことを実行しようとしています。
1つのセグメントをカットする方法は知っています新しい次のようなビデオ:
ffmpeg -i input.avi -ss 00:00:20 -t 00:00:05 -map 0 -codec copy output.avi
これは5秒のクリップを切り取って新しいビデオファイルとして保存しますが、逆にビデオ全体を保存するにはどうすればよいでしょうか。それなし指定されたクリップを削除するにはどうすればよいですか?また、複数のクリップを削除するように指定するにはどうすればよいですか?
たとえば、私のビデオが ABCDEFG で表せる場合、ACDFG で構成される新しいビデオを作成したいと思います。
答え1
まあ、まだ使えますtrim
それをフィルタリングします。ここでは例として、ビデオの最初と最後、および中間の 3 つのセグメントを切り取るとします。
ffmpeg -i in.ts -filter_complex \
"[0:v]trim=duration=30[a]; \
[0:v]trim=start=40:end=50,setpts=PTS-STARTPTS[b]; \
[a][b]concat[c]; \
[0:v]trim=start=80,setpts=PTS-STARTPTS[d]; \
[c][d]concat[out1]" -map [out1] out.ts
ここで何をしたかというと、最初の30秒、40~50秒、そして最後の80秒を切り取って、それをストリームに組み合わせてout1
、concat
30〜40秒(10秒)と50〜80秒(30秒)残してフィルターします。
setpts について: トリムは画像の表示時間を変更せず、10 秒を切り取るとデコーダー カウンターはこの 10 秒間のフレームを認識しないため、これが必要になります。
オーディオも使用したい場合は、オーディオ ストリームに対して同じことを行う必要があります。コマンドは次のようになります。
ffmpeg -i utv.ts -filter_complex \
"[0:v]trim=duration=30[av];[0:a]atrim=duration=30[aa];\
[0:v]trim=start=40:end=50,setpts=PTS-STARTPTS[bv];\
[0:a]atrim=start=40:end=50,asetpts=PTS-STARTPTS[ba];\
[av][bv]concat[cv];[aa][ba]concat=v=0:a=1[ca];\
[0:v]trim=start=80,setpts=PTS-STARTPTS[dv];\
[0:a]atrim=start=80,asetpts=PTS-STARTPTS[da];\
[cv][dv]concat[outv];[ca][da]concat=v=0:a=1[outa]" -map [outv] -map [outa] out.ts
答え2
ptQa のソリューションがうまく機能しないのは、主に、フィルターからのエラーの意味や修正方法がわからないからです。私のソリューションは、混乱が残る可能性があるため、少し扱いにくいようですが、スクリプトに組み込む場合は、クリーンアップを自動化できます。また、このアプローチは、ステップ 4 で問題が発生した場合でも、ステップ 1 ~ 3 が完了しているため、エラーからの回復が少し効率的になるという点でも気に入っています。
基本的な戦略は-t
、 と を使用し-ss
て必要な各セグメントのビデオを取得し、すべての部分を結合して最終バージョンを作成することです。
たとえば、それぞれ 5 秒の長さの 6 つのセグメント ABCDEF があり、A (0 ~ 5 秒)、C (10 ~ 15 秒)、E (20 ~ 25 秒) が必要な場合は、次のようにします。
ffmpeg -i abcdef.tvshow -t 5 a.tvshow -ss 10 -t 5 c.tvshow -ss 20 -t 5 e.tvshow
または
ffmpeg -i abcdef.tvshow -t 0:00:05 a.tvshow -ss 0:00:10 -t 0:00:05 c.tvshow -ss 0:00:20 -t 0:00:05 e.tvshow
これにより、ファイル a.tvshow、c.tvshow、e.tvshow が作成されます。 は-t
各クリップの長さを指定するため、c が 30 秒の場合は 30 または 0:00:30 を渡すことができます。-ss
オプションはソース ビデオにスキップする距離を指定するため、常にファイルの先頭からの相対位置になります。
たくさんのビデオ ファイルができたら、ace-files.txt
次のようなファイルを作成します。
file 'a.tvshow'
file 'c.tvshow'
file 'e.tvshow'
先頭の「file」とその後のエスケープされたファイル名に注意してください。
次にコマンド:
ffmpeg -f concat -i ace-files.txt -c copy ace.tvshow
これにより、すべてのファイルが連結されabe-files.txt
、オーディオおよびビデオ コーデックがコピーされ、セクション a、c、e のみのファイルが作成されます。次に、、、およびをace.tvshow
忘れずに削除してくださいace-files.txt
。a.tvshow
c.tvshow
e.tvshow
免責事項: 他のアプローチと比較して、これがどの程度効率的か非効率的かはわかりませんがffmpeg
、私の目的には適しています。誰かの役に立てば幸いです。
答え3
ptQa のアプローチに従うのが難しい人には、もう少し合理的な方法があります。各ステップを連結するのではなく、最後にすべて実行します。
入力ごとに、A/V ペアを定義します。
//Input1:
[0:v]trim=start=10:end=20,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10:end=20,asetpts=PTS-STARTPTS[0a];
//Input2:
[0:v]trim=start=30:end=40,setpts=PTS-STARTPTS,format=yuv420p[1v];
[0:a]atrim=start=30:end=40,asetpts=PTS-STARTPTS[1a];
//Input3:
[0:v]trim=start=30:end=40,setpts=PTS-STARTPTS,format=yuv420p[2v];
[0:a]atrim=start=30:end=40,asetpts=PTS-STARTPTS[2a];
必要な数のペアを定義し、それらをすべて 1 回のパスで連結します。ここで、n は合計入力数です。
[0v][0a][1v][1a][2v][2a]concat=n=3:v=1:a=1[outv][outa] -map [outv] -map [outa] out.mp4
これはループ内で簡単に構築できます。
2 つの入力を使用する完全なコマンドは次のようになります。
ffmpeg -i in.mp4 -filter_complex
[0:v]trim=start=10.0:end=15.0,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10.0:end=15.0,asetpts=PTS-STARTPTS[0a];
[0:v]trim=start=65.0:end=70.0,setpts=PTS-STARTPTS,format=yuv420p[1v];
[0:a]atrim=start=65.0:end=70.0,asetpts=PTS-STARTPTS[1a];[0v][0a][1v]
[1a]concat=n=2:v=1:a=1[outv][outa] -map [outv] -map [outa] out.mp4
答え4
録画したテレビ番組の編集を高速化するスクリプトを作成しました。このスクリプトは、保存したいセグメントの開始時間と終了時間を聞いて、それらをファイルに分割します。次のオプションがあります。
- 1 つまたは複数のセグメントを取得します。
- セグメントを 1 つのファイルに結合できます。
- 結合後、パーツファイルを保持または削除できます。
- 元のファイルをそのまま保存することも、新しいファイルに置き換えることもできます。
どう考えているか教えてください。
#!/bin/bash
/bin/date >>segmenter.log
function segment (){
while true; do
echo "Would you like to cut out a segment ?"
echo -e "1) Yes\n2) No\n3) Quit"
read CHOICE
if [ "$CHOICE" == "3" ]; then
exit
elif [ "$CHOICE" == "2" ]; then
clear
break
elif [ "$CHOICE" == "1" ]; then
clear
((segments++))
echo "What time does segment $segments start ?"
read SEGSTART
clear
echo -e "Segment $segments start set to $SEGSTART\n"
echo "What time does segment $segments end ?"
read SEGEND
clear
echo -e "Segment $segments end set to $SEGEND\n"
break
else
clear
echo -e "Bad option"
segment "$segments"
fi
done
if [ "$CHOICE" == "1" ]; then
echo "Cutting file $file video segment $segments starting at $SEGSTART and ending at $SEGEND"
ffmpeg -i "$file" -ss $SEGSTART -to $SEGEND -map 0:0 -map 0:1 -c:a copy -c:v copy "$filename-part$segments.$extension" >> segmenter.log 2>&1
clear
echo -e "Cut file $filename-part$segments.$extension starting at $SEGSTART and ending at $SEGEND\n"
segment "$segments"
fi
}
file="$1"
filename="${file%.*}"
extension="${file##*.}"
clear
segments=0
segment "$segments"
clear
if (("$segments"==1)); then
mv $filename"-part1."$extension "$filename-segmented.$extension"
elif (("$segments">1)); then
echo "Would you like to join the segments into one file ?"
OPTIONS="Yes No Quit"
select opt in $OPTIONS; do
clear
if [ "$opt" == "Quit" ]; then
exit
elif [ "$opt" == "Yes" ]; then
clear
echo "Joining segments"
ffmpeg -f concat -i <(for f in $filename"-part"*$extension; do echo "file '$(pwd)/$f'"; done) -c:a copy -c:v copy "$filename-segmented.$extension" >> segmenter.log 2>&1
clear
echo "Would you like to delete the part files ?"
select opt in $OPTIONS; do
clear
if [ "$opt" == "Quit" ]; then
exit
elif [ "$opt" == "Yes" ]; then
for f in $filename"-part"*$extension; do rm $f; done
break
elif [ "$opt" == "No" ]; then
break
else
clear
echo -e "Bad option\n"
fi
done
break
clear
elif [ "$opt" == "No" ]; then
exit
else
clear
echo -e "Bad option\n"
fi
done
fi
echo "Would you like to replace the original file with the result of your changes ?"
OPTIONS="Yes No Quit"
select opt in $OPTIONS; do
clear
if [ "$opt" == "Quit" ]; then
exit
elif [ "$opt" == "Yes" ]; then
rm $file
mv "$filename-segmented.$extension" $file
break
elif [ "$opt" == "No" ]; then
break
else
clear
echo -e "Bad option\n"
fi
done