Модуль:Research: различия между версиями

Материал из Space Station 14 Вики
мНет описания правки
мНет описания правки
 
(не показана 31 промежуточная версия этого же участника)
Строка 1: Строка 1:
local p = {}
local p = {}


-- Глобальная переменная для данных
-- Функция для загрузки данных исследований из JSON-файла
local data
local function loadResearchData()
return mw.text.jsonDecode(mw.title.new("User:IanComradeBot/research_prototypes.json"):getContent())
end


-- Хранилище для кэша перевода
-- Функция для поиска исследования по ID
local boardToMachineCache = {}
local function findResearchById(dataCache, id)
for _, research in ipairs(dataCache) do
if research.id == id then
return research
end
end
return nil
end
 
-- Кэш для ID машин
local machineIDCache = {}


-- Функция для перевода ID плат в ID машин
-- Функция для перевода ID плат в ID машин
local function translateBoardIDToMachineID(boardID)
local function translateBoardIDToMachineID(boardID)
-- Проверяем кэш
if machineIDCache[boardID] then
if boardToMachineCache[boardID] ~= nil then
return machineIDCache[boardID]
return boardToMachineCache[boardID]
end
end


-- Если данные еще не загружены, загружаем их
local data = mw.text.jsonDecode(mw.title.new("User:IanComradeBot/entity_prototypes.json"):getContent())
if not data then
data = mw.text.jsonDecode(mw.title.new("Участник:IanComradeBot/entity_prototypes.json"):getContent())
end
 
-- Поиск платы по ID и извлечение её названия
local board = data[boardID]
local board = data[boardID]
if not board or not board.name then
if not board or not board.name then
return nil -- Если платы нет в данных, возвращаем nil
return nil
end
end


-- Проверка наличия "(машинная плата)" или "(консольная плата)" в имени
if not board.name:find("%(машинная плата%)") and not board.name:find("%(консольная плата%)") then
if not (board.name:find("%(машинная плата%)") or board.name:find("%(консольная плата%)")) then
return nil
return nil -- Если в имени нет одной из фраз, возвращаем nil
end
end


-- Удаление "(машинная плата)" и "(консольная плата)" из имени платы
local machineName = board.name:gsub(" %(машинная плата%)", ""):gsub(" %(консольная плата%)", "")
local machineName = board.name:gsub(" %(машинная плата%)", ""):gsub(" %(консольная плата%)", ""):trim()
local excludeWords = {"Unanchored", "Debug", "Admin", "Enabled"}


-- Список исключений
local excludeWords = {"Unanchored", "Debug", "Admin"}
-- Оптимизация: создаем таблицу для быстрого поиска по имени
local machineID = nil
for _, entity in pairs(data) do
for _, entity in pairs(data) do
if entity.name == machineName then
if entity.name == machineName then
-- Проверка на наличие исключений в ID
local shouldExclude = false
local shouldExclude = false
for _, word in ipairs(excludeWords) do
for _, word in ipairs(excludeWords) do
Строка 48: Строка 47:
end
end
end
end
 
if not shouldExclude then
if not shouldExclude then
machineID = entity.id -- Сохраняем ID машины
machineIDCache[boardID] = entity.id
break
return entity.id
end
end
end
end
end
end


-- Кэшируем результат (nil, если не найдено)
return nil
boardToMachineCache[boardID] = machineID
return machineID -- Возвращаем ID машины или nil
end
end
-- Таблица для перевода названий дисциплин
local disciplineMapping = {
Arsenal = "Арсенал",
Industrial = "Промышленность",
Experimental = "Экспериментальное",
CivilianServices = "Обслуживание персонала"
}
-- Таблица для цветов по уровню
local tierColors = {
[1] = "#54d554",
[2] = "#ed9000",
[3] = "#d72a2a"
}


function p.main(frame)
function p.main(frame)
-- Подключение CSS файла
-- Подключение CSS
local cssLink = frame:extensionTag('templatestyles', '', {
local cssLink = frame:extensionTag('templatestyles', '', {
src = 'Шаблон:Research/styles.css'
src = 'Шаблон:Research/styles.css'
})
})


local id = frame.args.id or "" -- ID исследования
local dataCache = loadResearchData()
local icon = frame.args.icon or "" -- Изобрадения исследования
local id = frame.args.id or ""
local customRecipeUnlocks = frame.args.customRecipeUnlocks or nil -- Ручной ввод в список разблокируемых технологий
local icon = frame.args.icon or ""


-- Загрузка данных из JSON один раз
-- Получение кастомных рецептов с индексами (если они есть)
if not data then
local customRecipeUnlocks = {}
data = mw.text.jsonDecode(mw.title.new("User:IanComradeBot/research_prototypes.json"):getContent())
for i = 1, 100 do
local indexKey = "customRecipeUnlocksIndex" .. i
if frame.args[indexKey] then
table.insert(customRecipeUnlocks, frame.args[indexKey])
end
end
end


local out = cssLink
-- Получение кастомных рецептов без индексации
local found = false  -- Флаг для отслеживания наличия элементов
if frame.args.customRecipeUnlocks then
local disciplineName = ""  
local tempUnlocks = mw.text.split(frame.args.customRecipeUnlocks, " ")
for _, item in ipairs(tempUnlocks) do
table.insert(customRecipeUnlocks, item)
end
end


-- Определяем дисциплину на основе переданного id
local tech = findResearchById(dataCache, id)
for discipline, technologies in pairs(data) do
if not tech then
for _, tech in ipairs(technologies) do
return cssLink .. '<div style="color:red;">Исследование с ID "' .. id .. '" не найдено.</div>'
if tech.id == id then
end
found = true
disciplineName = ({
Arsenal = "Арсенал",
Industrial = "Промышленность",
Experimental = "Экспериментальное",
CivilianServices = "Обслуживание персонала"
})[discipline]


-- Цвета для номеров уровней
local disciplineName = disciplineMapping[tech.discipline] or "Неизвестное исследование"
local tierColor = ({
local tierColor = tierColors[tech.tier] or "#FFFFFF"
[1] = "#54d554",  -- зеленый
[2] = "#ed9000",  -- оранжевый
[3] = "#d72a2a"   -- красный
})[tech.tier]


-- Формирование HTML
-- Блок данных
out = out .. '<div class="research" id="' .. discipline .. '">'
local out = cssLink .. '<div class="research" id="' .. tech.discipline .. '">'
out = out .. '<div class="research__images">[[Файл:' .. icon .. '.png|64x64px|центр|link=]]</div>' -- Используем переданный icon
out = out .. '<div class="research__images">[[Файл:' .. icon .. '.png|64x64px|центр|link=]]</div>'
out = out .. '<div class="research__name">' .. tech.name .. '[[Файл:' .. discipline .. '.png|16px|link=]]</div>'
out = out .. '<div class="research__name">' .. tech.name .. '[[Файл:' .. tech.discipline .. '.png|16px|link=]]</div>'
out = out .. '<div class="research__type">'
out = out .. '<div class="research__type">'
out = out .. '<div>Уровень: <span style="color:' .. tierColor .. ';">' .. tech.tier .. '</span></div>'
out = out .. '<div>Уровень: <span style="color:' .. tierColor .. ';">' .. tech.tier .. '</span></div>'
out = out .. '<div class="research__technology">' .. disciplineName .. '</div>'
out = out .. '<div class="research__technology">' .. disciplineName .. '</div>'
out = out .. '<div>Стоимость: <span style="color:#DA70D6;">' .. tech.cost .. '</span></div>'
out = out .. '<div>Стоимость: <span style="color:#DA70D6;">' .. tech.cost .. '</span></div>'
out = out .. '</div>'
out = out .. '</div>'
out = out .. '<div class="research__unblocks">Разблокирует:'
out = out .. '<div class="research__technologies">'
out = out .. '<ul>'


-- Используем пользовательские recipeUnlocks, если они указаны, иначе используем значения из json файла
-- Блок необходимых технологий для открытия исследования
local recipeUnlocks
if tech.technologyPrerequisites and #tech.technologyPrerequisites > 0 then
if customRecipeUnlocks then
out = out .. '<div class="research__technologies-prerequisites">Необходимые исследования:'
recipeUnlocks = mw.text.split(customRecipeUnlocks, " ") -- Разбиваем по пробелам
else
for _, prerequisiteId in ipairs(tech.technologyPrerequisites) do
recipeUnlocks = tech.recipeUnlocks
local prerequisiteTech = findResearchById(dataCache, prerequisiteId)
end
out = out .. '<span> ' .. prerequisiteTech.name .. '</span>'
end
out = out .. '</div>'
end


-- Проверка каждого ID в recipeUnlocks
-- Блок открываемых технологий
for _, recipe in ipairs(recipeUnlocks) do
out = out .. '<div class="research__technologies-unblocks">Разблокирует:'
local machineID = translateBoardIDToMachineID(recipe) or recipe  -- Переводим только платы, остальные оставляем как есть
out = out .. '<ul>'


out = out .. frame:preprocess('<li>{{#invoke:Entity Lookup|createimagetooltip|Файл:' .. machineID .. '.png|' .. machineID .. '|Мета=32x32px,link=}} {{#invoke:Entity Lookup|getname|' .. machineID .. '}}</li>')
local recipeUnlocks = customRecipeUnlocks
end
if #customRecipeUnlocks == 0 then
recipeUnlocks = tech.recipeUnlocks
end


out = out .. '</ul>'
for _, recipe in ipairs(recipeUnlocks) do
out = out .. '</div>'
local machineID = translateBoardIDToMachineID(recipe) or recipe
out = out .. '</div>'
out = out .. frame:preprocess('<li>{{#invoke:Entity Lookup|createimagetooltip|Файл:' .. machineID .. '.png|' .. machineID .. '|Мета=32x32px,link=}} {{#invoke:Entity Lookup|getname|' .. machineID .. '}}</li>')
end
end
end
end


if not found then
out = out .. '</ul>'
out = out .. '<div style="color:red;">Нет доступных исследований.</div>'
out = out .. '</div>'
end
 
out = out .. '</div>'
out = out .. '</div>'


return out
return out

Текущая версия от 13:38, 21 ноября 2024

Для документации этого модуля может быть создана страница Модуль:Research/doc

local p = {}

-- Функция для загрузки данных исследований из JSON-файла
local function loadResearchData()
	return mw.text.jsonDecode(mw.title.new("User:IanComradeBot/research_prototypes.json"):getContent())
end

-- Функция для поиска исследования по ID
local function findResearchById(dataCache, id)
	for _, research in ipairs(dataCache) do
		if research.id == id then
			return research
		end
	end
	return nil
end

-- Кэш для ID машин
local machineIDCache = {}

-- Функция для перевода ID плат в ID машин
local function translateBoardIDToMachineID(boardID)
	if machineIDCache[boardID] then
		return machineIDCache[boardID]
	end

	local data = mw.text.jsonDecode(mw.title.new("User:IanComradeBot/entity_prototypes.json"):getContent())
	local board = data[boardID]
	if not board or not board.name then
		return nil
	end

	if not board.name:find("%(машинная плата%)") and not board.name:find("%(консольная плата%)") then
		return nil
	end

	local machineName = board.name:gsub(" %(машинная плата%)", ""):gsub(" %(консольная плата%)", "")
	local excludeWords = {"Unanchored", "Debug", "Admin", "Enabled"}

	for _, entity in pairs(data) do
		if entity.name == machineName then
			local shouldExclude = false
			for _, word in ipairs(excludeWords) do
				if entity.id:find(word) then
					shouldExclude = true
					break
				end
			end

			if not shouldExclude then
				machineIDCache[boardID] = entity.id
				return entity.id
			end
		end
	end

	return nil
end

-- Таблица для перевода названий дисциплин
local disciplineMapping = {
	Arsenal = "Арсенал",
	Industrial = "Промышленность",
	Experimental = "Экспериментальное",
	CivilianServices = "Обслуживание персонала"
}

-- Таблица для цветов по уровню
local tierColors = {
	[1] = "#54d554",
	[2] = "#ed9000",
	[3] = "#d72a2a"
}

function p.main(frame)
	-- Подключение CSS
	local cssLink = frame:extensionTag('templatestyles', '', {
		src = 'Шаблон:Research/styles.css'
	})

	local dataCache = loadResearchData()
	local id = frame.args.id or ""
	local icon = frame.args.icon or ""

	-- Получение кастомных рецептов с индексами (если они есть)
	local customRecipeUnlocks = {}
	for i = 1, 100 do 
		local indexKey = "customRecipeUnlocksIndex" .. i
		if frame.args[indexKey] then
			table.insert(customRecipeUnlocks, frame.args[indexKey])
		end
	end

	-- Получение кастомных рецептов без индексации
	if frame.args.customRecipeUnlocks then
		local tempUnlocks = mw.text.split(frame.args.customRecipeUnlocks, " ")
		for _, item in ipairs(tempUnlocks) do
			table.insert(customRecipeUnlocks, item)
		end
	end

	local tech = findResearchById(dataCache, id)
	if not tech then
		return cssLink .. '<div style="color:red;">Исследование с ID "' .. id .. '" не найдено.</div>'
	end

	local disciplineName = disciplineMapping[tech.discipline] or "Неизвестное исследование"
	local tierColor = tierColors[tech.tier] or "#FFFFFF"

	-- Блок данных
	local out = cssLink .. '<div class="research" id="' .. tech.discipline .. '">'
	out = out .. '<div class="research__images">[[Файл:' .. icon .. '.png|64x64px|центр|link=]]</div>'
	out = out .. '<div class="research__name">' .. tech.name .. '[[Файл:' .. tech.discipline .. '.png|16px|link=]]</div>'
	out = out .. '<div class="research__type">'
	out = out .. '<div>Уровень: <span style="color:' .. tierColor .. ';">' .. tech.tier .. '</span></div>'
	out = out .. '<div class="research__technology">' .. disciplineName .. '</div>'
	out = out .. '<div>Стоимость: <span style="color:#DA70D6;">' .. tech.cost .. '</span></div>'
	out = out .. '</div>'
	out = out .. '<div class="research__technologies">'

	-- Блок необходимых технологий для открытия исследования
	if tech.technologyPrerequisites and #tech.technologyPrerequisites > 0 then
		out = out .. '<div class="research__technologies-prerequisites">Необходимые исследования:'
		
		for _, prerequisiteId in ipairs(tech.technologyPrerequisites) do
			local prerequisiteTech = findResearchById(dataCache, prerequisiteId)
			
			out = out .. '<span> ' .. prerequisiteTech.name .. '</span>'
		end
	
		out = out .. '</div>'
	end

	-- Блок открываемых технологий
	out = out .. '<div class="research__technologies-unblocks">Разблокирует:'
	out = out .. '<ul>'

	local recipeUnlocks = customRecipeUnlocks
	if #customRecipeUnlocks == 0 then
		recipeUnlocks = tech.recipeUnlocks
	end

	for _, recipe in ipairs(recipeUnlocks) do
		local machineID = translateBoardIDToMachineID(recipe) or recipe
		out = out .. frame:preprocess('<li>{{#invoke:Entity Lookup|createimagetooltip|Файл:' .. machineID .. '.png|' .. machineID .. '|Мета=32x32px,link=}} {{#invoke:Entity Lookup|getname|' .. machineID .. '}}</li>')
	end

	out = out .. '</ul>'
	out = out .. '</div>'

	out = out .. '</div>'
	out = out .. '</div>'

	return out
end

return p