De Muysc cubun - Lengua Muisca

m
m
 
(No se muestran 29 ediciones intermedias del mismo usuario)
Línea 1: Línea 1:
 +
-- ============================================================
 +
-- Diego F. Gómez -- 2024-2026
 +
-- Módulo de conjugación para verbos muysca en mediawiki
 +
-- ============================================================
 +
 
local p = {}
 
local p = {}
  
-- Limpia el nombre de la página:
+
-- ============================================================
-- 1) quita espacios sobrantes
+
-- DICCIONARIO DE TEXTOS
-- 2) quita sufijos tipo "(2)", "(3)", etc.
+
-- ============================
 +
 
 +
local TEXTS = {
 +
    ui = {
 +
        expand = "Conjugar",
 +
        collapse = "Ocultar",
 +
        error_root = "Error: No se pudo extraer la raíz de ",
 +
        main_title = "Paradigma verbal: ",
 +
        col_person = "Persona"
 +
    },
 +
    series = {
 +
        s1 = {
 +
            title = "Serie de Flexión Verbal",
 +
            headers = {"Perfectivo", "Imperfectivo", "Irrealis", "Destinativo"}
 +
        },
 +
        s2 = {
 +
            title = "Serie de Nominalización",
 +
            headers = {"Resultativa", "Factual", "Irrealis", "Prospectiva"}
 +
        },
 +
        s3 = {
 +
            title = "Modo Imperativo",
 +
            headers = {"Forma Única"}
 +
        }
 +
    },
 +
    persons = {
 +
        sg1 = "1.ª sg.", sg2 = "2.ª sg.", sg3 = "3.ª",
 +
        pl1 = "1.ª pl.", pl2 = "2.ª pl.",
 +
        imp_sg = "2.ª sg. (Imp.)",  
 +
        imp_pl = "2.ª pl. (Imp.)"
 +
    }
 +
}
 +
 
 +
-- ============================================================
 +
-- LÓGICA PRIVADA
 +
-- ============================================================
 +
 
 
local function cleanPageName(pageName)
 
local function cleanPageName(pageName)
 +
    if not mw then return pageName:gsub("%s*%b()", ""):gsub("%d+", ""):gsub("^%s*(.-)%s*$", "%1") end
 
     pageName = mw.text.trim(pageName or "")
 
     pageName = mw.text.trim(pageName or "")
     pageName = mw.ustring.gsub(pageName, "%s*%b()", "") -- quita (2), (3), etc.
+
     pageName = mw.ustring.gsub(pageName, "%s*%b()", "")
     return pageName
+
    pageName = mw.ustring.gsub(pageName, "%d+", "")
 +
     return mw.text.trim(pageName)
 
end
 
end
  
-- Obtiene la raíz según el tipo verbal
+
local function getRoot(pageName, clase, manualRoot)
local function getRoot(pageName, clase)
+
    if manualRoot and manualRoot ~= "" then return manualRoot end
     local root = cleanPageName(pageName)
+
     local fullVerb = cleanPageName(pageName)
 +
    local root = fullVerb
 +
    if clase == "sq." then root = fullVerb:gsub("squa$", "")
 +
    elseif clase == "su." then root = fullVerb:gsub("suca$", "") end
 +
    if root == fullVerb then root = fullVerb:gsub("s[uq]ua$", "") end
 +
    return root
 +
end
  
    clase = mw.text.trim(clase or "")
+
-- ============================================================
 +
-- NUEVA LÓGICA DE CLASE (Parsing)
 +
-- ============================================================
  
     if clase == "sq." then
+
local function parseVerbClass(raw)
         -- quita "squa" al final
+
     raw = (raw or ""):lower()
         root = mw.ustring.gsub(root, "squa$", "")
+
    local info = {
     elseif clase == "su." then
+
        type = "",            -- "sq." o "su."
        -- quita "suca" al final
+
        isTransitive = false,
         root = mw.ustring.gsub(root, "suca$", "")
+
        isIntransitive = false,
 +
        isQuy = false,        -- Para iquy.
 +
         isQuySimple = false,  -- Para quy. (ua-squa)
 +
        hasRootI = false,      -- Para raíces en i. (hui-a)
 +
         raw = raw
 +
    }
 +
   
 +
    if raw:find("sq") then info.type = "sq." end
 +
    if raw:find("su") then info.type = "su." end
 +
    if raw:find("tr%.") then info.isTransitive = true end
 +
     if raw:find("intr%.") then info.isIntransitive = true end
 +
   
 +
    -- Detectamos raíces históricas
 +
    if raw:find("iquy") then
 +
         info.isQuy = true
 +
    elseif raw:find("quy%.") then
 +
        info.isQuySimple = true
 
     end
 
     end
 +
   
 +
    if raw:find(" i%.") or raw:find(" i ") or raw:sub(-2) == " i" then
 +
        info.hasRootI = true
 +
    end
 +
   
 +
    return info
 +
end
 +
 +
-- ============================================================
 +
-- EL CEREBRO: Lógica Pura
 +
-- ============================================================
 +
 +
function p.getConjugations(pageName, claseRaw, manualRoot)
 +
    local classInfo = parseVerbClass(claseRaw)
 +
    local root = getRoot(pageName, classInfo.type, manualRoot)
 +
   
 +
    local results = { root = root, classInfo = classInfo, series = {} }
 +
 +
    -- Configuración de Series
 +
    local config = {
 +
        {
 +
            id = "S1", -- Flexión Verbal
 +
            title = TEXTS.series.s1.title,
 +
            headers = TEXTS.series.s1.headers,
 +
            persons = {{TEXTS.persons.sg1, "z-"}, {TEXTS.persons.sg2, "m-"}, {TEXTS.persons.sg3, "a-"}, {TEXTS.persons.pl1, "chi-"}, {TEXTS.persons.pl2, "mi-"}},
 +
            suffixes = {
 +
                perf = "",
 +
                imperf = (classInfo.type == "su." and "-suca" or "-squa"),
 +
                irreal = (classInfo.type == "su." and "-nynga" or "-nga"),
 +
                destinativo = "-iua"
 +
            },
 +
            applyRule = true,
 +
            allowInfix = true,
 +
            hasNegation = true -- Habilita filas negativas
 +
        },
 +
        {
 +
            id = "S2", -- Nominalización
 +
            title = TEXTS.series.s2.title,
 +
            headers = TEXTS.series.s2.headers,
 +
            persons = {{TEXTS.persons.sg1, "cha-"}, {TEXTS.persons.sg2, "ma-"}, {TEXTS.persons.sg3, ""}, {TEXTS.persons.pl1, "chi-"}, {TEXTS.persons.pl2, "mi-"}},
 +
            suffixes = {
 +
                perf = (classInfo.type == "su." and "-ua" or "-ia"),
 +
                imperf = (classInfo.type == "su." and "-suca" or "-sca"),
 +
                irreal = (classInfo.type == "su." and "-nynga" or "-nga"),
 +
                prospectivo = (classInfo.type == "su." and "-nynguepqua" or "-nguepqua")
 +
            },
 +
            applyRule = false,
 +
            allowInfix = false
 +
        },
 +
        {
 +
            id = "S3", -- Imperativos
 +
            title = TEXTS.series.s3.title,
 +
            headers = TEXTS.series.s3.headers,
 +
            persons = {
 +
                {TEXTS.persons.imp_sg, "", "-u"},
 +
                {TEXTS.persons.imp_pl, "", "-ua"}
 +
            },
 +
            applyRule = false,
 +
            allowInfix = false
 +
        }
 +
    }
 +
 +
    local firstChar = root:sub(1,1):lower()
 +
    local firstTwo = root:sub(1,2):lower()
 +
 +
    for _, s in ipairs(config) do
 +
        local serieData = { id = s.id, title = s.title, headers = s.headers, rows = {} }
 +
       
 +
        for _, pData in ipairs(s.persons) do
 +
            local label, prefix, personalSuffix = pData[1], pData[2], pData[3]
 +
           
 +
            -- 1. Regla Especial de Prefijo (z- -> i-)
 +
            if s.applyRule and label == TEXTS.persons.sg1 then
 +
                if firstChar == "n" or firstChar == "z" or firstChar == "t" or firstTwo == "ch" then
 +
                    prefix = "i-"
 +
                end
 +
            end
 +
 +
            -- 2. LÓGICA DEL INFIJO -b- Y CAMBIOS EN LA RAÍZ
 +
            local currentInfix = ""
 +
            local currentRoot = root
 +
           
 +
            if s.allowInfix and classInfo.isTransitive and not classInfo.isIntransitive then
 +
                if prefix ~= "" then
 +
                    local r1 = firstChar
 +
                    if r1 == "u" then
 +
                        currentInfix = "g-"
 +
                    elseif r1 == "n" then
 +
                        currentInfix = "m-"
 +
                    elseif r1 == "h" and classInfo.type == "su." then
 +
                        local vocal = root:sub(2,2)
 +
                        currentInfix = "m" .. vocal .. "-"
 +
                        currentRoot = "h" .. root:sub(3)
 +
                    elseif r1 == "b" then
 +
                        currentInfix = "m-"
 +
                        currentRoot = "m" .. root:sub(2)
 +
                    else
 +
                        currentInfix = (prefix == "m-" and "m-" or "b-")
 +
                    end
 +
                end
 +
            end
  
    return root
+
          -- 3. LÓGICA DE SUFIJOS
end
+
            local perfSuffix = personalSuffix or (s.suffixes and s.suffixes.perf) or ""
 +
            local imperfSuffix = (s.suffixes and s.suffixes.imperf) or ""
 +
            local irrealSuffix = (s.suffixes and s.suffixes.irreal) or ""
 +
            local prospectivoSuffix = (s.suffixes and s.suffixes.prospectivo) or ""
 +
            local destinativoSuffix = (s.suffixes and s.suffixes.destinativo) or ""
 +
           
 +
            local rootPerf = currentRoot
 +
            local rootImperf = currentRoot
 +
            local rootIrreal = currentRoot
 +
            local rootProspectivo = currentRoot
 +
            local rootDestinativo = currentRoot
  
function p.render(frame)
+
            -- 4. LÓGICA DE RAÍCES EN -I (Serie 2)
    local args = frame:getParent().args
+
            if classInfo.hasRootI and s.id == "S2" then
 +
                rootPerf = currentRoot .. "i"
 +
                rootImperf = currentRoot .. "i"
 +
                rootIrreal = currentRoot .. "i"
 +
                rootProspectivo = currentRoot .. "i"
 +
            end
  
    -- clase = sq. o su.
+
            -- 5. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN IQUY (z/i)
    local clase = mw.text.trim(args.clase or "")
+
            if classInfo.isQuy then
 +
                if s.id == "S1" then
 +
                    perfSuffix = "quy"
 +
                elseif s.id == "S2" then
 +
                    rootPerf = currentRoot .. "z"
 +
                    perfSuffix = "-a"
 +
                    rootImperf = currentRoot .. "i"
 +
                    rootIrreal = currentRoot .. "i"
 +
                elseif s.id == "S3" then
 +
                    rootPerf = currentRoot .. "z"
 +
                end
 +
            end
  
    -- transitividad opcional (intr. / tr.), por si luego quieres mostrarla
+
            -- 6. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN -QUY SIMPLE (ej: ua-squa)
    local trans = mw.text.trim(args.trans or "")
+
            if classInfo.isQuySimple then
 +
                if s.id == "S1" then
 +
                    perfSuffix = "quy"
 +
                elseif s.id == "S2" then
 +
                    rootPerf = currentRoot .. "c"
 +
                    perfSuffix = "-a"
 +
                elseif s.id == "S3" then
 +
                    rootPerf = currentRoot .. "c"
 +
                end
 +
            end
 +
           
 +
            rootProspectivo = rootIrreal
  
    local pageName = mw.title.getCurrentTitle().text
+
            -- 7. CONSTRUCCIÓN DE FILA AFIRMATIVA
    local root = getRoot(pageName, clase)
+
            table.insert(serieData.rows, {
 +
                label = label,
 +
                status = "pos", -- Marca para estilo visual
 +
                perf = prefix .. currentInfix .. rootPerf .. perfSuffix,
 +
                imperf = prefix .. currentInfix .. rootImperf .. imperfSuffix,
 +
                irreal = prefix .. currentInfix .. rootIrreal .. irrealSuffix,
 +
                prospectivo = prefix .. currentInfix .. rootProspectivo .. prospectivoSuffix,
 +
                destinativo = prefix .. currentInfix .. rootDestinativo .. destinativoSuffix
 +
            })
  
    -- Si no hay raíz válida, no mostrar nada
+
            -- 8. LÓGICA DE NEGACIÓN (Solo para Serie 1)
    if root == "" then
+
            if s.hasNegation then
        return "<span style='color:red;'>Error: no se pudo identificar la raíz.</span>"
+
                -- REGLA DE RAÍZ PURA: La negación ignora extensiones históricas en S1
 +
                local negRoot = currentRoot
 +
               
 +
                table.insert(serieData.rows, {
 +
                    label = "", -- Se puede dejar vacío para usar rowspan o marcar con (-)
 +
                    status = "neg",
 +
                    perf = prefix .. currentInfix .. negRoot .. "-za",
 +
                    imperf = prefix .. currentInfix .. negRoot .. imperfSuffix .. "-za",
 +
                    irreal = prefix .. currentInfix .. negRoot .. "-zi-nga",
 +
                    destinativo = prefix .. currentInfix .. negRoot .. "-za-n iua"
 +
                })
 +
            end
 +
        end
 +
        table.insert(results.series, serieData)
 
     end
 
     end
 +
    return results
 +
end
  
     local forms = {
+
-- ============================================================
        ["1sg"] = "z-" .. root,
+
-- LA VISTA: Usado por la Wiki
        ["2sg"] = "m-" .. root,
+
-- ============================================================
        ["3"]  = "a-" .. root,
+
local function drawTable(serie)
        ["1pl"] = "chi-" .. root,
+
     local wrapper = mw.html.create("div"):css("margin-bottom", "1.5em"):css("font-size", "14px")
         ["2pl"] = "mi-" .. root
+
    wrapper:tag("div"):css("font-weight", "bold"):css("font-size", "1.1em")
    }
+
         :css("margin-bottom", "0.4em"):css("color", "#202122"):wikitext(serie.title)
  
     local html = mw.html.create("table")
+
     local tableHtml = wrapper:tag("table"):addClass("wikitable")
        :addClass("wikitable")
+
         :css("width", "100%"):css("max-width", "700px"):css("margin", "0")
         :css("width", "auto")
 
        :css("text-align", "left")
 
  
    -- Cabecera
+
     local headerRow = tableHtml:tag("tr")
     local header = html:tag("tr")
+
     headerRow:tag("th"):wikitext(TEXTS.ui.col_person)
     header:tag("th"):wikitext("Persona")
+
   
     header:tag("th"):wikitext("Forma")
+
    if serie.id == "S3" then
 +
        headerRow:tag("th"):wikitext(serie.headers[1])
 +
     else
 +
        for _, hName in ipairs(serie.headers) do
 +
            headerRow:tag("th"):wikitext(hName)
 +
        end
 +
    end
  
     -- Filas
+
     for i, row in ipairs(serie.rows) do
    local order = {"1sg", "2sg", "3", "1pl", "2pl"}
+
        local tr = tableHtml:tag("tr")
 +
       
 +
        -- Estética para negación
 +
        if row.status == "neg" then
 +
            tr:css("background-color", "#f8f9fa"):css("color", "#54595d")
 +
            local tdLabel = tr:tag("td"):css("text-align", "right"):css("font-size", "0.9em")
 +
                :css("font-style", "italic"):wikitext("neg.")
 +
        else
 +
            tr:tag("td"):css("font-weight", "bold"):wikitext(row.label)
 +
        end
  
    for _, person in ipairs(order) do
+
        tr:tag("td"):wikitext(row.perf)
         local row = html:tag("tr")
+
          
         row:tag("td"):wikitext(person)
+
        if serie.id == "S1" then
        row:tag("td"):wikitext(forms[person])
+
            tr:tag("td"):wikitext(row.imperf)
 +
            tr:tag("td"):wikitext(row.irreal)
 +
            tr:tag("td"):wikitext(row.destinativo)
 +
         elseif serie.id == "S2" then
 +
            tr:tag("td"):wikitext(row.imperf)
 +
            tr:tag("td"):wikitext(row.irreal)
 +
            tr:tag("td"):wikitext(row.prospectivo)
 +
        end
 
     end
 
     end
 +
    return wrapper
 +
end
  
     -- Puedes añadir info arriba si quieres
+
function p.render(frame)
     local output = ""
+
     local args = frame:getParent().args
 +
     local pageName = mw.title.getCurrentTitle().text
 +
    local claseInput = args.clase or args.tipo or args.cat or args[1] or ""
 +
    local data = p.getConjugations(pageName, claseInput, args.raiz)
  
     if clase ~= "" or trans ~= "" then
+
     if data.root == "" or data.root == pageName then
         output = output .. string.format(
+
         return "<span style='color:red;'>" .. TEXTS.ui.error_root .. pageName .. ".</span>"
            "<div><b>Clase:</b> %s %s</div>",
 
            clase,
 
            trans
 
        )
 
 
     end
 
     end
  
     output = output .. string.format(
+
     local mainDiv = mw.html.create("div"):addClass("mw-collapsible"):addClass("mw-collapsed")
         "<div><b>Raíz detectada:</b> %s</div>",
+
         :css("border", "1px solid #a2a9b1"):css("padding", "15px"):css("background-color", "#fcfcfc")
         root
+
         :attr("data-expandtext", TEXTS.ui.expand):attr("data-collapsetext", TEXTS.ui.collapse)
    )
 
  
     output = output .. tostring(html)
+
     mainDiv:tag("div"):css("font-weight", "bold"):css("border-bottom", "1px solid #a2a9b1")
 +
        :css("margin-bottom", "15px"):css("padding-bottom", "5px")
 +
        :wikitext(TEXTS.ui.main_title .. data.root .. " (" .. data.classInfo.raw .. ")")
  
     return output
+
    local contentDiv = mainDiv:tag("div"):addClass("mw-collapsible-content")
 +
    for _, serie in ipairs(data.series) do
 +
        contentDiv:node(drawTable(serie))
 +
    end
 +
     return tostring(mainDiv)
 
end
 
end
  
 
return p
 
return p

Revisión actual - 13:02 31 mar 2026

La documentación para este módulo puede ser creada en Módulo:Conjugador/doc

-- ============================================================
-- Diego F. Gómez -- 2024-2026
-- Módulo de conjugación para verbos muysca en mediawiki
-- ============================================================

local p = {}

-- ============================================================
-- DICCIONARIO DE TEXTOS 
-- ============================

local TEXTS = {
    ui = {
        expand = "Conjugar",
        collapse = "Ocultar",
        error_root = "Error: No se pudo extraer la raíz de ",
        main_title = "Paradigma verbal: ",
        col_person = "Persona"
    },
    series = {
        s1 = {
            title = "Serie de Flexión Verbal",
            headers = {"Perfectivo", "Imperfectivo", "Irrealis", "Destinativo"}
        },
        s2 = {
            title = "Serie de Nominalización",
            headers = {"Resultativa", "Factual", "Irrealis", "Prospectiva"}
        },
        s3 = {
            title = "Modo Imperativo",
            headers = {"Forma Única"}
        }
    },
    persons = {
        sg1 = "1.ª sg.", sg2 = "2.ª sg.", sg3 = "3.ª",
        pl1 = "1.ª pl.", pl2 = "2.ª pl.",
        imp_sg = "2.ª sg. (Imp.)", 
        imp_pl = "2.ª pl. (Imp.)"
    }
}

-- ============================================================
-- LÓGICA PRIVADA
-- ============================================================

local function cleanPageName(pageName)
    if not mw then return pageName:gsub("%s*%b()", ""):gsub("%d+", ""):gsub("^%s*(.-)%s*$", "%1") end
    pageName = mw.text.trim(pageName or "")
    pageName = mw.ustring.gsub(pageName, "%s*%b()", "")
    pageName = mw.ustring.gsub(pageName, "%d+", "")
    return mw.text.trim(pageName)
end

local function getRoot(pageName, clase, manualRoot)
    if manualRoot and manualRoot ~= "" then return manualRoot end
    local fullVerb = cleanPageName(pageName)
    local root = fullVerb
    if clase == "sq." then root = fullVerb:gsub("squa$", "")
    elseif clase == "su." then root = fullVerb:gsub("suca$", "") end
    if root == fullVerb then root = fullVerb:gsub("s[uq]ua$", "") end
    return root
end

-- ============================================================
-- NUEVA LÓGICA DE CLASE (Parsing)
-- ============================================================

local function parseVerbClass(raw)
    raw = (raw or ""):lower()
    local info = {
        type = "",             -- "sq." o "su."
        isTransitive = false,
        isIntransitive = false,
        isQuy = false,         -- Para iquy.
        isQuySimple = false,   -- Para quy. (ua-squa)
        hasRootI = false,      -- Para raíces en i. (hui-a)
        raw = raw
    }
    
    if raw:find("sq") then info.type = "sq." end
    if raw:find("su") then info.type = "su." end
    if raw:find("tr%.") then info.isTransitive = true end
    if raw:find("intr%.") then info.isIntransitive = true end
    
    -- Detectamos raíces históricas
    if raw:find("iquy") then 
        info.isQuy = true 
    elseif raw:find("quy%.") then
        info.isQuySimple = true
    end
    
    if raw:find(" i%.") or raw:find(" i ") or raw:sub(-2) == " i" then
        info.hasRootI = true
    end
    
    return info
end

-- ============================================================
-- EL CEREBRO: Lógica Pura
-- ============================================================

function p.getConjugations(pageName, claseRaw, manualRoot)
    local classInfo = parseVerbClass(claseRaw)
    local root = getRoot(pageName, classInfo.type, manualRoot)
    
    local results = { root = root, classInfo = classInfo, series = {} }

    -- Configuración de Series
    local config = {
        {
            id = "S1", -- Flexión Verbal
            title = TEXTS.series.s1.title, 
            headers = TEXTS.series.s1.headers,
            persons = {{TEXTS.persons.sg1, "z-"}, {TEXTS.persons.sg2, "m-"}, {TEXTS.persons.sg3, "a-"}, {TEXTS.persons.pl1, "chi-"}, {TEXTS.persons.pl2, "mi-"}},
            suffixes = {
                perf = "", 
                imperf = (classInfo.type == "su." and "-suca" or "-squa"),
                irreal = (classInfo.type == "su." and "-nynga" or "-nga"),
                destinativo = "-iua"
            },
            applyRule = true,
            allowInfix = true,
            hasNegation = true -- Habilita filas negativas
        },
        {
            id = "S2", -- Nominalización
            title = TEXTS.series.s2.title, 
            headers = TEXTS.series.s2.headers,
            persons = {{TEXTS.persons.sg1, "cha-"}, {TEXTS.persons.sg2, "ma-"}, {TEXTS.persons.sg3, ""}, {TEXTS.persons.pl1, "chi-"}, {TEXTS.persons.pl2, "mi-"}},
            suffixes = {
                perf = (classInfo.type == "su." and "-ua" or "-ia"), 
                imperf = (classInfo.type == "su." and "-suca" or "-sca"),
                irreal = (classInfo.type == "su." and "-nynga" or "-nga"),
                prospectivo = (classInfo.type == "su." and "-nynguepqua" or "-nguepqua")
            },
            applyRule = false,
            allowInfix = false
        },
        {
            id = "S3", -- Imperativos
            title = TEXTS.series.s3.title, 
            headers = TEXTS.series.s3.headers,
            persons = {
                {TEXTS.persons.imp_sg, "", "-u"}, 
                {TEXTS.persons.imp_pl, "", "-ua"}
            },
            applyRule = false,
            allowInfix = false
        }
    }

    local firstChar = root:sub(1,1):lower()
    local firstTwo = root:sub(1,2):lower()

    for _, s in ipairs(config) do
        local serieData = { id = s.id, title = s.title, headers = s.headers, rows = {} }
        
        for _, pData in ipairs(s.persons) do
            local label, prefix, personalSuffix = pData[1], pData[2], pData[3]
            
            -- 1. Regla Especial de Prefijo (z- -> i-)
            if s.applyRule and label == TEXTS.persons.sg1 then
                if firstChar == "n" or firstChar == "z" or firstChar == "t" or firstTwo == "ch" then
                    prefix = "i-"
                end
            end

            -- 2. LÓGICA DEL INFIJO -b- Y CAMBIOS EN LA RAÍZ
            local currentInfix = ""
            local currentRoot = root 
            
            if s.allowInfix and classInfo.isTransitive and not classInfo.isIntransitive then
                if prefix ~= "" then
                    local r1 = firstChar
                    if r1 == "u" then
                        currentInfix = "g-"
                    elseif r1 == "n" then
                        currentInfix = "m-"
                    elseif r1 == "h" and classInfo.type == "su." then
                        local vocal = root:sub(2,2)
                        currentInfix = "m" .. vocal .. "-"
                        currentRoot = "h" .. root:sub(3)
                    elseif r1 == "b" then
                        currentInfix = "m-"
                        currentRoot = "m" .. root:sub(2)
                    else
                        currentInfix = (prefix == "m-" and "m-" or "b-")
                    end
                end
            end

           -- 3. LÓGICA DE SUFIJOS
            local perfSuffix = personalSuffix or (s.suffixes and s.suffixes.perf) or ""
            local imperfSuffix = (s.suffixes and s.suffixes.imperf) or ""
            local irrealSuffix = (s.suffixes and s.suffixes.irreal) or ""
            local prospectivoSuffix = (s.suffixes and s.suffixes.prospectivo) or ""
            local destinativoSuffix = (s.suffixes and s.suffixes.destinativo) or ""
            
            local rootPerf = currentRoot
            local rootImperf = currentRoot
            local rootIrreal = currentRoot
            local rootProspectivo = currentRoot
            local rootDestinativo = currentRoot

            -- 4. LÓGICA DE RAÍCES EN -I (Serie 2)
            if classInfo.hasRootI and s.id == "S2" then
                rootPerf = currentRoot .. "i"
                rootImperf = currentRoot .. "i"
                rootIrreal = currentRoot .. "i"
                rootProspectivo = currentRoot .. "i"
            end

            -- 5. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN IQUY (z/i)
            if classInfo.isQuy then
                if s.id == "S1" then
                    perfSuffix = "quy"
                elseif s.id == "S2" then
                    rootPerf = currentRoot .. "z"
                    perfSuffix = "-a"
                    rootImperf = currentRoot .. "i"
                    rootIrreal = currentRoot .. "i"
                elseif s.id == "S3" then
                    rootPerf = currentRoot .. "z"
                end
            end

            -- 6. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN -QUY SIMPLE (ej: ua-squa)
            if classInfo.isQuySimple then
                if s.id == "S1" then
                    perfSuffix = "quy"
                elseif s.id == "S2" then
                    rootPerf = currentRoot .. "c"
                    perfSuffix = "-a"
                elseif s.id == "S3" then
                    rootPerf = currentRoot .. "c"
                end
            end
            
            rootProspectivo = rootIrreal

            -- 7. CONSTRUCCIÓN DE FILA AFIRMATIVA
            table.insert(serieData.rows, {
                label = label,
                status = "pos", -- Marca para estilo visual
                perf = prefix .. currentInfix .. rootPerf .. perfSuffix,
                imperf = prefix .. currentInfix .. rootImperf .. imperfSuffix,
                irreal = prefix .. currentInfix .. rootIrreal .. irrealSuffix,
                prospectivo = prefix .. currentInfix .. rootProspectivo .. prospectivoSuffix,
                destinativo = prefix .. currentInfix .. rootDestinativo .. destinativoSuffix
            })

            -- 8. LÓGICA DE NEGACIÓN (Solo para Serie 1)
            if s.hasNegation then
                -- REGLA DE RAÍZ PURA: La negación ignora extensiones históricas en S1
                local negRoot = currentRoot
                
                table.insert(serieData.rows, {
                    label = "", -- Se puede dejar vacío para usar rowspan o marcar con (-)
                    status = "neg",
                    perf = prefix .. currentInfix .. negRoot .. "-za",
                    imperf = prefix .. currentInfix .. negRoot .. imperfSuffix .. "-za",
                    irreal = prefix .. currentInfix .. negRoot .. "-zi-nga",
                    destinativo = prefix .. currentInfix .. negRoot .. "-za-n iua"
                })
            end
        end
        table.insert(results.series, serieData)
    end
    return results
end

-- ============================================================
-- LA VISTA: Usado por la Wiki
-- ============================================================
local function drawTable(serie)
    local wrapper = mw.html.create("div"):css("margin-bottom", "1.5em"):css("font-size", "14px")
    wrapper:tag("div"):css("font-weight", "bold"):css("font-size", "1.1em")
        :css("margin-bottom", "0.4em"):css("color", "#202122"):wikitext(serie.title)

    local tableHtml = wrapper:tag("table"):addClass("wikitable")
        :css("width", "100%"):css("max-width", "700px"):css("margin", "0")

    local headerRow = tableHtml:tag("tr")
    headerRow:tag("th"):wikitext(TEXTS.ui.col_person)
    
    if serie.id == "S3" then
        headerRow:tag("th"):wikitext(serie.headers[1])
    else
        for _, hName in ipairs(serie.headers) do
            headerRow:tag("th"):wikitext(hName)
        end
    end

    for i, row in ipairs(serie.rows) do
        local tr = tableHtml:tag("tr")
        
        -- Estética para negación
        if row.status == "neg" then
            tr:css("background-color", "#f8f9fa"):css("color", "#54595d")
            local tdLabel = tr:tag("td"):css("text-align", "right"):css("font-size", "0.9em")
                :css("font-style", "italic"):wikitext("neg.")
        else
            tr:tag("td"):css("font-weight", "bold"):wikitext(row.label)
        end

        tr:tag("td"):wikitext(row.perf)
        
        if serie.id == "S1" then
            tr:tag("td"):wikitext(row.imperf)
            tr:tag("td"):wikitext(row.irreal)
            tr:tag("td"):wikitext(row.destinativo)
        elseif serie.id == "S2" then
            tr:tag("td"):wikitext(row.imperf)
            tr:tag("td"):wikitext(row.irreal)
            tr:tag("td"):wikitext(row.prospectivo)
        end
    end
    return wrapper
end

function p.render(frame)
    local args = frame:getParent().args
    local pageName = mw.title.getCurrentTitle().text
    local claseInput = args.clase or args.tipo or args.cat or args[1] or ""
    local data = p.getConjugations(pageName, claseInput, args.raiz)

    if data.root == "" or data.root == pageName then
        return "<span style='color:red;'>" .. TEXTS.ui.error_root .. pageName .. ".</span>"
    end

    local mainDiv = mw.html.create("div"):addClass("mw-collapsible"):addClass("mw-collapsed")
        :css("border", "1px solid #a2a9b1"):css("padding", "15px"):css("background-color", "#fcfcfc")
        :attr("data-expandtext", TEXTS.ui.expand):attr("data-collapsetext", TEXTS.ui.collapse)

    mainDiv:tag("div"):css("font-weight", "bold"):css("border-bottom", "1px solid #a2a9b1")
        :css("margin-bottom", "15px"):css("padding-bottom", "5px")
        :wikitext(TEXTS.ui.main_title .. data.root .. " (" .. data.classInfo.raw .. ")")

    local contentDiv = mainDiv:tag("div"):addClass("mw-collapsible-content")
    for _, serie in ipairs(data.series) do
        contentDiv:node(drawTable(serie))
    end
    return tostring(mainDiv)
end

return p