Модуль:Песочница/Pok: различия между версиями
Pok (обсуждение | вклад) мНет описания правки |
Pok (обсуждение | вклад) Нет описания правки |
||
| (не показаны 24 промежуточные версии этого же участника) | |||
| Строка 1: | Строка 1: | ||
-- | local p = {} | ||
local | |||
local function trim(s) | |||
if not s then return s end | |||
return (s:gsub("^%s*(.-)%s*$", "%1")) | |||
end | |||
local function split(s, sep) | |||
if s == nil then return {} end | |||
local parts = {} | |||
sep = sep or "%." | |||
for part in string.gmatch(s, "([^" .. sep .. "]+)") do | |||
table.insert(parts, part) | |||
end | |||
return parts | |||
end | |||
local function parse_keys_from_template(content) | |||
if not content then return {} end | |||
local keys = {} | |||
local lower = content:lower() | |||
local titlePos = lower:find("|%s*title%s*=") | |||
if not titlePos then | |||
return keys | |||
end | |||
local startBrace = content:find("{{", titlePos) | |||
local region = nil | |||
if startBrace then | |||
local len = #content | |||
local i = startBrace | |||
local depth = 0 | |||
while i <= len - 1 do | |||
local two = content:sub(i, i+1) | |||
if two == "{{" then | |||
depth = depth + 1 | |||
i = i + 2 | |||
elseif two == "}}" then | |||
depth = depth - 1 | |||
i = i + 2 | |||
if depth == 0 then | |||
region = content:sub(startBrace, i-1) | |||
break | |||
end | |||
else | |||
i = i + 1 | |||
end | |||
end | |||
end | |||
if not region then | |||
local substr = content:sub(titlePos) | |||
local endPos = substr:find("}}") | |||
if endPos then | |||
region = substr:sub(1, endPos) | |||
else | |||
region = substr | |||
end | |||
end | |||
for key in string.gmatch(region, "|%s*([^=|%}]-)%s*=") do | |||
local k = trim(key) | |||
if k ~= "" then | |||
table.insert(keys, k) | |||
end | |||
end | |||
return keys | |||
end | |||
local function load_module_data(page) | |||
local baseUser = "IanComradeBot/" | |||
local moduleName = "Module:" .. baseUser .. page .. "/data" | |||
local ok, data = pcall(mw.loadData, moduleName) | |||
if not ok then return nil end | |||
return data | |||
end | |||
local function load_template_content(path) | |||
local title = mw.title.new("Template:" .. path) | |||
if not title then return nil end | |||
local ok, content = pcall(function() return title:getContent() end) | |||
if not ok then return nil end | |||
return content | |||
end | |||
function p.get(frame) | |||
local args = frame.args or {} | |||
local id = args[1] or "" | |||
if id == "" then return "" end | |||
local componentDefs = load_module_data("component.json") | |||
local prototypeDefs = load_module_data("prototype.json") | |||
if not componentDefs or not prototypeDefs then | |||
return "" | |||
end | |||
local foundComponents = {} | |||
local foundPrototypes = {} | |||
local compList = componentDefs[id] | |||
if type(compList) == "table" then | |||
for _, v in ipairs(compList) do | |||
if type(v) == "string" then | |||
foundComponents[v] = true | |||
end | |||
end | |||
end | |||
local protoList = prototypeDefs[id] | |||
if type(protoList) == "table" then | |||
for _, v in ipairs(protoList) do | |||
if type(v) == "string" then | |||
foundPrototypes[v] = true | |||
end | |||
end | |||
end | |||
local | for name in string.gmatch(id, "[^,]+") do | ||
local n = trim(name) | |||
if n ~= "" then | |||
if componentDefs[n] ~= nil then | |||
foundComponents[n] = true | |||
end | |||
if prototypeDefs[n] ~= nil then | |||
foundPrototypes[n] = true | |||
end | |||
if componentDefs[n] == nil and prototypeDefs[n] == nil then | |||
foundComponents[n] = true | |||
end | |||
end | |||
end | |||
local out = {} | |||
local errors = {} | |||
local keyOrder = {} | |||
local keyToTemplates = {} | |||
function | local function lcfirst(s) | ||
if not s or s == "" then return s end | |||
return string.lower(s:sub(1,1)) .. (s:sub(2) or "") | |||
end | |||
for compName,_ in pairs(foundComponents) do | |||
local compPathName = lcfirst(compName) | |||
local tplPath = "component/" .. compPathName | |||
local content = load_template_content(tplPath) | |||
if not content then | |||
table.insert(errors, "Ошибка: не найден шаблон component/" .. compPathName) | |||
else | |||
local keys = parse_keys_from_template(content) | |||
for _, key in ipairs(keys) do | |||
if not keyToTemplates[key] then | |||
keyToTemplates[key] = {} | |||
table.insert(keyOrder, key) | |||
end | |||
table.insert(keyToTemplates[key], "{{" .. tplPath .. "|title|" .. key .. "}}") | |||
end | |||
end | |||
end | |||
for protoName,_ in pairs(foundPrototypes) do | |||
local protoPathName = lcfirst(protoName) | |||
local tplPath = "prototype/" .. protoPathName | |||
local content = load_template_content(tplPath) | |||
if not content then | |||
table.insert(errors, "Ошибка: не найден шаблон prototype/" .. protoPathName) | |||
else | |||
local keys = parse_keys_from_template(content) | |||
for _, key in ipairs(keys) do | |||
if not keyToTemplates[key] then | |||
keyToTemplates[key] = {} | |||
table.insert(keyOrder, key) | |||
end | |||
table.insert(keyToTemplates[key], "{{" .. tplPath .. "|title|" .. key .. "}}") | |||
end | |||
end | |||
end | |||
for _, e in ipairs(errors) do | |||
table.insert(out, "<div class=\"error\">" .. mw.text.encode(e) .. "</div>") | |||
end | |||
for _, key in ipairs(keyOrder) do | |||
table.insert(out, "<h2>" .. mw.text.encode(key) .. "</h2>") | |||
for _, tplCall in ipairs(keyToTemplates[key]) do | |||
table.insert(out, tplCall) | |||
end | |||
end | |||
return frame:preprocess(table.concat(out, "\n\n")) | |||
end | end | ||
return p | return p | ||