<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="ru">
	<id>https://bukvica.org/w/index.php?action=history&amp;feed=atom&amp;title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C%3AWDFormat</id>
	<title>Модуль:WDFormat - История изменений</title>
	<link rel="self" type="application/atom+xml" href="https://bukvica.org/w/index.php?action=history&amp;feed=atom&amp;title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C%3AWDFormat"/>
	<link rel="alternate" type="text/html" href="https://bukvica.org/w/index.php?title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&amp;action=history"/>
	<updated>2026-04-14T23:10:36Z</updated>
	<subtitle>История изменений этой страницы в вики</subtitle>
	<generator>MediaWiki 1.43.8</generator>
	<entry>
		<id>https://bukvica.org/w/index.php?title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&amp;diff=238441&amp;oldid=prev</id>
		<title>Karaby: 1 версия импортирована</title>
		<link rel="alternate" type="text/html" href="https://bukvica.org/w/index.php?title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&amp;diff=238441&amp;oldid=prev"/>
		<updated>2025-07-27T15:36:37Z</updated>

		<summary type="html">&lt;p&gt;1 версия импортирована&lt;/p&gt;
&lt;table style=&quot;background-color: #fff; color: #202122;&quot; data-mw=&quot;interface&quot;&gt;
				&lt;col class=&quot;diff-marker&quot; /&gt;
				&lt;col class=&quot;diff-content&quot; /&gt;
				&lt;col class=&quot;diff-marker&quot; /&gt;
				&lt;col class=&quot;diff-content&quot; /&gt;
				&lt;tr class=&quot;diff-title&quot; lang=&quot;ru&quot;&gt;
				&lt;td colspan=&quot;2&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;← Предыдущая версия&lt;/td&gt;
				&lt;td colspan=&quot;2&quot; style=&quot;background-color: #fff; color: #202122; text-align: center;&quot;&gt;Версия от 15:36, 27 июля 2025&lt;/td&gt;
				&lt;/tr&gt;&lt;tr&gt;&lt;td colspan=&quot;4&quot; class=&quot;diff-notice&quot; lang=&quot;ru&quot;&gt;&lt;div class=&quot;mw-diff-empty&quot;&gt;(нет различий)&lt;/div&gt;
&lt;/td&gt;&lt;/tr&gt;
&lt;!-- diff cache key wiki:diff:1.41:old-238440:rev-238441 --&gt;
&lt;/table&gt;</summary>
		<author><name>Karaby</name></author>
	</entry>
	<entry>
		<id>https://bukvica.org/w/index.php?title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&amp;diff=238440&amp;oldid=prev</id>
		<title>Буквица&gt;The Fox Bot: Защитил страницу Модуль:WDFormat: критический шаблон или модуль: согласно Служебная:Постоянная ссылка/143835923#Промежуточный итог ([Редактирование=только автоподтверждённые] (бессрочно) [Переименование=только автоподтверждённые] (бессрочно))</title>
		<link rel="alternate" type="text/html" href="https://bukvica.org/w/index.php?title=%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&amp;diff=238440&amp;oldid=prev"/>
		<updated>2025-03-05T16:12:29Z</updated>

		<summary type="html">&lt;p&gt;Защитил страницу &lt;a href=&quot;/wiki/%D0%9C%D0%BE%D0%B4%D1%83%D0%BB%D1%8C:WDFormat&quot; title=&quot;Модуль:WDFormat&quot;&gt;Модуль:WDFormat&lt;/a&gt;: &lt;a href=&quot;/w/index.php?title=%D0%92%D0%9F:%D0%9A%D0%A8&amp;amp;action=edit&amp;amp;redlink=1&quot; class=&quot;new&quot; title=&quot;ВП:КШ (страница не существует)&quot;&gt;критический шаблон или модуль&lt;/a&gt;: согласно &lt;a href=&quot;/wiki/%D0%A1%D0%BB%D1%83%D0%B6%D0%B5%D0%B1%D0%BD%D0%B0%D1%8F:%D0%9F%D0%BE%D1%81%D1%82%D0%BE%D1%8F%D0%BD%D0%BD%D0%B0%D1%8F_%D1%81%D1%81%D1%8B%D0%BB%D0%BA%D0%B0/143835923#Промежуточный_итог&quot; title=&quot;Служебная:Постоянная ссылка/143835923&quot;&gt;Служебная:Постоянная ссылка/143835923&lt;/a&gt; ([Редактирование=только автоподтверждённые] (бессрочно) [Переименование=только автоподтверждённые] (бессрочно))&lt;/p&gt;
&lt;p&gt;&lt;b&gt;Новая страница&lt;/b&gt;&lt;/p&gt;&lt;div&gt;require(&amp;#039;strict&amp;#039;)&lt;br /&gt;
&lt;br /&gt;
local p = {}&lt;br /&gt;
&lt;br /&gt;
local NS_MODULE = 828 --: https://www.mediawiki.org/wiki/Extension_default_namespaces&lt;br /&gt;
local moduleNamespace = mw.site.namespaces[NS_MODULE].name&lt;br /&gt;
&lt;br /&gt;
local wikidata = require(moduleNamespace .. &amp;#039;:WDCommon&amp;#039;)&lt;br /&gt;
&lt;br /&gt;
local function dump(obj, level)&lt;br /&gt;
	if type(obj) ~= &amp;#039;table&amp;#039; then&lt;br /&gt;
		if type(obj) == &amp;#039;string&amp;#039; then&lt;br /&gt;
			return &amp;quot;&amp;#039;&amp;quot; .. obj .. &amp;quot;&amp;#039;&amp;quot;&lt;br /&gt;
		else&lt;br /&gt;
			return tostring(obj)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if not level then&lt;br /&gt;
		level = 0&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local indent = string.rep(&amp;#039; &amp;#039;, level)&lt;br /&gt;
	local s = &amp;#039;{&amp;#039;&lt;br /&gt;
	local isFirst = true&lt;br /&gt;
	for i, v in pairs(obj) do&lt;br /&gt;
		if not isFirst then&lt;br /&gt;
			s = s .. &amp;#039;,&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		s = s .. &amp;#039;\n&amp;#039;&lt;br /&gt;
		local currIndent = string.rep(&amp;#039; &amp;#039;, level + 4)&lt;br /&gt;
		s = s .. currIndent&lt;br /&gt;
		if type(i) == &amp;#039;string&amp;#039; then&lt;br /&gt;
			s = s .. i .. &amp;#039; = &amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		s = s .. dump(v, level + 4)&lt;br /&gt;
		isFirst = false&lt;br /&gt;
	end&lt;br /&gt;
	s = s .. &amp;#039;\n&amp;#039; .. indent .. &amp;#039;}&amp;#039;&lt;br /&gt;
	return s&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local Formatter = {&lt;br /&gt;
	profile = nil,&lt;br /&gt;
	processField = {},&lt;br /&gt;
}&lt;br /&gt;
&lt;br /&gt;
-- Transforms tag table hierarchy to mediawiki html container representation.&lt;br /&gt;
-- Returns deepest child.&lt;br /&gt;
function Formatter:tagToContainer(tag, parentContainer, source)&lt;br /&gt;
	if not tag or not tag.name then&lt;br /&gt;
		return parentContainer&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local container = parentContainer&lt;br /&gt;
	while tag do&lt;br /&gt;
		if container then&lt;br /&gt;
			container = container:tag(tag.name)&lt;br /&gt;
		else&lt;br /&gt;
			container = mw.html.create(tag.name)&lt;br /&gt;
		end&lt;br /&gt;
	&lt;br /&gt;
		if tag.classes then&lt;br /&gt;
			for _, currClass in ipairs(tag.classes) do&lt;br /&gt;
				container:addClass(currClass)&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	&lt;br /&gt;
		if tag.attr then&lt;br /&gt;
			for attr, value in pairs(tag.attr) do&lt;br /&gt;
				if type(value) == &amp;#039;function&amp;#039; then&lt;br /&gt;
					value = value(source)&lt;br /&gt;
				end&lt;br /&gt;
				container:attr(attr, value)&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
&lt;br /&gt;
		if tag.css then&lt;br /&gt;
			for key, value in pairs(tag.css) do&lt;br /&gt;
				container:css(key, value)&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
		&lt;br /&gt;
		tag = tag.tag&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return container&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:new(profile, source, langCode, params)&lt;br /&gt;
	local obj = {}&lt;br /&gt;
	setmetatable(obj, self)&lt;br /&gt;
	self.__index = self&lt;br /&gt;
&lt;br /&gt;
	self.profile = profile&lt;br /&gt;
	self.source = source&lt;br /&gt;
	self.lang = langCode&lt;br /&gt;
	self.params = params&lt;br /&gt;
	self.state = {&lt;br /&gt;
		empty = true,&lt;br /&gt;
		linkedEntities = {},&lt;br /&gt;
	}&lt;br /&gt;
	self.lastAddedText = &amp;#039;&amp;#039;&lt;br /&gt;
	self.container = mw.html.create()&lt;br /&gt;
	assert(self.profile.tag ~= nil, &amp;#039;Not root container found. Use tag profile field.&amp;#039;)&lt;br /&gt;
&lt;br /&gt;
	return obj&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.numericalRanges(source, processedData, result)&lt;br /&gt;
	local defaultLangObj = mw.getContentLanguage()&lt;br /&gt;
	local defaultLang = defaultLangObj:getCode()&lt;br /&gt;
	local rangeSign&lt;br /&gt;
	if defaultLang == &amp;#039;ru&amp;#039; then&lt;br /&gt;
		rangeSign = &amp;#039;—&amp;#039;&lt;br /&gt;
	elseif defaultLang == &amp;#039;ko&amp;#039; then&lt;br /&gt;
		rangeSign = &amp;#039;~&amp;#039;&lt;br /&gt;
	else&lt;br /&gt;
		rangeSign = &amp;#039;–&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	result.wikitext = mw.ustring.gsub(result.wikitext, &amp;#039;-&amp;#039;, rangeSign)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.squareBrackets(source, processedData, result)&lt;br /&gt;
	result.text = &amp;#039;&amp;amp;lsqb;&amp;#039; .. result.text .. &amp;#039;&amp;amp;rsqb;&amp;#039;&lt;br /&gt;
	result.wikitext = &amp;#039;&amp;amp;lsqb;&amp;#039; .. result.wikitext .. &amp;#039;&amp;amp;rsqb;&amp;#039;&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.nowiki(source, processedData, result)&lt;br /&gt;
	result.wikitext = mw.text.nowiki(result.wikitext)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.dash(source, processedData, result)&lt;br /&gt;
	local defaultLangObj = mw.getContentLanguage()&lt;br /&gt;
	local defaultLang = defaultLangObj:getCode()&lt;br /&gt;
	local dashSign&lt;br /&gt;
	if defaultLang == &amp;#039;ru&amp;#039; then&lt;br /&gt;
		dashSign = &amp;#039; — &amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	result.wikitext = mw.ustring.gsub(result.wikitext, &amp;#039; %- &amp;#039;, dashSign)&lt;br /&gt;
	result.text = mw.ustring.gsub(result.text, &amp;#039; %- &amp;#039;, dashSign)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.unit(source, processedData, result, state)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local unit = wikidata.unit(entity, processedData.langCode)&lt;br /&gt;
	if (unit and state.groupEmpty and processedData.capitalize) or processedData.forceCapitalize then&lt;br /&gt;
		unit = mw.ustring.gsub(unit, &amp;#039;^%l&amp;#039;, mw.ustring.upper)&lt;br /&gt;
	end&lt;br /&gt;
	result.text = unit&lt;br /&gt;
	result.wikitext = unit&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.abbr(source, processedData, result, state)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity or processedData.fieldTable.exact then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local abbr, _, ok = wikidata.abbrBiblio(entity, processedData.langCode)&lt;br /&gt;
	if (abbr and state.groupEmpty and processedData.capitalize) or processedData.forceCapitalize then&lt;br /&gt;
		abbr = mw.ustring.gsub(abbr, &amp;#039;^%l&amp;#039;, mw.ustring.upper)&lt;br /&gt;
	end&lt;br /&gt;
	if abbr and ok then&lt;br /&gt;
		abbr = mw.ustring.gsub(abbr, &amp;#039; &amp;#039;, &amp;#039;&amp;amp;nbsp;&amp;#039;)&lt;br /&gt;
	end&lt;br /&gt;
	result.text = abbr&lt;br /&gt;
	result.wikitext = abbr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.short(source, processedData, result, state)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity or processedData.fieldTable.exact then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local short = wikidata.short(entity, processedData.langCode)&lt;br /&gt;
	if (short and state.groupEmpty and processedData.capitalize) or processedData.forceCapitalize then&lt;br /&gt;
		short = mw.ustring.gsub(short, &amp;#039;^%l&amp;#039;, mw.ustring.upper)&lt;br /&gt;
	end&lt;br /&gt;
	if not short then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = short&lt;br /&gt;
	result.wikitext = short&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.abbrWithHint(source, processedData, result, state)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity or processedData.fieldTable.exact then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local abbr, _, ok = wikidata.abbrBiblio(entity, processedData.langCode)&lt;br /&gt;
	if (abbr and state.groupEmpty and processedData.capitalize) or processedData.forceCapitalize then&lt;br /&gt;
		abbr = mw.ustring.gsub(abbr, &amp;#039;^%l&amp;#039;, mw.ustring.upper)&lt;br /&gt;
	end&lt;br /&gt;
	if abbr and ok then&lt;br /&gt;
		abbr = mw.ustring.gsub(abbr, &amp;#039; &amp;#039;, &amp;#039;&amp;amp;nbsp;&amp;#039;)&lt;br /&gt;
	end&lt;br /&gt;
	if result.text ~= abbr then&lt;br /&gt;
		result.wikitext = &amp;#039;&amp;lt;abbr title=&amp;quot;&amp;#039; .. result.wikitext .. &amp;#039;&amp;quot;&amp;gt;&amp;#039; .. abbr .. &amp;#039;&amp;lt;/abbr&amp;gt;&amp;#039;&lt;br /&gt;
		result.text = abbr&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.date(source, processedData, result)&lt;br /&gt;
	if type(processedData.fieldTable.value) ~= &amp;#039;table&amp;#039; then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
	local langObj = mw.getLanguage(processedData.langCode)&lt;br /&gt;
	result.text = langObj:formatDate(&amp;#039;j xg Y&amp;#039;, processedData.fieldTable.value.timestamp)&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.dateISO(source, processedData, result)&lt;br /&gt;
	if type(processedData.fieldTable.value) ~= &amp;#039;table&amp;#039; then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
	local langObj = mw.getLanguage(processedData.langCode)&lt;br /&gt;
	result.text = langObj:formatDate(&amp;#039;Y-m-d&amp;#039;, processedData.fieldTable.value.timestamp)&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.uriScheme(source, processedData, result)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = wikidata.uriScheme(entity)&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.quantity(source, processedData, result)&lt;br /&gt;
	local fieldTable = processedData.fieldTable&lt;br /&gt;
	if not fieldTable.unitEntity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local unit = wikidata.unit(fieldTable.unitEntity, processedData.langCode)&lt;br /&gt;
	if unit then&lt;br /&gt;
		result.text = result.text .. &amp;#039; &amp;#039; .. unit&lt;br /&gt;
		result.wikitext = result.wikitext .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. unit&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function nameFromFieldTable(nameTable)&lt;br /&gt;
	local value = nameTable.value&lt;br /&gt;
	local ok = true&lt;br /&gt;
	if not value then&lt;br /&gt;
		local label = mw.wikibase.getLabel(nameTable.entity)&lt;br /&gt;
		if label then&lt;br /&gt;
			value = mw.wikibase.getLabel(nameTable.entity) .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. nameTable.entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		else&lt;br /&gt;
			value = &amp;#039;?&amp;#039; .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. nameTable.entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		ok = false&lt;br /&gt;
	end&lt;br /&gt;
	return value, ok&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function nameToInitial(nameTable)&lt;br /&gt;
	local value = nameTable.value&lt;br /&gt;
	local ok&lt;br /&gt;
	if not value then&lt;br /&gt;
		local label = mw.wikibase.getLabel(nameTable.entity)&lt;br /&gt;
		if label then&lt;br /&gt;
			value = label .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. nameTable.entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		else&lt;br /&gt;
			value = &amp;#039;?&amp;#039; .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. nameTable.entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		ok = false&lt;br /&gt;
	else&lt;br /&gt;
		value = mw.ustring.sub(value, 1, 1) .. &amp;#039;.&amp;#039;&lt;br /&gt;
		ok = true&lt;br /&gt;
	end&lt;br /&gt;
	return value, ok&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function namesToInitial(nameTables)&lt;br /&gt;
	local value, ok&lt;br /&gt;
	if table.getn(nameTables) == 0 then&lt;br /&gt;
		value, ok = nameToInitial(nameTables)&lt;br /&gt;
	else&lt;br /&gt;
		value = &amp;#039;&amp;#039;&lt;br /&gt;
		ok = true&lt;br /&gt;
		for i, nameTable in ipairs(nameTables) do&lt;br /&gt;
			if i &amp;gt; 1 then&lt;br /&gt;
				value = value .. &amp;#039;&amp;amp;nbsp;&amp;#039;&lt;br /&gt;
			end&lt;br /&gt;
			local currValue, currOk = nameToInitial(nameTable)&lt;br /&gt;
			value = value .. currValue&lt;br /&gt;
			ok = ok and currOk&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return value, ok&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.person(source, processedData, result)&lt;br /&gt;
	-- currently supports only names with givenName and pathronym/mathronym,&lt;br /&gt;
	-- middle names are not supported if they are a part of givenName&lt;br /&gt;
	local fieldTable = processedData.fieldTable&lt;br /&gt;
	local name&lt;br /&gt;
	if fieldTable.exact then&lt;br /&gt;
		name = result.text&lt;br /&gt;
		if name:match(&amp;#039;,&amp;#039;) then&lt;br /&gt;
			name = name:gsub(&amp;#039;([^,]+), *(.+)&amp;#039;, &amp;#039;%2 %1&amp;#039;)&lt;br /&gt;
			result.text = name&lt;br /&gt;
			result.wikitext = name&lt;br /&gt;
		end&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if fieldTable.components and fieldTable.components.familyName and fieldTable.components.givenName then&lt;br /&gt;
		local ok&lt;br /&gt;
		name, ok = namesToInitial(fieldTable.components.givenName)&lt;br /&gt;
		if fieldTable.components.ancestorName then&lt;br /&gt;
			local ancestorName, ancestorNameOk = namesToInitial(fieldTable.components.ancestorName)&lt;br /&gt;
			name = name .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. ancestorName&lt;br /&gt;
			ok = ok and ancestorNameOk&lt;br /&gt;
		end&lt;br /&gt;
&lt;br /&gt;
		local familyName, familyNameOk = nameFromFieldTable(fieldTable.components.familyName)&lt;br /&gt;
		name = name .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. familyName&lt;br /&gt;
		ok = ok and familyNameOk&lt;br /&gt;
		if not ok then&lt;br /&gt;
			result.linked = true&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		local entity = processedData.fieldTable.entity&lt;br /&gt;
		if entity then&lt;br /&gt;
			name = wikidata.abbr(entity, processedData.langCode)&lt;br /&gt;
		else&lt;br /&gt;
			name = processedData.value&lt;br /&gt;
		end&lt;br /&gt;
		if not name then&lt;br /&gt;
			name = mw.wikibase.getLabel(entity) .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = name&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.personReversed(source, processedData, result)&lt;br /&gt;
	-- currently supports only names with givenName and pathronym/mathronym,&lt;br /&gt;
	-- middle names are not supported if they are a part of givenName&lt;br /&gt;
	local fieldTable = processedData.fieldTable&lt;br /&gt;
	local name&lt;br /&gt;
	if fieldTable.exact then&lt;br /&gt;
		name = result.text&lt;br /&gt;
		if not name:match(&amp;#039;,&amp;#039;) then&lt;br /&gt;
			local initials, familyName = mw.ustring.match(name, &amp;#039;^(.+%.) ([^%. ]+)$&amp;#039;)&lt;br /&gt;
			if not initials or not familyName then&lt;br /&gt;
				return&lt;br /&gt;
			end&lt;br /&gt;
			name = familyName .. &amp;#039;,&amp;amp;nbsp;&amp;#039; .. initials&lt;br /&gt;
			result.text = name&lt;br /&gt;
			result.wikitext = name&lt;br /&gt;
		end&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if fieldTable.components and fieldTable.components.familyName and fieldTable.components.givenName then&lt;br /&gt;
		local ok&lt;br /&gt;
		local familyName, ok = nameFromFieldTable(fieldTable.components.familyName)&lt;br /&gt;
		local givenName, givenNameOk = namesToInitial(fieldTable.components.givenName)&lt;br /&gt;
		ok = ok and givenNameOk&lt;br /&gt;
		name = familyName .. &amp;#039;,&amp;amp;nbsp;&amp;#039; .. givenName&lt;br /&gt;
		if fieldTable.components.ancestorName then&lt;br /&gt;
			local ancestorName, ancestorNameOk = namesToInitial(fieldTable.components.ancestorName)&lt;br /&gt;
			ok = ok and ancestorNameOk&lt;br /&gt;
			name = name .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. ancestorName&lt;br /&gt;
		end&lt;br /&gt;
		if not ok then&lt;br /&gt;
			result.linked = true&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		local entity = processedData.fieldTable.entity&lt;br /&gt;
		if entity then&lt;br /&gt;
			name = wikidata.abbr(entity, processedData.langCode)&lt;br /&gt;
		else&lt;br /&gt;
			return&lt;br /&gt;
		end&lt;br /&gt;
		if not name then&lt;br /&gt;
			name = mw.wikibase.getLabel(entity) .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = name&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.personReversedNoComma(source, processedData, result)&lt;br /&gt;
	-- currently supports only names with givenName and pathronym/mathronym,&lt;br /&gt;
	-- middle names are not supported if they are a part of givenName&lt;br /&gt;
	local fieldTable = processedData.fieldTable&lt;br /&gt;
	local name&lt;br /&gt;
	if fieldTable.exact then&lt;br /&gt;
		name = result.text&lt;br /&gt;
		if name:match(&amp;#039;,&amp;#039;) then&lt;br /&gt;
			name = name:gsub(&amp;#039;([^,]+), *(.+)&amp;#039;, &amp;#039;%2&amp;amp;nbsp;%1&amp;#039;)&lt;br /&gt;
			result.text = name&lt;br /&gt;
			result.wikitext = name&lt;br /&gt;
		else&lt;br /&gt;
			local initials, familyName = mw.ustring.match(name, &amp;#039;^(.+%.) ([^%. ]+)$&amp;#039;)&lt;br /&gt;
			if not initials or not familyName then&lt;br /&gt;
				return&lt;br /&gt;
			end&lt;br /&gt;
			name = familyName .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. initials&lt;br /&gt;
			result.text = name&lt;br /&gt;
			result.wikitext = name&lt;br /&gt;
		end&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if fieldTable.exact then&lt;br /&gt;
		name = result.text&lt;br /&gt;
		if name:match(&amp;#039;,&amp;#039;) then&lt;br /&gt;
			name = name:gsub(&amp;#039;([^,]+), *(.+)&amp;#039;, &amp;#039;%2 %1&amp;#039;)&lt;br /&gt;
			result.text = name&lt;br /&gt;
			result.wikitext = name&lt;br /&gt;
		end&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if fieldTable.components and fieldTable.components.familyName and fieldTable.components.givenName then&lt;br /&gt;
		local ok&lt;br /&gt;
		local familyName, ok = nameFromFieldTable(fieldTable.components.familyName)&lt;br /&gt;
		local givenName, givenNameOk = namesToInitial(fieldTable.components.givenName)&lt;br /&gt;
		ok = ok and givenNameOk&lt;br /&gt;
		name = familyName .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. givenName&lt;br /&gt;
		if fieldTable.components.ancestorName then&lt;br /&gt;
			local ancestorName, ancestorNameOk = namesToInitial(fieldTable.components.ancestorName)&lt;br /&gt;
			ok = ok and ancestorNameOk&lt;br /&gt;
			name = name .. &amp;#039;&amp;amp;nbsp;&amp;#039; .. ancestorName&lt;br /&gt;
		end&lt;br /&gt;
		if not ok then&lt;br /&gt;
			result.linked = true&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		local entity = processedData.fieldTable.entity&lt;br /&gt;
		if entity then&lt;br /&gt;
			name = wikidata.abbr(entity, processedData.langCode)&lt;br /&gt;
		else&lt;br /&gt;
			return&lt;br /&gt;
		end&lt;br /&gt;
		if not name then&lt;br /&gt;
			name = mw.wikibase.getLabel(entity) .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = name&lt;br /&gt;
	result.wikitext = result.text&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.lowercase(source, processedData, result)&lt;br /&gt;
	result.text = mw.ustring.lower(result.text)&lt;br /&gt;
	result.wikitext = mw.ustring.lower(result.wikitext)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikisource(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local title = mw.wikibase.getSitelink(entity, processedData.langCode .. &amp;#039;wikisource&amp;#039;)&lt;br /&gt;
	if not title then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[[s:&amp;#039; .. title .. &amp;#039;|&amp;#039; .. result.text .. &amp;#039;]]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikiversity(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local title = mw.wikibase.getSitelink(entity, processedData.langCode .. &amp;#039;wikiversity&amp;#039;)&lt;br /&gt;
	if not title then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[[s:&amp;#039; .. title .. &amp;#039;|&amp;#039; .. result.text .. &amp;#039;]]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikibooks(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local title = mw.wikibase.getSitelink(entity, processedData.langCode .. &amp;#039;wikibooks&amp;#039;)&lt;br /&gt;
	if not title then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[[s:&amp;#039; .. title .. &amp;#039;|&amp;#039; .. result.text .. &amp;#039;]]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.link(source, processedData, result)&lt;br /&gt;
	if result.linked or not processedData.url then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[&amp;#039; .. processedData.url .. &amp;#039; &amp;#039; .. result.wikitext .. &amp;#039;]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.safeLink(source, processedData, result)&lt;br /&gt;
	if result.linked or not processedData.url then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
	if mw.ustring.match(result.wikitext, &amp;#039;%[&amp;#039;) then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[&amp;#039; .. processedData.url .. &amp;#039; &amp;#039; .. result.wikitext .. &amp;#039;]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikilink(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if processedData.wikilink then&lt;br /&gt;
		result.wikitext = &amp;#039;[[&amp;#039; .. processedData.wikilink .. &amp;#039;|&amp;#039; .. result.wikitext .. &amp;#039;]]&amp;#039;&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if result.state.linkedEntities[entity] then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local wikitext&lt;br /&gt;
	wikitext, entity = wikidata.base.wikilink(entity, result.wikitext)&lt;br /&gt;
	if result.state.linkedEntities[entity] or not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = wikitext&lt;br /&gt;
	result.state.linkedEntities[entity] = true&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.resolveWikilink(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if processedData.wikilink then&lt;br /&gt;
		result.wikitext = &amp;#039;[[&amp;#039; .. processedData.wikilink .. &amp;#039;|&amp;#039; .. result.wikitext .. &amp;#039;]]&amp;#039;&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if result.state.linkedEntities[entity] then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local wikitext&lt;br /&gt;
	wikitext, entity = wikidata.base.wikilink(entity, result.wikitext)&lt;br /&gt;
	if not entity then&lt;br /&gt;
		entity = wikidata.base.resolveParent(processedData.fieldTable.entity)&lt;br /&gt;
		if not entity then&lt;br /&gt;
			return&lt;br /&gt;
		end&lt;br /&gt;
		wikitext, entity = wikidata.base.wikilink(entity, result.wikitext)&lt;br /&gt;
	end&lt;br /&gt;
	if not entity or result.state.linkedEntities[entity] then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = wikitext&lt;br /&gt;
	result.state.linkedEntities[entity] = true&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.valuableWikidata(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity or result.state.linkedEntities[entity] then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entityObj = mw.wikibase.getEntity(entity)&lt;br /&gt;
	if entityObj.sitelinks then&lt;br /&gt;
		result.wikitext = result.wikitext .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|&amp;amp;#91;d&amp;amp;#93;]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
		result.state.linkedEntities[entity] = true&lt;br /&gt;
		result.linked = true&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikidata(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity or result.state.linkedEntities[entity] then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = result.wikitext .. &amp;#039;&amp;lt;sup class=&amp;quot;noprint&amp;quot;&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|&amp;amp;#91;d&amp;amp;#93;]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
	result.state.linkedEntities[entity] = true&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.forceWikidata(source, processedData, result)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = result.wikitext .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. entity .. &amp;#039;|&amp;amp;#91;d&amp;amp;#93;]]&amp;lt;/sup&amp;gt;&amp;#039; &lt;br /&gt;
	result.state.linkedEntities[entity] = true&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.wikidataLink(source, processedData, result)&lt;br /&gt;
	if result.linked then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[[d:&amp;#039; .. entity .. &amp;#039;|&amp;#039; .. result.wikitext .. &amp;#039;]]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.forceWikidataLink(source, processedData, result)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.wikitext = &amp;#039;[[d:&amp;#039; .. entity .. &amp;#039;|&amp;#039; .. result.wikitext .. &amp;#039;]]&amp;#039;&lt;br /&gt;
	result.linked = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter.processField.entity(source, processedData, result)&lt;br /&gt;
	local entity = processedData.fieldTable.entity&lt;br /&gt;
	if not entity then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	result.text = entity&lt;br /&gt;
	result.wikitext = entity&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function fieldTableByPath(source, path)&lt;br /&gt;
	if type(path) == &amp;#039;table&amp;#039; then&lt;br /&gt;
		local currField = nil&lt;br /&gt;
		local currFieldComponents = source&lt;br /&gt;
		for _, currFieldName in ipairs(path) do&lt;br /&gt;
			if type(currFieldName) == &amp;#039;string&amp;#039; then&lt;br /&gt;
				if not currFieldComponents then&lt;br /&gt;
					return nil&lt;br /&gt;
				end&lt;br /&gt;
				currField = currFieldComponents[currFieldName]&lt;br /&gt;
			else&lt;br /&gt;
				currField = currField[currFieldName]&lt;br /&gt;
			end&lt;br /&gt;
			if not currField then&lt;br /&gt;
				return nil&lt;br /&gt;
			end&lt;br /&gt;
&lt;br /&gt;
			currFieldComponents = currField.components&lt;br /&gt;
		end&lt;br /&gt;
		return currField&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return source[path]&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function fieldValueByPath(source, path)&lt;br /&gt;
	local t = fieldTableByPath(source, path)&lt;br /&gt;
	if not t then&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if type(path) == &amp;#039;table&amp;#039; and path.sub then&lt;br /&gt;
		if not t.value then&lt;br /&gt;
			return nil&lt;br /&gt;
		end&lt;br /&gt;
		return t.value[path.sub]&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return t.value&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function urlFromMaskByPart(part, source, fieldValue, langCode)&lt;br /&gt;
	if part.urlMaskProp then&lt;br /&gt;
		local urlMask = wikidata.urlMask(part.urlMaskProp, langCode)&lt;br /&gt;
		if urlMask then&lt;br /&gt;
			return urlMask:gsub(&amp;#039;%$1&amp;#039;, fieldValue)&lt;br /&gt;
		end&lt;br /&gt;
	elseif part.urlField then&lt;br /&gt;
		local urlTable = source[part.urlField]&lt;br /&gt;
		if urlTable then&lt;br /&gt;
			if type(urlTable) == &amp;#039;table&amp;#039; then&lt;br /&gt;
				return urlTable.value&lt;br /&gt;
			else&lt;br /&gt;
				return urlTable&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return nil&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function wikilinkFromMaskByPart(part, source, fieldValue)&lt;br /&gt;
	if part.wikilinkMask then&lt;br /&gt;
		return part.wikilinkMask:gsub(&amp;#039;%$1&amp;#039;, fieldValue)&lt;br /&gt;
	end&lt;br /&gt;
	return nil&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:forceLangByPart(part, source)&lt;br /&gt;
	local langCode&lt;br /&gt;
	if not part.forceLang then&lt;br /&gt;
		langCode = self.lang&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if langCode then&lt;br /&gt;
		return langCode&lt;br /&gt;
	elseif part.forceLang == &amp;#039;fallback&amp;#039; then&lt;br /&gt;
		langCode = &amp;#039;en&amp;#039;&lt;br /&gt;
	elseif part.forceLang == &amp;#039;default&amp;#039; or not part.forceLang then&lt;br /&gt;
		local defaultLangObj = mw.getContentLanguage()&lt;br /&gt;
		local defaultLang = defaultLangObj:getCode()&lt;br /&gt;
		langCode = defaultLang&lt;br /&gt;
	else&lt;br /&gt;
		langCode = part.forceLang&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return langCode&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Format field with all neccessary data supplied by arguments.&lt;br /&gt;
function Formatter:formatField(source, part, fieldTable, fieldValue, state)&lt;br /&gt;
	local processedData = {&lt;br /&gt;
		field = part.field,&lt;br /&gt;
		wikilink = wikilinkFromMaskByPart(part, source, fieldValue),&lt;br /&gt;
		langCode = self:forceLangByPart(part, source),&lt;br /&gt;
		fieldTable = fieldTable,&lt;br /&gt;
		capitalize = (part.capitalize or (part.capitalize == nil)) and (state.index == 1),&lt;br /&gt;
		forceCapitalize = (part.capitalize == true and state.index == 1),&lt;br /&gt;
	}&lt;br /&gt;
	processedData.url = urlFromMaskByPart(part, source, fieldValue, processedData.langCode)&lt;br /&gt;
	local value = fieldValue&lt;br /&gt;
	local linked = false&lt;br /&gt;
	if fieldTable.entity and (processedData.langCode ~= self.lang or not value) then&lt;br /&gt;
		if processedData.langCode then&lt;br /&gt;
			value = mw.wikibase.getLabelByLang(fieldTable.entity, processedData.langCode)&lt;br /&gt;
			if not value then&lt;br /&gt;
				value = mw.wikibase.getLabel(fieldTable.entity)&lt;br /&gt;
				if value then&lt;br /&gt;
					value = value .. &amp;#039;&amp;lt;sup&amp;gt;[[d:&amp;#039; .. fieldTable.entity .. &amp;#039;|?]]&amp;lt;/sup&amp;gt;&amp;#039;&lt;br /&gt;
					linked = true&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		else&lt;br /&gt;
			value = mw.wikibase.getLabel(fieldTable.entity)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if value and type(value) == &amp;#039;string&amp;#039; then&lt;br /&gt;
		if (state.groupEmpty and processedData.capitalize) or processedData.forceCapitalize then&lt;br /&gt;
			value = mw.ustring.gsub(value, &amp;#039;^%l&amp;#039;, mw.ustring.upper)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if type(value) == &amp;#039;table&amp;#039; then&lt;br /&gt;
		value = dump(value)&lt;br /&gt;
	elseif value == nil then&lt;br /&gt;
		value = &amp;#039;&amp;lt;b&amp;gt;&amp;lt;s&amp;gt;(nil)&amp;lt;/s&amp;gt;&amp;lt;/b&amp;gt;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	processedData.value = value&lt;br /&gt;
&lt;br /&gt;
	local result = {&lt;br /&gt;
		wikitext = value,&lt;br /&gt;
		text = value,&lt;br /&gt;
		state = state,&lt;br /&gt;
		linked = linked,&lt;br /&gt;
	}&lt;br /&gt;
	if not part.format then&lt;br /&gt;
		state.groupEmpty = false&lt;br /&gt;
		return result&lt;br /&gt;
	end&lt;br /&gt;
	for _, formatFunc in ipairs(part.format) do&lt;br /&gt;
		formatFunc(source, processedData, result, state)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	state.groupEmpty = false&lt;br /&gt;
	return result&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Used if fieldTable contains array.&lt;br /&gt;
function Formatter:formatFieldAsArray(source, part, fieldTable, state)&lt;br /&gt;
	local text = &amp;#039;&amp;#039;&lt;br /&gt;
	local wikitext = &amp;#039;&amp;#039;&lt;br /&gt;
&lt;br /&gt;
	local count = table.getn(fieldTable)&lt;br /&gt;
	local cutCount = 0&lt;br /&gt;
	if part.limits and count &amp;gt; part.limits.max then&lt;br /&gt;
		cutCount = count - part.limits.cutTo&lt;br /&gt;
		count = part.limits.cutTo&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local delimiter = part.itemsDelimiter&lt;br /&gt;
	if not delimiter then&lt;br /&gt;
		delimiter = &amp;#039;, &amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	for i=1, count do&lt;br /&gt;
		state.index = i&lt;br /&gt;
		local currFieldTable = fieldTable[i]&lt;br /&gt;
		local currResult = self:formatField(source, part, currFieldTable, currFieldTable.value, state)&lt;br /&gt;
		wikitext = wikitext .. currResult.wikitext&lt;br /&gt;
		text = text .. currResult.text&lt;br /&gt;
		if i &amp;lt; count then&lt;br /&gt;
			text = text .. delimiter&lt;br /&gt;
			wikitext = wikitext .. delimiter&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	if cutCount &amp;gt; 0 then&lt;br /&gt;
		local cutText = &amp;#039;&amp;#039;&lt;br /&gt;
		for i=count + 1, count + cutCount do&lt;br /&gt;
			state.index = i&lt;br /&gt;
			local currFieldTable = fieldTable[i]&lt;br /&gt;
			local currResult = self:formatField(source, part, currFieldTable, currFieldTable.value, state)&lt;br /&gt;
			cutText = cutText .. currResult.text&lt;br /&gt;
			if i &amp;lt; count + cutCount then&lt;br /&gt;
				cutText = cutText .. delimiter&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
&lt;br /&gt;
		local processedData = {&lt;br /&gt;
			field = part.field,&lt;br /&gt;
			langCode = self:forceLangByPart(part, source),&lt;br /&gt;
			fieldTable = fieldTable,&lt;br /&gt;
		}&lt;br /&gt;
		local othersText, othersWikitext = part.limits.replaceBy(source, processedData, cutText)&lt;br /&gt;
		text = text .. othersText&lt;br /&gt;
		wikitext = wikitext .. othersWikitext&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return text, wikitext&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local function fieldTableAndValueByPart(source, part)&lt;br /&gt;
	if part.entity or part.value then&lt;br /&gt;
		return { entity = part.entity, value = part.value }, part.value&lt;br /&gt;
	else&lt;br /&gt;
		return fieldTableByPath(source, part.field), fieldValueByPath(source, part.field)&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Formats the field and returns resulting text.&lt;br /&gt;
function Formatter:formatFieldComponents(source, part, state)&lt;br /&gt;
	local fieldTable, fieldValue = fieldTableAndValueByPart(source, part)&lt;br /&gt;
	if type(fieldTable) ~= &amp;#039;table&amp;#039; then&lt;br /&gt;
		error(&amp;#039;Field &amp;#039; .. dump(part.field) .. &amp;#039; is not a table. Its type is &amp;#039; .. type(fieldTable))&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local text&lt;br /&gt;
	local wikitext&lt;br /&gt;
	if table.getn(fieldTable) &amp;gt; 0 then&lt;br /&gt;
		text, wikitext = self:formatFieldAsArray(source, part, fieldTable, state)&lt;br /&gt;
	else&lt;br /&gt;
		state.index = 1&lt;br /&gt;
		local result = self:formatField(source, part, fieldTable, fieldValue, state)&lt;br /&gt;
		text = result.text&lt;br /&gt;
		wikitext = result.wikitext&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return { text = text, wikitext = wikitext }&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Format single field by its data.&lt;br /&gt;
function Formatter:commonFormatField(part, parentContainer, source)&lt;br /&gt;
	local result = self:formatFieldComponents(source, part, self.state)&lt;br /&gt;
&lt;br /&gt;
	parentContainer:wikitext(result.wikitext)&lt;br /&gt;
	self.lastAddedText = result.text&lt;br /&gt;
&lt;br /&gt;
	self.state.empty = false&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:ensureEndsWith(endsText, parentContainer)&lt;br /&gt;
	if self.lastAddedText and endsText then&lt;br /&gt;
		local len = mw.ustring.len(endsText)&lt;br /&gt;
		if mw.ustring.sub(self.lastAddedText, -len) ~= endsText then&lt;br /&gt;
			parentContainer:wikitext(endsText)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:ensureEndsAndAddDelimiter(group, part, parentContainer, groupEmpty, source)&lt;br /&gt;
	if self.state.empty or self.state.delimiterAdded then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local delimiter = part.delimiter&lt;br /&gt;
	local endsText = part.ensureEnds&lt;br /&gt;
	if (delimiter == nil or groupEmpty) and not part.forceDelimiter then&lt;br /&gt;
		if group.childDelimiter or group.childEnsureEnds then&lt;br /&gt;
			delimiter = group.childDelimiter&lt;br /&gt;
			endsText = group.childEnsureEnds&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if not delimiter and not endsText then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	self:ensureEndsWith(endsText, parentContainer)&lt;br /&gt;
&lt;br /&gt;
	if type(delimiter) == &amp;#039;function&amp;#039; then&lt;br /&gt;
		delimiter = delimiter(source)&lt;br /&gt;
	end&lt;br /&gt;
	parentContainer:wikitext(delimiter)&lt;br /&gt;
	self.state.delimiterAdded = true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Checks whether the field is set in the source.&lt;br /&gt;
local function fieldExists(source, path)&lt;br /&gt;
	local t = fieldTableByPath(source, path)&lt;br /&gt;
	if not t then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if type(path) == &amp;#039;table&amp;#039; and path.sub then&lt;br /&gt;
		if not t.value then&lt;br /&gt;
			return false&lt;br /&gt;
		end&lt;br /&gt;
		return (t.value[path.sub] ~= nil)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
local fieldConflicts, fieldDepends&lt;br /&gt;
&lt;br /&gt;
fieldConflicts = function(conflicts, source)&lt;br /&gt;
	if not conflicts then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if type(conflicts) == &amp;#039;table&amp;#039; and not conflicts.isPath then&lt;br /&gt;
		for _, currField in ipairs(conflicts) do&lt;br /&gt;
			if type(currField) == &amp;#039;string&amp;#039; or currField.isPath then&lt;br /&gt;
				if fieldExists(source, currField) then&lt;br /&gt;
					return true&lt;br /&gt;
				end&lt;br /&gt;
			else&lt;br /&gt;
				if fieldDepends(currField, source) then&lt;br /&gt;
					return true&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		return fieldExists(source, conflicts)&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	return false&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
fieldDepends = function(depends, source)&lt;br /&gt;
	if not depends then&lt;br /&gt;
		return true&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if type(depends) == &amp;#039;table&amp;#039; and not depends.isPath then&lt;br /&gt;
		for _, currField in ipairs(depends) do&lt;br /&gt;
			if type(currField) == &amp;#039;string&amp;#039; or currField.isPath then&lt;br /&gt;
				if not fieldExists(source, currField) then&lt;br /&gt;
					return false&lt;br /&gt;
				end&lt;br /&gt;
			else&lt;br /&gt;
				if not fieldConflicts(currField, source) then&lt;br /&gt;
					return false&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		return fieldExists(source, depends)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Look-a-head for groups by data row that can be displayed within&lt;br /&gt;
-- specified group. For the single field returns true if it can be displayed.&lt;br /&gt;
function Formatter:groupRowIsAvailable(group, childGroups, localSource)&lt;br /&gt;
	if table.getn(childGroups) &amp;gt; 0 then&lt;br /&gt;
		local found = false&lt;br /&gt;
		for _, subGroup in ipairs(childGroups) do&lt;br /&gt;
			if not subGroup.isStatic then&lt;br /&gt;
				if self:groupIsAvailable(subGroup, localSource) then&lt;br /&gt;
					found = true&lt;br /&gt;
					break&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
		if group.cond and not group.cond(localSource, self.params) then&lt;br /&gt;
			return false&lt;br /&gt;
		end&lt;br /&gt;
		return found&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if not fieldExists(localSource, group.field) and not group.entity and group.value == nil then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- TODO: тест на удаление этого условия&lt;br /&gt;
	if group.cond and not group.cond(localSource, self.params) then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return true&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Look-a-head for groups and data that can be displayed within specified group.&lt;br /&gt;
-- For the single field returns true if it can be displayed.&lt;br /&gt;
function Formatter:groupIsAvailable(group, source)&lt;br /&gt;
	if fieldConflicts(group.conflicts, source) then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if not fieldDepends(group.depends, source) then&lt;br /&gt;
		return false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local childGroups = group.groups or group&lt;br /&gt;
	local hasChildGroups = (table.getn(childGroups) &amp;gt; 0)&lt;br /&gt;
&lt;br /&gt;
	local localSource&lt;br /&gt;
	if hasChildGroups and group.field then&lt;br /&gt;
		localSource = source[group.field]&lt;br /&gt;
	else&lt;br /&gt;
		localSource = source&lt;br /&gt;
	end&lt;br /&gt;
	if not localSource then&lt;br /&gt;
		error(tostring(group.field))&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if table.getn(localSource) &amp;gt; 0 then&lt;br /&gt;
		local found = false&lt;br /&gt;
		if childGroups then&lt;br /&gt;
			for _, row in ipairs(localSource) do&lt;br /&gt;
				if not group.cond or group.cond(row, self.params) then&lt;br /&gt;
					for _, subGroup in ipairs(childGroups) do&lt;br /&gt;
						if not subGroup.isStatic then&lt;br /&gt;
							if self:groupIsAvailable(subGroup, row) then&lt;br /&gt;
								found = true&lt;br /&gt;
								break&lt;br /&gt;
							end&lt;br /&gt;
						end&lt;br /&gt;
					end&lt;br /&gt;
				end&lt;br /&gt;
				if found then&lt;br /&gt;
					break&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		else&lt;br /&gt;
			if not group.entity and group.value == nil then&lt;br /&gt;
				for _, row in ipairs(localSource) do&lt;br /&gt;
					if fieldExists(row, group.field) then&lt;br /&gt;
						found = true&lt;br /&gt;
						break&lt;br /&gt;
					end&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
		return found&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return self:groupRowIsAvailable(group, childGroups, source)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- Formats the group and puts it into the parentContainer.&lt;br /&gt;
function Formatter:formatGroup(parentGroup, group, parentContainer, source)&lt;br /&gt;
	if not self:groupIsAvailable(group, source) then&lt;br /&gt;
		return&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local state = self.state&lt;br /&gt;
&lt;br /&gt;
	local groupEmpty = state.groupEmpty&lt;br /&gt;
&lt;br /&gt;
	local childGroups = group.groups or group&lt;br /&gt;
	local hasChildGroups = (table.getn(childGroups) &amp;gt; 0)&lt;br /&gt;
	if hasChildGroups and not group.passthrough then&lt;br /&gt;
		state.groupEmpty = true&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local localSource&lt;br /&gt;
	if group.field and hasChildGroups then&lt;br /&gt;
		localSource = source[group.field]&lt;br /&gt;
	else&lt;br /&gt;
		localSource = source&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local prefix = group.prefix&lt;br /&gt;
	if parentContainer then&lt;br /&gt;
		self:ensureEndsAndAddDelimiter(parentGroup, group, parentContainer, state.groupEmpty, localSource)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if prefix and parentContainer then&lt;br /&gt;
		parentContainer:wikitext(prefix)&lt;br /&gt;
		self.lastAddedText = self.lastAddedText .. prefix&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local groupContainer&lt;br /&gt;
	if hasChildGroups then&lt;br /&gt;
		if table.getn(localSource) &amp;gt; 0 then&lt;br /&gt;
			groupContainer = parentContainer&lt;br /&gt;
			for _, row in ipairs(localSource) do&lt;br /&gt;
				if not group.cond or group.cond(row, self.params) then&lt;br /&gt;
					if self:groupRowIsAvailable(group, childGroups, row) then&lt;br /&gt;
						local currContainer = self:tagToContainer(group.tag, groupContainer, source) or parentContainer&lt;br /&gt;
						for _, part in ipairs(childGroups) do&lt;br /&gt;
							self:formatGroup(group, part, currContainer, row)&lt;br /&gt;
						end&lt;br /&gt;
					end&lt;br /&gt;
				end&lt;br /&gt;
			end&lt;br /&gt;
		else&lt;br /&gt;
			groupContainer = self:tagToContainer(group.tag, parentContainer, source) or parentContainer&lt;br /&gt;
			for _, part in ipairs(childGroups) do&lt;br /&gt;
				self:formatGroup(group, part, groupContainer, localSource)&lt;br /&gt;
			end&lt;br /&gt;
		end&lt;br /&gt;
	else&lt;br /&gt;
		groupContainer = self:tagToContainer(group.tag, parentContainer, source) or parentContainer&lt;br /&gt;
		self:commonFormatField(group, groupContainer, localSource)&lt;br /&gt;
		self.state.delimiterAdded = false&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	if not state.groupEmpty then&lt;br /&gt;
		local suffix = group.suffix&lt;br /&gt;
		if suffix and parentContainer then&lt;br /&gt;
			parentContainer:wikitext(suffix)&lt;br /&gt;
			self.lastAddedText = self.lastAddedText .. suffix&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return groupContainer&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:format()&lt;br /&gt;
	local rootContainer = self:formatGroup(nil, self.profile, self.container, self.source)&lt;br /&gt;
&lt;br /&gt;
	self:ensureEndsWith(self.profile.ensureEnds, rootContainer)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function Formatter:getAsText()&lt;br /&gt;
	return tostring(self.container:allDone())&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function p.format(profile, source, langCode, params)&lt;br /&gt;
		-- Temporary solution for backward compatibility&lt;br /&gt;
	if type(langCode) == &amp;#039;table&amp;#039; then&lt;br /&gt;
		params = langCode&lt;br /&gt;
		langCode = &amp;#039;ru&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	if not langCode and source.langCode then&lt;br /&gt;
		langCode = source.langCode.value&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local f = Formatter:new(profile, source, langCode, params)&lt;br /&gt;
	f:format()&lt;br /&gt;
	return f:getAsText()&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
p.f = Formatter.processField&lt;br /&gt;
&lt;br /&gt;
return p&lt;/div&gt;</summary>
		<author><name>Буквица&gt;The Fox Bot</name></author>
	</entry>
</feed>