Files
mpv/player/lua/console.lua
T
CogentRedTester 0af0c137bb console.lua: flush keybinds before sending mp.input opened events
This commit uses the undocumented `mp.flush_keybindings()` method
to ensure that all `console.lua` keybinds are set before
sending the `opened` event to `mp.input` clients.

Previously, the keybinds would only be flushed when `console.lua` went
idle. As the `opened` event is sent before this happens, clients
processing the event may end up flushing their own keybinds before this
happens. This makes it unreliable to use the `opened` event to override
keybinds set by `console.lua`; for example, adding different
behaviour to `input.select` when using `Shift+Enter` instead of `Enter`.

This commit allows the `opened` callback to be used to safely override
`console.lua` keybinds.

.luacheckrc: move flush_keybindings to globals
2026-02-24 20:12:57 +01:00

1887 lines
59 KiB
Lua

-- Copyright (C) 2019 the mpv developers
--
-- Permission to use, copy, modify, and/or distribute this software for any
-- purpose with or without fee is hereby granted, provided that the above
-- copyright notice and this permission notice appear in all copies.
--
-- THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
-- WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
-- MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
-- SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
-- WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
-- OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
-- CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
local utils = require "mp.utils"
local assdraw = require "mp.assdraw"
local function detect_platform()
local platform = mp.get_property_native("platform")
if platform == "darwin" or platform == "windows" then
return platform
elseif os.getenv("WAYLAND_DISPLAY") or os.getenv("WAYLAND_SOCKET") then
return "wayland"
end
return "x11"
end
local platform = detect_platform()
-- Default options
local opts = {
monospace_font = "",
font_size = 24,
border_size = 1.65,
background_alpha = 80,
gap = 0.2,
padding = 10,
menu_outline_size = 0,
menu_outline_color = "#FFFFFF",
corner_radius = 8,
margin_x = -1,
margin_y = -1,
scale_with_window = "auto",
focused_color = "#222222",
focused_back_color = "#FFFFFF",
match_color = "#0088FF",
exact_match = false,
case_sensitive = false,
history_dedup = true,
font_hw_ratio = "auto",
}
local styles = {
error = "{\\1c&H7a77f2&}",
completion = "{\\1c&Hcc99cc&}",
}
for key, style in pairs(styles) do
styles[key] = style .. "{\\3c&H111111&}"
end
local terminal_styles = {
error = "\027[31m",
selected_completion = "\027[7m",
default_item = "\027[1m",
disabled = "\027[38;5;8m",
matched_position = "\027[34m",
match_end = "\027[39m",
}
local open = false
local osd_msg_active = false
local insert_mode = false
local pending_update = false
local ime_active = false
local line = ""
local cursor = 1
local prompt
local id
local overlay = mp.create_osd_overlay("ass-events")
local width_overlay = mp.create_osd_overlay("ass-events")
width_overlay.compute_bounds = true
width_overlay.hidden = true
local histories = {}
local history = {}
local history_pos = 1
local searching_history = false
local history_paths = {}
local histories_to_save = {}
local MAX_LOG_LINES = 10000
local log_buffers = {}
local log_offset = 0
local key_bindings = {}
local dont_bind_up_down = false
local global_margins = { t = 0, b = 0 }
local input_caller
local input_caller_handler
local keep_open = false
local completion_buffer = {}
local selected_completion_index
local completion_pos
local completion_append
local completion_old_line
local completion_old_cursor
local autoselect_completion
local has_completions
local selectable_items
local matches = {}
local focused_match = 1
local first_match_to_print = 1
local default_item
local item_positions = {}
local max_item_width = 0
local horizontal_offset = 0
local complete
local cycle_through_completions
local set_active
local property_cache = {}
local function get_property_cached(name, def)
if property_cache[name] ~= nil then
return property_cache[name]
end
return def
end
local function get_font()
if not has_completions then
return
end
if opts.monospace_font ~= "" then
return opts.monospace_font
end
-- Pick a better default font for Windows and macOS
if platform == "windows" then
return "Consolas"
end
if platform == "darwin" then
return "Menlo"
end
return "monospace"
end
local function get_margin_x()
return opts.margin_x > -1 and opts.margin_x or mp.get_property_native("osd-margin-x")
end
local function get_margin_y()
return opts.margin_y > -1 and opts.margin_y or mp.get_property_native("osd-margin-y")
end
-- Naive helper function to find the next UTF-8 character in "str" after "pos"
-- by skipping continuation bytes. Assumes "str" contains valid UTF-8.
local function next_utf8(str, pos)
if pos > str:len() then return pos end
repeat
pos = pos + 1
until pos > str:len() or str:byte(pos) < 0x80 or str:byte(pos) > 0xbf
return pos
end
-- As above, but finds the previous UTF-8 character in "str" before "pos"
local function prev_utf8(str, pos)
if pos <= 1 then return pos end
repeat
pos = pos - 1
until pos <= 1 or str:byte(pos) < 0x80 or str:byte(pos) > 0xbf
return pos
end
local function len_utf8(str)
local len = 0
local pos = 1
while pos <= str:len() do
pos = next_utf8(str, pos)
len = len + 1
end
return len
end
local function utf8_positions(str)
local pos = 1
local positions = {true}
while pos <= str:len() do
pos = next_utf8(str, pos)
positions[pos] = true
end
return positions
end
-- Functions to calculate the font width.
local width_length_ratio = 0.5
local osd_width, osd_height = 0, 0
local text_osd = mp.create_osd_overlay("ass-events")
text_osd.compute_bounds, text_osd.hidden = true, true
local function measure_bounds(ass_text)
text_osd.res_x, text_osd.res_y = osd_width, osd_height
text_osd.data = ass_text
local res = text_osd:update()
return res.x0, res.y0, res.x1, res.y1
end
---Measure text width and normalize to a font size of 1
---text has to be ass safe
local function normalized_text_width(text, size, horizontal)
local align, rotation = horizontal and 7 or 1, horizontal and 0 or -90
local template = "{\\pos(0,0)\\rDefault\\blur0\\bord0\\shad0\\q2\\an%s\\fs%s\\fn%s\\frz%s}%s"
size = size / 0.8
local width
-- Limit to 5 iterations
local repetitions_left = 5
for i = 1, repetitions_left do
size = size * 0.8
local ass = assdraw.ass_new()
ass.text = template:format(align, size, get_font(), rotation, text)
local _, _, x1, y1 = measure_bounds(ass.text)
-- Check if nothing got clipped
if x1 and x1 < osd_width and y1 < osd_height then
width = horizontal and x1 or y1
break
end
if i == repetitions_left then
width = 0
end
end
return width / size, horizontal and osd_width or osd_height
end
local function fit_on_osd(text)
local estimated_width = #text * width_length_ratio
if osd_width >= osd_height then
-- Fill the osd as much as possible, bigger is more accurate.
return math.min(osd_width / estimated_width, osd_height), true
else
return math.min(osd_height / estimated_width, osd_width), false
end
end
local measured_font_hw_ratio = nil
local function get_font_hw_ratio()
local font_hw_ratio = tonumber(opts.font_hw_ratio)
if font_hw_ratio then
return font_hw_ratio
end
if not measured_font_hw_ratio then
local alphabet = "abcdefghijklmnopqrstuvwxyz"
local text = alphabet:rep(3)
local size, horizontal = fit_on_osd(text)
local normalized_width = normalized_text_width(text, size * 0.9, horizontal)
measured_font_hw_ratio = #text / normalized_width * 0.95
end
return measured_font_hw_ratio
end
-- Escape a string for verbatim display on the OSD
local function ass_escape(str)
return mp.command_native({"escape-ass", str})
end
local function should_scale()
return opts.scale_with_window == "yes" or
(opts.scale_with_window == "auto" and mp.get_property_native("osd-scale-by-window"))
end
local function scale_factor()
if should_scale() and osd_height > 0 then
return osd_height / 720
end
return get_property_cached("display-hidpi-scale", 1)
end
local function terminal_output()
-- Unlike vo-configured, current-vo doesn't become falsy while switching VO,
-- which would print the log to the OSD.
return not mp.get_property("current-vo") or not mp.get_property_native("video-osd")
end
local function get_scaled_osd_dimensions()
local scale = scale_factor()
return osd_width / scale, osd_height / scale
end
local function get_line_height()
if selectable_items then
return opts.font_size * (1 + opts.gap)
end
return opts.font_size
end
local function calculate_max_lines()
if terminal_output() then
-- Subtract 1 for the input line and for each line in the status line.
-- This does not detect wrapped lines.
return mp.get_property_native("term-size/h", 24) - 2 -
select(2, mp.get_property("term-status-msg"):gsub("\\n", ""))
end
return math.floor((select(2, get_scaled_osd_dimensions())
* (1 - global_margins.t - global_margins.b)
- get_margin_y() - (selectable_items and opts.padding * 2 or 0))
/ get_line_height()
-- Subtract 1 for the input line and 0.5 for the empty
-- line between the log and the input line.
- 1.5)
end
local function calculate_max_item_width()
if not selectable_items or terminal_output() then
return
end
local longest_item = prompt .. ("a"):rep(9)
for _, item in pairs(selectable_items) do
if #item > #longest_item then
longest_item = item
end
end
local osd_w, osd_h = get_scaled_osd_dimensions()
local font = get_font()
width_overlay.res_x = osd_w
width_overlay.res_y = osd_h
width_overlay.data = "{\\fs" .. opts.font_size ..
(font and "\\fn" .. font or "") .. "\\q2}" ..
ass_escape(longest_item)
local result = width_overlay:update()
if result.x0 then
max_item_width = math.min(result.x1 - result.x0,
osd_w - get_margin_x() * 2 - opts.padding * 2)
end
end
local function should_highlight_completion(i)
return i == selected_completion_index or
(i == 1 and selected_completion_index == 0 and autoselect_completion)
end
local function mpv_color_to_ass(color)
return color:sub(8,9) .. color:sub(6,7) .. color:sub(4,5),
string.format("%x", 255 - tonumber("0x" .. color:sub(2,3)))
end
local function color_option_to_ass(color)
return color:sub(6,7) .. color:sub(4,5) .. color:sub(2,3)
end
local function get_selected_ass()
local color, alpha = mpv_color_to_ass(mp.get_property("osd-selected-color"))
local outline_color, outline_alpha =
mpv_color_to_ass(mp.get_property("osd-selected-outline-color"))
return "{\\b1\\1c&H" .. color .. "&\\1a&H" .. alpha ..
"&\\3c&H" .. outline_color .. "&\\3a&H" .. outline_alpha .. "&}"
end
-- Takes a list of strings, a max width in characters and
-- optionally a max row count.
-- The result contains at least one column.
-- Rows are cut off from the top if rows_max is specified.
-- returns a string containing the formatted table and the row count
local function format_grid(list, width_max, rows_max)
if #list == 0 then
return "", 0
end
local spaces_min = 2
local spaces_max = 8
local list_size = #list
local column_count = 1
local row_count = list_size
local column_widths
-- total width without spacing
local width_total = 0
local list_widths = {}
for i, item in ipairs(list) do
list_widths[i] = len_utf8(item)
end
-- use as many columns as possible
for columns = 2, list_size do
local rows_lower_bound = math.min(rows_max, math.ceil(list_size / columns))
local rows_upper_bound = math.min(rows_max, list_size,
math.ceil(list_size / (columns - 1) - 1))
for rows = rows_upper_bound, rows_lower_bound, -1 do
local cw = {}
width_total = 0
-- find out width of each column
for column = 1, columns do
local width = 0
for row = 1, rows do
local i = row + (column - 1) * rows
local item_width = list_widths[i]
if not item_width then break end
if width < item_width then
width = item_width
end
end
cw[column] = width
width_total = width_total + width
if width_total + (columns - 1) * spaces_min > width_max then
break
end
end
if width_total + (columns - 1) * spaces_min <= width_max then
row_count = rows
column_count = columns
column_widths = cw
else
break
end
end
if width_total + (columns - 1) * spaces_min > width_max then
break
end
end
local spaces = math.floor((width_max - width_total) / (column_count - 1))
spaces = math.max(spaces_min, math.min(spaces_max, spaces))
local spacing = column_count > 1
and ass_escape(string.format("%" .. spaces .. "s", " "))
or ""
local rows = {}
for row = 1, row_count do
local columns = {}
for column = 1, column_count do
local i = row + (column - 1) * row_count
if i > #list then break end
-- more then 99 leads to "invalid format (width or precision too long)"
local format_string = column == column_count and "%s"
or "%-" .. math.min(column_widths[column], 99) .. "s"
columns[column] = ass_escape(string.format(format_string, list[i]))
if should_highlight_completion(i) then
columns[column] = get_selected_ass() .. columns[column] ..
"{\\b\\1a&\\3a&}" .. styles.completion
end
end
-- first row is at the bottom
rows[row_count - row + 1] = table.concat(columns, spacing)
end
return table.concat(rows, ass_escape("\n")), row_count
end
local function fuzzy_find(needle, haystacks)
local result = require "mp.fzy".filter(needle, haystacks)
table.sort(result, function (i, j)
if i[3] ~= j[3] then
return i[3] > j[3]
end
return i[1] < j[1]
end)
return result
end
local function find_matches(needle, haystacks)
if not opts.exact_match and needle:sub(1, 1) ~= "'" then
return fuzzy_find(needle, haystacks)
end
if not opts.exact_match then
needle = needle:sub(2)
end
if not opts.case_sensitive then
needle = needle:lower()
end
local result = {}
local needle_words = {}
for word in needle:gmatch("%S+") do
needle_words[#needle_words + 1] = word
end
for i, haystack in ipairs(haystacks) do
if not opts.case_sensitive then
haystack = haystack:lower()
end
local matching_positions = {}
for _, word in pairs(needle_words) do
local start, e = haystack:find(word, 1, true)
if start then
for j = start, e do
matching_positions[#matching_positions + 1] = j
end
else
matching_positions = nil
break
end
end
if matching_positions then
table.sort(matching_positions)
result[#result + 1] = { i, matching_positions }
end
end
return result
end
local function get_matches_to_print(terminal)
if not selectable_items or focused_match == 0 then
return {}
end
local items = {}
local max_lines = calculate_max_lines()
local escape = terminal and function (str) return str end or ass_escape
local highlight = terminal and terminal_styles.matched_position or
"{\\1c&H" .. color_option_to_ass(opts.match_color) .. "}"
if focused_match < first_match_to_print then
first_match_to_print = focused_match
elseif focused_match > first_match_to_print + max_lines - 1 then
first_match_to_print = focused_match - max_lines + 1
end
local last_match_to_print = math.min(first_match_to_print + max_lines - 1,
#matches)
if last_match_to_print - first_match_to_print + 1 < math.min(max_lines, #matches) and
last_match_to_print >= math.min(max_lines, #matches) then
first_match_to_print = last_match_to_print - math.min(max_lines, #matches) + 1
end
for i = first_match_to_print, last_match_to_print do
local item = ""
local end_highlight = terminal and terminal_styles.match_end or "{\\1c}"
if terminal then
if matches[i].index == default_item then
item = terminal_styles.default_item
end
if i == focused_match then
item = item .. terminal_styles.selected_completion
end
else
if i == focused_match then
if searching_history and
mp.get_property("osd-border-style") == "outline-and-shadow" then
item = get_selected_ass()
else
item = "{\\1c&H" .. color_option_to_ass(opts.focused_color) .. "&}"
end
end_highlight = item
end
end
local char_positions = utf8_positions(matches[i].text)
local start_of_last_match = matches[i].positions[1]
if not start_of_last_match then
item = item .. escape(matches[i].text)
elseif start_of_last_match > 1 then
item = item .. escape(matches[i].text:sub(1, start_of_last_match - 1))
end
for j, pos in ipairs(matches[i].positions) do
local last_pos = matches[i].positions[j - 1]
if last_pos and pos > last_pos + 1 then
if char_positions[start_of_last_match] and char_positions[last_pos + 1] then
item = item .. highlight ..
escape(matches[i].text:sub(start_of_last_match, last_pos)) ..
end_highlight ..
escape(matches[i].text:sub(last_pos + 1, pos - 1))
else
item = item .. escape(matches[i].text:sub(start_of_last_match, pos - 1))
end
start_of_last_match = pos
end
end
if start_of_last_match then
local last_pos = matches[i].positions[#matches[i].positions]
if char_positions[start_of_last_match] and char_positions[last_pos + 1] then
item = item .. highlight ..
escape(matches[i].text:sub(start_of_last_match, last_pos)) ..
end_highlight ..
escape(matches[i].text:sub(last_pos + 1))
else
item = item .. escape(matches[i].text:sub(start_of_last_match))
end
end
items[#items + 1] = item
end
return items
end
local function update_overlay(data, res_x, res_y, z)
if overlay.data == data and
overlay.res_x == res_x and
overlay.res_y == res_y and
overlay.z == z then
return
end
overlay.data = data
overlay.res_x = res_x
overlay.res_y = res_y
overlay.z = z
overlay:update()
end
local function print_to_terminal()
-- Clear the log after closing the console.
if not open then
if osd_msg_active then
mp.osd_message("")
end
osd_msg_active = false
return
end
local log = ""
local counter = ""
if selectable_items then
if #selectable_items > calculate_max_lines() then
local digits = math.ceil(math.log(#selectable_items, 10))
counter = terminal_styles.disabled ..
"[" .. string.format("%0" .. digits .. "d", focused_match) ..
"/" .. string.format("%0" .. digits .. "d", #matches) ..
"]\027[0m "
end
local clip = mp.get_property("term-clip-cc")
for _, item in ipairs(get_matches_to_print(true)) do
log = log .. clip .. item .. "\027[0m\n"
end
else
for _, log_line in ipairs(log_buffers[id]) do
log = log .. log_line.terminal_style .. log_line.text .. "\027[0m\n"
end
for i, completion in ipairs(completion_buffer) do
if should_highlight_completion(i) then
log = log .. terminal_styles.selected_completion ..
completion .. "\027[0m"
else
log = log .. completion
end
log = log .. (i < #completion_buffer and "\t" or "\n")
end
end
local before_cur = line:sub(1, cursor - 1)
local after_cur = line:sub(cursor)
-- Ensure there is a character with inverted colors to print.
if after_cur == "" then
after_cur = " "
end
mp.osd_message(log .. counter .. prompt .. " " .. before_cur .. "\027[7m" ..
after_cur:sub(1, 1) .. "\027[0m" .. after_cur:sub(2), 999)
osd_msg_active = true
end
local function render()
pending_update = false
if terminal_output() then
print_to_terminal()
return
end
-- Clear the OSD if the console was being printed to the terminal
if osd_msg_active then
mp.osd_message("")
osd_msg_active = false
end
-- Clear the OSD after closing the console
if not open then
update_overlay("", 0, 0, 0)
return
end
local ass = assdraw.ass_new()
local osd_w, osd_h = get_scaled_osd_dimensions()
local line_height = get_line_height()
local max_lines = calculate_max_lines()
local x, y, alignment, clipping_coordinates
if selectable_items and not searching_history then
x = (osd_w - max_item_width) / 2
y = osd_h *
(global_margins.t + (1 - global_margins.t - global_margins.b) / 2) -
(math.min(#selectable_items, max_lines) + 1.5) * line_height / 2
alignment = 7
clipping_coordinates = x .. ",0," .. x + max_item_width .. "," .. osd_h
else
x = get_margin_x()
y = osd_h * (1 - global_margins.b) - get_margin_y()
alignment = 1
-- Avoid drawing below topbar OSC when there are wrapped lines.
local coordinate_top = math.floor(global_margins.t * osd_h + 0.5)
clipping_coordinates = "0," .. coordinate_top .. "," .. osd_w .. "," .. osd_h
end
local font = get_font()
-- Use the same blur value as the rest of the OSD. 288 is the OSD's
-- PlayResY.
local blur = mp.get_property_native("osd-blur") * osd_h / 288
local border_style = mp.get_property("osd-border-style")
local style = "{\\r" ..
(font and "\\fn" .. font or "") ..
"\\fs" .. opts.font_size ..
"\\bord" .. opts.border_size .. "\\fsp0" ..
"\\blur" .. blur ..
(selectable_items and "\\q2" or "\\q1") ..
"\\clip(" .. clipping_coordinates .. ")}"
-- Create the cursor glyph as an ASS drawing. ASS will draw the cursor
-- inline with the surrounding text, but it sets the advance to the width
-- of the drawing. So the cursor doesn't affect layout too much, make it as
-- thin as possible and make it appear to be 1px wide by giving it 0.5px
-- horizontal borders.
local color, alpha = mpv_color_to_ass(mp.get_property("osd-color"))
local cheight = opts.font_size * 8
local cglyph = "{\\r\\blur0" ..
(get_property_cached("focused") == false
and "\\alpha&HFF&" or "\\3a&H" .. alpha .. "&") ..
"\\3c&H" .. color .. "&" ..
"\\xbord0.5\\ybord0\\xshad0\\yshad1\\p4\\pbo24}" ..
"m 0 0 l 1 0 l 1 " .. cheight .. " l 0 " .. cheight ..
"{\\p0}"
local before_cur = ass_escape(line:sub(1, cursor - 1))
local after_cur = ass_escape(line:sub(cursor))
local log_ass = ""
local log_buffer = log_buffers[id] or {}
log_offset = math.max(math.min(log_offset, #log_buffer - 1), 0)
local last = #log_buffer - log_offset
local first = math.max(last - math.min(max_lines, #log_buffer) + 1, 1)
for i = first, last do
log_ass = log_ass .. style .. log_buffer[i].style ..
ass_escape(log_buffer[i].text) .. "\\N"
end
local completion_ass = ""
if completion_buffer[1] and not selectable_items then
-- Estimate how many characters fit in one line
-- Even with bottom-left anchoring,
-- libass/ass_render.c:ass_render_event() subtracts --osd-margin-x from
-- the maximum text width twice.
-- TODO: --osd-margin-x should scale with osd-width and PlayResX to make
-- the calculation accurate.
local width_max = math.floor(
(osd_w - x - mp.get_property_native("osd-margin-x") * 2)
/ opts.font_size * get_font_hw_ratio())
local completions, rows = format_grid(completion_buffer, width_max, max_lines)
max_lines = max_lines - rows
completion_ass = style .. styles.completion .. completions .. "\\N"
end
-- Background
if selectable_items and
(not searching_history or border_style == "background-box") then
style = style .. "{\\bord0\\blur0\\4a&Hff&}"
local back_color, back_alpha = mpv_color_to_ass(mp.get_property(
border_style == "background-box" and "osd-back-color" or "osd-outline-color"))
if not searching_history then
back_alpha = string.format("%x", opts.background_alpha)
end
ass:new_event()
ass:an(alignment)
ass:pos(x, y)
ass:append("{\\1c&H" .. back_color .. "&\\1a&H" .. back_alpha ..
"&\\bord" .. opts.menu_outline_size .. "\\3c&H" ..
color_option_to_ass(opts.menu_outline_color) .. "\\blur0&}")
if border_style == "background-box" then
ass:append("{\\4a&Hff&}")
end
ass:draw_start()
ass:round_rect_cw(-opts.padding,
opts.padding * (alignment == 7 and -1 or 1),
max_item_width + opts.padding,
(1.5 + math.min(#matches, max_lines)) * line_height +
opts.padding * (alignment == 7 and 1 or 2),
opts.corner_radius, opts.corner_radius)
ass:draw_stop()
end
local items = get_matches_to_print()
item_positions = {}
for i, item in ipairs(items) do
local item_y = alignment == 7
and y + (1 + i) * line_height
or y - (1.5 + #items - i) * line_height
if (first_match_to_print - 1 + i == focused_match or
matches[first_match_to_print - 1 + i].index == default_item)
and (not searching_history or border_style == "background-box") then
ass:new_event()
ass:an(4)
ass:pos(x, item_y)
ass:append("{\\blur0\\bord0\\4aH&ff&\\1c&H" ..
color_option_to_ass(opts.focused_back_color) .. "&}")
if first_match_to_print - 1 + i ~= focused_match then
ass:append("{\\1aH&cc&}")
end
ass:draw_start()
ass:rect_cw(-opts.padding, 0, max_item_width + opts.padding, line_height)
ass:draw_stop()
end
ass:new_event()
ass:an(4)
ass:pos(x - horizontal_offset, item_y)
ass:append(style .. item)
item_positions[#item_positions + 1] =
{ item_y - line_height / 2, item_y + line_height / 2 }
end
-- Scrollbar
if selectable_items and #matches > max_lines then
ass:new_event()
ass:an(alignment + 2)
ass:pos(x + max_item_width, y)
ass:append(style)
if not searching_history or border_style == "background-box" then
ass:append("{\\bord0\\4a&Hff&\\blur0}")
end
ass:append(focused_match .. "/" .. #matches)
local start_percentage = (first_match_to_print - 1) / #matches
local end_percentage = (first_match_to_print - 1 + max_lines) / #matches
if end_percentage - start_percentage < 0.04 then
local diff = 0.04 - (end_percentage - start_percentage)
start_percentage = start_percentage * (1 - diff)
end_percentage = end_percentage + diff * (1 - end_percentage)
end
local max_height = max_lines * line_height
local bar_y = alignment == 7
and y + 1.5 * line_height + start_percentage * max_height
or y - 1.5 * line_height - max_height * (1 - end_percentage)
local height = max_height * (end_percentage - start_percentage)
ass:new_event()
ass:an(alignment)
ass:append("{\\blur0\\4a&Hff&\\bord1}")
ass:pos(x + max_item_width + opts.padding - 1, bar_y)
ass:draw_start()
ass:rect_cw(0, 0, -opts.padding / 2, height)
ass:draw_stop()
end
ass:new_event()
ass:an(alignment)
ass:pos(x, y)
if not selectable_items then
ass:append(log_ass .. "\\N" .. completion_ass)
end
ass:append(style .. ass_escape(prompt) .. " " .. before_cur)
ass:append(cglyph)
ass:append(style .. after_cur)
-- Redraw the cursor with the input text invisible. This will make the
-- cursor appear in front of the text.
ass:new_event()
ass:an(alignment)
ass:pos(x, y)
ass:append(style .. "{\\alpha&HFF&}" .. ass_escape(prompt) .. " " .. before_cur)
ass:append(cglyph)
ass:append(style .. "{\\alpha&HFF&}" .. after_cur)
-- z with selectable_items needs to be greater than the OSC's.
update_overlay(ass.text, osd_w, osd_h, selectable_items and 2000 or 0)
end
local update_timer = nil
update_timer = mp.add_periodic_timer(0.05, function()
if pending_update then
render()
else
update_timer:kill()
end
end)
update_timer:kill()
-- Add a line to the history and deduplicate
local function history_add(text)
if history[#history] == text or text == "" then
return
end
if opts.history_dedup then
-- More recent entries are more likely to be repeated
for i = #history, 1, -1 do
if history[i] == text then
table.remove(history, i)
break
end
end
end
history[#history + 1] = text
if history_paths[id] then
histories_to_save[id] = histories_to_save[id] .. text .. "\n"
end
end
local function handle_cursor_move()
-- Don't show completions after a command is entered because they move its
-- output up, and allow clearing completions by emptying the line.
if line == "" then
completion_buffer = {}
render()
else
complete()
end
end
local function handle_edit()
if not selectable_items then
handle_cursor_move()
mp.commandv("script-message-to", input_caller, input_caller_handler, "edited",
utils.format_json({line}))
return
end
matches = {}
for i, match in ipairs(find_matches(line, selectable_items)) do
matches[i] = {
index = match[1],
text = selectable_items[match[1]],
positions = match[2],
}
end
if line == "" and default_item then
focused_match = default_item
local max_lines = calculate_max_lines()
first_match_to_print = math.max(1, focused_match + 1 - math.ceil(max_lines / 2))
if first_match_to_print > #selectable_items - max_lines + 1 then
first_match_to_print = math.max(1, #selectable_items - max_lines + 1)
end
else
focused_match = 1
end
render()
end
-- Insert a character at the current cursor position (any_unicode)
local function handle_char_input(c)
if insert_mode then
line = line:sub(1, cursor - 1) .. c .. line:sub(next_utf8(line, cursor))
else
line = line:sub(1, cursor - 1) .. c .. line:sub(cursor)
end
cursor = cursor + #c
handle_edit()
end
-- Remove the character behind the cursor (Backspace)
local function handle_backspace()
if cursor <= 1 then return end
local prev = prev_utf8(line, cursor)
line = line:sub(1, prev - 1) .. line:sub(cursor)
cursor = prev
handle_edit()
end
-- Remove the character in front of the cursor (Del)
local function handle_del()
if cursor > line:len() then return end
line = line:sub(1, cursor - 1) .. line:sub(next_utf8(line, cursor))
handle_edit()
end
-- Toggle insert mode (Ins)
local function handle_ins()
insert_mode = not insert_mode
end
-- Move the cursor to the next character (Right)
local function next_char()
cursor = next_utf8(line, cursor)
handle_cursor_move()
end
-- Move the cursor to the previous character (Left)
local function prev_char()
cursor = prev_utf8(line, cursor)
handle_cursor_move()
end
-- Clear the current line (Ctrl+C)
local function clear()
line = ""
cursor = 1
insert_mode = false
history_pos = #history + 1
handle_edit()
end
-- Close the console if the current line is empty, otherwise delete the next
-- character (Ctrl+D)
local function maybe_exit()
if line == "" then
set_active(false)
else
handle_del()
end
end
local function unbind_mouse()
mp.remove_key_binding("_console_mouse_move")
mp.remove_key_binding("_console_mbtn_left")
end
local function after_cur_matches_completion()
local token = line:sub(completion_pos)
for _, completion in pairs(completion_buffer) do
if completion == token:sub(1, #completion) then
return true
end
end
end
-- Run the current command or select the current item
local function submit()
if searching_history then
searching_history = false
selectable_items = nil
line = #matches > 0 and matches[focused_match].text or ""
cursor = #line + 1
handle_edit()
unbind_mouse()
return
end
if selectable_items then
if #matches > 0 then
mp.commandv("script-message-to", input_caller, input_caller_handler, "submit",
utils.format_json({matches[focused_match].index}))
end
else
if selected_completion_index == 0 and autoselect_completion
and not after_cur_matches_completion() then
cycle_through_completions()
end
mp.commandv("script-message-to", input_caller, input_caller_handler, "submit",
utils.format_json({line}))
history_add(line)
end
if not keep_open then
set_active(false)
elseif not selectable_items then
clear()
end
end
local function determine_hovered_item()
local osd_w, _ = get_scaled_osd_dimensions()
local scale = scale_factor()
local mouse_pos = mp.get_property_native("mouse-pos")
local mouse_x = mouse_pos.x / scale
local mouse_y = mouse_pos.y / scale
local item_x0 = (searching_history and get_margin_x() or (osd_w - max_item_width) / 2)
- opts.padding
if mouse_x < item_x0 or mouse_x > item_x0 + max_item_width + opts.padding * 2 then
return
end
for i, positions in ipairs(item_positions) do
if mouse_y >= positions[1] and mouse_y <= positions[2] then
return first_match_to_print - 1 + i
end
end
end
local function bind_mouse()
mp.add_forced_key_binding("MOUSE_MOVE", "_console_mouse_move", function()
local item = determine_hovered_item()
if item and item ~= focused_match then
focused_match = item
render()
end
end)
mp.add_forced_key_binding("MBTN_LEFT", "_console_mbtn_left", function()
local item = determine_hovered_item()
if item then
focused_match = item
submit()
else
set_active(false)
end
end)
end
-- Go to the specified position in the command history
local function go_history(new_pos)
local old_pos = history_pos
history_pos = new_pos
-- Restrict the position to a legal value
if history_pos > #history + 1 then
history_pos = #history + 1
elseif history_pos < 1 then
history_pos = 1
end
-- Do nothing if the history position didn't actually change
if history_pos == old_pos then
return
end
-- If the user was editing a non-history line, save it as the last history
-- entry. This makes it much less frustrating to accidentally hit Up/Down
-- while editing a line.
if old_pos == #history + 1 then
history_add(line)
end
-- Now show the history line (or a blank line for #history + 1)
if history_pos <= #history then
line = history[history_pos]
else
line = ""
end
cursor = line:len() + 1
insert_mode = false
handle_edit()
end
-- Go to the specified relative position in the command history (Up, Down)
local function move_history(amount, is_wheel)
if not selectable_items then
go_history(history_pos + amount)
return
end
if is_wheel then
local max_lines = calculate_max_lines()
-- Update focused_match only if it's the first or last printed item and
-- there are hidden items.
if (amount > 0 and focused_match == first_match_to_print
and first_match_to_print - 1 + max_lines < #matches)
or (amount < 0 and focused_match == first_match_to_print - 1 + max_lines
and first_match_to_print > 1) then
focused_match = focused_match + amount
end
if amount > 0 and first_match_to_print < #matches - max_lines + 1
or amount < 0 and first_match_to_print > 1 then
-- math.min and math.max would only be needed with amounts other than
-- 1 and -1.
first_match_to_print = math.min(
math.max(first_match_to_print + amount, 1), #matches - max_lines + 1)
end
local item = determine_hovered_item()
if item then
focused_match = item
end
render()
return
end
focused_match = focused_match + amount
if focused_match > #matches then
focused_match = 1
elseif focused_match < 1 then
focused_match = #matches
end
render()
end
local function horizontal_scroll(amount)
horizontal_offset = math.max(horizontal_offset + amount, 0)
render()
end
-- Go to the first command in the command history (PgUp)
local function handle_pgup()
if selectable_items then
focused_match = math.max(focused_match - calculate_max_lines() + 1, 1)
render()
return
end
go_history(1)
end
-- Stop browsing history and start editing a blank line (PgDown)
local function handle_pgdown()
if selectable_items then
focused_match = math.min(focused_match + calculate_max_lines() - 1, #matches)
render()
return
end
go_history(#history + 1)
end
local function search_history()
if selectable_items or #history == 0 then
return
end
searching_history = true
selectable_items = {}
horizontal_offset = 0
for i = 1, #history do
selectable_items[i] = history[#history + 1 - i]
end
calculate_max_item_width()
handle_edit()
bind_mouse()
end
local function page_up_or_prev_char()
if selectable_items then
handle_pgup()
else
prev_char()
end
end
local function page_down_or_next_char()
if selectable_items then
handle_pgdown()
else
next_char()
end
end
-- Move to the start of the current word, or if already at the start, the start
-- of the previous word. (Ctrl+Left)
local function prev_word()
-- This is basically the same as next_word() but backwards, so reverse the
-- string in order to do a "backwards" find. This wouldn't be as annoying
-- to do if Lua didn't insist on 1-based indexing.
cursor = line:len() - select(2, line:reverse():find("%s*[^%s]*", line:len() - cursor + 2)) + 1
handle_cursor_move()
end
-- Move to the end of the current word, or if already at the end, the end of
-- the next word. (Ctrl+Right)
local function next_word()
cursor = select(2, line:find("%s*[^%s]*", cursor)) + 1
handle_cursor_move()
end
-- Move the cursor to the beginning of the line (HOME)
local function go_home()
cursor = 1
handle_cursor_move()
end
-- Move the cursor to the end of the line (END)
local function go_end()
cursor = line:len() + 1
handle_cursor_move()
end
-- Delete from the cursor to the beginning of the word (Ctrl+Backspace)
local function del_word()
local before_cur = line:sub(1, cursor - 1)
local after_cur = line:sub(cursor)
before_cur = before_cur:gsub("[^%s]+%s*$", "", 1)
line = before_cur .. after_cur
cursor = before_cur:len() + 1
handle_edit()
end
-- Delete from the cursor to the end of the word (Ctrl+Del)
local function del_next_word()
if cursor > line:len() then return end
local before_cur = line:sub(1, cursor - 1)
local after_cur = line:sub(cursor)
after_cur = after_cur:gsub("^%s*[^%s]+", "", 1)
line = before_cur .. after_cur
handle_edit()
end
-- Delete from the cursor to the end of the line (Ctrl+K)
local function del_to_eol()
line = line:sub(1, cursor - 1)
handle_edit()
end
-- Delete from the cursor back to the start of the line (Ctrl+U)
local function del_to_start()
line = line:sub(cursor)
cursor = 1
handle_edit()
end
-- Empty the log buffer of all messages (Ctrl+L)
local function clear_log_buffer()
if not selectable_items then
log_buffers[id] = {}
end
render()
end
local function scroll_log(amount)
if selectable_items then
return
end
log_offset = log_offset + amount
render()
end
-- Returns a string of UTF-8 text from the clipboard (or the primary selection)
local function get_clipboard(clip)
if platform == "x11" then
local property = clip and "clipboard/text" or "clipboard/text-primary"
return mp.get_property(property, "")
elseif platform == "wayland" then
if mp.get_property("current-clipboard-backend") == "wayland" then
local property = clip and "clipboard/text" or "clipboard/text-primary"
return mp.get_property(property, "")
end
-- Wayland VO clipboard is only updated on window focus
if clip and get_property_cached("focused") then
return mp.get_property("clipboard/text", "")
end
local res = utils.subprocess({
args = { "wl-paste", clip and "-n" or "-np" },
playback_only = false,
})
if not res.error then
return res.stdout
end
elseif platform == "windows" or platform == "darwin" then
return mp.get_property("clipboard/text", "")
end
return ""
end
-- Paste text from the window-system's clipboard. "clip" determines whether the
-- clipboard or the primary selection buffer is used (on X11 and Wayland only.)
local function paste(clip)
local text = get_clipboard(clip)
local before_cur = line:sub(1, cursor - 1)
local after_cur = line:sub(cursor)
line = before_cur .. text .. after_cur
cursor = cursor + text:len()
handle_edit()
end
local function copy()
if not selectable_items then
mp.set_property("clipboard/text", line)
mp.msg.info("Input line copied")
elseif matches[1] then
mp.set_property("clipboard/text", matches[focused_match].text)
if terminal_output() then
mp.msg.info("Item copied")
else
mp.osd_message("Item copied")
end
end
end
local function text_input(info)
if info.key_text and (info.event == "press" or info.event == "down"
or info.event == "repeat")
then
handle_char_input(info.key_text)
end
end
local function common_prefix_length(s1, s2)
local common_count = 0
for i = 1, #s1 do
if s1:byte(i) ~= s2:byte(i) then
break
end
common_count = common_count + 1
end
return common_count
end
local function max_overlap_length(s1, s2)
for s1_offset = 0, #s1 - 1 do
local match = true
for i = 1, #s1 - s1_offset do
if s1:byte(s1_offset + i) ~= s2:byte(i) then
match = false
break
end
end
if match then
return #s1 - s1_offset
end
end
return 0
end
-- If str starts with the first or last characters of prefix, strip them.
local function strip_common_characters(str, prefix)
return str:sub(1 + math.max(
common_prefix_length(prefix, str),
max_overlap_length(prefix, str)))
end
cycle_through_completions = function (backwards)
if #completion_buffer == 0 then
-- Allow Tab completion of commands before typing anything.
if line == "" then
complete()
end
return
end
selected_completion_index = selected_completion_index + (backwards and -1 or 1)
if selected_completion_index > #completion_buffer then
selected_completion_index = 1
elseif selected_completion_index < 1 then
selected_completion_index = #completion_buffer
end
local before_cur = line:sub(1, completion_pos - 1) ..
completion_buffer[selected_completion_index] .. completion_append
line = before_cur .. strip_common_characters(line:sub(cursor),
completion_buffer[selected_completion_index] .. completion_append)
cursor = before_cur:len() + 1
render()
end
-- Show autocompletions.
complete = function ()
completion_old_line = line
completion_old_cursor = cursor
mp.commandv("script-message-to", input_caller, input_caller_handler,
"complete", utils.format_json({line:sub(1, cursor - 1)}))
render()
end
-- List of input bindings. This is a weird mashup between common GUI text-input
-- bindings and readline bindings.
local function get_bindings()
local bindings = {
{ "esc", function() set_active(false) end },
{ "ctrl+[", function() set_active(false) end },
{ "enter", submit },
{ "kp_enter", submit },
{ "shift+enter", function() handle_char_input("\n") end },
{ "ctrl+j", submit },
{ "ctrl+m", submit },
{ "bs", handle_backspace },
{ "shift+bs", handle_backspace },
{ "ctrl+h", handle_backspace },
{ "del", handle_del },
{ "shift+del", handle_del },
{ "ins", handle_ins },
{ "shift+ins", function() paste(false) end },
{ "mbtn_mid", function() paste(false) end },
{ "left", function() prev_char() end },
{ "ctrl+b", function() page_up_or_prev_char() end },
{ "right", function() next_char() end },
{ "ctrl+f", function() page_down_or_next_char() end},
{ "up", function() move_history(-1) end },
{ "ctrl+p", function() move_history(-1) end },
{ "wheel_up", function() move_history(-1, true) end },
{ "down", function() move_history(1) end },
{ "ctrl+n", function() move_history(1) end },
{ "wheel_down", function() move_history(1, true) end },
{ "shift+up", function() scroll_log(1) end },
{ "shift+down", function() scroll_log(-1) end },
{ "wheel_left", function() end },
{ "wheel_right", function() end },
{ "shift+left", function() horizontal_scroll(-25) end },
{ "shift+right", function() horizontal_scroll( 25) end },
{ "wheel_left", function() horizontal_scroll(-25) end },
{ "wheel_right", function() horizontal_scroll( 25) end },
{ "shift+wheel_up", function() horizontal_scroll(-25) end },
{ "shift+wheel_down", function() horizontal_scroll( 25) end },
{ "ctrl+left", prev_word },
{ "alt+b", prev_word },
{ "ctrl+right", next_word },
{ "alt+f", next_word },
{ "tab", cycle_through_completions },
{ "ctrl+i", cycle_through_completions },
{ "shift+tab", function() cycle_through_completions(true) end },
{ "ctrl+a", go_home },
{ "home", go_home },
{ "ctrl+e", go_end },
{ "end", go_end },
{ "pgup", handle_pgup },
{ "pgdwn", handle_pgdown },
{ "ctrl+r", search_history },
{ "ctrl+c", clear },
{ "ctrl+d", maybe_exit },
{ "ctrl+k", del_to_eol },
{ "ctrl+l", clear_log_buffer },
{ "ctrl+u", del_to_start },
{ "ctrl+y", copy, },
{ "ctrl+v", function() paste(true) end },
{ "meta+v", function() paste(true) end },
{ "ctrl+bs", del_word },
{ "ctrl+w", del_word },
{ "ctrl+del", del_next_word },
{ "alt+d", del_next_word },
{ "kp_dec", function() handle_char_input(".") end },
{ "kp_add", function() handle_char_input("+") end },
{ "kp_subtract", function() handle_char_input("-") end },
{ "kp_multiply", function() handle_char_input("*") end },
{ "kp_divide", function() handle_char_input("/") end },
}
for i = 0, 9 do
bindings[#bindings + 1] =
{"kp" .. i, function() handle_char_input("" .. i) end}
end
return bindings
end
local function define_key_bindings()
if #key_bindings > 0 then
return
end
for _, bind in ipairs(get_bindings()) do
if not (dont_bind_up_down and (bind[1] == "up" or bind[1] == "down")) then
-- Generate arbitrary name for removing the bindings later.
local name = "_console_" .. (#key_bindings + 1)
key_bindings[#key_bindings + 1] = name
mp.add_forced_key_binding(bind[1], name, bind[2], {repeatable = true})
end
end
mp.add_forced_key_binding("any_unicode", "_console_text", text_input,
{repeatable = true, complex = true})
key_bindings[#key_bindings + 1] = "_console_text"
end
local function undefine_key_bindings()
for _, name in ipairs(key_bindings) do
mp.remove_key_binding(name)
end
key_bindings = {}
end
local function read_history()
if not history_paths[id] or history[1] then
return
end
local history_file = io.open(mp.command_native({"expand-path", history_paths[id]}))
if history_file == nil then
return
end
if opts.history_dedup then
local unfiltered_history = {}
for command in history_file:lines() do
unfiltered_history[#unfiltered_history + 1] = command
end
local history_map = {}
for i = #unfiltered_history, 1, -1 do
local command = unfiltered_history[i]
if not history_map[command] then
history[#history + 1] = command
history_map[command] = true
end
end
for i = 1, #history / 2, 1 do
history[i], history[#history - i + 1] = history[#history - i + 1], history[i]
end
else
for command in history_file:lines() do
history[#history + 1] = command
end
end
history_file:close()
end
-- Open or close the console
set_active = function (active)
if active == open then
return
end
if active then
open = true
insert_mode = false
define_key_bindings()
mp.set_property_bool("user-data/mpv/console/open", true)
ime_active = mp.get_property_bool("input-ime")
mp.set_property_bool("input-ime", true)
elseif searching_history then
searching_history = false
selectable_items = nil
unbind_mouse()
else
open = false
undefine_key_bindings()
unbind_mouse()
mp.set_property_bool("user-data/mpv/console/open", false)
mp.set_property_bool("input-ime", ime_active)
mp.commandv("script-message-to", input_caller, input_caller_handler,
"closed", utils.format_json({line, cursor}))
collectgarbage()
end
render()
end
mp.register_script_message("disable", function(message)
message = utils.parse_json(message or "")
if not message or message.client_name == input_caller then
set_active(false)
end
end)
mp.register_script_message("get-input", function (args)
if open then
mp.commandv("script-message-to", input_caller, input_caller_handler,
"closed", utils.format_json({line, cursor}))
end
args = utils.parse_json(args)
input_caller = args.client_name
input_caller_handler = args.handler_id
prompt = args.prompt or ""
line = args.default_text or ""
cursor = tonumber(args.cursor_position) or line:len() + 1
keep_open = args.keep_open
default_item = args.default_item
has_completions = args.has_completions
dont_bind_up_down = args.dont_bind_up_down
searching_history = false
if args.items then
selectable_items = {}
horizontal_offset = 0
-- Limit the number of characters to prevent libass from freezing mpv.
-- Not important for terminal output.
local limit
if osd_width == 0 or terminal_output() then
limit = 5000
else
limit = 5 * osd_width / opts.font_size
end
for i, item in ipairs(args.items) do
local last = next_utf8(item, limit) - 1
selectable_items[i] = item:gsub("[\r\n].*", ""):sub(1, last) ..
(last < #item and "" or "")
end
calculate_max_item_width()
handle_edit()
bind_mouse()
else
selectable_items = nil
unbind_mouse()
id = args.id
log_offset = 0
completion_buffer = {}
autoselect_completion = args.autoselect_completion
if histories[id] == nil then
histories[id] = {}
log_buffers[id] = {}
histories_to_save[id] = ""
end
history = histories[id]
history_paths[id] = args.history_path
read_history()
history_pos = #history + 1
handle_cursor_move()
end
set_active(true)
-- We want to ensure the keybindings have been set before sending the "opened" event
-- in case scripts want to override our bindings.
mp.flush_keybindings()
mp.commandv("script-message-to", input_caller, input_caller_handler, "opened")
end)
-- Add a line to the log buffer
mp.register_script_message("log", function (message)
message = utils.parse_json(message or "")
if not message or not message.log_id then
return
end
local log_buffer = log_buffers[message.log_id]
if not log_buffer then return end
log_buffer[#log_buffer + 1] = {
text = message.text,
style = message.error and styles.error or message.style or "",
terminal_style = message.error and terminal_styles.error or
message.terminal_style or "",
}
if #log_buffer > MAX_LOG_LINES then
table.remove(log_buffer, 1)
end
if not open or message.log_id ~= id then
return
end
if log_offset > 0 then
log_offset = log_offset + 1
end
if not update_timer:is_enabled() then
render()
update_timer:resume()
else
pending_update = true
end
end)
mp.register_script_message("set-log", function (log_id, log)
if not log_id or not log then
return
end
log = utils.parse_json(log)
log_buffers[log_id] = {}
for i = 1, #log do
if type(log[i]) == "table" then
log[i].text = log[i].text
log[i].style = log[i].style or ""
log[i].terminal_style = log[i].terminal_style or ""
log_buffers[id][i] = log[i]
else
log_buffers[id][i] = {
text = log[i],
style = "",
terminal_style = "",
}
end
end
if log_id == id then
render()
end
end)
mp.register_script_message("complete", function (message)
message = utils.parse_json(message)
if message.client_name ~= input_caller or message.handler_id ~= input_caller_handler
or line ~= completion_old_line or cursor ~= completion_old_cursor then
return
end
completion_buffer = {}
selected_completion_index = 0
local completions = message.list
table.sort(completions)
completion_pos = message.start_pos
completion_append = message.append
for i, match in ipairs(fuzzy_find(line:sub(completion_pos, cursor - 1),
completions)) do
completion_buffer[i] = completions[match[1]]
end
render()
end)
local function resize()
calculate_max_item_width()
render()
end
mp.observe_property("osd-dimensions", "native", function (_, value)
if value.w == osd_width and value.h == osd_height then
return
end
osd_width, osd_height = value.w, value.h
resize()
end)
mp.observe_property("display-hidpi-scale", "native", function (name, value)
property_cache[name] = value
resize()
end)
mp.observe_property("focused", "native", function (name, value)
property_cache[name] = value
render()
end)
mp.observe_property("user-data/osc/margins", "native", function(_, val)
if type(val) == "table" and type(val.t) == "number" and type(val.b) == "number" then
global_margins = val
else
global_margins = { t = 0, b = 0 }
end
render()
end)
mp.register_event("shutdown", function ()
mp.del_property("user-data/mpv/console")
for history_id, history_path in pairs(history_paths) do
history_path = mp.command_native({"expand-path", history_path})
local history_file, error_message = io.open(history_path, "ab")
if history_file then
history_file:write(histories_to_save[history_id])
history_file:close()
else
mp.msg.error("Failed to write history: " .. error_message)
end
end
end)
-- These are for backwards compatibility only.
mp.add_key_binding(nil, "enable", function ()
mp.command("script-message-to commands open")
end)
mp.register_script_message("type", function (...)
mp.commandv("script-message-to", "commands", "type", ...)
end)
require "mp.options".read_options(opts, nil, render)
collectgarbage()