Модуль:Сущность/data: различия между версиями
Pok (обсуждение | вклад) Нет описания правки |
Pok (обсуждение | вклад) Нет описания правки |
||
| (не показаны 4 промежуточные версии этого же участника) | |||
| Строка 14: | Строка 14: | ||
if not s then return s end | if not s then return s end | ||
return (s:gsub("^%s*(.-)%s*$", "%1")) | return (s:gsub("^%s*(.-)%s*$", "%1")) | ||
end | |||
local function each_csv_value(str, fn) | |||
if not str or str == "" then | |||
return | |||
end | |||
for item in string.gmatch(str, "[^,]+") do | |||
local value = trim(item) | |||
if value ~= "" then | |||
fn(value) | |||
end | |||
end | |||
end | end | ||
| Строка 162: | Строка 174: | ||
local function makeSourceLink(s) | local function makeSourceLink(s) | ||
local className = | local className = | ||
(s.name:sub(1,1):upper() .. s.name:sub(2)) .. | (s.name:sub(1, 1):upper() .. s.name:sub(2)) .. | ||
(s.kind and (s.kind:sub(1,1):upper() .. s.kind:sub(2)) or "") | (s.kind and (s.kind:sub(1, 1):upper() .. s.kind:sub(2)) or "") | ||
local tplLabel = "Template:" .. s.tplPath | local tplLabel = "Template:" .. s.tplPath | ||
| Строка 195: | Строка 207: | ||
end | end | ||
return table.concat(parts, "\n") | return table.concat(parts, "\n") | ||
end | |||
local function split_title_key(key) | |||
local main, sub = (key or ""):match("^([^_]+)_(.+)$") | |||
if main and sub then | |||
return main, sub | |||
end | |||
return key, nil | |||
end | |||
local function renderGroupedTitleBlocks(frame, keyOrder, keyToTemplates, noHeaders, showSource) | |||
local groups = {} | |||
local groupOrder = {} | |||
for _, key in ipairs(keyOrder or {}) do | |||
local mainTitle, subTitle = split_title_key(key) | |||
if mainTitle and mainTitle ~= "" then | |||
local group = groups[mainTitle] | |||
if not group then | |||
group = { blocks = {} } | |||
groups[mainTitle] = group | |||
groupOrder[#groupOrder + 1] = mainTitle | |||
end | |||
group.blocks[#group.blocks + 1] = { | |||
subTitle = subTitle, | |||
entries = keyToTemplates[key] or {} | |||
} | |||
end | |||
end | |||
local out = {} | |||
for _, mainTitle in ipairs(groupOrder) do | |||
local group = groups[mainTitle] | |||
local parts = {} | |||
if not noHeaders then | |||
parts[#parts + 1] = "<h2>" .. mw.text.encode(mainTitle) .. "</h2>" | |||
end | |||
for _, block in ipairs(group.blocks or {}) do | |||
local tplCalls, sources = collect_tpl_calls(block.entries or {}) | |||
local blockText = renderTitleBlock(block.subTitle or mainTitle, tplCalls, sources, false, frame, showSource) | |||
if blockText ~= "" then | |||
if block.subTitle and not noHeaders then | |||
parts[#parts + 1] = "<h3>" .. mw.text.encode(block.subTitle) .. "</h3>" | |||
end | |||
parts[#parts + 1] = blockText | |||
end | |||
end | |||
if #parts > 0 then | |||
out[#out + 1] = table.concat(parts, "\n") | |||
end | |||
end | |||
return table.concat(out, "\n") | |||
end | |||
local function matches_card_list(list, callKey, compositeKey) | |||
if not list then | |||
return false | |||
end | |||
return list[callKey] or list[compositeKey] or false | |||
end | end | ||
| Строка 246: | Строка 322: | ||
end | end | ||
local function cardWrapper(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders) | local function cardWrapper(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders, cardFilter) | ||
local merged = { | local merged = { | ||
sections = {}, | sections = {}, | ||
| Строка 257: | Строка 333: | ||
tagSet = {} | tagSet = {} | ||
} | } | ||
local rawContentParts = {} | |||
for _, callKey in ipairs(keyOrder or {}) do | for _, callKey in ipairs(keyOrder or {}) do | ||
local entries = keyToTemplates[callKey] or {} | local entries = keyToTemplates[callKey] or {} | ||
| Строка 271: | Строка 348: | ||
local section = (callKey:find("_", 1, true)) and callKey:match("^([^_]+)") or "Сущность" | local section = (callKey:find("_", 1, true)) and callKey:match("^([^_]+)") or "Сущность" | ||
if displayLabel ~= "" or content ~= "" then | local isWhitelisted = cardFilter and matches_card_list(cardFilter.whitelist, callKey, compositeKey) or | ||
false | |||
local isBlacklisted = cardFilter and matches_card_list(cardFilter.blacklist, callKey, compositeKey) or | |||
false | |||
if isWhitelisted and content ~= "" then | |||
rawContentParts[#rawContentParts + 1] = content | |||
end | |||
local allowCardEntry = not isWhitelisted and not isBlacklisted and | |||
((not cardFilter) or (not cardFilter.hasWhitelist) or | |||
matches_card_list(cardFilter.cardWhitelist, callKey, compositeKey)) | |||
if allowCardEntry and (displayLabel ~= "" or content ~= "") then | |||
if not merged.sectionsMap[section] then | if not merged.sectionsMap[section] then | ||
merged.sectionsMap[section] = true | merged.sectionsMap[section] = true | ||
| Строка 297: | Строка 386: | ||
end | end | ||
if tagText ~= "" then | if allowCardEntry and tagText ~= "" then | ||
if not merged.tagSet[tagText] then | if not merged.tagSet[tagText] then | ||
merged.tagSet[tagText] = true | merged.tagSet[tagText] = true | ||
| Строка 306: | Строка 395: | ||
end | end | ||
end | end | ||
each_csv_value(frame.args.cardTag or "", function(extraTag) | |||
if not merged.tagSet[extraTag] then | |||
merged.tagSet[extraTag] = true | |||
table.insert(merged.tags, extraTag) | |||
end | |||
end) | |||
local out = {} | |||
if #rawContentParts > 0 then | |||
out[#out + 1] = table.concat(rawContentParts, "\n") | |||
end | |||
local cardCall = buildCardCall(merged, entityId) | |||
if noHeaders then | if noHeaders then | ||
local hasLabel = false | local hasLabel = false | ||
for _, v in pairs(merged.labelOverrides or {}) do | for _, v in pairs(merged.labelOverrides or {}) do | ||
if v and v ~= "" then hasLabel = true break end | if v and v ~= "" then | ||
hasLabel = true | |||
break | |||
end | |||
end | end | ||
if not hasLabel then | if not hasLabel then | ||
for _, lst in pairs(merged.labelLists or {}) do | for _, lst in pairs(merged.labelLists or {}) do | ||
if #lst > 0 then hasLabel = true break end | if #lst > 0 then | ||
hasLabel = true | |||
break | |||
end | |||
end | end | ||
end | end | ||
| Строка 320: | Строка 429: | ||
local hasContent = false | local hasContent = false | ||
for _, v in pairs(merged.contentByKey or {}) do | for _, v in pairs(merged.contentByKey or {}) do | ||
if v and v ~= "" then hasContent = true break end | if v and v ~= "" then | ||
hasContent = true | |||
break | |||
end | |||
end | end | ||
if not hasLabel and not hasContent then | if not hasLabel and not hasContent then | ||
return "" | return table.concat(out, "\n") | ||
end | end | ||
end | end | ||
return | if cardCall ~= "" then | ||
out[#out + 1] = cardCall | |||
end | |||
return table.concat(out, "\n") | |||
end | end | ||
| Строка 361: | Строка 477: | ||
end | end | ||
end, | end, | ||
render_full = function(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders) | render_full = function(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders, showSource, cardFilter) | ||
return cardWrapper(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders) | return cardWrapper(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders, cardFilter) | ||
end | end | ||
}) | }) | ||
register_switch_mode("title", { | register_switch_mode("title", { | ||
full = true, | |||
build_entry = function(ctx, key) | build_entry = function(ctx, key) | ||
return { | return { | ||
| Строка 381: | Строка 498: | ||
} | } | ||
end, | end, | ||
render_full = function(frame, keyOrder, keyToTemplates, keySources, entityId, noHeaders, showSource) | |||
return renderGroupedTitleBlocks(frame, keyOrder, keyToTemplates, noHeaders, showSource) | |||
return | |||
end | end | ||
}) | }) | ||
| Строка 443: | Строка 559: | ||
filter.blacklist = parseListArg(args.blacklist or "") | filter.blacklist = parseListArg(args.blacklist or "") | ||
filter.whitelist = parseListArg(args.whitelist or "") | filter.whitelist = parseListArg(args.whitelist or "") | ||
filter.hasWhitelist = next(filter.whitelist) ~= nil | filter.hasWhitelist = false | ||
for _, sw in ipairs(switchModeOrder) do | |||
if filter.whitelist[sw] and next(filter.whitelist[sw]) ~= nil then | |||
filter.hasWhitelist = true | |||
break | |||
end | |||
end | |||
if not filter.hasWhitelist and filter.whitelist.cardContent and next(filter.whitelist.cardContent) ~= nil then | |||
filter.hasWhitelist = true | |||
end | |||
return filter | return filter | ||
end | |||
local function build_render_options(filter) | |||
return { | |||
noHeaders = false, | |||
cardFilter = { | |||
hasWhitelist = filter.whitelist.cardContent and next(filter.whitelist.cardContent) ~= nil or false, | |||
cardWhitelist = filter.whitelist.card or {}, | |||
blacklist = filter.blacklist.cardContent or {}, | |||
whitelist = filter.whitelist.cardContent or {} | |||
} | |||
} | |||
end | end | ||
local function should_include_key(filter, sw, key) | local function should_include_key(filter, sw, key) | ||
if filter.hasWhitelist then | if filter.hasWhitelist then | ||
if filter.whitelist[sw] and filter.whitelist[sw][key] then | |||
return true | |||
end | |||
if sw == "card" and filter.whitelist.cardContent and filter.whitelist.cardContent[key] then | |||
return true | |||
end | |||
return false | |||
end | end | ||
return not (filter.blacklist[sw] and filter.blacklist[sw][key]) | return not (filter.blacklist[sw] and filter.blacklist[sw][key]) | ||
end | end | ||
local function each_csv_value(str, | local function parse_csv_set(str) | ||
if not | local res = {} | ||
each_csv_value(str, function(name) | |||
res[name] = true | |||
end) | |||
return res | |||
end | |||
local function apply_entity_set_filters(foundSet, whitelistSet, blacklistSet) | |||
local hasWhitelist = next(whitelistSet or {}) ~= nil | |||
if hasWhitelist then | |||
for name in pairs(foundSet) do | |||
if not whitelistSet[name] then | |||
foundSet[name] = nil | |||
end | |||
end | |||
end | end | ||
for | |||
for name in pairs(blacklistSet or {}) do | |||
foundSet[name] = nil | |||
end | end | ||
end | end | ||
local function collect_entity_sets(id, componentDefs, prototypeStoreDefs, | local function collect_entity_sets(id, componentDefs, prototypeStoreDefs, | ||
componentWhitelist, componentBlacklist, prototypeWhitelist, prototypeBlacklist) | |||
local foundComponents, foundPrototypes = {}, {} | local foundComponents, foundPrototypes = {}, {} | ||
| Строка 488: | Строка 644: | ||
end) | end) | ||
apply_entity_set_filters(foundComponents, parse_csv_set(componentWhitelist), parse_csv_set(componentBlacklist)) | |||
apply_entity_set_filters(foundPrototypes, parse_csv_set(prototypeWhitelist), parse_csv_set(prototypeBlacklist)) | |||
return foundComponents, foundPrototypes | return foundComponents, foundPrototypes | ||
| Строка 535: | Строка 686: | ||
flattenExtraCache[cacheKey] = extra | flattenExtraCache[cacheKey] = extra | ||
return extra | return extra | ||
end | |||
local function add_card_tag_value(tags, seen, value) | |||
value = trim(value or "") | |||
if value == "" or seen[value] then | |||
return | |||
end | |||
seen[value] = true | |||
tags[#tags + 1] = value | |||
end | |||
local function merge_card_tag_text(...) | |||
local tags = {} | |||
local seen = {} | |||
for i = 1, select("#", ...) do | |||
each_csv_value(select(i, ...), function(value) | |||
add_card_tag_value(tags, seen, value) | |||
end) | |||
end | |||
return table.concat(tags, ", ") | |||
end | end | ||
| Строка 567: | Строка 740: | ||
end | end | ||
local function | local function each_entity_data(frame, id, onEntity, onMissing) | ||
local | local componentWhitelist = frame.args.componentWhitelist or frame.args.componentwhitelist or "" | ||
local componentBlacklist = frame.args.componentBlacklist or frame.args.componentblacklist or "" | |||
local prototypeWhitelist = frame.args.prototypeWhitelist or frame.args.prototypewhitelist or "" | |||
local prototypeBlacklist = frame.args.prototypeBlacklist or frame.args.prototypeblacklist or "" | |||
local | |||
local | |||
local componentDefs = load_module_data("component.json") | local componentDefs = load_module_data("component.json") | ||
local prototypeStoreDefs = load_module_data("prototype_store.json") | local prototypeStoreDefs = load_module_data("prototype_store.json") | ||
if not componentDefs or not prototypeStoreDefs then return | if not componentDefs or not prototypeStoreDefs then | ||
return false | |||
end | |||
local foundComponents, foundPrototypes = collect_entity_sets(id, componentDefs, prototypeStoreDefs | local foundComponents, foundPrototypes = collect_entity_sets(id, componentDefs, prototypeStoreDefs, | ||
componentWhitelist, componentBlacklist, prototypeWhitelist, prototypeBlacklist) | |||
local ok, dp = pcall(require, "Module:GetField") | local ok, dp = pcall(require, "Module:GetField") | ||
local function processEntity(kind, name, isStore) | local function processEntity(kind, name, isStore) | ||
| Строка 634: | Строка 766: | ||
local content = load_template_content(tplPath) | local content = load_template_content(tplPath) | ||
if not content then | if not content then | ||
if | if onMissing then | ||
onMissing(kind, name, isStore, tplPath) | |||
end | end | ||
return | return | ||
end | end | ||
| Строка 653: | Строка 784: | ||
end | end | ||
onEntity(parsed, { | |||
tplPath = tplPath, | tplPath = tplPath, | ||
id = id, | id = id, | ||
| Строка 659: | Строка 790: | ||
source = make_source(kind, name, pathName, tplPath), | source = make_source(kind, name, pathName, tplPath), | ||
priority = resolve_priority(parsed) | priority = resolve_priority(parsed) | ||
} | }) | ||
end | end | ||
| Строка 687: | Строка 818: | ||
end | end | ||
end | end | ||
return true | |||
end | |||
local function collect_card_tag_text(frame, args, id) | |||
local filter = build_key_filter(args) | |||
local cardFilter = build_render_options(filter).cardFilter | |||
local entries = {} | |||
local ok = each_entity_data(frame, id, function(parsed, ctx) | |||
local keys = parsed.card or {} | |||
if type(keys) == "table" then | |||
for _, key in ipairs(keys) do | |||
local compositeKey = (key:find("_", 1, true)) and key or ("Сущность_" .. key) | |||
local isWhitelisted = matches_card_list(cardFilter.whitelist, key, compositeKey) | |||
local isBlacklisted = matches_card_list(cardFilter.blacklist, key, compositeKey) | |||
local allowCardEntry = not isWhitelisted and not isBlacklisted and | |||
((not cardFilter.hasWhitelist) or matches_card_list(cardFilter.cardWhitelist, key, compositeKey)) | |||
if allowCardEntry then | |||
entries[#entries + 1] = { | |||
tplTag = makeTplCall(ctx.tplPath, "cardTag", key, ctx.id, ctx.extra), | |||
priority = ctx.priority, | |||
idx = #entries + 1 | |||
} | |||
end | |||
end | |||
end | |||
if parsed.cardTag and parsed.cardTag ~= "" then | |||
entries[#entries + 1] = { | |||
tplTag = makeTplCall(ctx.tplPath, "cardTag", "cardTag", ctx.id, ctx.extra), | |||
priority = ctx.priority, | |||
idx = #entries + 1 | |||
} | |||
end | |||
end) | |||
if not ok then | |||
return trim(args.tag or "") | |||
end | |||
sort_entries_by_priority(entries) | |||
local tags = {} | |||
local seen = {} | |||
for _, entry in ipairs(entries) do | |||
local tagText = trim(frame:preprocess(entry.tplTag or "") or "") | |||
add_card_tag_value(tags, seen, tagText) | |||
end | |||
return merge_card_tag_text(table.concat(tags, ", "), args.tag) | |||
end | |||
p.mergeCardTagText = merge_card_tag_text | |||
p.collectCardTagText = collect_card_tag_text | |||
local function build_missing_template_error(kind, name, isStore, tplPath) | |||
local baseType = (kind and (kind:sub(1, 1):upper() .. kind:sub(2)) or "") | |||
local classType = baseType | |||
if isStore then | |||
classType = classType .. "Store" | |||
end | |||
local className = name .. baseType | |||
local tplLabel = "Template:" .. tplPath | |||
return "{{сущность/infobox/base|тип=" .. classType .. "|название=" .. className .. "|ссылка=" .. tplLabel .. "}}" | |||
end | |||
local function renderBlocks(frame, state, renderOptions, entityId, showSource) | |||
local outLocal = {} | |||
local noHeaders = renderOptions and renderOptions.noHeaders | |||
local cardFilter = renderOptions and renderOptions.cardFilter | |||
for _, sw in ipairs(switchModeOrder) do | |||
local mode = switchModeRegistry[sw] or {} | |||
if mode.full then | |||
local outStr = "" | |||
if type(mode.render_full) == "function" then | |||
outStr = mode.render_full(frame, state.keyOrder[sw], state.keyToTemplates[sw], state.keySources[sw], | |||
entityId, noHeaders, showSource, cardFilter) | |||
end | |||
if outStr and outStr ~= "" then table.insert(outLocal, outStr) end | |||
else | |||
for _, key in ipairs(state.keyOrder[sw] or {}) do | |||
local entries = state.keyToTemplates[sw][key] or {} | |||
if type(mode.render_key) == "function" then | |||
local outStr = mode.render_key(frame, key, entries, noHeaders, showSource) | |||
if outStr and outStr ~= "" then table.insert(outLocal, outStr) end | |||
end | |||
end | |||
end | |||
end | |||
return outLocal | |||
end | |||
function p.get(frame) | |||
local args = getArgs(frame, { removeBlanks = false }) | |||
local id = args[1] or "" | |||
if id == "" then return "" end | |||
local showSource = trim(args.showSource or "") == "" | |||
local filter = build_key_filter(args) | |||
local renderOptions = build_render_options(filter) | |||
local state = new_switch_state() | |||
local errors = {} | |||
local ok = each_entity_data(frame, id, function(parsed, ctx) | |||
add_entries_from_meta(state, parsed, ctx, filter, false) | |||
end, function(kind, name, isStore, tplPath) | |||
if not filter.hasWhitelist then | |||
errors[#errors + 1] = build_missing_template_error(kind, name, isStore, tplPath) | |||
end | |||
end) | |||
if not ok then return "" end | |||
local out = {} | local out = {} | ||
| Строка 694: | Строка 940: | ||
end | end | ||
local blocks = renderBlocks(frame, state, | renderOptions.noHeaders = filter.hasWhitelist | ||
local blocks = renderBlocks(frame, state, renderOptions, id, showSource) | |||
for _, b in ipairs(blocks) do | for _, b in ipairs(blocks) do | ||
table.insert(out, b) | table.insert(out, b) | ||
| Строка 708: | Строка 956: | ||
local showSource = trim(args.nosource or "") == "" | local showSource = trim(args.nosource or "") == "" | ||
local previewFilter = build_key_filter(args) | |||
local renderOptions = build_render_options(previewFilter) | |||
local content = load_template_content(tplPath) | local content = load_template_content(tplPath) | ||
| Строка 728: | Строка 978: | ||
}, nil, true) | }, nil, true) | ||
local whitelist = | local whitelist = previewFilter.whitelist | ||
local hasWhitelist = | local hasWhitelist = previewFilter.hasWhitelist | ||
renderOptions.noHeaders = hasWhitelist | |||
local out = {} | local out = {} | ||
local blocks = renderBlocks(frame, state, | local blocks = renderBlocks(frame, state, renderOptions, "", showSource) | ||
for _, b in ipairs(blocks) do | for _, b in ipairs(blocks) do | ||
table.insert(out, b) | table.insert(out, b) | ||