mirror of
https://github.com/mpv-player/mpv
synced 2025-01-28 10:33:20 +00:00
d147a06e60
msg.info for when you're potentially adding hundreds of files is way too spammy and not really useful (autoload loads files, what?). Bump it up to msg.verbose. Also bump up the previously existing msg.verbose logs up to msg.debug to keep the difference in log levels. Fixes #12551.
329 lines
9.9 KiB
Lua
329 lines
9.9 KiB
Lua
-- This script automatically loads playlist entries before and after the
|
|
-- the currently played file. It does so by scanning the directory a file is
|
|
-- located in when starting playback. It sorts the directory entries
|
|
-- alphabetically, and adds entries before and after the current file to
|
|
-- the internal playlist. (It stops if it would add an already existing
|
|
-- playlist entry at the same position - this makes it "stable".)
|
|
-- Add at most 5000 * 2 files when starting a file (before + after).
|
|
|
|
--[[
|
|
To configure this script use file autoload.conf in directory script-opts (the "script-opts"
|
|
directory must be in the mpv configuration directory, typically ~/.config/mpv/).
|
|
|
|
Example configuration would be:
|
|
|
|
disabled=no
|
|
images=no
|
|
videos=yes
|
|
audio=yes
|
|
additional_image_exts=list,of,ext
|
|
additional_video_exts=list,of,ext
|
|
additional_audio_exts=list,of,ext
|
|
ignore_hidden=yes
|
|
same_type=yes
|
|
directory_mode=recursive
|
|
|
|
--]]
|
|
|
|
MAXENTRIES = 5000
|
|
MAXDIRSTACK = 20
|
|
|
|
local msg = require 'mp.msg'
|
|
local options = require 'mp.options'
|
|
local utils = require 'mp.utils'
|
|
|
|
o = {
|
|
disabled = false,
|
|
images = true,
|
|
videos = true,
|
|
audio = true,
|
|
additional_image_exts = "",
|
|
additional_video_exts = "",
|
|
additional_audio_exts = "",
|
|
ignore_hidden = true,
|
|
same_type = false,
|
|
directory_mode = "auto"
|
|
}
|
|
options.read_options(o, nil, function(list)
|
|
split_option_exts(list.additional_video_exts, list.additional_audio_exts, list.additional_image_exts)
|
|
if list.videos or list.additional_video_exts or
|
|
list.audio or list.additional_audio_exts or
|
|
list.images or list.additional_image_exts then
|
|
create_extensions()
|
|
end
|
|
if list.directory_mode then
|
|
validate_directory_mode()
|
|
end
|
|
end)
|
|
|
|
function Set (t)
|
|
local set = {}
|
|
for _, v in pairs(t) do set[v] = true end
|
|
return set
|
|
end
|
|
|
|
function SetUnion (a,b)
|
|
for k in pairs(b) do a[k] = true end
|
|
return a
|
|
end
|
|
|
|
function Split (s)
|
|
local set = {}
|
|
for v in string.gmatch(s, '([^,]+)') do set[v] = true end
|
|
return set
|
|
end
|
|
|
|
EXTENSIONS_VIDEO = Set {
|
|
'3g2', '3gp', 'avi', 'flv', 'm2ts', 'm4v', 'mj2', 'mkv', 'mov',
|
|
'mp4', 'mpeg', 'mpg', 'ogv', 'rmvb', 'webm', 'wmv', 'y4m'
|
|
}
|
|
|
|
EXTENSIONS_AUDIO = Set {
|
|
'aiff', 'ape', 'au', 'flac', 'm4a', 'mka', 'mp3', 'oga', 'ogg',
|
|
'ogm', 'opus', 'wav', 'wma'
|
|
}
|
|
|
|
EXTENSIONS_IMAGES = Set {
|
|
'avif', 'bmp', 'gif', 'j2k', 'jp2', 'jpeg', 'jpg', 'jxl', 'png',
|
|
'svg', 'tga', 'tif', 'tiff', 'webp'
|
|
}
|
|
|
|
function split_option_exts(video, audio, image)
|
|
if video then o.additional_video_exts = Split(o.additional_video_exts) end
|
|
if audio then o.additional_audio_exts = Split(o.additional_audio_exts) end
|
|
if image then o.additional_image_exts = Split(o.additional_image_exts) end
|
|
end
|
|
split_option_exts(true, true, true)
|
|
|
|
function create_extensions()
|
|
EXTENSIONS = {}
|
|
if o.videos then SetUnion(SetUnion(EXTENSIONS, EXTENSIONS_VIDEO), o.additional_video_exts) end
|
|
if o.audio then SetUnion(SetUnion(EXTENSIONS, EXTENSIONS_AUDIO), o.additional_audio_exts) end
|
|
if o.images then SetUnion(SetUnion(EXTENSIONS, EXTENSIONS_IMAGES), o.additional_image_exts) end
|
|
end
|
|
create_extensions()
|
|
|
|
function validate_directory_mode()
|
|
if o.directory_mode ~= "recursive" and o.directory_mode ~= "lazy" and o.directory_mode ~= "ignore" then
|
|
o.directory_mode = nil
|
|
end
|
|
end
|
|
validate_directory_mode()
|
|
|
|
function add_files(files)
|
|
local oldcount = mp.get_property_number("playlist-count", 1)
|
|
for i = 1, #files do
|
|
mp.commandv("loadfile", files[i][1], "append")
|
|
mp.commandv("playlist-move", oldcount + i - 1, files[i][2])
|
|
end
|
|
end
|
|
|
|
function get_extension(path)
|
|
match = string.match(path, "%.([^%.]+)$" )
|
|
if match == nil then
|
|
return "nomatch"
|
|
else
|
|
return match
|
|
end
|
|
end
|
|
|
|
table.filter = function(t, iter)
|
|
for i = #t, 1, -1 do
|
|
if not iter(t[i]) then
|
|
table.remove(t, i)
|
|
end
|
|
end
|
|
end
|
|
|
|
table.append = function(t1, t2)
|
|
local t1_size = #t1
|
|
for i = 1, #t2 do
|
|
t1[t1_size + i] = t2[i]
|
|
end
|
|
end
|
|
|
|
-- alphanum sorting for humans in Lua
|
|
-- http://notebook.kulchenko.com/algorithms/alphanumeric-natural-sorting-for-humans-in-lua
|
|
|
|
function alphanumsort(filenames)
|
|
local function padnum(n, d)
|
|
return #d > 0 and ("%03d%s%.12f"):format(#n, n, tonumber(d) / (10 ^ #d))
|
|
or ("%03d%s"):format(#n, n)
|
|
end
|
|
|
|
local tuples = {}
|
|
for i, f in ipairs(filenames) do
|
|
tuples[i] = {f:lower():gsub("0*(%d+)%.?(%d*)", padnum), f}
|
|
end
|
|
table.sort(tuples, function(a, b)
|
|
return a[1] == b[1] and #b[2] < #a[2] or a[1] < b[1]
|
|
end)
|
|
for i, tuple in ipairs(tuples) do filenames[i] = tuple[2] end
|
|
return filenames
|
|
end
|
|
|
|
local autoloaded = nil
|
|
local added_entries = {}
|
|
local autoloaded_dir = nil
|
|
|
|
function scan_dir(path, current_file, dir_mode, separator, dir_depth, total_files, extensions)
|
|
if dir_depth == MAXDIRSTACK then
|
|
return
|
|
end
|
|
msg.trace("scanning: " .. path)
|
|
local files = utils.readdir(path, "files") or {}
|
|
local dirs = dir_mode ~= "ignore" and utils.readdir(path, "dirs") or {}
|
|
local prefix = path == "." and "" or path
|
|
table.filter(files, function (v)
|
|
-- The current file could be a hidden file, ignoring it doesn't load other
|
|
-- files from the current directory.
|
|
if (o.ignore_hidden and not (prefix .. v == current_file) and string.match(v, "^%.")) then
|
|
return false
|
|
end
|
|
local ext = get_extension(v)
|
|
if ext == nil then
|
|
return false
|
|
end
|
|
return extensions[string.lower(ext)]
|
|
end)
|
|
table.filter(dirs, function(d)
|
|
return not ((o.ignore_hidden and string.match(d, "^%.")))
|
|
end)
|
|
alphanumsort(files)
|
|
alphanumsort(dirs)
|
|
|
|
for i, file in ipairs(files) do
|
|
files[i] = prefix .. file
|
|
end
|
|
|
|
table.append(total_files, files)
|
|
if dir_mode == "recursive" then
|
|
for _, dir in ipairs(dirs) do
|
|
scan_dir(prefix .. dir .. separator, current_file, dir_mode,
|
|
separator, dir_depth + 1, total_files, extensions)
|
|
end
|
|
else
|
|
for i, dir in ipairs(dirs) do
|
|
dirs[i] = prefix .. dir
|
|
end
|
|
table.append(total_files, dirs)
|
|
end
|
|
end
|
|
|
|
function find_and_add_entries()
|
|
local path = mp.get_property("path", "")
|
|
local dir, filename = utils.split_path(path)
|
|
msg.trace(("dir: %s, filename: %s"):format(dir, filename))
|
|
if o.disabled then
|
|
msg.debug("stopping: autoload disabled")
|
|
return
|
|
elseif #dir == 0 then
|
|
msg.debug("stopping: not a local path")
|
|
return
|
|
end
|
|
|
|
local pl_count = mp.get_property_number("playlist-count", 1)
|
|
this_ext = get_extension(filename)
|
|
-- check if this is a manually made playlist
|
|
if (pl_count > 1 and autoloaded == nil) or
|
|
(pl_count == 1 and EXTENSIONS[string.lower(this_ext)] == nil) then
|
|
msg.debug("stopping: manually made playlist")
|
|
return
|
|
else
|
|
if pl_count == 1 then
|
|
autoloaded = true
|
|
autoloaded_dir = dir
|
|
added_entries = {}
|
|
end
|
|
end
|
|
|
|
local extensions = {}
|
|
if o.same_type then
|
|
if EXTENSIONS_VIDEO[string.lower(this_ext)] ~= nil then
|
|
extensions = EXTENSIONS_VIDEO
|
|
elseif EXTENSIONS_AUDIO[string.lower(this_ext)] ~= nil then
|
|
extensions = EXTENSIONS_AUDIO
|
|
else
|
|
extensions = EXTENSIONS_IMAGES
|
|
end
|
|
else
|
|
extensions = EXTENSIONS
|
|
end
|
|
|
|
local pl = mp.get_property_native("playlist", {})
|
|
local pl_current = mp.get_property_number("playlist-pos-1", 1)
|
|
msg.trace(("playlist-pos-1: %s, playlist: %s"):format(pl_current,
|
|
utils.to_string(pl)))
|
|
|
|
local files = {}
|
|
do
|
|
local dir_mode = o.directory_mode or mp.get_property("directory-mode", "lazy")
|
|
local separator = mp.get_property_native("platform") == "windows" and "\\" or "/"
|
|
scan_dir(autoloaded_dir, path, dir_mode, separator, 0, files, extensions)
|
|
end
|
|
|
|
if next(files) == nil then
|
|
msg.debug("no other files or directories in directory")
|
|
return
|
|
end
|
|
|
|
-- Find the current pl entry (dir+"/"+filename) in the sorted dir list
|
|
local current
|
|
for i = 1, #files do
|
|
if files[i] == path then
|
|
current = i
|
|
break
|
|
end
|
|
end
|
|
if current == nil then
|
|
return
|
|
end
|
|
msg.trace("current file position in files: "..current)
|
|
|
|
-- treat already existing playlist entries, independent of how they got added
|
|
-- as if they got added by autoload
|
|
for _, entry in ipairs(pl) do
|
|
added_entries[entry.filename] = true
|
|
end
|
|
|
|
local append = {[-1] = {}, [1] = {}}
|
|
for direction = -1, 1, 2 do -- 2 iterations, with direction = -1 and +1
|
|
for i = 1, MAXENTRIES do
|
|
local pos = current + i * direction
|
|
local file = files[pos]
|
|
if file == nil or file[1] == "." then
|
|
break
|
|
end
|
|
|
|
-- skip files that are/were already in the playlist
|
|
if not added_entries[file] then
|
|
if direction == -1 then
|
|
msg.verbose("Prepending " .. file)
|
|
table.insert(append[-1], 1, {file, pl_current + i * direction + 1})
|
|
else
|
|
msg.verbose("Adding " .. file)
|
|
if pl_count > 1 then
|
|
table.insert(append[1], {file, pl_current + i * direction - 1})
|
|
else
|
|
mp.commandv("loadfile", file, "append")
|
|
end
|
|
end
|
|
end
|
|
added_entries[file] = true
|
|
end
|
|
if pl_count == 1 and direction == -1 and #append[-1] > 0 then
|
|
for i = 1, #append[-1] do
|
|
mp.commandv("loadfile", append[-1][i][1], "append")
|
|
end
|
|
mp.commandv("playlist-move", 0, current)
|
|
end
|
|
end
|
|
|
|
if pl_count > 1 then
|
|
add_files(append[1])
|
|
add_files(append[-1])
|
|
end
|
|
end
|
|
|
|
mp.register_event("start-file", find_and_add_entries)
|