Module:Navbox: Difference between revisions

From Xenharmonic Wiki
Jump to navigation Jump to search
Ganaram inukshuk (talk | contribs)
m remove default title
Ganaram inukshuk (talk | contribs)
rewrite navbox code to use a table of lines instead of concatenating to one big line
Line 6: Line 6:
local yesno = require("Module:Yesno")
local yesno = require("Module:Yesno")


-- Header/footer row
-- TODO: add option for dataless header row, to complement the existence of
-- a headerless data row.
 
-- Navbox header/footer row
function p.navbox_header_footer(row_content)
function p.navbox_header_footer(row_content)
local row = "|-\n"
local lines = {}
.. '| colspan="2" style="font-size: 0.8em; text-align: center; background-color: #eaecf0; padding: 0em; border: 1px solid white;" | '
table.insert(lines, "|-")
.. row_content .. "\n"
table.insert(lines, '| colspan="2" style="font-size: 0.8em; text-align: center; background-color: #eaecf0; padding: 0em; border: 1px solid white;" | ' .. row_content)
 
return row
return table.concat(lines, "\n")
end
end


-- Navbox row
-- Navbox row
function p.navbox_row(row_content, is_navbox)
function p.navbox_row(row_content, is_navbox)
local is_navbox = (is_navbox ~= nil and is_navbox or false)
local is_navbox = is_navbox or false
local lines = {}
table.insert(lines, "|-")
-- TODO: add option for a dataless header row


local row = "|-\n"
if #row_content == 1 then
if #row_content == 1 then
-- Headerless row; takes up two cells
-- Headerless row; holds data for simple navboxes or a nested navbox
if is_navbox then
if is_navbox then
-- Row data is a child navbox; data cell has no padding
table.insert(lines, '| style="padding: 0;" colspan="2" |')
row = row
table.insert(lines, row_content[1])
.. '| style="padding: 0;" colspan="2" | \n'
.. row_content[1] .. "\n"
else
else
-- Row data is normal data
table.insert(lines, '| style="font-size: 0.9em; padding: 0.25em 0.5em;" colspan="2" | ' .. row_content[1])
row = row .. '| style="font-size: 0.9em; padding: 0.25em 0.5em;" colspan="2" | ' .. row_content[1] .. "\n"
end
    end
else
else
-- Row with header and data
-- Header + Data row; typical usecase for a row
table.insert(lines, '! style="white-space: nowrap; font-size: 0.9em; width: 5%; text-align: right; background-color: #eaecf0; padding: 0.25em 0.5em; border: 1px solid white;" |' .. row_content[1])
if is_navbox then
if is_navbox then
-- Row data is a child navbox; data cell has no padding
table.insert(lines, '| style="padding: 0;" |')
row = row
.. '! style="white-space: nowrap; font-size: 0.9em; width: 5%; text-align: right; background-color: #eaecf0; padding: 0.25em 0.5em; border: 1px solid white;" |'
..  row_content[1] .. "\n"
.. '| style="padding: 0;" |\n'  
.. row_content[2] .. "\n"
else
else
-- Row data is normal data
table.insert(lines, '| style="font-size: 0.9em; padding: 0.25em 0.5em;" | ')
row = row
.. '! style="white-space: nowrap; font-size: 0.9em; width: 5%; text-align: right; background-color: #eaecf0; padding: 0.25em 0.5em; border: 1px solid white;" |'
..  row_content[1] .. "\n"
.. '| style="font-size: 0.9em; padding: 0.25em 0.5em;" | ' .. row_content[2] .. "\n"
end
end
table.insert(lines, row_content[2])
end
end
 
return row
return table.concat(lines, "\n")
end
end


-- Navbox title
-- Navbox title
function p.navbox_title(title, is_collapsible, name)
function p.navbox_title(title, is_collapsible, name)
local is_root_navbox = (is_root_navbox == nil and is_root_navbox or true) -- If not specified, default to TRUE
if not title then return "" end
local has_navbar = name ~= nil
local has_navbar = name ~= nil
local navbox_title = ''
if title ~= nil then
-- Assemble boilerplate lines separately
navbox_title = "|-\n"
local boilerplate_lines = {}
.. '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em;'
table.insert(boilerplate_lines, '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em; padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | ')
.. 'padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | '  
table.insert(boilerplate_lines, '<span style="display: inline-block; float: left; text-align: left; font-weight: normal; font-style: normal; min-width: 4em; padding: 0px; margin: 0px;">')
.. '<span style="display: inline-block; float: left; text-align: left; font-weight: normal; font-style: normal; min-width: 4em; padding: 0px; margin: 0px;">'
table.insert(boilerplate_lines, (has_navbar and navbar(name, "mini", "") or "") .. '</span>')
.. (has_navbar and navbar(name, "mini", "") or "") .. '</span>'
table.insert(boilerplate_lines, '<span style="font-size: 1.05em;">' .. title .. '</span>')
.. '<span style="font-size: 1.05em;">' .. title .. "</span>"
table.insert(boilerplate_lines, (is_collapsible and '' or '<span style="display: inline-block; float: right; font-size: 0.8em; width: 5em;">&nbsp;</span>'))
.. (is_collapsible and '' or '<span style="display: inline-block; float: right; font-size: 0.8em; width: 5em;">&nbsp;</span>')
 
.. '\n'
-- Then assemble lines, adding the boilerplate among those lines
end
local lines = {}
return navbox_title
table.insert(lines, "|-")
table.insert(lines, table.concat(boilerplate_lines, ""))
 
return table.concat(lines, "\n")
end
end


-- Navbox title for nested navboxes
-- Navbox title for nested navboxes
--[[
function p.nested_navbox_title(title, is_collapsible)
function p.nested_navbox_title(title, is_collapsible)
local navbox_title = ''
local navbox_title = ''
Line 82: Line 82:
end
end
return navbox_title
return navbox_title
end
]]--
function p.nested_navbox_title(title, is_collapsible)
if not title then return "" end
-- Assemble boilerplate lines separately
local boilerplate_lines = {}
table.insert(boilerplate_lines, '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em; padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | ')
if is_collapsible then table.insert(boilerplate_lines, '<span style="display: inline-block; float: left; min-width: 4em; padding: 0px; margin: 0px;">&nbsp;</span>') end
table.insert(boilerplate_lines, '<span style="font-size: 0.9em;">' .. title .. '</span>')
-- Then assemble lines, adding the boilerplate among those lines
local lines = {}
table.insert(lines, "|-")
table.insert(lines, table.concat(boilerplate_lines, ""))
return table.concat(lines, "\n")
end
end


Line 93: Line 111:
local is_collapsed    = yesno(args["Is Collapsed"  ], false)
local is_collapsed    = yesno(args["Is Collapsed"  ], false)
local navbox_type    = ((args["Navbox Type"] == nil) and "Normal" or args["Navbox Type"])
local navbox_type    = ((args["Navbox Type"] == nil) and "Normal" or args["Navbox Type"])
local header_row = args["Header Row"]
local header_row     = args["Header Row"]
local footer_row = args["Footer Row"]
local footer_row     = args["Footer Row"]
 
-- Start of table
-- Table that holds individual lines
local navbox = ''
local navbox = {}
 
if navbox_type == "Nested" then
if navbox_type == "Nested" then
-- Navbox has a title, wikitable border, collapse options, and extra
-- Build a navbox that lies within another navbox
-- margin.
table.insert(navbox, '<div class="wikitable" style="overflow-x: auto; padding: 0; margin: 2px;">')
-- This navbox is meant to be a sub-navbox, placed on a headerless row.
table.insert(navbox,
navbox = '<div class="wikitable" style="overflow-x: auto; padding: 0; margin: 2px;">\n'
'{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
.. '{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
.. (is_collapsible and (' class="mw-collapsible' .. (is_collapsed and ' mw-collapsed"' or '"')) or '')
.. (is_collapsible and ('class="mw-collapsible' .. (is_collapsed and ' mw-collapsed"' or '"')) or '') .. '\n'
)
.. p.nested_navbox_title(title, is_collapsible)
 
-- Title row
-- Add header
table.insert(navbox, p.nested_navbox_title(title, is_collapsible))
if header_row ~= nil then
navbox = navbox .. p.navbox_header_footer(header_row)
-- Header row
if header_row then
table.insert(navbox, p.navbox_header_footer(header_row))
end
end
 
-- Add rows
-- Data rows
for i = 1, #rows do
for i = 1, #rows do
navbox = navbox .. p.navbox_row(rows[i], is_data_navbox[i])
table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
end
end
 
-- Add footer
-- Footer rows
if footer_row ~= nil then
if footer_row then
navbox = navbox .. p.navbox_header_footer(footer_row)
table.insert(navbox, p.navbox_header_footer(footer_row))
end
end
 
-- End of table
-- End of table
navbox = navbox
table.insert(navbox, '|}')
.. '|}\n'
table.insert(navbox, '</div>')
.. '</div>'


elseif navbox_type == "Subheader" then
elseif navbox_type == "Subheader" then
-- Navbox has no title (by default), border, or collapse options.
-- Build a navbox that lies within another navbox
-- This navbox is meant to display subheaders. If placed on a headerless
-- This one serves as subrows for one row
-- row, the title row can serve as the header.
table.insert(navbox, '{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"')
navbox = '{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;\n'
.. p.nested_navbox_title(title, false)
-- Add rows
-- Title (optional)
table.insert(navbox, p.nested_navbox_title(title, false))
 
-- Rows
for i = 1, #rows do
for i = 1, #rows do
navbox = navbox .. p.navbox_row(rows[i], is_data_navbox[i])
table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
end
end
 
-- End of table
-- End of table
navbox = navbox .. '|}\n'
table.insert(navbox, '|}')
else
 
-- Navbox has a title, wikitable border, and collapse options.
else -- Normal
-- This navbox is a normal navbox or a nested navbox.
table.insert(navbox, '<div class="wikitable" style="overflow-x: auto; padding: 0">')
navbox = '<div class="wikitable" style="overflow-x: auto; padding: 0">\n'
table.insert(navbox,
.. '{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
'{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
.. (is_collapsible and ('class="mw-collapsible' .. (is_collapsed and 'mw-collapsed"' or '"')) or '') .. '\n'
.. (is_collapsible and (' class="mw-collapsible' .. (is_collapsed and ' mw-collapsed"' or '"')) or '')
.. p.navbox_title(title, is_collapsible, name)
)
 
-- Add header
-- Title
if header_row ~= nil then
table.insert(navbox, p.navbox_title(title, is_collapsible, name))
navbox = navbox .. p.navbox_header_footer(header_row)
 
-- Header
if header_row then
table.insert(navbox, p.navbox_header_footer(header_row))
end
end
 
-- Add rows
-- Rows
for i = 1, #rows do
for i = 1, #rows do
navbox = navbox .. p.navbox_row(rows[i], is_data_navbox[i])
table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
end
end
-- Add header
-- Footer
if footer_row ~= nil then
if footer_row then
navbox = navbox .. p.navbox_header_footer(footer_row)
table.insert(navbox, p.navbox_header_footer(footer_row))
end
end
 
-- End of table
-- End of table
navbox = navbox
table.insert(navbox, '|}')
.. '|}\n'
table.insert(navbox, '</div>')
.. '</div>'
end
end
 
return table.concat(navbox, '\n')
return navbox
end
end



Revision as of 20:27, 7 October 2025

Module documentation[view] [edit] [history] [purge]
This module implements a metatemplate, and may be invoked by templates using its corresponding template Template:Navbox, or used directly from other modules.

Module:Navbox is a module that implements the {{Navbox}} template. Navbox templates can be made by using the template or by calling the _navbox function from another module.

On templates, you can create a navbox by using {{Navbox}}, which calls this module's wrapper function.

On modules, you can include local navbox = require("Module:Navbox")._navbox to create a navbox.


Introspection summary for Module:Navbox 
Functions provided (6)
Line Function Params
12 navbox_header_footer (row_content)
21 navbox_row (row_content, is_navbox)
51 navbox_title (title, is_collapsible, name)
87 nested_navbox_title (title, is_collapsible)
105 _navbox (main) (args)
200 navbox (invokable) (frame)
Lua modules required (3)
Variable Module Functions used
getArgs Module:Arguments getArgs
navbar Module:Navbar _navbar
yesno Module:Yesno yesno

No function descriptions were provided. The Lua code may have further information.


-- Page is following provisonal style guide: User:Ganaram_inukshuk/Provisional_style_guide_for_Lua
local p = {}

local getArgs = require("Module:Arguments").getArgs
local navbar = require("Module:Navbar")._navbar
local yesno = require("Module:Yesno")

-- TODO: add option for dataless header row, to complement the existence of
-- a headerless data row.

-- Navbox header/footer row
function p.navbox_header_footer(row_content)
	local lines = {}
	table.insert(lines, "|-")
	table.insert(lines, '| colspan="2" style="font-size: 0.8em; text-align: center; background-color: #eaecf0; padding: 0em; border: 1px solid white;" | ' .. row_content)
	
	return table.concat(lines, "\n")
end

-- Navbox row
function p.navbox_row(row_content, is_navbox)
	local is_navbox = is_navbox or false
	local lines = {}
	table.insert(lines, "|-")
	
	-- TODO: add option for a dataless header row

	if #row_content == 1 then
		-- Headerless row; holds data for simple navboxes or a nested navbox
		if is_navbox then
			table.insert(lines, '| style="padding: 0;" colspan="2" |')
			table.insert(lines, row_content[1])
		else
			table.insert(lines, '| style="font-size: 0.9em; padding: 0.25em 0.5em;" colspan="2" | ' .. row_content[1])
		end
	else
		-- Header + Data row; typical usecase for a row
		table.insert(lines, '! style="white-space: nowrap; font-size: 0.9em; width: 5%; text-align: right; background-color: #eaecf0; padding: 0.25em 0.5em; border: 1px solid white;" |' .. row_content[1])
		if is_navbox then
			table.insert(lines, '| style="padding: 0;" |')
		else
			table.insert(lines, '| style="font-size: 0.9em; padding: 0.25em 0.5em;" | ')
		end
		table.insert(lines, row_content[2])
	end

	return table.concat(lines, "\n")
end

-- Navbox title
function p.navbox_title(title, is_collapsible, name)
	if not title then return "" end
	local has_navbar = name ~= nil
	
	-- Assemble boilerplate lines separately
	local boilerplate_lines = {}
	table.insert(boilerplate_lines, '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em; padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | ')
	table.insert(boilerplate_lines, '<span style="display: inline-block; float: left; text-align: left; font-weight: normal; font-style: normal; min-width: 4em; padding: 0px; margin: 0px;">')
	table.insert(boilerplate_lines, (has_navbar and navbar(name, "mini", "") or "") .. '</span>')
	table.insert(boilerplate_lines, '<span style="font-size: 1.05em;">' .. title .. '</span>')
	table.insert(boilerplate_lines, (is_collapsible and '' or '<span style="display: inline-block; float: right; font-size: 0.8em; width: 5em;">&nbsp;</span>'))

	-- Then assemble lines, adding the boilerplate among those lines
	local lines = {}
	table.insert(lines, "|-")
	table.insert(lines, table.concat(boilerplate_lines, ""))

	return table.concat(lines, "\n")
end

-- Navbox title for nested navboxes
--[[
function p.nested_navbox_title(title, is_collapsible)
	local navbox_title = ''
	if title ~= nil then
		navbox_title = "|-\n"
			.. '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em;'
			.. 'padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | ' 
			.. (is_collapsible and '<span style="display: inline-block; float: left; min-width: 4em; padding: 0px; margin: 0px;">&nbsp;</span>' or '')
			.. '<span style="font-size: 0.9em;">' .. title .. "</span>"
			.. '\n'
	end
	return navbox_title
end
]]--

function p.nested_navbox_title(title, is_collapsible)
	if not title then return "" end

	-- Assemble boilerplate lines separately
	local boilerplate_lines = {}
	table.insert(boilerplate_lines, '! style="text-align: center; background-color: #eaecf0; white-space: nowrap; margin: 0em 4em 0em 4em; padding: 0.25em 0.5em; border: 1px solid white;" colspan="2" | ')
	if is_collapsible then table.insert(boilerplate_lines, '<span style="display: inline-block; float: left; min-width: 4em; padding: 0px; margin: 0px;">&nbsp;</span>') end
	table.insert(boilerplate_lines, '<span style="font-size: 0.9em;">' .. title .. '</span>')
	
	-- Then assemble lines, adding the boilerplate among those lines
	local lines = {}
	table.insert(lines, "|-")
	table.insert(lines, table.concat(boilerplate_lines, ""))

	return table.concat(lines, "\n")
end

-- Navbox to be called by other modules; also called by wrapper function
function p._navbox(args)
	local title           = args["Title"]
	local name            = args["name"]
	local rows            = args["Rows"]
	local is_data_navbox  = args["Is Data Navbox"]
	local is_collapsible  = yesno(args["Is Collapsible"], true )
	local is_collapsed    = yesno(args["Is Collapsed"  ], false)
	local navbox_type     = ((args["Navbox Type"] == nil) and "Normal" or args["Navbox Type"])
	local header_row      = args["Header Row"]
	local footer_row      = args["Footer Row"]

	-- Table that holds individual lines
	local navbox = {}

	if navbox_type == "Nested" then
		-- Build a navbox that lies within another navbox
		table.insert(navbox, '<div class="wikitable" style="overflow-x: auto; padding: 0; margin: 2px;">')
		table.insert(navbox,
			'{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
			.. (is_collapsible and (' class="mw-collapsible' .. (is_collapsed and ' mw-collapsed"' or '"')) or '')
		)

		-- Title row
		table.insert(navbox, p.nested_navbox_title(title, is_collapsible))
		
		-- Header row
		if header_row then
			table.insert(navbox, p.navbox_header_footer(header_row))
		end

		-- Data rows
		for i = 1, #rows do
			table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
		end

		-- Footer rows
		if footer_row then
			table.insert(navbox, p.navbox_header_footer(footer_row))
		end

		-- End of table
		table.insert(navbox, '|}')
		table.insert(navbox, '</div>')

	elseif navbox_type == "Subheader" then
		-- Build a navbox that lies within another navbox
		-- This one serves as subrows for one row
		table.insert(navbox, '{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"')
		
		-- Title (optional)
		table.insert(navbox, p.nested_navbox_title(title, false))

		-- Rows
		for i = 1, #rows do
			table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
		end

		-- End of table
		table.insert(navbox, '|}')

	else -- Normal
		table.insert(navbox, '<div class="wikitable" style="overflow-x: auto; padding: 0">')
		table.insert(navbox,
			'{| style="mw-border-collapse: collapse; border-spacing: 0; margin: 0; width: 100%;"'
			.. (is_collapsible and (' class="mw-collapsible' .. (is_collapsed and ' mw-collapsed"' or '"')) or '')
		)

		-- Title
		table.insert(navbox, p.navbox_title(title, is_collapsible, name))

		-- Header
		if header_row then
			table.insert(navbox, p.navbox_header_footer(header_row))
		end

		-- Rows
		for i = 1, #rows do
			table.insert(navbox, p.navbox_row(rows[i], is_data_navbox[i]))
		end
		
		-- Footer
		if footer_row then
			table.insert(navbox, p.navbox_header_footer(footer_row))
		end

		-- End of table
		table.insert(navbox, '|}')
		table.insert(navbox, '</div>')
	end

	return table.concat(navbox, '\n')
end

-- Navbox to be invoked
-- Wrapper function for template-based navboxes
function p.navbox(frame)
	local args = getArgs(frame)
	
	-- Preprocess individual entries for, headers, data, and is-row-child into
	-- one single table.
	local rows = {}
	local is_data_navbox = {}
	for i = 1, 30 do
		local header = args["Header " .. i]
		local data   = args["Data "   .. i]
		local is_navbox = args["Is Data " .. i .. " Navbox"] ~= nil and args["Is Data " .. i .. " Navbox"] or false
		
		local row = nil
		if (header == nil and data ~= nil) then
			-- Headerless row
			row = {}
			table.insert(row, data)
		elseif (header ~= nil and data == nil) then
			-- Dataless row; permitted for placeholder purposes
			row = {}
			table.insert(row, header)
			table.insert(row, "")
		elseif (header ~= nil and data ~= nil) then
			-- Row with header and data
			row = {}
			table.insert(row, header)
			table.insert(row, data)
		end
		if row ~= nil then
			table.insert(rows, row)
			table.insert(is_data_navbox, is_navbox)
		end
		
		-- Remove original entries as cleanup
		args["Header " .. i] = nil
		args["Data "   .. i] = nil
		args["Is Data " .. i .. " Navbox"] = nil
	end
	args["Rows"] = rows
	args["Is Data Navbox"] = is_data_navbox
	
	local result = p._navbox(args)
	
	-- Debugger option
	local debugg = yesno(args["debug"])
	if debugg == true then
		result = "<syntaxhighlight lang=\"wikitext\">" .. result .. "</syntaxhighlight>"
	end
	
	return result
end

return p