This module creates a list with automatically balanced columns. It should not be used directly in entries, but in templates such as {{col2}}
or {{col3}}
. List entries are given as parameters to the template.
{{#invoke:columns|display|sort=1|collapse=1|columns=3}} -> {{col3|en|z|y|x|w|v|u|t}}
{{#invoke:columns|display|sort=1|collapse=1|columns=2}} -> {{col2|nl|a|b|c|d|e|f|g}}
export.create_list {
column_count = number,
content = list, alphabetize = boolean,
background_color = string, collapse = boolean,
toggle_category = string,
class = string, lang = language_object,
}
content
{ "term1", "term2", "term3", ... }
.lang
collapse
column_count
sort
toggle_category
class
div
tag that contains the list. Defaults to derivedterms
.background_color
The old name for the main function. It is now just a wrapper for create_list
.
The template-invokable function.
local export = {}
local html = mw.html.create
local m_links = require("Module:links")
local m_languages = require("Module:languages")
local m_table = require("Module:table")
local function format_list_items(list, args)
local function term_already_linked(term)
-- FIXME: "<span" is an ugly hack to prevent double-linking of terms already run through {{l|...}}:
-- ]
return term:find("<span")
end
for _, item in ipairs(args.content) do
if item == false then
-- omitted item; do nothing
else
if type(item) == "table" then
local link = term_already_linked(item.term.term) and item.term.term or m_links.full_link(item.term)
if item.q then
link = require("Module:qualifier").format_qualifier(item.q) .. " " .. link
end
if item.qq then
link = link .. " " .. require("Module:qualifier").format_qualifier(item.qq)
end
item = link
elseif args.lang and not term_already_linked(item) then
item = m_links.full_link {lang = args.lang, term = item, sc = args.sc}
end
list = list:node(html("li")
:wikitext(item)
)
end
end
return list
end
function export.create_list(args)
-- Fields in args that are used:
-- args.column_count, args.content, args.alphabetize, args.background_color,
-- args.collapse, args.toggle_category, args.class, args.lang
-- Check for required fields?
if type(args) ~= "table" then
error("expected table, got " .. type(args))
end
local class = args.class or "derivedterms"
local column_count = args.column_count or 1
local toggle_category = args.toggle_category or "derived terms"
local header = args.header
if header and args.format_header then
header = html("div")
:addClass("term-list-header")
:wikitext(header)
end
if args.alphabetize then
local function keyfunc(item)
if item == false then
item = "*" -- doesn't matter, will be omitted in format_list_items()
elseif type(item) == "table" then
item = item.term.term
end
return item
end
require("Module:collation").sort(args.content, args.lang, keyfunc)
end
local list = html("ul")
list = format_list_items(list, args)
local output = html("div")
:addClass(class)
:addClass("term-list")
:addClass("ul-column-count")
:attr("data-column-count", column_count)
:css("background-color", args.background_color)
:wikitext(mw.getCurrentFrame():extensionTag('templatestyles', nil, {src = 'Modul:columns/styles.css'}))
:node(list)
if args.collapse then
local nbsp = mw.ustring.char(0xA0)
output = html("div")
:node(output)
:addClass("list-switcher")
:attr("data-toggle-category", toggle_category)
:node(html("div")
:addClass("list-switcher-element")
:attr("data-showtext", nbsp .. "show more ▼" .. nbsp)
:attr("data-hidetext", nbsp .. "show less ▲" .. nbsp)
:css("display", "none")
:wikitext(nbsp)
)
end
return tostring(header or "") .. tostring(output)
end
-- This function is for compatibility with earlier version of ]
-- (now found in ]).
function export.create_table(...)
-- Earlier arguments to create_table:
-- n_columns, content, alphabetize, bg, collapse, class, title, column_width, line_start, lang
local args = {}
args.column_count, args.content, args.alphabetize, args.background_color,
args.collapse, args.class, args.header, args.column_width,
args.line_start, args.lang = ...
args.format_header = true
return export.create_list(args)
end
local param_mods = {"t", "alt", "tr", "ts", "pos", "lit", "id", "sc", "g", "q", "qq"}
local param_mod_set = m_table.listToSet(param_mods)
function export.display_from(frame_args, parent_args, frame)
local iparams = {
= {},
-- Default for auto-collapse. Overridable by template |collapse= param.
= {type = "boolean"},
-- If specified, this specifies the number of columns, and no columns
-- parameter is available on the template. Otherwise, the columns
-- parameter is the first available numbered param after the language-code
-- parameter.
= {type = "number"},
-- If specified, this specifies the language code, and no language-code
-- parameter is available on the template. Otherwise, the language-code
-- parameter can be specified as either |lang= or |1=.
= {},
-- Default for auto-sort. Overridable by template |sort= param.
= {type = "boolean"},
-- The following is accepted but currently ignored, per an extended discussion in
-- ].
= {default = ""},
= {},
}
local iargs = require("Module:parameters").process(frame_args, iparams, nil, "columns", "display_from")
local compat = iargs or parent_args
local lang_param = compat and "lang" or 1
local columns_param, first_content_param
-- New-style #columns specification is through parameter n= so we can transition to the situation where
-- omitting it results in auto-determination. Old-style #columns specification is through the first numbered
-- parameter after the lang parameter.
if parent_args then
columns_param = "n"
first_content_param = compat and 1 or 2
else
columns_param = compat and 1 or 2
first_content_param = columns_param + (iargs and 0 or 1)
end
local deprecated
local params = {
= not iargs and {required = true, default = "und"} or nil,
= not iargs and {required = true, default = 2} or nil,
= {list = true},
= {},
= {type = "boolean"},
= {type = "boolean"},
= {},
= {list = true}, -- used when calling from ] so the page displaying the synonyms/antonyms doesn't occur in the list
}
if lang_param == "lang" then
deprecated = true
end
local args = require("Module:parameters").process(parent_args, params, nil, "columns", "display_from")
local langcode = iargs or args
local lang = m_languages.getByCode(langcode, lang_param)
local sc = args and require("Module:scripts").getByCode(sc, "sc") or nil
local sort = iargs
if args ~= nil then
sort = args
end
local collapse = iargs
if args ~= nil then
collapse = args
end
local put
for i, item in ipairs(args) do
-- Parse off an initial language code (e.g. 'la:minūtia' or 'grc:]'). Don't parse if there's a spac
-- after the colon (happens e.g. if the user uses {{desc|...}} inside of {{col}}, grrr ...).
local termlangcode, actual_term = item:match("^(+):(.*)$")
local termlang
-- Make sure that only real language codes are handled as language links, so as to not catch interwiki
-- or namespaces links.
if termlangcode and (
mw.loadData("Module:languages/code to canonical name") or
mw.loadData("Module:etymology languages/code to canonical name")
) then
-- -1 since i is one-based
termlang = m_languages.getByCode(termlangcode, first_content_param + i - 1, "allow etym")
item = actual_term
else
termlang = lang
termlangcode = nil
end
local termobj = {term = {lang = termlang, sc = sc}}
-- Check for inline modifier, e.g. מרים<tr:Miryem>. But exclude HTML entry with <span ...>, <i ...>, <br/> or
-- similar in it, caused by wrapping an argument in {{l|...}}, {{af|...}} or similar. Basically, all tags of
-- the sort we parse here should consist of a less-than sign, plus letters, plus a colon, e.g. <tr:...>, so if
-- we see a tag on the outer level that isn't in this format, we don't try to parse it. The restriction to the
-- outer level is to allow generated HTML inside of e.g. qualifier tags, such as foo<q:similar to {{m|fr|bar}}>.
if item:find("<") and not item:find("^*<*") then
if not put then
put = require("Module:parse utilities")
end
local run = put.parse_balanced_segment_run(item, "<", ">")
local orig_param = first_content_param + i - 1
local function parse_err(msg)
error(msg .. ": " .. orig_param .. "= " .. table.concat(run))
end
termobj.term.term = run
for j = 2, #run - 1, 2 do
if run ~= "" then
parse_err("Extraneous text '" .. run .. "' after modifier")
end
local modtext = run:match("^<(.*)>$")
if not modtext then
parse_err("Internal error: Modifier '" .. modtext .. "' isn't surrounded by angle brackets")
end
local prefix, arg = modtext:match("^(+):(.*)$")
if not prefix then
parse_err("Modifier " .. run .. " lacks a prefix, should begin with one of '" ..
table.concat(param_mods, ":', '") .. ":'")
end
if param_mod_set then
local obj_to_set
if prefix == "q" or prefix == "qq" then
obj_to_set = termobj
else
obj_to_set = termobj.term
end
if prefix == "t" then
prefix = "gloss"
elseif prefix == "g" then
prefix = "genders"
arg = mw.text.split(arg, ",")
elseif prefix == "sc" then
arg = require("Module:scripts").getByCode(arg, orig_param .. ":sc")
end
if obj_to_set then
parse_err("Modifier '" .. prefix .. "' occurs twice, second occurrence " .. run)
end
obj_to_set = arg
else
parse_err("Unrecognized prefix '" .. prefix .. "' in modifier " .. run)
end
end
else
termobj.term.term = item
end
-- If a separate language code was given for the term, display the language name as a right qualifier.
-- Otherwise it may not be obvious that the term is in a separate language (e.g. if the main language is 'zh'
-- and the term language is a Chinese lect such as Min Nan). But don't do this for Translingual terms, which
-- are often added to the list of English and other-language terms.
if termlangcode and termlangcode ~= langcode and termlangcode ~= "mul" then
termobj.qq = {termlang:getCanonicalName(), termobj.qq}
end
local omitted = false
for _, omitted_item in ipairs(args.omit) do
if omitted_item == termobj.term.term then
omitted = true
break
end
end
if omitted then
-- signal create_list() to omit this item
args = false
else
args = termobj
end
end
local ret = export.create_list { column_count = iargs or args,
content = args,
alphabetize = sort,
header = args, background_color = "#F8F8FF",
collapse = collapse,
toggle_category = iargs,
class = iargs, lang = lang, sc = sc, format_header = true }
return deprecated and frame:expandTemplate{title = "check deprecated lang param usage", args = {ret, lang = args}} or ret
end
function export.display(frame)
return export.display_from(frame.args, frame:getParent().args, frame)
end
return export