Jump to content

Module:InfoboxBasic: Difference between revisions

From Yusupov's House
No edit summary
No edit summary
 
(2 intermediate revisions by the same user not shown)
Line 1: Line 1:
-- Keys to skip (already handled)
-- Module:InfoboxBasic
local skip = {
local p = {}
  name=true, Name=true, title=true, Title=true,
local trim = mw.text.trim
  image=true, Image=true,
  color=true, Color=true, colour=true, Colour=true,
  class=true, style=true
}


local function addRow(label, value)
local function isBlank(v) return v == nil or trim(tostring(v)) == '' end
  local row = tbl:tag('tr')
local function labelFromKey(k) return trim((k:gsub('_',' '))) end
  row:tag('th'):attr('scope','row'):wikitext(labelFromKey(label))
 
  row:tag('td'):wikitext(value)
-- Extract optional numeric prefix to force ordering, e.g. "01 Born", "002_Born"
local function splitOrderPrefix(key)
    local n, rest = tostring(key):match('^%s*(%d+)[%s_%-%.]*(.+)$')
    if n then
        return tonumber(n), trim(rest)
    end
    return nil, key
end
end


-- Preserve the editor’s order:
function p.render(frame)
if parent and parent.argumentPairs then
    local parent = frame:getParent() or frame
   -- argumentPairs returns last→first on many installs; collect then reverse
    local args  = parent.args or {}   -- never nil
   local seq = {}
 
   for key, value in parent:argumentPairs() do
    local title   = args.name or args.Name or args.title or args.Title or 'Infobox'
     if not skip[key] and not tonumber(key) and not isBlank(value) then
    local image  = args.image or args.Image
      seq[#seq+1] = {key=key, value=value}
    local caption = args.caption or args.Caption
    local color   = args.color or args.Color or args.colour or args.Colour
 
    -- Table
    local tbl = mw.html.create('table')
        :addClass('infobox')
        :css('float','right')
        :css('clear','right')
        :css('margin','0 0 1em 1em')
 
    -- Title
     local th = tbl:tag('tr'):tag('th')
        :attr('colspan','2')
        :addClass('infobox-title')
        :css('text-align','center')
        :css('font-size','125%')
        :wikitext(title)
 
    if not isBlank(color) then
        th:addClass('has-title-color')
        th:css('--ib-title-bg', color) -- only used by light-mode CSS
     end
     end
  end
 
  for i = #seq, 1, -1 do
    -- Optional image
    addRow(seq[i].key, seq[i].value)
    if not isBlank(image) then
  end
        local cell = tbl:tag('tr'):tag('td')
elseif parent and parent.getArgumentNames then
            :attr('colspan','2')
  -- Fallback: MediaWiki provides names in source order
            :addClass('infobox-image-cell')
  for _, key in ipairs(parent:getArgumentNames()) do
        cell:wikitext('[[File:' .. image .. '|frameless]]')
    local value = args[key]
 
    if not skip[key] and not tonumber(key) and not isBlank(value) then
        if not isBlank(caption) then
      addRow(key, value)
            cell:tag('div')
                :css('text-align','center')
                :css('font-size','90%')
                :css('font-style','italic')
                :css('margin-top','0.2em')
                :wikitext(caption)
        end
     end
     end
  end
 
else
    -- Keys to skip (already handled)
  -- Last-resort fallback (order may not be exact on very old setups)
    local skip = {
  local names = {}
        name=true, Name=true, title=true, Title=true,
  for k,_ in pairs(args or {}) do names[#names+1] = k end
        image=true, Image=true,
  for _, key in ipairs(names) do
        caption=true, Caption=true,
    local v = args[key]
        color=true, Color=true, colour=true, Colour=true,
    if not skip[key] and not tonumber(key) and not isBlank(v) then
        class=true, style=true
      addRow(key, v)
    }
 
    local function addRow(label, value)
        local row = tbl:tag('tr')
        row:tag('th'):attr('scope','row'):wikitext(labelFromKey(label))
        row:tag('td'):wikitext(value)
     end
     end
  end
 
    -- Build a stable order map from the editor's source order if possible
    local sourceOrder = {}
    if parent.argumentPairs then
        local temp = {}
        for key, _ in parent:argumentPairs() do
            temp[#temp+1] = key
        end
        for i = #temp, 1, -1 do
            sourceOrder[temp[i]] = i
        end
    elseif parent.getArgumentNames then
        local names = parent:getArgumentNames() or {}
        for i, key in ipairs(names) do
            sourceOrder[key] = i
        end
    end
 
    -- Collect rows with computed sort keys
    local rows = {}
    for key, value in pairs(args) do
        if not skip[key] and not tonumber(key) and not isBlank(value) then
            local forcedN, displayKey = splitOrderPrefix(key)
            local srcIndex = sourceOrder[key] or 10^9
            rows[#rows+1] = {
                forced = forcedN or 10^8,
                src    = srcIndex,
                key    = displayKey,
                rawkey = key,
                value  = value
            }
        end
    end
 
    -- Sort: first by explicit numeric prefix (if any), then by source order
    table.sort(rows, function(a,b)
        if a.forced ~= b.forced then
            return a.forced < b.forced
        end
        return a.src < b.src
    end)
 
    -- Render rows
    for _, r in ipairs(rows) do
        addRow(r.key, r.value)
    end
 
    return tostring(tbl)
end
end
return p

Latest revision as of 00:30, 2 October 2025

Documentation for this module may be created at Module:InfoboxBasic/doc

-- Module:InfoboxBasic
local p = {}
local trim = mw.text.trim

local function isBlank(v) return v == nil or trim(tostring(v)) == '' end
local function labelFromKey(k) return trim((k:gsub('_',' '))) end

-- Extract optional numeric prefix to force ordering, e.g. "01 Born", "002_Born"
local function splitOrderPrefix(key)
    local n, rest = tostring(key):match('^%s*(%d+)[%s_%-%.]*(.+)$')
    if n then
        return tonumber(n), trim(rest)
    end
    return nil, key
end

function p.render(frame)
    local parent = frame:getParent() or frame
    local args   = parent.args or {}   -- never nil

    local title   = args.name or args.Name or args.title or args.Title or 'Infobox'
    local image   = args.image or args.Image
    local caption = args.caption or args.Caption
    local color   = args.color or args.Color or args.colour or args.Colour

    -- Table
    local tbl = mw.html.create('table')
        :addClass('infobox')
        :css('float','right')
        :css('clear','right')
        :css('margin','0 0 1em 1em')

    -- Title
    local th = tbl:tag('tr'):tag('th')
        :attr('colspan','2')
        :addClass('infobox-title')
        :css('text-align','center')
        :css('font-size','125%')
        :wikitext(title)

    if not isBlank(color) then
        th:addClass('has-title-color')
        th:css('--ib-title-bg', color) -- only used by light-mode CSS
    end

    -- Optional image
    if not isBlank(image) then
        local cell = tbl:tag('tr'):tag('td')
            :attr('colspan','2')
            :addClass('infobox-image-cell')
        cell:wikitext('[[File:' .. image .. '|frameless]]')

        if not isBlank(caption) then
            cell:tag('div')
                :css('text-align','center')
                :css('font-size','90%')
                :css('font-style','italic')
                :css('margin-top','0.2em')
                :wikitext(caption)
        end
    end

    -- Keys to skip (already handled)
    local skip = {
        name=true, Name=true, title=true, Title=true,
        image=true, Image=true,
        caption=true, Caption=true,
        color=true, Color=true, colour=true, Colour=true,
        class=true, style=true
    }

    local function addRow(label, value)
        local row = tbl:tag('tr')
        row:tag('th'):attr('scope','row'):wikitext(labelFromKey(label))
        row:tag('td'):wikitext(value)
    end

    -- Build a stable order map from the editor's source order if possible
    local sourceOrder = {}
    if parent.argumentPairs then
        local temp = {}
        for key, _ in parent:argumentPairs() do
            temp[#temp+1] = key
        end
        for i = #temp, 1, -1 do
            sourceOrder[temp[i]] = i
        end
    elseif parent.getArgumentNames then
        local names = parent:getArgumentNames() or {}
        for i, key in ipairs(names) do
            sourceOrder[key] = i
        end
    end

    -- Collect rows with computed sort keys
    local rows = {}
    for key, value in pairs(args) do
        if not skip[key] and not tonumber(key) and not isBlank(value) then
            local forcedN, displayKey = splitOrderPrefix(key)
            local srcIndex = sourceOrder[key] or 10^9
            rows[#rows+1] = {
                forced = forcedN or 10^8,
                src    = srcIndex,
                key    = displayKey,
                rawkey = key,
                value  = value
            }
        end
    end

    -- Sort: first by explicit numeric prefix (if any), then by source order
    table.sort(rows, function(a,b)
        if a.forced ~= b.forced then
            return a.forced < b.forced
        end
        return a.src < b.src
    end)

    -- Render rows
    for _, r in ipairs(rows) do
        addRow(r.key, r.value)
    end

    return tostring(tbl)
end

return p