De Muysc cubun - Lengua Muisca

m
m
 
(No se muestran 11 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 = {}
  
-- 1. Limpieza y Normalización
+
-- ============================================================
 +
-- 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)
 
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()", "")
 
     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 = mw.text.trim(rawClass or "")
 
    rawClass = mw.ustring.lower(rawClass)
 
    if mw.ustring.find(rawClass, "sq") then return "sq."
 
    elseif mw.ustring.find(rawClass, "su") then return "su." end
 
    return ""
 
 
end
 
end
  
 
local function getRoot(pageName, clase, manualRoot)
 
local function getRoot(pageName, clase, manualRoot)
     manualRoot = mw.text.trim(manualRoot or "")
+
     if manualRoot and manualRoot ~= "" then return manualRoot end
    if manualRoot ~= "" then return manualRoot end
 
 
     local fullVerb = cleanPageName(pageName)
 
     local fullVerb = cleanPageName(pageName)
 
     local root = fullVerb
 
     local root = fullVerb
     if clase == "sq." then root = mw.ustring.gsub(fullVerb, "squa$", "")
+
     if clase == "sq." then root = fullVerb:gsub("squa$", "")
     elseif clase == "su." then root = mw.ustring.gsub(fullVerb, "suca$", "") end
+
     elseif clase == "su." then root = fullVerb:gsub("suca$", "") end
     if root == fullVerb then root = mw.ustring.gsub(fullVerb, "s[uq]ua$", "") end
+
     if root == fullVerb then root = fullVerb:gsub("s[uq]ua$", "") end
 
     return root
 
     return root
 
end
 
end
  
-- 2. Función que genera la tabla con el título ARRIBA
+
-- ============================================================
local function generateTableHtml(title, root, persons, suffixes, applySpecialRule, colNames)
+
-- NUEVA LÓGICA DE CLASE (Parsing)
     local wrapper = mw.html.create("div")
+
-- ============================================================
         :css("margin-bottom", "1.5em")
+
 
 +
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
 +
-- ============================================================
  
    -- Título encima de la tabla
+
function p.getConjugations(pageName, claseRaw, manualRoot)
    wrapper:tag("div")
+
    local classInfo = parseVerbClass(claseRaw)
        :css("font-weight", "bold")
+
    local root = getRoot(pageName, classInfo.type, manualRoot)
        :css("font-size", "1.1em")
+
   
        :css("margin-bottom", "0.4em")
+
    local results = { root = root, classInfo = classInfo, series = {} }
        :css("color", "#202122")
 
        :wikitext(title)
 
  
     local html = wrapper:tag("table")
+
    -- Configuración de Series
        :addClass("wikitable")
+
     local config = {
         :css("width", "100%")
+
        {
        :css("max-width", "600px")
+
            id = "S1", -- Flexión Verbal
         :css("margin", "0") -- Reset para que pegue al título
+
            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 header = html:tag("tr")
+
     local firstChar = root:sub(1,1):lower()
    header:tag("th"):wikitext("Persona")
+
     local firstTwo = root:sub(1,2):lower()
     header:tag("th"):wikitext(colNames[1])
 
    header:tag("th"):wikitext(colNames[2])
 
    header:tag("th"):wikitext(colNames[3])
 
  
     local firstChar = mw.ustring.lower(mw.ustring.sub(root, 1, 1))
+
     for _, s in ipairs(config) do
    local firstTwo = mw.ustring.lower(mw.ustring.sub(root, 1, 2))
+
        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
  
    for _, personData in ipairs(persons) do
+
            -- 2. LÓGICA DEL INFIJO -b- Y CAMBIOS EN LA RAÍZ
        local label, prefix = personData[1], personData[2]
+
            local currentInfix = ""
        if applySpecialRule and label == "1.ª sg." then
+
            local currentRoot = root
             if firstChar == "n" or firstChar == "z" or firstChar == "t" or firstTwo == "ch" then
+
           
                 prefix = "i-"
+
            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
 
         end
 
         end
         local row = html:tag("tr")
+
         table.insert(results.series, serieData)
        row:tag("td"):css("font-weight", "bold"):wikitext(label)
+
    end
         row:tag("td"):wikitext(prefix .. root .. (suffixes.perf or ""))
+
    return results
         row:tag("td"):wikitext(prefix .. root .. (suffixes.imperf or ""))
+
end
        row:tag("td"):wikitext(prefix .. root .. (suffixes.irreal or ""))
+
 
 +
-- ============================================================
 +
-- 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
 
     end
 
     return wrapper
 
     return wrapper
 
end
 
end
  
-- FUNCIÓN PRINCIPAL
 
 
function p.render(frame)
 
function p.render(frame)
 
     local args = frame:getParent().args
 
     local args = frame:getParent().args
    local clase = normalizeClass(args.clase or args.tipo or "")
 
 
     local pageName = mw.title.getCurrentTitle().text
 
     local pageName = mw.title.getCurrentTitle().text
     local root = getRoot(pageName, clase, args.raiz or "")
+
     local claseInput = args.clase or args.tipo or args.cat or args[1] or ""
 +
    local data = p.getConjugations(pageName, claseInput, args.raiz)
  
     if root == "" or root == pageName then
+
     if data.root == "" or data.root == pageName then
         return "<span style='color:red;'>Error: No se pudo extraer la raíz.</span>"
+
         return "<span style='color:red;'>" .. TEXTS.ui.error_root .. pageName .. ".</span>"
 
     end
 
     end
  
    -- Definición de Series con tus traducciones
+
     local mainDiv = mw.html.create("div"):addClass("mw-collapsible"):addClass("mw-collapsed")
     local serie1 = {
+
         :css("border", "1px solid #a2a9b1"):css("padding", "15px"):css("background-color", "#fcfcfc")
        title = "Serie de Flexión Verbal (Independiente)",
+
         :attr("data-expandtext", TEXTS.ui.expand):attr("data-collapsetext", TEXTS.ui.collapse)
        persons = {{"1.ª sg.", "z-"}, {"2.ª sg.", "m-"}, {"3.ª", "a-"}, {"1.ª pl.", "chi-"}, {"2.ª pl.", "mi-"}},
 
         suffixes = {perf = "", imperf = (clase == "su." and "-suca" or "-squa"), irreal = "-nga"},
 
        rule = true,
 
        cols = {"Perfectivo", "Imperfectivo", "Irrealis"}
 
    }
 
 
 
    local serie2 = {
 
         title = "Serie de Nominalización",
 
        persons = {{"1.ª sg.", "cha-"}, {"2.ª sg.", "ma-"}, {"3.ª", ""}, {"1.ª pl.", "chi-"}, {"2.ª pl.", "mi-"}},
 
        suffixes = {perf = "-ia", imperf = "-sca", irreal = "-nga"},
 
        rule = false,
 
        cols = {"Nominalización Resultativa", "Nominalización Factual", "Nominalización Irrealis"}
 
    }
 
  
    --- CONSTRUCCIÓN DEL CONTENEDOR COLAPSABLE ---
+
     mainDiv:tag("div"):css("font-weight", "bold"):css("border-bottom", "1px solid #a2a9b1")
    local mainDiv = mw.html.create("div")
+
         :css("margin-bottom", "15px"):css("padding-bottom", "5px")
        :addClass("mw-collapsible")
+
         :wikitext(TEXTS.ui.main_title .. data.root .. " (" .. data.classInfo.raw .. ")")
        :addClass("mw-collapsed")
 
        :css("border", "1px solid #a2a9b1")
 
        :css("padding", "15px")
 
        :css("background-color", "#fcfcfc")
 
        :attr("data-expandtext", "Conjugar")
 
        :attr("data-collapsetext", "Ocultar")
 
 
 
    -- Encabezado del bloque colapsable (Título del verbo)
 
     mainDiv:tag("div")
 
        :css("font-weight", "bold")
 
        :css("border-bottom", "1px solid #a2a9b1")
 
         :css("margin-bottom", "15px")
 
        :css("padding-bottom", "5px")
 
         :wikitext("Paradigma de " .. root)
 
 
 
    local contentDiv = mainDiv:tag("div")
 
        :addClass("mw-collapsible-content")
 
 
 
    -- Añadimos las tablas con los títulos arriba
 
    contentDiv:node(generateTableHtml(serie1.title, root, serie1.persons, serie1.suffixes, serie1.rule, serie1.cols))
 
    contentDiv:node(generateTableHtml(serie2.title, root, serie2.persons, serie2.suffixes, serie2.rule, serie2.cols))
 
  
 +
    local contentDiv = mainDiv:tag("div"):addClass("mw-collapsible-content")
 +
    for _, serie in ipairs(data.series) do
 +
        contentDiv:node(drawTable(serie))
 +
    end
 
     return tostring(mainDiv)
 
     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