Compare commits

13 Commits

Author SHA1 Message Date
ksyasuda
ff60b12e61 change helper functions to use mpv utils
- use utils.file_info.is_file to check if the file exits and is a file
- use utils.split_path to split path into directory and filename components
2023-08-08 00:47:19 -07:00
Kyle Yasuda
6ca5de8aa4 fix error due to adding video to queue already surrounded by quotes (#8)
- fix surround with quotes function to only surround with quotes if not already there
- remove quotes before adding to video_queue
2023-08-07 23:43:30 -07:00
Kyle Yasuda
64bde58672 update add_to_queue function to allow adding local files to the queue (#7)
* update add_to_queue function to allow adding local files to the queue
* update functions to work with local files

- change download to do nothing on local file
- change print function to not try to print channel name
- change move cursor up and down to be repeatable
2023-08-07 23:11:51 -07:00
Kyle Yasuda
6e3d782081 add ytdlp_file_format config variable (#6)
- allow choosing preferred file format for downloads
- update readme
2023-08-07 17:46:25 -07:00
Kyle Yasuda
cc8c677e0b Cleanup code (#5)
* cleanup code

- reorder variables
- set show_errors to true by default
- remove old variable from config file
- update readme
2023-08-07 02:17:05 -07:00
Kyle Yasuda
a39a7c6a95 Create luackeck.yml (#4) 2023-08-07 01:15:05 -07:00
Kyle Yasuda
af81562606 Dev (#3)
* add download_selected_video function
* add ytdlp_output_template to conf file

- add function to download the currently selected video in the queue
- update readme and conf file
- change ordering of options to have keybindings in alphabetical order
2023-08-06 23:30:28 -07:00
ksyasuda
8b4d2639e6 update readme 2023-08-06 20:41:42 -07:00
Kyle Yasuda
f5b5887616 Merge pull request #2 from ksyasuda/dev
update readme
2023-08-06 20:36:34 -07:00
ksyasuda
33918e87d0 update readme 2023-08-06 20:35:33 -07:00
Kyle Yasuda
fd742b00cb Merge pull request #1 from ksyasuda/remove-from-queue
add remove_from_queue function
2023-08-06 17:41:11 -07:00
ksyasuda
5e038d212e add remove_from_queue function
- add function/binding (ctrl+x) to remove currently selected video from queue
- update readme
- move get_video_info function into YouTubeQueue
2023-08-06 17:31:05 -07:00
c67ac189b0 add play_selected_video and update show_error 2023-08-06 12:11:34 -07:00
4 changed files with 251 additions and 123 deletions

10
.github/workflows/luackeck.yml vendored Normal file
View File

@@ -0,0 +1,10 @@
name: Luacheck
on: [push, pull_request]
jobs:
sile:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v3
- name: Luacheck linter
uses: lunarmodules/luacheck@v1

View File

@@ -1,7 +1,10 @@
# mpv-youtube-queue # mpv-youtube-queue
A Lua script for mpv that allows you to add YouTube videos to a queue, <div align="center">
navigate through the queue, and select a video to play.
A Lua script that implements the YouTube 'Add to Queue' functionality for mpv
</div>
![mpv-youtube-queue image](.assets/mpv-youtube-queue.png) ![mpv-youtube-queue image](.assets/mpv-youtube-queue.png)
@@ -12,9 +15,9 @@ navigate through the queue, and select a video to play.
- Select a video to play from the queue with an interactive menu, - Select a video to play from the queue with an interactive menu,
or navigate through the queue with keyboard shortcuts or navigate through the queue with keyboard shortcuts
- Edit the order of videos in the queue - Edit the order of videos in the queue
- Open the URL of the currently playing video in a new browser tab - Open the URL or channel page of the currently playing video in a new browser tab
- Open the channel page of the currently playing video
- Download the currently playing video - Download the currently playing video
- Download a video in the queue
## Notes ## Notes
@@ -44,33 +47,44 @@ This script requires the following software to be installed on the system
- `add_to_queue - ctrl+a`: Add a video in the clipboard to the queue - `add_to_queue - ctrl+a`: Add a video in the clipboard to the queue
- `download_current_video - ctrl+d`: Download the currently playing video - `download_current_video - ctrl+d`: Download the currently playing video
- `move_cursor_down - ctrl+DOWN`:N - Move the cursor down one row in the queue - `download_selected_video - ctrl+D`: Download the currently selected video
- `move_cursor_up - ctrl+UP`- Move the cursor up one row in the queue in the queue
- `move_cursor_down - ctrl+j`: Move the cursor down one row in the queue
- `move_cursor_up - ctrl+k`- Move the cursor up one row in the queue
- `move_video - ctrl+m`: Mark/move the selected video in the queue - `move_video - ctrl+m`: Mark/move the selected video in the queue
- `play_next_in_queue - ctrl+n`: Play the next video in the queue
- `open_video_in_browser - ctrl+o`: Open the currently playing video in the browser
- `open_channel_in_browser - ctrl+O`: Open the channel page for the currently - `open_channel_in_browser - ctrl+O`: Open the channel page for the currently
playing video in the browser playing video in the browser
- `open_video_in_browser - ctrl+o`: Open the currently playing video in the browser
- `play_next_in_queue - ctrl+n`: Play the next video in the queue
- `play_previous_in_queue - ctrl+p`: Play the previous video in the queue - `play_previous_in_queue - ctrl+p`: Play the previous video in the queue
- `print_current_video - ctrl+P`: Print the name and channel of the currently - `print_current_video - ctrl+P`: Print the name and channel of the currently
playing video to the OSD playing video to the OSD
- `print_queue - ctrl+q`: Print the contents of the queue to the OSD - `print_queue - ctrl+q`: Print the contents of the queue to the OSD
- `remove_from_queue - ctrl+x`: Remove the currently selected video from the
queue
- `play_selected_video - ctrl+ENTER`: Play the currently selected video in
the queue
### Default Option ### Default Option
- `clipboard_command - xclip -o`: The command to use to get the contents of the clipboard
- `browser - firefox`: The browser to use when opening a video or channel page - `browser - firefox`: The browser to use when opening a video or channel page
- `clipboard_command - xclip -o`: The command to use to get the contents of the clipboard
- `cursor_icon - ➤`: The icon to use for the cursor - `cursor_icon - ➤`: The icon to use for the cursor
- `marked_icon - `: The icon to use to mark a video as ready to be moved in - `display_limit - 6`: The maximum amount of videos to show on the OSD at once
the queue - `download_directory - ~/videos/YouTube`: The directory to use when
- `download_directory ~/videos/YouTube`: The directory to use when downloading downloading a video
a video
- `downloader - curl`: The name of the program to use to download the video
- `download_quality 720p`: The maximum download quality - `download_quality 720p`: The maximum download quality
- `downloader - curl`: The name of the program to use to download the video
- `font_name - JetBrains Mono`: The name of the font to use - `font_name - JetBrains Mono`: The name of the font to use
- `font_size - 12`: Size of the font - `font_size - 12`: Size of the font
- `display_limit - 6`: The maximum amount of videos to show on the OSD at once - `marked_icon - `: The icon to use to mark a video as ready to be moved in
- `show_errors - no`: Show error messages on the OSD the queue
- `show_errors - yes`: Show error messages on the OSD
- `ytdlp_file_format - mp4`: The preferred file format for downloaded videos
- `ytdlp_output_template - %(uploader)s/%(title)s.%(ext)s`: The [yt-dlp output
template string](https://github.com/yt-dlp/yt-dlp#output-template)
- Full path with the default `download_directory`
is: `~/videos/YouTube/<uploader>/<title>.<ext>`
## License ## License

View File

@@ -1,22 +1,27 @@
add_to_queue=ctrl+a add_to_queue=ctrl+a
download_current_video=ctrl+d
download_selected_video=ctrl+D
move_cursor_down=ctrl+j
move_cursor_up=ctrl+k
move_video=ctrl+m
play_next_in_queue=ctrl+n play_next_in_queue=ctrl+n
play_previous_in_queue=ctrl+p
print_queue=ctrl+q
move_cursor_up=ctrl+UP
move_cursor_down=ctrl+DOWN
play_selected_video=ctrl+ENTER
open_video_in_browser=ctrl+o open_video_in_browser=ctrl+o
open_channel_in_browser=ctrl+O open_channel_in_browser=ctrl+O
play_previous_in_queue=ctrl+p
print_current_video=ctrl+P print_current_video=ctrl+P
print_queue=ctrl+q
remove_from_queue=ctrl+x
play_selected_video=ctrl+ENTER
browser=firefox browser=firefox
clipboard_command=xclip -o clipboard_command=xclip -o
display_limit=6
cursor_icon=➤ cursor_icon=➤
marked_icon=⇅ display_limit=6
font_size=12
font_name=JetBrains Mono
download_quality=720p
download_directory=~/videos/YouTube download_directory=~/videos/YouTube
download_format_str=%(uploader)s/%(title)s.%(ext)s download_quality=720p
downloader=curl downloader=curl
show_errors=no font_name=JetBrains Mono
font_size=12
marked_icon=⇅
show_errors=yes
ytdlp_file_format=mp4
ytdlp_output_template=%(uploader)s/%(title)s.%(ext)s

View File

@@ -15,41 +15,38 @@
-- along with this program. If not, see <https://www.gnu.org/licenses/>. -- along with this program. If not, see <https://www.gnu.org/licenses/>.
local mp = require 'mp' local mp = require 'mp'
mp.options = require 'mp.options' mp.options = require 'mp.options'
local YouTubeQueue = {} local utils = require 'mp.utils'
local video_queue = {}
local current_video = nil
local index = 0
local selected_index = 1
local MSG_DURATION = 1.5
local marked_index = nil
local styleOn = mp.get_property("osd-ass-cc/0") local styleOn = mp.get_property("osd-ass-cc/0")
local styleOff = mp.get_property("osd-ass-cc/1") local styleOff = mp.get_property("osd-ass-cc/1")
local options = { local options = {
add_to_queue = "ctrl+a", add_to_queue = "ctrl+a",
download_current_video = "ctrl+d", download_current_video = "ctrl+d",
move_cursor_down = "ctrl+DOWN", download_selected_video = "ctrl+D",
move_cursor_up = "ctrl+UP", move_cursor_down = "ctrl+j",
move_cursor_up = "ctrl+k",
move_video = "ctrl+m", move_video = "ctrl+m",
open_channel_in_browser = "ctrl+O",
open_video_in_browser = "ctrl+o",
play_next_in_queue = "ctrl+n", play_next_in_queue = "ctrl+n",
open_video_in_browser = "ctrl+o",
open_channel_in_browser = "ctrl+O",
play_previous_in_queue = "ctrl+p", play_previous_in_queue = "ctrl+p",
play_selected_video = "ctrl+ENTER",
print_current_video = "ctrl+P", print_current_video = "ctrl+P",
print_queue = "ctrl+q", print_queue = "ctrl+q",
clipboard_command = "xclip -o", remove_from_queue = "ctrl+x",
play_selected_video = "ctrl+ENTER",
browser = "firefox", browser = "firefox",
clipboard_command = "xclip -o",
cursor_icon = "", cursor_icon = "",
marked_icon = "", display_limit = 6,
download_directory = "~/videos/YouTube", download_directory = "~/videos/YouTube",
download_format_str = "%(uploader)s/%(title)s.%(ext)s",
downloader = "curl",
download_quality = "720p", download_quality = "720p",
downloader = "curl",
font_name = "JetBrains Mono", font_name = "JetBrains Mono",
font_size = 12, font_size = 12,
display_limit = 6, marked_icon = "",
show_errors = false show_errors = true,
ytdlp_file_format = "mp4",
ytdlp_output_template = "%(uploader)s/%(title)s.%(ext)s"
} }
mp.options.read_options(options, "mpv-youtube-queue") mp.options.read_options(options, "mpv-youtube-queue")
@@ -83,11 +80,29 @@ local style = {
sortoftransparent .. "}" sortoftransparent .. "}"
} }
local YouTubeQueue = {}
local video_queue = {}
local MSG_DURATION = 1.5
local display_limit = options.display_limit local display_limit = options.display_limit
local index = 0
local selected_index = 1
local display_offset = 0 local display_offset = 0
local marked_index = nil
local current_video = nil
-- HELPERS {{{ -- HELPERS {{{
-- surround string with single quotes if it does not already have them
local function surround_with_quotes(s)
if string.sub(s, 0, 1) == "'" and string.sub(s, -1) == "'" then
return s
else
return "'" .. s .. "'"
end
end
local function remove_quotes(s) return string.gsub(s, "'", "") end
-- run sleep shell command for n seconds -- run sleep shell command for n seconds
local function sleep(n) os.execute("sleep " .. tonumber(n)) end local function sleep(n) os.execute("sleep " .. tonumber(n)) end
@@ -99,11 +114,31 @@ local function print_osd_message(message, duration, s)
duration) duration)
end end
-- returns true if the provided path exists and is a file
local function is_file(filepath)
mp.msg.info("FILEPATH: " .. filepath)
local result = utils.file_info(filepath)
if result == nil then
print_osd_message("File not found: " .. filepath, 3, style.error)
return false
end
return result.is_file
end
-- returns the filename given a path (e.g. /home/user/file.txt -> file.txt)
local function split_path(filepath)
if is_file(filepath) then return utils.split_path(filepath) end
end
local function print_current_video() local function print_current_video()
local current = YouTubeQueue.get_current_video() local current = YouTubeQueue.get_current_video()
if is_file(current.video_url) then
print_osd_message("Playing: " .. current.video_name, 3)
else
print_osd_message("Playing: " .. current.video_name .. ' by ' .. print_osd_message("Playing: " .. current.video_name .. ' by ' ..
current.channel_name, 3) current.channel_name, 3)
end end
end
local function expanduser(path) local function expanduser(path)
-- remove trailing slash if it exists -- remove trailing slash if it exists
@@ -121,7 +156,7 @@ local function expanduser(path)
end end
local function open_url_in_browser(url) local function open_url_in_browser(url)
local command = options.browser .. " " .. url local command = options.browser .. " " .. surround_with_quotes(url)
os.execute(command) os.execute(command)
end end
@@ -133,28 +168,6 @@ local function open_channel_in_browser()
open_url_in_browser(YouTubeQueue.get_current_video().channel_url) open_url_in_browser(YouTubeQueue.get_current_video().channel_url)
end end
local function get_video_info(url)
local command =
'yt-dlp --print channel_url --print uploader --print title --playlist-items 1 ' ..
url
local handle = io.popen(command)
if handle == nil then return nil, nil, nil end
local result = handle:read("*a")
handle:close()
-- Split the result into URL, name, and video title
local channel_url, channel_name, video_name = result:match(
"(.-)\n(.-)\n(.*)")
-- Remove trailing whitespace
if channel_url ~= nil then channel_url = channel_url:gsub("%s+$", "") end
if channel_name ~= nil then channel_name = channel_name:gsub("%s+$", "") end
if video_name ~= nil then video_name = video_name:gsub("%s+$", "") end
return channel_url, channel_name, video_name
end
-- local function is_valid_ytdlp_url(url) -- local function is_valid_ytdlp_url(url)
-- local command = 'yt-dlp --simulate \'' .. url .. '\' >/dev/null 2>&1' -- local command = 'yt-dlp --simulate \'' .. url .. '\' >/dev/null 2>&1'
-- local handle = io.popen(command .. "; echo $?") -- local handle = io.popen(command .. "; echo $?")
@@ -203,6 +216,28 @@ function YouTubeQueue.get_clipboard_content()
return result return result
end end
function YouTubeQueue.get_video_info(url)
local command =
'yt-dlp --print channel_url --print uploader --print title --playlist-items 1 ' ..
surround_with_quotes(url)
local handle = io.popen(command)
if handle == nil then return nil, nil, nil end
local result = handle:read("*a")
handle:close()
-- Split the result into URL, name, and video title
local channel_url, channel_name, video_name = result:match(
"(.-)\n(.-)\n(.*)")
-- Remove trailing whitespace
if channel_url ~= nil then channel_url = channel_url:gsub("%s+$", "") end
if channel_name ~= nil then channel_name = channel_name:gsub("%s+$", "") end
if video_name ~= nil then video_name = video_name:gsub("%s+$", "") end
return channel_url, channel_name, video_name
end
-- }}} -- }}}
-- QUEUE FUNCTIONS {{{ -- QUEUE FUNCTIONS {{{
@@ -386,6 +421,8 @@ function YouTubeQueue.play_next_in_queue()
return return
end end
local current_index = YouTubeQueue.get_current_index() local current_index = YouTubeQueue.get_current_index()
-- if the current video is not the first in the queue, then play the video
-- else, check if the video is playing and if not play the video with replace
if YouTubeQueue.size() > 1 then if YouTubeQueue.size() > 1 then
mp.set_property_number("playlist-pos", current_index - 1) mp.set_property_number("playlist-pos", current_index - 1)
else else
@@ -399,8 +436,10 @@ function YouTubeQueue.play_next_in_queue()
sleep(MSG_DURATION) sleep(MSG_DURATION)
end end
-- add the video to the queue from the clipboard -- add the video to the queue from the clipboard or call from script-message
function YouTubeQueue.add_to_queue(url) -- updates the internal playlist by default, pass 0 to disable
function YouTubeQueue.add_to_queue(url, update_internal_playlist)
if update_internal_playlist == nil then update_internal_playlist = 0 end
if url == nil or url == "" then if url == nil or url == "" then
url = YouTubeQueue.get_clipboard_content() url = YouTubeQueue.get_clipboard_content()
if url == nil or url == "" then if url == nil or url == "" then
@@ -413,26 +452,47 @@ function YouTubeQueue.add_to_queue(url)
print_osd_message("Video already in queue.", MSG_DURATION, style.error) print_osd_message("Video already in queue.", MSG_DURATION, style.error)
return return
end end
local channel_url, channel_name, video_name = get_video_info(url)
local video, channel_url, channel_name, video_name, video_url
if is_file(url) then
video_url = url
channel_url, video_name = split_path(video_url)
mp.msg.info("channel_url: " .. channel_url)
mp.msg.info("video_name: " .. video_name)
channel_name = "Local file"
video = {
video_url = video_url,
video_name = video_name,
channel_url = channel_url,
channel_name = channel_name
}
else
channel_url, channel_name, video_name = YouTubeQueue.get_video_info(url)
url = remove_quotes(url)
if (channel_url == nil or channel_name == nil or video_name == nil) or if (channel_url == nil or channel_name == nil or video_name == nil) or
(channel_url == "" or channel_name == "" or video_name == "") then (channel_url == "" or channel_name == "" or video_name == "") then
print_osd_message("Error getting video info.", MSG_DURATION, style.error) print_osd_message("Error getting video info.", MSG_DURATION,
return style.error)
end else
video = {
local video = {
video_url = url, video_url = url,
video_name = video_name, video_name = video_name,
channel_url = channel_url, channel_url = channel_url,
channel_name = channel_name channel_name = channel_name
} }
end
end
table.insert(video_queue, video) table.insert(video_queue, video)
-- if the queue was empty, start playing the video
-- otherwise, add the video to the playlist
if not YouTubeQueue.get_current_video() then if not YouTubeQueue.get_current_video() then
YouTubeQueue.play_next_in_queue() YouTubeQueue.play_next_in_queue()
else elseif update_internal_playlist == 0 then
mp.commandv("loadfile", url, "append-play") mp.commandv("loadfile", url, "append-play")
print_osd_message("Added " .. video_name .. " to queue.", MSG_DURATION)
end end
print_osd_message("Added " .. video_name .. " to queue.", MSG_DURATION)
end end
-- play the previous video in the queue -- play the previous video in the queue
@@ -450,46 +510,81 @@ function YouTubeQueue.play_previous_video()
sleep(MSG_DURATION) sleep(MSG_DURATION)
end end
function YouTubeQueue.download_current_video() function YouTubeQueue.download_video_at(idx)
if current_video and current_video ~= "" then
local o = options local o = options
local v = current_video local v = video_queue[idx]
local q = o.download_quality:sub(1, -2) local q = o.download_quality:sub(1, -2)
local dl_dir = expanduser(o.download_directory) local dl_dir = expanduser(o.download_directory)
local command = 'yt-dlp -f \'bestvideo[height<=' .. q .. local command = 'yt-dlp -f \'bestvideo[height<=' .. q .. '][ext=' ..
']+bestaudio/best[height<=' .. q .. ']\' -o "' .. options.ytdlp_file_format .. ']+bestaudio/best[height<=' ..
dl_dir .. "/" .. options.download_format_str .. q .. ']/bestvideo[height<=' .. q ..
'" --downloader ' .. o.downloader .. ' ' .. ']+bestaudio/best[height<=' .. q .. ']\' -o "' .. dl_dir ..
v.video_url "/" .. options.ytdlp_output_template ..
'" --downloader ' .. o.downloader .. ' ' .. v.video_url
-- Run the download command -- Run the download command
local handle = io.popen(command) local handle = io.popen(command)
if handle == nil then if handle == nil then
print_osd_message("Error starting download.", MSG_DURATION, print_osd_message("Error starting download.", MSG_DURATION, style.error)
style.error)
return return
end end
print_osd_message("Starting download for " .. v.video_name, MSG_DURATION) print_osd_message("Starting download for " .. v.video_name, MSG_DURATION)
local result = handle:read("*a") local result = handle:read("*a")
handle:close() handle:close()
if result == nil then if result == nil then
print_osd_message("Error starting download.", MSG_DURATION, print_osd_message("Error starting download.", MSG_DURATION, style.error)
style.error)
return return
end end
if result then if result then
print_osd_message("Finished downloading " .. v.video_name, print_osd_message("Finished downloading " .. v.video_name, MSG_DURATION)
MSG_DURATION)
else else
print_osd_message("Error downloading " .. v.video_name, print_osd_message("Error downloading " .. v.video_name, MSG_DURATION,
MSG_DURATION, style.error) style.error)
end end
end
function YouTubeQueue.download_current_video()
if is_file(current_video.video_url) then
print_osd_message("Current video is a local file... doing nothing.",
MSG_DURATION, style.error)
return
end
if current_video ~= nil and current_video ~= "" then
YouTubeQueue.download_video_at(index)
else else
print_osd_message("No video to download.", MSG_DURATION, style.error) print_osd_message("No video to download.", MSG_DURATION, style.error)
end end
end end
function YouTubeQueue.download_selected_video()
if selected_index == 1 and current_video == nil then
print_osd_message("No video to download.", MSG_DURATION, style.error)
return
end
if is_file(YouTubeQueue.get_video_at(selected_index)) then
print_osd_message("Current video is a local file... doing nothing.",
MSG_DURATION, style.error)
return
end
YouTubeQueue.download_video_at(selected_index)
end
function YouTubeQueue.remove_from_queue()
if index == selected_index then
print_osd_message("Cannot remove current video", MSG_DURATION,
style.error)
return
end
table.remove(video_queue, selected_index)
mp.commandv("playlist-remove", selected_index - 1)
print_osd_message("Deleted " .. current_video.video_name .. " from queue.",
MSG_DURATION)
if selected_index > 1 then selected_index = selected_index - 1 end
index = index - 1
YouTubeQueue.print_queue()
end
-- }}} -- }}}
-- LISTENERS {{{ -- LISTENERS {{{
@@ -506,7 +601,7 @@ local function on_track_changed() YouTubeQueue.update_current_index() end
-- Function to be called when the playback-restart event is triggered -- Function to be called when the playback-restart event is triggered
local function on_playback_restart() local function on_playback_restart()
local playlist_size = mp.get_property_number("playlist-count", 0) local playlist_size = mp.get_property_number("playlist-count", 0)
if playlist_size > 1 then if current_video ~= nil and playlist_size > 1 then
YouTubeQueue.update_current_index() YouTubeQueue.update_current_index()
elseif current_video == nil then elseif current_video == nil then
local url = mp.get_property("path") local url = mp.get_property("path")
@@ -525,9 +620,9 @@ mp.add_key_binding(options.play_previous_in_queue, "play_previous_video",
YouTubeQueue.play_previous_video) YouTubeQueue.play_previous_video)
mp.add_key_binding(options.print_queue, "print_queue", YouTubeQueue.print_queue) mp.add_key_binding(options.print_queue, "print_queue", YouTubeQueue.print_queue)
mp.add_key_binding(options.move_cursor_up, "move_cursor_up", mp.add_key_binding(options.move_cursor_up, "move_cursor_up",
YouTubeQueue.move_cursor_up) YouTubeQueue.move_cursor_up, { repeatable = true })
mp.add_key_binding(options.move_cursor_down, "move_cursor_down", mp.add_key_binding(options.move_cursor_down, "move_cursor_down",
YouTubeQueue.move_cursor_down) YouTubeQueue.move_cursor_down, { repeatable = true })
mp.add_key_binding(options.play_selected_video, "play_selected_video", mp.add_key_binding(options.play_selected_video, "play_selected_video",
YouTubeQueue.play_selected_video) YouTubeQueue.play_selected_video)
mp.add_key_binding(options.open_video_in_browser, "open_video_in_browser", mp.add_key_binding(options.open_video_in_browser, "open_video_in_browser",
@@ -538,8 +633,12 @@ mp.add_key_binding(options.open_channel_in_browser, "open_channel_in_browser",
open_channel_in_browser) open_channel_in_browser)
mp.add_key_binding(options.download_current_video, "download_current_video", mp.add_key_binding(options.download_current_video, "download_current_video",
YouTubeQueue.download_current_video) YouTubeQueue.download_current_video)
mp.add_key_binding(options.download_selected_video, "download_selected_video",
YouTubeQueue.download_selected_video)
mp.add_key_binding(options.move_video, "move_video", mp.add_key_binding(options.move_video, "move_video",
YouTubeQueue.mark_and_move_video) YouTubeQueue.mark_and_move_video)
mp.add_key_binding(options.remove_from_queue, "delete_video",
YouTubeQueue.remove_from_queue)
mp.register_event("end-file", on_end_file) mp.register_event("end-file", on_end_file)
mp.register_event("track-changed", on_track_changed) mp.register_event("track-changed", on_track_changed)