<?xml version="1.0"?>
<feed xmlns="http://www.w3.org/2005/Atom" xml:lang="ja">
	<id>https://bsd.neuroinf.jp/w/index.php?action=history&amp;feed=atom&amp;title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB%3AComplex_date</id>
	<title>モジュール:Complex date - 版の履歴</title>
	<link rel="self" type="application/atom+xml" href="https://bsd.neuroinf.jp/w/index.php?action=history&amp;feed=atom&amp;title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB%3AComplex_date"/>
	<link rel="alternate" type="text/html" href="https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;action=history"/>
	<updated>2026-05-02T05:20:51Z</updated>
	<subtitle>このウィキのこのページに関する変更履歴</subtitle>
	<generator>MediaWiki 1.43.8</generator>
	<entry>
		<id>https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=43583&amp;oldid=prev</id>
		<title>Nijcadmin: 1版 をインポートしました</title>
		<link rel="alternate" type="text/html" href="https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=43583&amp;oldid=prev"/>
		<updated>2020-01-29T04:27: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;ja&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;2020年1月29日 (水) 13:27時点における版&lt;/td&gt;
				&lt;/tr&gt;&lt;tr&gt;&lt;td colspan=&quot;4&quot; class=&quot;diff-notice&quot; lang=&quot;ja&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-mw_:diff:1.41:old-43582:rev-43583 --&gt;
&lt;/table&gt;</summary>
		<author><name>Nijcadmin</name></author>
	</entry>
	<entry>
		<id>https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=43582&amp;oldid=prev</id>
		<title>bsd&gt;4nn1l2: Changed protection level for &quot;Module:Complex date&quot;: downgraded protection level per consensus ([Move=Allow only template editors and administrators] (indefinite) [Edit=Allow only template editors and administrators]…</title>
		<link rel="alternate" type="text/html" href="https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=43582&amp;oldid=prev"/>
		<updated>2019-07-16T11:17:34Z</updated>

		<summary type="html">&lt;p&gt;Changed protection level for &amp;quot;&lt;a href=&quot;/wiki/%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&quot; title=&quot;モジュール:Complex date&quot;&gt;Module:Complex date&lt;/a&gt;&amp;quot;: downgraded protection level per &lt;a href=&quot;/wiki/%E7%89%B9%E5%88%A5:%E5%9B%BA%E5%AE%9A%E3%83%AA%E3%83%B3%E3%82%AF/357548402&quot; title=&quot;特別:固定リンク/357548402&quot;&gt;consensus&lt;/a&gt; ([Move=Allow only template editors and administrators] (indefinite) [Edit=Allow only template editors and administrators]…&lt;/p&gt;
&lt;a href=&quot;https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;amp;diff=43582&amp;amp;oldid=35027&quot;&gt;差分を表示&lt;/a&gt;</summary>
		<author><name>bsd&gt;4nn1l2</name></author>
	</entry>
	<entry>
		<id>https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=35027&amp;oldid=prev</id>
		<title>WikiSysop: 1版</title>
		<link rel="alternate" type="text/html" href="https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=35027&amp;oldid=prev"/>
		<updated>2016-03-05T00:55:49Z</updated>

		<summary type="html">&lt;p&gt;1版&lt;/p&gt;
&lt;p&gt;&lt;b&gt;新規ページ&lt;/b&gt;&lt;/p&gt;&lt;div&gt;--[[  &lt;br /&gt;
 &lt;br /&gt;
This module is intended for creation of complex date phrases in variety of languages.&lt;br /&gt;
 &lt;br /&gt;
Once deployed, please do not modify this code without applying the changes first at Module:Complex date/sandbox and testing &lt;br /&gt;
at Module:Complex date/sandbox/testcases.&lt;br /&gt;
 &lt;br /&gt;
Authors and maintainers:&lt;br /&gt;
* User:Sn1per - first draft of the original version &lt;br /&gt;
* User:Jarekt - corrections and expansion of the original version &lt;br /&gt;
 &lt;br /&gt;
]]&lt;br /&gt;
&lt;br /&gt;
-- List of external modules and functions&lt;br /&gt;
local p = {Error = nil}&lt;br /&gt;
local i18n       = require(&amp;#039;Module:i18n/complex date&amp;#039;)&lt;br /&gt;
local langSwitch = require(&amp;#039;Module:Fallback&amp;#039;)._langSwitch&lt;br /&gt;
local ISOdate    = require(&amp;#039;Module:ISOdate&amp;#039;)._ISOdate&lt;br /&gt;
local Date       = require(&amp;#039;Module:Date&amp;#039;)._Date&lt;br /&gt;
local formatnum  = require(&amp;#039;Module:Formatnum&amp;#039;).formatNum&lt;br /&gt;
local linguistic = require(&amp;#039;Module:Linguistic&amp;#039;)&lt;br /&gt;
&lt;br /&gt;
-- ==================================================&lt;br /&gt;
-- === Internal functions ===========================&lt;br /&gt;
-- ==================================================&lt;br /&gt;
&lt;br /&gt;
function formatnum1(numStr, lang)&lt;br /&gt;
-- mostly require(&amp;#039;Module:Formatnum&amp;#039;).formatNum function used to translate a number to use different numeral characters, &lt;br /&gt;
-- except that it it does not call  that function unless the language is on the list &amp;quot;LList&amp;quot;&lt;br /&gt;
	local LList = {bn=1,bpy=1,kn=1,hi=1,mr=1,new=1,pa=1,gu=1,fa=1,glk=1,mzn=1,ur=1,ar=1,ckb=1,ks=1,lo=1,[&amp;#039;or&amp;#039;]=1,bo=1,[&amp;#039;ml-old&amp;#039;]=1,mn=1,te=1,th=1}&lt;br /&gt;
	if LList[lang] then -- call only when the language is on the list&lt;br /&gt;
		numStr = formatnum(numStr, lang, 1)&lt;br /&gt;
	end&lt;br /&gt;
	return numStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function getISODate(datestr, datetype, lang, num, case)&lt;br /&gt;
-- translate dates in the format YYYY, YYYY-MM, and YYYY-MM-DD&lt;br /&gt;
	if  not case and i18n.Translations[datetype] then&lt;br /&gt;
		-- look up the grammatical case needed and call ISOdate module&lt;br /&gt;
		local rec = langSwitch(i18n.Translations[datetype], lang)&lt;br /&gt;
		if type(rec)==&amp;#039;table&amp;#039; then&lt;br /&gt;
			case = rec.case[num]&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return ISOdate(datestr, lang, case, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
-- use tables in Module:i18n/complex date to translate a phrase&lt;br /&gt;
	if not i18n.Translations[operation] then&lt;br /&gt;
		p.Error = string.format(&amp;#039;&amp;lt;span style=&amp;quot;background-color:red;&amp;quot;&amp;gt;Error in [[Module:Complex date]]: input parameter &amp;quot;%s&amp;quot; is not recognized.&amp;lt;/span&amp;gt;&amp;#039;, operation or &amp;#039;nil&amp;#039;)&lt;br /&gt;
		return &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	local dateStr = langSwitch(i18n.Translations[operation], lang)&lt;br /&gt;
	if type(dateStr)==&amp;#039;table&amp;#039; then&lt;br /&gt;
		dateStr = dateStr[1]&lt;br /&gt;
	end&lt;br /&gt;
	if type(dateStr)==&amp;#039;function&amp;#039; then&lt;br /&gt;
		local success&lt;br /&gt;
		local nDates = i18n.Translations[operation][&amp;#039;nDates&amp;#039;]&lt;br /&gt;
		if nDates==2 then -- 2 date phrase&lt;br /&gt;
			success, dateStr = pcall(dateStr, date1, date2, state)&lt;br /&gt;
		else  -- 1 date phrase&lt;br /&gt;
			success, dateStr = pcall(dateStr, date1, state)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	if type(dateStr)==&amp;#039;string&amp;#039; then&lt;br /&gt;
		-- replace parts of the string &amp;#039;$date1&amp;#039; and &amp;#039;$date2&amp;#039; with date1 and date2 strings&lt;br /&gt;
		dateStr = mw.ustring.gsub(dateStr, &amp;#039;$date1&amp;#039;, date1)&lt;br /&gt;
		dateStr = mw.ustring.gsub(dateStr, &amp;#039;$date2&amp;#039;, date2)&lt;br /&gt;
	else&lt;br /&gt;
		-- Special case of more complex phrases that can be build out of simple phrases&lt;br /&gt;
		-- If complex case is not translated to &amp;quot;lang&amp;quot; than build it out of simpler ones&lt;br /&gt;
		local x = dateStr&lt;br /&gt;
		dateStr = p._complex_date(x.conj, x.adj1, date1, x.units1, x.era1, x.adj2, date2, x.units2, x.era2, lang, 2)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function oneDatePhrase(dateStr, adj, era, units, lang, num, case, state)&lt;br /&gt;
-- translate a single date phrase&lt;br /&gt;
	if num==2 then&lt;br /&gt;
		state.adj, state.era, state.units, state.precision = state.adj2, state.era2, state.units2, state.precision2 &lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- dateStr can have many forms: ISO date, year or a number for &lt;br /&gt;
	-- decade, century or millennium&lt;br /&gt;
	if units == &amp;#039;&amp;#039; then -- unit is &amp;quot;year&amp;quot;, &amp;quot;month&amp;quot;, &amp;quot;day&amp;quot;&lt;br /&gt;
		dateStr = getISODate(dateStr, adj, lang, num, case)&lt;br /&gt;
	else -- units is &amp;quot;decade&amp;quot;, &amp;quot;century&amp;quot;, &amp;quot;millennium&amp;#039;&amp;#039;&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, units, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add adjective (&amp;quot;early&amp;quot;, &amp;quot;mid&amp;quot;, etc.) or preposition (&amp;quot;before&amp;quot;, &amp;quot;after&amp;quot;, &lt;br /&gt;
	-- &amp;quot;circa&amp;quot;, etc.) to the date&lt;br /&gt;
	if adj ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, adj, lang, state)&lt;br /&gt;
	else -- only era?&lt;br /&gt;
		dateStr = formatnum1(dateStr, lang)&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add era&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function twoDatePhrase(date1, date2, state, lang)&lt;br /&gt;
-- translate a double date phrase&lt;br /&gt;
	local dateStr, case&lt;br /&gt;
	local era=&amp;#039;&amp;#039;&lt;br /&gt;
	if state.era1 == state.era2 then&lt;br /&gt;
		-- if both eras are the same than add it only once&lt;br /&gt;
		era = state.era1&lt;br /&gt;
		state.era1 = &amp;#039;&amp;#039;&lt;br /&gt;
		state.era2 = &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	case = {nil, nil}&lt;br /&gt;
	if i18n.Translations[state.conj] then&lt;br /&gt;
		local rec = langSwitch(i18n.Translations[state.conj], lang)&lt;br /&gt;
		if type(rec)==&amp;#039;table&amp;#039; then&lt;br /&gt;
			case = rec.case&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	date1   = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, case[1], state)&lt;br /&gt;
	date2   = oneDatePhrase(date2, state.adj2, state.era2, state.units2, lang, 2, case[2], state)&lt;br /&gt;
	dateStr = translatePhrase(date1, date2, state.conj, lang, state)&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function otherPhrases(date1, date2, operation, era, lang, state)&lt;br /&gt;
-- translate specialized phrases&lt;br /&gt;
	local dateStr = &amp;#039;&amp;#039;&lt;br /&gt;
		&lt;br /&gt;
	if operation == &amp;#039;islamic&amp;#039; then&lt;br /&gt;
		if date2==&amp;#039;&amp;#039; then date2 = mw.getCurrentFrame():callParserFunction(&amp;#039;#time&amp;#039;, &amp;#039;xmY&amp;#039;, date1) end&lt;br /&gt;
		date1 = getISODate(date1, operation, lang, 1, nil)&lt;br /&gt;
		date2 = getISODate(date2, operation, lang, 2, nil)&lt;br /&gt;
		if era == &amp;#039;&amp;#039; then era = &amp;#039;ad&amp;#039; end&lt;br /&gt;
		dateStr = translatePhrase(date1, &amp;#039;&amp;#039;, era, lang, state) .. &amp;#039; (&amp;#039; .. translatePhrase(date2, &amp;#039;&amp;#039;, &amp;#039;ah&amp;#039;, lang, state) .. &amp;#039;)&amp;#039;&lt;br /&gt;
		era = &amp;#039;&amp;#039;&lt;br /&gt;
	elseif operation == &amp;#039;julian&amp;#039; then&lt;br /&gt;
		date1 = getISODate(date1, operation, lang, 1, nil)&lt;br /&gt;
		date2 = getISODate(date2, operation, lang, 2, nil)&lt;br /&gt;
		dateStr = translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;turn of the year&amp;#039; or operation == &amp;#039;turn of the decade&amp;#039; or operation == &amp;#039;turn of the century&amp;#039; then &lt;br /&gt;
		if operation == &amp;#039;turn of the decade&amp;#039; then dt=10 else dt=1 end&lt;br /&gt;
		if not date2 or date2==&amp;#039;&amp;#039; then date2=tostring(tonumber(date1)-dt) end&lt;br /&gt;
		if era~=&amp;#039;bp&amp;#039; and era~=&amp;#039;bc&amp;#039; then date1, date2 = date2, date1 end&lt;br /&gt;
		if operation == &amp;#039;turn of the year&amp;#039; then&lt;br /&gt;
			date1 = ISOdate(date1, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
			date2 = ISOdate(date2, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
		else&lt;br /&gt;
			date1 = formatnum1(date1, lang)&lt;br /&gt;
			date2 = formatnum1(date2, lang)&lt;br /&gt;
		end&lt;br /&gt;
		dateStr = translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;year unknown&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(&amp;#039;&amp;#039;, &amp;#039;&amp;#039;, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;unknown&amp;#039; then&lt;br /&gt;
		dateStr = tostring(mw.message.new( &amp;quot;exif-unknowndate&amp;quot; ):inLanguage( lang ))&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add era&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function checkAliases(str1, str2, sType)&lt;br /&gt;
-- some inputs have many aliases - reconcile them and ensure string is playing a proper role	&lt;br /&gt;
local out = &amp;#039;&amp;#039;&lt;br /&gt;
if str1 and str1~=&amp;#039;&amp;#039; then&lt;br /&gt;
	a = i18n.Synonyms[str1] -- look up synonyms of &amp;quot;str1&amp;quot;&lt;br /&gt;
	if a then&lt;br /&gt;
		out = a[1]&lt;br /&gt;
	else&lt;br /&gt;
		p.Error = string.format(&amp;#039;&amp;lt;span style=&amp;quot;background-color:red;&amp;quot;&amp;gt;Error in [[Module:Complex date]]: %s is not recognized.&amp;lt;/span&amp;gt;&amp;#039;, str1)&lt;br /&gt;
	end&lt;br /&gt;
elseif str2 and str2~=&amp;#039;&amp;#039; then -- if &amp;quot;str1&amp;quot; of type &amp;quot;sType&amp;quot; is empty than maybe ...&lt;br /&gt;
	a = i18n.Synonyms[str2]   -- ...&amp;quot;str2&amp;quot; is of the same type and is not empty&lt;br /&gt;
	if a and a[2]==sType then&lt;br /&gt;
		out  = a[1]&lt;br /&gt;
		str2 = &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
return out, str2&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function datePrecision(dateStr, units)	&lt;br /&gt;
-- &amp;quot;in this module &amp;quot;Units&amp;quot; is a string like millennium, century, or decade&lt;br /&gt;
--	&amp;quot;precision&amp;quot; is wikibase compatible date precision number: 6=millennium, 7=century, 8=decade, 9=year, 10=month, 11=day&lt;br /&gt;
-- based on string or numeric input calculate &amp;quot;Units&amp;quot; and &amp;quot;precision&amp;quot;&lt;br /&gt;
	local dateNum = tonumber(dateStr);&lt;br /&gt;
	if type(units)==&amp;#039;number&amp;#039; then&lt;br /&gt;
		precision = units&lt;br /&gt;
		if precision&amp;gt;11 then precision=11 end -- clip the range of precision values&lt;br /&gt;
		if     precision==6 then units=&amp;#039;millennium&amp;#039; 		&lt;br /&gt;
		elseif precision==7 then units=&amp;#039;century&amp;#039;&lt;br /&gt;
		elseif precision==8 then units=&amp;#039;decade&amp;#039;&lt;br /&gt;
		else units = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	elseif type(units)==&amp;#039;string&amp;#039; then&lt;br /&gt;
		units = string.lower(units);&lt;br /&gt;
		if     units==&amp;#039;millennium&amp;#039; then precision=6&lt;br /&gt;
		elseif units==&amp;#039;century&amp;#039;    then precision=7&lt;br /&gt;
		elseif units==&amp;#039;decade&amp;#039;     then precision=8&lt;br /&gt;
		else precision=9&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if units==&amp;#039;&amp;#039; or precision==9 then&lt;br /&gt;
		local sLen = mw.ustring.len(dateStr)&lt;br /&gt;
		if     sLen&amp;lt;= 4 then precision=9&lt;br /&gt;
		elseif sLen== 7 then precision=10&lt;br /&gt;
		elseif sLen&amp;gt;=10 then precision=11&lt;br /&gt;
		end&lt;br /&gt;
		units=&amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	if precision==6 and dateStr.match( dateStr, &amp;#039;%d000&amp;#039; )~=nil then &lt;br /&gt;
		dateStr = tostring(math.floor(tonumber(dateStr)/1000) +1)&lt;br /&gt;
	elseif precision==7 and mw.ustring.match( dateStr, &amp;#039;%d%d00&amp;#039; )~=nil then&lt;br /&gt;
		dateStr = tostring(math.floor(tonumber(dateStr)/100) +1)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return dateStr, units, precision&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- ==================================================&lt;br /&gt;
-- === External functions ===========================&lt;br /&gt;
-- ==================================================&lt;br /&gt;
&lt;br /&gt;
function p.Era(frame)&lt;br /&gt;
    -- process inputs&lt;br /&gt;
	local dateStr&lt;br /&gt;
	local args    = frame.args&lt;br /&gt;
	local dateStr = args[&amp;#039;date&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local eraType = string.lower(args[&amp;#039;era&amp;#039;]  or &amp;#039;&amp;#039;)&lt;br /&gt;
	local lang    = args[&amp;#039;lang&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	if lang == &amp;#039;&amp;#039; then&lt;br /&gt;
		lang = mw.message.new( &amp;quot;lang&amp;quot; ):plain()&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	dateStr = ISOdate(dateStr, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
	if eraType then &lt;br /&gt;
		eraType = checkAliases(eraType ,&amp;#039;&amp;#039;,&amp;#039;e&amp;#039;)&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, eraType, lang, {}) &lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function p.complex_date(frame)&lt;br /&gt;
    -- process inputs&lt;br /&gt;
	local dateStr, Error&lt;br /&gt;
	local args   = frame.args&lt;br /&gt;
	local date1  = args[&amp;#039;date1&amp;#039;] or args[&amp;#039;2&amp;#039;] or args[&amp;#039;date&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local date2  = args[&amp;#039;date2&amp;#039;] or args[&amp;#039;3&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local conj   = args[&amp;#039;conj&amp;#039;]  or args[&amp;#039;1&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local adj1   = args[&amp;#039;adj1&amp;#039;]  or args[&amp;#039;adj&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local adj2   = args[&amp;#039;adj2&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local units1 = args[&amp;#039;precision1&amp;#039;] or args[&amp;#039;precision&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local units2 = args[&amp;#039;precision2&amp;#039;] or args[&amp;#039;precision&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local era1   = args[&amp;#039;era1&amp;#039;] or args[&amp;#039;era&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local era2   = args[&amp;#039;era2&amp;#039;] or args[&amp;#039;era&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local lang   = args[&amp;#039;lang&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	if lang == &amp;#039;&amp;#039; then&lt;br /&gt;
		lang = mw.message.new( &amp;quot;lang&amp;quot; ):plain()&lt;br /&gt;
	end&lt;br /&gt;
	dateStr = p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, 1)&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		dateStr = p.Error .. &amp;#039;[[Category:Pages using Complex date template with incorrect parameter]]&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
	&lt;br /&gt;
function p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, passNr)&lt;br /&gt;
	local Output=&amp;#039;&amp;#039;&lt;br /&gt;
&lt;br /&gt;
    -- process inputs and save date in state array&lt;br /&gt;
	local state  = {} &lt;br /&gt;
	state.conj   = string.lower(conj   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.adj1   = string.lower(adj1   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.adj2   = string.lower(adj2   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.era1   = string.lower(era1   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.era2   = string.lower(era2   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.units1 = string.lower(units1 or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.units2 = string.lower(units2 or &amp;#039;&amp;#039;)&lt;br /&gt;
		  &lt;br /&gt;
	-- if date 1 is missing but date 2 is provided than swap them&lt;br /&gt;
	if date1 == &amp;#039;&amp;#039; and date2 ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		date1 = date2&lt;br /&gt;
		date2 = &amp;#039;&amp;#039;&lt;br /&gt;
		state = {adj1 = state.adj2, era1 = state.era2, units1 = state.units2, &lt;br /&gt;
		         adj2 = &amp;#039;&amp;#039;,         era2 = &amp;#039;&amp;#039;,         units2 = &amp;#039;&amp;#039;,  conj=state.conj, num=1}&lt;br /&gt;
	end&lt;br /&gt;
	if     date2 ~= &amp;#039;&amp;#039; then state.nDates = 2 &lt;br /&gt;
	elseif date1 ~= &amp;#039;&amp;#039; then state.nDates = 1 &lt;br /&gt;
	else	                state.nDates = 0&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- reconcile alternative names for text inputs&lt;br /&gt;
	local conj         = checkAliases(state.conj ,&amp;#039;&amp;#039;  ,&amp;#039;j&amp;#039;)&lt;br /&gt;
	state.adj1 ,conj   = checkAliases(state.adj1 ,conj,&amp;#039;a&amp;#039;)&lt;br /&gt;
	state.units1,conj  = checkAliases(state.units1,conj,&amp;#039;p&amp;#039;)&lt;br /&gt;
	state.era1 ,conj   = checkAliases(state.era1 ,conj,&amp;#039;e&amp;#039;)&lt;br /&gt;
	state.special,conj = checkAliases(&amp;#039;&amp;#039;,conj,&amp;#039;c&amp;#039;)&lt;br /&gt;
	state.adj2         = checkAliases(state.adj2 ,&amp;#039;&amp;#039;,&amp;#039;a&amp;#039;)&lt;br /&gt;
	state.units2       = checkAliases(state.units2,&amp;#039;&amp;#039;,&amp;#039;p&amp;#039;)&lt;br /&gt;
	state.era2         = checkAliases(state.era2 ,&amp;#039;&amp;#039;,&amp;#039;e&amp;#039;)&lt;br /&gt;
	state.conj         = conj&lt;br /&gt;
	state.lang         = lang&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- calculate date precision value&lt;br /&gt;
	date1, state.units1, state.precision1 = datePrecision(date1, state.units1)&lt;br /&gt;
	date2, state.units2, state.precision2 = datePrecision(date2, state.units2)&lt;br /&gt;
&lt;br /&gt;
	-- Handle special cases &lt;br /&gt;
	-- Some complex phrases can be created out of simpler ones. Therefore on pass # 1 we try to create &lt;br /&gt;
	-- the phrase using complex phrase and if that is not found than on the second pass we try to build&lt;br /&gt;
	-- the phrase out of the simpler ones&lt;br /&gt;
	if passNr==1 then&lt;br /&gt;
		if state.adj1==&amp;#039;circa&amp;#039; and state.nDates == 2 then&lt;br /&gt;
			state.conj = &amp;#039;circa2&amp;#039;&lt;br /&gt;
			state.adj1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.adj2 = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		if state.nDates == 2 and state.adj1==&amp;#039;late&amp;#039; and state.adj2==&amp;#039;early&amp;#039; and state.conj==&amp;#039;and&amp;#039; &lt;br /&gt;
		and state.units1==state.units2 and state.era1==state.era2 then&lt;br /&gt;
			if state.units1==&amp;#039;century&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the century&amp;#039;&lt;br /&gt;
			elseif state.units1==&amp;#039;decade&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the decade&amp;#039;&lt;br /&gt;
			elseif state.units1==&amp;#039;&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the year&amp;#039;&lt;br /&gt;
			end&lt;br /&gt;
			state.adj1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.adj2 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.units1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.units2 = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local errorStr = string.format(&lt;br /&gt;
	 &amp;#039;\n*conj=%s, adj1=%s, era1=%s, unit1=%s, prec1=%i, adj2=%s, era2=%s, unit2=%s, prec2=%i, special=%s&amp;#039;, &lt;br /&gt;
	  state.conj, state.adj1, state.era1, state.units1, state.precision1,&lt;br /&gt;
	  state.adj2, state.era2, state.units2, state.precision2, state.special)  &lt;br /&gt;
	state.adj, state.era, state.units, state.precision = state.adj1, state.era1, state.units1, state.precision1 &lt;br /&gt;
&lt;br /&gt;
	-- call specialized functions&lt;br /&gt;
	if state.special~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = otherPhrases(date1, date2, state.special, state.era1, lang, state)	&lt;br /&gt;
	elseif state.conj~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = twoDatePhrase(date1, date2, state, lang)&lt;br /&gt;
	elseif state.adj1~=&amp;#039;&amp;#039; or state.era1~=&amp;#039;&amp;#039; or state.units1~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, nil, state)&lt;br /&gt;
	elseif date1~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = ISOdate(date1, lang, &amp;#039;&amp;#039;, &amp;#039;dtstart&amp;#039;, &amp;#039;100-999&amp;#039;)&lt;br /&gt;
	end&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		return errorStr&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	-- if there is any wikicode in the string than execute it&lt;br /&gt;
	if mw.ustring.find(Output, &amp;#039;{&amp;#039;) then&lt;br /&gt;
		Output = mw.getCurrentFrame():preprocess(Output)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return Output&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
return p&lt;/div&gt;</summary>
		<author><name>WikiSysop</name></author>
	</entry>
	<entry>
		<id>https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=30434&amp;oldid=prev</id>
		<title>WikiSysop: 1版</title>
		<link rel="alternate" type="text/html" href="https://bsd.neuroinf.jp/w/index.php?title=%E3%83%A2%E3%82%B8%E3%83%A5%E3%83%BC%E3%83%AB:Complex_date&amp;diff=30434&amp;oldid=prev"/>
		<updated>2015-06-16T22:32:16Z</updated>

		<summary type="html">&lt;p&gt;1版&lt;/p&gt;
&lt;p&gt;&lt;b&gt;新規ページ&lt;/b&gt;&lt;/p&gt;&lt;div&gt;--[[  &lt;br /&gt;
 &lt;br /&gt;
This module is intended for creation of complex date phrases in variety of languages.&lt;br /&gt;
 &lt;br /&gt;
Once deployed, please do not modify this code without applying the changes first at Module:Complex date/sandbox and testing &lt;br /&gt;
at Module:Complex date/sandbox/testcases.&lt;br /&gt;
 &lt;br /&gt;
Authors and maintainers:&lt;br /&gt;
* User:Sn1per - first draft of the original version &lt;br /&gt;
* User:Jarekt - corrections and expansion of the original version &lt;br /&gt;
 &lt;br /&gt;
]]&lt;br /&gt;
&lt;br /&gt;
-- List of external modules and functions&lt;br /&gt;
local p = {Error = nil}&lt;br /&gt;
local i18n       = require(&amp;#039;Module:i18n/complex date&amp;#039;)&lt;br /&gt;
local langSwitch = require(&amp;#039;Module:Fallback&amp;#039;)._langSwitch&lt;br /&gt;
local ISOdate    = require(&amp;#039;Module:ISOdate&amp;#039;)._ISOdate&lt;br /&gt;
local Date       = require(&amp;#039;Module:Date&amp;#039;)._Date&lt;br /&gt;
local formatnum  = require(&amp;#039;Module:Formatnum &amp;#039;)._main&lt;br /&gt;
local linguistic = require(&amp;#039;Module:Linguistic&amp;#039;)&lt;br /&gt;
&lt;br /&gt;
-- ==================================================&lt;br /&gt;
-- === Internal functions ===========================&lt;br /&gt;
-- ==================================================&lt;br /&gt;
&lt;br /&gt;
function formatnum1(numStr, lang)&lt;br /&gt;
-- mostly require(&amp;#039;Module:Formatnum &amp;#039;)._main function used to translate a number to use different numeral characters, &lt;br /&gt;
-- except that it it does not call  that function unless the language is on the list &amp;quot;LList&amp;quot;&lt;br /&gt;
	local LList = {bn=1,bpy=1,kn=1,hi=1,mr=1,new=1,pa=1,gu=1,fa=1,glk=1,mzn=1,ur=1,ar=1,ckb=1,ks=1,lo=1,[&amp;#039;or&amp;#039;]=1,bo=1,[&amp;#039;ml-old&amp;#039;]=1,mn=1,te=1,th=1}&lt;br /&gt;
	if LList[lang] then -- call only when the language is on the list&lt;br /&gt;
		numStr = formatnum(numStr, lang, 1)&lt;br /&gt;
	end&lt;br /&gt;
	return numStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function getISODate(datestr, datetype, lang, num, case)&lt;br /&gt;
-- translate dates in the format YYYY, YYYY-MM, and YYYY-MM-DD&lt;br /&gt;
	if  not case and i18n.Translations[datetype] then&lt;br /&gt;
		-- look up the grammatical case needed and call ISOdate module&lt;br /&gt;
		local rec = langSwitch(i18n.Translations[datetype], lang)&lt;br /&gt;
		if type(rec)==&amp;#039;table&amp;#039; then&lt;br /&gt;
			case = rec.case[num]&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	return ISOdate(datestr, lang, case, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
-- use tables in Module:i18n/complex date to translate a phrase&lt;br /&gt;
	if not i18n.Translations[operation] then&lt;br /&gt;
		p.Error = string.format(&amp;#039;&amp;lt;span style=&amp;quot;background-color:red;&amp;quot;&amp;gt;Error in [[Module:Complex date]]: input parameter &amp;quot;%s&amp;quot; is not recognized.&amp;lt;/span&amp;gt;&amp;#039;, operation or &amp;#039;nil&amp;#039;)&lt;br /&gt;
		return &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	local dateStr = langSwitch(i18n.Translations[operation], lang)&lt;br /&gt;
	if type(dateStr)==&amp;#039;table&amp;#039; then&lt;br /&gt;
		dateStr = dateStr[1]&lt;br /&gt;
	end&lt;br /&gt;
	if type(dateStr)==&amp;#039;function&amp;#039; then&lt;br /&gt;
		local success&lt;br /&gt;
		local nDates = i18n.Translations[operation][&amp;#039;nDates&amp;#039;]&lt;br /&gt;
		if nDates==2 then -- 2 date phrase&lt;br /&gt;
			success, dateStr = pcall(dateStr, date1, date2, state)&lt;br /&gt;
		else  -- 1 date phrase&lt;br /&gt;
			success, dateStr = pcall(dateStr, date1, state)&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	if type(dateStr)==&amp;#039;string&amp;#039; then&lt;br /&gt;
		-- replace parts of the string &amp;#039;$date1&amp;#039; and &amp;#039;$date2&amp;#039; with date1 and date2 strings&lt;br /&gt;
		dateStr = mw.ustring.gsub(dateStr, &amp;#039;$date1&amp;#039;, date1)&lt;br /&gt;
		dateStr = mw.ustring.gsub(dateStr, &amp;#039;$date2&amp;#039;, date2)&lt;br /&gt;
	else&lt;br /&gt;
		-- Special case of more complex phrases that can be build out of simple phrases&lt;br /&gt;
		-- If complex case is not translated to &amp;quot;lang&amp;quot; than build it out of simpler ones&lt;br /&gt;
		local x = dateStr&lt;br /&gt;
		dateStr = p._complex_date(x.conj, x.adj1, date1, x.units1, x.era1, x.adj2, date2, x.units2, x.era2, lang, 2)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function oneDatePhrase(dateStr, adj, era, units, lang, num, case, state)&lt;br /&gt;
-- translate a single date phrase&lt;br /&gt;
	if num==2 then&lt;br /&gt;
		state.adj, state.era, state.units, state.precision = state.adj2, state.era2, state.units2, state.precision2 &lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- dateStr can have many forms: ISO date, year or a number for &lt;br /&gt;
	-- decade, century or millennium&lt;br /&gt;
	if units == &amp;#039;&amp;#039; then -- unit is &amp;quot;year&amp;quot;, &amp;quot;month&amp;quot;, &amp;quot;day&amp;quot;&lt;br /&gt;
		dateStr = getISODate(dateStr, adj, lang, num, case)&lt;br /&gt;
	else -- units is &amp;quot;decade&amp;quot;, &amp;quot;century&amp;quot;, &amp;quot;millennium&amp;#039;&amp;#039;&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, units, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add adjective (&amp;quot;early&amp;quot;, &amp;quot;mid&amp;quot;, etc.) or preposition (&amp;quot;before&amp;quot;, &amp;quot;after&amp;quot;, &lt;br /&gt;
	-- &amp;quot;circa&amp;quot;, etc.) to the date&lt;br /&gt;
	if adj ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, adj, lang, state)&lt;br /&gt;
	else -- only era?&lt;br /&gt;
		dateStr = formatnum1(dateStr, lang)&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add era&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function twoDatePhrase(date1, date2, state, lang)&lt;br /&gt;
-- translate a double date phrase&lt;br /&gt;
	local dateStr, case&lt;br /&gt;
	local era=&amp;#039;&amp;#039;&lt;br /&gt;
	if state.era1 == state.era2 then&lt;br /&gt;
		-- if both eras are the same than add it only once&lt;br /&gt;
		era = state.era1&lt;br /&gt;
		state.era1 = &amp;#039;&amp;#039;&lt;br /&gt;
		state.era2 = &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	case = {nil, nil}&lt;br /&gt;
	if i18n.Translations[state.conj] then&lt;br /&gt;
		local rec = langSwitch(i18n.Translations[state.conj], lang)&lt;br /&gt;
		if type(rec)==&amp;#039;table&amp;#039; then&lt;br /&gt;
			case = rec.case&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	date1   = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, case[1], state)&lt;br /&gt;
	date2   = oneDatePhrase(date2, state.adj2, state.era2, state.units2, lang, 2, case[2], state)&lt;br /&gt;
	dateStr = translatePhrase(date1, date2, state.conj, lang, state)&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function otherPhrases(date1, date2, operation, era, lang, state)&lt;br /&gt;
-- translate specialized phrases&lt;br /&gt;
	local dateStr = &amp;#039;&amp;#039;&lt;br /&gt;
		&lt;br /&gt;
	if operation == &amp;#039;islamic&amp;#039; then&lt;br /&gt;
		if date2==&amp;#039;&amp;#039; then date2 = mw.getCurrentFrame():callParserFunction(&amp;#039;#time&amp;#039;, &amp;#039;xmY&amp;#039;, date1) end&lt;br /&gt;
		date1 = getISODate(date1, operation, lang, 1, nil)&lt;br /&gt;
		date2 = getISODate(date2, operation, lang, 2, nil)&lt;br /&gt;
		if era == &amp;#039;&amp;#039; then era = &amp;#039;ad&amp;#039; end&lt;br /&gt;
		dateStr = translatePhrase(date1, &amp;#039;&amp;#039;, era, lang, state) .. &amp;#039; (&amp;#039; .. translatePhrase(date2, &amp;#039;&amp;#039;, &amp;#039;ah&amp;#039;, lang, state) .. &amp;#039;)&amp;#039;&lt;br /&gt;
		era = &amp;#039;&amp;#039;&lt;br /&gt;
	elseif operation == &amp;#039;julian&amp;#039; then&lt;br /&gt;
		date1 = getISODate(date1, operation, lang, 1, nil)&lt;br /&gt;
		date2 = getISODate(date2, operation, lang, 2, nil)&lt;br /&gt;
		dateStr = translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;turn of the year&amp;#039; or operation == &amp;#039;turn of the decade&amp;#039; or operation == &amp;#039;turn of the century&amp;#039; then &lt;br /&gt;
		if operation == &amp;#039;turn of the decade&amp;#039; then dt=10 else dt=1 end&lt;br /&gt;
		if not date2 or date2==&amp;#039;&amp;#039; then date2=tostring(tonumber(date1)-dt) end&lt;br /&gt;
		if era~=&amp;#039;bp&amp;#039; and era~=&amp;#039;bc&amp;#039; then date1, date2 = date2, date1 end&lt;br /&gt;
		if operation == &amp;#039;turn of the year&amp;#039; then&lt;br /&gt;
			date1 = ISOdate(date1, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
			date2 = ISOdate(date2, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
		else&lt;br /&gt;
			date1 = formatnum1(date1, lang)&lt;br /&gt;
			date2 = formatnum1(date2, lang)&lt;br /&gt;
		end&lt;br /&gt;
		dateStr = translatePhrase(date1, date2, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;year unknown&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(&amp;#039;&amp;#039;, &amp;#039;&amp;#039;, operation, lang, state)&lt;br /&gt;
	elseif operation == &amp;#039;unknown&amp;#039; then&lt;br /&gt;
		dateStr = tostring(mw.message.new( &amp;quot;exif-unknowndate&amp;quot; ):inLanguage( lang ))&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- add era&lt;br /&gt;
	if era ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, era, lang, state)&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function checkAliases(str1, str2, sType)&lt;br /&gt;
-- some inputs have many aliases - reconcile them and ensure string is playing a proper role	&lt;br /&gt;
local out = &amp;#039;&amp;#039;&lt;br /&gt;
if str1 and str1~=&amp;#039;&amp;#039; then&lt;br /&gt;
	a = i18n.Synonyms[str1] -- look up synonyms of &amp;quot;str1&amp;quot;&lt;br /&gt;
	if a then&lt;br /&gt;
		out = a[1]&lt;br /&gt;
	else&lt;br /&gt;
		p.Error = string.format(&amp;#039;&amp;lt;span style=&amp;quot;background-color:red;&amp;quot;&amp;gt;Error in [[Module:Complex date]]: %s is not recognized.&amp;lt;/span&amp;gt;&amp;#039;, str1)&lt;br /&gt;
	end&lt;br /&gt;
elseif str2 and str2~=&amp;#039;&amp;#039; then -- if &amp;quot;str1&amp;quot; of type &amp;quot;sType&amp;quot; is empty than maybe ...&lt;br /&gt;
	a = i18n.Synonyms[str2]   -- ...&amp;quot;str2&amp;quot; is of the same type and is not empty&lt;br /&gt;
	if a and a[2]==sType then&lt;br /&gt;
		out  = a[1]&lt;br /&gt;
		str2 = &amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
end&lt;br /&gt;
return out, str2&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function datePrecision(dateStr, units)	&lt;br /&gt;
-- &amp;quot;in this module &amp;quot;Units&amp;quot; is a string like millennium, century, or decade&lt;br /&gt;
--	&amp;quot;precision&amp;quot; is wikibase compatible date precision number: 6=millennium, 7=century, 8=decade, 9=year, 10=month, 11=day&lt;br /&gt;
-- based on string or numeric input calculate &amp;quot;Units&amp;quot; and &amp;quot;precision&amp;quot;&lt;br /&gt;
	local dateNum = tonumber(dateStr);&lt;br /&gt;
	if type(units)==&amp;#039;number&amp;#039; then&lt;br /&gt;
		precision = units&lt;br /&gt;
		if precision&amp;gt;11 then precision=11 end -- clip the range of precision values&lt;br /&gt;
		if     precision==6 then units=&amp;#039;millennium&amp;#039; 		&lt;br /&gt;
		elseif precision==7 then units=&amp;#039;century&amp;#039;&lt;br /&gt;
		elseif precision==8 then units=&amp;#039;decade&amp;#039;&lt;br /&gt;
		else units = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	elseif type(units)==&amp;#039;string&amp;#039; then&lt;br /&gt;
		units = string.lower(units);&lt;br /&gt;
		if     units==&amp;#039;millennium&amp;#039; then precision=6&lt;br /&gt;
		elseif units==&amp;#039;century&amp;#039;    then precision=7&lt;br /&gt;
		elseif units==&amp;#039;decade&amp;#039;     then precision=8&lt;br /&gt;
		else precision=9&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
	if units==&amp;#039;&amp;#039; or precision==9 then&lt;br /&gt;
		local sLen = mw.ustring.len(dateStr)&lt;br /&gt;
		if     sLen&amp;lt;= 4 then precision=9&lt;br /&gt;
		elseif sLen== 7 then precision=10&lt;br /&gt;
		elseif sLen&amp;gt;=10 then precision=11&lt;br /&gt;
		end&lt;br /&gt;
		units=&amp;#039;&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	if precision==6 and dateStr.match( dateStr, &amp;#039;%d000&amp;#039; )~=nil then &lt;br /&gt;
		dateStr = tostring(math.floor(tonumber(dateStr)/1000) +1)&lt;br /&gt;
	elseif precision==7 and mw.ustring.match( dateStr, &amp;#039;%d%d00&amp;#039; )~=nil then&lt;br /&gt;
		dateStr = tostring(math.floor(tonumber(dateStr)/100) +1)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return dateStr, units, precision&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
-- ==================================================&lt;br /&gt;
-- === External functions ===========================&lt;br /&gt;
-- ==================================================&lt;br /&gt;
&lt;br /&gt;
function p.Era(frame)&lt;br /&gt;
    -- process inputs&lt;br /&gt;
	local dateStr&lt;br /&gt;
	local args    = frame.args&lt;br /&gt;
	local dateStr = args[&amp;#039;date&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local eraType = string.lower(args[&amp;#039;era&amp;#039;]  or &amp;#039;&amp;#039;)&lt;br /&gt;
	local lang    = args[&amp;#039;lang&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	if lang == &amp;#039;&amp;#039; then&lt;br /&gt;
		lang = mw.message.new( &amp;quot;lang&amp;quot; ):plain()&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	dateStr = ISOdate(dateStr, lang, &amp;#039;&amp;#039;, &amp;#039;&amp;#039;, 1)&lt;br /&gt;
	if eraType then &lt;br /&gt;
		eraType = checkAliases(eraType ,&amp;#039;&amp;#039;,&amp;#039;e&amp;#039;)&lt;br /&gt;
		dateStr = translatePhrase(dateStr, &amp;#039;&amp;#039;, eraType, lang, {}) &lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
function p.complex_date(frame)&lt;br /&gt;
    -- process inputs&lt;br /&gt;
	local dateStr, Error&lt;br /&gt;
	local args   = frame.args&lt;br /&gt;
	local date1  = args[&amp;#039;date1&amp;#039;] or args[&amp;#039;2&amp;#039;] or args[&amp;#039;date&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local date2  = args[&amp;#039;date2&amp;#039;] or args[&amp;#039;3&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local conj   = args[&amp;#039;conj&amp;#039;]  or args[&amp;#039;1&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local adj1   = args[&amp;#039;adj1&amp;#039;]  or args[&amp;#039;adj&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local adj2   = args[&amp;#039;adj2&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local units1 = args[&amp;#039;precision1&amp;#039;] or args[&amp;#039;precision&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local units2 = args[&amp;#039;precision2&amp;#039;] or args[&amp;#039;precision&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local era1   = args[&amp;#039;era1&amp;#039;] or args[&amp;#039;era&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local era2   = args[&amp;#039;era2&amp;#039;] or args[&amp;#039;era&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	local lang   = args[&amp;#039;lang&amp;#039;] or &amp;#039;&amp;#039;&lt;br /&gt;
	if lang == &amp;#039;&amp;#039; then&lt;br /&gt;
		lang = mw.message.new( &amp;quot;lang&amp;quot; ):plain()&lt;br /&gt;
	end&lt;br /&gt;
	dateStr = p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, 1)&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		dateStr = p.Error .. &amp;#039;[[Category:Pages using Complex date template with incorrect parameter]]&amp;#039;&lt;br /&gt;
	end&lt;br /&gt;
	return dateStr&lt;br /&gt;
end&lt;br /&gt;
	&lt;br /&gt;
function p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, passNr)&lt;br /&gt;
	local Output=&amp;#039;&amp;#039;&lt;br /&gt;
&lt;br /&gt;
    -- process inputs and save date in state array&lt;br /&gt;
	local state  = {} &lt;br /&gt;
	state.conj   = string.lower(conj   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.adj1   = string.lower(adj1   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.adj2   = string.lower(adj2   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.era1   = string.lower(era1   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.era2   = string.lower(era2   or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.units1 = string.lower(units1 or &amp;#039;&amp;#039;)&lt;br /&gt;
	state.units2 = string.lower(units2 or &amp;#039;&amp;#039;)&lt;br /&gt;
		  &lt;br /&gt;
	-- if date 1 is missing but date 2 is provided than swap them&lt;br /&gt;
	if date1 == &amp;#039;&amp;#039; and date2 ~= &amp;#039;&amp;#039; then&lt;br /&gt;
		date1 = date2&lt;br /&gt;
		date2 = &amp;#039;&amp;#039;&lt;br /&gt;
		state = {adj1 = state.adj2, era1 = state.era2, units1 = state.units2, &lt;br /&gt;
		         adj2 = &amp;#039;&amp;#039;,         era2 = &amp;#039;&amp;#039;,         units2 = &amp;#039;&amp;#039;,  conj=state.conj, num=1}&lt;br /&gt;
	end&lt;br /&gt;
	if     date2 ~= &amp;#039;&amp;#039; then state.nDates = 2 &lt;br /&gt;
	elseif date1 ~= &amp;#039;&amp;#039; then state.nDates = 1 &lt;br /&gt;
	else	                state.nDates = 0&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- reconcile alternative names for text inputs&lt;br /&gt;
	local conj         = checkAliases(state.conj ,&amp;#039;&amp;#039;  ,&amp;#039;j&amp;#039;)&lt;br /&gt;
	state.adj1 ,conj   = checkAliases(state.adj1 ,conj,&amp;#039;a&amp;#039;)&lt;br /&gt;
	state.units1,conj  = checkAliases(state.units1,conj,&amp;#039;p&amp;#039;)&lt;br /&gt;
	state.era1 ,conj   = checkAliases(state.era1 ,conj,&amp;#039;e&amp;#039;)&lt;br /&gt;
	state.special,conj = checkAliases(&amp;#039;&amp;#039;,conj,&amp;#039;c&amp;#039;)&lt;br /&gt;
	state.adj2         = checkAliases(state.adj2 ,&amp;#039;&amp;#039;,&amp;#039;a&amp;#039;)&lt;br /&gt;
	state.units2       = checkAliases(state.units2,&amp;#039;&amp;#039;,&amp;#039;p&amp;#039;)&lt;br /&gt;
	state.era2         = checkAliases(state.era2 ,&amp;#039;&amp;#039;,&amp;#039;e&amp;#039;)&lt;br /&gt;
	state.conj         = conj&lt;br /&gt;
	state.lang         = lang&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		return nil&lt;br /&gt;
	end&lt;br /&gt;
	&lt;br /&gt;
	-- calculate date precision value&lt;br /&gt;
	date1, state.units1, state.precision1 = datePrecision(date1, state.units1)&lt;br /&gt;
	date2, state.units2, state.precision2 = datePrecision(date2, state.units2)&lt;br /&gt;
&lt;br /&gt;
	-- Handle special cases &lt;br /&gt;
	-- Some complex phrases can be created out of simpler ones. Therefore on pass # 1 we try to create &lt;br /&gt;
	-- the phrase using complex phrase and if that is not found than on the second pass we try to build&lt;br /&gt;
	-- the phrase out of the simpler ones&lt;br /&gt;
	if passNr==1 then&lt;br /&gt;
		if state.adj1==&amp;#039;circa&amp;#039; and state.nDates == 2 then&lt;br /&gt;
			state.conj = &amp;#039;circa2&amp;#039;&lt;br /&gt;
			state.adj1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.adj2 = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
		if state.nDates == 2 and state.adj1==&amp;#039;late&amp;#039; and state.adj2==&amp;#039;early&amp;#039; and state.conj==&amp;#039;and&amp;#039; &lt;br /&gt;
		and state.units1==state.units2 and state.era1==state.era2 then&lt;br /&gt;
			if state.units1==&amp;#039;century&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the century&amp;#039;&lt;br /&gt;
			elseif state.units1==&amp;#039;decade&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the decade&amp;#039;&lt;br /&gt;
			elseif state.units1==&amp;#039;&amp;#039; then&lt;br /&gt;
				state.conj=&amp;#039;turn of the year&amp;#039;&lt;br /&gt;
			end&lt;br /&gt;
			state.adj1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.adj2 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.units1 = &amp;#039;&amp;#039;&lt;br /&gt;
			state.units2 = &amp;#039;&amp;#039;&lt;br /&gt;
		end&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	local errorStr = string.format(&lt;br /&gt;
	 &amp;#039;\n*conj=%s, adj1=%s, era1=%s, unit1=%s, prec1=%i, adj2=%s, era2=%s, unit2=%s, prec2=%i, special=%s&amp;#039;, &lt;br /&gt;
	  state.conj, state.adj1, state.era1, state.units1, state.precision1,&lt;br /&gt;
	  state.adj2, state.era2, state.units2, state.precision2, state.special)  &lt;br /&gt;
	state.adj, state.era, state.units, state.precision = state.adj1, state.era1, state.units1, state.precision1 &lt;br /&gt;
&lt;br /&gt;
	-- call specialized functions&lt;br /&gt;
	if state.special~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = otherPhrases(date1, date2, state.special, state.era1, lang, state)	&lt;br /&gt;
	elseif state.conj~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = twoDatePhrase(date1, date2, state, lang)&lt;br /&gt;
	elseif state.adj1~=&amp;#039;&amp;#039; or state.era1~=&amp;#039;&amp;#039; or state.units1~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, nil, state)&lt;br /&gt;
	elseif date1~=&amp;#039;&amp;#039; then&lt;br /&gt;
		Output = ISOdate(date1, lang, &amp;#039;&amp;#039;, &amp;#039;dtstart&amp;#039;, &amp;#039;100-999&amp;#039;)&lt;br /&gt;
	end&lt;br /&gt;
	if p.Error~=nil then&lt;br /&gt;
		return errorStr&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	-- if there is any wikicode in the string than execute it&lt;br /&gt;
	if mw.ustring.find(Output, &amp;#039;{&amp;#039;) then&lt;br /&gt;
		Output = mw.getCurrentFrame():preprocess(Output)&lt;br /&gt;
	end&lt;br /&gt;
&lt;br /&gt;
	return Output&lt;br /&gt;
end&lt;br /&gt;
&lt;br /&gt;
&lt;br /&gt;
return p&lt;/div&gt;</summary>
		<author><name>WikiSysop</name></author>
	</entry>
</feed>