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

Материал из Space Stories Wiki
Нет описания правки
Нет описания правки
 
(не показана 1 промежуточная версия этого же участника)
Строка 27: Строка 27:
     if not recipeStr or recipeStr == "" then return {} end
     if not recipeStr or recipeStr == "" then return {} end
     local lines = splitLines(recipeStr)
     local lines = splitLines(recipeStr)
     local recipes = {}
     local steps = {}
     local currentInputs = {}
     local currentInputs = {}
     local i = 1
     local i = 1
Строка 36: Строка 36:
             if #currentInputs > 0 then
             if #currentInputs > 0 then
                 local outputs = {}
                 local outputs = {}
                 i = i + 1 -- переходим сразу после «смешайте»
                 i = i + 1
                 while i <= #lines do
                 while i <= #lines do
                     local nextLine = lines[i]
                     local nextLine = lines[i]
Строка 44: Строка 44:
                     i = i + 1
                     i = i + 1
                 end
                 end
                local action = mw.text.trim(line)
                 table.insert(steps, {
                 table.insert(recipes, {
                     inputs = currentInputs,
                     inputs = currentInputs,
                     outputs = outputs,
                     outputs = outputs,
                     action = action
                     action = mw.text.trim(line)
                 })
                 })
                 currentInputs = {}
                 currentInputs = {}
             else
             else
                 i = i + 1 -- пропускаем лишний «смешайте»
                 i = i + 1
             end
             end
         else
         else
Строка 60: Строка 59:
     end
     end
     if #currentInputs > 0 then
     if #currentInputs > 0 then
         table.insert(recipes, { inputs = currentInputs, outputs = {}, action = "" })
         table.insert(steps, { inputs = currentInputs, outputs = {}, action = "" })
     end
     end
     return recipes
     return steps
end
end
-- ====================== ИСПРАВЛЕННЫЙ БЛОК ======================
local FILE_PATTERN = "(%[%[File:[^]]+%]%])"  -- правильный Lua-паттерн


local function buildRecipeStepHtml(rec)
local function buildRecipeStepHtml(rec)
     if not rec or (#rec.inputs == 0 and #rec.outputs == 0) or (rec.action == "" and #rec.outputs == 0) then  
     if not rec or (#rec.inputs == 0 and #rec.outputs == 0) then return nil end
        return nil  
 
    end
     local inner = html.create("div"):addClass("chem-recipe")
     local inner = html.create("div"):addClass("chem-recipe")
   
 
     -- INPUTS
     -- INPUTS
     local inputsDiv = html.create("div"):addClass("recipe-inputs")
     local inputsDiv = html.create("div"):addClass("recipe-inputs")
Строка 78: Строка 79:
     end
     end
     inner:node(inputsDiv)
     inner:node(inputsDiv)
   
 
     -- ACTION + КОЛБА
     -- ACTION (спрайт строго сверху)
     if rec.action and rec.action ~= "" then
     if rec.action and rec.action ~= "" then
         local actionDiv = inner:tag("div"):addClass("recipe-action")
         local actionDiv = inner:tag("div"):addClass("recipe-action")
         local hasSprite = mw.ustring.find(rec.action, "%[%[File:")  
         local content = actionDiv:tag("div"):addClass("recipe-action-content")
        if hasSprite then
 
            actionDiv:wikitext(rec.action)
         local sprite = '[[File:Beaker.png|36px|class=chem-beaker]]'
         else
        local actionText = rec.action
            actionDiv:wikitext('[[File:Beaker.png|36px|class=chem-beaker]]<br>' .. rec.action)
 
        -- Ищем кастомный спрайт
        local customSprite = mw.ustring.match(actionText, FILE_PATTERN)
        if customSprite then
            sprite = customSprite
            actionText = mw.text.trim(mw.ustring.gsub(actionText, FILE_PATTERN, ""))
         end
         end
        content:tag("div"):addClass("action-sprite"):wikitext(sprite)
        content:tag("div"):addClass("action-text"):wikitext(actionText)
     end
     end
   
 
     -- OUTPUTS
     -- OUTPUTS
     local outputsDiv = html.create("div"):addClass("recipe-outputs")
     local outputsDiv = html.create("div"):addClass("recipe-outputs")
Строка 97: Строка 106:
     end
     end
     inner:node(outputsDiv)
     inner:node(outputsDiv)
   
 
     return inner
     return inner
end
end
-- ============================================================


local function buildEffectsHtml(effects)
local function buildEffectsHtml(effects)
Строка 109: Строка 119:
     hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
     hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
     container:node(heading)
     container:node(heading)
     local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "effects")
     local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "effects")
     local inner = html.create("div"):addClass("chem-effects")
     local inner = html.create("div"):addClass("chem-effects")
Строка 128: Строка 139:
     local border = args.border or "#444"
     local border = args.border or "#444"
     local desc = args.desc or ""
     local desc = args.desc or ""
   
 
     local recipes = {}
     local variants = {}
     local i = 1
     local i = 1
     while true do
     while true do
Строка 135: Строка 146:
         local recipeStr = args[key]
         local recipeStr = args[key]
         if not recipeStr then break end
         if not recipeStr then break end
         local parsedList = parseRecipeString(recipeStr)
         local steps = parseRecipeString(recipeStr)
         for _, rec in ipairs(parsedList) do
         if #steps > 0 then
             table.insert(recipes, rec)
             table.insert(variants, {id = i, steps = steps})
         end
         end
         i = i + 1
         i = i + 1
     end
     end
   
 
     local container = html.create("div"):addClass("chem-card")
     local container = html.create("div"):addClass("chem-card")
     container:attr("style", "border-color:" .. border .. "; --card-accent:" .. color)
     container:attr("style", "border-color:" .. border .. "; --card-accent:" .. color)
     container:tag("div"):addClass("chem-name-header"):wikitext(name)
     container:tag("div"):addClass("chem-name-header"):wikitext(name)
   
 
     if #recipes > 0 then
     if #variants > 0 then
         local recipeBlock = html.create("div"):addClass("chem-recipe-block")
         local recipeBlock = html.create("div"):addClass("chem-recipe-block")
         local heading = html.create("div"):addClass("chem-heading"):attr("data-kind", "recipe")
         local heading = html.create("div"):addClass("chem-heading"):attr("data-kind", "recipe")
Строка 153: Строка 164:
         hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
         hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
         recipeBlock:node(heading)
         recipeBlock:node(heading)
       
 
         local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "recipe")
         local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "recipe")
         local stepsContainer = html.create("div"):addClass("chem-recipe-steps")
         local stepsContainer = html.create("div"):addClass("chem-recipe-steps")
          
 
        for _, rec in ipairs(recipes) do
         for vidx, variant in ipairs(variants) do
            local step = buildRecipeStepHtml(rec)
            if #variants > 1 then
            if step then stepsContainer:node(step) end
                local varHeader = stepsContainer:tag("div"):addClass("recipe-variant-header")
                varHeader:wikitext("Вариант " .. vidx)
            end
 
            for _, rec in ipairs(variant.steps) do
                local step = buildRecipeStepHtml(rec)
                if step then stepsContainer:node(step) end
            end
         end
         end
       
 
         collapsible:node(stepsContainer)
         collapsible:node(stepsContainer)
         recipeBlock:node(collapsible)
         recipeBlock:node(collapsible)
         container:node(recipeBlock)
         container:node(recipeBlock)
     end
     end
   
 
     local effectsBlock = buildEffectsHtml(args.effects)
     local effectsBlock = buildEffectsHtml(args.effects)
     if effectsBlock then container:node(effectsBlock) end
     if effectsBlock then container:node(effectsBlock) end
   
 
     if desc ~= "" then
     if desc ~= "" then
         container:tag("div"):addClass("chem-desc"):wikitext(desc)
         container:tag("div"):addClass("chem-desc"):wikitext(desc)
     end
     end
     return tostring(container)
     return tostring(container)
end
end


return p
return p

Текущая версия от 17:45, 3 апреля 2026

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

local p = {}
local mw = mw
local html = mw.html

local function getArgs(frame)
    if type(frame) == "table" and frame.args then return frame.args end
    return mw.getCurrentFrame():getParent().args or {}
end

local function splitLines(s)
    local t = {}
    if not s then return t end
    for line in mw.text.gsplit(s, "\n") do
        line = mw.text.trim(line)
        if line ~= "" then table.insert(t, line) end
    end
    return t
end

local function parseLineToItem(l)
    local name, qty = mw.ustring.match(l, "^(.*)%s*%[%s*([%d,%.]+)%s*%]%s*$")
    if not name then name = l; qty = "1" end
    return { name = mw.text.trim(name), qty = qty or "1" }
end

local function parseRecipeString(recipeStr)
    if not recipeStr or recipeStr == "" then return {} end
    local lines = splitLines(recipeStr)
    local steps = {}
    local currentInputs = {}
    local i = 1
    while i <= #lines do
        local line = lines[i]
        local lower = mw.ustring.lower(line)
        if lower:find("смеш") then
            if #currentInputs > 0 then
                local outputs = {}
                i = i + 1
                while i <= #lines do
                    local nextLine = lines[i]
                    local nextLower = mw.ustring.lower(nextLine)
                    if nextLower:find("смеш") then break end
                    table.insert(outputs, parseLineToItem(nextLine))
                    i = i + 1
                end
                table.insert(steps, {
                    inputs = currentInputs,
                    outputs = outputs,
                    action = mw.text.trim(line)
                })
                currentInputs = {}
            else
                i = i + 1
            end
        else
            table.insert(currentInputs, parseLineToItem(line))
            i = i + 1
        end
    end
    if #currentInputs > 0 then
        table.insert(steps, { inputs = currentInputs, outputs = {}, action = "" })
    end
    return steps
end

-- ====================== ИСПРАВЛЕННЫЙ БЛОК ======================
local FILE_PATTERN = "(%[%[File:[^]]+%]%])"   -- правильный Lua-паттерн

local function buildRecipeStepHtml(rec)
    if not rec or (#rec.inputs == 0 and #rec.outputs == 0) then return nil end

    local inner = html.create("div"):addClass("chem-recipe")

    -- INPUTS
    local inputsDiv = html.create("div"):addClass("recipe-inputs")
    for _, it in ipairs(rec.inputs) do
        local item = inputsDiv:tag("div"):addClass("recipe-item")
        item:wikitext('[[File:Beaker.png|36px|class=chem-beaker]] ' .. it.name .. ' [' .. it.qty .. ']')
    end
    inner:node(inputsDiv)

    -- ACTION (спрайт строго сверху)
    if rec.action and rec.action ~= "" then
        local actionDiv = inner:tag("div"):addClass("recipe-action")
        local content = actionDiv:tag("div"):addClass("recipe-action-content")

        local sprite = '[[File:Beaker.png|36px|class=chem-beaker]]'
        local actionText = rec.action

        -- Ищем кастомный спрайт
        local customSprite = mw.ustring.match(actionText, FILE_PATTERN)
        if customSprite then
            sprite = customSprite
            actionText = mw.text.trim(mw.ustring.gsub(actionText, FILE_PATTERN, ""))
        end

        content:tag("div"):addClass("action-sprite"):wikitext(sprite)
        content:tag("div"):addClass("action-text"):wikitext(actionText)
    end

    -- OUTPUTS
    local outputsDiv = html.create("div"):addClass("recipe-outputs")
    for _, it in ipairs(rec.outputs) do
        local item = outputsDiv:tag("div"):addClass("recipe-item")
        item:wikitext('[[File:Beaker.png|36px|class=chem-beaker]] ' .. it.name .. ' [' .. it.qty .. ']')
    end
    inner:node(outputsDiv)

    return inner
end
-- ============================================================

local function buildEffectsHtml(effects)
    if not effects or effects == "" or effects == "Нет" then return nil end
    local container = html.create("div"):addClass("chem-effects-block")
    local heading = html.create("div"):addClass("chem-heading"):attr("data-kind", "effects")
    local hcontent = heading:tag("div"):addClass("chem-heading-content")
    hcontent:tag("span"):addClass("heading-text"):wikitext("Эффекты")
    hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
    container:node(heading)

    local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "effects")
    local inner = html.create("div"):addClass("chem-effects")
    for effect in mw.text.gsplit(effects, "[;\n]") do
        effect = mw.text.trim(effect)
        if effect ~= "" then
            inner:tag("div"):addClass("chem-effect-line"):wikitext(effect)
        end
    end
    collapsible:node(inner)
    container:node(collapsible)
    return container
end

function p.card(frame)
    local args = getArgs(frame)
    local name = args.name or "—"
    local color = args.color or "#8cf"
    local border = args.border or "#444"
    local desc = args.desc or ""

    local variants = {}
    local i = 1
    while true do
        local key = i == 1 and "recipe" or ("recipe" .. i)
        local recipeStr = args[key]
        if not recipeStr then break end
        local steps = parseRecipeString(recipeStr)
        if #steps > 0 then
            table.insert(variants, {id = i, steps = steps})
        end
        i = i + 1
    end

    local container = html.create("div"):addClass("chem-card")
    container:attr("style", "border-color:" .. border .. "; --card-accent:" .. color)
    container:tag("div"):addClass("chem-name-header"):wikitext(name)

    if #variants > 0 then
        local recipeBlock = html.create("div"):addClass("chem-recipe-block")
        local heading = html.create("div"):addClass("chem-heading"):attr("data-kind", "recipe")
        local hcontent = heading:tag("div"):addClass("chem-heading-content")
        hcontent:tag("span"):addClass("heading-text"):wikitext("Рецепты")
        hcontent:tag("span"):addClass("collapse-btn"):wikitext("развернуть")
        recipeBlock:node(heading)

        local collapsible = html.create("div"):addClass("collapsible collapsed"):attr("data-kind", "recipe")
        local stepsContainer = html.create("div"):addClass("chem-recipe-steps")

        for vidx, variant in ipairs(variants) do
            if #variants > 1 then
                local varHeader = stepsContainer:tag("div"):addClass("recipe-variant-header")
                varHeader:wikitext("Вариант " .. vidx)
            end

            for _, rec in ipairs(variant.steps) do
                local step = buildRecipeStepHtml(rec)
                if step then stepsContainer:node(step) end
            end
        end

        collapsible:node(stepsContainer)
        recipeBlock:node(collapsible)
        container:node(recipeBlock)
    end

    local effectsBlock = buildEffectsHtml(args.effects)
    if effectsBlock then container:node(effectsBlock) end

    if desc ~= "" then
        container:tag("div"):addClass("chem-desc"):wikitext(desc)
    end

    return tostring(container)
end

return p