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

мНет описания правки
мНет описания правки
 
(не показано 40 промежуточных версий этого же участника)
Строка 1: Строка 1:
-- Загрузка данных
local spriteData = mw.loadData("Модуль:IanComradeBot/prototypes/entity sprite.json/data")
local p = {}
local p = {}


-- Проверка равенства двух таблиц
-- Функция для получения таблицы данных
local function deepEqual(t1, t2)
function p.getData()
     if t1 == t2 then return true end
     return spriteData
    if type(t1) ~= "table" or type(t2) ~= "table" then return false end
end


    -- Если это массивы, проверяем их содержимое без учета порядка
-- Функция нечувствительного к регистру поиска поля
    local function isArray(t)
local function findFieldInsensitive(tbl, fieldName)
        local i = 0
    for key, value in pairs(tbl) do
        for _ in pairs(t) do
        if type(key) == "string" and mw.ustring.lower(key) == mw.ustring.lower(fieldName) then
            i = i + 1
            return value
            if t[i] == nil then return false end
         end
         end
        return true
     end
     end
    return nil
end


    if isArray(t1) and isArray(t2) then
-- Функция получения пути спрайта: сначала пытаем Icon.sprite, затем Sprite.sprite, затем первый layer.sprite
        if #t1 ~= #t2 then return false end
local function getSpritePath(entry)
        local matched = {}
    local iconBlock = findFieldInsensitive(entry, "Icon")
        for _, v1 in ipairs(t1) do
    if type(iconBlock) == "table" and iconBlock.sprite then
            local found = false
         return iconBlock.sprite
            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
     end


     -- Если это таблицы, проверяем их содержимое
     local spriteBlock = findFieldInsensitive(entry, "Sprite")
     for k, v in pairs(t1) do
     if not spriteBlock then return nil end
        if t2[k] == nil or not deepEqual(v, t2[k]) then
 
            return false
    if type(spriteBlock) == "string" then
        end
        return spriteBlock
     end
     end


     for k, v in pairs(t2) do
     if spriteBlock.sprite then
        if t1[k] == nil then
        return spriteBlock.sprite
            return false
        end
     end
     end


    return true
     if spriteBlock.layers then
end
         for _, layer in pairs(spriteBlock.layers) do
 
-- Получение пути спрайта
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
             if layer.sprite then
                 return layer.sprite
                 return layer.sprite
Строка 62: Строка 44:
         end
         end
     end
     end
     return nil
     return nil
end
end


-- Генерация шаблона repeat с оптимизированным вызовом preprocess (один раз в конце)
-- Проверка visible и shader
local function generateRepeatTemplate(data)
local function isLayerVisibleAndShaded(layer)
     local spriteGroups = {}
     if layer.visible == false then return false end
 
    if layer.shader and layer.shader == "unshaded" then return false end
    for _, entry in ipairs(data) do
    return true
        local found = false
end
        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
-- Функция определения, что поле state первое
            spriteGroups[entry.id] = {entry}
local function isStateFirstKey(layer)
         end
    for k, _ in pairs(layer) do
         return k == "state"
     end
     end
    return false
end


-- Функция получения списка состояний с учётом переопределяющего sprite в слое
local function getSpriteStates(entry)
     local result = {}
     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, "{{Entity Sprite/Repeat|" .. table.concat(idLinks, " ") .. "|" .. firstId .. "}}")
        end
    end


     return mw.getCurrentFrame():preprocess(table.concat(result, "\n"))
     local function addState(state, sprite)
end
        table.insert(result, { state = state, sprite = sprite })
 
-- Создаём индекс для путей
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
     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
     local iconBlock = findFieldInsensitive(entry, "Icon")
        return nil
    end


     if param == "image" then
     if iconBlock and iconBlock.state and isLayerVisibleAndShaded(iconBlock) then
         if secondaryParam then
         local s = (type(iconBlock) == "table" and iconBlock.sprite) or spritePath
            if tostring(entry.id) == tostring(secondaryParam) then
        addState(iconBlock.state, s)
                return spritePath
    else
            end
         local spriteBlock = findFieldInsensitive(entry, "Sprite")
            return nil
         if spriteBlock and spriteBlock.layers then
         else
            for _, layer in ipairs(spriteBlock.layers) do
            return mw.getCurrentFrame():preprocess("{{Entity Sprite/Image|" .. entry.id .. "|" .. spritePath .. "}}")
                if layer.state and isLayerVisibleAndShaded(layer) then
         end
                    local s = layer.sprite or spritePath
    elseif param == "path" then
                    addState(layer.state, s)
        if secondaryParam then
                    break
            local id = spritePathIndex[secondaryParam]
                 end
            if id then
                 return id
             end
             end
            return nil
         elseif spriteBlock and spriteBlock.state and isLayerVisibleAndShaded(spriteBlock) then
         else
            addState(spriteBlock.state, spritePath)
            return mw.getCurrentFrame():preprocess("{{Entity Sprite/Path|" .. entry.id .. "|" .. spritePath .. "}}")
         end
         end
     end
     end


     return nil
     local spriteBlock = findFieldInsensitive(entry, "Sprite")
end
     if spriteBlock and spriteBlock.layers then
 
        for _, layer in ipairs(spriteBlock.layers) do
-- Генерация шаблона по умолчанию
            local alreadyAdded = false
local function generateDefaultTemplate(data, params)
            for _, r in ipairs(result) do
     local id = params.Id
                if r.state == layer.state then
    local description = params.Description or ""
                    alreadyAdded = true
    local servers = params.Servers or ""
                    break
    local source = params.Source or ""
                end
    local tags = params.Tags or ""
            end
 
             if not alreadyAdded and layer.state and isStateFirstKey(layer) and isLayerVisibleAndShaded(layer) then
    local spritePath = nil
                 local s = layer.sprite or spritePath
    local path = params.Path
                 addState(layer.state, s)
 
    -- Поиск записи с указанным 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
         end
     end
     end


    -- Если запись не найдена, ничего не выводим
     if #result == 0 and spritePath then
     if entry then
         addState("icon", spritePath)
         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
     end


    -- Формирование шаблона
     return (#result > 0) and result or nil
     return mw.getCurrentFrame():preprocess(
        "{{Файл\n" ..
        "|Описание = " .. description .. "\n" ..
        "|Id      = " .. id .. "\n" ..
        "|Сервера  = " .. servers .. "\n" ..
        "|Источник = " .. source .. "\n" ..
        "|Путь    = " .. path .. "\n" ..
        "|Теги    = " .. tags .. "\n" ..
        "}}\n"
    )
end
end


-- Функция генерации шаблона по записи (mode: image, path, state)
function p.main(frame)
function p.main(frame)
     local param = frame.args[1]
     local mode = frame.args[1]
     local secondaryParam = frame.args[2]
     local id = frame.args[2]
    if not mode or not id then
        return "Ошибка: отсутствует режим или ID"
    end


     local data = mw.loadData("Модуль:Entity Sprite/data")
     for _, entry in ipairs(spriteData) do
 
        if entry.id == id then
    -- Индекс путей
            if mode == "image" then
    local spritePathIndex = createSpritePathIndex(data)
                local sprite = getSpritePath(entry)
 
                return sprite or "Ошибка: спрайт не найден"
    if param == "repeat" then
            elseif mode == "path" then
        return generateRepeatTemplate(data)
                if getSpritePath(entry) == id then
    elseif param == "path" and secondaryParam then
                    return entry.id
        for _, entry in ipairs(data) do
                end
            local template = generateTemplate(entry, param, secondaryParam, data, spritePathIndex)
            elseif mode == "state" then
            if template then
                local states = getSpriteStates(entry)
                 return template
                if not states then
                    return ""
                end
                local baseUrl = "https://github.com/space-syndicate/space-station-14/blob/master/Resources/Textures/"
                local links = {}
                for _, item in ipairs(states) do
                    local spritePath = item.sprite
                    local stateName = item.state
                    if spritePath then
                        local url = baseUrl .. spritePath .. "/" .. stateName .. ".png"
                        table.insert(links, "[" .. url .. " " .. stateName .. "]")
                    else
                        table.insert(links, stateName .. " Error: sprite not found")
                    end
                end
                 return "(state: " .. table.concat(links, ", ") .. ")"
             end
             end
         end
         end
        return nil
    elseif param == "image" and secondaryParam then
        for _, entry in ipairs(data) do
            if tostring(entry.id) == tostring(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
    if mode == "path" then
        return "Ошибка: путь не найден"
    elseif mode == "image" then
        return "Ошибка: ID не найден"
    elseif mode == "state" then
        return "Ошибка: ID не найден"
    end
    return "Ошибка: неизвестный режим"
end
end


return p
return p