Модуль:СтММ (Bk;rl,&VmBB)

Перейти к навигации Перейти к поиску
Документация
-- Этот модуль вызывается с помощью шаблонов СтММ и ЛММ

--[=[ Linter:

setmetatable(_G, {
	__newindex =
		function (_, n)
			error("attempt to write to undeclared variable "..n, 2)
		end,
	__index =
		function (_, n)
			error("attempt to read undeclared variable "..n, 2)
		end,
	})
--]=]

local p = {}
local jsontable = mw.loadJsonData('Шаблон:Интерактивная схема Московского метрополитена/data.json')

local function contains(tbl, x)
	local t
    local found = false
    for p, v in pairs(tbl) do
        if v == x then 
            found = p 
        end
    end
    return found
end

local function removeparen(text)
    return text:gsub( '^%s*(.+)%s+%b()%s*$', '%1' )
end

local function addquotes(text, icon)
	if (icon) then
		return text
	end
	return '<span class="stmm-quotes">«</span>' .. text .. '<span class="stmm-quotes">»</span>'
end

local function showicon(code, link)
	local iconsize = 15
	if (code == 'D4А') then
		iconsize = 30
	elseif (mw.ustring.find(code, 'D', 1, true)) then
		iconsize = 22
	end
	return '[[File:Moskwa Metro Line ' .. code .. '.svg|' .. iconsize .. 'px|'
		.. 'link=' .. link .. '|' .. removeparen(link) .. ']]'
end

local function splitname(text)
	if (text) then
		local slash = mw.ustring.find(text, '/', 1, true)
		if (slash) then
			return mw.ustring.sub(text, 1, slash - 1), mw.ustring.sub(text, slash + 1)
		end
		return text
	end
end

local function errors(text, template)
	error(text .. '[[Категория:Ошибки в использовании шаблона ' .. (template or 'СтММ') .. ']]' , 0)
end

local function findstation(stationparam)
	local csi, newline, newstation, category
	local lines = {}
	local allanswers = {}
	local newjson = {}
	local alllines = {}
	for _, value in pairs(jsontable.lines) do
		newline = {article = value.article, train = value.train, stations = {}}
		if (value.number and not value.hidden) then
			table.insert(alllines, value.number)
		end
		for _, value0 in pairs(value.stations or {}) do
			if (not value0.hidden) then
				newstation = {article = value0.article, line = value.number, rp = removeparen(value0.name or value0.article)}
				if (stationparam == newstation.rp or stationparam == value0.oldname) then
					if (stationparam == value0.oldname) then
						category = '[[Категория:Страницы, требующие переименования станций в шаблоне СтММ]]'
					end
					table.insert(allanswers, {value0.article, newstation.rp, newstation.line, value.article, value.train, category})
					table.insert(lines, newstation.line)
					if (value0.csi) then
						csi = {value.number, tostring(value0.csi), value.article, #allanswers}
					end
				end
				if (csi and csi[2] == value.number) then
					csi[5] = value.article
				end
				table.insert(newline.stations, newstation)
			end
		end
		if (value.number) then
			table.insert(newjson, newline)
		end
	end
	return lines, allanswers, csi, newjson, alllines
end

local function findnextstation(stationparam)
	local lines = {}
	local allanswers = {}
	for _, value in pairs(jsontable) do
		for _, value0 in pairs(value.stations or {}) do
			if (stationparam == value0.rp) then
				table.insert(allanswers, {value0.article, value0.rp, value0.line, value.article, value.train})
				table.insert(lines, value0.line)
			end
		end
	end
	return lines, allanswers
end

local function findstationhint(stationparam)
	local ahint, chint, rp, lowerp
	local lowerstation = mw.ustring.gsub(mw.ustring.lower(stationparam), ' +', ' ')
	for _, value in pairs(jsontable) do
		for _, value0 in pairs(value.stations or {}) do
			rp = value0.rp
			lowerp = mw.ustring.lower(rp)
			if (lowerp == lowerstation) then
				return rp
			elseif (mw.ustring.find(lowerstation, lowerp, 1, true)
					and (not ahint or (mw.ustring.len(rp) > mw.ustring.len(ahint)))) then
				ahint = rp
			elseif (not ahint and mw.ustring.find(lowerp, lowerstation, 1, true)
					and (not chint or (mw.ustring.len(rp) > mw.ustring.len(chint)))) then
				chint = rp
			end
		end
	end
	return ahint or chint
end

local function formatvariants(vars, cond)
	local err = ', возможные варианты: ' .. mw.ustring.gsub(mw.ustring.gsub(mw.text.jsonEncode(vars), '["%[%]]', ''),',', ', ')
	if (cond) then
		err = err .. '. Можно также использовать параметр «@@» вместо «@» для кроссплатформенных или общих станций'
	end
	return err
end

local function getargs(frame)
	local args = {}
	for p, v in pairs(frame:getParent().args) do
		args[p] = mw.text.trim(v)
	end
	return args
end

function p.station(frame)
	local startparam = 1
	local argsnumber = 0
	local iconparam = false
	local interchangeparam = false
	local hyphenparam = false
	local lineparam = ''
	local icon = ''
	local err = ''
	local ts = ''
	local args = getargs(frame)
	local stationparam, stationname, allanswers, hint, articleans, nameans,
		csiparam, csi, unknownpar, lines, hintlines, hintanswers, indexans

	for p, _ in ipairs(args) do
		argsnumber = math.max(argsnumber, p)
	end
	if (args[1] == '@') then
		iconparam = true
		startparam = 2
	elseif (args[1] == '-') then
		iconparam = true
		hyphenparam = true
		startparam = 2
	elseif (args[1] == '&') then
		iconparam = true
		interchangeparam = true
		startparam = 2
	elseif (args[1] == '@@') then
		csiparam = args[2]
		startparam = 2
	end
	if (not csiparam and argsnumber - startparam > 1) then
		return errors('Слишком много параметров')
	end
	for p, _ in pairs(args) do
		if (not contains({1, 2, 3}, p)) then
			if (csiparam) then
				unknownpar = true
			else
				return errors('Неизвестный параметр «' .. p .. '»')
			end
		end
	end
	stationparam, stationname = splitname(args[startparam])
	if (stationparam == '') then
		return errors('Пустое название станции')
	end
	if (stationname == '') then
		return errors('Пустой замещающий текст для отображения вместо названия станции')
	end
	if (csiparam and (args[startparam + 1] or unknownpar)) then
		return errors('Параметр «@@» должен использоваться исключительно с названием станции')
	end
	if (args[startparam + 1]) then
		lineparam = args[startparam + 1]
	end
	if (not stationparam) then
		return errors('Название станции не указано')
	end
	lines, allanswers, csi, jsontable, alllines = findstation(stationparam)
	if (not csi and csiparam) then
		return errors('Параметр «@@» должен использоваться только для кроссплатформенных или общих станций')
	end
	if (#allanswers == 0) then
		if (contains(alllines, stationparam) and lineparam ~= '') then
			errors('Возможно, вы используете старый синтаксис, станцию следует указывать до линии')
		end
		hint = findstationhint(stationparam)
		if (hint) then
			hintlines, hintanswers = findnextstation(hint)
			if (#hintanswers <= 1) then
				err = '. В номере линии нет необходимости'
			elseif (contains(hintlines, lineparam)) then
				err = ', она может использовать указанный номер линии: ' .. lineparam
			else
				err = '. Требуется также номер линии' .. formatvariants(hintlines)
			end
			err = ', но, возможно, вы имели в виду «' .. hint .. '»' .. err
		end
		return errors('Станция «' .. mw.ustring.gsub(stationparam, '  +', '<много пробелов>') .. '» неизвестна' .. err)
	end
	if (mw.text.jsonEncode(args[startparam + 1]) == '""') then
		if (#allanswers == 1) then
			return errors('Номер линии пуст и не требуется')
		elseif (#allanswers > 1) then
			return errors('Номер линии пуст' .. formatvariants(lines, csi and iconparam and not hyphenparam))
		end
	end
	if (#allanswers == 1 and lineparam ~= '') then
		return errors('Номер линии не требуется: ' .. lineparam)
	end
	if (not csiparam and ((#allanswers > 1 and not csi) or (#allanswers > 2 and csi)) and lineparam == '') then
		return errors('Существует несколько станций с таким названием. Требуется номер линии' .. formatvariants(lines, csi and iconparam and not hyphenparam))
	end
	if (not csiparam and #allanswers > 1) then
		for key, value in pairs (allanswers) do
			if (value[3] == lineparam) then
				indexans = key
			end
		end
		if (csi and lineparam == '') then
			indexans = 1
		end
	elseif (csi and csiparam) then
		indexans = csi[4]
	else
		indexans = 1
	end
	if (not indexans) then
		return errors('Неверный номер линии: ' .. lineparam .. formatvariants(lines, csi and iconparam and not hyphenparam))
	end
	articleans = allanswers[indexans][1]
	nameans = allanswers[indexans][2]
	if (iconparam or csiparam) then
		if (hyphenparam) then
			icon = '&nbsp;—'
		elseif (csi and lineparam == '') then
			icon = icon .. showicon(csi[1], csi[3]) .. showicon(csi[2], csi[5])
		else
			icon = showicon(allanswers[indexans][3], allanswers[indexans][4])
		end
	end
	if (iconparam) then
		ts = frame:extensionTag{
				name = 'templatestyles',
				args = { src = 'Шаблон:СтММ/styles.css' }
			}
	end
	return '<span style="white-space: nowrap">' .. icon .. ' ' .. addquotes('[[' .. articleans .. '|' .. (stationname or nameans)
			.. ']]', not tonumber(mw.ustring.sub(allanswers[indexans][3], 1, 1))
			or allanswers[indexans][5] or iconparam or csiparam) .. '</span>' .. ts .. (allanswers[indexans][6] or '')
end

function p.line(frame)
	local args = getargs(frame)
	local ans = ''
	local alllines = {}
	local allarticles = {}
	local found
	
	for p, v in pairs(args) do
		if (type(p) ~= 'number') then
			return errors('Неизвестный параметр «' .. p .. '»', 'ЛММ')
		end
		if (v == '') then
			return errors('Пустой параметр № ' .. p, 'ЛММ')
		end
	end
	if (not args[1]) then
		return errors('Нет параметров', 'ЛММ')
	end
	for _, value in ipairs(jsontable.lines) do
		if (value.number and not value.hidden) then
			table.insert(alllines, value.number)
			table.insert(allarticles, value.article)
		end
	end
	for _, value in ipairs(args) do
		found = contains(alllines, value)
		if (not found) then
			return errors('Неизвестная линия «' .. value .. '»', 'ЛММ')
		end
		ans = ans .. showicon(value, allarticles[found])
	end
	return '<span style="white-space: nowrap">' .. ans .. '&nbsp;</span>'
end

local function getprep(frame, text)
	return frame:preprocess(text)
end

local function fulldouble(frame)
	local cur
	local ans = '<table class=wikitable><tr><th colspan=2>@@</th></tr>' 
	for _, value in pairs(jsontable.lines) do
		for _, value0 in pairs(value.stations or {}) do
			if (value0.csi) then
				cur = '{{СтММ|@@|' .. removeparen(value0.name or value0.article) .. '}}'
				ans = ans .. '<tr><td>' .. cur .. '</td><td>' .. frame:preprocess(cur) .. '</td></tr>'
				--ans = ans .. '\n* ' .. removeparen(value0.article) .. ', ' .. value.number .. ', ' .. value0.csi
			end
		end
	end
	ans = ans .. '</table>'
	return ans
end

function p.full(frame)
	if (frame.args[1] == '0') then
		return fulldouble(frame)
	end
	local ans = '<table class=wikitable>'
	local rp
	local value
	local left, right, text
	value = jsontable.lines[tonumber(frame.args[1])]
	if (value.number and not value.hidden) then
		ans = ans .. '<tr><th>' .. value.number .. '</th><th>' .. (value.name or value.article) .. '</th></tr>'
		for _, value0 in pairs(value.stations or {}) do
			rp = removeparen(value0.name or value0.article)
			_, left = pcall(getprep, frame, '{{СтММ|@|' .. rp .. '}}')
			_, right = pcall(getprep, frame, '{{СтММ|@|' .. rp .. '|' .. value.number .. '}}')
			if (not mw.ustring.find(left, 'strong')) then
				ans = ans .. '<tr><td>{{СтММ|@|' .. rp .. '}}</td><td>' .. left .. '</td></tr>'
			end
			if (not mw.ustring.find(right, 'strong')) then
				ans = ans .. '<tr><td>{{СтММ|@|' .. rp .. '|' .. value.number .. '}}</td><td>' .. right .. '</td></tr>'
			end
		end
	end
	ans = ans .. '</table>'
	return ans
end

--[==[ Схема шаблона Московские маршруты
?param-text: сплиний(param-1)
 else: ?[[Файл:Logo train transilien.svg|15px|link=param-r/23|бсвл]]
       is-station(param-1): спстанций(param-1)
        else: [[Файл:Moskwa Metro Line param-1 (line?inv:alt)(искл).svg|15px|link=param-l1..сплиний(param-1)|бсвл]]
              ?param-line and ~= 0:
            		<span цвет=param-style..(param-1{12357}?white:black>[[сплиний(param-1)|param-2..сплиний(param-1)]]</span>
               else: ?[[Файл:Moskwa Metro Line param-2/34 alt.svg|15px|link=param-l2/34..сплиний(param-2/34)|бсвл]]
                     ?[[param-c (станция МЦК)|param-c]]
                     ?[[param-m (станция монорельса)|param-m]]
                     ?[[param-s (станция метро?(, param-d)(искл))|param-s]]

?[[Файл:Logo train transilien.svg|15px|link=param-r/23|бсвл]]
[[Файл:Moskwa Metro Line param-1 alt.svg|15px|link=param-l1..сплиний(param-1)|бсвл]]
?[[Файл:Moskwa Metro Line param-2/34 alt.svg|15px|link=param-l2/34..сплиний(param-2/34)|бсвл]]
?[[param-l1|param-cn]]
?[[param-c (станция МЦК)|param-c]]
?[[param-m (станция монорельса)|param-m]]
?nodisambig([[param-s (станция метро?(, param-d)(искл))|param-s]])
?param-sn

]==]
return p