Module:Navbox: Difference between revisions

From WIDEVERSE Wiki
Jump to navigation Jump to search
m (1 revision imported)
(Undo imported revision 14948 by user en>Uzume)
Tag: Undo
 
Line 1: Line 1:
require('strict')
-- <nowiki>
--
-- Implements {{navbox}}
--
 
local p = {}
local p = {}
local navbar = require('Module:Navbar')._navbar
local tnavbar = require( 'Module:Tnavbar' )
local cfg = mw.loadData('Module:Navbox/configuration')
local yesno = require( 'Module:Yesno' )
local getArgs -- lazily initialized
local onmain = require('Module:Mainonly').on_main()
local args
local page_title = mw.title.getCurrentTitle().fullText
local format = string.format
--
 
-- Helper for inserting a new row into the navbox
local function striped(wikitext, border)
--
-- Return wikitext with markers replaced for odd/even striping.
-- @param tbl {mw.html table}
-- Child (subgroup) navboxes are flagged with a category that is removed
-- @return tbl {mw.html table}
-- by parent navboxes. The result is that the category shows all pages
--
-- where a child navbox is not contained in a parent navbox.
local function insertRow( tbl )
local orphanCat = cfg.category.orphan
return tbl:tag( 'tr' )
if border == cfg.keyword.border_subgroup and args[cfg.arg.orphan] ~= cfg.keyword.orphan_yes then
-- No change; striping occurs in outermost navbox.
return wikitext .. orphanCat
end
local first, second = cfg.class.navbox_odd_part, cfg.class.navbox_even_part
if args[cfg.arg.evenodd] then
if args[cfg.arg.evenodd] == cfg.keyword.evenodd_swap then
first, second = second, first
else
first = args[cfg.arg.evenodd]
second = first
end
end
local changer
if first == second then
changer = first
else
local index = 0
changer = function (code)
if code == '0' then
-- Current occurrence is for a group before a nested table.
-- Set it to first as a valid although pointless class.
-- The next occurrence will be the first row after a title
-- in a subgroup and will also be first.
index = 0
return first
end
index = index + 1
return index % 2 == 1 and first or second
end
end
local regex = orphanCat:gsub('([%[%]])', '%%%1')
return (wikitext:gsub(regex, ''):gsub(cfg.marker.regex, changer)) -- () omits gsub count
end
end


local function processItem(item, nowrapitems)
--
if item:sub(1, 2) == '{|' then
-- Creates the navbox table
-- Applying nowrap to lines in a table does not make sense.
--
-- Add newlines to compensate for trim of x in |parm=x in a template.
-- @param args {table}
return '\n' .. item ..'\n'
-- @return tbl {mw.html table}
end
--
if nowrapitems == cfg.keyword.nowrapitems_yes then
local function createTbl( args )
local lines = {}
for line in (item .. '\n'):gmatch('([^\n]*)\n') do
local prefix, content = line:match('^([*:;#]+)%s*(.*)')
if prefix and not content:match(cfg.pattern.nowrap) then
line = format(cfg.nowrap_item, prefix, content)
end
table.insert(lines, line)
end
item = table.concat(lines, '\n')
end
if item:match('^[*:;#]') then
return '\n' .. item ..'\n'
end
return item
end


local function has_navbar()
local tbl = mw.html.create( 'table' )
return args[cfg.arg.navbar] ~= cfg.keyword.navbar_off
and args[cfg.arg.navbar] ~= cfg.keyword.navbar_plain
and (
args[cfg.arg.name]
or mw.getCurrentFrame():getParent():getTitle():gsub(cfg.pattern.sandbox, '')
~= cfg.pattern.navbox
)
end


local function renderNavBar(titleCell)
tbl
if has_navbar() then
:addClass( yesno( args.subgroup ) and 'navbox-subgroup' or 'navbox' )
titleCell:wikitext(navbar{
:addClass( 'nowraplinks' )
[cfg.navbar.name] = args[cfg.arg.name],
[cfg.navbar.mini] = 1,
[cfg.navbar.fontstyle] = (args[cfg.arg.basestyle] or '') .. ';' ..
(args[cfg.arg.titlestyle] or '') ..
';background:none transparent;border:none;box-shadow:none;padding:0;'
})
end


end
if not yesno( args.subgroup ) and
( args.state == 'collapsed' or
  args.state == 'uncollapsed' or
  args.state == 'autocollapse' or
  -- defaults to autocollapse
  args.state == nil )
then
tbl:addClass( 'mw-collapsible' )


local function renderTitleRow(tbl)
if args.state == 'collapsed' then
if not args[cfg.arg.title] then return end
tbl:addClass( 'mw-collapsed' )
elseif args.state == 'uncollapsed' then
tbl:addClass('navbox-uncollapsed')
end
end


local titleRow = tbl:tag('tr')
if yesno( args.collapsible ) then
tbl:addClass( 'navbox-collapsible' )
end


local titleCell = titleRow:tag('th'):attr('scope', 'col')
if args.style then
tbl:cssText( args.style )
end


local titleColspan = 2
-- manually set collapse/expand messages
if args[cfg.arg.imageleft] then titleColspan = titleColspan + 1 end
-- bug causing the default database messages to be used
if args[cfg.arg.image] then titleColspan = titleColspan + 1 end
tbl
:attr( {
['cellspacing'] = '0',
['data-expandtext'] = 'show',
['data-collapsetext'] = 'hide',
['data-navbox-name'] = args.name
} )


titleCell
return tbl
:cssText(args[cfg.arg.basestyle])
end
:cssText(args[cfg.arg.titlestyle])
:addClass(cfg.class.navbox_title)
:attr('colspan', titleColspan)
 
renderNavBar(titleCell)


titleCell
--
:tag('div')
-- Wrapper for [[Module:Tnavbar]]
-- id for aria-labelledby attribute
--
:attr('id', mw.uri.anchorEncode(args[cfg.arg.title]))
-- @param args {table}
:addClass(args[cfg.arg.titleclass])
-- @return {string}
:css('font-size', '114%')
--
:css('margin', '0 4em')
local function navbar( args )
:wikitext(processItem(args[cfg.arg.title]))
return tnavbar._collapsible( { [1] = '<span style="display:none;">RS3 </span>'..args.title, [2] = args.name } )
end
end


local function getAboveBelowColspan()
--
local ret = 2
-- Creates the header (what you see when the navbox is collapsed)
if args[cfg.arg.imageleft] then ret = ret + 1 end
--
if args[cfg.arg.image] then ret = ret + 1 end
-- @param tbl {mw.html table}
return ret
-- @param args {table}
end
-- @return {mw.html table}
--
local function header( tbl, args )
local div = insertRow( tbl )
:tag( 'th' )
:attr( 'colspan', '2' )
:addClass( 'navbox-title' )
:attr( 'id' , 'navbox-title' )
:tag( 'div' )


local function renderAboveRow(tbl)
-- @todo move this to site css so we can simplify this (hook off a class)
if not args[cfg.arg.above] then return end
-- to something like div:wikitext( args.name and navbar( args ) or args.title )
-- which can be appended to the above and returned straight away
if args.name then
div
:css( 'padding-right', args.state == 'plain' and '6em' or '0' )
:wikitext( navbar( args ) )
else
div
:css( 'padding-left', args.state == 'plain' and '0' or '6em' )
:wikitext( '<span style="display:none;">RS3 </span>'..args.title )
end


tbl:tag('tr')
return div:allDone()
:tag('td')
:addClass(cfg.class.navbox_abovebelow)
:addClass(args[cfg.arg.aboveclass])
:cssText(args[cfg.arg.basestyle])
:cssText(args[cfg.arg.abovestyle])
:attr('colspan', getAboveBelowColspan())
:tag('div')
-- id for aria-labelledby attribute, if no title
:attr('id', (not args[cfg.arg.title]) and mw.uri.anchorEncode(args[cfg.arg.above]) or nil)
:wikitext(processItem(args[cfg.arg.above], args[cfg.arg.nowrapitems]))
end
end


local function renderBelowRow(tbl)
--
if not args[cfg.arg.below] then return end
-- Inserts a row into the navbox
--
-- @param tbl {mw.html table}
-- @param gtitle {string}
-- @param group {string}
-- @param gtype {string}
-- @param gcats {table}
-- @param style {string}
-- @return {mw.html table}
--
local function row( tbl, gtitle, group, gtype, gcats, style, _name, subgroup )
local tr = insertRow( tbl )
local td
if gtitle then
td = tr
:addClass( 'navbox-group' )
:tag( 'td' )
:addClass( 'navbox-group-title' )
:wikitext( gtitle )
:done()
:tag( 'td' )
else
td = tr
:addClass( 'navbox-group' )
:addClass( 'navbox-group-split' )
:tag( 'td' )
:addClass( 'navbox-group-title-hidden' )
:attr( 'colspan', '0' )
:css( 'display', 'none' )
:done()
:tag( 'td' )
:attr( 'colspan', '2' )
end


tbl:tag('tr')
--[[
:tag('td')
  List styling
:addClass(cfg.class.navbox_abovebelow)
  This is unlikely to be implemented in the near future due to it requiring extra css to work
:addClass(args[cfg.arg.belowclass])
  and mobile currently not supporting that css.
:cssText(args[cfg.arg.basestyle])
  As an example, it lets you do the following instead if using {{*}} all the time
:cssText(args[cfg.arg.belowstyle])
  | group3 =
:attr('colspan', getAboveBelowColspan())
  * {{plink|foo}}
:tag('div')
  * {{plink|bar}}
:wikitext(processItem(args[cfg.arg.below], args[cfg.arg.nowrapitems]))
  * {{plink|baz}}
end
]]
if mw.ustring.match( group, '^%s*%*' ) then
td:newline()
 
-- trim whitespace on bullets
local spl = mw.text.split( group, '\n' )


local function renderListRow(tbl, index, listnum, listnums_size)
for i = 1, #spl do
local row = tbl:tag('tr')
spl[i] = mw.text.trim( spl[i] )
end


if index == 1 and args[cfg.arg.imageleft] then
group = '\n' .. table.concat( spl, '\n' )
row
:tag('td')
:addClass(cfg.class.noviewer)
:addClass(cfg.class.navbox_image)
:addClass(args[cfg.arg.imageclass])
:css('width', '1px')               -- Minimize width
:css('padding', '0 2px 0 0')
:cssText(args[cfg.arg.imageleftstyle])
:attr('rowspan', listnums_size)
:tag('div')
:wikitext(processItem(args[cfg.arg.imageleft]))
end
end


local group_and_num = format(cfg.arg.group_and_num, listnum)
--local group2 = group
local groupstyle_and_num = format(cfg.arg.groupstyle_and_num, listnum)
--local group3 = group2
if args[group_and_num] then
-- analytics
local groupCell = row:tag('th')


-- id for aria-labelledby attribute, if lone group with no title or above
--if _name then
if listnum == 1 and not (args[cfg.arg.title] or args[cfg.arg.above] or args[cfg.arg.group2]) then
-- local name = mw.ustring.gsub(_name,' ','_')
groupCell
-- for v in mw.ustring.gmatch(group,'%[%[[^%]]+%]%]') do
:attr('id', mw.uri.anchorEncode(args[cfg.arg.group1]))
-- if mw.ustring.match(v,'%[%[File:.+|link=') then
end
-- local link = mw.ustring.match(v,'|link=([^%]|]+)')
-- if link then
-- local linkrep = mw.ustring.gsub(link,'([%%%]%[%-^$*()+?])','%%%1')
-- local _link = mw.ustring.gsub(link,' ','_')
-- local newfile = mw.ustring.gsub(v,'|link='..linkrep,string.format('|link=https://runescape.wiki/w/%s?f=%s',_link,name))
-- local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')
-- group2 = mw.ustring.gsub(group2,w,newfile)
-- end
-- elseif mw.ustring.match(v,'%[%[Category:') then
-- nothing
-- else
-- local link = mw.ustring.match(v,'%[%[([^%]|]+)')
-- local txt = mw.ustring.match(v,'%|([^%]|]+)') or link


groupCell
-- local newlink = ''
:attr('scope', 'row')
:addClass(cfg.class.navbox_group)
:addClass(args[cfg.arg.groupclass])
:cssText(args[cfg.arg.basestyle])
-- If groupwidth not specified, minimize width
:css('width', args[cfg.arg.groupwidth] or '1%')


groupCell
-- black links if current page
:cssText(args[cfg.arg.groupstyle])
-- if link == page_title then
:cssText(args[groupstyle_and_num])
-- newlink = string.format('<b>%s</b>',txt)
:wikitext(args[group_and_num])
-- else
end
-- local _link = mw.ustring.gsub(link or '',' ','_')
-- newlink = string.format('[https://runescape.wiki.com/w/%s?n=%s %s]',_link,name,txt)
-- end
-- local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')
-- group2 = mw.ustring.gsub(group2,w,newlink)
-- end
-- end


local listCell = row:tag('td')
--[==[
fix [[these kind]]s of [[link]]s post analytics parse
]==]
-- group3 = group2


if args[group_and_num] then
-- for v in mw.ustring.gmatch(group2,'%[https://runescape.wiki.com/w[^%]]-%]%a') do
listCell
-- local rep = mw.ustring.gsub(v,'%]','')
:addClass(cfg.class.navbox_list_with_group)
-- rep = rep..']'
else
-- local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')
listCell:attr('colspan', 2)
end


if not args[cfg.arg.groupwidth] then
-- group3 = mw.ustring.gsub(group2,w,rep)
listCell:css('width', '100%')
-- end
end
--end


local rowstyle  -- usually nil so cssText(rowstyle) usually adds nothing
td
if index % 2 == 1 then
:addClass( 'navbox-list' )
rowstyle = args[cfg.arg.oddstyle]
:wikitext( group ) --group3
else
rowstyle = args[cfg.arg.evenstyle]
end


local list_and_num = format(cfg.arg.list_and_num, listnum)
if gtype and mw.ustring.lower( gtype ) == 'subgroup' then
local listText = args[list_and_num]
td
local oddEven = cfg.marker.oddeven
:addClass( 'navbox-parent' )
if listText:sub(1, 12) == '</div><table' then
:css( {
-- Assume list text is for a subgroup navbox so no automatic striping for this row.
padding = '0',
oddEven = listText:find(cfg.pattern.navbox_title) and cfg.marker.restart or cfg.class.navbox_odd_part
['border-bottom'] = '0'
} )
end
end


local liststyle_and_num = format(cfg.arg.liststyle_and_num, listnum)
if style then
local listclass_and_num = format(cfg.arg.listclass_and_num, listnum)
td:cssText( style )
listCell
:css('padding', '0')
:cssText(args[cfg.arg.liststyle])
:cssText(rowstyle)
:cssText(args[liststyle_and_num])
:addClass(cfg.class.navbox_list)
:addClass(cfg.class.navbox_part .. oddEven)
:addClass(args[cfg.arg.listclass])
:addClass(args[listclass_and_num])
:tag('div')
:css('padding',
(index == 1 and args[cfg.arg.list1padding]) or args[cfg.arg.listpadding] or '0 0.25em'
)
:wikitext(processItem(listText, args[cfg.arg.nowrapitems]))
 
if index == 1 and args[cfg.arg.image] then
row
:tag('td')
:addClass(cfg.class.noviewer)
:addClass(cfg.class.navbox_image)
:addClass(args[cfg.arg.imageclass])
:css('width', '1px')              -- Minimize width
:css('padding', '0 0 0 2px')
:cssText(args[cfg.arg.imagestyle])
:attr('rowspan', listnums_size)
:tag('div')
:wikitext(processItem(args[cfg.arg.image]))
end
end
end
local function has_list_class(htmlclass)
local patterns = {
'^' .. htmlclass .. '$',
'%s' .. htmlclass .. '$',
'^' .. htmlclass .. '%s',
'%s' .. htmlclass .. '%s'
}
for arg, _ in pairs(args) do
-- add subgroup categories
if type(arg) == 'string' and mw.ustring.find(arg, cfg.pattern.class) then
if next(gcats) and onmain then
for _, pattern in ipairs(patterns) do
first_char = page_title:sub(1,1)
if mw.ustring.find(args[arg] or '', pattern) then
title_pattern = "[" .. first_char:upper() .. first_char:lower() .. "]" .. page_title:sub(2)
return true
title_pattern = title_pattern :gsub("%(","%%(")
:gsub("%)","%%)")
:gsub("_"," ")
:gsub(" ","[_ ]")
:gsub("%-","%%-")
local link_patterns = {
"%[%["..title_pattern.."%]%]", "%[%["..title_pattern.."%|", "{{[Pp]link%|"..title_pattern.."}}",
"{{[Pp]link%|"..title_pattern.."%|", "{{[Pp]linkp%|"..title_pattern.."}}",
"{{[Pp]linkp%|"..title_pattern.."%|", "{{[Ii]linkp%|"..title_pattern.."}}",
"{{[Ii]linkp%|"..title_pattern.."%|", "{{[Cc]hatl%|"..title_pattern.."}}",
"{{[Cc]hatl%|"..title_pattern.."%|"
}
for _,v in ipairs(link_patterns) do
if group:match(v) then
for _,cat in ipairs(gcats) do
td:wikitext('[[Category:'..cat..']]')
end
end
break
end
end
end
end
end
end
return false
return td:allDone()
end
end


-- there are a lot of list classes in the wild, so we add their TemplateStyles
--
local function add_list_styles()
-- Inserts a footer into the navbox
local frame = mw.getCurrentFrame()
--
local function add_list_templatestyles(htmlclass, templatestyles)
-- @param tbl {mw.html table}
if has_list_class(htmlclass) then
-- @param args {table}
return frame:extensionTag{
-- @return {mw.html table}
name = 'templatestyles', args = { src = templatestyles }
--
}
local function footer( tbl, args )
else
local th = insertRow( tbl )
return ''
:tag( 'th' )
end
:attr( 'colspan', '2' )
end
:addClass( 'navbox-footer' )
local hlist_styles = add_list_templatestyles('hlist', cfg.hlist_templatestyles)
local plainlist_styles = add_list_templatestyles('plainlist', cfg.plainlist_templatestyles)
-- a second workaround for [[phab:T303378]]
-- when that issue is fixed, we can actually use has_navbar not to emit the
-- tag here if we want
if has_navbar() and hlist_styles == '' then
hlist_styles = frame:extensionTag{
name = 'templatestyles', args = { src = cfg.hlist_templatestyles }
}
end
-- hlist -> plainlist is best-effort to preserve old Common.css ordering.
-- this ordering is not a guarantee because most navboxes will emit only
-- one of these classes [hlist_note]
return hlist_styles .. plainlist_styles
end


local function needsHorizontalLists(border)
if args.fstyle then
if border == cfg.keyword.border_subgroup or args[cfg.arg.tracking] == cfg.keyword.tracking_no then
th:cssText( args.fstyle )
return false
end
end
return not has_list_class(cfg.pattern.hlist) and not has_list_class(cfg.pattern.plainlist)
end


local function hasBackgroundColors()
if mw.ustring.match( args.footer, '^%s*%*' ) then
for _, key in ipairs({cfg.arg.titlestyle, cfg.arg.groupstyle,
th:newline()
cfg.arg.basestyle, cfg.arg.abovestyle, cfg.arg.belowstyle}) do
if tostring(args[key]):find('background', 1, true) then
return true
end
end
return false
end


local function hasBorders()
-- trim whitespace on bullets
for _, key in ipairs({cfg.arg.groupstyle, cfg.arg.basestyle,
local spl = mw.text.split( args.footer, '\n' )
cfg.arg.abovestyle, cfg.arg.belowstyle}) do
if tostring(args[key]):find('border', 1, true) then
return true
end
end
return false
end


local function isIllegible()
for i = 1, #spl do
local styleratio = require('Module:Color contrast')._styleratio
spl[i] = mw.text.trim( spl[i] )
for key, style in pairs(args) do
if tostring(key):match(cfg.pattern.style) then
if styleratio{mw.text.unstripNoWiki(style)} < 4.5 then
return true
end
end
end
end
return false
end
local function getTrackingCategories(border)
local cats = {}
if needsHorizontalLists(border) then table.insert(cats, cfg.category.horizontal_lists) end
if hasBackgroundColors() then table.insert(cats, cfg.category.background_colors) end
if isIllegible() then table.insert(cats, cfg.category.illegible) end
if hasBorders() then table.insert(cats, cfg.category.borders) end
return cats
end


local function renderTrackingCategories(builder, border)
args.footer = table.concat( spl, '\n' )
local title = mw.title.getCurrentTitle()
if title.namespace ~= 10 then return end -- not in template space
local subpage = title.subpageText
if subpage == cfg.keyword.subpage_doc or subpage == cfg.keyword.subpage_sandbox
or subpage == cfg.keyword.subpage_testcases then return end


for _, cat in ipairs(getTrackingCategories(border)) do
th:addClass( 'navbox-list' )
builder:wikitext('[[Category:' .. cat .. ']]')
end
end
end


local function renderMainTable(border, listnums)
th:wikitext( args.footer )
local tbl = mw.html.create('table')
:addClass(cfg.class.nowraplinks)
:addClass(args[cfg.arg.bodyclass])


local state = args[cfg.arg.state]
return th:allDone()
if args[cfg.arg.title] and state ~= cfg.keyword.state_plain and state ~= cfg.keyword.state_off then
end
if state == cfg.keyword.state_collapsed then
state = cfg.class.collapsed
end
tbl
:addClass(cfg.class.collapsible)
:addClass(state or cfg.class.autocollapse)
end


tbl:css('border-spacing', 0)
--
if border == cfg.keyword.border_subgroup or border == cfg.keyword.border_none then
-- Adds [[Category:Navigational templates]] to navbox template pages
tbl
--
:addClass(cfg.class.navbox_subgroup)
-- @return {string}
:cssText(args[cfg.arg.bodystyle])
--
:cssText(args[cfg.arg.style])
local function categories()
else  -- regular navbox - bodystyle and style will be applied to the wrapper table
local title = mw.title.getCurrentTitle()
tbl
local page = title.text
:addClass(cfg.class.navbox_inner)
local ns = title.nsText
:css('background', 'transparent')
:css('color', 'inherit')
end
tbl:cssText(args[cfg.arg.innerstyle])


renderTitleRow(tbl)
if ns == 'Template' then
renderAboveRow(tbl)
-- sort in category by pagename
local listnums_size = #listnums
return '[[Category:Navigational templates|' .. page .. ']]'
for i, listnum in ipairs(listnums) do
else
renderListRow(tbl, i, listnum, listnums_size)
return ''
end
end
renderBelowRow(tbl)


return tbl
end
end


local function add_navbox_styles(hiding_templatestyles)
--
-- Adds [[Template:Navbox/doc]] to navbox template pages
--
-- @param args {table}
-- @return {string}
--
local function docs( args )
local frame = mw.getCurrentFrame()
local frame = mw.getCurrentFrame()
-- This is a lambda so that it doesn't need the frame as a parameter
local title = mw.title.getCurrentTitle()
local function add_user_styles(templatestyles)
local base = title.baseText
if templatestyles and templatestyles ~= '' then
local ns = title.nsText
return frame:extensionTag{
 
name = 'templatestyles', args = { src = templatestyles }
-- not if a subpage of [[Template:Navbox]]
}
if base ~= 'Navbox' and
end
-- in template ns
ns == 'Template' and
-- not a navbox group within a navbox
not yesno( args.subgroup ) and
-- not a collapsible navbox within a navbox
not yesno( args.collapsible ) and
-- not if the doc argument is not set to "yes"
yesno( args.doc, false )
then
return frame:expandTemplate{ title = 'Navbox/doc' }
else
return ''
return ''
end
end


-- get templatestyles. load base from config so that Lua only needs to do
-- the work once of parser tag expansion
local base_templatestyles = cfg.templatestyles
local templatestyles = add_user_styles(args[cfg.arg.templatestyles])
local child_templatestyles = add_user_styles(args[cfg.arg.child_templatestyles])
-- The 'navbox-styles' div exists to wrap the styles to work around T200206
-- more elegantly. Instead of combinatorial rules, this ends up being linear
-- number of CSS rules.
return mw.html.create('div')
:addClass(cfg.class.navbox_styles)
:wikitext(
add_list_styles() .. -- see [hlist_note] applied to 'before base_templatestyles'
base_templatestyles ..
templatestyles ..
child_templatestyles ..
table.concat(hiding_templatestyles)
)
:done()
end
end


-- work around [[phab:T303378]]
--
-- for each arg: find all the templatestyles strip markers, insert them into a
-- Navbox method to allow it to be called by other modules
-- table. then remove all templatestyles markers from the arg
--
local function move_hiding_templatestyles(args)
-- @param _args {table}
local gfind = string.gfind
-- @return {string}
local gsub = string.gsub
--
local templatestyles_markers = {}
function p._navbox( _args )
local strip_marker_pattern = '(\127[^\127]*UNIQ%-%-templatestyles%-%x+%-QINU[^\127]*\127)'
local args = {}
for k, arg in pairs(args) do
local wkCss = ''
for marker in gfind(arg, strip_marker_pattern) do
local wkDiv = ''
table.insert(templatestyles_markers, marker)
local j
-- preserves parser function behaviour where an empty string is considered undefined
-- or nil in lua's case
for k, v in pairs( _args ) do
if v ~= '' then
args[k] = v
end
end
args[k] = gsub(arg, strip_marker_pattern, '')
end
end
return templatestyles_markers
end


function p._navbox(navboxArgs)
local tbl = createTbl( args )
args = navboxArgs
local hiding_templatestyles = move_hiding_templatestyles(args)
local listnums = {}


for k, _ in pairs(args) do
if not yesno( args.subgroup ) then
if type(k) == 'string' then
tbl = header( tbl, args )
local listnum = k:match(cfg.pattern.listnum)
if listnum then table.insert(listnums, tonumber(listnum)) end
end
end
end
table.sort(listnums)


local border = mw.text.trim(args[cfg.arg.border] or args[1] or '')
-- insert up to 20 rows
if border == cfg.keyword.border_child then
--
border = cfg.keyword.border_subgroup
-- 20 is a limit inherited from wikipedia when we copied this over
end
-- and we've never had a reason to extend it
for i = 1, 20 do
j = tostring( i )


-- render the main body of the navbox
if args['group' .. j] then
local tbl = renderMainTable(border, listnums)
local gcats = {}
 
for p = 1, 20 do
local res = mw.html.create()
local q = tostring ( p )
-- render the appropriate wrapper for the navbox, based on the border param
 
if args['g' .. j .. 'cat' .. q] then
if border == cfg.keyword.border_none then
table.insert(gcats, args['g' .. j .. 'cat' .. q])
res:node(add_navbox_styles(hiding_templatestyles))
else
local nav = res:tag('div')
break
:attr('role', 'navigation')
end
:node(tbl)
end
-- aria-labelledby title, otherwise above, otherwise lone group
if args[cfg.arg.title] or args[cfg.arg.above] or (args[cfg.arg.group1]
tbl = row( tbl, args['gtitle' .. j], args['group' .. j], args['gtype' .. j], gcats, args['style' .. j], args.name, args.subgroup )
and not args[cfg.arg.group2]) then
nav:attr(
'aria-labelledby',
mw.uri.anchorEncode(
args[cfg.arg.title] or args[cfg.arg.above] or args[cfg.arg.group1]
)
)
else
nav:attr('aria-label', cfg.aria_label)
end
elseif border == cfg.keyword.border_subgroup then
-- We assume that this navbox is being rendered in a list cell of a
-- parent navbox, and is therefore inside a div with padding:0em 0.25em.
-- We start with a </div> to avoid the padding being applied, and at the
-- end add a <div> to balance out the parent's </div>
res
:wikitext('</div>')
:node(tbl)
:wikitext('<div>')
else
res:node(add_navbox_styles(hiding_templatestyles))
local nav = res:tag('div')
:attr('role', 'navigation')
:addClass(cfg.class.navbox)
:addClass(args[cfg.arg.navboxclass])
:cssText(args[cfg.arg.bodystyle])
:cssText(args[cfg.arg.style])
:css('padding', '3px')
:node(tbl)
-- aria-labelledby title, otherwise above, otherwise lone group
if args[cfg.arg.title] or args[cfg.arg.above]
or (args[cfg.arg.group1] and not args[cfg.arg.group2]) then
nav:attr(
'aria-labelledby',
mw.uri.anchorEncode(args[cfg.arg.title] or args[cfg.arg.above] or args[cfg.arg.group1])
)
else
else
nav:attr('aria-label', cfg.aria_label)
break
end
end
end
end


if (args[cfg.arg.nocat] or cfg.keyword.nocat_false):lower() == cfg.keyword.nocat_false then
if args.footer then
renderTrackingCategories(res, border)
tbl = footer( tbl, args )
end
end
return striped(tostring(res), border)
end


function p.navbox(frame)
tbl = tostring( tbl )
if not getArgs then
 
getArgs = require('Module:Arguments').getArgs
local cats = ''
if not yesno(args.subgroup) and not yesno(args.hidecat) then
cats = categories()
end
end
args = getArgs(frame, {wrappers = {cfg.pattern.navbox}})
local docs = docs( args )


-- Read the arguments in the order they'll be output in, to make references
return tbl .. cats .. docs
-- number in the right order.
end
local _
_ = args[cfg.arg.title]
_ = args[cfg.arg.above]
-- Limit this to 20 as covering 'most' cases (that's a SWAG) and because
-- iterator approach won't work here
for i = 1, 20 do
_ = args[format(cfg.arg.group_and_num, i)]
_ = args[format(cfg.arg.list_and_num, i)]
end
_ = args[cfg.arg.below]


return p._navbox(args)
--
-- Main navbox method accessed through #invoke
--
-- @param frame {table}
-- @return {string}
--
function p.navbox( frame )
local args = frame:getParent().args
return p._navbox( args )
end
end


return p
return p
-- </nowiki>

Latest revision as of 16:08, 14 November 2023

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

-- <nowiki>
--
-- Implements {{navbox}}
--

local p = {}
local tnavbar = require( 'Module:Tnavbar' )
local yesno = require( 'Module:Yesno' )
local onmain = require('Module:Mainonly').on_main()
local page_title = mw.title.getCurrentTitle().fullText
--
-- Helper for inserting a new row into the navbox
--
-- @param tbl {mw.html table}
-- @return tbl {mw.html table}
--
local function insertRow( tbl )
	return tbl:tag( 'tr' )
end

--
-- Creates the navbox table
--
-- @param args {table}
-- @return tbl {mw.html table}
--
local function createTbl( args )

	local tbl = mw.html.create( 'table' )

	tbl
		:addClass( yesno( args.subgroup ) and 'navbox-subgroup' or 'navbox' )
		:addClass( 'nowraplinks' )

	if not yesno( args.subgroup ) and
		( args.state == 'collapsed' or
		  args.state == 'uncollapsed' or
		  args.state == 'autocollapse' or
		  -- defaults to autocollapse
		  args.state == nil )
	then
		tbl:addClass( 'mw-collapsible' )

		if args.state == 'collapsed' then
			tbl:addClass( 'mw-collapsed' )
		elseif args.state == 'uncollapsed' then
			tbl:addClass('navbox-uncollapsed')
		end
	end

	if yesno( args.collapsible ) then
		tbl:addClass( 'navbox-collapsible' )
	end

	if args.style then
		tbl:cssText( args.style )
	end

	-- manually set collapse/expand messages
	-- bug causing the default database messages to be used
	tbl
		:attr( {
			['cellspacing'] = '0',
			['data-expandtext'] = 'show',
			['data-collapsetext'] = 'hide',
			['data-navbox-name'] = args.name
		} )

	return tbl
end

--
-- Wrapper for [[Module:Tnavbar]]
--
-- @param args {table}
-- @return {string}
--
local function navbar( args )
	return tnavbar._collapsible( { [1] = '<span style="display:none;">RS3 </span>'..args.title, [2] = args.name } )
end

--
-- Creates the header (what you see when the navbox is collapsed)
--
-- @param tbl {mw.html table}
-- @param args {table}
-- @return {mw.html table}
--
local function header( tbl, args )
	local div = insertRow( tbl )
		:tag( 'th' )
			:attr( 'colspan', '2' )
			:addClass( 'navbox-title' )
			:attr( 'id' , 'navbox-title' )
				:tag( 'div' )

	-- @todo move this to site css so we can simplify this (hook off a class)
	-- to something like div:wikitext( args.name and navbar( args ) or args.title )
	-- which can be appended to the above and returned straight away
	if args.name then
		div
			:css( 'padding-right', args.state == 'plain' and '6em' or '0' )
			:wikitext( navbar( args ) )
	else
		div
			:css( 'padding-left', args.state == 'plain' and '0' or '6em' )
			:wikitext( '<span style="display:none;">RS3 </span>'..args.title )
	end

	return div:allDone()
end

--
-- Inserts a row into the navbox
--
-- @param tbl {mw.html table}
-- @param gtitle {string}
-- @param group {string}
-- @param gtype {string}
-- @param gcats {table}
-- @param style {string}
-- @return {mw.html table}
--
local function row( tbl, gtitle, group, gtype, gcats, style, _name, subgroup )
	local tr = insertRow( tbl )
	local td
	
	if gtitle then
		td = tr
			:addClass( 'navbox-group' )
			:tag( 'td' )
				:addClass( 'navbox-group-title' )
				:wikitext( gtitle )
				:done()
			:tag( 'td' )
	else
		td = tr
			:addClass( 'navbox-group' )
			:addClass( 'navbox-group-split' )
			:tag( 'td' )
				:addClass( 'navbox-group-title-hidden' )
				:attr( 'colspan', '0' )
				:css( 'display', 'none' )
				:done()
			:tag( 'td' )
				:attr( 'colspan', '2' )
	end

	--[[
	   List styling
	   This is unlikely to be implemented in the near future due to it requiring extra css to work
	   and mobile currently not supporting that css.
	   As an example, it lets you do the following instead if using {{*}} all the time
	   | group3 =
	   * {{plink|foo}}
	   * {{plink|bar}}
	   * {{plink|baz}}
	]]
	if mw.ustring.match( group, '^%s*%*' ) then
		td:newline()

		-- trim whitespace on bullets
		local spl = mw.text.split( group, '\n' )

		for i = 1, #spl do
			spl[i] = mw.text.trim( spl[i] )
		end

		group = '\n' .. table.concat( spl, '\n' )		
	end

	--local group2 = group
	--local group3 = group2
	-- analytics

	--if _name then
	--	local name = mw.ustring.gsub(_name,' ','_')
	--	for v in mw.ustring.gmatch(group,'%[%[[^%]]+%]%]') do
	--		if mw.ustring.match(v,'%[%[File:.+|link=') then
	--			local link = mw.ustring.match(v,'|link=([^%]|]+)')
	--			if link then
	--				local linkrep = mw.ustring.gsub(link,'([%%%]%[%-^$*()+?])','%%%1')
	--				local _link = mw.ustring.gsub(link,' ','_')
	--				local newfile = mw.ustring.gsub(v,'|link='..linkrep,string.format('|link=https://runescape.wiki/w/%s?f=%s',_link,name))
	--				local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')
	--				group2 = mw.ustring.gsub(group2,w,newfile)
	--			end
	--		elseif mw.ustring.match(v,'%[%[Category:') then
				-- nothing
	--		else
	--			local link = mw.ustring.match(v,'%[%[([^%]|]+)')
	--			local txt = mw.ustring.match(v,'%|([^%]|]+)') or link

	--			local newlink = ''

				-- black links if current page
	--			if link == page_title then
	--				newlink = string.format('<b>%s</b>',txt)
	--			else
	--				local _link = mw.ustring.gsub(link or '',' ','_')
	--				newlink = string.format('[https://runescape.wiki.com/w/%s?n=%s %s]',_link,name,txt)
	--			end
	--			local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')
	--			group2 = mw.ustring.gsub(group2,w,newlink)
	--		end
	--	end

		--[==[
			fix [[these kind]]s of [[link]]s post analytics parse
			]==]
	--	group3 = group2

	--	for v in mw.ustring.gmatch(group2,'%[https://runescape.wiki.com/w[^%]]-%]%a') do
	--		local rep = mw.ustring.gsub(v,'%]','')
	--		rep = rep..']'
	--		local w = mw.ustring.gsub(v,'([%%%]%[%-^$*()+?])','%%%1')

	--		group3 = mw.ustring.gsub(group2,w,rep)
	--	end
	--end

	td
		:addClass( 'navbox-list' )
		:wikitext( group ) --group3

	if gtype and mw.ustring.lower( gtype ) == 'subgroup' then
		td
			:addClass( 'navbox-parent' )
			:css( {
				padding = '0',
				['border-bottom'] = '0'
			} )
	end

	if style then
		td:cssText( style )
	end
	
	-- add subgroup categories
	if next(gcats) and onmain then
		first_char = page_title:sub(1,1)
		title_pattern = "[" .. first_char:upper() .. first_char:lower() .. "]" .. page_title:sub(2)
		title_pattern = title_pattern :gsub("%(","%%(")
								:gsub("%)","%%)")
								:gsub("_"," ")
								:gsub(" ","[_ ]")
								:gsub("%-","%%-")
		local link_patterns = {
			"%[%["..title_pattern.."%]%]", "%[%["..title_pattern.."%|", "{{[Pp]link%|"..title_pattern.."}}",
			"{{[Pp]link%|"..title_pattern.."%|", "{{[Pp]linkp%|"..title_pattern.."}}",
			"{{[Pp]linkp%|"..title_pattern.."%|", "{{[Ii]linkp%|"..title_pattern.."}}",
			"{{[Ii]linkp%|"..title_pattern.."%|", "{{[Cc]hatl%|"..title_pattern.."}}",
			"{{[Cc]hatl%|"..title_pattern.."%|"
		}
		for _,v in ipairs(link_patterns) do
			if group:match(v) then
				for _,cat in ipairs(gcats) do
					td:wikitext('[[Category:'..cat..']]')
				end
				break
			end
		end
	end
	return td:allDone()
end

--
-- Inserts a footer into the navbox
--
-- @param tbl {mw.html table}
-- @param args {table}
-- @return {mw.html table}
--
local function footer( tbl, args )
	local th = insertRow( tbl )
		:tag( 'th' )
			:attr( 'colspan', '2' )
			:addClass( 'navbox-footer' )

	if args.fstyle then
		th:cssText( args.fstyle )
	end

	if mw.ustring.match( args.footer, '^%s*%*' ) then
		th:newline()

		-- trim whitespace on bullets
		local spl = mw.text.split( args.footer, '\n' )

		for i = 1, #spl do
			spl[i] = mw.text.trim( spl[i] )
		end

		args.footer = table.concat( spl, '\n' )

		th:addClass( 'navbox-list' )
	end

	th:wikitext( args.footer )

	return th:allDone()
end

--
-- Adds [[Category:Navigational templates]] to navbox template pages
--
-- @return {string}
--
local function categories()
	local title = mw.title.getCurrentTitle()
	local page = title.text
	local ns = title.nsText

	if ns == 'Template' then
		-- sort in category by pagename
		return '[[Category:Navigational templates|' .. page .. ']]'
	else
		return ''
	end

end

--
-- Adds [[Template:Navbox/doc]] to navbox template pages
--
-- @param args {table}
-- @return {string}
--
local function docs( args )
	local frame = mw.getCurrentFrame()
	local title = mw.title.getCurrentTitle()
	local base = title.baseText
	local ns = title.nsText

		-- not if a subpage of [[Template:Navbox]]
	if base ~= 'Navbox' and
		-- in template ns
		ns == 'Template' and
		-- not a navbox group within a navbox
		not yesno( args.subgroup ) and
		-- not a collapsible navbox within a navbox
		not yesno( args.collapsible ) and
		-- not if the doc argument is not set to "yes"
		yesno( args.doc, false )
	then
		return frame:expandTemplate{ title = 'Navbox/doc' }
	else
		return ''
	end

end

--
-- Navbox method to allow it to be called by other modules
--
-- @param _args {table}
-- @return {string}
--
function p._navbox( _args )
	local args = {}
	local wkCss = ''
	local wkDiv = ''
	local j
	
	-- preserves parser function behaviour where an empty string is considered undefined
	-- or nil in lua's case
	for k, v in pairs( _args ) do
		if v ~= '' then
			args[k] = v
		end
	end

	local tbl = createTbl( args )

	if not yesno( args.subgroup ) then
		tbl = header( tbl, args )
	end

	-- insert up to 20 rows
	--
	-- 20 is a limit inherited from wikipedia when we copied this over
	-- and we've never had a reason to extend it
	for i = 1, 20 do
		j = tostring( i )

		if args['group' .. j] then
			local gcats = {}
			for p = 1, 20 do
				local q = tostring ( p )
				
				if args['g' .. j .. 'cat' .. q] then
					table.insert(gcats, args['g' .. j .. 'cat' .. q])
				else
					break
				end
			end
					
			tbl = row( tbl, args['gtitle' .. j], args['group' .. j], args['gtype' .. j], gcats, args['style' .. j], args.name, args.subgroup )
		else
			break
		end
	end

	if args.footer then
		tbl = footer( tbl, args )
	end

	tbl = tostring( tbl )

	local cats = ''
	if not yesno(args.subgroup) and not yesno(args.hidecat) then
		cats = categories()
	end
	local docs = docs( args )

	return tbl .. cats .. docs
end

--
-- Main navbox method accessed through #invoke
--
-- @param frame {table}
-- @return {string}
--
function p.navbox( frame )
	local args = frame:getParent().args
	return p._navbox( args )
end

return p
-- </nowiki>