videoclip/videoclip.lua

366 lines
9.9 KiB
Lua
Raw Normal View History

2020-08-21 04:19:06 +00:00
local msg = require('mp.msg')
local mpopt = require('mp.options')
-- Options can be changed here or in a separate config file.
-- Config path: ~/.config/mpv/script-opts/videoclip.conf
local config = {
-- absolute paths
2020-08-21 04:19:06 +00:00
-- relative paths (e.g. ~ for home dir) do NOT work.
video_folder_path = string.format('%s/Videos/', os.getenv("HOME")),
audio_folder_path = string.format('%s/Music/', os.getenv("HOME")),
2020-08-21 04:19:06 +00:00
2020-08-21 04:47:46 +00:00
font_size = 24,
2020-08-21 04:19:06 +00:00
audio_bitrate = '32k',
-- The range of the CRF scale is 051, where 0 is lossless,
-- 23 is the default, and 51 is worst quality possible.
video_quality = 23,
-- Use the slowest preset that you have patience for.
-- https://trac.ffmpeg.org/wiki/Encode/H.264
preset = 'faster',
video_width = -2,
video_height = 480,
}
mpopt.read_options(config, 'videoclip')
local overlay = mp.create_osd_overlay('ass-events')
local menu
local ffmpeg
local OSD
local Timings
2020-08-21 04:19:06 +00:00
------------------------------------------------------------
-- utility functions
2020-08-27 07:13:48 +00:00
function string:endswith(suffix)
return self:match(string.format('%s$', suffix))
end
local function add_extension(filename, extension)
2020-08-21 04:19:06 +00:00
return filename .. extension
end
local function remove_extension(filename)
2020-08-21 04:19:06 +00:00
return filename:gsub('%.%w+$','')
end
local function remove_text_in_brackets(str)
2020-08-21 04:19:06 +00:00
return str:gsub('%b[]','')
end
local function remove_special_characters(str)
2020-08-21 04:19:06 +00:00
return str:gsub('[%c%p%s]','')
end
local function human_readable_time(seconds)
if type(seconds) ~= 'number' or seconds < 0 then
2020-08-21 04:19:06 +00:00
return 'empty'
end
local parts = {}
2020-08-21 04:19:06 +00:00
parts.h = math.floor(seconds / 3600)
parts.m = math.floor(seconds / 60) % 60
parts.s = math.floor(seconds % 60)
parts.ms = math.floor((seconds * 1000) % 1000)
local ret = string.format("%02dm%02ds%03dms", parts.m, parts.s, parts.ms)
if parts.h > 0 then
ret = string.format('%dh%s', parts.h, ret)
2020-08-21 04:19:06 +00:00
end
return ret
2020-08-21 04:19:06 +00:00
end
local function construct_filename()
2020-08-21 04:19:06 +00:00
local filename = mp.get_property("filename") -- filename without path
filename = remove_extension(filename)
filename = remove_text_in_brackets(filename)
filename = remove_special_characters(filename)
filename = string.format(
'%s_(%s-%s)',
filename,
human_readable_time(menu.timings['start']),
human_readable_time(menu.timings['end'])
2020-08-21 04:19:06 +00:00
)
return filename
end
local function get_audio_track_number()
2020-08-21 04:19:06 +00:00
local audio_track_number = 0
local tracks_count = mp.get_property_number("track-list/count")
for i = 1, tracks_count do
local track_type = mp.get_property(string.format("track-list/%d/type", i))
local track_index = mp.get_property_number(string.format("track-list/%d/ff-index", i))
local track_selected = mp.get_property(string.format("track-list/%d/selected", i))
if track_type == "audio" and track_selected == "yes" then
2020-09-03 00:12:22 +00:00
audio_track_number = track_index
2020-08-21 04:19:06 +00:00
break
end
end
return audio_track_number
end
------------------------------------------------------------
-- ffmpeg helper
ffmpeg = {prefix = {"ffmpeg", "-hide_banner", "-nostdin", "-y", "-loglevel", "quiet"}}
ffmpeg.execute = function(args)
if next(args) == nil then
return
end
for i, value in ipairs(ffmpeg.prefix) do
table.insert(args, i, value)
end
2020-08-27 03:29:23 +00:00
local ret = mp.command_native{
name = "subprocess",
playback_only = false,
capture_stdout = true,
args = args
}
return ret
2020-08-21 04:19:06 +00:00
end
2020-08-22 00:44:37 +00:00
ffmpeg.create_videoclip = function(clip_filename, video_path, track_number)
local clip_path = add_extension(config.video_folder_path .. clip_filename, '.mp4')
2020-08-27 03:29:23 +00:00
return ffmpeg.execute{
'-ss', tostring(menu.timings['start']),
'-to', tostring(menu.timings['end']),
'-i', video_path,
'-map_metadata', '-1',
2020-09-03 00:12:22 +00:00
'-map', string.format("0:%d", track_number),
2020-08-27 03:29:23 +00:00
'-map', '0:v:0',
'-codec:a', 'libopus',
'-codec:v', 'libx264',
'-preset', config.preset,
'-vbr', 'on',
'-compression_level', '10',
'-application', 'voip',
'-ac', '2',
'-b:a', tostring(config.audio_bitrate),
'-crf', tostring(config.video_quality),
'-vf', string.format("scale=%d:%d", config.video_width, config.video_height),
clip_path
2020-08-21 04:19:06 +00:00
}
end
2020-08-22 00:44:37 +00:00
ffmpeg.create_audioclip = function(clip_filename, video_path, track_number)
local clip_path = add_extension(config.audio_folder_path .. clip_filename, '.ogg')
2020-08-22 00:44:37 +00:00
2020-08-27 03:29:23 +00:00
return ffmpeg.execute{
'-vn',
'-ss', tostring(menu.timings['start']),
'-to', tostring(menu.timings['end']),
'-i', video_path,
'-map_metadata', '-1',
2020-09-03 00:12:22 +00:00
'-map', string.format("0:%d", track_number),
2020-08-27 03:29:23 +00:00
'-ac', '2',
'-codec:a', 'libopus',
'-vbr', 'on',
'-compression_level', '10',
'-application', 'voip',
'-b:a', tostring(config.audio_bitrate),
clip_path
2020-08-22 00:44:37 +00:00
}
end
ffmpeg.create_clip = function(clip_type)
if clip_type == nil then return end
if not menu.timings:validate() then
2020-08-22 00:44:37 +00:00
mp.osd_message("Wrong timings. Aborting.", 2)
return
end
local clip_filename = construct_filename()
local video_path = mp.get_property("path")
local track_number = get_audio_track_number()
2020-08-27 03:29:23 +00:00
mp.osd_message("Please wait...", 9999)
local ret
local location
2020-08-27 03:29:23 +00:00
2020-08-22 00:44:37 +00:00
if clip_type == 'video' then
location = config.video_folder_path
2020-08-27 03:29:23 +00:00
ret = ffmpeg.create_videoclip(clip_filename, video_path, track_number)
2020-08-22 00:44:37 +00:00
elseif clip_type == 'audio' then
location = config.audio_folder_path
2020-08-27 03:29:23 +00:00
ret = ffmpeg.create_audioclip(clip_filename, video_path, track_number)
end
if ret.status == 0 then
mp.osd_message(string.format("Clip saved to %s.", location), 2)
2020-08-27 03:29:23 +00:00
else
mp.osd_message(string.format("Error: couldn't create the clip.\nDoes %s exist?", location), 5)
2020-08-22 00:44:37 +00:00
end
end
2020-08-21 04:19:06 +00:00
------------------------------------------------------------
-- main menu
menu = {}
menu.keybinds = {
{ key = 's', fn = function() menu.set_time('start') end },
{ key = 'e', fn = function() menu.set_time('end') end },
{ key = 'S', fn = function() menu.set_time_sub('start') end },
{ key = 'E', fn = function() menu.set_time_sub('end') end },
2020-08-27 03:29:23 +00:00
{ key = 'c', fn = function() menu.close(); ffmpeg.create_clip('video'); menu.timings:reset() end },
{ key = 'a', fn = function() menu.close(); ffmpeg.create_clip('audio'); menu.timings:reset() end },
2020-08-21 04:19:06 +00:00
{ key = 'o', fn = function() mp.commandv('run', 'xdg-open', 'https://streamable.com/') end },
{ key = 'ESC', fn = function() menu.close() end },
}
menu.set_time = function(property)
menu.timings[property] = mp.get_property_number('time-pos')
2020-08-21 04:19:06 +00:00
menu.update()
end
menu.set_time_sub = function(property)
local sub_delay = mp.get_property_native("sub-delay")
local time_pos = mp.get_property_number(string.format("sub-%s", property))
if time_pos == nil then
menu.update("Warning: No subtitles visible.")
return
end
menu.timings[property] = time_pos + sub_delay
2020-08-21 04:19:06 +00:00
menu.update()
end
menu.update = function(message)
2020-09-06 15:51:36 +00:00
local osd = OSD:new():size(config.font_size):align(4)
osd:bold('Clip creator'):newline():newline()
2020-08-21 04:19:06 +00:00
osd:bold('Start time: '):append(human_readable_time(menu.timings['start'])):newline()
osd:bold('End time: '):append(human_readable_time(menu.timings['end'])):newline()
osd:newline()
2020-08-21 04:19:06 +00:00
osd:bold('Bindings:'):newline()
osd:tab():bold('s: '):append('Set start time'):newline()
osd:tab():bold('e: '):append('Set end time'):newline()
osd:newline()
osd:tab():bold('S: '):append('Set start time based on subtitles'):newline()
osd:tab():bold('E: '):append('Set end time based on subtitles'):newline()
osd:newline()
osd:tab():bold('c: '):append('Create video clip'):newline()
2020-08-22 00:44:37 +00:00
osd:tab():bold('a: '):append('Create audio clip'):newline()
2020-08-21 04:19:06 +00:00
osd:tab():bold('o: '):append('Open `streamable.com`'):newline()
osd:tab():bold('ESC: '):append('Close'):newline()
if message ~= nil then
osd:newline():append(message):newline()
end
osd:draw()
end
menu.close = function()
for _, val in pairs(menu.keybinds) do
mp.remove_key_binding(val.key)
end
overlay:remove()
end
menu.open = function()
menu.timings = Timings:new()
2020-08-21 04:19:06 +00:00
for _, val in pairs(menu.keybinds) do
mp.add_key_binding(val.key, val.key, val.fn)
end
menu.update()
end
------------------------------------------------------------
-- Helper class for styling OSD messages
OSD = {}
OSD.__index = OSD
function OSD:new()
return setmetatable({text=''}, self)
end
function OSD:append(s)
self.text = self.text .. s
return self
end
function OSD:bold(s)
return self:append('{\\b1}' .. s .. '{\\b0}')
end
function OSD:newline()
return self:append('\\N')
end
function OSD:tab()
return self:append('\\h\\h\\h\\h')
end
function OSD:size(size)
return self:append('{\\fs' .. size .. '}')
end
2020-09-06 15:51:36 +00:00
function OSD:align(number)
return self:append('{\\an' .. number .. '}')
end
2020-08-21 04:19:06 +00:00
function OSD:draw()
overlay.data = self.text
overlay:update()
end
------------------------------------------------------------
-- Timings class
Timings = {
['start'] = -1,
['end'] = -1,
}
function Timings:new(o)
o = o or {}
setmetatable(o, self)
self.__index = self
return o
end
function Timings:reset()
self['start'] = -1
self['end'] = -1
end
function Timings:validate()
return self['start'] >= 0 and self['start'] < self['end']
2020-08-21 04:19:06 +00:00
end
2020-08-27 07:13:48 +00:00
------------------------------------------------------------
-- Validate config
if not config.video_folder_path:endswith('/') then
config.video_folder_path = config.video_folder_path .. '/'
end
if not config.audio_folder_path:endswith('/') then
config.audio_folder_path = config.audio_folder_path .. '/'
end
2020-08-21 04:19:06 +00:00
------------------------------------------------------------
-- Finally, set an 'entry point' in mpv
2020-08-31 19:45:35 +00:00
mp.add_key_binding('c', 'videoclip-menu-open', menu.open)