Basculer le menu
Changer de menu des préférences
Basculer le menu personnel
Non connecté(e)
Votre adresse IP sera visible au public si vous faites des modifications.

« Module:Infobox/Core » : différence entre les versions

De Nefald
Hiob (discussion | contributions)
m Catégorie multiple selon champ 'type'
Hiob (discussion | contributions)
Aucun résumé des modifications
Balise : Révoqué
Ligne 2 : Ligne 2 :
local getArgs = require('Module:Arguments').getArgs
local getArgs = require('Module:Arguments').getArgs


-- Fonction de validation du champ "type"
-- Fonction pour créer l'HTML de l'infobox
local function validateTypeField(args, config)
function p.createInfobox(config, args)
     local typeValue = args.type or args['type']
    local root = mw.html.create('div')
        :addClass('infobox')
        :attr('role', 'region')
        :attr('aria-label', 'Infobox')
 
    -- En-tête avec titre
     local header = root:tag('div'):addClass('infobox-header')
      
      
     -- Si pas de configuration pour le type, pas de validation
     -- Conteneur pour l'icône et le titre
     if not config.typeField then
     local headerContent = header:tag('div'):addClass('infobox-header-content')
        return { valid = true }
    end
      
      
     -- Si le champ est requis et vide
     -- Gestion de l'icône
     if config.typeField.required and (not typeValue or typeValue == '') then
     if config.icone then
        return {
        local iconSpan = headerContent:tag('span'):addClass('infobox-icon')
             error = 'Le champ "type" est obligatoire.',
       
             valid = false
        if type(config.icone) == 'table' then
         }
            -- Icône FontAwesome
            local iconType = config.icone.type or 'fas'
             local iconName = config.icone.nom or 'info-circle'
            iconSpan:addClass(iconType):addClass('fa-' .. iconName)
        elseif type(config.icone) == 'string' then
            -- Image personnalisée
             iconSpan:wikitext(string.format('[[Fichier:%s|20px|link=]]', config.icone))
         end
     end
     end
      
      
     -- Si pas de valeur, mais pas requis, c'est valide
     -- Titre
     if not typeValue or typeValue == '' then
     local titre = type(config.titre) == 'function' and config.titre(args) or config.titre
        return { valid = true }
    headerContent:tag('span')
    end
         :addClass('infobox-title')
   
         :wikitext(titre)
    -- Nettoyer la valeur
    local cleanType = mw.text.trim(mw.ustring.lower(typeValue))
   
    -- Vérifier si la valeur est dans la liste autorisée
    if config.typeField.allowedValues and not config.typeField.allowedValues[cleanType] then
        local errorMsg = config.typeField.errorMessage or 'Valeur de type non autorisée : ' .. typeValue
         return {
            error = errorMsg,
            valid = false
         }
    end
   
    return {
        valid = true,
        cleanValue = cleanType,
        displayValue = config.typeField.allowedValues and config.typeField.allowedValues[cleanType] and config.typeField.allowedValues[cleanType].display or typeValue
    }
end


-- Fonction pour gérer la catégorisation automatique
     -- Image si définie
local function addCategories(args, config, frame)
     if config.image then
     -- Vérifier qu'on est dans le namespace principal (0)
         local imageData = type(config.image) == 'function' and config.image(args) or config.image
    local title = mw.title.getCurrentTitle()
         if imageData and imageData.nom then
     if title.namespace ~= 0 then
             local imageDiv = root:tag('div'):addClass('infobox-image')
         return ''
            local taille = imageData.taille or '300px'
    end
            local legende = imageData.legende
   
           
    local categoryString = ''
            -- Construction du wikitext pour l'image
   
            local imageWikitext = string.format('[[Fichier:%s|%s|center', imageData.nom, taille)
    -- Nouvelles catégories configurables (fonction categories)
            if legende and legende ~= '' then
    if type(config.categories) == 'function' then
                imageWikitext = imageWikitext .. '|' .. legende
        local cats = config.categories(args, config, frame)
         if cats then
             for _, cat in ipairs(cats) do
                categoryString = categoryString .. '[[Catégorie:' .. cat .. ']]'
             end
             end
            imageWikitext = imageWikitext .. ']]'
           
            imageDiv:wikitext(imageWikitext)
         end
         end
     end
     end
   
    return categoryString
end
function p.build(args, config, frame)
    if not args.nom or args.nom == '' then
        return '<div class="error">Nom requis pour l\'infobox</div>'
    end
    -- Validation du champ type
    local typeValidation = validateTypeField(args, config)
    if not typeValidation.valid then
        return '<div class="error">' .. typeValidation.error .. '</div>'
    end
    local root = mw.html.create('div')
        :addClass('infobox-table')
        :attr('role', 'table')


     -- En-tête avec nom
     -- Corps de l'infobox avec les sections
     local header = mw.html.create('div')
     local body = root:tag('div'):addClass('infobox-body')
        :addClass('infobox-header')
      
      
     if args.image and args.image ~= '' then
     -- Variable pour suivre si les coordonnées ont été affichées
        local imageDiv = mw.html.create('div')
     local coordinatesDisplayed = false
            :addClass('infobox-image')
       
        local imageSize = args.tailleimage or '280px'
        local imageCode = string.format('[[Fichier:%s|%s|center', args.image, imageSize)
        if args.legende and args.legende ~= '' then
            imageCode = imageCode .. '|' .. args.legende
        end
        imageCode = imageCode .. ']]'
       
        imageDiv:wikitext(imageCode)
        header:node(imageDiv)
    end
 
    -- Titre sans icône (icône déplacée en bas à droite)
     local titleDiv = mw.html.create('div')
        :addClass('infobox-title')
        :wikitext(args.nom)
      
      
    header:node(titleDiv)
     if config.sections then
 
         for _, section in ipairs(config.sections) do
     if args.soustitre and args.soustitre ~= '' then
             -- Créer la section seulement si elle a des champs à afficher
         local subtitleDiv = mw.html.create('div')
            local hasVisibleFields = false
             :addClass('infobox-subtitle')
            for _, champ in ipairs(section.champs) do
            :wikitext(args.soustitre)
                if args[champ.cle] and args[champ.cle] ~= '' then
        header:node(subtitleDiv)
                    hasVisibleFields = true
    end
                    break
 
                 end
    root:node(header)
             end
 
           
    -- Corps avec les champs
            -- Vérifier aussi les coordonnées groupées
    local body = mw.html.create('div')
            if not coordinatesDisplayed and args.x and args.y and args.z then
        :addClass('infobox-body')
                for _, champ in ipairs(section.champs) do
 
                    if champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z' then
    for _, section in ipairs(config.sections or {}) do
                        hasVisibleFields = true
        if section.titre then
                        break
            local sectionDiv = mw.html.create('div')
                    end
                :addClass('infobox-section-title')
                end
                 :wikitext(section.titre)
            end
             body:node(sectionDiv)
           
        end
            if hasVisibleFields then
 
                 local sectionDiv = body:tag('div'):addClass('infobox-section')
        -- Vérifier si on doit afficher les coordonnées automatiquement
        local hasCoordinates = args.x and args.x ~= '' and args.y and args.y ~= '' and args.z and args.z ~= ''
        local coordinatesDisplayed = false
 
        for _, champ in ipairs(section.champs or {}) do
            -- Détection automatique des coordonnées
            if hasCoordinates and not coordinatesDisplayed and (champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z' or champ.cle == 'spawn' or champ.cle == 'coordonnees') then
                 local rowDiv = mw.html.create('div')
                    :addClass('infobox-row')
                  
                  
                 local labelDiv = mw.html.create('div')
                 -- Titre de section
                    :addClass('infobox-label')
                if section.titre then
                    :wikitext('Coordonnées')
                    sectionDiv:tag('div')
                        :addClass('infobox-section-title')
                        :wikitext(section.titre)
                end
                  
                  
                 local valueDiv = mw.html.create('div')
                 -- Vérifier si cette section contient les coordonnées (x, y, z)
                    :addClass('infobox-value')
                 local hasCoordinates = false
                  
                 if not coordinatesDisplayed and args.x and args.y and args.z then
                -- Construction du lien vers la carte si le paramètre monde existe
                     for _, champ in ipairs(section.champs) do
                 if args.monde and args.monde ~= '' then
                        if champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z' then
                     local mapUrl = string.format('[https://map.nefald.fr/?worldname=%s&mapname=surface&zoom=4&x=%s&y=%s&z=%s %s, %s, %s]',
                            hasCoordinates = true
                        args.monde, args.x, args.y, args.z, args.x, args.y, args.z)
                            break
                    valueDiv:wikitext("''" .. mapUrl .. "''")
                        end
                else
                     end
                    -- Affichage simple sans lien
                     valueDiv:wikitext(string.format('%s, %s, %s', args.x, args.y, args.z))
                 end
                 end
                  
                  
                 rowDiv:node(labelDiv):node(valueDiv)
                 -- Si la section contient les coordonnées, les afficher groupées
                body:node(rowDiv)
                 if hasCoordinates and not coordinatesDisplayed then
                 coordinatesDisplayed = true
                    -- Vérifier si on peut créer un lien (worldname obligatoire)
               
                    local canCreateLink = args.worldname and args.worldname ~= ''
            -- *** NOUVELLE LOGIQUE POUR TRAITEMENT DES CHAMPS ***
                   
            elseif not (hasCoordinates and (champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z')) then
                    local labelDiv = mw.html.create('div')
                local rawValue = args[champ.cle]
                        :addClass('infobox-label')
               
                        :wikitext("'''Coordonnées'''")
                -- Si le champ a une valeur OU une fonction process qui peut générer une valeur
                   
                if (rawValue and rawValue ~= '') or champ.process then
                    local valueDiv = mw.html.create('div')
                    local finalValue = rawValue
                        :addClass('infobox-value')
                      
                      
                     -- Appliquer la fonction process si elle existe
                     if canCreateLink then
                    if champ.process and type(champ.process) == 'function' then
                        -- Construction du lien avec worldname
                         finalValue = champ.process(rawValue, args, frame)
                        local mapname = args.mapname and args.mapname ~= '' and args.mapname or 'flat'
                        local zoom = args.zoom and args.zoom ~= '' and args.zoom or '4'
                       
                        local mapUrl = string.format(
                            'https://map.nefald.fr/?worldname=%s&mapname=%s&zoom=%s&x=%s&y=%s&z=%s',
                            mw.uri.encode(args.worldname),
                            mw.uri.encode(mapname),
                            mw.uri.encode(zoom),
                            mw.uri.encode(args.x),
                            mw.uri.encode(args.y),
                            mw.uri.encode(args.z)
                        )
                       
                        local displayText = string.format('%s, %s, %s', args.x, args.y, args.z)
                        valueDiv:wikitext(string.format('[%s %s]', mapUrl, displayText))
                    else
                        -- Affichage simple sans lien si worldname manque
                         local displayText = string.format('%s, %s, %s', args.x, args.y, args.z)
                        valueDiv:wikitext(displayText)
                     end
                     end
                      
                      
                     -- N'afficher que si on a une valeur finale
                     local rowDiv = mw.html.create('div'):addClass('infobox-row')
                     if finalValue and finalValue ~= '' then
                    rowDiv:node(labelDiv):node(valueDiv)
                         local rowDiv = mw.html.create('div')
                    sectionDiv:node(rowDiv)
                            :addClass('infobox-row')
                    coordinatesDisplayed = true
 
                end
                         local labelDiv = mw.html.create('div')
               
                -- Champs de la section
                for _, champ in ipairs(section.champs) do
                    local value = args[champ.cle]
                   
                    -- Ignorer les coordonnées individuelles si elles ont été affichées groupées
                    if coordinatesDisplayed and (champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z') then
                        -- Ne rien faire, on a déjà affiché les coordonnées groupées
                     elseif value and value ~= '' then
                         local rowDiv = sectionDiv:tag('div'):addClass('infobox-row')
                       
                         -- Label
                        rowDiv:tag('div')
                             :addClass('infobox-label')
                             :addClass('infobox-label')
                             :wikitext(champ.label)
                             :wikitext("'''" .. champ.label .. "'''")
                          
                          
                         local valueDiv = mw.html.create('div')
                        -- Valeur
                            :addClass('infobox-value')
                         local valueDiv = rowDiv:tag('div'):addClass('infobox-value')
                          
                          
                         -- Formatage spécial pour le champ "type"
                         -- Traiter la valeur (fonction ou chaîne)
                         if champ.cle == 'type' and typeValidation.displayValue then
                         if type(champ.format) == 'function' then
                             valueDiv:wikitext(typeValidation.displayValue)
                             valueDiv:wikitext(champ.format(value, args))
                         else
                         else
                             valueDiv:wikitext(finalValue)
                             valueDiv:wikitext(value)
                         end
                         end
                        rowDiv:node(labelDiv):node(valueDiv)
                        body:node(rowDiv)
                     end
                     end
                 end
                 end
Ligne 200 : Ligne 174 :
     end
     end


     root:node(body)
     return tostring(root)
end
 
-- Fonction pour valider le type
function p.validateType(args, config)
    if not config.typeField then
        return true, nil
    end
      
      
     -- Traiter l'icône APRÈS avoir construit tout le HTML
     local typeValue = args.type
     local finalResult = tostring(root)
    if not typeValue or typeValue == '' then
        if config.typeField.required then
            return false, "Le champ 'type' est obligatoire"
        end
        return true, nil
     end
      
      
     if config.icone then
    -- Vérifier si le type est dans les valeurs autorisées
         local iconeType = config.icone.type or 'fas'
     if config.typeField.allowedValues then
        local iconeNom = config.icone.nom
         local typeConfig = config.typeField.allowedValues[typeValue:lower()]
       
         if not typeConfig then
        -- Utiliser mw.getCurrentFrame() si frame n'est pas disponible
             return false, config.typeField.errorMessage or "Type non reconnu"
        local currentFrame = frame or mw.getCurrentFrame()
        local iconeContent = '{{#' .. iconeType .. ':' .. iconeNom .. '}}'
       
         if currentFrame then
             iconeContent = currentFrame:preprocess(iconeContent)
         end
         end
          
         return true, typeConfig
         -- Créer la div d'icône séparément et l'ajouter au résultat
    end
        local iconeHtml = string.format(
   
            '<div class="infobox-icon-bottom">%s</div>',
    return true, nil
            iconeContent
end
        )
 
       
-- Fonction pour générer les catégories
         -- Injecter l'icône juste avant la fermeture de la div principale
function p.generateCategories(args, config, typeConfig)
        finalResult = finalResult:gsub('</div>$', iconeHtml .. '</div>')
    if not config.categories then
         return ''
    end
   
    return config.categories(args, typeConfig)
end
 
-- Fonction principale appelée par {{#invoke:}}
function p.main(frame)
    local args = getArgs(frame, {parentFirst = true})
   
    -- Récupérer le type d'infobox depuis les arguments du frame
    local infoboxType = frame.args[1] or frame.args.type
    if not infoboxType or infoboxType == '' then
        return '<div class="error">Type d\'infobox requis</div>'
    end
   
    -- Charger la configuration appropriée
    local configModule = 'Module:Infobox/Configs/' .. infoboxType
    local success, config = pcall(require, configModule)
   
    if not success then
         return '<div class="error">Configuration introuvable pour : ' .. infoboxType .. '</div>'
    end
   
    -- Valider le type si nécessaire
    local isValid, typeConfig = p.validateType(args, config)
    if not isValid then
        return '<div class="error">' .. typeConfig .. '</div>'
     end
     end
      
      
     -- Ajouter les catégories automatiques (ancien + nouveau système)
    -- Générer l'infobox
     local categories = addCategories(args, config, frame)
    local infobox = p.createInfobox(config, args)
   
     -- Ajouter les catégories
     local categories = p.generateCategories(args, config, typeConfig)
      
      
     return finalResult .. categories
     return infobox .. categories
end
end


return p
return p

Version du 23 octobre 2025 à 13:59

La documentation pour ce module peut être créée à Module:Infobox/Core/doc

local p = {}
local getArgs = require('Module:Arguments').getArgs

-- Fonction pour créer l'HTML de l'infobox
function p.createInfobox(config, args)
    local root = mw.html.create('div')
        :addClass('infobox')
        :attr('role', 'region')
        :attr('aria-label', 'Infobox')

    -- En-tête avec titre
    local header = root:tag('div'):addClass('infobox-header')
    
    -- Conteneur pour l'icône et le titre
    local headerContent = header:tag('div'):addClass('infobox-header-content')
    
    -- Gestion de l'icône
    if config.icone then
        local iconSpan = headerContent:tag('span'):addClass('infobox-icon')
        
        if type(config.icone) == 'table' then
            -- Icône FontAwesome
            local iconType = config.icone.type or 'fas'
            local iconName = config.icone.nom or 'info-circle'
            iconSpan:addClass(iconType):addClass('fa-' .. iconName)
        elseif type(config.icone) == 'string' then
            -- Image personnalisée
            iconSpan:wikitext(string.format('[[Fichier:%s|20px|link=]]', config.icone))
        end
    end
    
    -- Titre
    local titre = type(config.titre) == 'function' and config.titre(args) or config.titre
    headerContent:tag('span')
        :addClass('infobox-title')
        :wikitext(titre)

    -- Image si définie
    if config.image then
        local imageData = type(config.image) == 'function' and config.image(args) or config.image
        if imageData and imageData.nom then
            local imageDiv = root:tag('div'):addClass('infobox-image')
            local taille = imageData.taille or '300px'
            local legende = imageData.legende
            
            -- Construction du wikitext pour l'image
            local imageWikitext = string.format('[[Fichier:%s|%s|center', imageData.nom, taille)
            if legende and legende ~= '' then
                imageWikitext = imageWikitext .. '|' .. legende
            end
            imageWikitext = imageWikitext .. ']]'
            
            imageDiv:wikitext(imageWikitext)
        end
    end

    -- Corps de l'infobox avec les sections
    local body = root:tag('div'):addClass('infobox-body')
    
    -- Variable pour suivre si les coordonnées ont été affichées
    local coordinatesDisplayed = false
    
    if config.sections then
        for _, section in ipairs(config.sections) do
            -- Créer la section seulement si elle a des champs à afficher
            local hasVisibleFields = false
            for _, champ in ipairs(section.champs) do
                if args[champ.cle] and args[champ.cle] ~= '' then
                    hasVisibleFields = true
                    break
                end
            end
            
            -- Vérifier aussi les coordonnées groupées
            if not coordinatesDisplayed and args.x and args.y and args.z then
                for _, champ in ipairs(section.champs) do
                    if champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z' then
                        hasVisibleFields = true
                        break
                    end
                end
            end
            
            if hasVisibleFields then
                local sectionDiv = body:tag('div'):addClass('infobox-section')
                
                -- Titre de section
                if section.titre then
                    sectionDiv:tag('div')
                        :addClass('infobox-section-title')
                        :wikitext(section.titre)
                end
                
                -- Vérifier si cette section contient les coordonnées (x, y, z)
                local hasCoordinates = false
                if not coordinatesDisplayed and args.x and args.y and args.z then
                    for _, champ in ipairs(section.champs) do
                        if champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z' then
                            hasCoordinates = true
                            break
                        end
                    end
                end
                
                -- Si la section contient les coordonnées, les afficher groupées
                if hasCoordinates and not coordinatesDisplayed then
                    -- Vérifier si on peut créer un lien (worldname obligatoire)
                    local canCreateLink = args.worldname and args.worldname ~= ''
                    
                    local labelDiv = mw.html.create('div')
                        :addClass('infobox-label')
                        :wikitext("'''Coordonnées'''")
                    
                    local valueDiv = mw.html.create('div')
                        :addClass('infobox-value')
                    
                    if canCreateLink then
                        -- Construction du lien avec worldname
                        local mapname = args.mapname and args.mapname ~= '' and args.mapname or 'flat'
                        local zoom = args.zoom and args.zoom ~= '' and args.zoom or '4'
                        
                        local mapUrl = string.format(
                            'https://map.nefald.fr/?worldname=%s&mapname=%s&zoom=%s&x=%s&y=%s&z=%s',
                            mw.uri.encode(args.worldname),
                            mw.uri.encode(mapname),
                            mw.uri.encode(zoom),
                            mw.uri.encode(args.x),
                            mw.uri.encode(args.y),
                            mw.uri.encode(args.z)
                        )
                        
                        local displayText = string.format('%s, %s, %s', args.x, args.y, args.z)
                        valueDiv:wikitext(string.format('[%s %s]', mapUrl, displayText))
                    else
                        -- Affichage simple sans lien si worldname manque
                        local displayText = string.format('%s, %s, %s', args.x, args.y, args.z)
                        valueDiv:wikitext(displayText)
                    end
                    
                    local rowDiv = mw.html.create('div'):addClass('infobox-row')
                    rowDiv:node(labelDiv):node(valueDiv)
                    sectionDiv:node(rowDiv)
                    coordinatesDisplayed = true
                end
                
                -- Champs de la section
                for _, champ in ipairs(section.champs) do
                    local value = args[champ.cle]
                    
                    -- Ignorer les coordonnées individuelles si elles ont été affichées groupées
                    if coordinatesDisplayed and (champ.cle == 'x' or champ.cle == 'y' or champ.cle == 'z') then
                        -- Ne rien faire, on a déjà affiché les coordonnées groupées
                    elseif value and value ~= '' then
                        local rowDiv = sectionDiv:tag('div'):addClass('infobox-row')
                        
                        -- Label
                        rowDiv:tag('div')
                            :addClass('infobox-label')
                            :wikitext("'''" .. champ.label .. "'''")
                        
                        -- Valeur
                        local valueDiv = rowDiv:tag('div'):addClass('infobox-value')
                        
                        -- Traiter la valeur (fonction ou chaîne)
                        if type(champ.format) == 'function' then
                            valueDiv:wikitext(champ.format(value, args))
                        else
                            valueDiv:wikitext(value)
                        end
                    end
                end
            end
        end
    end

    return tostring(root)
end

-- Fonction pour valider le type
function p.validateType(args, config)
    if not config.typeField then
        return true, nil
    end
    
    local typeValue = args.type
    if not typeValue or typeValue == '' then
        if config.typeField.required then
            return false, "Le champ 'type' est obligatoire"
        end
        return true, nil
    end
    
    -- Vérifier si le type est dans les valeurs autorisées
    if config.typeField.allowedValues then
        local typeConfig = config.typeField.allowedValues[typeValue:lower()]
        if not typeConfig then
            return false, config.typeField.errorMessage or "Type non reconnu"
        end
        return true, typeConfig
    end
    
    return true, nil
end

-- Fonction pour générer les catégories
function p.generateCategories(args, config, typeConfig)
    if not config.categories then
        return ''
    end
    
    return config.categories(args, typeConfig)
end

-- Fonction principale appelée par {{#invoke:}}
function p.main(frame)
    local args = getArgs(frame, {parentFirst = true})
    
    -- Récupérer le type d'infobox depuis les arguments du frame
    local infoboxType = frame.args[1] or frame.args.type
    if not infoboxType or infoboxType == '' then
        return '<div class="error">Type d\'infobox requis</div>'
    end
    
    -- Charger la configuration appropriée
    local configModule = 'Module:Infobox/Configs/' .. infoboxType
    local success, config = pcall(require, configModule)
    
    if not success then
        return '<div class="error">Configuration introuvable pour : ' .. infoboxType .. '</div>'
    end
    
    -- Valider le type si nécessaire
    local isValid, typeConfig = p.validateType(args, config)
    if not isValid then
        return '<div class="error">' .. typeConfig .. '</div>'
    end
    
    -- Générer l'infobox
    local infobox = p.createInfobox(config, args)
    
    -- Ajouter les catégories
    local categories = p.generateCategories(args, config, typeConfig)
    
    return infobox .. categories
end

return p