Модуль:Prototypes/Хранилище/Предмет: различия между версиями
Pok (обсуждение | вклад) Нет описания правки |
Pok (обсуждение | вклад) Нет описания правки |
||
| Строка 1: | Строка 1: | ||
local p = {} | local p = {} | ||
-- Кэш данных | -- Кэш для данных и предварительно обработанных структур | ||
local | local cache = { | ||
data = nil, | |||
allSelectors = nil, | |||
indexedData = {} | |||
} | |||
-- | -- Функции как локальные переменные | ||
local loadData, findDataById, formatContent, getContentsOutput, processNestedSelectors, getTableOutput, getContainedOutput, getChemOutput, handleGroupSelector, handleAllSelector, handleNestedSelector | local loadData, findDataById, formatContent, getContentsOutput, processNestedSelectors, getTableOutput, getContainedOutput, getChemOutput, handleGroupSelector, handleAllSelector, handleNestedSelector, indexData, processRolls | ||
-- | -- Загрузка и кэширование данных | ||
loadData = function(filePath) | loadData = function(filePath) | ||
if | if cache[filePath] then | ||
return | return cache[filePath] | ||
end | end | ||
local page = mw.title.new(filePath) | local page = mw.title.new(filePath) | ||
local content = page:getContent() | local content = page:getContent() | ||
local | local data = content and mw.text.jsonDecode(content) or nil | ||
cache[filePath] = data | |||
return | |||
return data | |||
end | end | ||
-- | -- Индексация данных для быстрого доступа | ||
indexData = function(data) | |||
cache.indexedData = {} | |||
if data then | if not data then return end | ||
for _, item in ipairs(data) do | |||
if item.id then | |||
cache.indexedData[item.id] = item | |||
end | end | ||
end | end | ||
end | end | ||
-- Поиск данных по ID | -- Поиск данных по ID (с использованием индексации) | ||
findDataById = function( | findDataById = function(data, id) | ||
return | if not data or not id then return nil end | ||
return cache.indexedData[id] or nil | |||
end | end | ||
-- Форматирование содержимого | -- Форматирование одного содержимого | ||
formatContent = function(content) | formatContent = function(content) | ||
if not content.id then | if not content.id then | ||
| Строка 46: | Строка 53: | ||
local prob = "" | local prob = "" | ||
-- Преобразование веса в вероятность | |||
if content.weight then | if content.weight then | ||
content.prob = content.weight / 100 | content.prob = content.weight / 100 | ||
| Строка 52: | Строка 60: | ||
if content.prob then | if content.prob then | ||
local percentage = content.prob * 100 | local percentage = content.prob * 100 | ||
prob = string.format(" <div>% | prob = percentage >= 1 and string.format(" <div>%d%%</div>", math.floor(percentage)) | ||
or string.format(" <div>%g%%</div>", percentage) | |||
end | end | ||
| Строка 61: | Строка 70: | ||
end | end | ||
-- Получение содержимого | -- Получение содержимого | ||
getContentsOutput = function(contents) | getContentsOutput = function(contents) | ||
local result = {} | local result = {} | ||
| Строка 87: | Строка 96: | ||
-- Обработка таблиц | -- Обработка таблиц | ||
getTableOutput = function(tableId) | getTableOutput = function(tableId) | ||
cache.allSelectors = cache.allSelectors or loadData('User:IanComradeBot/prototypes/AllSelector.json') | |||
local tableData = findDataById( | local tableData = findDataById(cache.allSelectors, tableId) | ||
if not tableData then return 'Таблица не найдена.' end | if not tableData then return 'Таблица не найдена.' end | ||
| Строка 102: | Строка 111: | ||
-- Формирование списка содержащихся предметов или таблиц | -- Формирование списка содержащихся предметов или таблиц | ||
getContainedOutput = function(data, id | getContainedOutput = function(data, id) | ||
local item = findDataById(data, id) | local item = findDataById(data, id) | ||
if not item then return '' end | if not item then return '' end | ||
local result = | local result = {} | ||
if item.StorageFill and item.StorageFill.contents then | if item.StorageFill and item.StorageFill.contents then | ||
result | table.insert(result, getContentsOutput(item.StorageFill.contents)) | ||
elseif item.EntityTableContainerFill and item.EntityTableContainerFill.containers then | elseif item.EntityTableContainerFill and item.EntityTableContainerFill.containers then | ||
local containers = item.EntityTableContainerFill.containers | local containers = item.EntityTableContainerFill.containers | ||
if containers.entity_storage then | if containers.entity_storage then | ||
local children = containers.entity_storage.children | local children = containers.entity_storage.children | ||
if children then | if children then | ||
for _, child in ipairs(children) do | for _, child in ipairs(children) do | ||
if child["!type"] == "NestedSelector" and child.tableId then | if child["!type"] == "NestedSelector" and child.tableId then | ||
result | table.insert(result, handleNestedSelector(child, false)) | ||
elseif child["!type"] == "GroupSelector" then | elseif child["!type"] == "GroupSelector" then | ||
result | table.insert(result, handleGroupSelector(child)) | ||
elseif child["!type"] == "AllSelector" then | elseif child["!type"] == "AllSelector" then | ||
result | table.insert(result, handleAllSelector(child)) | ||
elseif child.id then | elseif child.id then | ||
result | table.insert(result, formatContent(child)) | ||
end | end | ||
end | end | ||
end | end | ||
if containers.entity_storage.tableId then | if containers.entity_storage.tableId then | ||
result | table.insert(result, getTableOutput(containers.entity_storage.tableId)) | ||
end | end | ||
end | end | ||
if containers.storagebase and containers.storagebase.tableId then | if containers.storagebase and containers.storagebase.tableId then | ||
result | table.insert(result, getTableOutput(containers.storagebase.tableId)) | ||
end | end | ||
end | end | ||
return result | return table.concat(result) | ||
end | end | ||
-- Обработка AllSelector | -- Обработка AllSelector | ||
handleAllSelector = function(allSelector) | handleAllSelector = function(allSelector) | ||
local result = | local result = {} | ||
if not allSelector.children then return | if not allSelector.children then return '' end | ||
for _, child in ipairs(allSelector.children) do | for _, child in ipairs(allSelector.children) do | ||
if child["!type"] == "NestedSelector" and child.tableId then | if child["!type"] == "NestedSelector" and child.tableId then | ||
result | table.insert(result, handleNestedSelector(child, true)) | ||
elseif child.id then | elseif child.id then | ||
result | table.insert(result, formatContent(child)) | ||
end | end | ||
end | end | ||
return result | return table.concat(result) | ||
end | end | ||
-- Обработка GroupSelector | -- Обработка GroupSelector | ||
handleGroupSelector = function(groupSelector) | handleGroupSelector = function(groupSelector) | ||
local result = '' | local result = '' | ||
-- Проверка на существование children | -- Проверка на существование children | ||
if not groupSelector.children then | if not groupSelector.children then | ||
return "" | return "" | ||
end | end | ||
local weight = groupSelector.weight or "default" | local weight = groupSelector.weight or "default" | ||
for _, child in ipairs(groupSelector.children) do | |||
if child["!type"] == "GroupSelector" then | |||
result = result .. handleGroupSelector(child) | |||
elseif child["!type"] == "AllSelector" then | |||
result = result .. '<div class="AllSelector">' | |||
result = result .. handleAllSelector(child) | |||
result = result .. '</div>' | |||
elseif child.id then | |||
result = result .. formatContent(child) | |||
else | |||
result = result .. "<div>Ошибка: отсутствует id у элемента.</div>" | |||
end | end | ||
end | end | ||
-- Если weight не равен "default", оборачиваем в div | |||
if weight ~= "default" then | |||
result = string.format('<div class="together" id="%s">%s</div>', weight, result) | |||
end | |||
return result | return result | ||
end | end | ||
-- Обработка NestedSelector | -- Обработка NestedSelector | ||
handleNestedSelector = function(nestedSelector, wrapped) | handleNestedSelector = function(nestedSelector, wrapped) | ||
local tableId = nestedSelector.tableId | local tableId = nestedSelector.tableId | ||
if not tableId then return | if not tableId then return '' end | ||
if wrapped then | if wrapped then | ||
local classes = {} | local classes = {} | ||
if nestedSelector.rolls and nestedSelector.rolls.range then | if nestedSelector.rolls and nestedSelector.rolls.range then | ||
table.insert(classes, 'rolls-' .. tostring(nestedSelector.rolls.range):gsub(',', '-')) | table.insert(classes, 'rolls-' .. tostring(nestedSelector.rolls.range):gsub(',', '-')) | ||
| Строка 232: | Строка 215: | ||
table.insert(classes, 'prob-' .. tostring(math.floor(nestedSelector.prob * 100)) .. 'p') | table.insert(classes, 'prob-' .. tostring(math.floor(nestedSelector.prob * 100)) .. 'p') | ||
end | end | ||
return string.format('<div class="%s">%s</div>', table.concat(classes, ' '), getTableOutput(tableId)) | |||
else | else | ||
return getTableOutput(tableId) | |||
end | end | ||
end | end | ||
-- Формирование списка химии | -- Формирование списка химии | ||
getChemOutput = function(data, id) | getChemOutput = function(data, id) | ||
local item = findDataById(data, id) | local item = findDataById(data, id) | ||
if not item then return '' end | if not item then return '' end | ||
local solutions = item.SolutionContainerManager and item.SolutionContainerManager.solutions | local solutions = item.SolutionContainerManager and item.SolutionContainerManager.solutions | ||
if not solutions then return '' end | if not solutions then return '' end | ||
local result = '<ul | local result = {} | ||
table.insert(result, '<ul>') | |||
for _, solution in pairs(solutions) do | for _, solution in pairs(solutions) do | ||
for _, reagent in ipairs(solution.reagents) do | for _, reagent in ipairs(solution.reagents) do | ||
result | table.insert(result, string.format('<li>[[Химия#chem_%s|%s]] (%d ед.)</li>', reagent.ReagentId, reagent.ReagentId, reagent.Quantity)) | ||
end | end | ||
end | end | ||
result | table.insert(result, '</ul>') | ||
return result | return table.concat(result) | ||
end | end | ||
-- Обработка rolls | -- Обработка rolls | ||
processRolls = function(rolls) | processRolls = function(rolls) | ||
if rolls.range then | if rolls.range then | ||
local min, max = rolls.range:match("(%d+),%s*(%d+)") | local min, max = rolls.range:match("(%d+),%s*(%d+)") | ||
min, max = tonumber(min), tonumber(max) | min, max = tonumber(min), tonumber(max) | ||
return min and max and string.format('Максимальное количество предметов: от %d до %d.', min, max) or 'Некорректный формат для range.' | |||
elseif rolls.value then | elseif rolls.value then | ||
return string.format('Будет сгенерировано %d предметов.', rolls.value) | |||
else | else | ||
return 'Не указан параметр rolls.' | |||
end | end | ||
end | end | ||
| Строка 292: | Строка 261: | ||
if not id then return 'Не указан ID.' end | if not id then return 'Не указан ID.' end | ||
cache.data = cache.data or loadData('User:IanComradeBot/prototypes/fills/Item.json') | |||
if not cache.data then return 'Не удалось загрузить данные.' end | |||
indexData(cache.data) | |||
if mode == 'framing' then | if mode == 'framing' then | ||
local subMode = frame.args[2] | local subMode = frame.args[2] | ||
id = frame.args[3] | |||
if not id then | if not id then | ||
| Строка 306: | Строка 275: | ||
if subMode == 'chem' then | if subMode == 'chem' then | ||
return frame:preprocess('{{СollapsibleMenu|' .. getChemOutput( | return frame:preprocess('{{СollapsibleMenu|' .. getChemOutput(cache.data, id) .. '}}') | ||
elseif subMode == 'contained' then | elseif subMode == 'contained' then | ||
return frame:preprocess('{{СollapsibleMenu|' .. getContainedOutput( | return frame:preprocess('{{СollapsibleMenu|' .. getContainedOutput(cache.data, id) .. '}}') | ||
else | else | ||
return 'Неизвестный подрежим для framing: ' .. subMode | return 'Неизвестный подрежим для framing: ' .. subMode | ||
| Строка 315: | Строка 284: | ||
if mode == 'chem' then | if mode == 'chem' then | ||
return frame:preprocess(getChemOutput( | return frame:preprocess(getChemOutput(cache.data, id)) | ||
elseif mode == 'contained' then | elseif mode == 'contained' then | ||
return frame:preprocess(getContainedOutput( | return frame:preprocess(getContainedOutput(cache.data, id)) | ||
elseif mode == 'rolls' then | elseif mode == 'rolls' then | ||
local entity = findDataById( | local entity = findDataById(cache.data, id) | ||
if not entity then return 'ID не найден в данных.' end | if not entity then return 'ID не найден в данных.' end | ||