De Muysc cubun - Lengua Muisca

m
m
 
(No se muestran 4 ediciones intermedias del mismo usuario)
Línea 8: Línea 8:
 
-- ============================================================
 
-- ============================================================
 
-- DICCIONARIO DE TEXTOS  
 
-- DICCIONARIO DE TEXTOS  
-- ============================================================
+
-- ============================
 +
 
 
local TEXTS = {
 
local TEXTS = {
 
     ui = {
 
     ui = {
Línea 19: Línea 20:
 
     series = {
 
     series = {
 
         s1 = {
 
         s1 = {
             title = "Serie de Flexión Verbal (Independiente)",
+
             title = "Serie de Flexión Verbal",
             headers = {"Perfectivo", "Imperfectivo", "Irrealis"}
+
             headers = {"Perfectivo", "Imperfectivo", "Irrealis", "Destinativo"}
 
         },
 
         },
 
         s2 = {
 
         s2 = {
 
             title = "Serie de Nominalización",
 
             title = "Serie de Nominalización",
             headers = {"Resultativa", "Factual", "Irrealis"}
+
             headers = {"Resultativa", "Factual", "Irrealis", "Prospectiva"}
 
         },
 
         },
 
         s3 = {
 
         s3 = {
Línea 44: Línea 45:
  
 
local function cleanPageName(pageName)
 
local function cleanPageName(pageName)
     if not mw then return pageName:gsub("%s*%b()", ""):gsub("%d+", ""):trim() end
+
     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()", "")
 
     pageName = mw.ustring.gsub(pageName, "%s*%b()", "")
 
     pageName = mw.ustring.gsub(pageName, "%d+", "")
 
     pageName = mw.ustring.gsub(pageName, "%d+", "")
 
     return mw.text.trim(pageName)
 
     return mw.text.trim(pageName)
end
 
 
local function normalizeClass(rawClass)
 
    rawClass = (rawClass or ""):lower()
 
    if rawClass:find("sq") then return "sq."
 
    elseif rawClass:find("su") then return "su." end
 
    return ""
 
 
end
 
end
  
Línea 78: Línea 72:
 
         isTransitive = false,
 
         isTransitive = false,
 
         isIntransitive = false,
 
         isIntransitive = false,
         isQuy = false,
+
         isQuy = false,         -- Para iquy.
 +
        isQuySimple = false,  -- Para quy. (ua-squa)
 +
        hasRootI = false,      -- Para raíces en i. (hui-a)
 
         raw = raw
 
         raw = raw
 
     }
 
     }
Línea 86: Línea 82:
 
     if raw:find("tr%.") then info.isTransitive = true end
 
     if raw:find("tr%.") then info.isTransitive = true end
 
     if raw:find("intr%.") then info.isIntransitive = true end
 
     if raw:find("intr%.") then info.isIntransitive = true end
     if raw:find("quy%.") then info.isQuy = 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
 
     return info
Línea 108: Línea 114:
 
             headers = TEXTS.series.s1.headers,
 
             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-"}},
 
             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 = "-nga"},
+
             suffixes = {
 +
                perf = "",  
 +
                imperf = (classInfo.type == "su." and "-suca" or "-squa"),
 +
                irreal = (classInfo.type == "su." and "-nynga" or "-nga"),
 +
                destinativo = "-iua"
 +
            },
 
             applyRule = true,
 
             applyRule = true,
             allowInfix = true
+
             allowInfix = true,
 +
            hasNegation = true -- Habilita filas negativas
 
         },
 
         },
 
         {
 
         {
Línea 117: Línea 129:
 
             headers = TEXTS.series.s2.headers,
 
             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-"}},
 
             persons = {{TEXTS.persons.sg1, "cha-"}, {TEXTS.persons.sg2, "ma-"}, {TEXTS.persons.sg3, ""}, {TEXTS.persons.pl1, "chi-"}, {TEXTS.persons.pl2, "mi-"}},
             suffixes = {perf = "-ia", imperf = "-sca", irreal = "-nga"},
+
             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,
 
             applyRule = false,
 
             allowInfix = false
 
             allowInfix = false
Línea 141: Línea 158:
 
          
 
          
 
         for _, pData in ipairs(s.persons) do
 
         for _, pData in ipairs(s.persons) do
            -- CORRECCIÓN AQUÍ: Capturamos el tercer valor (sufijo personal)
 
 
             local label, prefix, personalSuffix = pData[1], pData[2], pData[3]
 
             local label, prefix, personalSuffix = pData[1], pData[2], pData[3]
 
              
 
              
Línea 162: Línea 178:
 
                     elseif r1 == "n" then
 
                     elseif r1 == "n" then
 
                         currentInfix = "m-"
 
                         currentInfix = "m-"
                     elseif r1 == "h" then
+
                     elseif r1 == "h" and classInfo.type == "su." then
 
                         local vocal = root:sub(2,2)
 
                         local vocal = root:sub(2,2)
 
                         currentInfix = "m" .. vocal .. "-"
 
                         currentInfix = "m" .. vocal .. "-"
Línea 170: Línea 186:
 
                         currentRoot = "m" .. root:sub(2)
 
                         currentRoot = "m" .. root:sub(2)
 
                     else
 
                     else
                         if prefix == "m-" then
+
                         currentInfix = (prefix == "m-" and "m-" or "b-")
                            currentInfix = "m-"
 
                        else
 
                            currentInfix = "b-"
 
                        end
 
 
                     end
 
                     end
 
                 end
 
                 end
 
             end
 
             end
  
            -- 3. LÓGICA DE SUFIJOS
+
          -- 3. LÓGICA DE SUFIJOS
            -- Priorizamos personalSuffix (S3) sobre los sufijos generales de serie
 
 
             local perfSuffix = personalSuffix or (s.suffixes and s.suffixes.perf) or ""
 
             local perfSuffix = personalSuffix or (s.suffixes and s.suffixes.perf) or ""
 
             local imperfSuffix = (s.suffixes and s.suffixes.imperf) or ""
 
             local imperfSuffix = (s.suffixes and s.suffixes.imperf) or ""
 
             local irrealSuffix = (s.suffixes and s.suffixes.irreal) 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 rootPerf = currentRoot
 
             local rootImperf = currentRoot
 
             local rootImperf = currentRoot
 
             local rootIrreal = 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
  
             -- 4. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN IQUY (-quy / -z / -i)
+
             -- 5. LÓGICA DE VERBOS CON RAÍZ HISTÓRICA EN IQUY (z/i)
 
             if classInfo.isQuy then
 
             if classInfo.isQuy then
 
                 if s.id == "S1" then
 
                 if s.id == "S1" then
                    -- Serie de Flexión: Mantiene raíz u + sufijo quy
 
 
                     perfSuffix = "quy"
 
                     perfSuffix = "quy"
 
                 elseif s.id == "S2" then
 
                 elseif s.id == "S2" then
                    -- Serie de Nominalización: Preserva rasgos históricos
 
 
                     rootPerf = currentRoot .. "z"
 
                     rootPerf = currentRoot .. "z"
 
                     perfSuffix = "-a"
 
                     perfSuffix = "-a"
Línea 201: Línea 222:
 
                     rootIrreal = currentRoot .. "i"
 
                     rootIrreal = currentRoot .. "i"
 
                 elseif s.id == "S3" then
 
                 elseif s.id == "S3" then
                    -- NUEVO: El Imperativo de verbos iquy también palataliza (u -> uz)
 
 
                     rootPerf = currentRoot .. "z"
 
                     rootPerf = currentRoot .. "z"
                    -- perfSuffix ya trae "-u" o "-ua" del personalSuffix definido arriba
 
 
                 end
 
                 end
 
             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, {
 
             table.insert(serieData.rows, {
 
                 label = label,
 
                 label = label,
 +
                status = "pos", -- Marca para estilo visual
 
                 perf = prefix .. currentInfix .. rootPerf .. perfSuffix,
 
                 perf = prefix .. currentInfix .. rootPerf .. perfSuffix,
 
                 imperf = prefix .. currentInfix .. rootImperf .. imperfSuffix,
 
                 imperf = prefix .. currentInfix .. rootImperf .. imperfSuffix,
                 irreal = prefix .. currentInfix .. rootIrreal .. irrealSuffix
+
                 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
 
         end
 
         table.insert(results.series, serieData)
 
         table.insert(results.series, serieData)
Línea 220: Línea 272:
  
 
-- ============================================================
 
-- ============================================================
-- LA VISTA: Usado por la Wiki (drawTable y render se mantienen igual)
+
-- LA VISTA: Usado por la Wiki
 
-- ============================================================
 
-- ============================================================
 
local function drawTable(serie)
 
local function drawTable(serie)
Línea 228: Línea 280:
  
 
     local tableHtml = wrapper:tag("table"):addClass("wikitable")
 
     local tableHtml = wrapper:tag("table"):addClass("wikitable")
         :css("width", "100%"):css("max-width", "600px"):css("margin", "0")
+
         :css("width", "100%"):css("max-width", "700px"):css("margin", "0")
  
 
     local headerRow = tableHtml:tag("tr")
 
     local headerRow = tableHtml:tag("tr")
Línea 241: Línea 293:
 
     end
 
     end
  
     for _, row in ipairs(serie.rows) do
+
     for i, row in ipairs(serie.rows) do
 
         local tr = tableHtml:tag("tr")
 
         local tr = tableHtml:tag("tr")
         tr:tag("td"):css("font-weight", "bold"):wikitext(row.label)
+
          
 +
        -- 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)
 
         tr:tag("td"):wikitext(row.perf)
         if serie.id ~= "S3" then
+
       
 +
         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.imperf)
 
             tr:tag("td"):wikitext(row.irreal)
 
             tr:tag("td"):wikitext(row.irreal)
 +
            tr:tag("td"):wikitext(row.prospectivo)
 
         end
 
         end
 
     end
 
     end
Línea 256: Línea 323:
 
     local args = frame:getParent().args
 
     local args = frame:getParent().args
 
     local pageName = mw.title.getCurrentTitle().text
 
     local pageName = mw.title.getCurrentTitle().text
     local data = p.getConjugations(pageName, args.clase or args.tipo, args.raiz)
+
    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
 
     if data.root == "" or data.root == pageName then

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