Як я можу видалити кілька сегментів із відео за допомогою 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, якщо ви запитували, як вирізати кілька кліпів із відео та залишити решту так, як є, це було б одне, але ви хочете зняти з нього кілька кліпів і поєднати їх із кліпами з інших відео, що робить це не окреме, унікальне питання. Ви повинні виконати те, що поставили інші запитання, щоб отримати окремі сегменти, а потім поєднати їх.
Synetech

1
@Synetech дякую за відповіді. Я не хочу поєднувати їх із кліпами з інших відео. Я просто хочу видалити деякі частини з відео. Наприклад, якби моє відео могло бути представлене ABCDEFG, я б хотів створити нове, яке складалося б з ACDFG.
Матіас

@Synetech Це був не я, це Тог повинен був неправильно зрозуміти.
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фільтром.

Про параметри: нам це потрібно, тому що обробка не змінює час відображення зображення, а коли ми вирізаємо лічильник декодера 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

Чудовий! Навіщо нам потрібні септи? Чи можете ви додати пояснення?
Раджиб

Добре, дивіться оновлену відповідь.
ptQa

4
+1 Це дійсно слід вибрати як відповідь. Він більше підходить для "декількох сегментів" і знімає необхідність виконувати кілька команд одна за одною. (якщо тільки інший спосіб не є більш швидким)
Knossos

1
Як би ви ставилися до збереження даних субтитрів, коли пропускаєте кадри?
Ендрю Скагнеллі

1
Це дуже нелюдсько.
Голопот

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Каже , як довго кожен кліп, так що якщо з 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який повинен бути просто розділами a, c та e. Тоді тільки НЕ забудьте видалити ace-files.txt, a.tvshow, c.tvshowі e.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

Це було дуже корисно, але мені довелося видалити єдині лапки з імен файлів, інакше це не спрацювало.
Чаймор

Під час з'єднання відеозаписів, між ними виникає коротке чорне відео та звук приглушеного звуку, близько 25 мс. Будь-які ідеї, щоб позбутися цього?
Антоніо Боніфаті '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
Цей сценарій чудовий! Лише одна модифікація, оскільки ви використовуєте абсолютні шляхи для конкатенації: add safe=0, тобто ffmpeg -f concat -safe 0 -i ...дивіться ffmpeg.org/ffmpeg-all.html#Options-36
pkSML

2

Хоча відповідь, надана ptQa, здається, працює, я розробив інше рішення, яке підтвердило свою роботу.

По суті, те, що я роблю, - це вирізати одне відео для кожної частини оригінального відео, яку я хочу включити в свій результат. Пізніше я з'єднав їх із поясненим тут Демаксером Concat .

Рішення те саме, що я спробував спочатку і представив проблеми синхронізації. Я додав команду -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
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.