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

мНет описания правки
Нет описания правки
 
(не показано 59 промежуточных версий этого же участника)
Строка 1: Строка 1:
-- Загрузка данных
local JsonPaths = require('Module:JsonPaths')
local spriteData = mw.loadData(JsonPaths.get("prototype/sprite.json"))
local p = {}
local p = {}


-- Загрузка данных
local function normalizeSpritePath(path)
local function loadData(filePath)
if path == nil then
    local page = mw.title.new(filePath)
return nil
    local content = page and page:getContent()
end
    return content and mw.text.jsonDecode(content) or nil
 
path = mw.text.trim(tostring(path))
path = path:gsub("^/Textures/?", "")
 
return path
end
 
local function normalizeState(state)
if state == nil then
return nil
end
 
state = mw.text.trim(tostring(state))
if state == "" then
return nil
end
 
return state
end
end


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


    -- Если это массивы, проверяем их содержимое без учета порядка
-- Функция для получения таблицы данных
    local function isArray(t)
function p.getData()
        local i = 0
return getIdTable() or {}
        for _ in pairs(t) do
end
            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
-- 1) сначала entry.sprite
        local matched = {}
-- 2) потом первый layers[*].sprite
        for _, v1 in ipairs(t1) do
local function getSpritePath(entry)
            local found = false
if not entry then
            for j, v2 in ipairs(t2) do
return nil
                if not matched[j] and deepEqual(v1, v2) then
end
                    matched[j] = true
                    found = true
                    break
                end
            end
            if not found then return false end
        end
        return true
    end


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


    for k, v in pairs(t2) do
if type(entry.layers) == "table" then
        if t1[k] == nil then
for _, layer in ipairs(entry.layers) do
            return false
if layer and layer.sprite then
        end
return normalizeSpritePath(layer.sprite)
    end
end
end
end


    return true
return nil
end
end


-- Получение пути спрайта
-- Проверка visible
local function getSpritePath(entry)
local function isLayerVisible(layer)
    if entry.Sprite and entry.Sprite.sprite then
if not layer then
        return entry.Sprite.sprite
return false
    elseif entry.Icon and entry.Icon.sprite then
end
        return entry.Icon.sprite
return layer.visible ~= false
    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
end


-- Генерация шаблона repeat
-- Функция получения списка состояний с учётом переопределяющего sprite в слое
local function generateRepeatTemplate(data)
local function getSpriteStates(entry)
    local spriteGroups = {}
local result = {}
local spritePath = getSpritePath(entry)
 
if entry and type(entry.layers) == "table" then
for _, layer in ipairs(entry.layers) do
if isLayerVisible(layer) and layer.state then
table.insert(result, {
state = tostring(layer.state),
sprite = normalizeSpritePath(layer.sprite or spritePath)
})
end
end
elseif entry and entry.state and spritePath then
table.insert(result, {
state = tostring(entry.state),
sprite = spritePath
})
end


    for _, entry in ipairs(data) do
if #result == 0 and spritePath then
        local found = false
table.insert(result, {
        for key, group in pairs(spriteGroups) do
state = "icon",
            if deepEqual(entry.Sprite, group[1].Sprite) and
sprite = spritePath
              deepEqual(entry.EntityStorageVisuals, group[1].EntityStorageVisuals) and
})
              deepEqual(entry.Icon, group[1].Icon) then
end
                table.insert(group, entry)
                found = true
                break
            end
        end


        if not found then
return (#result > 0) and result or nil
            spriteGroups[entry.id] = {entry}
end
        end
    end


    local result = {}
local function findByPathAndState(data, targetPath, targetState)
    for _, group in pairs(spriteGroups) do
for entityId, entry in pairs(data) do
        if #group > 1 then
if getSpritePath(entry) == targetPath then
            local idLinks = {}
local states = getSpriteStates(entry)
            for _, entry in ipairs(group) do
if states then
                table.insert(idLinks, "[[:Файл:" .. entry.id .. ".png]]")
for _, item in ipairs(states) do
            end
if normalizeState(item.state) == targetState then
            local firstId = group[1].id
return entityId
            table.insert(result, mw.getCurrentFrame():preprocess("{{Entity Sprite/Repeat|" .. table.concat(idLinks, " ") .. "|" .. firstId .. "}}"))
end
        end
end
    end
end
end
end


    return table.concat(result, "\n")
return nil
end
end


-- Создаём индекс для путей
local function findByPath(data, targetPath)
local function createSpritePathIndex(data)
for entityId, entry in pairs(data) do
    local index = {}
if getSpritePath(entry) == targetPath then
    for _, entry in ipairs(data) do
return entityId
        local spritePath = getSpritePath(entry)
end
        if spritePath then
end
            index[spritePath] = entry.id
 
        end
return nil
    end
    return index
end
end


-- Обновляем основную функцию
-- Функция генерации шаблона по записи (mode: image, path, state)
local function generateTemplate(entry, param, secondaryParam, data, spritePathIndex)
function p.main(frame)
    local spritePath = getSpritePath(entry)
local mode = frame.args[1]
    if not entry.id or not spritePath then
local id = frame.args[2]
        return nil
local stateArg = frame.args[3]
    end


    if param == "image" then
if not mode or not id then
        if secondaryParam then
return "Ошибка: отсутствует режим или ID"
            if tostring(entry.id) == tostring(secondaryParam) then
end
                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
local data = getIdTable()
end
if not data then
return "Ошибка: неверный формат JSON"
end
 
if mode == "image" or mode == "state" then
local entry = data[id]
if not entry then
return "Ошибка: ID не найден"
end
 
if mode == "image" then
local sprite = getSpritePath(entry)
return sprite or "Ошибка: спрайт не найден"
elseif mode == "state" then
local states = getSpriteStates(entry)
if not states then
return ""
end


-- Генерация шаблона по умолчанию
local baseUrl = JsonPaths.git() .. "/Resources/Textures/"
local function generateDefaultTemplate(data, params)
local links = {}
    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
for _, item in ipairs(states) do
    local path = params.Path
local spritePath = item.sprite
local stateName = item.state


    -- Поиск записи с указанным ID
if spritePath then
    local entry = nil
local url = baseUrl .. normalizeSpritePath(spritePath) .. "/" .. stateName .. ".png"
    if id and id ~= "" then
table.insert(links, "[" .. url .. " " .. stateName .. "]")
        for _, item in ipairs(data) do
else
            if tostring(item.id) == tostring(id) then
table.insert(links, stateName .. " Error: sprite not found")
                entry = item
end
                break
end
            end
        end
    end


    -- Если запись не найдена, ничего не выводим
return "(state: " .. table.concat(links, ", ") .. ")"
    if entry then
end
        spritePath = getSpritePath(entry)
end
        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


    -- Формирование шаблона
if mode == "path" then
    return mw.getCurrentFrame():preprocess(
local targetPath = normalizeSpritePath(id)
        "{{Файл\n" ..
local targetState = normalizeState(stateArg)
        "|Описание = " .. description .. "\n" ..
        "|Id      = " .. id .. "\n" ..
        "|Сервера  = " .. servers .. "\n" ..
        "|Источник = " .. source .. "\n" ..
        "|Путь    = " .. path .. "\n" ..
        "|Теги    = " .. tags .. "\n" ..
        "}}\n"
    )
end


function p.main(frame)
if targetState then
    local param = frame.args[1]
local byState = findByPathAndState(data, targetPath, targetState)
    local secondaryParam = frame.args[2]
if byState then
return byState
end
end


    local data = loadData('User:IanComradeBot/prototypes/entity sprite.json')
local byPath = findByPath(data, targetPath)
    if not data or type(data) ~= 'table' then
if byPath then
        return 'Ошибка: Невозможно загрузить данные из JSON.'
return byPath
    end
end


    -- Индекс путей
return "Ошибка: путь не найден"
    local spritePathIndex = createSpritePathIndex(data)
end


    if param == "repeat" then
return "Ошибка: неизвестный режим"
        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
end


return p
return p