Hoppa till innehållet

Modul:Sports results

Från Wikipedia

Dokumentation [visa] [redigera] [historik] [rensa sidcachen]


Dubbelmöten

[redigera wikitext]
Vad du skriver
{{#invoke:Sports results|main
|update=1 januari 1900
|source=[https://sv.wikipedia.org/wiki/Portal:Huvudsida Wikipedia]

|team1=AAA |name_AAA=Länka lag 1 |short_AAA=KN 1
|team2=BBB |name_BBB=Länka lag 2 |short_BBB=KN 2
|team3=CCC |name_CCC=Länka lag 3 |short_CCC=KN 3
|team4=DDD |name_DDD=Länka lag 4 |short_DDD=KN 4

<!--Lag 1-->
|match_AAA_BBB=1–0
|match_AAA_CCC=0–1
|match_AAA_DDD=<small>16 nov '14</small>

<!--Lag 2-->
|match_BBB_AAA=B–A
|match_BBB_CCC=<small>3 sep '15</small>
|match_BBB_DDD=B–D

<!--Lag 3-->
|match_CCC_AAA=<small>16 nov '14</small>
|match_CCC_BBB=C–B
|match_CCC_DDD=<small>16 nov '14</small>

<!--Lag 4-->
|match_DDD_AAA=D–A
|match_DDD_BBB=<small>3 sep '15</small>
|match_DDD_CCC=D–C

}}
Hur det ser ut
Hemma \ Borta KN 1 KN 2 KN 3 KN 4
Länka lag 1 1–0 0–1 16 nov '14
Länka lag 2 B–A 3 sep '15 B–D
Länka lag 3 16 nov '14 C–B 16 nov '14
Länka lag 4 D–A 3 sep '15 D–C
Uppdaterad efter match(er) spelad(e) den 1 januari 1900. Källa: Wikipedia

Mer än två möten

[redigera wikitext]

Modulen kan användas för ligor som möter varandra mer än två gånger, för detta använd |legs=2 och |match1_AAA_BBB=/|match2_AAA_BBB= istället för |match_AAA_BBB=

Vad du skriver
{{#invoke:sports results|main
|update = complete
|source = [http://www.aleague.com.au/results aleague.com.au]
|font_size=85%
|solid_cell=grey|legs=2

|team1 =ADE |name_ADE=[[Adelaide United FC|Adelaide United]]
|team2 =BRI |name_BRI=[[Brisbane Roar FC|Brisbane Roar]]
|team3 =CCM |name_CCM=[[Central Coast Mariners FC|Central Coast Mariners]]
|team4 =MCY |name_MCY=[[Melbourne City FC|Melbourne City]]
|team5 =MVC |name_MVC=[[Melbourne Victory FC|Melbourne Victory]]
|team6 =NEW |name_NEW=[[Newcastle United Jets FC|Newcastle United Jets]]
|team7 =PER |name_PER=[[Perth Glory FC|Perth Glory]]
|team8 =SYD |name_SYD=[[Sydney FC|Sydney]]
|team9 =WEL |name_WEL=[[Wellington Phoenix FC|Wellington Phoenix]] {{flagdeco|NZL}}|short_WEL=[[Wellington Phoenix FC|WEL]]
|team10=WSW |name_WSW=[[Western Sydney Wanderers FC|Western Sydney Wanderers]]

<!--Adelade United-->
|match1_ADE_BRI= 2–1
|match1_ADE_CCM= 2–1
|match1_ADE_MCY= 0–2
|match1_ADE_MVC= 2–0
|match1_ADE_NEW= 1–1
|match1_ADE_PER= 0–2
|match1_ADE_SYD= 1–1
|match1_ADE_WEL= 0–0
|match1_ADE_WSW= 2–2

|match2_ADE_BRI= 4–3
|match2_ADE_CCM= null 
|match2_ADE_MCY= null
|match2_ADE_MVC= 1–0
|match2_ADE_NEW= null 
|match2_ADE_PER= 0–2
|match2_ADE_SYD= null 
|match2_ADE_WEL= 3–1
|match2_ADE_WSW= 1–3

<!--Brisbane Roar-->
|match1_BRI_ADE= 3–5
|match1_BRI_CCM= 1–1
|match1_BRI_MCY= 2–0
|match1_BRI_MVC= 2–4
|match1_BRI_NEW= 1–6
|match1_BRI_PER= 2–4
|match1_BRI_SYD= 2–1
|match1_BRI_WEL= 0–0
|match1_BRI_WSW= 2–2

|match2_BRI_ADE= null
|match2_BRI_CCM= null
|match2_BRI_MCY= null
|match2_BRI_MVC= 0–5
|match2_BRI_NEW= null
|match2_BRI_PER= null
|match2_BRI_SYD= 1–3
|match2_BRI_WEL= 2–1
|match2_BRI_WSW= 1–4

<!--Central Coast Mariners-->
|match1_CCM_ADE= 0–3
|match1_CCM_BRI= 1–1
|match1_CCM_MCY= 1–1
|match1_CCM_MVC= 2–3
|match1_CCM_NEW= 1–2
|match1_CCM_PER= 1–4
|match1_CCM_SYD= 1–2
|match1_CCM_WEL= 2–8
|match1_CCM_WSW= 3–1

|match2_CCM_ADE= 0–1
|match2_CCM_BRI= 3–5
|match2_CCM_MCY= 2–1
|match2_CCM_MVC= null 
|match2_CCM_NEW= null
|match2_CCM_PER= 0–3
|match2_CCM_SYD= null
|match2_CCM_WEL= null
|match2_CCM_WSW= null

<!--Melbourne City-->
|match1_MCY_ADE= 1–1
|match1_MCY_BRI= 1–0
|match1_MCY_CCM= 5–0
|match1_MCY_MVC= 1–1
|match1_MCY_NEW= 3–0
|match1_MCY_PER= 0–0
|match1_MCY_SYD= 0–3
|match1_MCY_WEL= 2–0
|match1_MCY_WSW= 4–3

|match2_MCY_ADE= 0–0
|match2_MCY_BRI= 4–1
|match2_MCY_CCM= null
|match2_MCY_MVC= null
|match2_MCY_NEW= 2–1
|match2_MCY_PER= 2–2
|match2_MCY_SYD= null
|match2_MCY_WEL= null
|match2_MCY_WSW= null

<!--Melbourne Victory-->
|match1_MVC_ADE= 2–0
|match1_MVC_BRI= 2–1
|match1_MVC_CCM= 4–1
|match1_MVC_MCY= 1–2
|match1_MVC_NEW= 2–1
|match1_MVC_PER= 2–3
|match1_MVC_SYD= 2–1
|match1_MVC_WEL= 1–1
|match1_MVC_WSW= 4–0

|match2_MVC_ADE= null
|match2_MVC_BRI= null
|match2_MVC_CCM= 2–1
|match2_MVC_MCY= 1–1
|match2_MVC_NEW= 0–2
|match2_MVC_PER= 1–2
|match2_MVC_SYD= null
|match2_MVC_WEL= 3–3
|match2_MVC_WSW= null

<!--Newcastle United Jets-->
|match1_NEW_ADE= 1–2
|match1_NEW_BRI= 2–0
|match1_NEW_CCM= 1–0
|match1_NEW_MCY= 3–1
|match1_NEW_MVC= 0–1
|match1_NEW_PER= 0–2
|match1_NEW_SYD= 1–1
|match1_NEW_WEL= 1–1
|match1_NEW_WSW= 3–2

|match2_NEW_ADE= 0–0
|match2_NEW_BRI= 2–2
|match2_NEW_CCM= 2–3
|match2_NEW_MCY= null
|match2_NEW_MVC= null
|match2_NEW_PER= null
|match2_NEW_SYD= 2–0
|match2_NEW_WEL= null
|match2_NEW_WSW= null

<!--Perth Glory-->
|match1_PER_ADE= 0–0
|match1_PER_BRI= 2–1
|match1_PER_CCM= 3–2
|match1_PER_MCY= 1–0
|match1_PER_MVC= 0–2
|match1_PER_NEW= 2–0
|match1_PER_SYD= 1–2
|match1_PER_WEL= 3–0
|match1_PER_WSW= 1–1

|match2_PER_ADE= null
|match2_PER_BRI= 4–0
|match2_PER_CCM= null
|match2_PER_MCY= null
|match2_PER_MVC= null
|match2_PER_NEW= 1–0
|match2_PER_SYD= 3–1
|match2_PER_WEL= 5–0
|match2_PER_WSW= 4–3

<!--Sydney-->
|match1_SYD_ADE= 2–1
|match1_SYD_BRI= 2–1
|match1_SYD_CCM= 5–2
|match1_SYD_MCY= 2–0
|match1_SYD_MVC= 1–2
|match1_SYD_NEW= 1–0
|match1_SYD_PER= 1–0
|match1_SYD_WEL= 1–3
|match1_SYD_WSW= 2–0

|match2_SYD_ADE= 2–0
|match2_SYD_BRI= null
|match2_SYD_CCM= 1–1
|match2_SYD_MCY= 0–2
|match2_SYD_MVC= 2–1
|match2_SYD_NEW= null
|match2_SYD_PER= null
|match2_SYD_WEL= null
|match2_SYD_WSW= null

<!--Wellington Phoenix-->
|match1_WEL_ADE= 1–3
|match1_WEL_BRI= 4–1
|match1_WEL_CCM= 2–0
|match1_WEL_MCY= 1–0
|match1_WEL_MVC= 1–1
|match1_WEL_NEW= 2–1
|match1_WEL_PER= 1–1
|match1_WEL_SYD= 0–1
|match1_WEL_WSW= 0–3

|match2_WEL_ADE= null
|match2_WEL_BRI= null
|match2_WEL_CCM= 3–2
|match2_WEL_MCY= 3–2
|match2_WEL_MVC= null
|match2_WEL_NEW= 4–0
|match2_WEL_PER= null
|match2_WEL_SYD= 0–1
|match2_WEL_WSW= 3–1

<!--Western Sydney Wanderers-->
|match1_WSW_ADE= 1–2
|match1_WSW_BRI= 2–2
|match1_WSW_CCM= 2–0
|match1_WSW_MCY= 0–2
|match1_WSW_MVC= 1–2
|match1_WSW_NEW= 0–2
|match1_WSW_PER= 1–1
|match1_WSW_SYD= 1–3
|match1_WSW_WEL= 2–3

|match2_WSW_ADE= null
|match2_WSW_BRI= null
|match2_WSW_CCM= 2–0
|match2_WSW_MCY= 3–0
|match2_WSW_MVC= 0–1
|match2_WSW_NEW= 1–5 
|match2_WSW_PER= null
|match2_WSW_SYD= 1–1
|match2_WSW_WEL= null

}}
Hur det ser ut
Hemma \ Borta ADE BRI CCM MCY MVC NEW PER SYD WEL WSW ADE BRI CCM MCY MVC NEW PER SYD WEL WSW
Adelaide United 2–1 2–1 0–2 2–0 1–1 0–2 1–1 0–0 2–2 4–3 1–0 0–2 3–1 1–3
Brisbane Roar 3–5 1–1 2–0 2–4 1–6 2–4 2–1 0–0 2–2 0–5 1–3 2–1 1–4
Central Coast Mariners 0–3 1–1 1–1 2–3 1–2 1–4 1–2 2–8 3–1 0–1 3–5 2–1 0–3
Melbourne City 1–1 1–0 5–0 1–1 3–0 0–0 0–3 2–0 4–3 0–0 4–1 2–1 2–2
Melbourne Victory 2–0 2–1 4–1 1–2 2–1 2–3 2–1 1–1 4–0 2–1 1–1 0–2 1–2 3–3
Newcastle United Jets 1–2 2–0 1–0 3–1 0–1 0–2 1–1 1–1 3–2 0–0 2–2 2–3 2–0
Perth Glory 0–0 2–1 3–2 1–0 0–2 2–0 1–2 3–0 1–1 4–0 1–0 3–1 5–0 4–3
Sydney 2–1 2–1 5–2 2–0 1–2 1–0 1–0 1–3 2–0 2–0 1–1 0–2 2–1
Wellington Phoenix Nya Zeeland 1–3 4–1 2–0 1–0 1–1 2–1 1–1 0–1 0–3 3–2 3–2 4–0 0–1 3–1
Western Sydney Wanderers 1–2 2–2 2–0 0–2 1–2 0–2 1–1 1–3 2–3 2–0 3–0 0–1 1–5 1–1
-- Module to build results cross-tables for standings in Sports
-- See documentation for details

require('strict')

local p = {}

-- Main function
function p.main(frame)
	-- Get the args, stripping out blank values
	local getArgs = require('Module:Arguments').getArgs
	local Args = getArgs(frame, {parentFirst = true})

	-- Exit early if we are using section transclusion for a different section
	if (Args['transcludesection'] and Args['section'])
		and Args['transcludesection'] ~= Args['section'] then
		return ''
	end

	-- Declare locals
	local t = {}
	local t_footer = {}
	local t_return = {}
	local team_list = {}
	local notes_exist = false
	local ii, ii_fw, bg_col, team_name, team_code_ii, ii_start, ii_end
	-- Optional custom team header
	local team_header = Args['team_header'] or '[[fil:ArrowDown.svg|9px|link=]] Hemma \\ Borta [[fil:ArrowRight.svg|9px|link=]]'
	-- Number of legs
	local legs = tonumber(Args['legs']) or 1
	local multirowlegs = (Args['multirowlegs'] or 'no') ~= 'no'
	
	-- Edit links if requested
	local baselink = frame:getParent():getTitle()
	if mw.title.getCurrentTitle().text == baselink then	baselink = '' end
	local template_name = Args['template_name']
		or (baselink ~= '' and (':' .. baselink))
		or ''
	local edit_links = template_name == '' and ''
		or frame:expandTemplate{ title = 'navbar',
			args = { mini=1, style='float:right', template_name} }
		
	-- Get the custom start point for the table (most will start by default at 1)
	local top_pos = tonumber(Args['highest_pos']) or 1
	-- Get the custom end point for the table (unrestricted if bottom_pos is < top_pos)
	local bottom_pos = tonumber(Args['lowest_pos']) or 0
	local N_teams = top_pos - 1 -- Default to 0 at start, but higher number needed to skip certain entries

	-- Load some other modules
	local p_sub = require('Module:Sports table/sub')
	
	-- Alternative syntax for team list
	if Args['team_order'] and Args['team_order'] ~= '' then
		local tlist = mw.text.split(Args['team_order'], '%s*[;,]%s*')
		for k, tname in ipairs(tlist) do
			if tname ~= '' then
				Args['team' .. k] = tname
			end
		end
	end
	
	if Args['team_header_note'] then
		notes_exist=true
		local note_string = frame:expandTemplate{ title = 'efn',
								args = { group='lower-alpha', Args['team_header_note']} }
		team_header = team_header .. note_string
	end

	-- Read in number of consecutive teams (ignore entries after skipping a spot)
	ii_start = N_teams
	while Args['team'..N_teams+1] ~= nil and (bottom_pos < top_pos or N_teams < bottom_pos) do
		N_teams = N_teams+1
		-- Sneakily add it twice to the team_list parameter, once for the actual
		-- ranking, the second for position lookup in sub-tables
		-- This is possible because Lua allows both numbers and strings as indices.
		team_list[N_teams] = Args['team'..N_teams] -- i^th entry is team X
		team_list[Args['team'..N_teams]] = N_teams -- team X entry is position i
	end
	ii_end = N_teams
	-- Get team to show
	local ii_show = team_list[Args['showteam']] -- nil if non-existant

	-- Set the font size
	local font_size=Args['font_size'] or '100%'

	-- Create header
	-- Open table
	table.insert(t,'{|class="wikitable plainrowheaders" style="text-align:center;font-size:'..font_size..';"\n')
	-- Table title
	if Args['title'] then
		table.insert(t,'|+ ' .. Args['title'] .. '\n')
	end
	-- First column
	t_return.count = 0 			-- Dummy parameter, using subfunction call seems best at this point because both module are intertwined
	t_return.tab_text = t		-- Actual text
	t_return = p_sub.colhead(t_return,'auto', edit_links .. ' ' .. team_header)
	-- Other columns passed to subfunction
	t_return = p.header(t_return,Args,p_sub,N_teams,team_list,legs,multirowlegs)
	t = t_return.tab_text

	-- Random value used for uniqueness
	math.randomseed( os.clock() * 10^8 )
	local rand_val = math.random()

	local note_string, note_id
	local note_id_list = {}

	-- Now create individual rows
	ii_start = tonumber(Args['highest_row']) and (tonumber(Args['highest_row']) > top_pos) and tonumber(Args['highest_row']) or top_pos
	ii_end = tonumber(Args['lowest_row']) and (tonumber(Args['lowest_row']) < N_teams) and tonumber(Args['lowest_row']) or N_teams
	for ii=ii_start,ii_end do
		-- Get team info
		team_code_ii = team_list[ii]
		team_name = Args['name_'..team_code_ii] or team_code_ii
		local ii_style = 'text-align:' .. (Args['team_align'] or 'right') .. ';'
			.. (ii and ii == ii_show and 'font-weight:bold;' or '')
			.. (Args['team_nowrap'] and 'white-space:nowrap;' or '')
		local team_note = Args['note_'..team_code_ii]
		if team_note then
			notes_exist = true
			-- Only when it exist
			-- First check for existence of reference for note
			if not Args['note_'..team_note] then
				-- It's the entry
				-- Add random end for unique ID if more tables are present on article (which might otherwise share an ID)
				note_id = '"table_note_'..team_code_ii..rand_val..'"'
				note_id_list[team_code_ii] = note_id
				note_string = frame:expandTemplate{ title = 'efn',
					args = { group='lower-alpha', name=note_id,  team_note} }
			else
				-- Check for existence elsewhere
				local note_local_num = team_list[team_note] or ii_end + 1
				if note_id_list[team_note] or ((note_local_num >= ii_start) and (note_local_num <= ii_end)) then
					-- It exists
					note_id = '"table_note_'..team_note..rand_val..'"' -- Identifier
					note_string = frame:extensionTag{ name = 'ref',
						args = { group = 'lower-alpha', name = note_id} }
				else
					-- Now define the identifier for this
					-- Add random end for unique ID
					note_id = '"table_note_'..team_note..rand_val..'"'
					note_id_list[team_note] = note_id
					-- Call refn template
					note_string = frame:expandTemplate{ title = 'efn',
						args = { group='lower-alpha', name=note_id, Args['note_'..team_note]} }
				end
			end
			-- Now append this to the team_name string
			team_name = team_name..note_string
		end
		-- Team names
		table.insert(t,'|- \n')  -- New row
		table.insert(t,'! scope="row"'.. (multirowlegs and ' rowspan=' .. legs or '') 
			.. 'style="'.. ii_style ..'"| '..team_name..'\n')  -- Position number

		-- Now include note to match results if needed
		for jj=top_pos,N_teams do
			local team_code_jj = team_list[jj]
			if ii == jj then
				-- Nothing
			else
				for l=1,legs do
					local m = (legs == 1) and 'match_' or 'match' .. l .. '_'
					local match_note = Args[m ..team_code_ii..'_'..team_code_jj..'_note']
					if match_note then
						notes_exist = true
						-- Only when it exist
						-- First check for existence of reference for note
						if not (Args['note_'..match_note] or Args[m ..match_note..'_note']) then
							-- It's the entry
							-- Add random end for unique ID if more tables are present on article (which might otherwise share an ID)
							note_id = '"table_note_'..team_code_ii..'_'..team_code_jj..rand_val..'"'
							note_id_list[team_code_ii..'_'..team_code_jj] = note_id
							note_string = frame:expandTemplate{ title = 'efn',
								args = { group='lower-alpha', name=note_id,  match_note} }
						else
							-- Check for existence elsewhere
							local note_local_num = team_list[match_note] or ii_end + 1
							if note_id_list[match_note] or ((note_local_num >= ii_start) and (note_local_num <= ii_end)) then
								-- It exists
								note_id = '"table_note_'..match_note..rand_val..'"' -- Identifier
								note_string = frame:extensionTag{ name = 'ref',
									args = { group = 'lower-alpha', name = note_id} }
							else
								-- Now define the identifier for this
								-- Add random end for unique ID
								note_id = '"table_note_'..match_note..rand_val..'"'
								note_id_list[match_note] = note_id
								-- Call refn template
								note_string = frame:expandTemplate{ title = 'efn',
									args = { group='lower-alpha', name=note_id, Args['note_'..match_note]} }
							end
						end
						-- Now append this to the match result string
						Args[m..team_code_ii..'_'..team_code_jj] = (Args[m..team_code_ii..'_'..team_code_jj] or '–')..note_string
					end
				end
			end
		end
		-- Then individual results
		t = p.row(t,Args,N_teams,team_list,ii,ii_show,legs,multirowlegs)
	end

	-- Close table
	table.insert(t, '|}\n')

	-- Get info for footer
	local update = Args['update']
		or 'unknown'
	local start_date = Args['start_date']
		or 'unknown'
	local source = Args['source']
		or frame:expandTemplate{ title = 'citation needed',
			args = { reason='No source parameter defined', date=os.date('%B %Y') } }

	-- Create footer text
	-- Date updating
	if string.lower(update)=='complete' then
		-- Do nothing
	elseif update=='' then
		-- Empty parameter
		table.insert(t_footer,'Uppdaterad efter match(er) spelade okänt datum. ')
	elseif string.lower(update)=='future' then
		-- Future start date
		table.insert(t_footer,'Första matchen/matcherna kommer spelas den '..start_date..'. ')
	else
		table.insert(t_footer,'Uppdaterad efter match(er) spelad(e) den '..update..'. ')
	end
	table.insert(t_footer,'Källa: '..source)
	if (Args['matches_style'] or '') == 'FBR' then
		table.insert(t_footer, Args['team_header']
			and '<br />Färger: Blå = Vinst för hemmalaget; Gul = oavgjort; Röd = Vinst för bortalaget.' 
			or '<br />Färger: Blå = Vinst för hemmalaget; Gul = oavgjort; Röd = Vinst för bortalaget.')
	elseif (Args['matches_style'] or '') == 'BSR' then
		table.insert(t_footer, Args['team_header']
			and '<br />Färger: Blå = Vinst för hemmalaget; Röd = Vinst för bortalaget.'
			or '<br />Färger: Blå = Vinst för hemmalaget; Röd = Vinst för bortalaget.')
	end
	if Args['a_note'] then
		table.insert(t_footer, '<br />För kommande matcher, ett "a" indikerar på att det finns en artikel om lagens tidigare möten.')
	end
	if Args['ot_note'] then
		table.insert(t_footer, '<br />Matcher med en ljusare bakgrundsfärg avgjordes efter förlängning.')
	end
	
	
	local templatestyles = frame:extensionTag{
		name = 'templatestyles', args = { src = 'Module:Sports results/styles.css' }
	}
	
	-- Add notes (if applicable)
	if notes_exist then
		table.insert(t_footer,'<br>Notes:')
		t_footer = templatestyles .. '<div class="sports-results-notes">'..table.concat(t_footer)..'</div>'
		t_footer = t_footer..frame:expandTemplate{ title = 'notelist', args = { group='lower-alpha'} }
	else
		t_footer = templatestyles .. '<div class="sports-results-notes">'..table.concat(t_footer)..'</div>'
	end

	-- Add footer to main text table
	table.insert(t,t_footer)
	
	-- Rewrite anchor links
	for k=1,#t do
		if t[k]:match('%[%[#[^%[%]]*%|') then
			t[k] = mw.ustring.gsub(t[k], '(%[%[)(#[^%[%]]*%|)', '%1' .. baselink .. '%2')
		end
	end
	
	return '<div style="overflow:hidden">'
		.. '<div class="noresize overflowbugx" style="overflow:auto">\n'
		.. table.concat(t) .. '</div></div>'
end

-- Other functions
local function get_short_name(s, t, n, ss)
	-- return short name if defined
	if s and s ~= '' then
		return s
	end
	-- deflag if necessary
	if ss and n then
		if ss == 'noflag' then
			n = mw.ustring.gsub(n, '%[%[[Ff][Ii][Ll][Ee]:[^%[%]]*%]%]', '')
		elseif ss == 'flag' then
			n = mw.ustring.gsub(n, '(<span class="flagicon">%s*%[%[[Ff][Ii][Ll][Ee]:[^%[%]]*link=)[^%|%[%]]*(%]%][^<>]*</span>)%s*%[%[([^%[%]%|]*)%|[^%[%]]*%]%]', '%1%3%2')
			n = mw.ustring.gsub(n, '.*(<span class="flagicon">%s*%[%[[Ff][Ii][Ll][Ee]:[^%[%]]*%]%][^<>]*</span>).*', '%1')
			n = mw.ustring.gsub(n, '&nbsp;(</span>)', '%1')
		end
	end
	
	-- replace link text in name with team abbr if possible
	if n and t and n:match('(%[%[[^%[%]]*%]%])') then
		n = mw.ustring.gsub(n, '(%[%[[^%|%]]*%|)[^%|%]]*(%]%])', '%1' .. t .. '%2')
		n = mw.ustring.gsub(n, '(%[%[[^%|%]]*)(%]%])', '%1|' .. t .. '%2')
		n = mw.ustring.gsub(n, '(%[%[[^%|%]]*%|)([A-Z][A-Z][A-Z])(%]%])&nbsp;<span[^<>]*>%([A-Z][A-Z][A-Z]%)</span>', '%1%2%3')
		return n
	end
	-- nothing worked, so just return the unlinked team abbr
	return t or ''
end

local function get_score_background(s, c)
	local s1, s2
	-- Define the colouring
	local wc, lc, tc
	if c == 'level2' then
	wc, lc, tc = '#CCF9FF', '#FCC', '#FFC' -- blue2, red2, yellow2
	elseif c == 'level3' then
	wc, lc, tc = '#DDFCFF', '#FDD', '#FFD' -- blue3, red3, yellow3
	elseif c == 'level4' then
	wc, lc, tc = '#EEFFFF', '#FEE', '#FFE' -- blue4, red4, yellow4
	else
	wc, lc, tc = '#BBF3FF', '#FBB', '#FFB' -- blue1, red1, yellow1
	end

	-- check for override
	if s:match('^%s*<span%s%s*style%s*=["\'%s]*background[%-colr]*%s*:([^\'";<>]*).-$') then
		local c = mw.ustring.gsub(s,'^%s*<span%s%s*style%s*=["\'%s]*background[%-colr]*%s*:([^\'";<>]*).-$', '%1')
		return c
	end
	
	-- delink if necessary
	if s:match('^%s*%[%[[^%[%]]*%|([^%[%]]*)%]%]') then
		s = s:match('^%s*%[%[[^%[%]]*%|([^%[%]]*)%]%]')
	end
	if s:match('^%s*%[[^%[%]%s]*%s([^%[%]]*)%]') then
		s = s:match('^%s*%[[^%[%]%s]*%s([^%[%]]*)%]')
	end
	if s:match('<span[^<>]*>(.-)</span>') then
		s = s:match('<span[^<>]*>(.-)</span>')
	end

	-- get the scores
	s1 = tonumber(mw.ustring.gsub( s or '',
		'^%s*([%d%.][%d%.]*)%s*–%s*([%d%.][%d%.]*).*', '%1' ) or '') or ''
	s2 = tonumber(mw.ustring.gsub( s or '',
		'^%s*([%d%.][%d%.]*)%s*–%s*([%d%.][%d%.]*).*', '%2' ) or '') or ''

	-- return colouring if possible
	if s1 ~= '' and s2 ~= '' then
		return (s1 > s2) and wc or ((s2 > s1) and lc or tc)
	else
		return 'transparent'
	end
end

local function format_score(s)
	s = mw.ustring.gsub(s or '', '^%s*([%d%.]+)%s*[–−—%-]%s*([%d%.]+)', '%1–%2')
	s = mw.ustring.gsub(s, '^%s*([%d%.]+)%s*&[MmNn][Dd][Aa][Ss][Hh];%s*([%d%.]+)', '%1–%2')
	s = mw.ustring.gsub(s, '^%s*(%[%[[^%[%]]*%|[%d%.]+)%s*%-%s*([%d%.]+)', '%1–%2')
	s = mw.ustring.gsub(s, '^%s*(%[[^%[%]%s]*%s+[%d%.]+)%s*%-%s*([%d%.]+)', '%1–%2')
	s = mw.ustring.gsub(s, '^%s*(%[%[[^%[%]]*%|[%d%.]+)%s*&[MmNn][Dd][Aa][Ss][Hh];%s*([%d%.]+)', '%1–%2')
	s = mw.ustring.gsub(s, '^%s*(%[[^%[%]%s]*%s+[%d%.]+)%s*&[MmNn][Dd][Aa][Ss][Hh];%s*([%d%.]+)', '%1–%2')
	return s
end

function p.header(tt,Args,p_sub,N_teams,team_list,legs,multirowlegs)
	local ii, team_code_ii, short_name
	legs = legs or 1

	-- Set match column width
	local col_width = Args['match_col_width'] or '28'

	-- Get some default values in case it doesn't start at 1
	local top_pos = tonumber(Args['highest_pos']) or 1

	for l=1,legs do
		if multirowlegs and l > 1 then
			break
		end
		for ii=top_pos,N_teams do
			team_code_ii = team_list[ii]
			short_name = get_short_name(Args['short_'..team_code_ii],
				team_code_ii, Args['name_'..team_code_ii], Args['short_style'] or '')
			local bl = legs > 1 and ii == top_pos and ' style="border-left:2px solid #aaa;"' or ''
			tt = p_sub.colhead(tt,col_width .. bl,short_name)
		end
	end
	return tt
end

function p.row(tt,Args,N_teams,team_list,ii,ii_show,legs,multirowlegs)
	-- Note ii is the row number being shown
	local jj, fw, bg, result, result_extra, team_code_ii, team_code_jj
	legs = legs or 1

	-- Set score cell style
	local matches_style = Args['matches_style'] or ''

	team_code_ii = team_list[ii]

	-- Get some default values in case it doesn't start at 1
	local top_pos = tonumber(Args['highest_pos']) or 1
	for l=1,legs do
		if multirowlegs and l > 1 then
			table.insert(tt,'|- \n')  -- New row
		end
		for jj=top_pos,N_teams do
			team_code_jj = team_list[jj]
			local m = (legs == 1) and 'match_' or 'match' .. l .. '_'
			result = Args[m..team_code_ii..'_'..team_code_jj] or ''
			result_extra = Args['result_'..team_code_ii..'_'..team_code_jj] or ''
			local bl = legs > 1 and jj == top_pos and 'border-left:2px solid #aaa;' or ''

			if ii == jj or result == 'null' then
				-- Solid cell
				fw = 'font-weight:' .. (ii==ii_show and 'bold' or 'normal') .. ';'
				bg = 'background:transparent;'

				-- Grey background color for solid cell
				if Args['solid_cell'] == 'grey' then
					table.insert(tt,'| style="'..fw..bl..'background:#bbb;" |\n')
				else
					table.insert(tt,'| style="'..fw..bl..bg..'" | &mdash;\n')
				end
			else
				-- Content cell
				-- Set bolding and background
				fw = 'font-weight:' .. ((ii==ii_show or jj == ii_show) and 'bold' or 'normal') .. ';'
				bg = 'background:transparent;'

				-- Reformat dashes
				if result ~= '' then
					result = format_score(result)
				end
				-- Background coloring if enabled
				if matches_style == 'FBR' and result ~= '' then
					if result_extra == 'OT' then
						bg = 'background:' .. get_score_background(result,'level2') .. ';'
					elseif result_extra == 'PK' then
						bg = 'background:' .. get_score_background(result,'level3') .. ';'
					else
						bg = 'background:' .. get_score_background(result,'') .. ';'
					end
				elseif matches_style == 'BSR' and result ~= '' then
					if result_extra == 'OT' then
						bg = 'background:' .. get_score_background(result,'level3') .. ';'
					elseif result_extra == 'OTL' then
						bg = 'background:' .. get_score_background('0–1','level3') .. ';'
					elseif result_extra == 'OTW' then
						bg = 'background:' .. get_score_background('1–0','level3') .. ';'
					elseif result_extra == 'L' then
						bg = 'background:' .. get_score_background('0–1','') .. ';'
					elseif result_extra == 'W' then
						bg = 'background:' .. get_score_background('1–0','') .. ';'
					else
						bg = 'background:' .. get_score_background(result,'') .. ';'
					end
				end
				table.insert(tt,'| style="white-space:nowrap;'..fw..bl..bg..'" |'..result..'\n')
			end
		end
	end
	
	return tt
end

return p