Module:TimeAgo: Difference between revisions
Jump to navigation
Jump to search
m (1 revision imported) |
m (1 revision imported: Merging) |
||
Line 1: | Line 1: | ||
-- | -- <nowiki> | ||
-- | |||
-- Implements {{time ago}} | |||
-- | |||
local p = {} | local p = {} | ||
local yesno = require( 'Module:Yesno' ) | |||
-- | -- assumes 31 days in a month (might need tweaking?) | ||
-- assumes 365.25 days in a year to account for leap years | |||
local convert = {60, 3600, 86400, 604800, 2678400, 31557600} | |||
-- used to convert units to magnitudes | |||
local magnitudes = { | |||
years = 6, | |||
months = 5, | |||
weeks = 4, | |||
days = 3, | |||
hours = 2, | |||
minutes = 1, | |||
seconds = 0 | |||
} | } | ||
-- | -- units to append to time diff | ||
local units = { | |||
{'second', 'seconds', 'second\'s', 'seconds\''}, | |||
{'minute', 'minutes', 'minute\'s', 'minutes\''}, | |||
{'hour', 'hours', 'hour\'s', 'hours\''}, | |||
{'day', 'days', 'day\'s', 'days\''}, | |||
{'week', 'weeks', 'week\'s', 'weeks\''}, | |||
{'month', 'months', 'month\'s', 'months\''}, | |||
{'year', 'years', 'year\'s', 'years\''} | |||
} | } | ||
-- | |||
-- Converts the input values to the returned string | |||
-- | |||
local function core( diff, abs_diff, magnitude, ago ) | |||
local num = math.floor( abs_diff ) | |||
local unit = 1 | |||
local plural = 1 | |||
if magnitude > 0 then | |||
num = math.floor( abs_diff / convert[magnitude] ) | |||
end | |||
if abs_diff > 1 or abs_diff == 0 then | |||
plural = plural + 1 | |||
end | |||
if diff >= 0 then | |||
ago = ago or 'ago' | |||
else | |||
plural = plural + 2 | |||
ago = 'time' | |||
end | |||
return num .. ' ' .. units[magnitude + 1][plural] .. ' ' .. ago | |||
end | |||
-- | |||
-- Compares arg1 to arg2 and returns the larger number | |||
-- | |||
local function max_( arg1, arg2 ) | |||
if arg1 > arg2 then | |||
return arg1 | |||
end | |||
return arg2 | |||
end | |||
-- | |||
-- Wrapper for use through #invoke | |||
-- | |||
function p.ago( frame ) | |||
return p._ago( frame:getParent().args ) | |||
end | end | ||
function p. | -- | ||
-- Validates arguments and converts them to something that can be process by core | |||
-- | |||
-- @param args[1] {str} time string | |||
-- @param args.magnitude {str} (optional) override the output time's units | |||
-- @param args.min_magnitude {str} (optional) require a minimum unit for the output time | |||
-- @param args.ago {str} (optional) Replace 'ago' with a different string | |||
-- only used for times in the past | |||
-- @param args.purge {str} (optional) add a purge link to the end of the resulting string | |||
-- will not work when testing from debug console | |||
-- | |||
function p._ago( args ) | |||
local lang = mw.language.getContentLanguage() | |||
local frame = mw.getCurrentFrame() | |||
local cur_time = lang:formatDate( 'U' ) | |||
-- check time argument is a valid time string | |||
local no_err, time = pcall( lang.formatDate, lang, 'U', args[1] ) | |||
if not no_err then | |||
return '<strong class="error">Error: first parameter cannot be parsed as a date or time.</strong>' | |||
end | |||
-- calculate time diff in seconds | |||
local diff = cur_time - time | |||
local abs_diff = math.abs( diff ) | |||
-- calculate magnitude | |||
local auto = 0 | |||
local min_ = -1 | |||
if args.magnitude then | |||
-- use the specified magnitude | |||
min_ = magnitudes[mw.ustring.lower( args.magnitude )] | |||
else | |||
-- use the specified minimum magnitude | |||
-- will only be used if it's higher than the auto-detected magnitude calculated below | |||
if args.min_magnitude then | |||
min_ = magnitudes[mw.ustring.lower( args.min_magnitude )] | |||
end | |||
-- auto detects the magnitude to be used | |||
-- multiples by two as it's preferred to have something like 43 hours instead of 2 days | |||
for i = 1, 6 do | |||
if math.floor( abs_diff / ( convert[i] * 2 ) ) > 0 then | |||
auto = auto + 1 | |||
else | |||
break | |||
end | |||
end | |||
-- for some reason the original template didn't detect weeks, using days instead | |||
-- so preserve that behaviour here | |||
if auto == 4 then | |||
auto = 3 | |||
end | |||
end | |||
magnitude = max_( auto, min_ ) | |||
local ret = core( diff, abs_diff, magnitude, args.ago ) | |||
if yesno( args.purge ) then | |||
-- @todo use mw.title for this | |||
local purgeTxt = args.purgeText or 'update' | |||
ret = string.format('%s <span class="plainlinks jsPurgeLink">([%s %s])</span>', ret, tostring(mw.uri.canonicalUrl(mw.title.getCurrentTitle().fullText, {action= 'purge'})), purgeTxt) | |||
end | |||
return ret | |||
end | end | ||
return p | return p |
Revision as of 17:46, 1 October 2018
Documentation for this module may be created at Module:TimeAgo/doc
-- <nowiki>
--
-- Implements {{time ago}}
--
local p = {}
local yesno = require( 'Module:Yesno' )
-- assumes 31 days in a month (might need tweaking?)
-- assumes 365.25 days in a year to account for leap years
local convert = {60, 3600, 86400, 604800, 2678400, 31557600}
-- used to convert units to magnitudes
local magnitudes = {
years = 6,
months = 5,
weeks = 4,
days = 3,
hours = 2,
minutes = 1,
seconds = 0
}
-- units to append to time diff
local units = {
{'second', 'seconds', 'second\'s', 'seconds\''},
{'minute', 'minutes', 'minute\'s', 'minutes\''},
{'hour', 'hours', 'hour\'s', 'hours\''},
{'day', 'days', 'day\'s', 'days\''},
{'week', 'weeks', 'week\'s', 'weeks\''},
{'month', 'months', 'month\'s', 'months\''},
{'year', 'years', 'year\'s', 'years\''}
}
--
-- Converts the input values to the returned string
--
local function core( diff, abs_diff, magnitude, ago )
local num = math.floor( abs_diff )
local unit = 1
local plural = 1
if magnitude > 0 then
num = math.floor( abs_diff / convert[magnitude] )
end
if abs_diff > 1 or abs_diff == 0 then
plural = plural + 1
end
if diff >= 0 then
ago = ago or 'ago'
else
plural = plural + 2
ago = 'time'
end
return num .. ' ' .. units[magnitude + 1][plural] .. ' ' .. ago
end
--
-- Compares arg1 to arg2 and returns the larger number
--
local function max_( arg1, arg2 )
if arg1 > arg2 then
return arg1
end
return arg2
end
--
-- Wrapper for use through #invoke
--
function p.ago( frame )
return p._ago( frame:getParent().args )
end
--
-- Validates arguments and converts them to something that can be process by core
--
-- @param args[1] {str} time string
-- @param args.magnitude {str} (optional) override the output time's units
-- @param args.min_magnitude {str} (optional) require a minimum unit for the output time
-- @param args.ago {str} (optional) Replace 'ago' with a different string
-- only used for times in the past
-- @param args.purge {str} (optional) add a purge link to the end of the resulting string
-- will not work when testing from debug console
--
function p._ago( args )
local lang = mw.language.getContentLanguage()
local frame = mw.getCurrentFrame()
local cur_time = lang:formatDate( 'U' )
-- check time argument is a valid time string
local no_err, time = pcall( lang.formatDate, lang, 'U', args[1] )
if not no_err then
return '<strong class="error">Error: first parameter cannot be parsed as a date or time.</strong>'
end
-- calculate time diff in seconds
local diff = cur_time - time
local abs_diff = math.abs( diff )
-- calculate magnitude
local auto = 0
local min_ = -1
if args.magnitude then
-- use the specified magnitude
min_ = magnitudes[mw.ustring.lower( args.magnitude )]
else
-- use the specified minimum magnitude
-- will only be used if it's higher than the auto-detected magnitude calculated below
if args.min_magnitude then
min_ = magnitudes[mw.ustring.lower( args.min_magnitude )]
end
-- auto detects the magnitude to be used
-- multiples by two as it's preferred to have something like 43 hours instead of 2 days
for i = 1, 6 do
if math.floor( abs_diff / ( convert[i] * 2 ) ) > 0 then
auto = auto + 1
else
break
end
end
-- for some reason the original template didn't detect weeks, using days instead
-- so preserve that behaviour here
if auto == 4 then
auto = 3
end
end
magnitude = max_( auto, min_ )
local ret = core( diff, abs_diff, magnitude, args.ago )
if yesno( args.purge ) then
-- @todo use mw.title for this
local purgeTxt = args.purgeText or 'update'
ret = string.format('%s <span class="plainlinks jsPurgeLink">([%s %s])</span>', ret, tostring(mw.uri.canonicalUrl(mw.title.getCurrentTitle().fullText, {action= 'purge'})), purgeTxt)
end
return ret
end
return p