web-dev-qa-db-ja.com

FFmpegを使用してビデオから複数のセグメントを削除するにはどうすればよいですか?

FFmpegを使用してビデオのいくつかのセクションを削除しようとしています。

たとえば、テレビで番組を録画し、コマーシャルを切り取りたい場合を想像してみてください。これは、GUIビデオエディタを使用すると簡単です。削除する各クリップの開始と終了をマークし、削除を選択するだけです。私はFFmpegでコマンドラインから同じことをしようとしています。

私は単一セグメントをnewビデオにカットする方法を知っています:

ffmpeg -i input.avi -ss 00:00:20 -t 00:00:05 -map 0 -codec copy output.avi

これは5秒のクリップをカットして新しいビデオファイルとして保存しますが、反対の方法でビデオ全体を保存するにはどうすればよいですかなし指定したクリップ、および複数のクリップを指定するには削除されましたか?

たとえば、自分のビデオがABCDEFGで表現できる場合、ACDFGで構成される新しいビデオを作成します。

56
Matias

PtQaによって提供された答えは機能するようですが、私は正常に機能することを証明した他のソリューションを開発しました。

基本的に、私が行うことは、結果に含めたい元のビデオの各部分に対して1つのビデオをカットすることです。後で、私はそれらをConcat Demuxerで説明した here と連結します。

解決策は、私が最初に試し、同期の問題を提示した場合と同じです。追加したのは、さまざまなビデオを生成するときのコマンド-avoid_negative_ts 1です。このソリューションでは、同期の問題は解消されます。

3
Matias

まあ、それでも trim フィルターを使用できます。以下に例を示します。セグメントを30〜40秒(10秒)および50〜80秒(30秒)切り取ると仮定します。

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秒をトリミングして終了し、それらを組み合わせてストリームout1concat フィルターを使用します。

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
49
ptQa

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'

先頭の「ファイル」とその後のエスケープされたファイル名に注意してください。

次に、コマンド:

ffmpeg -f concat -i ace-files.txt -c copy ace.tvshow

これにより、abe-files.txt内のすべてのファイルが連結され、それらのオーディオコーデックとビデオコーデックがコピーされて、ファイルace.tvshowが作成されます。次に、ace-files.txta.tvshowc.tvshowおよびe.tvshowを削除することを忘れないでください。

免責事項:これがffmpegの点で他のアプローチと比較して(非)効率的であるかどうかはわかりませんが、私の目的では、より効果的に機能します。それが誰かを助けることを願っています。

18
xbakesx

録画したテレビの編集をスピードアップするスクリプトを作成しました。スクリプトは、保持したいセグメントの開始時間と終了時間を尋ね、ファイルに分割します。次のオプションがあります。

  • 1つまたは複数のセグメントを取得します。
  • セグメントを1つの結果ファイルに結合できます。
  • 参加後、パーツファイルを保持または削除できます。
  • 元のファイルを保持するか、新しいファイルに置き換えることができます。

実際の動画: here

どう考えているか教えてください。

 #!/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
5
rocuinneagain

PtQaのアプローチで問題が発生した場合は、少し効率的な方法で対処できます。方法の各ステップを連結するのではなく、すべてを最後に実行してください。

入力ごとに、A/Vペアを定義します。

//Input1:
[0:v]trim=start=10:end=20,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10:end=10,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つの入力を使用する完全なコマンドは次のようになります。

 -y -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
3
shawnblais