611 lines
17 KiB
Lua
611 lines
17 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/).
|
|
|
|
Option `ignore_patterns` is a comma-separated list of patterns (see lua.org/pil/20.2.html).
|
|
Additionally to the standard lua patterns, you can also escape commas with `%`,
|
|
for example, the option `bak%,x%,,another` will be resolved as patterns `bak,x,` and `another`.
|
|
But it does not mean you need to escape all lua patterns twice,
|
|
so the option `bak%%,%。mp4,` will be resolved as two patterns `bak%%` and `%.mp4`.
|
|
|
|
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
|
|
same_series=yes
|
|
directory_mode=recursive
|
|
ignore_patterns=^~,^bak-,%.bak$
|
|
|
|
--]]
|
|
|
|
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,
|
|
same_series = false,
|
|
directory_mode = "ignore",
|
|
ignore_patterns = ""
|
|
}
|
|
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
|
|
|
|
-- Returns first and last positions in string or past-to-end indices
|
|
function FindOrPastTheEnd (string, pattern, start_at)
|
|
local pos1, pos2 = string.find(string, pattern, start_at)
|
|
return pos1 or #string + 1,
|
|
pos2 or #string + 1
|
|
end
|
|
|
|
function Split (list)
|
|
local set = {}
|
|
|
|
local item_pos = 1
|
|
local item = ""
|
|
|
|
while item_pos <= #list do
|
|
local pos1, pos2 = FindOrPastTheEnd(list, "%%*,", item_pos)
|
|
|
|
local pattern_length = pos2 - pos1
|
|
local is_comma_escaped = pattern_length % 2
|
|
|
|
local pos_before_escape = pos1 - 1
|
|
local item_escape_count = pattern_length - is_comma_escaped
|
|
|
|
item = item .. string.sub(list, item_pos, pos_before_escape + item_escape_count)
|
|
|
|
if is_comma_escaped == 1 then
|
|
item = item .. ","
|
|
else
|
|
set[item] = true
|
|
item = ""
|
|
end
|
|
|
|
item_pos = pos2 + 1
|
|
end
|
|
|
|
set[item] = true
|
|
|
|
-- exclude empty items
|
|
set[""] = nil
|
|
|
|
return set
|
|
end
|
|
|
|
EXTENSIONS_VIDEO_DEFAULT = Set {
|
|
'3g2', '3gp', 'avi', 'flv', 'm2ts', 'm4v', 'mj2', 'mkv', 'mov',
|
|
'mp4', 'mpeg', 'mpg', 'ogv', 'rmvb', 'webm', 'wmv', 'y4m'
|
|
}
|
|
|
|
EXTENSIONS_AUDIO_DEFAULT = Set {
|
|
'aiff', 'ape', 'au', 'flac', 'm4a', 'mka', 'mp3', 'oga', 'ogg',
|
|
'ogm', 'opus', 'wav', 'wma'
|
|
}
|
|
|
|
EXTENSIONS_IMAGES_DEFAULT = 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 split_patterns()
|
|
o.ignore_patterns = Split(o.ignore_patterns)
|
|
end
|
|
split_patterns()
|
|
|
|
function create_extensions()
|
|
EXTENSIONS = {}
|
|
EXTENSIONS_VIDEO = {}
|
|
EXTENSIONS_AUDIO = {}
|
|
EXTENSIONS_IMAGES = {}
|
|
if o.videos then
|
|
SetUnion(SetUnion(EXTENSIONS_VIDEO, EXTENSIONS_VIDEO_DEFAULT), o.additional_video_exts)
|
|
SetUnion(EXTENSIONS, EXTENSIONS_VIDEO)
|
|
end
|
|
if o.audio then
|
|
SetUnion(SetUnion(EXTENSIONS_AUDIO, EXTENSIONS_AUDIO_DEFAULT), o.additional_audio_exts)
|
|
SetUnion(EXTENSIONS, EXTENSIONS_AUDIO)
|
|
end
|
|
if o.images then
|
|
SetUnion(SetUnion(EXTENSIONS_IMAGES, EXTENSIONS_IMAGES_DEFAULT), o.additional_image_exts)
|
|
SetUnion(EXTENSIONS, EXTENSIONS_IMAGES)
|
|
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
|
|
|
|
function get_filename_without_ext(filename)
|
|
local idx = filename:match(".+()%.%w+$")
|
|
if idx then
|
|
filename = filename:sub(1, idx - 1)
|
|
end
|
|
return filename
|
|
end
|
|
|
|
function utf8_char_bytes(str, i)
|
|
local char_byte = str:byte(i)
|
|
if char_byte < 0xC0 then
|
|
return 1
|
|
elseif char_byte < 0xE0 then
|
|
return 2
|
|
elseif char_byte < 0xF0 then
|
|
return 3
|
|
elseif char_byte < 0xF8 then
|
|
return 4
|
|
else
|
|
return 1
|
|
end
|
|
end
|
|
|
|
function utf8_iter(str)
|
|
local byte_start = 1
|
|
return function()
|
|
local start = byte_start
|
|
if #str < start then return nil end
|
|
local byte_count = utf8_char_bytes(str, start)
|
|
byte_start = start + byte_count
|
|
return start, str:sub(start, start + byte_count - 1)
|
|
end
|
|
end
|
|
|
|
function utf8_to_table(str)
|
|
local t = {}
|
|
for _, ch in utf8_iter(str) do
|
|
t[#t + 1] = ch
|
|
end
|
|
return t
|
|
end
|
|
|
|
function jaro(s1, s2)
|
|
local match_window = math.floor(math.max(#s1, #s2) / 2.0) - 1
|
|
local matches1 = {}
|
|
local matches2 = {}
|
|
|
|
local m = 0;
|
|
local t = 0;
|
|
|
|
for i = 0, #s1, 1 do
|
|
local start = math.max(0, i - match_window)
|
|
local final = math.min(i + match_window + 1, #s2)
|
|
|
|
for k = start, final, 1 do
|
|
if not (matches2[k] or s1[i] ~= s2[k]) then
|
|
matches1[i] = true
|
|
matches2[k] = true
|
|
m = m + 1
|
|
break
|
|
end
|
|
end
|
|
end
|
|
|
|
if m == 0 then
|
|
return 0.0
|
|
end
|
|
|
|
local k = 0
|
|
for i = 0, #s1, 1 do
|
|
if matches1[i] then
|
|
while not matches2[k] do
|
|
k = k + 1
|
|
end
|
|
|
|
if s1[i] ~= s2[k] then
|
|
t = t + 1
|
|
end
|
|
|
|
k = k + 1
|
|
end
|
|
end
|
|
|
|
t = t / 2.0
|
|
|
|
return (m / #s1 + m / #s2 + (m - t) / m) / 3.0
|
|
end
|
|
|
|
function jaro_winkler_distance(s1, s2)
|
|
if #s1 + #s2 == 0 then
|
|
return 0.0
|
|
end
|
|
|
|
if s1 == s2 then
|
|
return 1.0
|
|
end
|
|
|
|
s1 = utf8_to_table(s1)
|
|
s2 = utf8_to_table(s2)
|
|
|
|
local d = jaro(s1, s2)
|
|
local p = 0.1
|
|
local l = 0;
|
|
while (s1[l] == s2[l] and l < 4) do
|
|
l = l + 1
|
|
end
|
|
|
|
return d + l * p * (1 - d)
|
|
end
|
|
|
|
function is_same_series(f1, f2)
|
|
local f1, f2 = get_filename_without_ext(f1), get_filename_without_ext(f2)
|
|
if f1 ~= f2 then
|
|
-- by episode
|
|
local sub1 = f1:gsub("^[%[%(]+.-[%]%)]+[%s%[]*", ""):match("(.-%D+)0*%d+")
|
|
local sub2 = f2:gsub("^[%[%(]+.-[%]%)]+[%s%[]*", ""):match("(.-%D+)0*%d+")
|
|
if sub1 and sub2 and sub1 == sub2 then
|
|
return true
|
|
end
|
|
|
|
-- by similarity
|
|
local threshold = 0.8
|
|
local similarity = jaro_winkler_distance(f1, f2)
|
|
if similarity > threshold then
|
|
return true
|
|
end
|
|
end
|
|
|
|
return false
|
|
end
|
|
|
|
function is_ignored(file)
|
|
for pattern, _ in pairs(o.ignore_patterns) do
|
|
if string.match(file, pattern) then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
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
|
|
|
|
----- winapi start -----
|
|
-- in windows system, we can use the sorting function provided by the win32 API
|
|
-- see https://learn.microsoft.com/en-us/windows/win32/api/shlwapi/nf-shlwapi-strcmplogicalw
|
|
-- this function was taken from https://github.com/mpvnet-player/mpv.net/issues/575#issuecomment-1817413401
|
|
local winapi = {}
|
|
local is_windows = mp.get_property_native("platform") == "windows"
|
|
|
|
if is_windows then
|
|
-- is_ffi_loaded is false usually means the mpv builds without luajit
|
|
local is_ffi_loaded, ffi = pcall(require, "ffi")
|
|
|
|
if is_ffi_loaded then
|
|
winapi = {
|
|
ffi = ffi,
|
|
C = ffi.C,
|
|
CP_UTF8 = 65001,
|
|
shlwapi = ffi.load("shlwapi"),
|
|
}
|
|
|
|
-- ffi code from https://github.com/po5/thumbfast, Mozilla Public License Version 2.0
|
|
ffi.cdef[[
|
|
int __stdcall MultiByteToWideChar(unsigned int CodePage, unsigned long dwFlags, const char *lpMultiByteStr,
|
|
int cbMultiByte, wchar_t *lpWideCharStr, int cchWideChar);
|
|
int __stdcall StrCmpLogicalW(wchar_t *psz1, wchar_t *psz2);
|
|
]]
|
|
|
|
winapi.utf8_to_wide = function(utf8_str)
|
|
if utf8_str then
|
|
local utf16_len = winapi.C.MultiByteToWideChar(winapi.CP_UTF8, 0, utf8_str, -1, nil, 0)
|
|
|
|
if utf16_len > 0 then
|
|
local utf16_str = winapi.ffi.new("wchar_t[?]", utf16_len)
|
|
|
|
if winapi.C.MultiByteToWideChar(winapi.CP_UTF8, 0, utf8_str, -1, utf16_str, utf16_len) > 0 then
|
|
return utf16_str
|
|
end
|
|
end
|
|
end
|
|
|
|
return ""
|
|
end
|
|
end
|
|
end
|
|
----- winapi end -----
|
|
|
|
function alphanumsort_windows(filenames)
|
|
table.sort(filenames, function(a, b)
|
|
local a_wide = winapi.utf8_to_wide(a)
|
|
local b_wide = winapi.utf8_to_wide(b)
|
|
return winapi.shlwapi.StrCmpLogicalW(a_wide, b_wide) == -1
|
|
end)
|
|
|
|
return filenames
|
|
end
|
|
|
|
-- alphanum sorting for humans in Lua
|
|
-- http://notebook.kulchenko.com/algorithms/alphanumeric-natural-sorting-for-humans-in-lua
|
|
function alphanumsort_lua(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
|
|
|
|
function alphanumsort(filenames)
|
|
local is_ffi_loaded = pcall(require, "ffi")
|
|
if is_windows and is_ffi_loaded then
|
|
alphanumsort_windows(filenames)
|
|
else
|
|
alphanumsort_lua(filenames)
|
|
end
|
|
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.
|
|
local current = prefix .. v == current_file
|
|
if o.ignore_hidden and not current and string.match(v, "^%.") then
|
|
return false
|
|
end
|
|
if not current and is_ignored(v) then
|
|
return false
|
|
end
|
|
|
|
local ext = get_extension(v)
|
|
if ext == nil then
|
|
return false
|
|
end
|
|
local name = mp.get_property("filename")
|
|
if o.same_series then
|
|
local name = mp.get_property("filename")
|
|
for ext, _ in pairs(extensions) do
|
|
if name:match(ext .. "$") ~= nil and v ~= name and
|
|
not is_same_series(name, v)
|
|
then
|
|
return false
|
|
end
|
|
end
|
|
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)
|