Module:Navbox: Difference between revisions

1,977 bytes added ,  11 years ago
rewrite with htmlbuilder
m>Toohool
(fix for lists in image and imageleft)
m>Toohool
(rewrite with htmlbuilder)
Line 2: Line 2:
-- This module will implement {{Navbox}}
-- This module will implement {{Navbox}}
--
--
 
local p = {}
local p = {}
local HtmlBuilder = require('Module:Sandbox/Toohool/HtmlBuilder')


local gutterRow = '<tr style="height:2px;"><td></td></tr>'
local args
local tableRowAdded = false
local border
local border
local listnums = {}
local listnums = {}
local ret = {}
 
function add(...)
    local args = {...}
    for i = 1, #args do
        if args[i] then
            table.insert(ret, args[i])
        end
    end
end
 
function trim(s)
function trim(s)
     return (mw.ustring.gsub(s, "^%s*(.-)%s*$", "%1"))
     return (mw.ustring.gsub(s, "^%s*(.-)%s*$", "%1"))
end
function addTableRow(tbl)
    -- If any other rows have already been added, then we add a 2px gutter row.
    if tableRowAdded then
tbl
.tag('tr')
.css('height', '2px')
.tag('td')
end
tableRowAdded = true
return tbl.tag('tr')
end
end


function renderTitleRow(args)
 
--
--  Title row
--
function renderTitleRow(tbl)
     if not args.title then return end
     if not args.title then return end
local titleRow = addTableRow(tbl)
    if args.titlegroup then
    titleRow
    .tag('th')
    .attr('scope', 'row')
    .addClass('navbox-group')
    .addClass(args.titlegroupclass)
    .cssText(args.basestyle)
    .cssText(args.groupstyle)
    .cssText(args.titlegroupstyle)
    .wikitext(args.titlegroup)
    end
      
      
     add('<tr>')
     local titleCell = titleRow.tag('th').attr('scope', 'col')
   
     if args.titlegroup then
     if args.titlegroup then
        add('<th scope="row" class="navbox-group ', args.titlegroupclass, '" style="', args.basestyle, ';', args.groupstyle, ';', args.titlegroupstyle, '">', args.titlegroup, '</th><th scope="col" style="border-left:2px solid #fdfdfd;width:100%;')
    titleCell
    else
    .css('border-left', '2px solid #fdfdfd')
        add('<th scope="col" style="')
    .css('width', '100%')
     end
     end
     local colspan = 2
   
     if args.imageleft then colspan = colspan + 1 end
     local titleColspan = 2
     if args.image then colspan = colspan + 1 end
     if args.imageleft then titleColspan = titleColspan + 1 end
     if args.titlegroup then colspan = colspan - 1 end
     if args.image then titleColspan = titleColspan + 1 end
     add(args.basestyle, ';', args.titlestyle, '" class="navbox-title" colspan=', colspan, '>')
     if args.titlegroup then titleColspan = titleColspan - 1 end
 
      
    local stateLinkPlaceholder = '<span style="float:right;width:6em;">&nbsp;</span>'
    titleCell
    .cssText(args.basestyle)
    .cssText(args.titlestyle)
    .addClass('navbox-title')
    .attr('colspan', titleColspan)
renderNavBar(titleCell)
titleCell
.tag('div')
.addClass(args.titleclass)
.css('font-size', '110%')
.newline()
.wikitext(args.title)
end
function renderNavBar(titleCell)
local addStateLinkPlaceholder = false
     if args.navbar == 'plain' or args.navbar == 'off' or (not args.name and (border == 'subgroup' or border == 'child' or border == 'none')) then
     if args.navbar == 'plain' or args.navbar == 'off' or (not args.name and (border == 'subgroup' or border == 'child' or border == 'none')) then
         if args.navbar == 'off' then
         if args.navbar == 'off' then
            if args.state == 'plain' then add(stateLinkPlaceholder) end
        addStateLinkPlaceholder = (args.state == 'plain')
         else
         else
            if args.state ~= 'plain' then add(stateLinkPlaceholder) end
        addStateLinkPlaceholder = (args.state ~= 'plain')
         end
         end
     else
     else
         if args.name then
         if args.name then
             add(mw.getCurrentFrame():expandTemplate{ title = 'navbar', args = {  
             titleCell.wikitext(mw.getCurrentFrame():expandTemplate{ title = 'navbar', args = {  
                 args.name,  
                 args.name,  
                 mini = 1,  
                 mini = 1,  
Line 53: Line 96:
             }})
             }})
         else
         else
            add('<span class="error" style="float:left;white-space:nowrap;">Error: No name provided</span>')
        titleCell
             if args.state == 'plain' then add(stateLinkPlaceholder) end
        .tag('span')
        .addClass('error')
        .css('float', 'left')
        .css('white-space', 'nowrap')
        .wikitext('Error: No name provided')
             addStateLinkPlaceholder = (args.state == 'plain')
         end
         end
     end
     end
   
    if addStateLinkPlaceholder then
    titleCell
    .tag('span')
    .css('float', 'right')
    .css('width', '6em')
    .wikitext('&nbsp;')
    end
end


    add('<div class="', args.titleclass, '" style="font-size:110%;">')
    add(args.title)
    add('</div></th></tr>')
end


function renderAboveRow(args)
--
--  Above/Below rows
--
function renderAboveRow(tbl)
     if not args.above then return end
     if not args.above then return end
   
    if args.title then add(gutterRow) end
addTableRow(tbl)
    add('<tr><td class="navbox-abovebelow ', args.aboveclass, '" style="', args.basestyle, ';', args.abovestyle, '" colspan="', getAboveBelowColspan(args), '">')
.tag('td')
    add('<div>\n', args.above, '</div></td></tr>')
.addClass('navbox-abovebelow')
.addClass(args.aboveclass)
.cssText(args.basestyle)
.cssText(args.abovestyle)
.attr('colspan', getAboveBelowColspan())
.tag('div')
.newline()
.wikitext(args.above)
end
end
function renderBelowRow(args)
 
    if args.below then
function renderBelowRow(tbl)
        if args.title or args.above or #listnums > 0 then add(gutterRow) end
if not args.below then return end
        add('<tr><td class="navbox-abovebelow ', args.belowclass, '" style="', args.basestyle, ';', args.belowstyle, 'colspan="', getAboveBelowColspan(args), '">')
        add('<div>\n', args.below, '</div></td></tr>')
addTableRow(tbl)
    end
.tag('td')
.addClass('navbox-abovebelow')
.addClass(args.belowclass)
.cssText(args.basestyle)
.cssText(args.belowstyle)
.attr('colspan', getAboveBelowColspan())
.tag('div')
.newline()
.wikitext(args.below)
end
end
function getAboveBelowColspan(args)
 
function getAboveBelowColspan()
     local ret = 2
     local ret = 2
     if args.imageleft then ret = ret + 1 end
     if args.imageleft then ret = ret + 1 end
Line 83: Line 155:
     return ret
     return ret
end
end
 
function renderFirstListRow(args)
--
--  List rows
--
function renderFirstListRow(tbl)
     if not args.list1 then return end
     if not args.list1 then return end
   
    if args.title or args.above then add(gutterRow) end
local row = addTableRow(tbl)
    add('<tr>')
     if args.imageleft then
     if args.imageleft then
        add('<td class="navbox-image ', args.imageclass, '" style="width:0%;padding:0px 2px 0px 0px;', args.imageleftstyle, '"')
    row
        add(' rowspan=', (2 * #listnums - 1), '><div>\n', args.imageleft, '</div></td>')
    .tag('td')
    .addClass('navbox-image')
    .addClass(args.imageclass)
    .css('width', '0%')
    .css('padding', '0px 2px 0px 0px')
    .cssText(args.imageleftstyle)
    .attr('rowspan', 2 * #listnums - 1)
    .tag('div')
    .newline()
    .wikitext(args.imageleft)
     end
     end
    if args.group1 then
    local groupCell = row.tag('th')
   
    groupCell
  .attr('scope', 'row')
  .addClass('navbox-group')
  .addClass(args.groupclass)
  .cssText(args.basestyle)
 
  if args.groupwidth then
  groupCell.css('width', args.groupwidth)
  end


groupCell
.cssText(args.groupstyle)
.cssText(args.group1style)
.wikitext(args.group1)
    end
   
    local listCell = row.tag('td')
   
     if args.group1 then
     if args.group1 then
        add('<th scope="row" class="navbox-group ', args.groupclass, '" style="', args.basestyle, ';')
    listCell
        if args.groupwidth then add('width:', args.groupwidth, ';') end
    .css('text-align', 'left')
        add(args.groupstyle, ';', args.group1style, '">')
    .css('border-left-width', '2px')
        add(args.group1, '</th>')
    .css('border-left-style', 'solid')
        add('<td style="text-align:left;border-left-width:2px;border-left-style:solid;')
     else
     else
        add('<td colspan=2 style="')
    listCell.attr('colspan', 2)
     end
     end
     if not args.groupwidth then add('width:100%;') end
   
    add('padding:0px;', args.liststyle, ';', args.oddstyle, ';', args.list1style, '" class="navbox-list navbox-')
     if not args.groupwidth then  
    if args.evenodd == 'swap' then
    listCell.css('width', '100%')
        add('even')
    else
        add(args.evenodd or 'odd')
     end
     end
     add(' ', args.listclass, '">')
 
    add('<div style="padding:', args.list1padding or args.listpadding or '0em 0.25em', '">\n')
    local evenOdd = args.evenodd or 'odd'
    add(args.list1)
     if args.evenodd == 'swap' then evenOdd = 'even' end
    add('</div></td>')
   
    listCell
    .css('padding', '0px')
    .cssText(args.liststyle)
    .cssText(args.oddstyle)
    .cssText(args.list1style)
    .addClass('navbox-list')
    .addClass('navbox-' .. evenOdd)
    .addClass(args.listclass)
    .tag('div')
    .css('padding', args.list1padding or args.listpadding or '0em 0.25em')
    .newline()
    .wikitext(args.list1)


     if args.image then
     if args.image then
        add('<td class="navbox-image ', args.imageclass, '" style="width:0%;padding:0px 0px 0px 2px;', args.imagestyle, '" ')
    row
        add(' rowspan=', (2 * #listnums - 1), '>')
    .tag('td')
        add('<div>\n', args.image, '</div></td>')
    .addClass('navbox-image')
    .addClass(args.imageclass)
    .css('width', '0%')
    .css('padding', '0px 0px 0px 2px')
    .cssText(args.imagestyle)
    .attr('rowspan', 2 * #listnums - 1)
    .tag('div')
    .newline()
    .wikitext(args.image)
    end
end
function renderNthListRow(tbl, listnum)
local row = addTableRow(tbl)
    if args['group' .. listnum] then
    local groupCell = row.tag('th')
   
    groupCell
  .attr('scope', 'row')
  .addClass('navbox-group')
  .addClass(args.groupclass)
  .cssText(args.basestyle)
 
  if args.groupwidth then
  groupCell.css('width', args.groupwidth)
  end
 
  groupCell
  .cssText(args.groupstyle)
  .cssText(args['group' .. listnum .. 'style'])
  .wikitext(args['group' .. listnum])
     end
     end
      
      
     add('</tr>')
     local listCell = row.tag('td')
end


function renderNthListRow(args, listnum)
    if args.title or args.above or args.list1 then
        add(gutterRow)
    end
    add('<tr>')
     if args['group' .. listnum] then
     if args['group' .. listnum] then
        add('<th scope="row" class="navbox-group ', args.groupclass, '" style="', args.basestyle, ';')
    listCell
        if args.groupwidth then add('width:', args.groupwidth, ';') end
    .css('text-align', 'left')
        add(args.groupstyle, ';', args['group' .. listnum .. 'style'], '">')
    .css('border-left-width', '2px')
        add(args['group' .. listnum])
    .css('border-left-style', 'solid')
        add('</th><td style="text-align:left;border-left-width:2px;border-left-style:solid;')
     else
     else
        add('<td colspan=2 style="')
    listCell.attr('colspan', 2)
    end
   
    if not args.groupwidth then
    listCell.css('width', '100%')
     end
     end
    if not args.groupwidth then add('width:100%;') end
      
      
     local isOdd = (listnum % 2) == 1
     local isOdd = (listnum % 2) == 1
     local rowstyle = args.evenstyle
     local rowstyle = args.evenstyle
     if isOdd then rowstyle = args.oddstyle end
     if isOdd then rowstyle = args.oddstyle end
       
    add('padding:0px;', args.liststyle, ';', rowstyle, ';', args['list' .. listnum .. 'style'], '" ')
local evenOdd
    add('class="navbox-list navbox-')
if args.evenodd == 'swap' then
    if args.evenodd == 'swap' then
if isOdd then evenOdd = 'even' else evenOdd = 'odd' end
        if isOdd then add('even') else add('odd') end
else
    else
if isOdd then evenOdd = args.evenodd or 'odd' else evenOdd = args.evenodd or 'even' end
        if isOdd then add(args.evenodd or 'odd') else add(args.evenodd or 'even') end
end
    end
     add(' ', args.listclass, '">')
     listCell
   
    .css('padding', '0px')
    add('<div style="padding:', args.listpadding or '0em 0.25em', '">\n', args['list' .. listnum], '</div></td></tr>')
    .cssText(args.liststyle)
    .cssText(rowstyle)
    .cssText(args['list' .. listnum .. 'style'])
    .addClass('navbox-list')
    .addClass('navbox-' .. evenOdd)
    .addClass(args.listclass)
    .tag('div')
    .css('padding', args.listpadding or '0em 0.25em')
    .newline()
    .wikitext(args['list' .. listnum])
end
end


function p._navbox(args)
 
    for k, v in pairs(args) do
--
        local listnum = ('' .. k):match('^list(%d+)$')
--  Main navbox tables
        if listnum then table.insert(listnums, tonumber(listnum)) end
--
    end
function renderMainTable()
    table.sort(listnums)
local tbl = HtmlBuilder.create('table')
   
.attr('cellspacing', 0)
    border = trim(args.border or args[1] or '')
.addClass('nowraplinks')
    if border == 'subgroup' or border == 'child' then
.addClass(args.bodyclass)
        add('</div>')
    elseif border ~= 'none' then
        add('<table cellspacing="0" class="navbox" style="border-spacing:0;', args.bodystyle, ';', args.style, '"><tr><td style="padding:2px;">')
    end
   
    add('<table cellspacing="0" class="nowraplinks ', args.bodyclass, ' ')
   
     if args.title and (args.state ~= 'plain' and args.state ~= 'off') then
     if args.title and (args.state ~= 'plain' and args.state ~= 'off') then
        add('collapsible ', args.state or 'autocollapse', ' ')
    tbl
    .addClass('collapsible')
    .addClass(args.state or 'autocollapse')
     end
     end
   
tbl.css('border-spacing', 0)
     if border == 'subgroup' or border == 'child' or border == 'none' then
     if border == 'subgroup' or border == 'child' or border == 'none' then
        add('navbox-subgroup" style="border-spacing:0;', args.bodystyle, ';', args.style)
    tbl
     else
    .addClass('navbox-subgroup')
        add('navbox-inner" style="border-spacing:0;background:transparent;color:inherit')
    .cssText(args.bodyStyle)
    .cssText(args.style)
     else -- regular navobx - bodyStyle and style will be applied to the wrapper table
    tbl
    .addClass('navbox-inner')
    .css('background', 'transparent')
    .css('color', 'inherit')
     end
     end
     add(';', args.innerstyle, ';">')
     tbl.cssText(args.innerstyle)
   
     renderTitleRow(args)
     renderTitleRow(tbl)
     renderAboveRow(args)
     renderAboveRow(tbl)
     renderFirstListRow(args)
     renderFirstListRow(tbl)
 
     -- render lists 2 through N
     -- render lists 2 through N
     for i, listnum in ipairs(listnums) do
     for i, listnum in ipairs(listnums) do
         if listnum > 1 then
         if listnum > 1 then
             renderNthListRow(args, listnum)  
             renderNthListRow(tbl, listnum)  
         end
         end
     end
     end
    renderBelowRow(tbl)
   
    return tbl
end
function p._navbox(navboxArgs)
args = navboxArgs
    for k, v in pairs(args) do
        local listnum = ('' .. k):match('^list(%d+)$')
        if listnum then table.insert(listnums, tonumber(listnum)) end
    end
    table.sort(listnums)
    border = trim(args.border or args[1] or '')


    renderBelowRow(args)
-- render the main body of the navbox
local tbl = renderMainTable()


    add('</table>')
-- render the appropriate wrapper around the navbox, depending on the border param
   
local res = HtmlBuilder.create()
     if border == 'subgroup' or border == 'child' then
     if border == 'subgroup' or border == 'child' then
         add('<div>')
         res
      .tag('/div', {unclosed = true})
      .done()
      .node(tbl)
      .tag('div', {unclosed = true})
     elseif border ~= 'none' then
     elseif border ~= 'none' then
        add('</td></tr></table>')
    res
    .tag('table')
  .attr('cellspacing', 0)
    .addClass('navbox')
    .css('border-spacing', 0)
    .cssText(args.bodystyle)
    .cssText(args.style)
    .tag('tr')
    .tag('td')
    .css('padding', '2px')
    .node(tbl)
     end
     end
 
     -- TODO: add tracking categories
     -- TODO: add tracking categories
   
     return table.concat(ret, '')
     return tostring(res)
end
end
 
function p.navbox(frame)
function p.navbox(frame)
     return p._navbox(frame:getParent().args)
     return p._navbox(frame:getParent().args)
end
end
 
return p
return p
Anonymous user