Модуль:Message box

Из Википедии, бесплатной энциклопедии

Документация
Меташаблоны сообщений
  • {{ambox}} — статьи
  • {{cmbox}} — категории
  • {{imbox}} — файлы
  • {{tmbox}} — страницы обсуждений
  • {{fmbox}} — системные сообщения
  • {{ombox}} — другие страницы

Этот модуль предназначен для создания шаблонов-сообщений ({{mbox}}, {{ambox}}, {{cmbox}}, {{fmbox}}, {{imbox}}, {{ombox}} и {{tmbox}}), также он может использоваться другими модулями. Этот модуль не должен вызываться напрямую из статей, для этого используйте один из вышеперечисленных шаблонов-сообщений.

Использование

Чтобы использовать данный модуль в другом модуле, нужно для начала загрузить его:

local messageBox = require('Module:Message box') 

Чтобы создать шаблон-сообщение, используйте функцию main. У неё есть 2 параметра: первый — тип шаблона, второй — таблица с параметрами шаблона.

local box = messageBox.main( boxType, {     param1 = param1,     param2 = param2,     -- Еще параметры... }) 

Существует 7 вариантов оформления в зависимости от пространства имен, в котором располагается шаблон:

Функция Шаблон Тип страницы
mbox {{mbox}} определяется автоматически
ambox {{ambox}} статьи
cmbox {{cmbox}} категории
fmbox {{fmbox}} страницы интерфейса
imbox {{imbox}} файлы
tmbox {{tmbox}} страницы обсуждения
ombox {{ombox}} другие пространства имен

Подробнее о дополнительных параметрах шаблонов см. их документации.

Вызов через #invoke

Помимо функции main, в модуле есть отдельные функции для каждого типа сообщений. Они вызываются кодом {{#invoke:Message box|mbox|...}}, {{#invoke:Message box|ambox|...}} и т. д. Также эти функции могут быть вызваны другими модулями, однако это менее эффективно, чем вызов функции main.

Технические детали

Все перечисленные выше шаблоны основаны на одном базовом коде, а различия между каждым из них настраиваются на отдельной странице. Ниже перечислены различные параметры конфигурации:

  • types — таблица, содержащая данные, используемые типами сообщения. Ключи таблицы — это значения, которые можно передать определенному типу сообщения, а значения таблицы — это таблицы, содержащие класс и изображение, используемые этим типом.
  • default — тип, используемый, если параметру типа не было передано значение или если было указано недопустимое значение.
  • showInvalidTypeError — показывать ли ошибку, если значение, переданное параметру типа, было недопустимым.
  • allowBlankParams — обычно пустые значения удаляются из параметров, передаваемых модулю. Однако пробел сохраняется для параметров, включенных в таблицу allowBlankParams.
  • allowSmall — можно ли создать уменьшенную версию сообщения с помощью |small = yes.
  • smallParam — пользовательское имя для малого параметра. Например, если установлено значение left, вы можете создать малое сообщение, используя |small = left.
  • smallClass — класс, используемый для малых сообщений.
  • substCheck — выполнять проверку подстановки или нет.
  • classes — массив классов для использования в сообщении.
  • imageEmptyCell — использовать ли пустую ячейку <td></td>, если нет набора изображений. Это используется для сохранения интервалов между окнами сообщений шириной менее 100 % экрана.
  • imageEmptyCellStyle — следует ли стилизовать пустые ячейки изображения.
  • imageCheckBlank — приводит ли |image = blank к отсутствию изображения.
  • imageSmallSize — обычно изображения, используемые в малых сообщениях, имеют размер 30x30px. Этот параметр устанавливает пользовательский размер.
  • imageCellDiv — следует ли заключать изображение в div, обеспечивая максимальный размер изображения.
  • useCollapsibleTextFields — использовать ли текстовые поля, которые можно свернуть, то есть |issue =, |fix =, |talk = и т. д. В настоящее время используется только в ambox.
  • imageRightNone — если |imageright = none приводит к тому, что изображение не отображается в правой части окна сообщения.
  • allowMainspaceCategories — разрешить категоризацию в основном пространстве имен.
  • templateCategory — название категории для размещения на странице шаблона.
  • templateCategoryRequireName — требуется ли параметр |name = для отображения категории шаблона.
  • templateErrorCategory — имя категории ошибок, которая будет использоваться на странице шаблона.
  • templateErrorParamsToCheck — массив имен параметров для проверки. Если таковые отсутствуют, к странице шаблона применяется templateErrorCategory.
-- This is a meta-module for producing message box templates, including -- {{mbox}}, {{ambox}}, {{imbox}}, {{tmbox}}, {{ombox}}, {{cmbox}} and {{fmbox}}.  -- Load necessary modules. require('strict') local getArgs local categoryHandler = require('Module:Category handler')._main local yesno = require('Module:Yesno') local boxDate = require('Module:Calendar').bxDate;  -- Get a language object for formatDate and ucfirst. local lang = mw.language.getContentLanguage()  -- Define constants local CONFIG_MODULE = 'Module:Message box/configuration'  -------------------------------------------------------------------------------- -- Helper functions --------------------------------------------------------------------------------  local function getTitleObject(...) 	-- Get the title object, passing the function through pcall 	-- in case we are over the expensive function count limit. 	local success, title = pcall(mw.title.new, ...) 	if success then 		return title 	end end  local function union(t1, t2) 	-- Returns the union of two arrays. 	local vals = {} 	for i, v in ipairs(t1) do 		vals[v] = true 	end 	for i, v in ipairs(t2) do 		vals[v] = true 	end 	local ret = {} 	for k in pairs(vals) do 		table.insert(ret, k) 	end 	table.sort(ret) 	return ret end  local function getArgNums(args, prefix) 	local nums = {} 	for k, v in pairs(args) do 		local num = mw.ustring.match(tostring(k), '^' .. prefix .. '([1-9]%d*)$') 		if num then 			table.insert(nums, tonumber(num)) 		end 	end 	table.sort(nums) 	return nums end  -- локальная обёртка, игнорирует таблицу с номерами дня, месяца и года local function formatDate(txtDateIn, strFormat, params) 	local txtDateOut, date, status = boxDate(txtDateIn, strFormat, params) 	if status.brk then 		return error(status.errorText) 	else 		return txtDateOut 	end end  -------------------------------------------------------------------------------- -- Box class definition --------------------------------------------------------------------------------  local MessageBox = {} MessageBox.__index = MessageBox  function MessageBox.new(boxType, args, cfg) 	args = args or {} 	local obj = {}  	-- Set the title object and the namespace. 	obj.title = getTitleObject(args.page) or mw.title.getCurrentTitle()  	-- Set the config for our box type. 	obj.cfg = cfg[boxType] 	if not obj.cfg then 		local ns = obj.title.namespace 		-- boxType is "mbox" or invalid input 		if ns == 0 then 			obj.cfg = cfg.ambox -- main namespace 		elseif ns == 6 then 			obj.cfg = cfg.imbox -- file namespace 		elseif ns == 14 then 			obj.cfg = cfg.cmbox -- category namespace 		else 			local nsTable = mw.site.namespaces[ns] 			if nsTable and nsTable.isTalk then 				obj.cfg = cfg.tmbox -- any talk namespace 			else 				obj.cfg = cfg.ombox -- other namespaces or invalid input 			end 		end 	end  	-- Set the arguments, and remove all blank arguments except for the ones 	-- listed in cfg.allowBlankParams. 	do 		local newArgs = {} 		for k, v in pairs(args) do 			if v ~= '' then 				newArgs[k] = v 			end 		end 		for i, param in ipairs(obj.cfg.allowBlankParams or {}) do 			newArgs[param] = args[param] 		end 		obj.args = newArgs 	end  	-- Define internal data structure. 	obj.categories = {} 	obj.classes = {}  	return setmetatable(obj, MessageBox) end  function MessageBox:addCat(ns, cat, sort) 	if not cat then 		return nil 	end 	if sort then 		cat = string.format('[[Категория:%s|%s]]', cat, sort) 	else 		cat = string.format('[[Категория:%s]]', cat) 	end 	self.categories[ns] = self.categories[ns] or {} 	table.insert(self.categories[ns], cat) end  function MessageBox:addClass(class) 	if not class then 		return nil 	end 	table.insert(self.classes, class) end  function MessageBox:setParameters() 	local args = self.args 	local cfg = self.cfg  	-- Get type data. 	self.type = args.type 	local typeData = cfg.types[self.type] 	self.invalidTypeError = cfg.showInvalidTypeError 		and self.type 		and not typeData 	typeData = typeData or cfg.types[cfg.default] 	self.typeClass = typeData.class 	self.typeImage = typeData.image  	-- Find if the box has been wrongly substituted. 	self.isSubstituted = cfg.substCheck and args.subst == 'SUBST'  	-- Find whether we are using a small message box. 	self.isSmall = cfg.allowSmall and ( 		cfg.smallParam and args.small == cfg.smallParam 		or not cfg.smallParam and yesno(args.small) 	)  	-- Add attributes, classes and styles. 	self.id = args.id 	self.name = args.name 	if self.name then 		self:addClass('mbox-' .. string.gsub(self.name,' ','_')) 	end 	if yesno(args.plainlinks) ~= false then 		self:addClass('plainlinks') 	end 	for _, class in ipairs(cfg.classes or {}) do 		self:addClass(class) 	end 	if self.isSmall then 		self:addClass(cfg.smallClass or 'mbox-small') 	end 	self:addClass(self.typeClass) 	self:addClass(args.class) 	self.style = args.style 	self.attrs = args.attrs  	self.dataLabel1 = args['data-label-1'] 	self.dataLabel2 = args['data-label-2'] 	self.dataLabel3 = args['data-label-3'] 	self.dataValue1 = args['data-value-1'] 	self.dataValue2 = args['data-value-2'] 	self.dataValue3 = args['data-value-3'] 	 	-- Set text style. 	self.textstyle = args.textstyle  	-- Find if we are on the template page or not. This functionality is only 	-- used if useCollapsibleTextFields is set, or if both cfg.templateCategory 	-- and cfg.templateCategoryRequireName are set. 	self.useCollapsibleTextFields = cfg.useCollapsibleTextFields 	if self.useCollapsibleTextFields 		or cfg.templateCategory 		and cfg.templateCategoryRequireName 	then 		self.name = args.name 		if self.name then 			local templateName = mw.ustring.match( 				self.name, 				'^[tT][eE][mM][pP][lL][aA][tT][eE][%s_]*:[%s_]*(.*)$' 			) or self.name 			templateName = ('Template:' .. templateName) or ('Шаблон:' .. templateName) 			self.templateTitle = getTitleObject(templateName) 		end 		self.isTemplatePage = self.templateTitle 			and mw.title.equals(self.title, self.templateTitle) 	end  	-- Process data for collapsible text fields. At the moment these are only 	-- used in {{ambox}}. 	if self.useCollapsibleTextFields then 		-- Get the self.issue value. 		if self.isSmall and args.smalltext then 			self.issue = args.smalltext 		else 			if args.sect and args.sect ~= '' or nil then 				local issue_sect = args.issue_sect 				issue_sect = type(issue_sect) == 'string' and issue_sect ~= '' and issue_sect or nil 				local text_sect = args.text_sect 				text_sect = type(text_sect) == 'string' and text_sect ~= '' and text_sect or nil 				local issues = {} 				table.insert(issues, issue_sect) 				table.insert(issues, text_sect) 				self.issue = table.concat(issues, ' ') 			else 				local issue = args.issue 				issue = type(issue) == 'string' and issue ~= '' and issue or nil 				local text = args.text 				text = type(text) == 'string' and text or nil 				local issues = {} 				table.insert(issues, issue) 				table.insert(issues, text) 				self.issue = table.concat(issues, ' ') 			end 		end  		-- Get the self.talk value. 		local talk = args.talk 		-- Show talk links on the template page or template subpages if the talk 		-- parameter is blank. 		if talk == '' 			and self.templateTitle 			and ( 				mw.title.equals(self.templateTitle, self.title) 				or self.title:isSubpageOf(self.templateTitle) 			) 		then 			talk = '#' 		elseif talk == '' then 			talk = nil 		end 		if talk then 			-- If the talk value is a talk page, make a link to that page. Else 			-- assume that it's a section heading, and make a link to the talk 			-- page of the current page with that section heading. 			local talkTitle = getTitleObject(talk) 			local talkArgIsTalkPage = true 			if not talkTitle or not talkTitle.isTalkPage then 				talkArgIsTalkPage = false 				talkTitle = getTitleObject( 					self.title.text, 					mw.site.namespaces[self.title.namespace].talk.id 				) 			end 			if talkTitle and talkTitle.exists then 				local talkText = 'Соответствующую дискуссию можно найти на' 				if talkArgIsTalkPage then 					talkText = string.format( 						'%s [[%s|%s]].', 						talkText, 						talk, 						talkTitle.prefixedText 					) 				else 					talkText = string.format( 						'%s [[%s#%s|странице обсуждения]].', 						talkText, 						talkTitle.prefixedText, 						talk 					) 				end 				self.talk = talkText 			end 		end  		-- Get other values. 		self.fix = args.fix ~= '' and args.fix or nil 		local date 		if args.date and args.date ~= '' then 			local status, result = pcall(formatDate, args.date) 			if status then 				date = string.format("(<span class='date'>%s</span>)", result) 			else  				date = string.format("<span class='error'>(Строка «%s» не является верной датой, пожалуйста, укажите дату в формате <code>ГГГГ-ММ-ДД</code>)</span>", args.date) 			end  		elseif args.date == '' and self.isTemplatePage then 			date = string.format("(<span class='date'>%s</span>)", formatDate( lang:formatDate('Y-m-d') ) ) -- тут возникновения ошибки, связанной с пользовательским вводом, не будет 		end 		if date then 			self.date = string.format(" <span class='mbox-date'>''%s''</span>", date) 		end 		self.info = args.info 		if yesno(args.removalnotice) then 			self.removalNotice = cfg.removalNotice 		end 		if args.shortFix then 			self.shortFix = args.shortFix 		end 	end  	-- Set the non-collapsible text field. At the moment this is used by all box 	-- types other than ambox, and also by ambox when small=yes. 	if self.isSmall then 		self.text = args.smalltext or args.text 	else 		self.text = args.text 		self.textsmall = args['text-small'] 	end  	-- Set the below row. 	self.below = cfg.below and args.below  	-- General image settings. 	self.imageCellDiv = not self.isSmall and cfg.imageCellDiv 	self.imageEmptyCell = cfg.imageEmptyCell 	if cfg.imageEmptyCellStyle then 		self.imageEmptyCellStyle = 'border:none;padding:0px;width:1px' 	end  	-- Left image settings. 	local imageLeft = self.isSmall and args.smallimage or args.image 	if cfg.imageCheckBlank and imageLeft ~= 'blank' and imageLeft ~= 'none' and imageLeft ~= '' 		or not cfg.imageCheckBlank and imageLeft ~= 'none' 	then 		self.imageLeft = imageLeft 		if not imageLeft then 			local imageSize = self.isSmall 				and (cfg.imageSmallSize or '30x30px') 			    or cfg.imageSize 				or '40x40px' 			self.imageLeft = string.format('[[File:%s|%s|alt=]]', self.typeImage 				or 'Information icon4.svg', imageSize) 		end 	end  	-- Right image settings. 	local imageRight = self.isSmall and args.smallimageright or args.imageright 	if not (cfg.imageRightNone and imageRight == 'none') then 		self.imageRight = imageRight 	end 	 	-- set templatestyles 	self.base_templatestyles = cfg.templatestyles end  function MessageBox:setMainspaceCategories() 	local args = self.args 	local cfg = self.cfg     local date = nil  	if not cfg.allowMainspaceCategories then 		return nil 	end  	local nums = {} 	for _, prefix in ipairs{'cat', 'category', 'all'} do 		args[prefix .. '1'] = args[prefix] 		nums = union(nums, getArgNums(args, prefix)) 	end  	-- The following is roughly equivalent to the old {{Ambox/category}}. 	local status, result = pcall(formatDate, args.date, 'xg Y') 	if status then 		date = result 	end  	date = type(date) == 'string' and date 	local preposition = 'с' 	local suffix = 'года' 	for _, num in ipairs(nums) do 		local mainCat = args['cat' .. tostring(num)] 			or args['category' .. tostring(num)] 		local allCat = args['all' .. tostring(num)] 		mainCat = type(mainCat) == 'string' and mainCat 		allCat = type(allCat) == 'string' and allCat 		if mainCat and date and date ~= '' then 			local catTitle = string.format('%s %s %s %s', mainCat, preposition, date, suffix) 			self:addCat(0, catTitle) 			catTitle = getTitleObject('Категория:' .. catTitle) 			local status, result = pcall(formatDate, args.date) 			if not status then 				self:addCat(0, 'Википедия:Статьи с недопустимым параметром даты в шаблоне-сообщении') 			end  		elseif mainCat and (not date or date == '') then 			self:addCat(0, mainCat) 		end 		if allCat then 			self:addCat(0, allCat) 		end 	end end  function MessageBox:setTemplateCategories() 	local args = self.args 	local cfg = self.cfg  	-- Add template categories. 	if cfg.templateCategory then 		if cfg.templateCategoryRequireName then 			if self.isTemplatePage then 				self:addCat(10, cfg.templateCategory) 			end 		elseif not self.title.isSubpage then 			self:addCat(10, cfg.templateCategory) 		end 	end  	-- Add template error categories. 	if cfg.templateErrorCategory then 		local templateErrorCategory = cfg.templateErrorCategory 		local templateCat, templateSort 		if not self.name and not self.title.isSubpage then 			templateCat = templateErrorCategory 		elseif self.isTemplatePage then 			local paramsToCheck = cfg.templateErrorParamsToCheck or {} 			local count = 0 			for i, param in ipairs(paramsToCheck) do 				if not args[param] then 					count = count + 1 				end 			end 			if count > 0 then 				templateCat = templateErrorCategory 				templateSort = tostring(count) 			end 			if self.categoryNums and #self.categoryNums > 0 then 				templateCat = templateErrorCategory 				templateSort = 'C' 			end 		end 		self:addCat(10, templateCat, templateSort) 	end end  function MessageBox:setAllNamespaceCategories() 	-- Set categories for all namespaces. 	if self.invalidTypeError then 		local allSort = (self.title.namespace == 0 and 'Main:' or '') .. self.title.prefixedText 		self:addCat('all', 'Википедия:Необходимо исправить параметр в шаблоне-сообщении', allSort) 	end 	if self.isSubstituted then 		self:addCat('all', 'Википедия:Страницы с ошибочно подставленными шаблонами') 	end 	if self.isSmall then 		self:addCat(0, 'Википедия:Страницы с малыми шаблонами-сообщениями') 	end end  function MessageBox:setCategories() 	if self.title.namespace == 0 then 		self:setMainspaceCategories() 	elseif self.title.namespace == 10 then 		self:setTemplateCategories() 	end 	self:setAllNamespaceCategories() end  function MessageBox:renderCategories() 	-- Convert category tables to strings and pass them through 	-- [[Module:Category handler]]. 	return categoryHandler{ 		main = table.concat(self.categories[0] or {}), 		template = table.concat(self.categories[10] or {}), 		all = table.concat(self.categories.all or {}), 		nocat = self.args.nocat, 		page = self.args.page 	} end  function MessageBox:export() 	local root = mw.html.create()  	-- Add the subst check error. 	if self.isSubstituted and self.name then 		root:tag('b') 			:addClass('error') 			:wikitext(string.format( 				'Шаблон <code>%s[[Шаблон:%s|%s]]%s</code> был неккоректно подставлен.', 				mw.text.nowiki('{{'), self.name, self.name, mw.text.nowiki('}}') 			)) 	end 	 	-- Conditional TemplateStyles loading 	if self.base_templatestyles then 		local frame = mw.getCurrentFrame() 		root:wikitext(frame:extensionTag{ 			name = 'templatestyles', 			args = { src = self.base_templatestyles }, 		}) 	end  	-- Create the box table. 	local boxTable = root:tag('table') 	boxTable:attr('id', self.id or nil) 	for i, class in ipairs(self.classes or {}) do 		boxTable:addClass(class or nil) 	end 	boxTable 		:cssText(self.style or nil) 		:attr('role', 'presentation') 			 	if self.dataLabel1 then 		boxTable:attr('data-' .. self.dataLabel1, self.dataValue1) 	end 	if self.dataLabel2 then 		boxTable:attr('data-' .. self.dataLabel2, self.dataValue2) 	end 	if self.dataLabel3 then 		boxTable:attr('data-' .. self.dataLabel3, self.dataValue3) 	end 	 	if self.attrs then 		boxTable:attr(self.attrs) 	end  	-- Add the left-hand image. 	local row = boxTable:tag('tr') 	if self.imageLeft then 		local imageLeftCell = row:tag('td'):addClass('mbox-image') 		if self.imageCellDiv then 			-- If we are using a div, redefine imageLeftCell so that the image 			-- is inside it. Divs use style="width: 52px;", which limits the 			-- image width to 52px. If any images in a div are wider than that, 			-- they may overlap with the text or cause other display problems. 			imageLeftCell = imageLeftCell:tag('div'):css('width', '52px') 		end 		imageLeftCell:wikitext(self.imageLeft or nil) 	elseif self.imageEmptyCell then 		-- Some message boxes define an empty cell if no image is specified, and 		-- some don't. The old template code in templates where empty cells are 		-- specified gives the following hint: "No image. Cell with some width 		-- or padding necessary for text cell to have 100% width." 		row:tag('td') 			:addClass('mbox-empty-cell') 			:cssText(self.imageEmptyCellStyle or nil) 	end  	-- Add the text. 	local textCell = row:tag('td'):addClass('mbox-text') 	if self.useCollapsibleTextFields then 		-- The message box uses advanced text parameters that allow things to be 		-- collapsible. At the moment, only ambox uses this. 		textCell:cssText(self.textstyle or nil) 		local textCellDiv = textCell:tag('div') 		textCellDiv 			:addClass('mbox-text-div') 			:wikitext(self.issue or nil) 		local textsmallCellDiv = textCell:tag('div') 		textsmallCellDiv 			:addClass('mbox-textsmall-div hide-when-compact') 			:cssText(self.textsmallstyle) 			:wikitext(self.textsmall or nil)	 		if (self.talk or self.fix) and not self.isSmall then 			textsmallCellDiv:tag('span') 				:addClass('hide-when-compact') 				:wikitext(self.fix and (' ' .. self.fix) or nil) 				:wikitext(self.talk and (' ' .. self.talk) or nil) 		end 		if self.textsmall or self.fix or self.talk then 			textsmallCellDiv:wikitext(self.date and (' ' .. self.date) or nil) 		else 			textCellDiv:wikitext(self.date and (' ' .. self.date) or nil)	 		end 		if self.info and not self.isSmall then 			textsmallCellDiv 				:tag('span') 				:addClass('hide-when-compact') 				:wikitext(self.info and (' ' .. self.info) or nil) 		end 		if self.removalNotice then 			textsmallCellDiv:tag('small') 				:addClass('hide-when-compact') 				:tag('i') 					:wikitext(string.format(" (%s)", self.removalNotice)) 		end 		if self.shortFix then 			textCell:tag('div') 				:addClass('mbox-multiply') 					:tag('span') 						:wikitext(string.format("%s", self.shortFix)) 					:tag('span') 						:wikitext(self.date and (' ' .. self.date) or nil) 		end 	else 		-- Default text formatting - anything goes. 		textCell 			:cssText(self.textstyle or nil) 			:wikitext(self.text or nil) 	end  	-- Add the right-hand image. 	if self.imageRight then 		local imageRightCell = row:tag('td'):addClass('mbox-imageright') 		if self.imageCellDiv then 			-- If we are using a div, redefine imageRightCell so that the image 			-- is inside it. 			imageRightCell = imageRightCell:tag('div'):css('width', '52px') 		end 		imageRightCell 			:wikitext(self.imageRight or nil) 	end  	-- Add the below row. 	if self.below then 		boxTable:tag('tr') 			:tag('td') 				:attr('colspan', self.imageRight and '3' or '2') 				:addClass('mbox-text') 				:cssText(self.textstyle or nil) 				:wikitext(self.below or nil) 	end  	-- Add error message for invalid type parameters. 	if self.invalidTypeError then 		root:tag('div') 			:css('text-align', 'center') 			:wikitext(string.format( 				'Этот шаблон-сообщение использует неверный параметр "type=%s", необходимо исправить.', 				self.type or '' 			)) 	end  	-- Add categories. 	root:wikitext(self:renderCategories() or nil)  	return tostring(root) end  -------------------------------------------------------------------------------- -- Exports --------------------------------------------------------------------------------  local p, mt = {}, {}  function p._exportClasses() 	-- For testing. 	return { 		MessageBox = MessageBox 	} end  function p.main(boxType, args, cfgTables) 	local box = MessageBox.new(boxType, args, cfgTables or mw.loadData(CONFIG_MODULE)) 	box:setParameters() 	box:setCategories() 	return box:export() end  function mt.__index(t, k) 	return function (frame) 		if not getArgs then 			getArgs = require('Module:Arguments').getArgs 		end 		return t.main(k, getArgs(frame, {trim = false, removeBlanks = false})) 	end end  return setmetatable(p, mt)