720 lines
22 KiB
Lua
720 lines
22 KiB
Lua
local msg = require 'mp.msg'
|
|
local utils = require 'mp.utils'
|
|
local s2t = require("dicts/s2t_chars")
|
|
local t2s = require("dicts/t2s_chars")
|
|
|
|
local function ass_escape(text)
|
|
return text:gsub("\\", "\\\\")
|
|
:gsub("{", "\\{")
|
|
:gsub("}", "\\}")
|
|
:gsub("\n", "\\N")
|
|
end
|
|
|
|
local function xml_unescape(str)
|
|
return str:gsub(""", "\"")
|
|
:gsub("'", "'")
|
|
:gsub(">", ">")
|
|
:gsub("<", "<")
|
|
:gsub("&", "&")
|
|
end
|
|
|
|
local function decode_html_entities(text)
|
|
return text:gsub("&#x([%x]+);", function(hex)
|
|
local codepoint = tonumber(hex, 16)
|
|
return unicode_to_utf8(codepoint)
|
|
end):gsub("&#(%d+);", function(dec)
|
|
local codepoint = tonumber(dec, 10)
|
|
return unicode_to_utf8(codepoint)
|
|
end)
|
|
end
|
|
|
|
-- 加载黑名单模式
|
|
local function load_blacklist_patterns(filepath)
|
|
local patterns = {}
|
|
if not file_exists(filepath) then
|
|
return patterns
|
|
end
|
|
local file = io.open(filepath, "r")
|
|
if not file then
|
|
msg.error("无法打开黑名单文件: " .. filepath)
|
|
return patterns
|
|
end
|
|
|
|
if string.match(filepath, "%.xml$") then
|
|
-- xml文件格式示例
|
|
--<?xml version="1.0" encoding="utf-8"?>
|
|
--<filters>
|
|
-- <item enabled="true">t=卡在</item>
|
|
-- <item enabled="true">t=进度条</item>
|
|
--</filters>
|
|
print("加载黑名单文件: " .. filepath)
|
|
for line in file:lines() do
|
|
local pattern = line:match('<item%s+enabled="true">t=(.-)</item>')
|
|
if pattern then
|
|
print("加载黑名单模式: " .. pattern)
|
|
table.insert(patterns, pattern)
|
|
end
|
|
end
|
|
end
|
|
|
|
if string.match(filepath, "%.json$") then
|
|
-- json文件格式示例
|
|
-- [{"type":0,"filter":"开门","opened":true,"id":15628936}
|
|
-- ,{"type":0,"filter":"tony","opened":true,"id":15628939}
|
|
-- ,{"type":1,"filter":"0+.1","opened":true,"id":15628951}]
|
|
local content = read_file(filepath)
|
|
if content then
|
|
local json = utils.parse_json(content)
|
|
if json and type(json) == "table" then
|
|
for _, entry in ipairs(json) do
|
|
if entry.opened and entry.filter and entry.type == 0 then
|
|
table.insert(patterns, entry.filter)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
if string.match(filepath, "%.txt$") then
|
|
-- 文本文件格式示例
|
|
-- 卡在
|
|
-- 进度条
|
|
for line in file:lines() do
|
|
line = line:match("^%s*(.-)%s*$")
|
|
if line ~= "" then
|
|
table.insert(patterns, line)
|
|
end
|
|
end
|
|
end
|
|
|
|
file:close()
|
|
return patterns
|
|
end
|
|
|
|
local blacklist_file = mp.command_native({ "expand-path", options.blacklist_path })
|
|
local black_patterns = load_blacklist_patterns(blacklist_file)
|
|
|
|
-- 检查字符串是否在黑名单中
|
|
function is_blacklisted(str, patterns)
|
|
for _, pattern in ipairs(patterns) do
|
|
local ok, result = pcall(function()
|
|
return str:match(pattern)
|
|
end)
|
|
|
|
if ok and result then
|
|
return true, pattern
|
|
elseif not ok then
|
|
-- msg.debug("黑名单规则错误,跳过: " .. pattern .. ",错误信息:" .. result)
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
-- 简繁转换
|
|
local function convert(text, dict)
|
|
return text:gsub("[%z\1-\127\194-\244][\128-\191]*", function(c)
|
|
return dict[c] or c
|
|
end)
|
|
end
|
|
|
|
local function ch_convert(str)
|
|
if options.chConvert == 1 then
|
|
return convert(str, t2s)
|
|
elseif options.chConvert == 2 then
|
|
return convert(str, s2t)
|
|
end
|
|
return str
|
|
end
|
|
|
|
local ch_convert_cache = {}
|
|
local ch_cache_keys = {}
|
|
local ch_cache_max = 5000
|
|
|
|
local function ch_convert_cached(text)
|
|
if type(text) ~= "string" or text == "" then return text end
|
|
local cached = ch_convert_cache[text]
|
|
if cached ~= nil then return cached end
|
|
|
|
local converted = ch_convert(text)
|
|
ch_convert_cache[text] = converted
|
|
ch_cache_keys[#ch_cache_keys+1] = text
|
|
|
|
if #ch_cache_keys > ch_cache_max then
|
|
local old_key = table.remove(ch_cache_keys, 1)
|
|
ch_convert_cache[old_key] = nil
|
|
end
|
|
|
|
return converted
|
|
end
|
|
|
|
-- 合并重复弹幕
|
|
local function merge_duplicate_danmaku(danmakus, threshold)
|
|
if not threshold or tonumber(threshold) < 0 then return danmakus end
|
|
|
|
local groups = {}
|
|
|
|
for _, d in ipairs(danmakus) do
|
|
local tkey = tostring(d.type or "")
|
|
local ckey = tostring(d.color or "")
|
|
local text = d.text or ""
|
|
|
|
groups[tkey] = groups[tkey] or {}
|
|
groups[tkey][ckey] = groups[tkey][ckey] or {}
|
|
groups[tkey][ckey][text] = groups[tkey][ckey][text] or {}
|
|
table.insert(groups[tkey][ckey][text], d)
|
|
end
|
|
|
|
local merged = {}
|
|
local abs = math.abs
|
|
|
|
for _, bytype in pairs(groups) do
|
|
for _, bycolor in pairs(bytype) do
|
|
for _, group in pairs(bycolor) do
|
|
table.sort(group, function(a, b) return a.time < b.time end)
|
|
|
|
local i = 1
|
|
while i <= #group do
|
|
local base = group[i]
|
|
local times = { base.time }
|
|
local count = 1
|
|
local j = i + 1
|
|
|
|
while j <= #group and abs(group[j].time - base.time) <= threshold do
|
|
times[#times+1] = group[j].time
|
|
count = count + 1
|
|
j = j + 1
|
|
end
|
|
|
|
local same_time = true
|
|
for k = 2, #times do
|
|
if times[k] ~= times[1] then
|
|
same_time = false
|
|
break
|
|
end
|
|
end
|
|
|
|
local danmaku = {
|
|
time = base.time,
|
|
type = base.type,
|
|
size = base.size,
|
|
color = base.color,
|
|
text = base.text,
|
|
source = base.source,
|
|
orig_time = base.orig_time,
|
|
}
|
|
if count > 2 or not same_time then
|
|
danmaku.text = danmaku.text .. string.format("x%d", count)
|
|
end
|
|
|
|
table.insert(merged, danmaku)
|
|
i = j
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
table.sort(merged, function(a, b) return a.time < b.time end)
|
|
return merged
|
|
end
|
|
|
|
-- 限制每屏弹幕条数
|
|
local function limit_danmaku(danmakus, limit)
|
|
if not limit or limit <= 0 then
|
|
return danmakus
|
|
end
|
|
|
|
local window = {}
|
|
for _, d in ipairs(danmakus) do
|
|
for i = #window, 1, -1 do
|
|
if window[i].end_time <= d.start_time then
|
|
table.remove(window, i)
|
|
end
|
|
end
|
|
|
|
if #window < limit then
|
|
table.insert(window, d)
|
|
else
|
|
local max_idx = 1
|
|
for i = 2, #window do
|
|
if window[i].end_time > window[max_idx].end_time then
|
|
max_idx = i
|
|
end
|
|
end
|
|
if window[max_idx].end_time > d.end_time then
|
|
window[max_idx].drop = true
|
|
window[max_idx] = d
|
|
else
|
|
d.drop = true
|
|
end
|
|
end
|
|
end
|
|
|
|
local result = {}
|
|
for _, d in ipairs(danmakus) do
|
|
if not d.drop then
|
|
table.insert(result, d)
|
|
end
|
|
end
|
|
return result
|
|
end
|
|
|
|
-- 解析 XML 弹幕
|
|
local function parse_xml_danmaku(xml_string)
|
|
local danmakus = {}
|
|
-- [^>]* 匹配其他 attributes
|
|
-- %f[^%s] 确保 p= 前面是空白字符
|
|
for p_attr, text in xml_string:gmatch('<d%s+[^>]*%f[^%s]p="([^"]+)"[^>]*>([^<]+)</d>') do
|
|
local params = {}
|
|
local i = 1
|
|
for val in p_attr:gmatch("([^,]+)") do
|
|
params[i] = tonumber(val)
|
|
i = i + 1
|
|
end
|
|
|
|
if params[1] and params[2] and params[3] and params[4] then
|
|
table.insert(danmakus, {
|
|
time = params[1],
|
|
type = params[2] or 1,
|
|
size = params[3] or 25,
|
|
color = params[4] or 0xFFFFFF,
|
|
text = xml_unescape(text)
|
|
})
|
|
end
|
|
end
|
|
|
|
table.sort(danmakus, function(a, b) return a.time < b.time end)
|
|
return danmakus
|
|
end
|
|
|
|
-- 解析 JSON 弹幕
|
|
local function parse_json_danmaku(json_string)
|
|
local danmakus = {}
|
|
if json_string:sub(1, 3) == "\239\187\191" then
|
|
json_string = json_string:sub(4)
|
|
end
|
|
|
|
local json = utils.parse_json(json_string)
|
|
if not json or type(json) ~= "table" then
|
|
msg.info("JSON 解析失败")
|
|
return danmakus
|
|
end
|
|
|
|
for _, entry in ipairs(json) do
|
|
local c = entry.c
|
|
local text = entry.m or ""
|
|
if type(c) == "string" then
|
|
local params = {}
|
|
local i = 1
|
|
for val in c:gmatch("([^,]+)") do
|
|
params[i] = tonumber(val)
|
|
i = i + 1
|
|
end
|
|
|
|
if params[1] and params[2] and params[3] and params[4] then
|
|
table.insert(danmakus, {
|
|
time = params[1],
|
|
color = params[2] or 0xFFFFFF,
|
|
type = params[3] or 1,
|
|
size = params[4] or 25,
|
|
text = text
|
|
})
|
|
end
|
|
end
|
|
end
|
|
|
|
table.sort(danmakus, function(a, b) return a.time < b.time end)
|
|
return danmakus
|
|
end
|
|
|
|
-- 解析弹幕文件
|
|
function parse_danmaku_file(danmaku_input)
|
|
local danmakus = {}
|
|
|
|
if file_exists(danmaku_input) then
|
|
local content = read_file(danmaku_input)
|
|
if content then
|
|
local parsed = {}
|
|
if danmaku_input:match("%.xml$") then
|
|
parsed = parse_xml_danmaku(content)
|
|
elseif danmaku_input:match("%.json$") then
|
|
parsed = parse_json_danmaku(content)
|
|
end
|
|
|
|
for _, d in ipairs(parsed) do
|
|
table.insert(danmakus, d)
|
|
end
|
|
else
|
|
msg.info("无法读取文件内容: " .. danmaku_input)
|
|
end
|
|
else
|
|
msg.info("文件不存在: " .. danmaku_input)
|
|
end
|
|
|
|
for _, d in ipairs(danmakus) do
|
|
if d.orig_time == nil then d.orig_time = d.time end
|
|
end
|
|
|
|
if #danmakus == 0 then
|
|
msg.info("未能解析任何弹幕")
|
|
return nil
|
|
end
|
|
|
|
return danmakus
|
|
end
|
|
|
|
--# 弹幕数组与布局算法 (Danmaku Array & Layout Algorithms)
|
|
local DanmakuArray = {}
|
|
DanmakuArray.__index = DanmakuArray
|
|
|
|
function DanmakuArray:new(res_x, res_y, font_size)
|
|
local obj = {
|
|
solution_y = res_y,
|
|
font_size = font_size,
|
|
rows = math.floor(res_y / font_size),
|
|
time_length_array = {}
|
|
}
|
|
for i = 1, obj.rows do
|
|
obj.time_length_array[i] = { time = 0, length = 0, empty = true }
|
|
end
|
|
setmetatable(obj, self)
|
|
return obj
|
|
end
|
|
|
|
function DanmakuArray:set_time_length(row, time, length)
|
|
if row > 0 and row <= self.rows then
|
|
self.time_length_array[row] = { time = time, length = length, empty = false }
|
|
end
|
|
end
|
|
|
|
function DanmakuArray:get_time(row)
|
|
if row > 0 and row <= self.rows then
|
|
return self.time_length_array[row].time
|
|
end
|
|
return -1
|
|
end
|
|
|
|
function DanmakuArray:get_length(row)
|
|
if row > 0 and row <= self.rows then
|
|
return self.time_length_array[row].length
|
|
end
|
|
return 0
|
|
end
|
|
|
|
function DanmakuArray:is_empty(row)
|
|
if row > 0 and row <= self.rows then
|
|
return self.time_length_array[row].empty
|
|
end
|
|
return false
|
|
end
|
|
|
|
-- 滚动弹幕 Y 坐标算法
|
|
function get_position_y(font_size, appear_time, text_length, resolution_x, roll_time, array)
|
|
local velocity = (text_length + resolution_x) / roll_time
|
|
|
|
for i = 1, array.rows do
|
|
local previous_appear_time = array:get_time(i)
|
|
if array:is_empty(i) then
|
|
array:set_time_length(i, appear_time, text_length)
|
|
return 1 + (i - 1) * font_size
|
|
end
|
|
|
|
local previous_length = array:get_length(i)
|
|
local previous_velocity = (previous_length + resolution_x) / roll_time
|
|
local delta_velocity = velocity - previous_velocity
|
|
local delta_x = (appear_time - previous_appear_time) * previous_velocity - previous_length
|
|
|
|
if delta_x >= 0 then
|
|
if delta_velocity <= 0 then
|
|
array:set_time_length(i, appear_time, text_length)
|
|
return 1 + (i - 1) * font_size
|
|
end
|
|
|
|
local delta_time = delta_x / delta_velocity
|
|
if delta_time >= roll_time then
|
|
array:set_time_length(i, appear_time, text_length)
|
|
return 1 + (i - 1) * font_size
|
|
end
|
|
end
|
|
end
|
|
-- 所有行都被占用,放弃渲染
|
|
return nil
|
|
end
|
|
|
|
-- 固定弹幕 Y 坐标算法
|
|
function get_fixed_y(font_size, appear_time, fixtime, array, from_top)
|
|
local row_start, row_end, row_step
|
|
if from_top then
|
|
row_start, row_end, row_step = 1, array.rows, 1
|
|
else
|
|
row_start, row_end, row_step = array.rows, 1, -1
|
|
end
|
|
|
|
for i = row_start, row_end, row_step do
|
|
local previous_appear_time = array:get_time(i)
|
|
if array:is_empty(i) then
|
|
array:set_time_length(i, appear_time, 0)
|
|
return (i - 1) * font_size + 1
|
|
else
|
|
local delta_time = appear_time - previous_appear_time
|
|
if delta_time > fixtime then
|
|
array:set_time_length(i, appear_time, 0)
|
|
return (i - 1) * font_size + 1
|
|
end
|
|
end
|
|
end
|
|
-- 所有行都被占用,放弃渲染
|
|
return nil
|
|
end
|
|
|
|
-- 将弹幕转换为 XML 格式
|
|
function convert_danmaku_to_xml(danmaku_out)
|
|
local danmakus = {}
|
|
for _, source in pairs(DANMAKU.sources) do
|
|
if not source.blocked and source.data then
|
|
for _, d in ipairs(source.data) do
|
|
table.insert(danmakus, d)
|
|
end
|
|
end
|
|
end
|
|
|
|
if #danmakus == 0 then
|
|
show_message("弹幕内容为空,无法保存", 3)
|
|
msg.verbose("弹幕内容为空,无法保存")
|
|
COMMENTS = {}
|
|
return false
|
|
end
|
|
|
|
-- 拼接为 XML 内容
|
|
local xml = { '<?xml version="1.0" encoding="UTF-8"?><i>\n' }
|
|
for _, d in ipairs(danmakus) do
|
|
local time = d.time
|
|
local type = d.type or 1
|
|
local size = d.size or 25
|
|
local color = d.color or 0xFFFFFF
|
|
local text = d.text or ""
|
|
|
|
text = text:gsub("&", "&")
|
|
:gsub("<", "<")
|
|
:gsub(">", ">")
|
|
:gsub("\"", """)
|
|
:gsub("'", "'")
|
|
|
|
table.insert(xml, string.format('<d p="%s,%s,%s,%s">%s</d>\n', time, type, size, color, text))
|
|
end
|
|
table.insert(xml, '</i>')
|
|
|
|
-- 写入 XML 文件
|
|
local file = io.open(danmaku_out, "w")
|
|
if not file then
|
|
show_message("无法写入目标 XML 文件", 3)
|
|
msg.info("无法写入目标 XML 文件: " .. danmaku_out)
|
|
return false
|
|
end
|
|
file:write(table.concat(xml))
|
|
file:close()
|
|
show_message("转换 XML 弹幕成功: " .. danmaku_out, 3)
|
|
msg.info("转换 XML 弹幕成功: " .. danmaku_out)
|
|
return true
|
|
end
|
|
|
|
function convert_danmaku_to_ass_events(force)
|
|
local per_source_lists = {}
|
|
for url, source in pairs(DANMAKU.sources) do
|
|
if not source.blocked and source.data then
|
|
local segments = nil
|
|
local prefix = nil
|
|
if source.delay_segments and #source.delay_segments > 0 then
|
|
segments = {}
|
|
for i, v in ipairs(source.delay_segments) do segments[i] = v end
|
|
table.sort(segments, function(a, b) return (a.start or 0) < (b.start or 0) end)
|
|
prefix = {}
|
|
local s = 0
|
|
for i, v in ipairs(segments) do
|
|
s = s + (v.delay or 0)
|
|
prefix[i] = s
|
|
end
|
|
end
|
|
|
|
local function get_cached_delay(t)
|
|
local segs = segments or {}
|
|
local pre = prefix or {}
|
|
if #segs == 0 then return 0 end
|
|
local idx = binary_search(segs, t, function(s) return (s and s.start) or 0 end)
|
|
local target = idx - 1
|
|
if target < 1 then return 0 end
|
|
return pre[target] or 0
|
|
end
|
|
|
|
local list = {}
|
|
for _, d in ipairs(source.data) do
|
|
local base_time = d.orig_time or d.time
|
|
if d.orig_time == nil then d.orig_time = base_time end
|
|
local adjusted_time = base_time + get_cached_delay(base_time)
|
|
local entry = {
|
|
orig_time = d.orig_time,
|
|
time = adjusted_time,
|
|
type = d.type,
|
|
size = d.size,
|
|
color = d.color,
|
|
text = d.text,
|
|
source = url,
|
|
}
|
|
if not is_blacklisted(d.text, black_patterns) then
|
|
table.insert(list, entry)
|
|
end
|
|
end
|
|
|
|
if #list > 0 then
|
|
table.sort(list, function(a, b) return a.time < b.time end)
|
|
per_source_lists[#per_source_lists + 1] = list
|
|
end
|
|
end
|
|
end
|
|
|
|
local danmakus = {}
|
|
|
|
local heap = new_min_heap()
|
|
for li = 1, #per_source_lists do
|
|
local lst = per_source_lists[li]
|
|
if lst and #lst > 0 then
|
|
heap.push({ time = lst[1].time, list_idx = li, pos = 1, entry = lst[1] })
|
|
end
|
|
end
|
|
|
|
while true do
|
|
local node = heap.pop()
|
|
if not node then break end
|
|
table.insert(danmakus, node.entry)
|
|
local li = node.list_idx
|
|
local next_pos = node.pos + 1
|
|
local lst = per_source_lists[li]
|
|
if lst and lst[next_pos] then
|
|
heap.push({ time = lst[next_pos].time, list_idx = li, pos = next_pos, entry = lst[next_pos] })
|
|
end
|
|
end
|
|
|
|
if options.max_screen_danmaku > 0 and options.merge_tolerance <= 0 then
|
|
options.merge_tolerance = options.scrolltime
|
|
end
|
|
|
|
danmakus = merge_duplicate_danmaku(danmakus, options.merge_tolerance)
|
|
|
|
if #danmakus == 0 then
|
|
if not force then
|
|
show_message("该集弹幕内容为空,结束加载", 3)
|
|
msg.verbose("该集弹幕内容为空,结束加载")
|
|
end
|
|
COMMENTS = {}
|
|
return
|
|
end
|
|
|
|
if not force then
|
|
msg.info("已解析 " .. #danmakus .. " 条弹幕")
|
|
end
|
|
|
|
local fontsize = tonumber(options.fontsize) or 50
|
|
local scrolltime = tonumber(options.scrolltime) or 15
|
|
local fixtime = tonumber(options.fixtime) or 5
|
|
|
|
local res_x = 1920
|
|
local res_y = 1080
|
|
|
|
local roll_array = DanmakuArray:new(res_x, res_y, fontsize)
|
|
local top_array = DanmakuArray:new(res_x, res_y, fontsize)
|
|
|
|
-- 预处理弹幕,先计算时间段以便进行数量限制
|
|
local pre_events = {}
|
|
for _, d in ipairs(danmakus) do
|
|
local time = d.type == 1 and math.floor(d.time + 0.5) or d.time
|
|
local orig_time = d.type == 1 and math.floor(d.orig_time + 0.5) or d.orig_time
|
|
local appear_time = time
|
|
local danmaku_type = d.type
|
|
|
|
local end_time = nil
|
|
if danmaku_type >= 1 and danmaku_type <= 3 then
|
|
end_time = appear_time + scrolltime
|
|
elseif danmaku_type == 5 or danmaku_type == 4 then
|
|
end_time = appear_time + fixtime
|
|
end
|
|
|
|
if end_time then
|
|
table.insert(pre_events, {orig_time = orig_time, start_time = appear_time, end_time = end_time, danmaku = d})
|
|
end
|
|
end
|
|
|
|
if options.max_screen_danmaku > 0 then
|
|
pre_events = limit_danmaku(pre_events, options.max_screen_danmaku)
|
|
end
|
|
|
|
local ass_events = {}
|
|
for _, ev in ipairs(pre_events) do
|
|
local d = ev.danmaku
|
|
local appear_time = ev.start_time
|
|
local danmaku_type = d.type
|
|
local clean_text = ch_convert_cached(decode_html_entities(d.text))
|
|
local text = ass_escape(clean_text)
|
|
:gsub("x(%d+)$", "{\\b1\\i1}x%1")
|
|
|
|
-- 颜色从十进制转为 BGR Hex
|
|
local color = math.max(0, math.min(d.color or 0xFFFFFF, 0xFFFFFF))
|
|
local color_hex = string.format("%06X", color)
|
|
local r = string.sub(color_hex, 1, 2)
|
|
local g = string.sub(color_hex, 3, 4)
|
|
local b = string.sub(color_hex, 5, 6)
|
|
local color_text = string.format("{\\c&H%s%s%s&}", b, g, r)
|
|
|
|
local style, effect
|
|
local pos, move = nil, nil
|
|
|
|
-- 滚动弹幕 (类型 1, 2, 3)
|
|
if danmaku_type >= 1 and danmaku_type <= 3 then
|
|
style = "R2L"
|
|
local text_length = get_str_width(text, fontsize)
|
|
local x1 = res_x + text_length / 2
|
|
local x2 = -text_length / 2
|
|
local y = get_position_y(fontsize, appear_time, text_length, res_x, scrolltime, roll_array)
|
|
if y then
|
|
effect = string.format("{\\move(%d, %d, %d, %d)}", x1, y, x2, y)
|
|
move = {x1, y, x2, y}
|
|
end
|
|
|
|
-- 顶部弹幕 (类型 5)
|
|
elseif danmaku_type == 5 then
|
|
style = "TOP"
|
|
local x = res_x / 2
|
|
local y = get_fixed_y(fontsize, appear_time, fixtime, top_array, true)
|
|
if y then
|
|
effect = string.format("{\\pos(%d, %d)}", x, y)
|
|
pos = {x, y}
|
|
end
|
|
|
|
-- 底部弹幕 (类型 4)
|
|
elseif danmaku_type == 4 then
|
|
style = "BTM"
|
|
local x = res_x / 2
|
|
local y = get_fixed_y(fontsize, appear_time, fixtime, top_array, false)
|
|
if y then
|
|
effect = string.format("{\\pos(%d, %d)}", x, y)
|
|
pos = {x, y}
|
|
end
|
|
end
|
|
|
|
if style and effect then
|
|
text = effect .. color_text .. text
|
|
local event = {
|
|
orig_time = ev.orig_time,
|
|
start_time = ev.start_time,
|
|
end_time = ev.end_time,
|
|
delay = ev.start_time - (ev.orig_time or ev.start_time),
|
|
style = style,
|
|
text = text,
|
|
clean_text = clean_text,
|
|
pos = pos,
|
|
move = move,
|
|
source = d.source,
|
|
}
|
|
table.insert(ass_events, event)
|
|
COMMENTS = ass_events
|
|
end
|
|
end
|
|
end |