https://de.wikipedia.org/w/index.php?action=history&feed=atom&title=Module%3AComplex_dateModul:Complex date - Versionsgeschichte2025-06-06T18:01:36ZVersionsgeschichte dieser Seite in WikipediaMediaWiki 1.45.0-wmf.4https://de.wikipedia.org/w/index.php?title=Modul:Complex_date&diff=195600313&oldid=prevEderporto: Modul:WikidataIB hat dieses Modul als Abhängigkeit, daher importiere ich das englische Äquivalent, damit es richtig funktioniert.2020-01-07T16:46:38Z<p>Modul:WikidataIB hat dieses Modul als Abhängigkeit, daher importiere ich das englische Äquivalent, damit es richtig funktioniert.</p>
<p><b>Neue Seite</b></p><div>--[[ <br />
__ __ _ _ ____ _ _ _ <br />
| \/ | ___ __| |_ _| | ___ _ / ___|___ _ __ ___ _ __ | | _____ __ __| | __ _| |_ ___ <br />
| |\/| |/ _ \ / _` | | | | |/ _ (_) | / _ \| '_ ` _ \| '_ \| |/ _ \ \/ / / _` |/ _` | __/ _ \<br />
| | | | (_) | (_| | |_| | | __/_| |__| (_) | | | | | | |_) | | __/> < | (_| | (_| | || __/<br />
|_| |_|\___/ \__,_|\__,_|_|\___(_)\____\___/|_| |_| |_| .__/|_|\___/_/\_\ \__,_|\__,_|\__\___|<br />
|_| <br />
<br />
This module is intended for creation of complex date phrases in variety of languages.<br />
<br />
Once deployed, please do not modify this code without applying the changes first at Module:Complex date/sandbox and testing <br />
at Module:Complex date/sandbox/testcases.<br />
<br />
Authors and maintainers:<br />
* User:Sn1per - first draft of the original version <br />
* User:Jarekt - corrections and expansion of the original version <br />
<br />
]]<br />
<br />
-- List of external modules and functions<br />
local p = {Error = nil}<br />
local i18n = require('Module:i18n/complex date')<br />
local ISOdate = require('Module:ISOdate')._ISOdate<br />
local Calendar -- loaded lazily<br />
<br />
-- ==================================================<br />
-- === Internal functions ===========================<br />
-- ==================================================<br />
<br />
local function langSwitch(list,lang)<br />
local langList = mw.language.getFallbacksFor(lang)<br />
table.insert(langList,1,lang)<br />
table.insert(langList,math.max(#langList,2),'default')<br />
for i,language in ipairs(langList) do<br />
if list[language] then<br />
return list[language]<br />
end<br />
end<br />
end<br />
<br />
local function formatnum1(numStr, lang)<br />
-- mostly require('Module:Formatnum').formatNum function used to translate a number to use different numeral characters, <br />
-- except that it it does not call that function unless the language is on the list "LList"<br />
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,['or']=1,bo=1,['ml-old']=1,mn=1,te=1,th=1}<br />
if LList[lang] then -- call only when the language is on the list<br />
numStr = require('Module:Formatnum').formatNum(numStr, lang, 1)<br />
end<br />
return numStr<br />
end<br />
<br />
local function getISODate(datestr, datetype, lang, num, case)<br />
-- translate dates in the format YYYY, YYYY-MM, and YYYY-MM-DD<br />
if not case and i18n.Translations[datetype] then<br />
-- look up the grammatical case needed and call ISOdate module<br />
local rec = langSwitch(i18n.Translations[datetype], lang)<br />
if type(rec)=='table' then<br />
case = rec.case[num]<br />
end<br />
end<br />
return ISOdate(datestr, lang, case, '', 1)<br />
end<br />
<br />
local function translatePhrase(date1, date2, operation, lang, state)<br />
-- use tables in Module:i18n/complex date to translate a phrase<br />
if not i18n.Translations[operation] then<br />
p.Error = string.format('<span style="background-color:red;">Error in [[Module:Complex date]]: input parameter "%s" is not recognized.</span>', operation or 'nil')<br />
return ''<br />
end<br />
local dateStr = langSwitch(i18n.Translations[operation], lang)<br />
if type(dateStr)=='table' then<br />
dateStr = dateStr[1]<br />
end<br />
if type(dateStr)=='function' then<br />
local success<br />
local nDates = i18n.Translations[operation]['nDates']<br />
if nDates==2 then -- 2 date phrase<br />
success, dateStr = pcall(dateStr, date1, date2, state)<br />
else -- 1 date phrase<br />
success, dateStr = pcall(dateStr, date1, state)<br />
end<br />
end<br />
<br />
if type(dateStr)=='string' then<br />
-- replace parts of the string '$date1' and '$date2' with date1 and date2 strings<br />
dateStr = mw.ustring.gsub(dateStr, '$date1', date1)<br />
dateStr = mw.ustring.gsub(dateStr, '$date2', date2)<br />
else<br />
-- Special case of more complex phrases that can be build out of simple phrases<br />
-- If complex case is not translated to "lang" than build it out of simpler ones<br />
local x = dateStr<br />
dateStr = p._complex_date(x.conj, x.adj1, date1, x.units1, x.era1, x.adj2, date2, x.units2, x.era2, lang, 2)<br />
end<br />
return dateStr<br />
end<br />
<br />
local function oneDatePhrase(dateStr, adj, era, units, lang, num, case, state)<br />
-- translate a single date phrase<br />
if num==2 then<br />
state.adj, state.era, state.units, state.precision = state.adj2, state.era2, state.units2, state.precision2 <br />
end<br />
<br />
-- dateStr can have many forms: ISO date, year or a number for <br />
-- decade, century or millennium<br />
if units == '' then -- unit is "year", "month", "day"<br />
dateStr = getISODate(dateStr, adj, lang, num, case)<br />
else -- units is "decade", "century", "millennium''<br />
dateStr = translatePhrase(dateStr, '', units, lang, state)<br />
end<br />
<br />
-- add adjective ("early", "mid", etc.) or preposition ("before", "after", <br />
-- "circa", etc.) to the date<br />
if adj ~= '' then<br />
dateStr = translatePhrase(dateStr, '', adj, lang, state)<br />
else -- only era?<br />
dateStr = formatnum1(dateStr, lang)<br />
end<br />
<br />
-- add era<br />
if era ~= '' then<br />
dateStr = translatePhrase(dateStr, '', era, lang, state)<br />
end<br />
return dateStr<br />
end<br />
<br />
local function twoDatePhrase(date1, date2, state, lang)<br />
-- translate a double date phrase<br />
local dateStr, case<br />
local era=''<br />
if state.era1 == state.era2 then<br />
-- if both eras are the same than add it only once<br />
era = state.era1<br />
state.era1 = ''<br />
state.era2 = ''<br />
end<br />
case = {nil, nil}<br />
if i18n.Translations[state.conj] then<br />
local rec = langSwitch(i18n.Translations[state.conj], lang)<br />
if type(rec)=='table' then<br />
case = rec.case<br />
end<br />
end<br />
date1 = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, case[1], state)<br />
date2 = oneDatePhrase(date2, state.adj2, state.era2, state.units2, lang, 2, case[2], state)<br />
dateStr = translatePhrase(date1, date2, state.conj, lang, state)<br />
if era ~= '' then<br />
dateStr = translatePhrase(dateStr, '', era, lang, state)<br />
end<br />
return dateStr<br />
end<br />
<br />
local function otherPhrases(date1, date2, operation, era, lang, state)<br />
-- translate specialized phrases<br />
local dateStr = ''<br />
<br />
if operation == 'islamic' then<br />
if date2=='' then date2 = mw.getCurrentFrame():callParserFunction('#time', 'xmY', date1) end<br />
date1 = getISODate(date1, operation, lang, 1, nil)<br />
date2 = getISODate(date2, operation, lang, 2, nil)<br />
if era == '' then era = 'ad' end<br />
dateStr = translatePhrase(date1, '', era, lang, state) .. ' (' .. translatePhrase(date2, '', 'ah', lang, state) .. ')'<br />
era = ''<br />
elseif operation == 'julian' then<br />
if not date2 and date1 then -- Convert from Julian to Gregorian calendar date<br />
if Calendar == nil then<br />
Calendar = require("Module:Calendar")<br />
end<br />
local JDN = Calendar._date2jdn(date1, 0)<br />
if JDN then<br />
date2 = date1 -- first date is assumed to be Julian<br />
date1 = Calendar._jdn2date(JDN, 1)<br />
end<br />
end<br />
date1 = getISODate(date1, operation, lang, 1, nil)<br />
date2 = getISODate(date2, operation, lang, 2, nil)<br />
dateStr = translatePhrase(date1, date2, operation, lang, state)<br />
dateStr = mw.ustring.gsub(mw.ustring.gsub(dateStr, '%( ', '('), ' %)', ')') -- in case date2 is empty<br />
elseif operation == 'turn of the year' or operation == 'turn of the decade' or operation == 'turn of the century' then<br />
local dt = 1<br />
if operation == 'turn of the decade' then dt=10 end<br />
if not date2 or date2=='' then date2=tostring(tonumber(date1)-dt) end<br />
if era~='bp' and era~='bc' then date1, date2 = date2, date1 end<br />
if operation == 'turn of the year' then<br />
date1 = ISOdate(date1, lang, '', '', 1)<br />
date2 = ISOdate(date2, lang, '', '', 1)<br />
else<br />
date1 = formatnum1(date1, lang)<br />
date2 = formatnum1(date2, lang)<br />
end<br />
dateStr = translatePhrase(date1, date2, operation, lang, state)<br />
elseif operation == 'year unknown' then<br />
dateStr = translatePhrase('', '', operation, lang, state)<br />
elseif operation == 'unknown' then<br />
dateStr = tostring(mw.message.new( "exif-unknowndate" ):inLanguage( lang ))<br />
end<br />
<br />
-- add era<br />
if era ~= '' then<br />
dateStr = translatePhrase(dateStr, '', era, lang, state)<br />
end<br />
return dateStr<br />
end<br />
<br />
local function checkAliases(str1, str2, sType)<br />
-- some inputs have many aliases - reconcile them and ensure string is playing a proper role <br />
local out = ''<br />
if str1 and str1~='' then<br />
a = i18n.Synonyms[str1] -- look up synonyms of "str1"<br />
if a then<br />
out = a[1]<br />
else<br />
p.Error = string.format('<span style="background-color:red;">Error in [[Module:Complex date]]: %s is not recognized.</span>', str1)<br />
end<br />
elseif str2 and str2~='' then -- if "str1" of type "sType" is empty than maybe ...<br />
a = i18n.Synonyms[str2] -- ..."str2" is of the same type and is not empty<br />
if a and a[2]==sType then<br />
out = a[1]<br />
str2 = ''<br />
end<br />
end<br />
return out, str2<br />
end<br />
<br />
local function datePrecision(dateStr, units) <br />
-- "in this module "Units" is a string like millennium, century, or decade<br />
-- "precision" is wikibase compatible date precision number: 6=millennium, 7=century, 8=decade, 9=year, 10=month, 11=day<br />
-- based on string or numeric input calculate "Units" and "precision"<br />
local dateNum = tonumber(dateStr);<br />
if type(units)=='number' then<br />
precision = units<br />
if precision>11 then precision=11 end -- clip the range of precision values<br />
if precision==6 then units='millennium' <br />
elseif precision==7 then units='century'<br />
elseif precision==8 then units='decade'<br />
else units = ''<br />
end<br />
elseif type(units)=='string' then<br />
units = string.lower(units);<br />
if units=='millennium' then precision=6<br />
elseif units=='century' then precision=7<br />
elseif units=='decade' then precision=8<br />
else precision=9<br />
end<br />
end<br />
if units=='' or precision==9 then<br />
local sLen = mw.ustring.len(dateStr)<br />
if sLen<= 4 then precision=9<br />
elseif sLen== 7 then precision=10<br />
elseif sLen>=10 then precision=11<br />
end<br />
units=''<br />
end<br />
if precision==6 and dateStr.match( dateStr, '%d000' )~=nil then <br />
dateStr = tostring(math.floor(tonumber(dateStr)/1000) +1)<br />
elseif precision==7 and mw.ustring.match( dateStr, '%d%d00' )~=nil then<br />
dateStr = tostring(math.floor(tonumber(dateStr)/100) +1)<br />
end<br />
<br />
return dateStr, units, precision<br />
end<br />
<br />
<br />
local function isodate2timestamp(dateStr, precision, era)<br />
-- convert date string to timestamps used by Quick Statements<br />
local tStamp = nil<br />
if era == 'ah' or precision<6 then<br />
return nil<br />
elseif era ~= '' then<br />
eraLUT = {ad='+', bc='-', bp='-' }<br />
era = eraLUT[era]<br />
else<br />
era='+'<br />
end<br />
<br />
-- convert isodate to timestamp used by quick statements<br />
if precision>=9 then <br />
if string.match(dateStr,"^%d%d%d%d$") then -- if YYYY format <br />
tStamp = era .. dateStr .. '-00-00T00:00:00Z/9'<br />
elseif string.match(dateStr,"^%d%d%d%d%-%d%d$") then -- if YYYY-MM format <br />
tStamp = era .. dateStr .. '-00T00:00:00Z/10'<br />
elseif string.match(dateStr,"^%d%d%d%d%-%d%d%-%d%d$") then -- if YYYY-MM-DD format <br />
tStamp = era .. dateStr .. 'T00:00:00Z/11'<br />
end<br />
elseif precision==8 then -- decade<br />
tStamp = era .. dateStr .. '-00-00T00:00:00Z/8'<br />
elseif precision==7 then -- century<br />
local d = tostring(tonumber(dateStr)-1)<br />
tStamp = era .. d .. '50-00-00T00:00:00Z/7'<br />
elseif precision==6 then<br />
local d = tostring(tonumber(dateStr)-1)<br />
tStamp = era .. d .. '500-00-00T00:00:00Z/6'<br />
end<br />
<br />
return tStamp<br />
end<br />
<br />
local function oneDateQScode(dateStr, adj, era, precision)<br />
-- create QuickStatements string for "one date" dates<br />
local outputStr = ''<br />
<br />
local d = isodate2timestamp(dateStr, precision, era)<br />
if not d then<br />
return ''<br />
end<br />
local rLUT = { early='Q40719727' , mid='Q40719748', late='Q40719766',<br />
['1quarter']='Q40690303' , ['2quarter']='Q40719649' , ['3quarter']='Q40719662', ['4quarter']='Q40719674',<br />
spring='Q40720559' , summer='Q40720564' , autumn='Q40720568' , winter='Q40720553',<br />
firsthalf='Q40719687', secondhalf='Q40719707' }<br />
local qLUT = {['from']='P580', ['until']='P582', ['after']='P1319', ['before']='P1326'}<br />
<br />
local refine = rLUT[adj]<br />
local qualitier = qLUT[adj]<br />
<br />
if adj=='' then<br />
outputStr = d<br />
elseif adj=='circa' then<br />
outputStr = d..",P1480,Q5727902"<br />
elseif refine then<br />
outputStr = d..",P4241,"..refine<br />
elseif precision>7 and qualitier then<br />
local century = string.gsub(d, 'Z%/%d+', 'Z/7')<br />
outputStr = century ..",".. qualitier ..","..d<br />
end<br />
return outputStr <br />
end<br />
<br />
local function twoDateQScode(date1, date2, state)<br />
-- create QuickStatements string for "two date" dates<br />
if state.adj1~='' or state.adj2~='' or state.era1~=state.era2 then<br />
return ''<br />
end<br />
local outputStr = ''<br />
local d1 = isodate2timestamp(date1, state.precision1, state.era1)<br />
local d2 = isodate2timestamp(date2, state.precision2, state.era2)<br />
if (not d1) or (not d2) then<br />
return ''<br />
end <br />
-- find date with lower precision in common to both dates<br />
local cd<br />
local year1 = string.sub(d1,2,5)<br />
local k = 0<br />
for i = 1,10,1 do <br />
if string.sub(d1,1,i)==string.sub(d2,1,i) then <br />
k = i -- find last matching letter<br />
end<br />
end<br />
if k>=9 then -- same month, since "+YYYY-MM-" is in common<br />
cd = isodate2timestamp(string.sub(d1,2,8), 10, state.era1)<br />
elseif k>=6 and k<9 then -- same year, since "+YYYY-" is in common<br />
cd = isodate2timestamp(year1, 9, state.era1)<br />
elseif k==4 then -- same decade(k=4, precision=8), since "+YYY" is in common<br />
cd = isodate2timestamp(year1, 8, state.era1)<br />
elseif k==3 then -- same century(k=3, precision=7) since "+YY" is in common<br />
local d = tostring(math.floor(tonumber(year1)/100) +1) -- convert 1999 -> 20<br />
cd = isodate2timestamp( d, 7, state.era1)<br />
elseif k==2 then -- same millennium (k=2, precision=6), since "+Y" is in common<br />
local d = tostring(math.floor(tonumber(year1)/1000) +1) -- convert 1999 -> 2<br />
cd = isodate2timestamp( d, 6, state.era1)<br />
else<br />
return ''<br />
end<br />
--if not cd then<br />
-- return ' <br/>error: ' .. d1.." / " .. d2.." / ".. (cd or '') .." / ".. string.sub(d1,2,5).." / " .. string.sub(d2,2,5).." / " .. tostring(k)<br />
--end<br />
<br />
--<br />
if state.conj=='from-until' then<br />
outputStr = cd ..",P580,".. d1 ..",P582,".. d2<br />
elseif state.conj=='between' then<br />
outputStr = cd ..",P1319,".. d1 ..",P1326,".. d2<br />
elseif state.conj=='circa2' then<br />
outputStr = cd ..",P1319,".. d1 ..",P1326,".. d2 ..",P1480,Q5727902"<br />
end<br />
<br />
return outputStr<br />
end<br />
<br />
-- ==================================================<br />
-- === External functions ===========================<br />
-- ==================================================<br />
<br />
function p.Era(frame)<br />
-- process inputs<br />
local dateStr<br />
local args = frame.args<br />
if not (args.lang and mw.language.isSupportedLanguage(args.lang)) then <br />
args.lang = frame:callParserFunction( "int", "lang" ) -- get user's chosen language <br />
end<br />
local lang = args['lang']<br />
local dateStr = args['date'] or ''<br />
local eraType = string.lower(args['era'] or '')<br />
<br />
dateStr = ISOdate(dateStr, lang, '', '', 1)<br />
if eraType then <br />
eraType = checkAliases(eraType ,'','e')<br />
dateStr = translatePhrase(dateStr, '', eraType, lang, {}) <br />
end<br />
return dateStr<br />
end<br />
<br />
function p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, passNr)<br />
local Output=''<br />
<br />
-- process inputs and save date in state array<br />
local state = {} <br />
state.conj = string.lower(conj or '')<br />
state.adj1 = string.lower(adj1 or '')<br />
state.adj2 = string.lower(adj2 or '')<br />
state.era1 = string.lower(era1 or '')<br />
state.era2 = string.lower(era2 or '')<br />
state.units1 = string.lower(units1 or '')<br />
state.units2 = string.lower(units2 or '')<br />
<br />
-- if date 1 is missing but date 2 is provided than swap them<br />
if date1 == '' and date2 ~= '' then<br />
date1 = date2<br />
date2 = ''<br />
state = {adj1 = state.adj2, era1 = state.era2, units1 = state.units2, <br />
adj2 = '', era2 = '', units2 = '', conj=state.conj, num=1}<br />
end<br />
if date2 ~= '' then state.nDates = 2 <br />
elseif date1 ~= '' then state.nDates = 1 <br />
else state.nDates = 0<br />
end<br />
<br />
-- reconcile alternative names for text inputs<br />
local conj = checkAliases(state.conj ,'' ,'j')<br />
state.adj1 ,conj = checkAliases(state.adj1 ,conj,'a')<br />
state.units1,conj = checkAliases(state.units1,conj,'p')<br />
state.era1 ,conj = checkAliases(state.era1 ,conj,'e')<br />
state.special,conj = checkAliases('',conj,'c')<br />
state.adj2 = checkAliases(state.adj2 ,'','a')<br />
state.units2 = checkAliases(state.units2,'','p')<br />
state.era2 = checkAliases(state.era2 ,'','e')<br />
state.conj = conj<br />
state.lang = lang<br />
if p.Error~=nil then<br />
return nil<br />
end<br />
<br />
-- calculate date precision value<br />
date1, state.units1, state.precision1 = datePrecision(date1, state.units1)<br />
date2, state.units2, state.precision2 = datePrecision(date2, state.units2)<br />
<br />
-- Handle special cases <br />
-- Some complex phrases can be created out of simpler ones. Therefore on pass # 1 we try to create <br />
-- the phrase using complex phrase and if that is not found than on the second pass we try to build<br />
-- the phrase out of the simpler ones<br />
if passNr==1 then<br />
if state.adj1=='circa' and state.nDates == 2 then<br />
state.conj = 'circa2'<br />
state.adj1 = ''<br />
state.adj2 = ''<br />
end<br />
if state.nDates == 2 and state.adj1=='late' and state.adj2=='early' and state.conj=='and' <br />
and state.units1==state.units2 and state.era1==state.era2 then<br />
if state.units1=='century' then<br />
state.conj='turn of the century'<br />
elseif state.units1=='decade' then<br />
state.conj='turn of the decade'<br />
elseif state.units1=='' then<br />
state.conj='turn of the year'<br />
end<br />
state.adj1 = ''<br />
state.adj2 = ''<br />
state.units1 = ''<br />
state.units2 = ''<br />
end<br />
end<br />
<br />
local errorStr = string.format(<br />
'\n*conj=%s, adj1=%s, era1=%s, unit1=%s, prec1=%i, adj2=%s, era2=%s, unit2=%s, prec2=%i, special=%s', <br />
state.conj, state.adj1, state.era1, state.units1, state.precision1,<br />
state.adj2, state.era2, state.units2, state.precision2, state.special) <br />
state.adj, state.era, state.units, state.precision = state.adj1, state.era1, state.units1, state.precision1 <br />
<br />
-- call specialized functions<br />
local QScode = ''<br />
if state.special~='' then<br />
Output = otherPhrases(date1, date2, state.special, state.era1, lang, state) <br />
elseif state.conj~='' then<br />
QScode = twoDateQScode(date1, date2, state)<br />
Output = twoDatePhrase(date1, date2, state, lang)<br />
elseif state.adj1~='' or state.era1~='' or state.units1~='' then<br />
Output = oneDatePhrase(date1, state.adj1, state.era1, state.units1, lang, 1, nil, state)<br />
QScode = oneDateQScode(date1, state.adj1, state.era1, state.precision1)<br />
elseif date1~='' then<br />
Output = ISOdate(date1, lang, '', 'dtstart', '100-999')<br />
end<br />
if p.Error~=nil then<br />
return errorStr<br />
end<br />
<br />
-- if there is any wikicode in the string than execute it<br />
if mw.ustring.find(Output, '{') then<br />
Output = mw.getCurrentFrame():preprocess(Output)<br />
end<br />
if QScode and #QScode>0 then<br />
QScode = ' <div style="display: none;">date QS:P,' .. QScode .. '</div>'<br />
end<br />
<br />
return Output .. QScode<br />
end<br />
<br />
function p.complex_date(frame)<br />
-- process inputs<br />
local dateStr, Error<br />
local args = frame.args<br />
if not (args.lang and mw.language.isSupportedLanguage(args.lang)) then <br />
args.lang = frame:callParserFunction( "int", "lang" ) -- get user's chosen language <br />
end<br />
local date1 = args['date1'] or args['2'] or args['date'] or ''<br />
local date2 = args['date2'] or args['3'] or ''<br />
local conj = args['conj'] or args['1'] or ''<br />
local adj1 = args['adj1'] or args['adj'] or ''<br />
local adj2 = args['adj2'] or ''<br />
local units1 = args['precision1'] or args['precision'] or ''<br />
local units2 = args['precision2'] or args['precision'] or ''<br />
local era1 = args['era1'] or args['era'] or ''<br />
local era2 = args['era2'] or args['era'] or ''<br />
local lang = args['lang']<br />
<br />
dateStr = p._complex_date(conj, adj1, date1, units1, era1, adj2, date2, units2, era2, lang, 1)<br />
if p.Error~=nil then<br />
dateStr = p.Error .. '[[Category:Pages using Complex date template with incorrect parameter]]'<br />
end<br />
return dateStr<br />
end<br />
<br />
return p</div>Ederporto