Module:Navbox with collapsible groups: Difference between revisions

From WikiPasokh
m (1 revision imported)
(pseudo-ternary operator doesn't work with "and nil")
Line 1: Line 1:
-- This module implements {{Navbox with collapsible groups}}
-- This module implements {{Navbox with collapsible groups}}
local q = {}
 
local Navbox = require('Module:Navbox')
require('strict')
local p = {}
local parentCfg = mw.loadData('Module:Navbox/configuration')
local thisCfg = mw.loadData('Module:Navbox with collapsible groups/configuration')
local cfg = {}
for k, v in pairs(thisCfg) do
if type(v) == 'table' then
cfg[k] = {}
if type(parentCfg[k]) == 'table' then
for kk, vv in pairs(parentCfg[k]) do cfg[k][kk] = vv end
end
for kk, vv in pairs(v) do cfg[k][kk] = vv end
end
end
local inArray = require("Module:TableTools").inArray
local getArgs -- lazily initialized


-- helper functions
-- helper functions
local andnum = function(s, n) return string.format(cfg.arg[s..'_and_num'], n) end
local isblank = function(v) return (v or '') == '' end
local function concatstrings(s)
local function concatstrings(s)
local r = table.concat(s, '')
local r = table.concat(s, '')
if r:match('^%s*$') then r = nil end
if r:match('^%s*$') then return nil end
return r
return r
end
end
 
local function concatstyles(s)
local function concatstyles(s)
local r = table.concat(s, ';')
local r = ''
while r:match(';%s*;') do
for _, v in ipairs(s) do
r = mw.ustring.gsub(r, ';%s*;', ';')
v = mw.text.trim(v, "%s;")
if not isblank(v) then r = r .. v .. ';' end
end
end
if r:match('^%s*;%s*$') then r = nil end
if isblank(r) then return nil end
return r
return r
end
end
local function getSubgroup(args, listnum, listText)
local subArgs = {
[cfg.arg.border] = cfg.keyword.border_subgroup,
[cfg.arg.navbar] = cfg.keyword.navbar_plain
}
local hasSubArgs = false
for k, v in pairs(args) do
k = tostring(k)
for _, w in ipairs(cfg.keyword.subgroups) do
w = w .. listnum .. "_"
if (#k > #w) and (k:sub(1, #w) == w) then
subArgs[k:sub(#w + 1)] = v
hasSubArgs = true
end
end
end
return hasSubArgs and p._navbox(subArgs) or listText
end
-- Main functions
p._navbox = require('Module:Navbox')._navbox


function q._navbox(pargs)
function p._withCollapsibleGroups(pargs)
-- table for args passed to navbox
-- table for args passed to navbox
local targs = {}
local targs = {}
Line 25: Line 67:
-- process args
-- process args
local passthrough = {
local passthrough = {
['name']=true,['navbar']=true,['state']=true,['border']=true,
[cfg.arg.above]=true,[cfg.arg.aboveclass]=true,[cfg.arg.abovestyle]=true,
['bodyclass']=true,['groupclass']=true,['listclass']=true,
[cfg.arg.basestyle]=true,
['style']=true,['bodystyle']=true,['basestyle']=true,
[cfg.arg.below]=true,[cfg.arg.belowclass]=true,[cfg.arg.belowstyle]=true,
['title']=true,['titleclass']=true,['titlestyle']=true,
[cfg.arg.bodyclass]=true,[cfg.arg.bodystyle]=true,
['above']=true,['aboveclass']=true,['abovestyle']=true,
[cfg.arg.border]=true,
['below']=true,['belowclass']=true,['belowstyle']=true,
[cfg.arg.groupclass]=true,
['image']=true,['imageclass']=true,['imagestyle']=true,
[cfg.arg.image]=true,[cfg.arg.imageclass]=true,[cfg.arg.imagestyle]=true,
['imageleft']=true,['imageleftstyle']=true
[cfg.arg.imageleft]=true,[cfg.arg.imageleftstyle]=true,
[cfg.arg.listclass]=true,
[cfg.arg.name]=true,
[cfg.arg.navbar]=true,
[cfg.arg.state]=true,
[cfg.arg.style]=true,
[cfg.arg.title]=true,[cfg.arg.titleclass]=true,[cfg.arg.titlestyle]=true
}
}
for k,v in pairs(pargs) do
for k,v in pairs(pargs) do
Line 38: Line 86:
if passthrough[k] then
if passthrough[k] then
targs[k] = v
targs[k] = v
elseif (k:match('^list[0-9][0-9]*$')  
elseif (k:match(cfg.pattern.num)) then
or k:match('^content[0-9][0-9]*$') ) then
local n = k:match(cfg.pattern.num)
local n = mw.ustring.gsub(k, '^[a-z]*([0-9]*)$', '%1')
local list_and_num = andnum('list', n)
if (targs['list' .. n] == nil and pargs['group' .. n] == nil
if ((k:match(cfg.pattern.listnum) or k:match(cfg.pattern.contentnum))
and pargs['sect' .. n] == nil and pargs['section' .. n] == nil) then
and targs[list_and_num] == nil
targs['list' .. n] = concatstrings(
and pargs[andnum('group', n)] == nil
{pargs['list' .. n] or '', pargs['content' .. n] or ''})
and pargs[andnum('sect', n)] == nil
end
and pargs[andnum('section', n)] == nil) then
elseif (k:match('^group[0-9][0-9]*$')  
targs[list_and_num] = concatstrings({
or k:match('^sect[0-9][0-9]*$')  
pargs[list_and_num] or '',
or k:match('^section[0-9][0-9]*$') ) then
pargs[andnum('content', n)] or ''
local n = mw.ustring.gsub(k, '^[a-z]*([0-9]*)$', '%1')
})
if targs['list' .. n] == nil then
if (targs[list_and_num] and inArray(cfg.keyword.subgroups, targs[list_and_num])) then
local titlestyle = concatstyles(
targs[list_and_num] = getSubgroup(pargs, n, targs[list_and_num])
{pargs['groupstyle'] or '',pargs['secttitlestyle'] or '',  
end
pargs['group' .. n .. 'style'] or '',  
elseif ((k:match(cfg.pattern.groupnum) or k:match(cfg.pattern.sectnum) or k:match(cfg.pattern.sectionnum))
pargs['section' .. n ..'titlestyle'] or ''})
and targs[list_and_num] == nil) then
local liststyle = concatstyles(
local titlestyle = concatstyles({
{pargs['liststyle'] or '', pargs['contentstyle'] or '',  
pargs[cfg.arg.groupstyle] or '',
pargs['list' .. n .. 'style'] or '',  
pargs[cfg.arg.secttitlestyle] or '',  
pargs['content' .. n .. 'style'] or ''})
pargs[andnum('groupstyle', n)] or '',  
local title = concatstrings(
pargs[andnum('sectiontitlestyle', n)] or ''
{pargs['group' .. n] or '',  
})
pargs['sect' .. n] or '',
local liststyle = concatstyles({
pargs['section' .. n] or ''})
pargs[cfg.arg.liststyle] or '',
local list = concatstrings(
pargs[cfg.arg.contentstyle] or '',  
{pargs['list' .. n] or '',  
pargs[andnum('liststyle', n)] or '',  
pargs['content' .. n] or ''})
pargs[andnum('contentstyle', n)] or ''
local state = (pargs['abbr' .. n] and pargs['abbr' .. n] == pargs['selected'])  
})
and 'uncollapsed' or pargs['state' .. n] or 'collapsed'
local title = concatstrings({
pargs[andnum('group', n)] or '',
pargs[andnum('sect', n)] or '',
pargs[andnum('section', n)] or ''
})
local list = concatstrings({
pargs[list_and_num] or '',  
pargs[andnum('content', n)] or ''
})
if list and inArray(cfg.keyword.subgroups, list) then
list = getSubgroup(pargs, n, list)
end
local abbr_and_num = andnum('abbr', n)
local state = (pargs[abbr_and_num] and pargs[abbr_and_num] == pargs[cfg.arg.selected])  
and cfg.keyword.state_uncollapsed
or (pargs[andnum('state', n)] or cfg.keyword.state_collapsed)
targs['list' .. n] = Navbox._navbox(
targs[list_and_num] =p._navbox({
{'child', navbar = 'plain', state = state,
cfg.keyword.border_child,
basestyle = pargs['basestyle'],
[cfg.arg.navbar] = cfg.keyword.navbar_plain,
title = title, titlestyle = titlestyle,
[cfg.arg.state] = state,
list1 = list, liststyle = liststyle,
[cfg.arg.basestyle] = pargs[cfg.arg.basestyle],
listclass = pargs['list' .. n .. 'class'],
[cfg.arg.title] = title,
image = pargs['image' .. n],
[cfg.arg.titlestyle] = titlestyle,
imageleft = pargs['imageleft' .. n],
[andnum('list', 1)] = list,
listpadding = pargs['listpadding']})
[cfg.arg.liststyle] = liststyle,
[cfg.arg.listclass] = pargs[andnum('listclass', n)],
[cfg.arg.image] = pargs[andnum('image', n)],
[cfg.arg.imageleft] = pargs[andnum('imageleft', n)],
[cfg.arg.listpadding] = pargs[cfg.arg.listpadding]
})
end
end
end
end
Line 83: Line 151:
end
end
-- ordering of style and bodystyle
-- ordering of style and bodystyle
targs['style'] = concatstyles({targs['style'] or '', targs['bodystyle'] or ''})
targs[cfg.arg.style] = concatstyles({targs[cfg.arg.style] or '', targs[cfg.arg.bodystyle] or ''})
targs['bodystyle'] = nil
targs[cfg.arg.bodystyle] = nil
-- child or subgroup
-- child or subgroup
if targs['border'] == nil then targs['border'] = pargs[1] end
if targs[cfg.arg.border] == nil then targs[cfg.arg.border] = pargs[1] end


return Navbox._navbox(targs)
return p._navbox(targs)
end
end


function q.navbox(frame)
-- Template entry points
local pargs = require('Module:Arguments').getArgs(frame, {wrappers = {'Template:Navbox with collapsible groups'}})
function p.navbox (frame, boxtype)
local function readArgs(args, prefix)
-- Read the arguments in the order they'll be output in, to make references
-- number in the right order.
local _
_ = args[prefix .. cfg.arg.title]
_ = args[prefix .. 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[prefix .. andnum('group', i)]
if inArray(cfg.keyword.subgroups, args[prefix .. andnum('list', i)]) then
for _, v in ipairs(cfg.keyword.subgroups) do
readArgs(args, prefix .. v .. i .. "_")
end
end
end
_ = args[prefix .. cfg.arg.below]
end


-- Read the arguments in the order they'll be output in, to make references number in the right order.
if not getArgs then
local _
getArgs = require('Module:Arguments').getArgs
_ = pargs.title
_ = pargs.above
for i = 1, 20 do
_ = pargs["group" .. tostring(i)]
_ = pargs["list" .. tostring(i)]
end
end
_ = pargs.below
local args = getArgs(frame, {wrappers = {cfg.pattern[boxtype or 'navbox']}})
readArgs(args, "")
return p['_'..(boxtype or 'navbox')](args)
end


return q._navbox(pargs)
p['with collapsible groups'] = function (frame)
return p.navbox(frame, 'withCollapsibleGroups')
end
end


local q = {}
q._navbox = p._withCollapsibleGroups
q.navbox = p['with collapsible groups']
return q
return q

Revision as of 23:36, 17 September 2024

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

-- This module implements {{Navbox with collapsible groups}}

require('strict')
local p = {}
local parentCfg = mw.loadData('Module:Navbox/configuration')
local thisCfg = mw.loadData('Module:Navbox with collapsible groups/configuration')
local cfg = {}
for k, v in pairs(thisCfg) do
	if type(v) == 'table' then
		cfg[k] = {}
		if type(parentCfg[k]) == 'table' then
			for kk, vv in pairs(parentCfg[k]) do cfg[k][kk] = vv end
		end
		for kk, vv in pairs(v) do cfg[k][kk] = vv end
	end
end
local inArray = require("Module:TableTools").inArray
local getArgs -- lazily initialized

-- helper functions
local andnum = function(s, n) return string.format(cfg.arg[s..'_and_num'], n) end
	
local isblank = function(v) return (v or '') == '' end

local function concatstrings(s)
	local r = table.concat(s, '')
	if r:match('^%s*$') then return nil end
	return r
end
	
local function concatstyles(s)
	local r = ''
	for _, v in ipairs(s) do
		v = mw.text.trim(v, "%s;")
		if not isblank(v) then r = r .. v .. ';' end
	end
	if isblank(r) then return nil end
	return r
end
	
local function getSubgroup(args, listnum, listText)
	local subArgs = {
		[cfg.arg.border] = cfg.keyword.border_subgroup,
		[cfg.arg.navbar] = cfg.keyword.navbar_plain
	}
	local hasSubArgs = false
	for k, v in pairs(args) do
		k = tostring(k)
		for _, w in ipairs(cfg.keyword.subgroups) do
			w = w .. listnum .. "_"
			if (#k > #w) and (k:sub(1, #w) == w) then
				subArgs[k:sub(#w + 1)] = v
				hasSubArgs = true
			end
		end
	end
	return hasSubArgs and p._navbox(subArgs) or listText
end

-- Main functions
p._navbox = require('Module:Navbox')._navbox

function p._withCollapsibleGroups(pargs)
	-- table for args passed to navbox
	local targs = {}

	-- process args
	local passthrough = {
		[cfg.arg.above]=true,[cfg.arg.aboveclass]=true,[cfg.arg.abovestyle]=true,
		[cfg.arg.basestyle]=true,
		[cfg.arg.below]=true,[cfg.arg.belowclass]=true,[cfg.arg.belowstyle]=true,
		[cfg.arg.bodyclass]=true,[cfg.arg.bodystyle]=true,
		[cfg.arg.border]=true,
		[cfg.arg.groupclass]=true,
		[cfg.arg.image]=true,[cfg.arg.imageclass]=true,[cfg.arg.imagestyle]=true,
		[cfg.arg.imageleft]=true,[cfg.arg.imageleftstyle]=true,
		[cfg.arg.listclass]=true,
		[cfg.arg.name]=true,
		[cfg.arg.navbar]=true,
		[cfg.arg.state]=true,
		[cfg.arg.style]=true,
		[cfg.arg.title]=true,[cfg.arg.titleclass]=true,[cfg.arg.titlestyle]=true
	}
	for k,v in pairs(pargs) do
		if k and type(k) == 'string' then
			if passthrough[k] then
				targs[k] = v
			elseif (k:match(cfg.pattern.num)) then
				local n = k:match(cfg.pattern.num)
				local list_and_num = andnum('list', n)
				if ((k:match(cfg.pattern.listnum) or k:match(cfg.pattern.contentnum))
						and targs[list_and_num] == nil
						and pargs[andnum('group', n)] == nil
						and pargs[andnum('sect', n)] == nil
						and pargs[andnum('section', n)] == nil) then
					targs[list_and_num] = concatstrings({
						pargs[list_and_num] or '',
						pargs[andnum('content', n)] or ''
					})
					if (targs[list_and_num] and inArray(cfg.keyword.subgroups, targs[list_and_num])) then
						targs[list_and_num] = getSubgroup(pargs, n, targs[list_and_num])
					end
				elseif ((k:match(cfg.pattern.groupnum) or k:match(cfg.pattern.sectnum) or k:match(cfg.pattern.sectionnum))
						and targs[list_and_num] == nil) then
					local titlestyle = concatstyles({
						pargs[cfg.arg.groupstyle] or '',
						pargs[cfg.arg.secttitlestyle] or '', 
						pargs[andnum('groupstyle', n)] or '', 
						pargs[andnum('sectiontitlestyle', n)] or ''
					})
					local liststyle = concatstyles({
						pargs[cfg.arg.liststyle] or '',
						pargs[cfg.arg.contentstyle] or '', 
						pargs[andnum('liststyle', n)] or '', 
						pargs[andnum('contentstyle', n)] or ''
					})
					local title = concatstrings({
						pargs[andnum('group', n)] or '',
						pargs[andnum('sect', n)] or '',
						pargs[andnum('section', n)] or ''
					})
					local list = concatstrings({
						pargs[list_and_num] or '', 
						pargs[andnum('content', n)] or ''
					})
					if list and inArray(cfg.keyword.subgroups, list) then
						list = getSubgroup(pargs, n, list)
					end
					local abbr_and_num = andnum('abbr', n)
					local state = (pargs[abbr_and_num] and pargs[abbr_and_num] == pargs[cfg.arg.selected]) 
						and cfg.keyword.state_uncollapsed
						or (pargs[andnum('state', n)] or cfg.keyword.state_collapsed)
					
					targs[list_and_num] =p._navbox({
						cfg.keyword.border_child,
						[cfg.arg.navbar] = cfg.keyword.navbar_plain,
						[cfg.arg.state] = state,
						[cfg.arg.basestyle] = pargs[cfg.arg.basestyle],
						[cfg.arg.title] = title,
						[cfg.arg.titlestyle] = titlestyle,
						[andnum('list', 1)] = list,
						[cfg.arg.liststyle] = liststyle,
						[cfg.arg.listclass] = pargs[andnum('listclass', n)],
						[cfg.arg.image] = pargs[andnum('image', n)],
						[cfg.arg.imageleft] = pargs[andnum('imageleft', n)],
						[cfg.arg.listpadding] = pargs[cfg.arg.listpadding]
					})
				end
			end
		end
	end
	-- ordering of style and bodystyle
	targs[cfg.arg.style] = concatstyles({targs[cfg.arg.style] or '', targs[cfg.arg.bodystyle] or ''})
	targs[cfg.arg.bodystyle] = nil
	
	-- child or subgroup
	if targs[cfg.arg.border] == nil then targs[cfg.arg.border] = pargs[1] end

	return p._navbox(targs)
end

-- Template entry points
function p.navbox (frame, boxtype)
	local function readArgs(args, prefix)
		-- Read the arguments in the order they'll be output in, to make references
		-- number in the right order.
		local _
		_ = args[prefix .. cfg.arg.title]
		_ = args[prefix .. 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[prefix .. andnum('group', i)]
			if inArray(cfg.keyword.subgroups, args[prefix .. andnum('list', i)]) then
				for _, v in ipairs(cfg.keyword.subgroups) do
					readArgs(args, prefix .. v .. i .. "_")
				end
			end
		end
		_ = args[prefix .. cfg.arg.below]
	end

	if not getArgs then
		getArgs = require('Module:Arguments').getArgs
	end
	local args = getArgs(frame, {wrappers = {cfg.pattern[boxtype or 'navbox']}})
	readArgs(args, "")
	return p['_'..(boxtype or 'navbox')](args)
end

p['with collapsible groups'] = function (frame)
	return p.navbox(frame, 'withCollapsibleGroups')
end

local q = {}
q._navbox = p._withCollapsibleGroups
q.navbox = p['with collapsible groups']
return q