|
|
| Строка 1: |
Строка 1: |
| -- Загрузка данных | | -- Загрузка данных |
| local itemData = mw.loadData("Модуль:IanComradeBot/prototypes/fills/Item.json/data") | | local latheData = mw.loadData("Модуль:IanComradeBot/prototypes/lathe.json/data") -- Функция для загрузки данных станков |
| local itemSlotsData = mw.loadData("Модуль:IanComradeBot/prototypes/ItemSlots.json/data") | | local recipeData = mw.loadData("Модуль:IanComradeBot/prototypes/lathe/recipes.json/data") -- Функция для загрузки данных рецептов |
| local chemData = mw.loadData("Модуль:IanComradeBot/prototypes/fills/chem.json/data") | | local researchData = mw.loadData("Модуль:IanComradeBot/prototypes/research.json/data") -- Функция для загрузки данных исследований |
| | local materialData = mw.loadData("Модуль:IanComradeBot/prototypes/materials.json/data") -- Функция для загрузки данных материалов |
| | local chemData = mw.loadData("Модуль:IanComradeBot/chem prototypes.json/data") -- Функция для загрузки данных химических веществ |
|
| |
|
| local p = {} | | local p = {} |
|
| |
|
| -- Функция для загрузки данных | | -- Функция для форматирования времени |
| local loadData = function(filePath) | | local function format_seconds_to_short_string(input_seconds) |
| local page = mw.title.new(filePath)
| | local minutes = math.floor(input_seconds / 60) |
| local content = page:getContent()
| | local seconds = input_seconds % 60 |
| return content and mw.text.jsonDecode(content) or nil
| |
| end
| |
|
| |
|
| -- Функция processRolls для преобразования диапазона
| | local minutes_part = minutes > 0 and (minutes .. " мин.") or nil |
| local processRolls = function(rolls) | | local seconds_part = seconds > 0 and (seconds .. " сек.") or nil |
| local result = ""
| |
| if rolls and rolls.range then
| |
| -- Если указан range
| |
| local min, max = rolls.range:match("(%d+),%s*(%d+)")
| |
| min, max = tonumber(min), tonumber(max)
| |
| if min and max then
| |
| result = result .. string.format('[%d-%d]', min + 1, max + 1)
| |
| else
| |
| result = result .. 'Некорректный формат для range.'
| |
| end
| |
| elseif rolls and rolls.value then
| |
| -- Если указано value
| |
| result = result .. string.format('[%d]', rolls.value)
| |
| else
| |
| result = result .. 'Не указан параметр rolls.'
| |
| end
| |
| return result
| |
| end
| |
|
| |
|
| -- Функция для поиска первого startingItem в slots
| | if minutes_part and seconds_part then |
| local function getFirstStartingItem(data)
| | return minutes_part .. " " .. seconds_part |
| if not data or not data.ItemSlots or not data.ItemSlots.slots then
| | elseif seconds_part then |
| return nil
| | return seconds_part |
| end
| | elseif minutes_part then |
|
| | return minutes_part |
| for _, slot in pairs(data.ItemSlots.slots) do
| | else |
| if slot.startingItem and slot.startingItem ~= "" then
| | return '0 сек.' |
| return slot.startingItem
| | end |
| end
| |
| end
| |
|
| |
| return nil
| |
| end | | end |
|
| |
|
| | -- Функция для сортировки рецептов |
| | local function sortRecipesByPriority(recipes) |
| | table.sort(recipes, function(a, b) |
| | local priority = { Static = 1, EMAG = 3 } |
| | local aPriority = priority[a.discipline] or 2 |
| | local bPriority = priority[b.discipline] or 2 |
|
| |
|
| -- Локальные функции
| | if a.isEmag ~= b.isEmag then |
| local findDataById, formatContent, getContentsOutput, processNestedSelectors, getTableOutput, getContainedOutput, getChemOutput, handleGroupSelector, handleAllSelector, handleNestedSelector
| | return not a.isEmag |
| | |
| -- Поиск данных по ID через индекс
| |
| findDataById = function(itemDataIndex, id)
| |
| if not itemDataIndex then return nil end
| |
| for _, item in ipairs(itemDataIndex) do
| |
| if item.id == id then | |
| return item | |
| end | | end |
| end
| |
| end
| |
|
| |
| -- Форматирование содержимого
| |
| formatContent = function(content)
| |
| if type(content) == "table" and not content.id then
| |
| return "Ошибка: отсутствует id у элемента."
| |
| end
| |
|
| |
| -- Если передан объект с id
| |
| local id = content.id or content
| |
| local name = string.format('{{#invoke:Entity Lookup|getname|%s}}', id)
| |
| local image = string.format('%s.png', id)
| |
| local amount = (content.amount and content.amount ~= 1) and string.format(" [%d]", content.amount) or ""
| |
| local prob = ""
| |
|
| |
| if content.weight then
| |
| content.prob = content.weight / 100
| |
| end
| |
|
| |
| if content.prob then
| |
| prob = string.format(" <div>%s%%</div>", content.prob * 100 >= 1 and math.floor(content.prob * 100) or content.prob * 100)
| |
| end
| |
|
| |
|
| return string.format(
| | if aPriority == bPriority then |
| '{{LinkСard|SideStyle=1|background-color=#cbcbff0a|image=%s|name=%s%s%s {{#invoke:Песочница/Pok|main|framing|contained|%s}} {{#invoke:Песочница/Pok|main|framing|slot|%s}} {{#invoke:Песочница/Pok|main|framing|chem|%s}} }}',
| | if a.tier == b.tier then |
| image, name, amount, prob, id, id, id
| | return a.discipline < b.discipline |
| )
| |
| end
| |
| | |
| -- Получение содержимого через таблицу
| |
| getContentsOutput = function(contents)
| |
| local result = ""
| |
| for _, content in ipairs(contents) do
| |
| result = result .. formatContent(content)
| |
| end
| |
| return result
| |
| end
| |
| | |
| -- Обработка вложенных таблиц
| |
| processNestedSelectors = function(children, visited)
| |
| visited = visited or {}
| |
| if not children or #children == 0 then return "" end
| |
| | |
| local result = ""
| |
| | |
| for _, child in ipairs(children) do
| |
| if child.id then
| |
| if not visited[child.id] then
| |
| result = result .. formatContent(child)
| |
| end | | end |
| elseif child["!type"] == "NestedSelector" then
| | return a.tier < b.tier |
| result = result .. handleNestedSelector(child, true, visited)
| | end |
| elseif child["!type"] == "GroupSelector" then
| |
| result = result .. handleGroupSelector(child, visited)
| |
| end
| |
| end
| |
|
| |
|
| return result
| | return aPriority < bPriority |
| | end) |
| end | | end |
|
| |
|
| -- Обработка таблиц | | function p.main(frame) |
| getTableOutput = function(tableId, visited)
| | -- Подключение CSS |
| visited = visited or {}
| | local cssLink = frame:extensionTag('templatestyles', '', { |
| | src = 'Шаблон:Prototypes/Машина/Станок/styles.css' |
| | }) |
|
| |
|
| if visited[tableId] then
| | local latheId = frame.args[1] or "" |
| return ''
| | if latheId == "" then |
| end
| | return '<div style="color:red;">Не указан ID станка.</div>' |
| | end |
|
| |
|
| visited[tableId] = true
| | local lathe = nil |
| | for _, data in ipairs(latheData) do |
| | if data.id == latheId then |
| | lathe = data |
| | break |
| | end |
| | end |
|
| |
|
| local tableData = loadData('User:IanComradeBot/prototypes/table.json')
| | if not lathe then |
| local tableDataIndex = findDataById(tableData, tableId)
| | return '<div style="color:red;">Станок с ID "' .. latheId .. '" не найден.</div>' |
| | end |
|
| |
|
| if not tableDataIndex then return 'Таблица не найдена.' end
| | local materialMapping = {} |
| | for _, material in ipairs(materialData) do |
| | materialMapping[material.material.id] = material.material.stackEntity |
| | end |
|
| |
|
| if tableDataIndex['!type:GroupSelector'] then
| | local chemMapping = {} |
| return handleGroupSelector(tableDataIndex['!type:GroupSelector'], visited)
| | for id, chem in pairs(chemData) do |
| elseif tableDataIndex['!type:AllSelector'] then
| | chemMapping[id] = chem.name |
| return processNestedSelectors(tableDataIndex['!type:AllSelector'].children, visited)
| | end |
| end
| |
|
| |
|
| return 'Таблица не содержит элементов.'
| | local out = cssLink |
| end
| | local recipes = {} |
|
| |
|
| -- Формирование списка содержащихся предметов или таблиц
| | local function getRecipeDetails(recipeId) |
| getContainedOutput = function(itemData, id, visited)
| | for _, recipe in ipairs(recipeData) do |
| visited = visited or {}
| | if recipe.id == recipeId then |
| | return recipe.latheRecipe |
| | end |
| | end |
| | return nil |
| | end |
|
| |
|
| if visited[id] then
| | local function findInResearch(recipeId) |
| return ''
| | for _, research in ipairs(researchData) do |
| end
| | if research.technology and research.technology.recipeUnlocks then |
| | for _, unlock in ipairs(research.technology.recipeUnlocks) do |
| | if unlock == recipeId then |
| | return { |
| | name = research.technology.name, |
| | tier = research.technology.tier, |
| | discipline = research.technology.discipline |
| | } |
| | end |
| | end |
| | end |
| | end |
| | return nil |
| | end |
|
| |
|
| visited[id] = true
| | -- Обработка staticRecipes |
| | if lathe.Lathe.staticRecipes then |
| | for _, recipeId in ipairs(lathe.Lathe.staticRecipes) do |
| | local recipe = getRecipeDetails(recipeId) |
| | if recipe and recipe.result then |
| | table.insert(recipes, { |
| | result = recipe.result, |
| | completetime = recipe.completetime, |
| | materials = recipe.materials, |
| | discipline = "Static", |
| | tier = 0 |
| | }) |
| | elseif recipe and recipe.resultReagents then |
| | for reagent, amount in pairs(recipe.resultReagents) do |
| | if type(reagent) == "string" and reagent ~= "category" then |
| | local reagentName = chemMapping[reagent] or reagent |
| | table.insert(recipes, { |
| | result = reagentName .. "|amount=" .. amount .. "ед.|mode-chem=1", |
| | completetime = recipe.completetime, |
| | materials = recipe.materials, |
| | discipline = "Static", |
| | tier = 0 |
| | }) |
| | break |
| | end |
| | end |
| | else |
| | out = out .. '<div style="color:red;">Ошибка: Рецепт с ID "' .. recipeId .. '" не найден или поля result/resultReagents отсутствуют.</div>' |
| | end |
| | end |
| | end |
|
| |
|
| local item = findDataById(itemData, id)
| | -- Обработка dynamicRecipes |
| if not item then return '' end
| | if lathe.Lathe.dynamicRecipes then |
| | for _, recipeId in ipairs(lathe.Lathe.dynamicRecipes) do |
| | local recipe = getRecipeDetails(recipeId) |
| | if recipe then |
| | local researchInfo = findInResearch(recipeId) |
| | if researchInfo then |
| | table.insert(recipes, { |
| | result = recipe.result, |
| | completetime = recipe.completetime, |
| | materials = recipe.materials, |
| | discipline = researchInfo.discipline, |
| | tier = researchInfo.tier, |
| | researchName = researchInfo.name |
| | }) |
| | end |
| | end |
| | end |
| | end |
|
| |
|
| local result = {}
| | -- Обработка emagStaticRecipes |
| | if lathe.EmagLatheRecipes and lathe.EmagLatheRecipes.emagStaticRecipes then |
| | for _, recipeId in ipairs(lathe.EmagLatheRecipes.emagStaticRecipes) do |
| | local recipe = getRecipeDetails(recipeId) |
| | if recipe then |
| | table.insert(recipes, { |
| | result = recipe.result, |
| | completetime = recipe.completetime, |
| | materials = recipe.materials, |
| | discipline = "Static", |
| | tier = 0, |
| | isEmag = true |
| | }) |
| | end |
| | end |
| | end |
|
| |
|
| -- Обработка StorageFill
| | -- Обработка emagDynamicRecipes |
| if item.StorageFill and item.StorageFill.contents then
| | if lathe.EmagLatheRecipes and lathe.EmagLatheRecipes.emagDynamicRecipes then |
| result[#result + 1] = getContentsOutput(item.StorageFill.contents)
| | for _, recipeId in ipairs(lathe.EmagLatheRecipes.emagDynamicRecipes) do |
| | local recipe = getRecipeDetails(recipeId) |
| | if recipe then |
| | local researchInfo = findInResearch(recipeId) |
| | if researchInfo then |
| | table.insert(recipes, { |
| | result = recipe.result, |
| | completetime = recipe.completetime, |
| | materials = recipe.materials, |
| | discipline = researchInfo.discipline, |
| | tier = researchInfo.tier, |
| | researchName = researchInfo.name, |
| | isEmag = true |
| | }) |
| | end |
| | end |
| | end |
| | end |
|
| |
|
| -- Обработка EntityTableContainerFill
| | sortRecipesByPriority(recipes) |
| elseif item.EntityTableContainerFill and item.EntityTableContainerFill.containers then
| |
| local containers = item.EntityTableContainerFill.containers
| |
|
| |
|
| -- Обработка entity_storage
| | -- Таблица для перевода названий дисциплин |
| if containers.entity_storage then
| | local disciplineMapping = { |
| -- Если есть просто элементы с id, то форматируем их как обычные предметы
| | Arsenal = "Арсенал", |
| if containers.entity_storage.children then
| | Industrial = "Промышленность", |
| for _, child in ipairs(containers.entity_storage.children) do
| | Experimental = "Экспериментальное", |
| if child.id then
| | CivilianServices = "Обслуживание персонала" |
| result[#result + 1] = formatContent(child) -- Теперь одиночные id не игнорируются
| | } |
| elseif child["!type"] == "GroupSelector" then
| |
| result[#result + 1] = handleGroupSelector(child, visited)
| |
| elseif child["!type"] == "AllSelector" then
| |
| result[#result + 1] = processNestedSelectors(child.children, visited)
| |
| end
| |
| end
| |
| end
| |
|
| |
|
| -- Если есть таблица, то загружаем её
| | -- Таблица для цветов по уровням |
| if containers.entity_storage.tableId then
| | local tierColors = { |
| result[#result + 1] = getTableOutput(containers.entity_storage.tableId, visited)
| | [1] = "#54d554", |
| end
| | [2] = "#ed9000", |
| end
| | [3] = "#d72a2a" |
| | } |
|
| |
|
| -- Обработка storagebase (аналогично entity_storage)
| | local materialUseMultiplier = lathe.Lathe.materialUseMultiplier or 1 |
| if containers.storagebase then
| | local timeMultiplier = lathe.Lathe.timeMultiplier or 1 |
| if containers.storagebase.children then
| |
| for _, child in ipairs(containers.storagebase.children) do
| |
| if child.id then
| |
| result[#result + 1] = formatContent(child)
| |
| elseif child["!type"] == "GroupSelector" then
| |
| result[#result + 1] = handleGroupSelector(child, visited)
| |
| elseif child["!type"] == "AllSelector" then
| |
| result[#result + 1] = processNestedSelectors(child.children, visited)
| |
| end
| |
| end
| |
| end
| |
|
| |
|
| if containers.storagebase.tableId then
| | for _, recipe in ipairs(recipes) do |
| result[#result + 1] = getTableOutput(containers.storagebase.tableId, visited)
| | local scaledTime = format_seconds_to_short_string(recipe.completetime * timeMultiplier) |
| end
| | out = out .. '{{Шаблон:Prototypes/Машина/Станок|product=' .. recipe.result |
| end
| | out = out .. '|complete-time=' .. scaledTime |
| end
| | out = out .. '|materials=' |
|
| |
|
| return table.concat(result)
| | if next(recipe.materials) then |
| end | | for material, amount in pairs(recipe.materials) do |
| | local stackEntity = materialMapping[material] or material |
| | local scaledAmount = (amount * materialUseMultiplier) / 100 |
| | out = out .. '<b>[[File:' .. stackEntity .. '.png|32x32px|link=]] ' .. scaledAmount .. ' {{#invoke:Entity Lookup|getname|' .. stackEntity .. '}}</b>' |
| | end |
| | else |
| | out = out .. 'Нет данных о материалах' |
| | end |
|
| |
|
| -- Обработка AllSelector | | -- Информация об исследовании |
| handleAllSelector = function(allSelector)
| | if recipe.discipline ~= "Static" then |
| if not allSelector.children then return '' end
| | local tierColor = tierColors[recipe.tier] or "#FFFFFF" |
| return processNestedSelectors(allSelector.children)
| | local disciplineName = disciplineMapping[recipe.discipline] or "Неизвестная дисциплина" |
| end
| |
|
| |
|
| -- Обработка GroupSelector
| | out = out .. '|info=<div style="font-weight:600;"><span style="margin:8px;">[[File:' .. recipe.discipline .. '.png|16x16px|link=]]</span> [[Руководство по исследованию и разработке|' .. disciplineName |
| handleGroupSelector = function(groupSelector)
| | out = out .. ']], уровень: <span style="color: ' .. tierColor .. '">' .. recipe.tier .. '</span> </div>' |
| if not groupSelector.children then return '' end
| | end |
| local result = ""
| |
| local wrapperStart, wrapperEnd = "", ""
| |
|
| |
|
| -- Проверка для контейнера EntityTableContainerFill
| | -- Пометка при взломе EMAG |
| if groupSelector.weight and groupSelector.weight ~= "default" then
| | if recipe.isEmag then |
| wrapperStart = string.format('{{LinkСard/Сollapsible|name=Группа предметов %s%%|content=', groupSelector.weight)
| | out = out .. '|mode-emag=1' |
| wrapperEnd = "}}"
| | end |
| elseif groupSelector["!type"] == "GroupSelector" and not groupSelector.weight then
| |
| wrapperStart = '{{LinkСard/Сollapsible|name=Может выпасть лишь один из:|content='
| |
| wrapperEnd = "}}"
| |
| end
| |
|
| |
|
| for _, child in ipairs(groupSelector.children) do
| | -- Пометка для исследуемой технологии |
| if child["!type"] == "GroupSelector" then
| | if recipe.discipline ~= "Static" then |
| result = result .. handleGroupSelector(child)
| | out = out .. '|mode-research=1' |
| elseif child["!type"] == "AllSelector" then
| | end |
| result = result .. string.format('{{LinkСard/Сollapsible|name=Выпадают только вместе:|content=%s}}', handleAllSelector(child))
| |
| elseif child.id then
| |
| result = result .. formatContent(child)
| |
| else
| |
| result = result .. "<div>Ошибка: отсутствует id у элемента.</div>"
| |
| end
| |
| end
| |
|
| |
|
| return wrapperStart .. result .. wrapperEnd
| | out = out .. '}}' |
| end
| | end |
| | |
| -- Обработка NestedSelector
| |
| handleNestedSelector = function(nestedSelector, wrapped, visited)
| |
| visited = visited or {}
| |
| if not nestedSelector.tableId then return '' end
| |
| | |
| local result = ""
| |
| local classesRolls, classesProb
| |
| | |
| if wrapped then
| |
| if nestedSelector.rolls and nestedSelector.rolls.range then
| |
| local rollsResult = processRolls(nestedSelector.rolls)
| |
| if rollsResult and #rollsResult > 0 then
| |
| classesRolls = ', максимум может выпасть: ' .. rollsResult
| |
| end
| |
| end
| |
| if nestedSelector.prob then
| |
| classesProb = string.format(" <div>%s%%</div>", nestedSelector.prob * 100 >= 1 and math.floor(nestedSelector.prob * 100) or nestedSelector.prob * 100)
| |
| end
| |
| end
| |
| | |
| if wrapped and (classesRolls or classesProb) then
| |
| result = result .. string.format('{{LinkСard/Сollapsible|name=Группа предметов%s%s|content=', classesRolls or "", classesProb or "")
| |
| end
| |
| | |
| result = result .. getTableOutput(nestedSelector.tableId, visited)
| |
| | |
| if wrapped and (classesRolls or classesProb) then
| |
| result = result .. "}}"
| |
| end
| |
| | |
| return result
| |
| end
| |
| | |
| -- Формирование списка химии
| |
| getChemOutput = function(itemData, id)
| |
| local item = findDataById(itemData, id)
| |
| if not item
| |
| or not item.SolutionContainerManager
| |
| or not item.SolutionContainerManager.solutions then
| |
| return ''
| |
| end
| |
| | |
| local foodSolution = item.SolutionContainerManager.solutions["food"]
| |
| if not foodSolution or not foodSolution.reagents then
| |
| return ''
| |
| end
| |
| | |
| local result = ""
| |
| for _, reagent in ipairs(foodSolution.reagents) do
| |
| result = result .. string.format(
| |
| '<li>[[Химия#chem_%s|%s]] (%d ед.)</li>',
| |
| reagent.ReagentId, reagent.ReagentId, reagent.Quantity
| |
| )
| |
| end
| |
| | |
| return string.format('<ul class="1">%s</ul>', result)
| |
| end
| |
| | |
| -- Основная функция модуля
| |
| function p.main(frame)
| |
| local mode = frame.args[1]
| |
| local id = frame.args[2]
| |
| | |
| if not id then return 'Не указан ID.' end
| |
|
| |
| local itemDataIndex = itemData
| |
|
| |
| if not itemData then return 'Не удалось загрузить данные.' end
| |
| | |
| if mode == 'framing' then
| |
| local subMode = frame.args[2]
| |
| local id = frame.args[3]
| |
| | |
| if not id then
| |
| return 'Не указан ID для режима framing.'
| |
| end
| |
| | |
| if subMode == 'chem' then
| |
| return frame:preprocess('{{СollapsibleMenu|color=#3e7c82|' .. getChemOutput(chemData, id) .. '}}')
| |
| elseif subMode == 'contained' then
| |
| return frame:preprocess('{{СollapsibleMenu|' .. getContainedOutput(itemDataIndex, id) .. '}}')
| |
| elseif subMode == "slot" then
| |
| local itemDataEntry = findDataById(itemSlotsData, id)
| |
| if not itemDataEntry then return "" end
| |
| | |
| local startingItem = getFirstStartingItem(itemDataEntry)
| |
| if not startingItem then return "" end
| |
|
| |
| return frame:preprocess('{{СollapsibleMenu|color=#71702a|' .. formatContent(startingItem) .. '}}')
| |
| else
| |
| return 'Неизвестный подрежим для framing: ' .. subMode
| |
| end
| |
| elseif mode == 'chem' then
| |
| return frame:preprocess(getChemOutput(chemData, id))
| |
| elseif mode == 'contained' then
| |
| return frame:preprocess(getContainedOutput(itemDataIndex, id))
| |
| elseif mode == "slot" then
| |
| local itemDataEntry = findDataById(itemSlotsData, id)
| |
| if not itemDataEntry then return "" end
| |
| | |
| local startingItem = getFirstStartingItem(itemDataEntry)
| |
| if not startingItem then return "" end
| |
|
| |
| return frame:preprocess(formatContent(startingItem))
| |
| elseif mode == 'rolls' then
| |
| local entity = findDataById(itemDataIndex, id)
| |
| if not entity then return 'ID не найден в данных.' end
| |
| | |
| if entity.EntityTableContainerFill then
| |
| local containers = entity.EntityTableContainerFill.containers
| |
| if containers.entity_storage and containers.entity_storage.rolls then
| |
| return processRolls(containers.entity_storage.rolls)
| |
| end
| |
| end
| |
|
| |
|
| return ''
| | return mw.getCurrentFrame():preprocess(out) |
| else
| |
| return 'Неизвестный режим: ' .. mode
| |
| end
| |
| end | | end |
|
| |
|
| return p | | return p |