local m_utilities = require("Module:utilities")
local m_links = require("Module:links")
local m_common = require("Module:ine-common")
local export = {}
local lang = require("Module:languages").getByCode("ine-pro")
local endings = {
["anim"] = {
["athem"] = {
["nom_sg"] = {"s"},
["voc_sg"] = {""},
["acc_sg"] = {"m̥"},
["gen_sg"] = {{stressed = "és", unstressed = "s"}},
["abl_sg"] = {{stressed = "és", unstressed = "s"}},
["dat_sg"] = {"éy"},
["loc_sg"] = {"", "i"},
["ins_sg"] = {{stressed = "éh₁", unstressed = "h₁"}},
["nom_du"] = {"h₁(e)"},
["voc_du"] = {"h₁(e)"},
["acc_du"] = {"h₁(e)"},
["gen_du"] = "?",
["abl_du"] = "?",
["dat_du"] = "?",
["loc_du"] = "?",
["ins_du"] = "?",
["nom_pl"] = {"es"},
["voc_pl"] = {"es"},
["acc_pl"] = {"m̥s"},
["gen_pl"] = {"óHom"},
["abl_pl"] = {"mós"},
["dat_pl"] = {"mós"},
["loc_pl"] = {"sú"},
["ins_pl"] = {"bʰí"},
},
["them"] = {
["nom_sg"] = {"ós"},
["voc_sg"] = {"é"},
["acc_sg"] = {"óm"},
["gen_sg"] = {"ósyo"},
["abl_sg"] = {"éad"},
["dat_sg"] = {"óey"},
["loc_sg"] = {"éy", "óy"},
["ins_sg"] = {"óh₁"},
["nom_du"] = {"óh₁"},
["voc_du"] = {"óh₁"},
["acc_du"] = {"óh₁"},
["gen_du"] = "?",
["abl_du"] = "?",
["dat_du"] = "?",
["loc_du"] = "?",
["ins_du"] = "?",
["nom_pl"] = {"óes"},
["voc_pl"] = {"óes"},
["acc_pl"] = {"óms"},
["gen_pl"] = {"óHom"},
["abl_pl"] = {"ómos"},
["dat_pl"] = {"ómos"},
["loc_pl"] = {"óysu"},
["ins_pl"] = {"ṓys"},
},
},
["inan"] = {
["athem"] = {
["nom_sg"] = {""},
["voc_sg"] = {""},
["acc_sg"] = {""},
["nom_du"] = {"ih₁"},
["voc_du"] = {"ih₁"},
["acc_du"] = {"ih₁"},
["nom_pl"] = {"h₂"},
["voc_pl"] = {"h₂"},
["acc_pl"] = {"h₂"},
},
["them"] = {
["nom_sg"] = {"óm"},
["voc_sg"] = {"óm"},
["acc_sg"] = {"óm"},
["nom_du"] = {"óy(h₁)"},
["voc_du"] = {"óy(h₁)"},
["acc_du"] = {"óy(h₁)"},
["nom_pl"] = {"éh₂"},
["voc_pl"] = {"éh₂"},
["acc_pl"] = {"éh₂"},
},
},
["coll"] = {
["athem"] = {
["nom_sg"] = {"h₂"},
["voc_sg"] = {"h₂"},
["acc_sg"] = {"h₂"},
},
["them"] = {
["nom_sg"] = {"éh₂"},
["voc_sg"] = {"éh₂"},
["acc_sg"] = {"éh₂"},
},
},
}
-- Copy over endings which are the same for animate and inanimate.
for _, t in ipairs({"athem", "them"}) do
for cn, _ in pairs(endings["anim"][t]) do
if not endings["inan"][t][cn] then
endings["inan"][t][cn] = endings["anim"][t][cn]
end
if cn:find("_sg$") and not endings["coll"][t][cn] then
endings["coll"][t][cn] = endings["anim"][t][cn]
end
end
end
local endings_pron = {
["m"] = {
["nom_sg"] = {"ós"},
["acc_sg"] = {"óm"},
["gen_sg"] = {"ósyo"},
["abl_sg"] = {"ósmead"},
["dat_sg"] = {"ósmey"},
["loc_sg"] = {"ósmi"},
["ins_sg"] = {"ónoh₁"},
["nom_du"] = {"óh₁"},
["acc_du"] = {"óh₁"},
["gen_du"] = "?",
["abl_du"] = "?",
["dat_du"] = "?",
["loc_du"] = "?",
["ins_du"] = "?",
["nom_pl"] = {"óy"},
["acc_pl"] = {"óms"},
["gen_pl"] = {"óysoHom"},
["abl_pl"] = {"óymos"},
["dat_pl"] = {"óymos"},
["loc_pl"] = {"óysu"},
["ins_pl"] = {"ṓys"},
},
["f"] = {
["nom_sg"] = {"éh₂"},
["acc_sg"] = {"éh₂m̥"},
["gen_sg"] = {"ósyeh₂s"},
["abl_sg"] = {"ósyeh₂s"},
["dat_sg"] = {"ósyeh₂ey"},
["loc_sg"] = {"ósyeh₂"},
["ins_sg"] = {"éh₂(e)h₁"},
["nom_du"] = "?",
["acc_du"] = "?",
["gen_du"] = "?",
["abl_du"] = "?",
["dat_du"] = "?",
["loc_du"] = "?",
["ins_du"] = "?",
["nom_pl"] = {"éh₂es"},
["acc_pl"] = {"éh₂m̥s"},
["gen_pl"] = {"éh₂soHom"},
["abl_pl"] = {"éh₂mos"},
["dat_pl"] = {"éh₂mos"},
["loc_pl"] = {"éh₂su"},
["ins_pl"] = {"éh₂bʰi"},
},
["n"] = {
["nom_sg"] = {"ód"},
["acc_sg"] = {"ód"},
["nom_du"] = {"óy"},
["acc_du"] = {"óy"},
["nom_pl"] = {"éh₂"},
["acc_pl"] = {"éh₂"},
},
}
-- Copy over endings which are the same for animate and inanimate.
for cn, _ in pairs(endings_pron["m"]) do
if not endings_pron["n"][cn] then
endings_pron["n"][cn] = endings_pron["m"][cn]
end
end
local function inflect(data, prefix, endings, stem1, stem2, loc_sg_stem)
stem2 = stem2 or stem1
-- Is the stem thematic?
local thematic_unstressed = false
if mw.ustring.find(stem1, "[eoéó]$") then
endings = mw.clone(endings.them or endings)
if mw.ustring.find(stem1, "[eo]$") then
thematic_unstressed = true
end
stem1 = mw.ustring.gsub(stem1, "[eoéó]$", "")
stem2 = stem1
else
endings = endings.athem or endings
if mw.ustring.find(stem1, "[eoéó]h₂$") and stem1 == stem2 then
thematic_unstressed = true
end
end
if not loc_sg_stem then
if mw.ustring.find(stem2, "[áéíĺḿńóŕúḗṓ́]") then
loc_sg_stem = stem2
else
loc_sg_stem = stem1
end
end
local stem2_zero = stem2
if mw.ustring.find(stem1, "[iu]$") then
stem2_zero = mw.ustring.gsub(stem2, "é[wy]$", {["éw"] = "ú", ["éy"] = "í"})
end
local stem1_full = stem1
if stem1 ~= stem2 and mw.ustring.gsub(stem1, "os$", "es") == stem2 then
stem1_full = stem2
end
local nom_pl_stem = mw.ustring.gsub(stem1, "[íiu]$", {["í"] = "éy", ["i"] = "ey", ["u"] = "ew"})
-- Go over each case-number combination
for cn, cnendings in pairs(endings) do
if cnendings == "?" then
data.forms[(prefix and prefix .. "_" or "") .. cn] = {"?"}
elseif cnendings then
data.forms[(prefix and prefix .. "_" or "") .. cn] = {}
for _, cnending in ipairs(cnendings) do
local stem = stem1
-- Use stem2 if the ending can be stressed, stem1 otherwise
if cn == "loc_sg" then
stem = loc_sg_stem
elseif cn == "ins_sg" or cn == "abl_pl" or cn == "dat_pl" or cn == "loc_pl" or cn == "ins_pl" then
stem = stem2_zero
elseif (cn == "nom_du" or cn == "voc_du" or cn == "acc_du") then
stem = stem1_full
elseif (cn == "nom_pl" or cn == "voc_pl") and cnending == "es" then
stem = nom_pl_stem
elseif cnending.stressed or mw.ustring.find(cnending, "[áéíĺḿńóŕúḗṓ́]") then
stem = stem2
end
local ending = (thematic_unstressed and (cnending.unstressed or m_common.destress(cnending))) or cnending.stressed or cnending
local ending_unstr = cnending.unstressed
if (cn == "gen_sg" or cn == "abl_sg") and ending_unstr == "s" and mw.ustring.find(stem, "s$") then
ending_unstr = "os"
end
if mw.ustring.find(stem, "[iu]$") then
if cn == "acc_sg" and ending == "m̥" then
ending = "m"
elseif cn == "acc_pl" and ending == "m̥s" then
ending = "ms"
end
end
if mw.ustring.find(stem, "[eéoóaáiíuú]h₂$") and cn == "nom_sg" and ending == "s" then
ending = ""
end
if not (cnending == "i" and mw.ustring.find(stem, "i$")) then
table.insert(data.forms[(prefix and prefix .. "_" or "") .. cn], m_common.add_ending(stem, ending, ending_unstr, cn == "nom_sg" and ending == "s"))
end
end
end
end
end
local function postprocess(data)
for key, form in pairs(data.forms) do
-- Do not show singular, dual or plural forms for nouns that don't have them
if (not data.sg and key:find("_sg$")) or (not data.du and key:find("_du$")) or (not data.pl and key:find("_pl$")) or (not data.coll and key:find("_coll$")) then
form = nil
end
data.forms[key] = form
end
end
function export.masc(frame)
local params = {
[1] = {required = true},
[2] = {},
["nom_sg"] = {},
["loc_sg"] = {},
["coll"] = {},
["coll2"] = {},
["coll_loc"] = {},
["ac"] = {},
["n"] = {},
}
local args = require("Module:parameters").process(frame:getParent().args, params)
if mw.title.getCurrentTitle().nsText == "Template" then
args[1] = args[1] or "h₂éǵro"
end
args[2] = args[2] or args[1]
local data = {forms = {}, title = nil, categories = {}}
data.sg = true
data.du = true
data.pl = true
data.coll = false
data.voc = true
if args["n"] then
if not mw.ustring.find(args["n"], "s") then
data.sg = false
end
if not mw.ustring.find(args["n"], "d") then
data.du = false
end
if not mw.ustring.find(args["n"], "p") then
data.pl = false
end
end
local stem_type = {}
if mw.ustring.find(args[1], "[eoéó]$") then
data.info = "有构幹元音"
table.insert(data.categories, lang:getCanonicalName() .. "有构幹元音o-詞幹名詞")
elseif mw.ustring.find(args[1], "[eoéó]h₂$") and args[1] == args[2] then
data.info = "Thematic in " .. m_links.full_link({lang = lang, alt = "*-eh₂"}, "term")
table.insert(data.categories, lang:getCanonicalName() .. "有构幹元音eh₂-詞幹名詞")
else
data.info = "無构幹元音"
if mw.ustring.find(args[1], "ti$") then
stem_type = " ti-詞幹"
elseif mw.ustring.find(args[1], "tu$") then
stem_type = " tu-詞幹"
elseif mw.ustring.find(args[1], "i$") then
stem_type = " i-詞幹"
elseif mw.ustring.find(args[1], "ih₂$") then
stem_type = " ih₂-詞幹"
elseif mw.ustring.find(args[1], "u$") then
stem_type = " u-詞幹"
elseif mw.ustring.find(args[1], "uh₂$") then
stem_type = " uh₂-詞幹"
elseif mw.ustring.find(args[1], "[eéoóaáiíuú][%a₁₂₃]+[eéoó]m$") then
stem_type = " m-詞幹"
elseif mw.ustring.find(args[1], "[eéoóaáiíuú][%a₁₂₃]+m[eéoó]n$") then
stem_type = " men-詞幹"
elseif mw.ustring.find(args[1], "[eéoóaáiíuú][%a₁₂₃]+[eéoó]n$") then
stem_type = " n-詞幹"
elseif mw.ustring.find(args[1], "[eéoóaáiíuú][%a₁₂₃]+[eéoó][lr]$") then
stem_type = " r-詞幹"
else
stem_type = "詞根"
end
if args["ac"] then
data.info = data.info .. ", " .. args["ac"]
table.insert(data.categories, lang:getCanonicalName() .. args["ac"] .. stem_type .."名詞")
else
table.insert(data.categories, lang:getCanonicalName() .. "無构幹元音名詞")
end
end
-- Create the forms
inflect(data, nil, endings["anim"], args[1], args[2], args["loc_sg"])
if args["nom_sg"] then
data.forms["nom_sg"] = {args["nom_sg"]}
end
-- Collective
if args["coll"] then
args["coll2"] = args["coll2"] or args["coll"]
data.coll = true
local tempdata = {forms = {}}
inflect(tempdata, nil, endings["coll"], args["coll"], args["coll2"], args["coll_loc"])
for cn, forms in pairs(tempdata.forms) do
if cn:find("_sg$") then
local coll = cn:gsub("_sg$", "_coll")
data.forms[coll] = forms
end
end
end
postprocess(data)
return make_table(data)
end
function export.neut(frame)
local params = {
[1] = {required = true},
[2] = {},
["loc_sg"] = {},
["coll"] = {},
["coll2"] = {},
["coll_loc"] = {},
["ac"] = {},
["n"] = {},
}
local args = require("Module:parameters").process(frame:getParent().args, params)
if mw.title.getCurrentTitle().nsText == "Template" then
args[1] = args[1] or "wérǵo"
end
args[2] = args[2] or args[1]
local data = {forms = {}, title = nil, categories = {}}
data.sg = true
data.du = true
data.pl = true
data.coll = false
data.voc = true
if args["n"] then
if not mw.ustring.find(args["n"], "s") then
data.sg = false
end
if not mw.ustring.find(args["n"], "d") then
data.du = false
end
if not mw.ustring.find(args["n"], "p") then
data.pl = false
end
end
local stem_type = {}
if mw.ustring.find(args[1], "[eoéó]$") then
data.info = "有构幹元音"
table.insert(data.categories, lang:getCanonicalName() .. "有构幹元音中性o-詞幹名詞")
else
data.info = "無构幹元音"
if mw.ustring.find(args[1], "i$") then
stem_type = " i-詞幹"
elseif mw.ustring.find(args[1], "u$") then
stem_type = " u-詞幹"
elseif mw.ustring.find(args[1], "mn̥$") then
stem_type = " men-詞幹"
elseif mw.ustring.find(args[1], "[lr]̥$") then
stem_type = " r/n-詞幹"
elseif mw.ustring.find(args[1], "os$") then
stem_type = " s-詞幹"
else
stem_type = "詞根"
end
if args["ac"] then
data.info = data.info .. ", " .. args["ac"]
table.insert(data.categories, lang:getCanonicalName() .. args["ac"] .. " neuter".. stem_type .. "名詞")
else
table.insert(data.categories, lang:getCanonicalName() .. "無构幹元音名詞")
end
end
-- Create the forms
inflect(data, nil, endings["inan"], args[1], args[2], args["loc_sg"])
-- Collective
if args["coll"] then
args["coll2"] = args["coll2"] or args["coll"]
data.coll = true
local tempdata = {forms = {}}
inflect(tempdata, nil, endings["coll"], args["coll"], args["coll2"], args["coll_loc"])
for cn, forms in pairs(tempdata.forms) do
if cn:find("_sg$") then
local coll = cn:gsub("_sg$", "_coll")
data.forms[coll] = forms
end
end
end
postprocess(data)
return make_table(data)
end
function export.adj(frame)
local params = {
[1] = {required = true},
[2] = {},
["loc_sg"] = {},
[3] = {},
[4] = {},
["ac"] = {},
["n"] = {},
}
local args = require("Module:parameters").process(frame:getParent().args, params)
if mw.title.getCurrentTitle().nsText == "Template" then
args[1] = args[1] or "néwo"
end
args[2] = args[2] or args[1]
local data = {forms = {}, title = nil, categories = {}}
data.genders = true
data.sg = true
data.du = true
data.pl = true
data.voc = true
if args["n"] then
if not mw.ustring.find(args["n"], "s") then
data.sg = false
end
if not mw.ustring.find(args["n"], "d") then
data.du = false
end
if not mw.ustring.find(args["n"], "p") then
data.pl = false
end
end
if mw.ustring.find(args[1], "[eoéó]$") then
data.info = "有构幹元音"
table.insert(data.categories, lang:getCanonicalName() .. "有构幹元音形容詞")
if not args[3] then
args[3] = mw.ustring.gsub(args[1], "[eoéó]$", {["e"] = "eh₂", ["o"] = "eh₂", ["é"] = "éh₂", ["ó"] = "éh₂"})
args[4] = args[3]
end
else
data.info = "無构幹元音"
if args["ac"] then
data.info = data.info .. ", " .. args["ac"]
table.insert(data.categories, lang:getCanonicalName() .. args["ac"] .. "形容詞")
else
table.insert(data.categories, lang:getCanonicalName() .. "無构幹元音形容詞")
end
end
-- Create the forms
inflect(data, "m", endings["anim"], args[1], args[2], args["loc_sg"])
inflect(data, "f", endings["anim"], args[3], args[4])
inflect(data, "n", endings["inan"], args[1], args[2], args["loc_sg"])
postprocess(data)
return make_table(data)
end
function export.pron_adj(frame)
local params = {
[1] = {required = true},
["m_nom_sg"] = {},
["f_nom_sg"] = {},
}
local args = require("Module:parameters").process(frame:getParent().args, params)
if mw.title.getCurrentTitle().nsText == "Template" then
args[1] = args[1] or "h₂élyo"
end
local data = {forms = {}, title = nil, categories = {}}
data.genders = true
data.sg = true
data.du = true
data.pl = true
data.voc = false
if mw.ustring.find(args[1], "[eoéó]$") then
data.info = "Thematic pronominal"
else
error("Thematic only for now.")
end
-- Create the forms
inflect(data, "m", endings_pron["m"], args[1])
inflect(data, "f", endings_pron["f"], args[1])
inflect(data, "n", endings_pron["n"], args[1])
data.forms["m_nom_sg"] = args["m_nom_sg"] and {args["m_nom_sg"]} or data.forms["m_nom_sg"]
data.forms["f_nom_sg"] = args["f_nom_sg"] and {args["f_nom_sg"]} or data.forms["f_nom_sg"]
postprocess(data)
return make_table(data)
end
local names = {
["nom"] = "主格",
["voc"] = "呼格",
["acc"] = "賓格",
["gen"] = "屬格",
["abl"] = "奪格",
["dat"] = "與格",
["loc"] = "方位格",
["ins"] = "工具格",
["sg"] = "單數",
["du"] = "雙數",
["pl"] = "複數",
["coll"] = "集合名詞",
["m"] = "陽性",
["f"] = "陰性",
["n"] = "中性",
}
-- Make the table
function make_table(data)
local function repl(param)
if param == "info" then
return mw.getContentLanguage():ucfirst(data.info or "")
end
local form = data.forms[param]
if not form or #form == 0 then
return "—"
end
local ret = {}
for key, subform in ipairs(form) do
table.insert(ret, m_links.full_link({lang = lang, alt = "*" .. subform}))
end
return table.concat(ret, ", ")
end
local numbers = {"sg", "du", "pl"}
local genders = {"x"}
local cases = {"nom", "acc", "gen", "abl", "dat", "loc", "ins"}
if data.voc then
table.insert(cases, 2, "voc")
end
if data.genders then
genders = {"m", "f", "n"}
elseif data.coll then
table.insert(numbers, "coll")
end
local first_number = "sg"
if not data.sg then
first_number = "du"
if not data.du then
first_number = "pl"
end
end
local wikicode = {}
table.insert(wikicode, "{| class=\"inflection-table vsSwitcher\" data-toggle-category=\"inflection\" style=\"background: #FAFAFA; border: 1px solid #d0d0d0; text-align: left;\" cellspacing=\"1\" cellpadding=\"2\"")
table.insert(wikicode, "|- style=\"background: #CCCCFF;\"\n! class=\"vsToggleElement\" colspan=\"" .. (#numbers + 1) .. "\" | {{{info}}}")
table.insert(wikicode, "|- class=\"vsShow\" style=\"background: #CCCCFF;\"")
table.insert(wikicode, "!")
if data.genders then
table.insert(wikicode, "! style=\"min-width: 11em; background: #CCCCFF;\" | " .. names["m"])
table.insert(wikicode, "! style=\"min-width: 11em; background: #CCCCFF;\" | " .. names["f"])
else
table.insert(wikicode, "! style=\"min-width: 11em; background: #CCCCFF;\" | " .. names[first_number])
if data.coll then
table.insert(wikicode, "! style=\"min-width: 11em; background: #CCCCFF;\" | " .. names["coll"])
end
end
table.insert(wikicode, "|- class=\"vsShow\" style=\"background: #F2F2FF;\"")
table.insert(wikicode, "! style=\"min-width: 8em; background: #E6E6FF;\" | " .. names["nom"])
if data.genders then
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{m_nom_" .. first_number .. "}}}")
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{f_nom_" .. first_number .. "}}}")
else
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{nom_" .. first_number .. "}}}")
if data.coll then
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{nom_coll}}}")
end
end
table.insert(wikicode, "|- class=\"vsShow\" style=\"background: #F2F2FF;\"")
table.insert(wikicode, "! style=\"min-width: 8em; background: #E6E6FF;\" | " .. names["gen"])
if data.genders then
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{m_gen_" .. first_number .. "}}}")
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{f_gen_" .. first_number .. "}}}")
else
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{gen_" .. first_number .. "}}}")
if data.coll then
table.insert(wikicode, "| style=\"min-width: 11em;\" | {{{gen_coll}}}")
end
end
for _, gender in ipairs(genders) do
table.insert(wikicode, "|- class=\"vsHide\" style=\"background: #CCCCFF;\"")
if data.genders then
table.insert(wikicode, "! " .. names[gender])
else
table.insert(wikicode, "!")
end
for _, number in ipairs(numbers) do
table.insert(wikicode, "! style=\"min-width: 11em; background: #CCCCFF;\" | " .. names[number])
end
for _, case in ipairs(cases) do
table.insert(wikicode, "|- class=\"vsHide\" style=\"background-color: #F2F2FF;\"\n! style=\"min-width: 8em; background-color: #E6E6FF;\" | " .. names[case])
for _, number in ipairs(numbers) do
if data.genders then
table.insert(wikicode, "| {{{" .. gender .. "_" .. case .. "_" .. number .. "}}}")
else
table.insert(wikicode, "| {{{" .. case .. "_" .. number .. "}}}")
end
end
end
end
table.insert(wikicode, [=[|}]=])
wikicode = table.concat(wikicode, "\n")
return (mw.ustring.gsub(wikicode, "{{{([a-z0-9_]+)}}}", repl)) .. m_utilities.format_categories(data.categories, lang)
end
return export