------------------------------------------------------------------------------------------ ---------------------------------File/Playlist Opening------------------------------------ ------------------------------------------------------------------------------------------ ------------------------------------------------------------------------------------------ local mp = require 'mp' local msg = require 'mp.msg' local utils = require 'mp.utils' local o = require 'modules.options' local g = require 'modules.globals' local fb_utils = require 'modules.utils' local ass = require 'modules.ass' local cursor = require 'modules.navigation.cursor' local controls = require 'modules.controls' local scanning = require 'modules.navigation.scanning' local movement = require 'modules.navigation.directory-movement' local state = g.state -- In mpv v0.38 a new index argument was added to the loadfile command. -- For some crazy reason this new argument is placed before the existing options -- argument, breaking any scripts that used it. This function finds the correct index -- for the options argument using the `command-list` property. local function get_loadfile_options_arg_index() local command_list = mp.get_property_native('command-list', {}) for _, command in ipairs(command_list) do if command.name == 'loadfile' then for i, arg in ipairs(command.args or {}) do if arg.name == 'options' then return i end end end end return 3 end local LEGACY_LOADFILE_SYNTAX = get_loadfile_options_arg_index() == 3 -- A wrapper around loadfile to handle the syntax changes introduced in mpv v0.38. local function legacy_loadfile_wrapper(file, flag, options) if LEGACY_LOADFILE_SYNTAX then return mp.command_native({"loadfile", file, flag, options}) else return mp.command_native({"loadfile", file, flag, -1, options}) end end --adds a file to the playlist and changes the flag to `append-play` in preparation --for future items local function loadfile(file, opts, mpv_opts) if o.substitute_backslash and not fb_utils.get_protocol(file) then file = file:gsub("/", "\\") end if opts.flag == "replace" then msg.verbose("Playling file", file) else msg.verbose("Appending", file, "to the playlist") end if mpv_opts then msg.debug('Settings opts on', file, ':', utils.to_string(mpv_opts)) end if not legacy_loadfile_wrapper(file, opts.flag, mpv_opts) then msg.warn(file) end opts.flag = "append-play" opts.items_appended = opts.items_appended + 1 end --this function recursively loads directories concurrently in separate coroutines --results are saved in a tree of tables that allows asynchronous access local function concurrent_loadlist_parse(directory, load_opts, prev_dirs, item_t) --prevents infinite recursion from the item.path or opts.directory fields if prev_dirs[directory] then return end prev_dirs[directory] = true local list, list_opts = scanning.scan_directory(directory, { source = "loadlist" }) if list == g.root then return end --if we can't parse the directory then append it and hope mpv fares better if list == nil then msg.warn("Could not parse", directory, "appending to playlist anyway") item_t.type = "file" return end directory = list_opts.directory or directory if directory == "" then return end --we must declare these before we start loading sublists otherwise the append thread will --need to wait until the whole list is loaded (when synchronous IO is used) item_t._sublist = list or {} list._directory = directory --launches new parse operations for directories, each in a different coroutine for _, item in ipairs(list) do if fb_utils.parseable_item(item) then fb_utils.coroutine.run(concurrent_loadlist_wrapper, fb_utils.get_new_directory(item, directory), load_opts, prev_dirs, item) end end return true end --a wrapper function that ensures the concurrent_loadlist_parse is run correctly function concurrent_loadlist_wrapper(directory, opts, prev_dirs, item) --ensures that only a set number of concurrent parses are operating at any one time. --the mpv event queue is seemingly limited to 1000 items, but only async mpv actions like --command_native_async should use that, events like mp.add_timeout (which coroutine.sleep() uses) should --be handled enturely on the Lua side with a table, which has a significantly larger maximum size. while (opts.concurrency > o.max_concurrency) do fb_utils.coroutine.sleep(0.1) end opts.concurrency = opts.concurrency + 1 local success = concurrent_loadlist_parse(directory, opts, prev_dirs, item) opts.concurrency = opts.concurrency - 1 if not success then item._sublist = {} end if coroutine.status(opts.co) == "suspended" then fb_utils.coroutine.resume_err(opts.co) end end --recursively appends items to the playlist, acts as a consumer to the previous functions producer; --if the next directory has not been parsed this function will yield until the parse has completed local function concurrent_loadlist_append(list, load_opts) local directory = list._directory for _, item in ipairs(list) do if not g.sub_extensions[ fb_utils.get_extension(item.name, "") ] and not g.audio_extensions[ fb_utils.get_extension(item.name, "") ] then while (not item._sublist and fb_utils.parseable_item(item)) do coroutine.yield() end if fb_utils.parseable_item(item) then concurrent_loadlist_append(item._sublist, load_opts) else loadfile(fb_utils.get_full_path(item, directory), load_opts, item.mpv_options) end end end end --recursive function to load directories using the script custom parsers --returns true if any items were appended to the playlist local function custom_loadlist_recursive(directory, load_opts, prev_dirs) --prevents infinite recursion from the item.path or opts.directory fields if prev_dirs[directory] then return end prev_dirs[directory] = true local list, opts = scanning.scan_directory(directory, { source = "loadlist" }) if list == g.root then return end --if we can't parse the directory then append it and hope mpv fares better if list == nil then msg.warn("Could not parse", directory, "appending to playlist anyway") loadfile(directory, load_opts.flag) return true end directory = opts.directory or directory if directory == "" then return end for _, item in ipairs(list) do if not g.sub_extensions[ fb_utils.get_extension(item.name, "") ] and not g.audio_extensions[ fb_utils.get_extension(item.name, "") ] then if fb_utils.parseable_item(item) then custom_loadlist_recursive( fb_utils.get_new_directory(item, directory) , load_opts, prev_dirs) else local path = fb_utils.get_full_path(item, directory) loadfile(path, load_opts, item.mpv_options) end end end end --a wrapper for the custom_loadlist_recursive function local function loadlist(item, opts) local dir = fb_utils.get_full_path(item, opts.directory) local num_items = opts.items_appended if o.concurrent_recursion then item = fb_utils.copy_table(item) opts.co = fb_utils.coroutine.assert() opts.concurrency = 0 --we need the current coroutine to suspend before we run the first parse operation, so --we schedule the coroutine to run on the mpv event queue mp.add_timeout(0, function() fb_utils.coroutine.run(concurrent_loadlist_wrapper, dir, opts, {}, item) end) concurrent_loadlist_append({item, _directory = opts.directory}, opts) else custom_loadlist_recursive(dir, opts, {}) end if opts.items_appended == num_items then msg.warn(dir, "contained no valid files") end end --load playlist entries before and after the currently playing file local function autoload_dir(path, opts) if o.autoload_save_current and path == g.current_file.path then mp.commandv("write-watch-later-config") end --loads the currently selected file, clearing the playlist in the process loadfile(path, opts) local pos = 1 local file_count = 0 for _,item in ipairs(state.list) do if item.type == "file" and not g.sub_extensions[ fb_utils.get_extension(item.name, "") ] and not g.audio_extensions[ fb_utils.get_extension(item.name, "") ] then local p = fb_utils.get_full_path(item) if p == path then pos = file_count else loadfile( p, opts, item.mpv_options) end file_count = file_count + 1 end end mp.commandv("playlist-move", 0, pos+1) end --runs the loadfile or loadlist command local function open_item(item, opts) if fb_utils.parseable_item(item) then return loadlist(item, opts) end local path = fb_utils.get_full_path(item, opts.directory) if g.sub_extensions[ fb_utils.get_extension(item.name, "") ] then mp.commandv("sub-add", path, opts.flag == "replace" and "select" or "auto") elseif g.audio_extensions[ fb_utils.get_extension(item.name, "") ] then mp.commandv("audio-add", path, opts.flag == "replace" and "select" or "auto") else if opts.autoload then autoload_dir(path, opts) else loadfile(path, opts, item.mpv_options) end end end --handles the open options as a coroutine --once loadfile has been run we can no-longer guarantee synchronous execution - the state values may change --therefore, we must ensure that any state values that could be used after a loadfile call are saved beforehand local function open_file_coroutine(opts) if not state.list[state.selected] then return end if opts.flag == 'replace' then controls.close() end --we want to set the idle option to yes to ensure that if the first item --fails to load then the player has a chance to attempt to load further items (for async append operations) local idle = mp.get_property("idle", "once") mp.set_property("idle", "yes") --handles multi-selection behaviour if next(state.selection) then local selection = fb_utils.sort_keys(state.selection) --reset the selection after state.selection = {} cursor.disable_select_mode() ass.update_ass() --the currently selected file will be loaded according to the flag --the flag variable will be switched to append once a file is loaded for i=1, #selection do open_item(selection[i], opts) end else local item = state.list[state.selected] if opts.flag == "replace" then movement.down_dir() end open_item(item, opts) end if mp.get_property("idle") == "yes" then mp.set_property("idle", idle) end end --opens the selelected file(s) local function open_file(flag, autoload) fb_utils.coroutine.run(open_file_coroutine, { flag = flag, autoload = (autoload ~= o.autoload and flag == "replace"), directory = state.directory, items_appended = 0 }) end return { add_files = open_file, }