2014-04-29 12:11:39 +00:00
|
|
|
-- This script uses the lavfi cropdetect filter to automatically
|
|
|
|
-- insert a crop filter with appropriate parameters for the currently
|
|
|
|
-- playing video.
|
|
|
|
--
|
|
|
|
-- It registers the key-binding "C" (shift+c), which when pressed,
|
|
|
|
-- inserts the filter vf=lavfi=cropdetect. After 1 second, it then
|
|
|
|
-- inserts the filter vf=crop=w:h:x:y, where w,h,x,y are determined
|
|
|
|
-- from the vf-metadata gathered by cropdetect. The cropdetect filter
|
|
|
|
-- is removed immediately after the crop filter is inserted as it is
|
|
|
|
-- no longer needed.
|
|
|
|
--
|
|
|
|
-- If the "C" key is pressed again, the crop filter is removed
|
|
|
|
-- restoring playback to its original state.
|
|
|
|
--
|
|
|
|
-- Since the crop parameters are determined from the 1 second of video
|
|
|
|
-- between inserting the cropdetect and crop filters, the "C" key
|
|
|
|
-- should be pressed at a position in the video where the crop region
|
|
|
|
-- is unambiguous (i.e., not a black frame, black background title
|
|
|
|
-- card, or dark scene).
|
|
|
|
--
|
|
|
|
-- The default delay between insertion of the cropdetect and
|
|
|
|
-- crop filters may be overridden by adding
|
|
|
|
--
|
2014-12-14 23:31:30 +00:00
|
|
|
-- --script-opts=autocrop.detect_seconds=<number of seconds>
|
2014-04-29 12:11:39 +00:00
|
|
|
--
|
|
|
|
-- to mpv's arguments. This may be desirable to allow cropdetect more
|
|
|
|
-- time to collect data.
|
2014-05-04 19:54:03 +00:00
|
|
|
require "mp.msg"
|
2014-04-29 12:11:39 +00:00
|
|
|
|
2014-05-06 16:59:59 +00:00
|
|
|
script_name = mp.get_script_name()
|
|
|
|
cropdetect_label = string.format("%s-cropdetect", script_name)
|
|
|
|
crop_label = string.format("%s-crop", script_name)
|
2014-04-11 11:50:06 +00:00
|
|
|
|
2014-04-29 11:07:25 +00:00
|
|
|
-- number of seconds to gather cropdetect data
|
|
|
|
detect_seconds = tonumber(mp.get_opt(string.format("%s.detect_seconds", script_name)))
|
|
|
|
if not detect_seconds then
|
|
|
|
detect_seconds = 1
|
2014-04-11 11:50:06 +00:00
|
|
|
end
|
2014-04-29 11:07:25 +00:00
|
|
|
|
|
|
|
function del_filter_if_present(label)
|
|
|
|
-- necessary because mp.command('vf del @label:filter') raises an
|
|
|
|
-- error if the filter doesn't exist
|
2014-05-04 19:59:21 +00:00
|
|
|
local vfs = mp.get_property_native("vf")
|
2014-04-29 11:07:25 +00:00
|
|
|
for i,vf in pairs(vfs) do
|
2014-10-20 06:12:34 +00:00
|
|
|
if vf["label"] == label then
|
|
|
|
table.remove(vfs, i)
|
|
|
|
mp.set_property_native("vf", vfs)
|
|
|
|
return true
|
|
|
|
end
|
2014-04-29 11:07:25 +00:00
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
function autocrop_start()
|
2014-05-04 19:54:03 +00:00
|
|
|
-- exit if cropdetection is already in progress
|
|
|
|
if timer then
|
|
|
|
mp.msg.warn("already cropdetecting!")
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
-- if there's a crop filter, remove it and exit
|
2014-04-29 11:07:25 +00:00
|
|
|
if del_filter_if_present(crop_label) then
|
|
|
|
return
|
|
|
|
end
|
2014-05-04 19:54:03 +00:00
|
|
|
|
2014-04-29 11:07:25 +00:00
|
|
|
-- insert the cropdetect filter
|
|
|
|
ret=mp.command(
|
2014-10-20 06:12:34 +00:00
|
|
|
string.format(
|
2017-08-17 19:43:25 +00:00
|
|
|
'vf add @%s:cropdetect=limit=%f:round=2:reset=0',
|
|
|
|
cropdetect_label, 24/255
|
2014-10-20 06:12:34 +00:00
|
|
|
)
|
2014-04-29 11:07:25 +00:00
|
|
|
)
|
|
|
|
-- wait to gather data
|
2014-05-04 19:54:03 +00:00
|
|
|
timer=mp.add_timeout(detect_seconds, do_crop)
|
2014-04-29 11:07:25 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function do_crop()
|
|
|
|
-- get the metadata
|
|
|
|
local cropdetect_metadata = mp.get_property_native(
|
2014-10-20 06:12:34 +00:00
|
|
|
string.format("vf-metadata/%s", cropdetect_label)
|
2014-04-29 11:07:25 +00:00
|
|
|
)
|
2017-08-17 19:43:25 +00:00
|
|
|
|
2014-04-29 11:07:25 +00:00
|
|
|
-- use it to crop if its valid
|
|
|
|
if cropdetect_metadata then
|
2014-10-20 06:12:34 +00:00
|
|
|
if cropdetect_metadata["lavfi.cropdetect.w"]
|
|
|
|
and cropdetect_metadata["lavfi.cropdetect.h"]
|
|
|
|
and cropdetect_metadata["lavfi.cropdetect.x"]
|
|
|
|
and cropdetect_metadata["lavfi.cropdetect.y"]
|
|
|
|
then
|
2017-08-17 19:43:25 +00:00
|
|
|
mp.command(string.format("vf add @%s:lavfi-crop=w=%s:h=%s:x=%s:y=%s",
|
2014-04-29 11:07:25 +00:00
|
|
|
crop_label,
|
2014-05-04 19:59:21 +00:00
|
|
|
cropdetect_metadata["lavfi.cropdetect.w"],
|
|
|
|
cropdetect_metadata["lavfi.cropdetect.h"],
|
|
|
|
cropdetect_metadata["lavfi.cropdetect.x"],
|
|
|
|
cropdetect_metadata["lavfi.cropdetect.y"]))
|
2014-04-29 11:07:25 +00:00
|
|
|
else
|
|
|
|
mp.msg.error(
|
2014-10-20 06:12:34 +00:00
|
|
|
"Got empty crop data. You might need to increase detect_seconds."
|
|
|
|
)
|
|
|
|
end
|
2014-04-29 11:07:25 +00:00
|
|
|
else
|
|
|
|
mp.msg.error(
|
|
|
|
"No crop data. Was the cropdetect filter successfully inserted?"
|
|
|
|
)
|
|
|
|
mp.msg.error(
|
|
|
|
"Does your version of ffmpeg/libav support AVFrame metadata?"
|
|
|
|
)
|
|
|
|
end
|
|
|
|
-- remove the cropdetect filter
|
|
|
|
del_filter_if_present(cropdetect_label)
|
2014-05-04 19:54:03 +00:00
|
|
|
timer=nil
|
2014-04-29 11:07:25 +00:00
|
|
|
end
|
|
|
|
|
2014-05-06 16:59:59 +00:00
|
|
|
mp.add_key_binding("C", "auto_crop", autocrop_start)
|