-- HELPER FUNCTIONS --
local function notblank(v) return (v or '' ) ~= '' end
local function ifblank(v, alt) return notblank(v) and v or alt end
local function red(v) return '<span style="color: red;">' .. v .. '</span>' end
local function getContent(page) return string.gsub(mw.title.new(page).content, '^Date,"[^"]+"', '') end
local function diffDays(dargs)
return require('Module:diff days')['diff_days'](
mw.getCurrentFrame():newChild{ title = 'Template:Diff_days', args = dargs }
)
end
local showTimeLine = require('Module:Horizontal timeline')._showTimeLine
local Text = require('Module:Text').Text()
-- EXPORTED CONSTANTS --
local p = {
gSTALE = function() return 30 end, --global for representing the age in days after which a pageviews file, and any chart based on it, becomes stale and no longer appropriate for display on a Talk page. Stale pages should have their pageviews file updated.
gCOUNT = function() return 75 end, --global for representing the default number of days to display in the chart; overridable with param {{para|ct}}.
}
function p._xviews(args) -- {{xviews}}
-- args contains stale, days, ct, log, and mode
local page = ifblank(args[1], mw.title.getCurrentTitle().fullText .. '/pageviews')
if mw.title.new(page).exists then
local stale = tonumber(ifblank(args.stale, p.gSTALE()))
if tonumber(p._age(page)) > stale then --stale:
return red(mw.ustring.format(
'The pageviews file [[%s]] is more than %d days old; please see [[Template:Xviews#Instructions|Instructions]].',
page, stale
))
else --not stale:
return p._xvmain(page, {ct = ifblank(args.days, ifblank(args.ct, p.gCOUNT()))}) ..
((notblank(args.log) or notblank(args.mode)) and '' or p._xAxis(page, {})) -- p._xAxis() doesn't handle log values or vertical bars yet
end
else
return red('Missing required pageviews file.') ..
' See [[Template:Xviews#Instructions|Instructions]].'
end
end
function p.xviews(frame)
return p._xviews(frame.args)
end
function p._age(page) -- {{xviews/age}}
local nthview = p._nthView(page, p._viewCount(page))
nthview = Text.split(nthview, ':', true)[1]
return diffDays({nthview, mw.getContentLanguage():formatDate('j F Y'), precision=0})
end
function p.age(frame)
return p._age(ifblank(frame.args[1] or 'Talk:Liberation of France/pageviews'))
end
function p._nthView(page, index) -- {{xviews/nth view}}
local content = string.gsub(getContent(page), '.?(202%d%-%d%d%-%d%d)%,?(%d+)', '%1:%2,')
content = string.gsub(content, "^%s*,%s*(.-)%s*,$", "%1")
local items = {}
for item in string.gmatch(content, "([^,]+)") do
table.insert(items, mw.text.trim(item))
end
index = tonumber(index)
if index < 0 then
index = #items + index + 1
end
return items[index]
end
function p.nthView(frame)
return p._nthView(
ifblank(frame.args[1] or 'Talk:Liberation of France/pageviews'),
ifblank(frame.args[2] or 3)
)
end
function p._viewCount(page) -- {{xviews/view count}}
local str = string.gsub(getContent(page), '.?202%d%-%d%d%-%d%d', '')
str = string.gsub(str, ',', '', 1)
-- Remove leading and trailing delimiters (along with any surrounding whitespace)
str = str:gsub("^%s*,%s*", ""):gsub("%s*,%s*$", "")
-- Normalize internal consecutive delimiters to a single delimiter (replace ",," with ",")
str = str:gsub("%s*,%s*,%s*", ',')
return select(2, str:gsub(',', '')) + 1
end
function p.viewCount(frame)
return p._viewCount(ifblank(frame.args[1], 'Talk:Liberation of France/pageviews'))
end
function p._xvmain(page, xargs) -- {{xviews/xvmain}}
-- xargs contains ct, log, mode, and debug
local output = {}
local stop = p._viewCount(page)
local start = stop - math.min(152, ifblank(xargs.ct, 93)) + 1
local iargs = {
page = page,
max = p._maxViews(page),
log = xargs.log, -- logarithmic scale (not implemented yet)
mode = xargs.mode, -- bar display orientation (mode=vert not implemented yet)
debug = xargs.debug, -- set to non-empty for loop debug output emitted by p._itemBar()
}
for n = start, stop do
iargs.n = n
table.insert(output, p._itemBar(iargs))
end
return table.concat(output)
end
function p.xvmain(frame)
local xargs = frame.args
xargs.ct = ifblank(xargs.ct, 93)
return p._xvmain(ifblank(xargs[1], 'Talk:Liberation of France/pageviews'), xargs)
end
function p._itemBar(iargs) -- {{xviews/item bar}}
-- iargs contains n, page, max, log, mode, and debug
if iargs.debug then
return mw.ustring.format("page=%s, n=%s, label=%s",
ifblank(iargs.page, 'missing'),
ifblank(iargs.n, 'missing'),
ifblank(iargs.label, 'missing')
)
end
local nthview = p._nthView(iargs.page, ifblank(iargs.n, 1))
if not nthview then
error("Error generating item bar for item #" .. ifblank(iargs.n, 1) .. ". Range may be invalid")
end
local val = Text.split(nthview, ':', true)[2]
local width = val * 100 * 0.88 / p._maxViews(iargs.page)
width = tonumber(string.format("%.2f", width))
local label = p._xlabel(Text.split(nthview, ':', true)[1], 'y')
return p._vbar(width, val, {label = label})
end
function p.itemBar(frame)
local iargs = frame.args
iargs.page = ifblank(iargs.page, 'Talk:House of the Dragon/pageviews')
iargs.n = ifblank(iargs.n, 1)
return p._itemBar(iargs)
end
function p._vbar(width, val, vargs) -- {{xviews/vbar}}
-- vargs contains label, label-style, thick, style, val, and val-style
local ts = mw.getCurrentFrame():extensionTag{
name = 'templatestyles',
args = {src = 'Template:Xviews/vbar/styles.css'}
}
local div = mw.html.create('div'):attr('class', 'vbar')
if notblank(vargs.label) then
div:tag('span')
:attr('class', 'vbar-labelh')
:cssText(ifblank(vargs['label-style']))
:wikitext(vargs.label)
end
div:tag('span')
:attr('class', 'vbar-h')
:attr('alt', 'Page views for (date) = (views).')
:css('height', ifblank(vargs['thick']))
:css('width', width .. '%')
:cssText(ifblank(vargs.style))
if notblank(val) then
div:tag('span')
:attr('class', 'vbar-valh')
:cssText(ifblank(vargs['val-style']))
:wikitext(val)
end
return ts .. tostring(div)
end
function p.vbar(frame)
local vargs = frame.args
return p._vbar(ifblank(vargs[1], 75), ifblank(vargs[2], ifblank(vargs.val, 2136)), vargs)
end
function p._xlabel(date, bold01) -- {{xviews/xlabel}}
local replace = ((date:sub(-3) == '-01') and notblank(bold01)) and "'''%1'''" or "%1"
return string.gsub(date, '20[23]%d%-(%d%d%-%d%d)', replace)
end
function p.xlabel(frame)
return p._xlabel(ifblank(frame.date, '2024-06-01'), frame.bold01)
end
function p._maxViews(page) -- {{xviews/max views}}
local views = 0
local content = string.gsub(getContent(page), '.?202%d%-%d%d%-%d%d', '')
for v in Text.gsplit(content, ',', true) do
v = tonumber(v)
if (v and v > views) then views = v end
end
return views
end
function p.maxViews(frame)
return p._maxViews(ifblank(frame.args[1], 'Talk:Liberation of France/pageviews'))
end
function p._xAxis(page, xargs) -- {{xviews/x-axis}}
-- xargs contains shift, caption, ff
local div = mw.html.create('div')
:css('margin', '-22px 0 0 ' .. ifblank(xargs.shift, '16px'))
:wikitext(
showTimeLine({
caption = ifblank(xargs.caption, 'page views for ' .. mw.title.new(page).rootText),
border = 'none',
from = 0,
to = p._maxViews(page),
inc = tonumber(string.format("%.0f", ifblank(xargs.ff, 1.0) * p._maxViews(page)/100)) * 10,
row1 = 'scale',
['axis-nudge'] = '-0.8em',
['axis-0'] = '<span style="padding-left:10px">0</span>'
})
)
return tostring(div)
end
function p.xAxis(frame)
return p._xArgs(ifblank(frame.args[1], 'Talk:Liberation of France/pageviews'), frame.args)
end
return p