ฉันจะลบหลายส่วนออกจากวิดีโอโดยใช้ FFmpeg ได้อย่างไร


51

ฉันกำลังพยายามลบบางส่วนของวิดีโอโดยใช้ FFmpeg

ตัวอย่างเช่นสมมติว่าคุณบันทึกรายการโทรทัศน์และต้องการตัดโฆษณาออก นี่เป็นเรื่องง่ายด้วยโปรแกรมแก้ไขวิดีโอ GUI คุณเพียงทำเครื่องหมายจุดเริ่มต้นและจุดสิ้นสุดของแต่ละคลิปที่จะลบและเลือกลบ ฉันพยายามทำสิ่งเดียวกันจากบรรทัดคำสั่งด้วย FFmpeg

ฉันรู้วิธีตัดส่วนเดียวไปยังวิดีโอใหม่เช่น:

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

วิธีนี้จะตัดคลิปห้าวินาทีและบันทึกเป็นไฟล์วิดีโอใหม่ แต่ฉันจะทำตรงข้ามและบันทึกวิดีโอทั้งหมดโดยไม่ต้องใช้คลิปที่ระบุได้อย่างไรและฉันจะระบุคลิปที่จะลบได้อย่างไร

ตัวอย่างเช่นหากวิดีโอของฉันสามารถแสดงโดย ABCDEFG ฉันต้องการสร้างวิดีโอใหม่ที่ประกอบด้วย ACDFG



1
นั่นไม่ใช่สิ่งที่ฉันขอฉันต้องการได้รับทั้งหมดในตอนเดียว แต่จะมีส่วนต่างจากวิดีโอต้นฉบับ
Matias

@Matias ถ้าคุณถามว่าจะตัดคลิปบางส่วนออกจากวิดีโอแล้วปล่อยให้ส่วนที่เหลืออยู่นั่นเป็นเรื่องหนึ่ง แต่คุณต้องการที่จะเอาคลิปสองสามคลิปนั้นมารวมกับคลิปจากวิดีโออื่น ๆ ที่ทำให้ นี่ไม่ใช่คำถามที่แตกต่างและไม่เหมือนใคร คุณต้องทำสิ่งที่คำถามอื่น ๆ ถามเพื่อให้ได้เซ็กเมนต์แยกจากนั้นรวมเข้าด้วยกัน
Synetech

1
@Synetech ขอบคุณสำหรับคำตอบของคุณ ฉันไม่ต้องการรวมเข้ากับคลิปจากวิดีโออื่น ๆ ฉันต้องการลบบางส่วนออกจากวิดีโอ ตัวอย่างเช่นหากวิดีโอของฉันสามารถแสดงโดย ABCDEFG ฉันต้องการสร้างวิดีโอใหม่ที่ประกอบด้วย ACDFG
Matias

@Synetech นั่นไม่ใช่ฉันมันเป็น Tog ที่ต้องมีความเข้าใจผิด
Matias

คำตอบ:


47

คุณยังสามารถใช้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 วินาทีเพื่อสิ้นสุดจากนั้นรวมเข้าout1กับสตรีมด้วยconcatตัวกรอง

เกี่ยวกับ 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

เยี่ยมมาก! ทำไมเราต้องตั้งค่า คุณสามารถเพิ่มคำอธิบายได้ไหม?
Rajib

ตกลงดูคำตอบที่อัปเดต
ptQa

4
+1 ควรเลือกคำตอบนี้จริงๆ เหมาะสำหรับ "หลายเซกเมนต์" และลบความจำเป็นในการดำเนินการหลายคำสั่งทีละคำสั่ง (เว้นแต่จะมีวิธีอื่นที่มีประสิทธิภาพความเร็วสูงกว่า)
Knossos

1
คุณจะรักษาข้อมูลคำบรรยายอย่างไรเมื่อคุณข้ามเฟรม
Andrew Scagnelli

1
นี่มันไร้มนุษยธรรมมาก
golopot

15

ฉันไม่สามารถหาวิธีแก้ปัญหาของ ptQa ได้ส่วนใหญ่เป็นเพราะฉันไม่สามารถเข้าใจได้ว่าข้อผิดพลาดจากตัวกรองหมายถึงอะไรหรือจะแก้ไขได้อย่างไร วิธีการแก้ปัญหาของฉันดูแปลก ๆ เล็กน้อยเพราะมันสามารถทิ้งความยุ่งเหยิงไว้ได้ แต่ถ้าคุณโยนมันเข้าไปในสคริปต์การทำความสะอาดอาจเป็นไปโดยอัตโนมัติ ฉันชอบวิธีนี้เพราะถ้ามีอะไรผิดพลาดในขั้นตอนที่ 4 คุณจะจบด้วยขั้นตอนที่ 1-3 เสร็จแล้วดังนั้นการกู้คืนจากข้อผิดพลาดจะมีประสิทธิภาพมากกว่าเล็กน้อย

กลยุทธ์พื้นฐานคือการใช้-tและ-ssการรับวิดีโอของแต่ละกลุ่มที่คุณต้องการจากนั้นเข้าร่วมทุกส่วนสำหรับรุ่นสุดท้ายของคุณ

สมมติว่าคุณมี 6 เซกเมนต์ ABCDEF ทุกๆ 5 วินาทีและคุณต้องการ 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 หรือ 00:00:30 -ssตัวเลือกกล่าวว่าวิธีที่จะมองข้ามเป็นวิดีโอแหล่งที่มาจึงเสมอเมื่อเทียบกับจุดเริ่มต้นของไฟล์

จากนั้นเมื่อคุณมีไฟล์วิดีโอมากมายฉันจะสร้างไฟล์ace-files.txtเช่นนี้:

file 'a.tvshow'
file 'c.tvshow'
file 'e.tvshow'

หมายเหตุ "ไฟล์" ที่จุดเริ่มต้นและชื่อไฟล์ที่ถูก Escape หลังจากนั้น

จากนั้นคำสั่ง:

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

นั่นเป็นการรวมไฟล์ทั้งหมดabe-files.txtเข้าด้วยกันคัดลอกตัวแปลงสัญญาณเสียงและวิดีโอของพวกเขาและสร้างไฟล์ace.tvshowซึ่งควรจะเป็นส่วน a, c และ e แล้วก็อย่าลืมที่จะลบace-files.txt, a.tvshow, และc.tvshowe.tvshow

ข้อสงวนสิทธิ์ : ฉันไม่รู้ว่าจะมีประสิทธิภาพอย่างไรเมื่อเทียบกับแนวทางอื่น ๆ ในแง่ของffmpegแต่สำหรับวัตถุประสงค์ของฉันมันทำงานได้ดีขึ้น หวังว่าจะช่วยใครซักคน


4
อันนี้เป็นที่เข้าใจได้ดีสำหรับมือใหม่อย่างฉันขอบคุณ
juanpastas

4
หมายเหตุถ้าคุณกำลังใช้ bash คุณสามารถหลีกเลี่ยงการสร้างไฟล์ ( ace-files.txtในตัวอย่างของคุณ) ด้วย:ffmpeg -f concat -i <(printf "file '%s'\n" $(pwd)/prefix_*.tvshow) -c copy output.tvshow
bufh

สิ่งนี้มีประโยชน์จริง ๆ แต่ฉันต้องลบเครื่องหมายอัญประกาศเดี่ยวจากชื่อไฟล์รอบ ๆ หรือไม่ได้ผล
tchaymore

เมื่อต่อวิดีโอเข้าด้วยกันจะมีวิดีโอสีดำสั้น ๆ และเสียงเงียบระหว่างพวกเขาประมาณ 25ms ความคิดใดที่จะกำจัดได้?
Antonio Bonifati 'Farmboy'

อืม ... ฉันแค่ตรวจสอบคณิตศาสตร์ของคุณเกี่ยวกับการชดเชยเวลาและการกระโดดข้ามเวลาและตรวจสอบให้แน่ใจว่าคุณไม่ได้เว้นช่องว่าง 25 มิลลิวินาที ...
xbakesx

5

ฉันสร้างสคริปต์เพื่อเพิ่มความเร็วในการแก้ไขรายการทีวีที่บันทึกไว้ สคริปต์จะขอเวลาเริ่มต้นและสิ้นสุดของกลุ่มที่คุณต้องการเก็บและแยกออกเป็นไฟล์ มันมีตัวเลือกให้คุณคุณสามารถ:

  • ใช้เวลาหนึ่งหรือหลายส่วน
  • คุณสามารถรวมกลุ่มเป็นไฟล์เดียวได้
  • หลังจากเข้าร่วมคุณสามารถเก็บหรือลบไฟล์ส่วนได้
  • คุณสามารถเก็บไฟล์ต้นฉบับหรือแทนที่ด้วยไฟล์ใหม่ของคุณ

วิดีโอการใช้งาน: ที่นี่

แจ้งให้เราทราบสิ่งที่คุณคิด.

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

2
สคริปต์นี้ยอดเยี่ยม! เพียงหนึ่งการดัดแปลงเนื่องจากคุณกำลังใช้พา ธ สัมบูรณ์สำหรับการต่อข้อมูล: เพิ่มsafe=0เช่นffmpeg -f concat -safe 0 -i ...ดูffmpeg.org/ffmpeg-all.html#Options-36
pkSML

2

แม้ว่าคำตอบของ ptQa นั้นดูเหมือนว่าจะทำงานได้ แต่ฉันได้พัฒนาวิธีการแก้ปัญหาอื่นที่ทำงานได้ดี

โดยพื้นฐานแล้วสิ่งที่ฉันทำคือตัดหนึ่งวิดีโอสำหรับแต่ละส่วนของวิดีโอต้นฉบับที่ฉันต้องการรวมไว้ในผลลัพธ์ของฉัน ต่อมาผม concatenate พวกเขาด้วย Concat demuxer อธิบายที่นี่

วิธีการแก้ปัญหาเดียวกันกับที่ฉันได้ลองครั้งแรกและนำเสนอปัญหาการซิงค์ สิ่งที่ฉันเพิ่มคือคำสั่ง-avoid_negative_ts 1เมื่อสร้างวิดีโอที่แตกต่างกัน ด้วยวิธีนี้ปัญหาการซิงค์จะหายไป


1

สำหรับผู้ที่มีปัญหาในการติดตามแนวทางของ 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];

กำหนดคู่ให้มากที่สุดเท่าที่คุณต้องการจากนั้นต่อเข้าด้วยกันในครั้งเดียวโดยที่ 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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.