summaryrefslogtreecommitdiffstats
path: root/.config/mpv/scripts/uosc_shared/lib/std.lua
diff options
context:
space:
mode:
Diffstat (limited to '.config/mpv/scripts/uosc_shared/lib/std.lua')
-rw-r--r--.config/mpv/scripts/uosc_shared/lib/std.lua181
1 files changed, 181 insertions, 0 deletions
diff --git a/.config/mpv/scripts/uosc_shared/lib/std.lua b/.config/mpv/scripts/uosc_shared/lib/std.lua
new file mode 100644
index 0000000..1261666
--- /dev/null
+++ b/.config/mpv/scripts/uosc_shared/lib/std.lua
@@ -0,0 +1,181 @@
+--[[ Stateless utilities missing in lua standard library ]]
+
+---@param number number
+function round(number) return math.floor(number + 0.5) end
+
+---@param min number
+---@param value number
+---@param max number
+function clamp(min, value, max) return math.max(min, math.min(value, max)) end
+
+---@param rgba string `rrggbb` or `rrggbbaa` hex string.
+function serialize_rgba(rgba)
+ local a = rgba:sub(7, 8)
+ return {
+ color = rgba:sub(5, 6) .. rgba:sub(3, 4) .. rgba:sub(1, 2),
+ opacity = clamp(0, tonumber(#a == 2 and a or 'ff', 16) / 255, 1),
+ }
+end
+
+-- Trim any `char` from the end of the string.
+---@param str string
+---@param char string
+---@return string
+function trim_end(str, char)
+ local char, end_i = char:byte(), 0
+ for i = #str, 1, -1 do
+ if str:byte(i) ~= char then
+ end_i = i
+ break
+ end
+ end
+ return str:sub(1, end_i)
+end
+
+---@param str string
+---@param pattern string
+---@return string[]
+function split(str, pattern)
+ local list = {}
+ local full_pattern = '(.-)' .. pattern
+ local last_end = 1
+ local start_index, end_index, capture = str:find(full_pattern, 1)
+ while start_index do
+ list[#list + 1] = capture
+ last_end = end_index + 1
+ start_index, end_index, capture = str:find(full_pattern, last_end)
+ end
+ if last_end <= (#str + 1) then
+ capture = str:sub(last_end)
+ list[#list + 1] = capture
+ end
+ return list
+end
+
+-- Get index of the last appearance of `sub` in `str`.
+---@param str string
+---@param sub string
+---@return integer|nil
+function string_last_index_of(str, sub)
+ local sub_length = #sub
+ for i = #str, 1, -1 do
+ for j = 1, sub_length do
+ if str:byte(i + j - 1) ~= sub:byte(j) then break end
+ if j == sub_length then return i end
+ end
+ end
+end
+
+---@param itable table
+---@param value any
+---@return integer|nil
+function itable_index_of(itable, value)
+ for index, item in ipairs(itable) do
+ if item == value then return index end
+ end
+end
+
+---@param itable table
+---@param compare fun(value: any, index: number)
+---@param from_end? boolean Search from the end of the table.
+---@return number|nil index
+---@return any|nil value
+function itable_find(itable, compare, from_end)
+ local from, to, step = from_end and #itable or 1, from_end and 1 or #itable, from_end and -1 or 1
+ for index = from, to, step do
+ if compare(itable[index], index) then return index, itable[index] end
+ end
+end
+
+---@param itable table
+---@param decider fun(value: any, index: number)
+function itable_filter(itable, decider)
+ local filtered = {}
+ for index, value in ipairs(itable) do
+ if decider(value, index) then filtered[#filtered + 1] = value end
+ end
+ return filtered
+end
+
+---@param itable table
+---@param value any
+function itable_remove(itable, value)
+ return itable_filter(itable, function(item) return item ~= value end)
+end
+
+---@param itable table
+---@param start_pos? integer
+---@param end_pos? integer
+function itable_slice(itable, start_pos, end_pos)
+ start_pos = start_pos and start_pos or 1
+ end_pos = end_pos and end_pos or #itable
+
+ if end_pos < 0 then end_pos = #itable + end_pos + 1 end
+ if start_pos < 0 then start_pos = #itable + start_pos + 1 end
+
+ local new_table = {}
+ for index, value in ipairs(itable) do
+ if index >= start_pos and index <= end_pos then
+ new_table[#new_table + 1] = value
+ end
+ end
+ return new_table
+end
+
+---@generic T
+---@param a T[]|nil
+---@param b T[]|nil
+---@return T[]
+function itable_join(a, b)
+ local result = {}
+ if a then for _, value in ipairs(a) do result[#result + 1] = value end end
+ if b then for _, value in ipairs(b) do result[#result + 1] = value end end
+ return result
+end
+
+---@param target any[]
+---@param source any[]
+function itable_append(target, source)
+ for _, value in ipairs(source) do target[#target + 1] = value end
+ return target
+end
+
+---@param target any[]
+---@param source any[]
+---@param props? string[]
+function table_assign(target, source, props)
+ if props then
+ for _, name in ipairs(props) do target[name] = source[name] end
+ else
+ for prop, value in pairs(source) do target[prop] = value end
+ end
+ return target
+end
+
+---@generic T
+---@param table T
+---@return T
+function table_shallow_copy(table)
+ local result = {}
+ for key, value in pairs(table) do result[key] = value end
+ return result
+end
+
+--[[ EASING FUNCTIONS ]]
+
+function ease_out_quart(x) return 1 - ((1 - x) ^ 4) end
+function ease_out_sext(x) return 1 - ((1 - x) ^ 6) end
+
+--[[ CLASSES ]]
+
+---@class Class
+Class = {}
+function Class:new(...)
+ local object = setmetatable({}, {__index = self})
+ object:init(...)
+ return object
+end
+function Class:init() end
+function Class:destroy() end
+
+function class(parent) return setmetatable({}, {__index = parent or Class}) end