Module:Ranking movements

Permanently protected module
From Wikipedia, the free encyclopedia

-- This module implements [[Template:Ranking movements]]
local p = {}

local templatestyles = 'Module:Ranking movements/styles.css'

local colors = {
	['+'] = '#D8FFEB',
	['-'] = '#FFE6E6',
	['not released'] = '#999999'
}
local labels = {
	['title'] = 'Ranking movements',
	['legend'] = 'Legend',
	['week'] = 'Week',
	['poll'] = 'Poll',
	['pre'] = 'Pre',
	['final'] = 'Final',
	['not released'] = 'Not released',
	['+'] = 'Increase in ranking',
	['-'] = 'Decrease in ranking',
	['—'] = 'Not ranked',
	['RV'] = 'Received votes',
	['т'] = 'Tied with team above or below',
	['( )'] = 'First-place votes'
}

local legend = {
	['+'] = false,
	['-'] = false,
	['—'] = false,
	['RV'] = false,
	['т'] = false,
	['( )'] = false
}

local function format_cell_text(s)
	if s then
		if s:match('^[Nn][Rr]') or s:match('^—') then
			legend['—'] = true
		end
		if s:match('^[Rr][Vv]') then
			legend['RV'] = true
		end
		if s:match('%d[%-%s]*[Tt]') or s:match('[Tт]') then
			legend['т'] = true
		end
		if s:match('[%(%)]') then
			legend['( )'] = true
		end
		s = mw.ustring.gsub(s, '^NR', '—')
	end
	return s
end

local function get_rank(s)
	-- This is intended to remove any extra stuff after the ranking
	local result = mw.ustring.match(s or '', '^%d*') or '' -- Is it numbers?
	if result == '' then
		result = mw.ustring.match(s or '', '^[%a/—]*') or '' -- Is it letters?
	end
	return result
end

local function get_color(lastweek, thisweek)
	-- No coloring if first week or this week is blank
	if (lastweek == 'first week') or (thisweek == '') or (lastweek == '')
		or (lastweek == 'N/A') or (thisweek == 'N/A') then
		return nil
	end
	-- No coloring if the rank has not changes
	if (lastweek == thisweek) then
		return nil
	end
	-- If last week was NR then rank has increased
	if (lastweek == 'NR' or lastweek == '—') then
		legend['+'] = true
		return colors['+']
	end
	-- If last week was RV
	if (lastweek == 'RV') then
		-- If this week is NR then rank has decreased, otherwise increased
		if (thisweek == 'NR' or thisweek == '—') then
			legend['-'] = true
			return colors['-']
		else
			legend['+'] = true
			return colors['+']
		end
	end
	-- In all other cases, just compare the numbers
	lastweek = tonumber(lastweek) or 999999
	thisweek = tonumber(thisweek) or 999999
	if (lastweek > thisweek) then
		legend['+'] = true
		return colors['+']
	end
	if (thisweek > lastweek) then
		legend['-'] = true
		return colors['-']
	end
	return nil
end

local tracking, preview = {}, {}

local function checkargs(args)
    for k, v in pairs(args) do
    	if v ~= '' then
			if k and type(k) == 'string' then
				if k:match('^poll[1-6]firstweek$') or k:match('^poll[1-6]lastweek$') 
					or k:match('^poll[1-6]title$') or k:match('^poll[1-6]_final$') or k == 'title' or k == 'nocat'
					or k:match('^wk1?%d$') or k:match('^wk2[01]') then
					-- valid
				elseif k:match('^poll[1-6]_1?%d$') or k:match('^poll[1-6]_2[01]$') then
					-- valid
					local i = mw.ustring.gsub(k, '^poll([1-6])_%d+$', '%1')
					local j = mw.ustring.gsub(k, '^poll[1-6]_(%d+)$', '%1')
					local maxweeks = tonumber(args['poll' .. i .. 'lastweek']) or 20
					if tonumber(j) > maxweeks then
						table.insert(tracking, '[[Category:Pages using ranking movements with poll after last week|' .. i .. ']]')
					end
				else
					-- invalid
					local vlen = mw.ustring.len(k)
					k = mw.ustring.sub(k, 1, (vlen < 25) and vlen or 25) 
					k = mw.ustring.gsub(k, '[^%w\-_ ]', '?')
					table.insert(tracking, '[[Category:Pages using ranking movements with unknown parameters|' .. k .. ']]')
					table.insert(preview, '"' .. k .. '"')
				end
			end
		end
	end
end


function p.main(frame)
	local args = require('Module:Arguments').getArgs(frame)
	
	local maxpolls = 6
	-- Compute the maximum number of columns
	local minweeks = 20
	local maxweeks = 0
	for k = 1,maxpolls do
		local n = tonumber(args['poll' .. k .. 'lastweek']) or (args['poll' .. k .. 'title'] and 20 or 0)
		maxweeks = (n > maxweeks) and n or maxweeks
		local n = tonumber(args['poll' .. k .. 'firstweek']) or (args['poll' .. k .. 'title'] and 0 or 20)
		minweeks = (n < minweeks) and n or minweeks
	end

	-- Start table
	local root = mw.html.create('table')
		:addClass('wikitable ranking-movements')

	-- Table caption title
	local caption = root:tag('caption'):wikitext(args['title'] or labels['title'])

	-- Week header row
	local row = root:tag('tr')
	row:tag('th') -- Blank space in corner
	row:tag('th'):attr('colspan', maxweeks - minweeks + 1):wikitext(labels['week'])

	-- Poll header row
	row = root:tag('tr')
	row:tag('th')
		:wikitext(labels['poll'])
	for k = minweeks,maxweeks do
		local text = args['wk' .. k] or k
		if (k == 0) then
			text = args['wk' .. k] or labels['pre']
		elseif (k == maxweeks) then
			text = args['wk' .. k] or labels['final']
		end
		row:tag('th')
			:wikitext(text)
	end
	
	-- Poll data rows
	for i = 1,maxpolls do
		local ptitle = args['poll' .. i .. 'title'] or ''
		local pfirstweek = tonumber(args['poll' .. i .. 'firstweek']) or 0
		local plastweek = tonumber(args['poll' .. i .. 'lastweek']) or 20
		if ptitle ~= '' then
			row = root:tag('tr')
			row:tag('th')
				:wikitext(ptitle)
			-- Before the first poll
			if pfirstweek > minweeks then
				row:tag('td')
					:attr('colspan', pfirstweek - minweeks)
					:css('background', colors['not released'])
					:wikitext(labels['not released'])
			end
			-- Results
			local lastweek = 'first week'
			for k = pfirstweek,plastweek do
				local rank = args['poll' .. i .. '_' .. k]
				if rank then
					rank = mw.ustring.gsub(rank, '^&[MNmn][Dd][Aa][Ss][Hh];', '—')
					rank = mw.ustring.gsub(rank, '^[–%-]', '—')
				end
				local thisweek = get_rank(rank)
				row:tag('td')
					:css('background', get_color(lastweek, thisweek))
					:wikitext(format_cell_text(rank))
				lastweek = thisweek
			end
			-- After the last poll
			if plastweek < maxweeks then
				rank = args['poll' .. i .. '_final']
				local colspan = (maxweeks - plastweek) - (rank and 1 or 0)
				row:tag('td')
					:attr('colspan', (colspan > 1) and colspan or nil)
					:css('background', colors['not released'])
					:wikitext(labels['not released'])
				if rank and colspan > 0 then
					rank = mw.ustring.gsub(rank, '^&[MNmn][Dd][Aa][Ss][Hh];', '—')
					rank = mw.ustring.gsub(rank, '^[–%-]', '—')
					local thisweek = get_rank(rank)
					row:tag('td')
						:css('background', get_color(lastweek, thisweek))
						:wikitext(format_cell_text(rank))
				end				
			end
		end
	end
	-- Add legend
	local line1, line2 = '', ''
	if legend['+'] or legend['-'] then
		for k,v in ipairs({'+', '-'}) do
			line1 = line1 .. ' <span style="color:' .. colors[v] .. '">██</span>'
			line1 = line1 .. ' ' .. labels[v]
		end
	end
	for k,v in ipairs({'—', 'RV', 'т', '( )'}) do
		if legend[v] then line2 = line2  .. ' ' .. v .. ' = ' .. labels[v] end
	end
	if line1 ~= '' or line2 ~= '' then
		caption:wikitext('<br /><small>\'\'\'' .. labels['legend'] .. ':\'\'\'' .. line1)
		if line1 ~= '' and line2 ~= '' then
			caption:wikitext('<br />')
		end
		caption:wikitext(line2 .. '</small>')
	end
	
	checkargs(args)

	local trackstr = (#tracking > 0) and table.concat(tracking, '') or ''
	if #preview > 0 then
		trackstr = require('Module:If preview')._warning({
			'Unknown parameters ' .. table.concat(preview, '; ') .. '.'
		}) .. trackstr
	end

	return frame:extensionTag{ name = 'templatestyles', args = { src = templatestyles} } 
		.. tostring(root) .. trackstr
end

return p