Модуль:Entity Sprite: различия между версиями
Материал из Space Station 14 Вики
Pok (обсуждение | вклад) мНет описания правки |
Pok (обсуждение | вклад) мНет описания правки |
||
Строка 1: | Строка 1: | ||
local p = {} | local p = {} | ||
local | -- Загрузка данных | ||
local function loadData(filePath) | |||
local page = mw.title.new(filePath) | |||
local content = page and page:getContent() | |||
return content and mw.text.jsonDecode(content) or nil | |||
end | end | ||
-- | -- Проверка равенства двух таблиц | ||
local function deepEqual(t1, t2) | local function deepEqual(t1, t2) | ||
if t1 == t2 then return true end | if t1 == t2 then return true end | ||
if type(t1) ~= "table" or type(t2) ~= "table" then return false end | if type(t1) ~= "table" or type(t2) ~= "table" then return false end | ||
-- Если это массивы, проверяем их содержимое без учета порядка | |||
local function isArray(t) | local function isArray(t) | ||
local i = 0 | local i = 0 | ||
Строка 50: | Строка 40: | ||
end | end | ||
-- Если это таблицы, проверяем их содержимое | |||
for k, v in pairs(t1) do | for k, v in pairs(t1) do | ||
if t2[k] == nil or not deepEqual(v, t2[k]) then | if t2[k] == nil or not deepEqual(v, t2[k]) then | ||
Строка 56: | Строка 47: | ||
end | end | ||
for k, | for k, v in pairs(t2) do | ||
if t1[k] == nil then | if t1[k] == nil then | ||
return false | return false | ||
Строка 65: | Строка 56: | ||
end | end | ||
-- | -- Получение пути спрайта | ||
local function getSpritePath(entry) | local function getSpritePath(entry) | ||
if entry.Sprite and entry.Sprite.sprite then | if entry.Sprite and entry.Sprite.sprite then | ||
Строка 79: | Строка 70: | ||
end | end | ||
return nil | return nil | ||
end | end | ||
Строка 149: | Строка 89: | ||
if not found then | if not found then | ||
spriteGroups[entry.id] = { entry } | spriteGroups[entry.id] = {entry} | ||
end | end | ||
end | end | ||
Строка 168: | Строка 108: | ||
end | end | ||
-- | -- Создаём индекс для путей | ||
local function generateTemplate(entry, param, secondaryParam, spritePathIndex) | local function createSpritePathIndex(data) | ||
local index = {} | |||
for _, entry in ipairs(data) do | |||
local spritePath = getSpritePath(entry) | |||
if spritePath then | |||
index[spritePath] = entry.id | |||
end | |||
end | |||
return index | |||
end | |||
-- Обновляем основную функцию | |||
local function generateTemplate(entry, param, secondaryParam, data, spritePathIndex) | |||
local spritePath = getSpritePath(entry) | local spritePath = getSpritePath(entry) | ||
if not entry.id or not spritePath then | if not entry.id or not spritePath then | ||
Строка 186: | Строка 138: | ||
elseif param == "path" then | elseif param == "path" then | ||
if secondaryParam then | if secondaryParam then | ||
-- Используем индекс для быстрого поиска | |||
local id = spritePathIndex[secondaryParam] | local id = spritePathIndex[secondaryParam] | ||
if id then | if id then | ||
Строка 210: | Строка 163: | ||
local path = params.Path | local path = params.Path | ||
-- Поиск записи с указанным ID | |||
local entry = nil | local entry = nil | ||
if id and id ~= "" | if id and id ~= "" then | ||
for _, item in ipairs(data) do | for _, item in ipairs(data) do | ||
if tostring(item.id) == tostring(id) then | if tostring(item.id) == tostring(id) then | ||
Строка 222: | Строка 174: | ||
end | end | ||
-- Если запись не найдена, ничего не выводим | |||
if entry then | if entry then | ||
spritePath = getSpritePath(entry) | spritePath = getSpritePath(entry) | ||
Строка 228: | Строка 181: | ||
end | end | ||
end | end | ||
-- Если Path не указан, подставляем путь из User:IanComradeBot/prototypes/entity sprite.json | |||
if not path or path == "" then | if not path or path == "" then | ||
path = "Resources/Textures/" .. (spritePath or "") | path = "Resources/Textures/" .. (spritePath or "") | ||
end | end | ||
-- Формирование шаблона | |||
return mw.getCurrentFrame():preprocess( | return mw.getCurrentFrame():preprocess( | ||
"{{Файл\n" .. | "{{Файл\n" .. | ||
Строка 241: | Строка 196: | ||
"|Путь = " .. path .. "\n" .. | "|Путь = " .. path .. "\n" .. | ||
"|Теги = " .. tags .. "\n" .. | "|Теги = " .. tags .. "\n" .. | ||
"}}\n" | "}}\n" | ||
) | ) | ||
end | end | ||
Строка 254: | Строка 209: | ||
end | end | ||
-- Индекс путей | |||
local spritePathIndex = createSpritePathIndex(data) | local spritePathIndex = createSpritePathIndex(data) | ||
Строка 260: | Строка 216: | ||
elseif param == "path" and secondaryParam then | elseif param == "path" and secondaryParam then | ||
for _, entry in ipairs(data) do | for _, entry in ipairs(data) do | ||
local template = generateTemplate(entry, param, secondaryParam, spritePathIndex) | local template = generateTemplate(entry, param, secondaryParam, data, spritePathIndex) | ||
if template then | if template then | ||
return template | return template | ||
end | end | ||
end | end | ||
return nil | return nil | ||
elseif param == "image" and secondaryParam then | elseif param == "image" and secondaryParam then | ||
for _, entry in ipairs(data) do | |||
if entry.id == secondaryParam then | |||
return getSpritePath(entry) or "Ошибка: Спрайт не найден." | |||
end | |||
end | end | ||
return "Ошибка: ID не найден." | |||
elseif param == "image" or param == "path" then | elseif param == "image" or param == "path" then | ||
local result = {} | local result = {} | ||
for _, entry in ipairs(data) do | for _, entry in ipairs(data) do | ||
local template = generateTemplate(entry, param, secondaryParam, spritePathIndex) | local template = generateTemplate(entry, param, secondaryParam, data, spritePathIndex) | ||
if template then | if template then | ||
table.insert(result, template) | table.insert(result, template) | ||
Строка 283: | Строка 239: | ||
return table.concat(result, "\n") | return table.concat(result, "\n") | ||
else | else | ||
-- Если нет режима, генерируем шаблон по умолчанию | |||
return generateDefaultTemplate(data, frame.args) | return generateDefaultTemplate(data, frame.args) | ||
end | end |
Версия от 12:16, 3 февраля 2025
local p = {}
-- Загрузка данных
local function loadData(filePath)
local page = mw.title.new(filePath)
local content = page and page:getContent()
return content and mw.text.jsonDecode(content) or nil
end
-- Проверка равенства двух таблиц
local function deepEqual(t1, t2)
if t1 == t2 then return true end
if type(t1) ~= "table" or type(t2) ~= "table" then return false end
-- Если это массивы, проверяем их содержимое без учета порядка
local function isArray(t)
local i = 0
for _ in pairs(t) do
i = i + 1
if t[i] == nil then return false end
end
return true
end
if isArray(t1) and isArray(t2) then
if #t1 ~= #t2 then return false end
local matched = {}
for _, v1 in ipairs(t1) do
local found = false
for j, v2 in ipairs(t2) do
if not matched[j] and deepEqual(v1, v2) then
matched[j] = true
found = true
break
end
end
if not found then return false end
end
return true
end
-- Если это таблицы, проверяем их содержимое
for k, v in pairs(t1) do
if t2[k] == nil or not deepEqual(v, t2[k]) then
return false
end
end
for k, v in pairs(t2) do
if t1[k] == nil then
return false
end
end
return true
end
-- Получение пути спрайта
local function getSpritePath(entry)
if entry.Sprite and entry.Sprite.sprite then
return entry.Sprite.sprite
elseif entry.Icon and entry.Icon.sprite then
return entry.Icon.sprite
elseif entry.Sprite and entry.Sprite.layers then
for _, layer in ipairs(entry.Sprite.layers) do
if layer.sprite then
return layer.sprite
end
end
end
return nil
end
-- Генерация шаблона repeat
local function generateRepeatTemplate(data)
local spriteGroups = {}
for _, entry in ipairs(data) do
local found = false
for key, group in pairs(spriteGroups) do
if deepEqual(entry.Sprite, group[1].Sprite) and
deepEqual(entry.EntityStorageVisuals, group[1].EntityStorageVisuals) and
deepEqual(entry.Icon, group[1].Icon) then
table.insert(group, entry)
found = true
break
end
end
if not found then
spriteGroups[entry.id] = {entry}
end
end
local result = {}
for _, group in pairs(spriteGroups) do
if #group > 1 then
local idLinks = {}
for _, entry in ipairs(group) do
table.insert(idLinks, "[[:Файл:" .. entry.id .. ".png]]")
end
local firstId = group[1].id
table.insert(result, mw.getCurrentFrame():preprocess("{{Entity Sprite/Repeat|" .. table.concat(idLinks, " ") .. "|" .. firstId .. "}}"))
end
end
return table.concat(result, "\n")
end
-- Создаём индекс для путей
local function createSpritePathIndex(data)
local index = {}
for _, entry in ipairs(data) do
local spritePath = getSpritePath(entry)
if spritePath then
index[spritePath] = entry.id
end
end
return index
end
-- Обновляем основную функцию
local function generateTemplate(entry, param, secondaryParam, data, spritePathIndex)
local spritePath = getSpritePath(entry)
if not entry.id or not spritePath then
return nil
end
if param == "image" then
if secondaryParam then
if tostring(entry.id) == tostring(secondaryParam) then
return spritePath
end
return nil
else
return mw.getCurrentFrame():preprocess("{{Entity Sprite/Image|" .. entry.id .. "|" .. spritePath .. "}}")
end
elseif param == "path" then
if secondaryParam then
-- Используем индекс для быстрого поиска
local id = spritePathIndex[secondaryParam]
if id then
return id
end
return nil
else
return mw.getCurrentFrame():preprocess("{{Entity Sprite/Path|" .. entry.id .. "|" .. spritePath .. "}}")
end
end
return nil
end
-- Генерация шаблона по умолчанию
local function generateDefaultTemplate(data, params)
local id = params.Id
local description = params.Description or ""
local servers = params.Servers or ""
local source = params.Source or ""
local tags = params.Tags or ""
local spritePath = nil
local path = params.Path
-- Поиск записи с указанным ID
local entry = nil
if id and id ~= "" then
for _, item in ipairs(data) do
if tostring(item.id) == tostring(id) then
entry = item
break
end
end
end
-- Если запись не найдена, ничего не выводим
if entry then
spritePath = getSpritePath(entry)
if not spritePath then
return ""
end
end
-- Если Path не указан, подставляем путь из User:IanComradeBot/prototypes/entity sprite.json
if not path or path == "" then
path = "Resources/Textures/" .. (spritePath or "")
end
-- Формирование шаблона
return mw.getCurrentFrame():preprocess(
"{{Файл\n" ..
"|Описание = " .. description .. "\n" ..
"|Id = " .. id .. "\n" ..
"|Сервера = " .. servers .. "\n" ..
"|Источник = " .. source .. "\n" ..
"|Путь = " .. path .. "\n" ..
"|Теги = " .. tags .. "\n" ..
"}}\n"
)
end
function p.main(frame)
local param = frame.args[1]
local secondaryParam = frame.args[2]
local data = loadData('User:IanComradeBot/prototypes/entity sprite.json')
if not data or type(data) ~= 'table' then
return 'Ошибка: Невозможно загрузить данные из JSON.'
end
-- Индекс путей
local spritePathIndex = createSpritePathIndex(data)
if param == "repeat" then
return generateRepeatTemplate(data)
elseif param == "path" and secondaryParam then
for _, entry in ipairs(data) do
local template = generateTemplate(entry, param, secondaryParam, data, spritePathIndex)
if template then
return template
end
end
return nil
elseif param == "image" and secondaryParam then
for _, entry in ipairs(data) do
if entry.id == secondaryParam then
return getSpritePath(entry) or "Ошибка: Спрайт не найден."
end
end
return "Ошибка: ID не найден."
elseif param == "image" or param == "path" then
local result = {}
for _, entry in ipairs(data) do
local template = generateTemplate(entry, param, secondaryParam, data, spritePathIndex)
if template then
table.insert(result, template)
end
end
return table.concat(result, "\n")
else
-- Если нет режима, генерируем шаблон по умолчанию
return generateDefaultTemplate(data, frame.args)
end
end
return p