Jump to content

Module:fi-dialects/template/map

From Wiktionary, the free dictionary

Implements {{fi-dial-map}}.


local export = {}
local m_dial = require("Module:fi-dialects")
local m_common = require("Module:fi-dialects/template/common")

local USE_MAPFRAME = false -- experimental
local MAP_SIZE = 1200
local m_map, map_size, map_width, map_height

if USE_MAPFRAME then
	m_map = require("Module:fi-dialects/map/mapframe")
	local ASPECT_RATIO = 1.1875
	map_width = tostring(MAP_SIZE)
	map_height = tostring(MAP_SIZE * ASPECT_RATIO)
else
	m_map = require("Module:fi-dialects/map")
	map_size = MAP_SIZE .. "px"
end

local dots = {
	"FF150F", "0D8AFF", "59FF0D", "FF0FF1", "E4FF10", 
	"10C7FF", "8210FF", "FF8E0A", "07FFD1", "380DFF", 
	"874F4E", "4C7183", "62814D", "804C7A", "78814D", 
	"4D8178", "694D86", "826949", "57844F", "574D82",
	"FF4E45", "4ABBFF", "8DFF49", "FF42F9", "DAFF4C", 
	"4AFFEA", "9F48FF", "FFB14C", "68FF4C", "6549FF", 
	"783017", "164F73", "307313", "7E187D", "5D7612", 
	"167364", "411777", "734E16", "227416", "31167A", 
}

-- go through synonyms, assign colors to each term, and compile lists to syns.
local function visit(syns, visited, parish, ...)
	local terms = {...}
	if parish then syns[parish] = terms end
	for _, term_w in ipairs(terms) do
		local term = term_w
		if not visited[term] then
			local next_index = #visited + 1
			table.insert(visited, term)
			visited[term] = dots[next_index]
		end
	end
end

-- makes a "chip" used for the color legend.
local function make_chip(text, color, qualifier)
	if qualifier then text = text .. " " .. require("Module:qualifier").format_qualifier(qualifier) end
	return '<span class="color" style="white-space:nowrap;margin:4px;padding:4px;border:1px solid #' .. color .. ';border-left:2em solid #' .. color .. ';line-height:2.5em">' .. text .. "</span>"
end

-- same as make_chip, but tag "term".
local function make_chip_tag(text, color, qualifier)
	if qualifier then
		suffix = " " .. require("Module:qualifier").format_qualifier(qualifier)
	else
		suffix = ""
	end
	return '<span class="color" style="white-space:nowrap;margin:4px;padding:4px;border:1px solid #' .. color .. ';border-left:2em solid #' .. color .. ';line-height:2.5em"><span class="Latn" lang="fi">' .. text .. "</span>" .. suffix .. "</span>"
end

-- same as make_chip, but link "term".
local function make_chip_link(target, text, id, color, qualifier)
	return make_chip(m_common.link(target, text, id), color, qualifier)
end

local function make_wiki_link(target, alt, html)
	if not target then return html end
	return "[[w:" .. target .. "|" .. tostring(mw.html.create("span"):attr("title", alt):wikitext(html)) .. "]]"
end

local function make_formatted_link(term, alt, html)
	local text, id = m_common.parse_term(term)
	local id_suffix = ""
	if id then id_suffix = ":_" .. mw.ustring.gsub(id, " ", "_") end
	return tostring(mw.html.create("span"):attr("title", alt):wikitext("[[" .. text .. "#Finnish" .. id_suffix .. "|" .. html .. "]]"))
end

local function colors_to_css(colors)
	if type(colors) == "string" then return colors end
	if #colors == 1 then return colors[1] end

	-- create "pie chart"
	local sector = 360 / #colors
	local result = nil
	local fmt = sector == math.floor(sector) and "%0.0f" or "%0.4f"

	for factor, color in ipairs(colors) do
		result = (result and result .. "," or "") .. color .. " " .. string.format(fmt, sector * (factor - 1)) .. "deg " .. string.format(fmt, sector * factor) .. "deg"
	end

	return "conic-gradient(" .. result .. ")"
end

local function fi_sort(a, b)
	return m_common.get_sort_key(a) < m_common.get_sort_key(b)
end

local function make_map_parish_table(parishes_by_form, chip_by_term, label_order)
	local parish_table = nil
    local get_parish = function (p) return m_dial.getParish(p, true) end
	local visited = {}

	local function visit_for_table(form)
		if not form or visited[form] then return end
		visited[form] = true
		local parishes = parishes_by_form[form]
		if not parishes or #parishes == 0 then return end
		table.sort(parishes, function (a, b)
			local pa = get_parish(a)
			local pb = get_parish(b)
			if pa then pa = pa:getFormattedName() end
			if pb then pb = pb:getFormattedName() end
			return (pa or a) < (pb or b)
		end)

		if parish_table then
			parish_table = parish_table .. '\n|-\n'
		else
			parish_table = '{| class="wikitable"\n'
		end

		local formatted_parishes = {}
		for _, parish in ipairs(parishes) do
			local parish_name, parish_tooltip
			local obj = get_parish(parish)
			if obj then
				parish_name = obj:getFormattedName()
				parish_tooltip = obj:getFormattedName() .. ', ' .. obj:getArea():getFormattedName()
			else
				parish_name = parish
				parish_tooltip = nil
			end
			if parish_tooltip then
				table.insert(formatted_parishes, tostring(mw.html.create('span'):wikitext(parish_name):attr('title', parish_tooltip)))
			else
				table.insert(formatted_parishes, parish_name)
			end
		end
		parish_table = parish_table .. '| ' .. (chip_by_term[form] or form) .. '\n| ' .. table.concat(formatted_parishes, ', ')
	end

	if label_order then
		for _, form in ipairs(label_order) do
			visit_for_table(form)
		end
	end

	local forms_list = {}
	for form, parishes in pairs(parishes_by_form) do
		table.insert(forms_list, form)
	end

	table.sort(forms_list, fi_sort)
	for _, form in ipairs(forms_list) do
		visit_for_table(form)
	end

	if parish_table then
		parish_table = parish_table .. '\n|}'
	end
	return parish_table or "''(no forms)''"
end

local function synonym_map(frame, term, word_id, data, is_feature)
	local synonyms = data[is_feature and "data" or "syns"]
	local source = data.source and (type(data.source) == "table" and data.source or { data.source }) or { }
	
	local parishes = { }
	local syns = { }
	local colors = { }
	
	local has_custom_order = is_feature and data.label_order
	
	if has_custom_order then
		for _, label in ipairs(data.label_order) do
			visit(syns, colors, nil, label)
		end
	end
	
	-- gather parishes and their terms
	for parish, terms in pairs(synonyms) do
		if not m_common.special[parish] then
			local result = m_dial.getParish(parish, true)
			if result then
				table.insert(parishes, result)
			end
	
			-- assign each term a color
			if type(terms) == "string" then
				visit(syns, colors, parish, terms)
			else
				-- loadData breaks unpack, we must make a copy
				local terms_copy = {}
				for i, term in ipairs(terms) do terms_copy[i] = term end
				visit(syns, colors, parish, unpack(terms_copy))
			end
		end
	end
	
	if not has_custom_order then
		-- sort and make chips
		table.sort(colors, fi_sort)
	end
	
	local qualifiers = data.qualifiers or { }
	local chips = { }
	local chip_by_term = { }
	for _, term in ipairs(colors) do
		local chip
		if is_feature or data.semantic then
			chip = make_chip(data.labels[term] or term, colors[term], qualifiers[term])
		else
			local text, id = m_common.parse_term(term)
			if data.nolink then
				chip = make_chip_tag(data.labels and data.labels[term] or text, colors[term], qualifiers[term])
			else
				local target = data.links and data.links[term] or text
				local label = data.labels and (data.labels and data.labels[term] or nil) or text
				chip = make_chip_link(target, label, id, colors[term], qualifiers[term])	
			end
		end
		chip_by_term[term] = chip
		table.insert(chips, chip)
	end
	chips = table.concat(chips, " ")
	
	local map

	local function pin_text(parish)
		local terms = syns[parish:getCode()]
		local term_texts = {}
		if is_feature then
			for i, term in ipairs(terms) do term_texts[i] = data.labels[term] or term end
		else
			for i, term in ipairs(terms) do term_texts[i] = m_common.extract_text(term) end
		end
		return parish:getFormattedName() .. ', ' .. parish:getArea():getFormattedName() .. ':\n' .. table.concat(term_texts, is_feature and "; " or ", ")
	end

	local function pin_color(term)
		return '#' .. (colors[term] or "000000")
	end

	if USE_MAPFRAME then
		local function make_pin(parish)
			local area = parish:getArea()
			local group = area:getGroup()
			if not group then return nil end
			local branch = group:getBranch()

			local lat, lon = parish:getCoordinates()
			local terms = syns[parish:getCode()]
			
			if #terms < 1 then return nil end

			local features = {}
			-- TODO: need a better solution
			
			local i = 0
			local f = 2 * math.pi / #terms
			if #terms > 1 then
				a = 0.0001
			else
				a = 0.0
			end
			
			for _, term in ipairs(terms) do
				table.insert(features, {
					type = "Feature",
					properties = {
						title = pin_text(parish, area, group, branch),
						["marker-color"] = pin_color(term),
						["marker-size"] = "small"
					},
					geometry = {
						type = "Point",
						coordinates = { lon + a * math.sin(i * f), lat + a * math.cos(i * f) }
					}
				})
				i = i + 1
			end
			return features		
		end
		map = m_map.show{frame = frame, parishes = parishes, pin = make_pin, width = map_width, height = map_height}

	else
		-- complicated peg
		local function render_dot(parish, top, left)
			local terms = syns[parish:getCode()]
			local alt = pin_text(parish)

			local outer = mw.html.create('div')
						:attr('class', 'dot_outer')
						:css('position', 'absolute')
						:css('top', top)		-- positioning
						:css('left', left)
						:tag('div')
							:css('position', 'relative')
							:css('left', '-4px')		-- center (8px / 2 = 4px)
							:css('top', '-4px')
							:css('width', '8px')
							:css('height', '8px')
							:attr('title', alt)

			local color = pin_color(terms[1])
			local multiple_targets = false
			if #terms > 1 then
				local first_link = data.links and data.links[terms[1]] or terms[1]
				color = { color }
				for i = 2, #terms do
					table.insert(color, pin_color(terms[i]))
					multiple_targets = multiple_targets or (not data.links or (data.links[terms[i]] or terms[i]) ~= first_link)
				end
			end

			local dot = outer:tag('span')
					:css('width', '8px')
					:css('height', '8px')
					:css('border-radius', '50%')
					:css('user-select', 'none')
					:css('display', 'inline-block')
					:css('background', colors_to_css(color))
					:css('border', '0.5px solid rgba(0,0,0,0.25)')
					:wikitext('&nbsp;')
			if is_feature or multiple_targets or data.nolink or data.semantic then return tostring(dot:allDone()) end
			return make_formatted_link(terms[1], alt, tostring(dot:allDone()))
		end

		map = m_map.show{frame = frame, parishes = parishes, peg = render_dot, size = map_size} .. require("Module:TemplateStyles")("Module:fi-dialects/style.css")
	end
		
	local heading
	if is_feature then
		heading = data.title or 'Feature map'
	elseif data.semantic then
		heading = 'Dialectal meanings for ' .. m_common.mention(term, data.gloss, data.usage)
	else
		heading = 'Dialectal synonyms for ' .. m_common.mention(term, data.gloss, data.usage)
	end
	
	local note
	if not is_feature and synonyms.common then
		note = "''" .. 'The most commonly found form in dialects is ' .. m_common.mention(synonyms.common) .. '. The map below might not show all parishes where this form is attested.' .. "''"
	end

	local parishes_by_form = { }
	for parish_name, parish_forms in pairs(syns) do
		for _, parish_form in ipairs(parish_forms) do
			parishes_by_form[parish_form] = (parishes_by_form[parish_form] or {})
			table.insert(parishes_by_form[parish_form], parish_name)
		end
	end
	local parish_table = make_map_parish_table(parishes_by_form, chip_by_term, has_custom_order)
	
	return '<div style="float:right;"><small>([[Module:fi-dialects/data/' .. (is_feature and 'feature' or 'word') .. '/' .. word_id .. '|edit data]])</small></div>' ..
			"<p>''" .. m_common.disclaimer .. "''</p>" ..
			'<div style="float:right;"><small>([[commons:File:Finnish dialect location map.svg|background image]])</small></div>' ..
			m_common.format_sources(source, true) .. '\n\n' ..
			'== ' .. heading .. " ==\n" .. (note and note .. "\n" or "") .. [[
{| style="width:100%;" cellspacing="3" cellpadding="5"
|-
| align="center" |
<div style="display:inline-block;">

<p>]] .. chips .. [[</p>

]] .. map .. [[

</div>
|}

=== List ===
]] .. parish_table .. [[

]] .. "\n" .. (is_feature and "" or ("[[Category:Finnish dialect maps|" .. word_id .. "]]"))
end

local function west_east_map(frame)
	local fallback_colors = { ["west"] = "1192a6", ["east"] = "8a06a1" }

	local branch_labels = { ["west"] = "Western Finnish", ["east"] = "Eastern Finnish" }

	local parish_data = mw.loadData("Module:fi-dialects/data/parish").parishes
	
	local west_chips = {}
	local east_chips = {}
	
	local display_groups = frame.args["groups"]

	local palette, xref

	if display_groups then
		local group_colors = {
			-- west (cold colors)
			["Southwest"] = "0080ff",
			["SouthwestTransitional"] = "00d0ff",
			["Tavastia"] = "1cff68",
			["SouthOstrobothnia"] = "2a00fc", 
			["NorthOstrobothnia"] = "5193fc",
			["Lapland"] = "b3d2ff",
			
			-- east (warm colors)
			["Savonia"] = "ffa200",
			["Southeast"] = "e81f00",
		}

		local group_keys = {
			"Southwest", "SouthwestTransitional", "Tavastia",
			"SouthOstrobothnia", "NorthOstrobothnia", "Lapland",
			"Savonia", "Southeast",
		}

		for _, group_code in ipairs(group_keys) do
			local group = m_dial.getGroup(group_code)
			local chips
    		if group:getBranch() == "east" then
    			chips = east_chips
    		elseif group:getBranch() == "west" then
    			chips = west_chips
    		end
    		if chips then
    			table.insert(chips, make_chip(group:getFormattedName(), group_colors[group_code] or fallback_colors[group:getBranch()]))
    		end
		end

		palette = group_colors
		xref = "For a map showing dialect areas rather than groups, see [[Template:fi-dial-map]]."
	else
		local area_colors = {
			-- west (cold colors)
			["Länsi-Pohja"] = "7682cf",
			["Peräpohjola"] = "b3d2ff",
			["Pohjanmaa/Pohjoinen"] = "448bfc", 
			["Pohjanmaa/Keski"] = "095fe8",
			["Pohjanmaa/Etelä"] = "043eb3",
			["Satakunta/Länsi"] = "6be1f2", 
			["Satakunta/Pohjoinen"] = "09bd87",
			["Satakunta/Etelä"] = "2898a8",
			["Varsinais-Suomi/Etelä"] = "448bfc",
			["Varsinais-Suomi/Itä"] = "0677bf",
			["Varsinais-Suomi/Pohjoinen"] = "40aff5", 
			["Varsinais-Suomi/Ylämaa"] = "07c5e0", 
			["Häme/Pohjoinen"] = "52eb82", 
			["Häme/Etelä"] = "21d133",
			["Häme/Kaakko"] = "089908",
			["Kymenlaakso"] = "9acf29",
			
			-- east (warm colors)
			["Keski-Suomi/Pohjoinen"] = "e6c053",
			["Keski-Suomi/Länsi"] = "d1a62a",
			["Keski-Suomi/Etelä"] = "b88c0f",
			["Kainuu"] = "e0de92",
			["Savo/Pohjoinen"] = "e8913a",
			["Savo/Etelä"] = "c26c15",
			["Karjala/Pohjoinen"] = "fc7b12",
			["Karjala/Keski"] = "f2272d",
			["Karjala/Etelä"] = "b81102",
			["Inkeri"] = "9c3b68",
			["Vermlanti"] = "fcf40d",
		}

		local area_colors_sorted = {}
	
		for k, _ in pairs(area_colors) do
			table.insert(area_colors_sorted, k)
		end
	
		table.sort(area_colors_sorted, function (a_code, b_code) 
			local a = m_dial.getArea(a_code)
			local b = m_dial.getArea(b_code)
	
			local a_key = a:getEnglishName()
			local b_key = b:getEnglishName()
			
			-- make sure larger areas like Tavastia stay together
			if a:getSuperArea() then
				a_key = a:getSuperArea():getEnglishName() .. "/" .. a_key
			end
			if b:getSuperArea() then
				b_key = b:getSuperArea():getEnglishName() .. "/" .. b_key
			end
			
			return a_key < b_key
		end)

		for _, area_code in ipairs(area_colors_sorted) do
			local area = m_dial.getArea(area_code)
			local chips = area:getBranch() == "east" and east_chips or west_chips
			table.insert(chips, make_chip(area:getFormattedName(), area_colors[area_code] or fallback_colors[area:getBranch()]))
		end

		palette = area_colors
		xref = "For a map showing dialect groups rather than areas, see [[Template:fi-dial-map/groups]]."
	end

	west_chips = table.concat(west_chips, " ")
	east_chips = table.concat(east_chips, " ")
	
	local parishes = {}
	for k, v in pairs(parish_data) do
		table.insert(parishes, m_dial.getParish(k))
	end
	
	local map

	local function pin_text(parish, area, group, branch)
		area = area or parish:getArea()
		group = group or area:getGroup()
		branch = branch or group:getBranch()
		return parish:getFormattedName() .. ",\n" .. area:getFormattedName() .. ",\n" .. group:getFormattedName() .. ",\n" .. branch_labels[branch]
	end

	local function pin_color(parish, area, group, branch)
		area = area or parish:getArea()
		group = group or area:getGroup()
		return palette[display_groups and group:getCode() or area:getCode()] or fallback_colors[branch]
	end

	if USE_MAPFRAME then
		local function make_pin(parish)
			local area = parish:getArea()
			local group = area:getGroup()
			if not group then return nil end
			local branch = group:getBranch()

			local lat, lon = parish:getCoordinates()

			return {
				type = "Feature",
				properties = {
					title = pin_text(parish, area, group, branch),
					["marker-color"] = pin_color(parish, area, group, branch),
					["marker-size"] = "small"
				},
				geometry = {
					type = "Point",
					coordinates = { lon, lat }
				}
			}		
		end
		map = m_map.show{frame = frame, parishes = parishes, pin = make_pin, width = map_width, height = map_height}

	else
		local function color_peg(parish, top, left)
			local area = parish:getArea()
			local group = area:getGroup()
			if not group then return nil end

			local branch = group:getBranch()
			local alt = pin_text(parish, area, group, branch)
			local color = pin_color(parish, area, group, branch)

			return make_wiki_link(parish:getWikipediaArticle(true), alt, tostring(mw.html.create('div')
						:attr('class', 'dot_outer')
						:css('position', 'absolute')
						:css('top', top)		-- positioning
						:css('left', left)
						:tag('div')
							:css('position', 'relative')
							:css('left', '-4px')		-- center (8px / 2 = 4px)
							:css('top', '-4px')
							:css('width', '8px')
							:css('height', '8px')
							:attr('title', alt)
							:tag('span')
								:css('width', '8px')
								:css('height', '8px')
								:css('border-radius', '50%')
								:css('user-select', 'none')
								:css('display', 'inline-block')
								:css('border', '0.5px solid rgba(0,0,0,0.25)')
								:css('background', '#' .. color)
								:wikitext('&nbsp;')
								:done()
							:done()))
		end
		map = m_map.show{frame = frame, parishes = parishes, peg = color_peg, size = map_size}
	end
	
	return '<div style="float:right;"><small>([[Module:fi-dialects/data/parish|edit data]])</small></div>' .. 
			"<p>''" .. m_common.disclaimer .. "''</p>\n\n" ..
			'<p>Each spot on the map is a parish, with some minor exceptions; see [[Appendix:Finnish dialects]]. ' .. xref .. '</p>\n\n' ..
			'<small>Sources: Data of parishes and their areas is based on data from [https://kaino.kotus.fi/sms/ Suomen murteiden sanakirja] © Kotimaisten kielten keskus, under the CC BY 4.0 license. Location data is partially extracted from [https://www.openstreetmap.org/ OpenStreetMap] © OpenStreetMap contributors, under the Open Database license. See the information for the [[commons:File:Finnish dialect location map.svg#Summary|background image]] for its sources and licensing.</small>\n' ..
			'== Map of Finnish dialects ==\n' .. [[
{| style="width:100%;" cellspacing="3" cellpadding="5"
|-
| align="center" |
<div style="display:inline-block;">

<p>'''Western Finnish''': ]] .. west_chips .. [[</p>

<p>'''Eastern Finnish''': ]] .. east_chips .. [[</p>

<div style="display:inline-block;">

]] .. map .. [[

</div></div>
|}
__NOTOC__
== Dialect list ==
]] .. require("Module:fi-dialects/list").embed_dialect_list("===") .. "\n[[Category:Finnish dialect maps| ]]"
end

function export.show_map(frame)
	local word_id
	local title_text = mw.title.getCurrentTitle().text
	local is_feature = false
	if mw.title.getCurrentTitle().namespace == 10 and title_text == "fi-dial-map/groups" then
		if not frame.args["groups"] then error("groups=1 required") end
		return west_east_map(frame)
	elseif mw.title.getCurrentTitle().namespace == 10 and mw.ustring.find(title_text, "^fi%-dial%-map/") then
		word_id = mw.ustring.gsub(title_text, "^fi%-dial%-map/", "")
	elseif mw.title.getCurrentTitle().namespace == 10 and title_text == "fi-dial-map" then
		if frame.args["groups"] then error("groups not allowed") end
		return west_east_map(frame)
	else
		error("This template can only be used in subpages of [[Template:fi-dial-map]]")
	end
	
	if mw.ustring.find(word_id, "^feature/") then
		is_feature = true
		word_id = mw.ustring.gsub(word_id, "^feature/", "")
	end
	
	local title = word_id
	if mw.ustring.find(title, "%(") then
		title = mw.ustring.match(title, "^[^(]+")
	end
	local module_name = "Module:fi-dialects/data/" .. (is_feature and "feature" or "word") .. "/" .. word_id
	local data_ok, data = pcall(function() return mw.loadData(module_name) end)
	if not data_ok then
		return "<div><em>No data found. ([[" .. module_name .. "|Add some]].)</em></div>" .. require("Module:utilities").format_categories("fi-dial-map missing data")
	end
	return synonym_map(frame, title, word_id, data, is_feature)
end

return export