模块:线路时刻表

local p = {}

local getArgs = require('Module:Arguments').getArgs

---@param funcName string
---@return fun(frame: frame)
local function makeInvokeFunction(funcName)
	return function(frame)
		local args = getArgs(frame, { parentOnly = true })
		return p[funcName](args, frame)
	end
end

---@type fun(_: {[1]: string}): string
local delink = require('Module:delink').delink

---@alias spanTerminus { [1]: string[], [2]: string[] }
---@alias spanDay { [1]: spanTerminus, [2]: spanTerminus }
---@alias span { line: string, type: string?, stations: string[], data: { [string]: spanDay } }
---@alias spanLTerminus { [1]: string[], [2]: string[][] }
---@alias spanLDay spanLTerminus[]
---@alias spanL { line: string, type: string?, stations: string[], data: { [string]: spanLDay }, override_termini: string[], sub_termini: { [1]: string[], [2]: string[], scope: 'last' } }
---@alias periodDay { [1]: string[], [2]: string[] }
---@alias pointDay { [1]: { [1]: integer }, [2]: { [1]: integer } }
---@alias point { line: string|string[], type: string|string[]?, periods: periodDay[], data: { [string]: pointDay[] } }
---@alias timeEntry { days: string[], ref: string, timespans: (span|spanL)[], timepoints: point[] }
---@type timeEntry
local data

---@alias adjType { title: string, color: string, circular: boolean, ['left terminus']: string, ['right terminus']: string }
---@alias adjLine { types: { [string]: adjType }, title: string, color: string, circular: boolean, ['left terminus']: string, ['right terminus']: string }
---@alias adjEntry { lines: { [string]: adjLine }, aliases: { [string]: string } }
---@type adjEntry
local adj_data

---加载时刻数据
---@param system string
---@return timeEntry?
local function getData(system)
	local success, result = pcall(mw.loadData, 'Module:线路时刻表/' .. system)
	if success then
		return result
	else
		return nil
	end
end

---加载线网数据
---@param system string
---@return adjEntry?
local function getAdjData(system)
	local success, result = pcall(mw.loadData, 'Module:Adjacent stations/' .. system)
	if success then
		return result
	else
		return nil
	end
end

---加载数据
---@param system string
local function initData(system)
	local d = getData(system)
	if d then
		data = d
	else
		error('不存在模块:线路时刻表/' .. system)
	end
	local a = getAdjData(system)
	if a then
		adj_data = a
	else
		error('不存在模块:Adjacent stations/' .. system)
	end
end

---获取线路数据
---@param line string
---@return adjLine
local function getLineData(line)
	if line then
		if adj_data['aliases'] then
			line = adj_data['aliases'][mw.ustring.lower(line)] or line
		end
		local default = adj_data['lines']['_default'] or {}
		local line_data = {}
		-- Shadow clone to avoid edit immutable objects
		for k, v in pairs(adj_data['lines'][line] or {}) do
			line_data[k] = v
		end
		for k, v in pairs(default) do
			if v then line_data[k] = line_data[k] or v end
		end
		line_data['title'] = line_data['title'] and mw.ustring.gsub(line_data['title'], '%%1', line)
		return line_data
	else
		error('Null argument')
	end
end

---获取支线数据
---@param line_data adjLine
---@param _type string
---@return adjType
local function getTypeData(line_data, _type)
	_type = adj_data.aliases and adj_data.aliases[mw.ustring.lower(_type)] or _type
	return line_data.types and line_data.types[_type] or {}
end

---获取线路标题
---@param line string
---@param _type string
---@return string
local function getTitle(line, _type)
	local line_data = getLineData(line)
	if _type then
		local type_data = getTypeData(line_data, _type)
		return line_data.title .. (type_data.title or _type)
	else
		return line_data.title
	end
end

---获取支线标题
---@param line string
---@param _type string
---@return string
local function getSubtitle(line, _type)
	local line_data = getLineData(line)
	if _type then
		local type_data = getTypeData(line_data, _type)
		return type_data.title or _type
	else
		error('Null argument')
	end
end

---获取线路颜色
---@param line string
---@param _type string
---@return string
local function getColor(line, _type)
	local line_data = getLineData(line)
	if _type then
		local type_data = getTypeData(line_data, _type)
		return '#' .. (type_data.color or line_data.color)
	else
		return '#' .. line_data.color
	end
end

---@alias terminus string|{ [integer]: string, via: string? }
---@alias termini { [1]: terminus, [2]: terminus }

---获取线路终点
---@param line string
---@param _type string
---@return termini
local function getTermini(line, _type)
	local line_data = getLineData(line)
	if _type then
		local type_data = getTypeData(line_data, _type)
		return {
			type_data['right terminus'] or line_data['right terminus'],
			type_data['left terminus'] or line_data['left terminus']
		}
	else
		return {
			line_data['right terminus'],
			line_data['left terminus']
		}
	end
end

---获取车站链接
---@param frame frame
---@param name string
---@param system string
---@param line string
---@param _type string
---@return string
local function getStation(frame, name, system, line, _type)
	return require('Module:Adjacent stations')._station(
		{ system, name, line, _type },
		frame)
end

---构造终点站表头
---@param frame frame
---@param terminus terminus
---@param system string
---@param line string
---@param _type string
---@return string
local function makeTerminus(frame, terminus, system, line, _type)
	local circular
	if _type and _type ~= '' then
		circular = adj_data.lines[line].types[_type].circular
		if circular == nil then
			circular = adj_data.lines[line].circular
		end
	else
		circular = adj_data.lines[line].circular
	end
	if type(terminus) == 'table' then
		if circular then
			return table.concat(terminus, '或')
		else
			local names = {}
			for _, name in ipairs(terminus) do
				table.insert(names, getStation(frame, name, system, line, _type))
			end
			return '往' .. table.concat(names, '或')
		end
	else
		if circular then
			return terminus
		else
			return '往' .. getStation(frame, terminus, system, line, _type)
		end
	end
end

---构造次级终点站表头
---@param frame frame
---@param subTerminus string
---@param system string
---@param line string
---@param _type string
---@return string
local function makeSubTerminus(frame, subTerminus, system, line, _type)
	return '至' .. getStation(frame, subTerminus, system, line, _type)
end

---检测车站是否运营
---@param t span|spanDay|spanTerminus|spanL|spanLDay|spanLTerminus|string
---@return boolean
local function checkOperated(t)
	if type(t) == 'table' then
		return checkOperated(t[1])
	else
		return t ~= nil
	end
end

---平面化数组
---@param input table
---@param flattened string[]?
---@return string[]
local function flattenMatrix(input, flattened)
	flattened = flattened or {}
	for i, element in ipairs(input) do
		if type(element) == 'table' then
			flattenMatrix(element, flattened)
		else
			table.insert(flattened, element)
		end
	end
	return flattened
end

---克隆单层数组
---@generic T
---@param input T[]?
---@return T[]?
local function cloneArray(input)
	if not input then return nil end

	local result = {}
	for i, item in ipairs(input) do
		result[i] = item
	end
	return result
end

---构建车站信息框时刻
---@param args string[]
---@param frame frame
---@return html
function p._as_station(args, frame)
	local system = args[1]
	local station = args[2]
	local all_days = args[3] -- 使用完整时刻表

	initData(system)
	local days
	if all_days then
		days = cloneArray(data.days)
	else
		days = { data.days[1] }
	end

	local tbl = mw.html.create('table')
		:addClass('wikitable')
		:addClass('station-infobox-timetable')

	--表头
	if #days > 1 then
		local tr_day = mw.html.create('tr')
		tr_day
			:tag('th')
			:attr('colspan', 3):attr('rowspan', 2)
			:wikitext(frame:preprocess(data.ref)):done()
		for _, day in ipairs(days) do
			tr_day
				:tag('th')
				:attr('colspan', 2):attr('scope', 'colgroup')
				:wikitext(day):done()
		end
		tr_day:allDone()
		tbl:node(tr_day)

		local tr_type = mw.html.create('tr')
		for _ = 1, #days do
			tr_type
				:tag('th')
				:attr('scope', 'col')
				:wikitext('首班'):done()
				:tag('th')
				:attr('scope', 'col')
				:wikitext('末班'):done()
		end
		tr_type:allDone()
		tbl:node(tr_type)
	elseif #days == 1 then
		tbl:tag('tr')
			:tag('th')
			:attr('colspan', 3)
			:wikitext(days[1], frame:preprocess(data.ref)):done()
			:tag('th')
			:attr('scope', 'col')
			:wikitext('首班'):done()
			:tag('th')
			:attr('scope', 'col')
			:wikitext('末班'):done()
			:done()
	else
		error('缺少运行图元信息')
	end

	for _, route_data in ipairs(data.timespans) do
		local _data = route_data.data[station]
		if _data and checkOperated(_data) then
			-- 允许系统内部分线路退化为单day
			if #days > 1 and #_data == 1 then
				for _ = 1, #days - #_data do
					table.insert(_data, _data[1])
				end
			end

			local color = getColor(route_data.line, route_data.type)
			local termini = route_data.override_termini or getTermini(route_data.line, route_data.type)

			-- 有子终点站
			if route_data.sub_termini then
				if route_data.sub_termini.scope == 'last' then
					---@cast _data spanLDay

					local sum_row = 0
					local t_map = {}
					local s_map = {}

					-- 预处理:有效行
					for t, terminus in ipairs(termini) do
						table.insert(s_map, {})

						for s, _ in ipairs(route_data.sub_termini[t]) do
							for d = 1, #days do
								if _data[d][2][t][s] and _data[d][2][t][s] ~= '' then
									table.insert(s_map[#s_map], s)
									break
								end
							end
						end

						-- 处理终点站(末班为空)
						if #s_map[#s_map] == 0 then
							if terminus == station then
								table.insert(s_map[#s_map], 0) -- 标记终点站
							end
						end

						if #s_map[#s_map] == 0 then
							table.remove(s_map)
						else
							table.insert(t_map, t)
						end

						sum_row = sum_row + #s_map[#s_map]
					end

					-- 线路色条
					local tr = mw.html.create('tr')
					tr:tag('td')
						:addClass('bar')
						:attr('rowspan', sum_row):attr('role', 'rowheader')
						:attr('aria-label', delink { getTitle(route_data.line, route_data.type) })
						:css('background-color', color):done()

					for it, t in ipairs(t_map) do
						local terminus = termini[t]

						if it > 1 then
							tr = mw.html.create('tr')
						end

						-- 终点站
						if s_map[it][1] == 0 then
							tr:tag('td')
								:attr('colspan', 2):addClass('table-rh'):attr('role', 'rowheader')
								:wikitext(makeTerminus(frame, terminus, system, route_data.line, route_data.type)):done()
								:tag('td')
								:addClass('note'):attr('colspan', 2 * #days)
								:wikitext('终点站'):done()
								:allDone()

							tbl:node(tr)

							-- 仅全程
						elseif s_map[it][1] == 1 and #s_map[it] == 1 and terminus == route_data.sub_termini[t][1] then
							tr:tag('td')
								:attr('colspan', 2):addClass('table-rh'):attr('role', 'rowheader')
								:wikitext(makeTerminus(frame, terminus, system, route_data.line, route_data.type)):done()

							for d = 1, #days do
								tr:tag('td')
									:attr('rowspan', #s_map[it])
									:wikitext(_data[d][1][t]):done()
									:tag('td')
									:wikitext(_data[d][2][t][1]):done()
							end

							tbl:node(tr:allDone())
						elseif s_map[it][1] > 0 then
							tr:tag('td')
								:attr('rowspan', #s_map[it]):addClass('table-rh-parent'):attr('role', 'rowheader')
								:wikitext(makeTerminus(frame, terminus, system, route_data.line, route_data.type)):done()

							for is, s in ipairs(s_map[it]) do
								local sub_terminus = route_data.sub_termini[t][s]

								if is > 1 then
									tr = mw.html.create('tr')
								end

								if terminus == sub_terminus then
									tr:tag('td')
										:addClass('table-rh-child'):attr('role', 'rowheader')
										:wikitext('全程'):done()
								else
									tr:tag('td')
										:addClass('table-rh-child'):attr('role', 'rowheader')
										:wikitext(makeSubTerminus(frame, sub_terminus, system, route_data.line,
											route_data.type)):done()
								end

								for d = 1, #days do
									if is == 1 then
										tr:tag('td')
											:attr('rowspan', #s_map[it])
											:wikitext(_data[d][1][t]):done()
									end
									tr:tag('td')
										:wikitext(_data[d][2][t][s]):done()
								end

								tbl:node(tr:allDone())
							end
						else
							error('无效的索引值')
						end
					end
				else
					error('不支持的子终点站作用域') -- 目前仅实现末班车的子终点站
				end

				-- 无子终点站
			else
				---@cast _data spanDay

				local t_map = {}

				-- 预处理:有效行
				for t, terminus in ipairs(termini) do
					if terminus == station then
						table.insert(t_map, t)
					else
						for d = 1, #days do
							if (_data[d][1][t] and _data[d][1][t] ~= '') or (_data[d][2][t] and _data[d][2][t] ~= '') then
								table.insert(t_map, t)
								break
							end
						end
					end
				end

				local tr = mw.html.create('tr')
				tr:tag('td')
					:addClass('bar')
					:attr('rowspan', #t_map):attr('role', 'rowheader')
					:attr('aria-label', delink { getTitle(route_data.line, route_data.type) })
					:css('background-color', color):done()

				for it, t in ipairs(t_map) do
					local terminus = termini[t]

					if it > 1 then
						tr = mw.html.create('tr')
					end

					tr:tag('td')
						:attr('colspan', 2):addClass('table-rh'):attr('role', 'rowheader')
						:wikitext(makeTerminus(frame, terminus, system, route_data.line, route_data.type)):done()

					if terminus == station then
						tr:tag('td')
							:addClass('note'):attr('colspan', 2 * #days)
							:wikitext('终点站'):done()
					else
						for d = 1, #days do
							tr:tag('td')
								:wikitext(_data[d][1][t]):done()
							tr:tag('td')
								:wikitext(_data[d][2][t]):done()
						end
					end

					tr:allDone()
					tbl:node(tr)
				end
			end
		end
	end

	return tbl:allDone()
end

---构建线路时刻
---@param args string[]
---@param frame frame
---@return html?
function p._as_line(args, frame)
	local system = args[1]
	local line = args[2]
	local _type = args[3]

	initData(system)
	for _, route_data in ipairs(data.timespans) do
		if line == route_data.line and _type == route_data.type then
			local tbl = mw.html.create('table')
				:addClass('wikitable')
				:addClass('line-timetable')

			local color = getColor(route_data.line, route_data.type)
			local termini = cloneArray(route_data.override_termini) or getTermini(route_data.line, route_data.type)

			local num_cols
			local num_ref_rows = 2

			local days
			if #cloneArray(route_data.data[route_data.stations[1]]) == 1 then -- 允许线路退化为无 day
				days = { data.days[1] }
			else
				days = cloneArray(data.days)
			end

			if #days > 1 then num_ref_rows = num_ref_rows + 1 end
			if route_data.sub_termini then num_ref_rows = num_ref_rows + 1 end
			local ref = mw.html.create('th')
				:addClass('ref'):attr('rowspan', num_ref_rows)
				:wikitext('车站<br/>', frame:preprocess(data.ref))
				:allDone()

			local bar = mw.html.create('tr')
			---@type html[]
			local hr = {}
			if route_data.sub_termini then
				if route_data.sub_termini.scope == 'last' then
					local num_f_cols = #termini
					local num_l_cols = #flattenMatrix(route_data.sub_termini)
					num_cols = #days * (num_f_cols + num_l_cols)
					bar:tag('td')
						:addClass('bar'):attr('colspan', num_cols + 1):css('background-color', color)
						:allDone()

					local hT1 = mw.html.create()
					local hT2 = mw.html.create()
					local hS = mw.html.create()
					for t, terminus in ipairs(termini) do
						local sub_termini = cloneArray(route_data.sub_termini[t])

						local th_text = makeTerminus(frame, terminus, system, line, _type)
						hT1:tag('th')
							:addClass('major'):attr('rowspan', 2)
							:wikitext(th_text)
							:done()
						hT2:tag('th')
							:addClass('major'):attr('colspan', #sub_termini)
							:wikitext(th_text)
							:done()
						for s, sub_terminus in ipairs(sub_termini) do
							if terminus == sub_terminus then
								hS:tag('th'):addClass('minor'):wikitext('全程'):done()
							else
								hS:tag('th')
									:addClass('minor')
									:wikitext(makeSubTerminus(frame, sub_terminus, system, line, _type))
									:done()
							end
						end
					end

					local hFL = mw.html.create()
						:tag('th'):attr('colspan', num_f_cols):wikitext('首班'):done()
						:tag('th'):attr('colspan', num_l_cols):wikitext('末班')
						:allDone()

					hr[1] = mw.html.create('tr')
					hr[2] = mw.html.create('tr')
					hr[3] = mw.html.create('tr')
					hr[4] = mw.html.create('tr')
					if #days > 1 then
						hr[1]:node(ref)
					else
						hr[2]:node(ref)
					end
					for d, day in ipairs(days) do
						hr[1]:tag('th'):attr('colspan', num_f_cols + num_l_cols):wikitext(day):done()
						hr[2]:node(hFL)
						hr[3]:node(hT1):node(hT2)
						hr[4]:node(hS)
					end
					hr[1]:allDone()
					hr[2]:allDone()
					hr[3]:allDone()
					hr[4]:allDone()
				else
					error('不支持的子终点站作用域')
				end
			else
				num_cols = 2 * #days * #termini
				bar:tag('td')
					:addClass('bar'):attr('colspan', num_cols + 1):css('background-color', color)
					:allDone()

				local hT = mw.html.create('')
				for t, terminus in ipairs(termini) do
					hT:tag('th'):addClass('th'):wikitext(makeTerminus(frame, terminus, system, line, _type)):done()
				end
				hT:allDone()

				local hFL = mw.html.create()
					:tag('th'):attr('colspan', #termini):wikitext('首班'):done()
					:tag('th'):attr('colspan', #termini):wikitext('末班')
					:allDone()

				hr[1] = mw.html.create('tr')
				hr[2] = mw.html.create('tr')
				hr[3] = mw.html.create('tr')
				if #days > 1 then
					hr[1]:node(ref)
				else
					hr[2]:node(ref)
				end
				for _, day in ipairs(days) do
					hr[1]:tag('th'):attr('colspan', 2 * #termini):wikitext(day):done()
					hr[2]:node(hFL)
					hr[3]:node(hT):node(hT)
				end
				hr[1]:allDone()
				hr[2]:allDone()
				hr[3]:allDone()
			end

			if #days == 1 then table.remove(hr, 1) end
			tbl:node(bar)
			for _, row in ipairs(hr) do
				tbl:node(row)
			end
			tbl:node(bar)

			local tr
			for _, station in ipairs(route_data.stations) do
				local _data = flattenMatrix(route_data.data[station])
				tr = mw.html.create('tr'):tag('td'):addClass('table-rh'):wikitext(station):done()

				if checkOperated(_data) then
					for _, time in ipairs(_data) do
						tr:tag('td'):wikitext(time):done()
					end
				else
					tr:tag('td'):addClass('closed'):attr('colspan', num_cols):wikitext('未运营'):done()
				end

				tbl:node(tr:allDone())
			end

			tbl:node(bar)
			return tbl:allDone()
		end
	end
end

---@param str string
---@return integer
---@return integer
local function parseTime(str)
	return tonumber(string.sub(str, 1, 2)) or 0, tonumber(string.sub(str, 4, 5)) or 0
end

---@param h integer
---@param m integer
---@param other integer
---@return integer
---@return integer
local function addMinutes(h, m, other)
	h = h
	m = m + other
	if m >= 60 then
		local x = math.floor(m / 60)
		h = h + x
		m = m - 60 * x
	end
	return h, m
end

---构建车站车次时刻
---@param args string[]
---@param frame frame
---@return html
function p._as_station_ext(args, frame)
	local system = args[1]
	local station = args[2]

	initData(system)
	local days = cloneArray(data.days)

	local tbl = mw.html.create('table')
		:addClass('wikitable')
		:addClass('station-route-timetable')

	for _, route_data in ipairs(data.timepoints) do
		local _data = route_data.data[station]
		local route_periods = cloneArray(route_data.periods)

		if _data then
			if type(route_data.line) == 'table' then
				if type(route_data.type) == 'table' then
					for l, line in ipairs(route_data.line --[[ @as string[] ]]) do
						tbl:tag('tr')
							:tag('td')
							:addClass('bar'):addClass('composed'):attr('colspan', 2 * #days + 1)
							:css('background-color', getColor(line, route_data.type[l]))
							:allDone()
					end
				else
					for _, line in ipairs(route_data.line --[[ @as string[] ]]) do
						tbl:tag('tr')
							:tag('td')
							:addClass('bar'):addClass('composed'):attr('colspan', 2 * #days + 1)
							:css('background-color', getColor(line, route_data.type --[[ @as string ]]))
							:allDone()
					end
				end
			else
				tbl:tag('tr')
					:tag('td')
					:addClass('bar'):attr('colspan', 2 * #days + 1)
					:css('background-color',
						getColor(route_data.line --[[ @as string ]], route_data.type --[[ @as string ]]))
					:allDone()
			end

			local num_days = #route_periods
			local day_span = 1
			if num_days < #days then
				assert(num_days == 1, #route_periods)
				day_span = #days
			end

			---@type { [1]: { [integer]: string[] }, [2]: { [integer]: string[] } }[]
			local points = {}
			local h_min = 23
			local h_max = 0
			local update_h_bound = function(h)
				h_min = math.min(h, h_min)
				h_max = math.max(h, h_max)
			end
			for d = 1, num_days, 1 do
				local _points = {}

				_points[1] = {}
				for _, point in ipairs(route_periods[d][1]) do
					local h, m = parseTime(point)
					h, m = addMinutes(h, m, _data[d][1][1])
					h = h % 24
					if _points[1][h] then
						table.insert(_points[1][h], string.format('%02d', m))
					else
						_points[1][h] = { string.format('%02d', m) }
						update_h_bound(h)
					end
				end

				_points[2] = {}
				for _, point in ipairs(route_periods[d][2]) do
					local h, m = parseTime(point)
					h, m = addMinutes(h, m, _data[d][2][1])
					h = h % 24
					if _points[2][h] then
						table.insert(_points[2][h], string.format('%02d', m))
					else
						_points[2][h] = { string.format('%02d', m) }
						update_h_bound(h)
					end
				end

				table.insert(points, _points)
			end

			tbl:tag('tr'):tag('th')
				:addClass('ref'):wikitext(frame:preprocess(getSubtitle(
				route_data.line[1] or route_data.line,
				route_data.type[1] or route_data.type)))
				:attr('colspan', 2 * #days + 1)
				:done():done()

			local termini = getTermini(route_data.line[1] or route_data.line,
				route_data.type[1] or route_data.type)
			tbl:tag('tr')
				:tag('th')
				:wikitext(makeTerminus(frame, termini[1], system, route_data.line[1] or route_data.line,
					route_data.type[1] or route_data.type))
				:attr('colspan', #days)
				:done()
				:tag('th'):wikitext('时')
				:attr('rowspan', num_days > 1 and 2 or 1)
				:done()
				:tag('th')
				:wikitext(makeTerminus(frame, termini[2], system, route_data.line[1] or route_data.line,
					route_data.type[1] or route_data.type))
				:attr('colspan', #days)
				:done()
				:done()

			if num_days > 1 then
				local tr = mw.html.create('tr')
				for d = #days, 1, -1 do
					tr:tag('th'):wikitext(days[d]):done()
				end
				for d = 1, #days do
					tr:tag('th'):wikitext(days[d]):done()
				end
				tr:allDone()
				tbl:node(tr)
			end

			for h = h_min, h_max do
				local tr = mw.html.create('tr')
				for d = num_days, 1, -1 do
					tr:tag('td')
						:attr('style', 'direction: rtl')
						:wikitext(
							frame:expandTemplate {
								title = 'hlist',
								args = points[d][1][h]
							})
						:attr('colspan', day_span)
						:done()
				end
				tr:tag('td'):addClass('table-rh'):wikitext(string.format('%02d', h)):done()
				for d = 1, num_days do
					tr:tag('td')
						:wikitext(
							frame:expandTemplate {
								title = 'hlist',
								args = points[d][2][h]
							})
						:attr('colspan', day_span)
						:done()
				end
				tr:allDone()
				tbl:node(tr)
			end
		end
	end

	return tbl:allDone()
end

p.as_station = makeInvokeFunction('_as_station')
p.as_line = makeInvokeFunction('_as_line')
p.as_station_ext = makeInvokeFunction('_as_station_ext')

return p