This extension allows you to:
-
Quickly cut videos both losslessly and re-encoded-ly.
-
Specify custom actions in a
config.lua
file to support your own use cases without having to modify the script itself or write your own extension. -
Bookmark timestamps to a
.book
file and load them as chapters. -
Save cut information to a
.list
file for backup and make cuts later. -
Choose meaningful channel names to organize the aforementioned actions.
All directly in the fantastic media player mpv.
Besides mpv, you must have ffmpeg
in your PATH.
git clone -b release --single-branch "https://github.com/jankozik/mpvplugin-videocut.git" ~/.config/mpv/scripts/mpv-cut
In
%AppData%\Roaming\mpv\scripts
or Users\user\scoop\persist\mpv\scripts
run:
git clone -b release --single-branch "https://github.com/jankozik/mpvplugin-videocut.git"
That's all you have to do, next time you run mpv the script will be automatically loaded.
-
Press
c
to begin a cut. -
Seek to a later time in the video.
-
Press
c
again to make the cut.
The resulting cut will be placed in the same directory as the source file.
You can press a
to cycle between three default actions:
-
Copy (lossless cut, rounds to keyframes).
-
Encode (re-encoded cut, exact).
-
List (simply add the timestamps for the cut to a
.list
file).
mpv-cut
uses an extensible list of actions that you can modify in your
config.lua
. This makes it easy to change the ffmpeg
command (or any command
for that matter) to suit your specific situation. It should be possible to
create custom actions with limited coding knowledge. More details in
config.
Press i
to append the current timestamp to a .book
file. This
automatically reloads the timestamps as chapters in mpv. You can navigate
between these chapters with the default mpv bindings, !
and @
.
The resulting cuts and bookmark files will be prefixed a channel number. This
is to help you categorize cuts and bookmarks. You can press -
to decrement
the channel and =
to increment the channel.
You can configure a name for each channel as shown below.
If you want to make all the cuts stored in a cut list, simply press 0
.
You can configure settings by creating a config.lua
file in the same
directory as main.lua
.
You can include or omit any of the following:
-- Key config
KEY_CUT = "c"
KEY_CYCLE_ACTION = "a"
KEY_BOOKMARK_ADD = "i"
KEY_CHANNEL_INC = "="
KEY_CHANNEL_DEC = "-"
KEY_MAKE_CUTS = "0"
-- The list of channel names, you can choose whatever you want.
CHANNEL_NAMES[1] = "FUNNY"
CHANNEL_NAMES[2] = "COOL"
-- The default channel
CHANNEL = 1
-- The default action
ACTION = ACTIONS.ENCODE
-- The action to use when making cuts from a cut list
MAKE_CUT = ACTIONS.COPY
-- Delete a default action
ACTIONS.LIST = nil
-- Specify custom actions
ACTIONS.ENCODE = function(d)
local args = {
"ffmpeg",
"-nostdin", "-y",
"-loglevel", "error",
"-i", d.inpath,
"-ss", d.start_time,
"-t", d.duration,
"-pix_fmt", "yuv420p",
"-crf", "16",
"-preset", "superfast",
utils.join_path(d.indir, "ENCODE_" .. d.channel .. "_" .. d.infile_noext .. "_FROM_" .. d.start_time_hms .. "_TO_" .. d.end_time_hms .. d.ext)
}
mp.command_native_async({
name = "subprocess",
args = args,
playback_only = false,
}, function() print("Done") end)
end
-- The table that gets passed to an action will have the following properties:
-- inpath, indir, infile, infile_noext, ext
-- channel
-- start_time, end_time, duration
-- start_time_hms, end_time_hms, duration_hms
This is my input.conf
file, and it is optimized for both normal playback and
quickly editing videos.
RIGHT seek 2 exact
LEFT seek -2 exact
] add speed 0.5
[ add speed -0.5
} add speed 0.25
{ add speed -0.25
SPACE cycle pause; set speed 1 # Reset speed whenever pausing.
BS script-binding osc/visibility # Make progress bar stay visible.
UP seek 0.01 keyframes # Seek by keyframes only.
DOWN seek -0.01 keyframes # Seek by keyframes only.
You may also want to change your key repeat delay and rate by tweaking
input-ar-delay
and input-ar-rate
to your liking in mpv.conf
.
There are plenty of reasons, but to give some examples:
-
Video seems to be pretty complex, at least to me. One video file may cause certain issues, and another may not, which makes writing an ffmpeg command that accounts for all scenarios difficult. If you spend a ton of time making many cuts in a long movie only to find that the colors look off because of some 10-bit h265 dolby mega surround whatever the fuck, with a cut list it's trivial to edit the ffmpeg command and re-make the cuts.
-
Maybe you forget that the foreign language video you're cutting has softsubs rather than hardsubs, and you make a bunch of encode cuts resulting in cuts that have no subtitles.
-
You delete the source video for storage reasons, but still want to have a back up of the cut timestamps in the event you need to remake the cuts.
Suppose you're watching a movie or show for your own enjoyment, but you also want to compile funny moments to post online or send to your friends. It would ruin your viewing experience to wait for a funny moment to be over in order to make a cut. Instead, you can quickly make a bookmark whenever you laugh, and once you're done watching you can go back and make actual cuts.
This refers to ffmpeg's -copy
flag which copies the input stream instead of
re-encoding it, meaning that the cut will process extremely quickly and the
resulting video will retain 100% of the original quality. The main drawback is
that the cut may have some extra video at the beginning and end, and as a
result of that there may be some slightly wonky behavior with video players and
editors.
-
As mentioned above, copying the input stream is very fast and lossless but the cuts are not exact. Sometimes you want a cut to be exact.
-
If you want to change the framerate.
-
If you want to encode hardsubs.
-
If the video's compression isn't efficient enough to upload to a messaging platform or something, you may want to compress it more.
To concatenate videos with ffmpeg, you need to create a file with content like this:
file cut_1.mp4
file cut_2.mp4
file cut_3.mp4
file cut_4.mp4
You can name the file whatever you want, here I named it concat.txt
.
Then run the command:
ffmpeg -f concat -safe 0 -i concat.txt -c copy out.mp4
That's annoying though, so you can skip manually creating the file by using bash. This command will concatenate all files in the current directory that begin with "COPY_":
ffmpeg -f concat -safe 0 -i <(printf 'file %q\n' "$PWD"/COPY_*) -c copy lol.mp4
-
You need to escape apostrophes which is why we are using
printf %q "$string"
. -
Instead of actually creating a file we just use process substitution
<(whatever)
to create a temporary file, which is why we need the$PWD
in there for the absolute path.
You can also do it in vim, among other things.
ls | vim -
:%s/'/\\'/g
:%norm Ifile
:wq concat.txt
This substitution might not cover all cases, but whatever, if you're
concatenating a file named [{}1;']["!.mp4
you can figure it out yourself.
Depending on the encoding of the video file being played, the following may be quite slow:
-
The use of
exact
ininput.conf
. -
The use of the
.
and,
keys to go frame by frame. -
The holding down of the
,
key to play the video in reverse.
Long story short, if the video uses an encoding that is difficult for mpv to
decode, exact seeking and backwards playback won't be smooth, which for normal
playback is not a problem at all, since by default mpv very quickly seeks
keyframe-wise when you press left arrow
or right arrow
.
However if we are very intensively cutting a video, it may be useful to be able to quickly seek to an exact time, and to quickly play in reverse. In this case, it is useful to first make a proxy of the original video which is very easy to decode, generate a cut list with the proxy, and then apply the cut list to the original video.
To create a proxy which will be very easy to decode, you can use this ffmpeg command:
ffmpeg -noautorotate -i input.mp4 -pix_fmt yuv420p -g 1 -sn -an -vf colormatrix=bt601:bt709,scale=w=1280:h=1280:force_original_aspect_ratio=decrease:force_divisible_by=2 -c:v libx264 -crf 16 -preset superfast -tune fastdecode proxy.mp4
The important options here are the -g 1
and the scale filter. The other
options are more or less irrelevant. The resulting video file should seek
extremely quickly and play backwards just fine.
Once you are done generating the cut list, simply open the cut_list.txt
file,
substitute the proxy file name for the original file name, and run make_cuts
on it.