local export = {}
local lang = require("Module:languages").getByCode("se")
local m_str_utils = require("Module:string utilities")
local find = m_str_utils.find
local gmatch = m_str_utils.gmatch
local gsub = m_str_utils.gsub
local len = m_str_utils.len
local lower = m_str_utils.lower
local sub = m_str_utils.sub
local u = require("Module:string/char")
local BREVE = u(0x0306)
local letters_phonemes = {
["ˈ"] = "ː",
["a"] = "a", ["ạ"] = "a",
["á"] = "aː", ["áˈ"] = "aˑ", ["á" .. BREVE ] = "a",
["b"] = "b",
["c"] = "t͡s",
["č"] = "t͡ʃ",
["d"] = "d",
["đ"] = "ð",
["ẹ"] = "e", ["ē"] = "eː",
["g"] = "ɡ",
["ī"] = "iː",
["kh"] = "kʰ",
["ọ"] = "o", ["ō"] = "oː",
["ph"] = "pʰ",
["š"] = "ʃ",
["th"] = "tʰ",
["ŧ"] = "θ",
["ū"] = "uː",
["z"] = "d͡z",
["ž"] = "d͡ʒ",
["ea"] = "ea̯", ["eaˈ"] = "e̯a", ["ea" .. BREVE] = "ĕă̯",
["ie"] = "ie̯", ["ieˈ"] = "i̯e", ["ie" .. BREVE] = "ĭĕ̯",
["oa"] = "oɑ̯", ["oaˈ"] = "o̯ɑ", ["oa" .. BREVE] = "ŏɑ̯̆",
["uo"] = "uo̯", ["uoˈ"] = "u̯o", ["uo" .. BREVE] = "ŭŏ̯",
["dj"] = "ɟ",
["lj"] = "ʎ",
["nj"] = "ɲ",
["hj"] = "j̥", ["hjj"] = "j̥.j̥", ["hjˈj"] = "j̥ː.j̥",
["hl"] = "l̥", ["hll"] = "l̥.l̥", ["hlˈl"] = "l̥ː.l̥",
["hm"] = "m̥", ["hmm"] = "m̥.m̥", ["hmˈm"] = "m̥ː.m̥",
["hn"] = "n̥", ["hnn"] = "n̥.n̥", ["hnˈn"] = "n̥ː.n̥",
["hr"] = "r̥", ["hrr"] = "r̥.r̥", ["hrˈr"] = "r̥ː.r̥",
}
-- This adds letters_phonemes["e"] = "e", letters_phonemes["i"] = "i", etc.
for letter in gmatch("efhijklmnŋoprstuv", ".") do
letters_phonemes[letter] = letter
end
-- Preaspirated
for letter in gmatch("ptcčk", ".") do
letters_phonemes["h" .. letter] = "h" .. letters_phonemes[letter]
letters_phonemes["h" .. letter .. letter] = "hː" .. letters_phonemes[letter]
end
local function get_phoneme(remainder)
-- Find the longest string of letters that matches a recognised sequence in the list
local longestmatch = ""
for letter, _ in pairs(letters_phonemes) do
if sub(remainder, 1, len(letter)) == letter and len(letter) > len(longestmatch) then
longestmatch = letter
end
end
if len(longestmatch) > 0 then
return longestmatch, sub(remainder, len(longestmatch) + 1)
else
return sub(remainder, 1, 1), sub(remainder, 2)
end
end
local function get_syllable(remainder)
local syll = {cons = {}, vowel = ""}
local cons
while find(remainder, "^([^aạáeẹēiīoọōuū]+)") do
cons, remainder = get_phoneme(remainder)
if cons == "nˈnj" then
require("Module:debug").track("se-IPA/nnj")
end
if cons == "ˈ" then
syll.cons.quantity = 3
else
if cons == "dj" or cons == "lj" then
if syll.cons[#syll.cons] == string.sub(cons, 1, 1) then
syll.cons[#syll.cons] = cons
syll.cons.quantity = 3
else
table.insert(syll.cons, cons)
end
elseif cons == "nj" and syll.cons[#syll.cons] == "n" then
syll.cons[#syll.cons] = "nj"
end
table.insert(syll.cons, cons)
end
end
if find(remainder, "^([aạáeẹēiīoọōuū]+)") then
syll.vowel, remainder = get_phoneme(remainder)
end
if remainder == "" then
remainder = nil
end
-- Determine consonant quantity
if not syll.cons.quantity then
if not syll.cons[2] then
syll.cons.quantity = 1
else
if find(syll.cons[#syll.cons], "(.)%1$") or (syll.cons[#syll.cons] == syll.cons[#syll.cons - 1] and not find(syll.cons[#syll.cons], "^[bdgzž]$")) or (syll.cons[#syll.cons - 1] == "p" and syll.cons[#syll.cons] == "m") or (syll.cons[#syll.cons - 1] == "t" and syll.cons[#syll.cons] == "n") or (syll.cons[#syll.cons - 1] == "t" and syll.cons[#syll.cons] == "nj") or (syll.cons[#syll.cons - 1] == "k" and syll.cons[#syll.cons] == "ŋ") then
syll.cons.quantity = 2
else
syll.cons.quantity = 3
end
end
end
return syll, remainder
end
-- Split the word into syllables of C(C)V shape
local function split_syllables(remainder)
remainder = lower(remainder)
remainder = gsub(remainder, "([aáeēiīoōuū])i", "%1j")
local syllables = {}
local syll
while remainder do
syll, remainder = get_syllable(remainder)
table.insert(syllables, syll)
end
syllables.count = #syllables
if syllables[#syllables].vowel == "" then
syllables.count = syllables.count - 1
end
return syllables
end
local function shorten(vowel)
vowel = gsub(vowel, "^[ēīōū]$", {["ē"] = "e", ["ī"] = "i", ["ō"] = "o", ["ū"] = "u"})
for _, v in ipairs({"á", "ea", "ie", "oa", "uo"}) do
vowel = gsub(vowel, v, v .. BREVE)
end
return vowel
end
local function shift(vowel)
for _, v in ipairs({"á", "ea", "ie", "oa", "uo"}) do
vowel = gsub(vowel, v, v .. "ˈ")
end
return vowel
end
local function lengthen(vowel)
vowel = gsub(vowel, "^[eiou]$", {["e"] = "ē", ["i"] = "ī", ["o"] = "ō", ["u"] = "ū"})
vowel = gsub(vowel, BREVE, "")
return vowel
end
-- Determine whether long vowels should be shortened before certain consonants
local function should_shorten(syll, nextsyll)
if nextsyll.cons[1] then
if find(nextsyll.cons[1], "^h([ptcčk])%1$") then
-- Long preaspirate
return true
elseif find(nextsyll.cons[1], "^([đflmnŋrsšv])ˈ%1$") then
-- Overlong vowel
return true
elseif (syll.vowel == "ie" or syll.vowel == "uo") and find(nextsyll.vowel, "^[áīū]$") then
if find(nextsyll.cons[1], "^([bcčdgkptzž])%1$") then
-- Geminate stop
return true
elseif nextsyll.cons[1] == "pm" or nextsyll.cons[1] == "tn" or nextsyll.cons[1] == "tnj" or nextsyll.cons[1] == "kŋ" then
-- Glottalised nasal
return true
elseif nextsyll.cons[2] and not find(nextsyll.cons[2], "^h[ptcčk]$") then
-- Clusters, except when the second element is a strong-grade preaspirate
return true
end
elseif (syll.vowel == "ie" or syll.vowel == "uo") and nextsyll.vowel == "a" then
if find(nextsyll.cons[1], "^([bdgzž])%1$") then
-- Geminate voiced stop
return true
elseif nextsyll.cons[2] and not find(nextsyll.cons[2], "(.)%1$") and not find(nextsyll.cons[2], "^h[ptcčk]$") and not (nextsyll.cons[2] == "pm" or nextsyll.cons[2] == "tn" or nextsyll.cons[2] == "tnj" or nextsyll.cons[2] == "kŋ") then
-- Clusters, except when the second element is long, or a preaspirate, or a preglottalised nasal
return true
end
end
end
return false
end
local function convert_spelling(syllables)
local foot = 0
for i, syll in ipairs(syllables) do
if syll.vowel == "" then
if syll.cons[#syll.cons] == "t" then
syll.cons[#syll.cons] = "ht"
elseif syll.cons[#syll.cons] == "d" then
syll.cons[#syll.cons] = "t"
end
break
end
local nextsyll = syllables[i + 1] or {cons = {}, vowel = ""}
foot = foot + 1
if foot == 3 and i ~= syllables.count then
foot = 1
end
-- Make i and u long in even syllables
if foot == 2 and (syll.vowel == "i" or syll.vowel == "u") and nextsyll.cons[1] ~= "j" then
syll.vowel = lengthen(syll.vowel)
end
if #syll.cons == 1 then
if foot == 1 then
-- Postaspiration
syll.cons[1] = gsub(syll.cons[1], "^([kpt])$", "%1h")
elseif foot == 3 then
-- d is đ between two unstressed vowels
syll.cons[1] = gsub(syll.cons[1], "d", "đ")
end
elseif #syll.cons > 1 then
if syll.cons[#syll.cons] == syll.cons[#syll.cons - 1] and syll.cons[#syll.cons - 2] and find(syll.cons[#syll.cons - 2], "[cčkpsšt]$") then
-- Ungeminate last consonant after voiceless
syll.cons[#syll.cons] = nil
elseif find(syll.cons[#syll.cons], "[cčkpsšt]$") then
-- Ungeminate last consonant after voiceless
syll.cons[#syll.cons] = gsub(syll.cons[#syll.cons], "(.)%1$", "%1")
else
-- Preaspirate final voiceless consonant after voiced
syll.cons[#syll.cons] = gsub(syll.cons[#syll.cons], "^([cčkpt])$", "h%1")
syll.cons[#syll.cons] = gsub(syll.cons[#syll.cons], "^([cčkpt])%1$", "h%1%1")
end
-- Devoice final geminates
if syll.cons[#syll.cons] == "bb" then
syll.cons[#syll.cons] = "pp"
elseif syll.cons[#syll.cons] == "dd" then
syll.cons[#syll.cons] = "tt"
elseif syll.cons[#syll.cons] == "gg" then
syll.cons[#syll.cons] = "kk"
elseif syll.cons[#syll.cons] == "zz" then
syll.cons[#syll.cons] = "cc"
elseif syll.cons[#syll.cons] == "žž" then
syll.cons[#syll.cons] = "čč"
end
end
-- Devoice remaining single voiced consonants
for j, cons in ipairs(syll.cons) do
if cons == "b" and syll.cons[j - 1] ~= "b" and (j ~= 1 or syll.cons[2] ~= "b" and syll.cons[2] ~= "m") then
syll.cons[j] = "p"
elseif cons == "d" and syll.cons[j - 1] ~= "d" and (j ~= 1 or syll.cons[2] ~= "d" and syll.cons[2] ~= "n" and syll.cons[2] ~= "nj") then
syll.cons[j] = "t"
elseif cons == "g" and syll.cons[j - 1] ~= "g" and (j ~= 1 or syll.cons[2] ~= "g" and syll.cons[2] ~= "ŋ") then
syll.cons[j] = "k"
elseif cons == "z" and syll.cons[j - 1] ~= "z" and (j ~= 1 or syll.cons[2] ~= "z") then
syll.cons[j] = "c"
elseif cons == "ž" and syll.cons[j - 1] ~= "ž" and (j ~= 1 or syll.cons[2] ~= "ž") then
syll.cons[j] = "č"
end
end
-- Regularise divergent spellings in clusters
--if #syll.cons > 2 then
-- error("Clusters with more than 2 consonants are not yet supported.")
--end
if foot == 2 and syll.cons.quantity == 3 then
-- Lengthen initial sonorant in quantity 3
table.insert(syll.cons, 2, "ˈ")
end
-- Secondary stress
if foot == 1 and i > 1 then
if #syll.cons == 1 then
table.insert(syll.cons, 1, "ˌ")
elseif #syll.cons == 2 then
table.insert(syll.cons, 2, "ˌ")
end
end
end
-- This needs to be a separate pass because otherwise unstressed ī and ū won't have been lengthened yet
for i, syll in ipairs(syllables) do
local nextsyll = syllables[i + 1] or {cons = {}, vowel = ""}
-- if should_shorten(syll, nextsyll) then
-- syll.vowel = shorten(syll.vowel)
if find(nextsyll.vowel, "^[ạẹọ]$") then
syll.vowel = shift(syll.vowel)
end
end
end
-- Dialect-specific conversions
local function dialect(syllables)
for i, syll in ipairs(syllables) do
-- Western Finnmark dialect
if syll.cons[1] then
if syll.cons[#syll.cons] == "ŋ" then
syll.cons[#syll.cons] = "nj"
if syll.cons[#syll.cons - 1] == "ˈ" then
if syll.cons[#syll.cons - 2] then
syll.cons[#syll.cons - 2] = gsub(syll.cons[#syll.cons - 2], "^[gkŋ]$", {["g"] = "d", ["k"] = "t", ["ŋ"] = "nj"})
end
else
if syll.cons[#syll.cons - 1] then
syll.cons[#syll.cons - 1] = gsub(syll.cons[#syll.cons - 1], "^[gk]$", {["g"] = "d", ["k"] = "t", ["ŋ"] = "nj"})
end
end
end
end
end
end
-- Convert word to IPA
local function to_IPA(syllables)
for i, syll in ipairs(syllables) do
for j, cons in ipairs(syll.cons) do
if syll.vowel == "" and cons == "ht" then
syll.cons[j] = "h(t)"
elseif letters_phonemes[cons] then
-- Drop the final part after the tie bar
if string.find(letters_phonemes[cons], "͡", nil, true) and syll.cons[j] == syll.cons[j + (syll.cons[j + 1] == "ˈ" and 2 or 1)] then
syll.cons[j] = gsub(letters_phonemes[cons], "͡.*$", "")
else
syll.cons[j] = letters_phonemes[cons]
end
end
end
syll.vowel = letters_phonemes[syll.vowel] or syll.vowel
syllables[i] = table.concat(syll.cons) .. syll.vowel
end
return "ˈ" .. table.concat(syllables)
end
function export.IPA(frame)
local params = {
[1] = {default = mw.title.getCurrentTitle().text},
}
local args = require("Module:parameters").process(frame:getParent().args, params)
local syllables = split_syllables(args[1])
convert_spelling(syllables)
dialect(syllables)
return
require("Module:accent qualifier").format_qualifiers(lang, {"凱-{于}-圖凱努"}) .. " " ..
require("Module:IPA").format_IPA_full { lang = lang, items = {{pron = "/" .. to_IPA(syllables) .. "/"}} } ..
require("Module:utilities").format_categories({lang:getCanonicalName() .. tostring(syllables.count) .. "音節詞"}, lang)
end
return export