local export = {}

local lang = require("Module:languages").getByCode("pl")

local function collect_numbered(args, prefix, target)
	local was_empty = false
	if args[prefix] then
		table.insert(target, args[prefix])
		local i = 2
		while args[prefix .. i] do
			table.insert(target, args[prefix .. i])
			i = i + 1
		end
	end
	
	return target
end

local function add_if_nonempty(infls, item)
	if #item > 0 then
		table.insert(infls, item)	
	end
end

-- verbs

-- Aspect normalization table.
-- Note that determinate, indeterminate and frequentative aspects mainly make
-- sense for verbs of motion. Most verbs will have either perfective or
-- imperfective aspect.
local norm_aspect = {
	-- perfective aspect: zrobić, dojechać, pójść
	["p"]            = "pf",
	["pf"]           = "pf",
	["perf"]         = "pf",
	["perfective"]   = "pf",
	-- imperfective aspect: robić, karać
	["i"]            = "impf",
	["impf"]         = "impf",
	["imperf"]       = "impf",
	["imperfective"] = "impf",
	-- determinate aspect: jechać, iść
	-- implies imperfective
	["impf-det"]     = "impf-det",
	["det"]          = "impf-det",
	-- indeterminate aspect: jeździć, chodzić
	-- implies imperfective
	["impf-indet"]   = "impf-indet",
	["indet"]        = "impf-indet",
	-- frequentative aspect: chadzać
	-- implies imperfective and indeterminate
	["freq"]         = "impf-freq",
	["if"]           = "impf-freq",
	["impf-freq"]    = "impf-freq",
}

function export.show_verb(frame)
	local args = frame:getParent().args
	
	local data = {lang = lang, pos_category = "動詞", categories = {}, sort_key = args.sort, heads = {args.head}, genders = {}, inflections = {}}
	local tracking_categories = {}
	local aspect = norm_aspect[args.aspect or args.asp or args.a or false] or "?"

	if aspect == "pf" then
		table.insert(data.genders, "pf")
		table.insert(data.categories, "波蘭語完整體動詞")
	elseif (aspect == "impf") or (aspect == "impf-det") or (aspect == "impf-indet") or (aspect == "impf-freq") then
		table.insert(data.genders, "impf")
		table.insert(data.categories, "波蘭語非完整體動詞")
	else
		table.insert(data.genders, "?")
		table.insert(data.categories, "Polish verbs missing aspect")
	end

	if aspect == "impf-det" then
		table.insert(data.inflections, {label = "determinate"})
		table.insert(data.categories, "Polish determinate verbs")
	elseif aspect == "impf-indet" then
		table.insert(data.inflections, {label = "indeterminate"})
		table.insert(data.categories, "Polish indeterminate verbs")
	elseif aspect == "impf-freq" then
		table.insert(data.inflections, {label = "indeterminate"})
		table.insert(data.inflections, {label = "frequentative"})
		table.insert(data.categories, "Polish indeterminate verbs")
		table.insert(data.categories, "Polish frequentative verbs")
	end

	local perf = { label = "完整體" }
	local imperf = { label = "未完整體" }
	local det = { label = "imperfective determinate" }
	local indet = { label = "indeterminate" }
	local freq = { label = "frequentative" }
	
	-- legacy parameter
	if args[1] then
		table.insert(tracking_categories, "pl-verb with positional parameters")
		if (aspect == "impf") then
			table.insert(perf, args[1])
		elseif (aspect == "pf") then
			table.insert(imperf, args[1])
		end
	end

	collect_numbered(args, "perf", perf)
	collect_numbered(args, "pf", perf)
	collect_numbered(args, "imperf", imperf)
	collect_numbered(args, "impf", imperf)
	collect_numbered(args, "det", det)
	collect_numbered(args, "indet", indet)
	collect_numbered(args, "freq", freq)
	
	-- "pref" would never come before "det" (pf; det-inedt, impf; det-indet, indet; det-pf, etc.)
	add_if_nonempty(data.inflections, det)
	add_if_nonempty(data.inflections, perf)
	add_if_nonempty(data.inflections, imperf)
	add_if_nonempty(data.inflections, indet)
	add_if_nonempty(data.inflections, freq)

	return require("Module:headword").full_headword(data) ..
		require("Module:utilities").format_categories(tracking_categories, lang, args.sort)
end

function export.show_participle(frame)
	local args = frame:getParent().args
	
	local data = {lang = lang, pos_category = "分詞", categories = {}, sort_key = args.sort, heads = {args.head}, genders = {}, inflections = {}}
	local tracking_categories = {}
	local aspect = norm_aspect[args.aspect or args.asp or args.a or false] or "?"
	
	-- type of participle
	local ptype = args[1]
	if not ptype then
		local PAGENAME = mw.title.getCurrentTitle().fullText
		
		if PAGENAME:match("ąc[yae]$") then -- biegnący, 
			ptype = "aadj"			
		elseif PAGENAME:match("[nt][yae]$") then -- otwarty, uwielbiany
			ptype = "padj"
		elseif PAGENAME:match("ąc$") then
			ptype = "cadv"
		elseif PAGENAME:match("szy$") then
			ptype = "aadv"
		end
	end

	if ptype then
		if ptype == "pasv-adj" or ptype == "padj" then
			table.insert(data.categories, "波蘭語被動形容詞性分詞")
		elseif ptype == "actv-adj" or ptype == "aadj" then
			table.insert(data.categories, "波蘭語主動形容詞性分詞")
		elseif ptype == "antr-adv" or ptype == "aadv" then
			table.insert(data.categories, "Polish anterior adverbial participles")
		elseif ptype == "cont-adv" or ptype == "cadv" then
			table.insert(data.categories, "Polish contemporary adverbial participles")
		else
			table.insert(tracking_categories, "pl-participle with unrecognized type")
		end
	else
		table.insert(tracking_categories, "pl-participle without type")
	end

	if (aspect == "pf") or (aspect == "pf-it") or (aspect == "pf-sem") then
		table.insert(data.genders, "pf")
	elseif (aspect == "impf") or (aspect == "impf-it") or (aspect == "impf-dur") then
		table.insert(data.genders, "impf")
	end

	if aspect == "impf-dur" then
		table.insert(data.inflections, {label = "durative"})
	elseif aspect == "impf-it" then
		table.insert(data.inflections, {label = "iterative"})
	elseif aspect == "pf-sem" then
		table.insert(data.inflections, {label = "semelfactive"})
	elseif aspect == "pf-it" then
		table.insert(data.inflections, {label = "iterative"})
	end

	return require("Module:headword").full_headword(data) ..
		require("Module:utilities").format_categories(tracking_categories, lang, args.sort)
end

-- nouns and proper nouns

local noun_gender_cat = {
	["m"     ] = "波蘭語陽性名詞",
	["m-an"  ] = "波蘭語陽性名詞",
	["m-in"  ] = "波蘭語陽性名詞",
	["m-pr"  ] = "波蘭語陽性名詞",
	["f"     ] = "波蘭語陰性名詞",
	["n"     ] = "波蘭語中性名詞",
	["vr"	 ] = "波蘭語男性名詞",
	["nv"	 ] = "波蘭語非男性名詞",
	["m-p"   ] = false,
	["m-pr-p"] = false,
	["m-an-p"] = false,
	["m-in-p"] = false,
	["f-p"   ] = false,
	["n-p"   ] = false,
	["p"     ] = false,
}

function export.show_noun(frame)
	local args = frame:getParent().args
	
	local data = {lang = lang, pos_category = frame.args.proper and "專有名詞" or "名詞", categories = {}, sort_key = args.sort, heads = {args.head}, genders = {}, inflections = {}}
	local tracking_categories = {}
	
	if args.indecl and args.indecl ~= '' then
		table.insert(data.inflections, { label = "無變格" })
		table.insert(data.categories, "波蘭語無變格名詞")
	end

	add_if_nonempty(data.inflections, collect_numbered(args, "abbr", { label = "縮寫" }))
	add_if_nonempty(data.inflections, collect_numbered(args, "dim", { label = "指小詞" }))
	add_if_nonempty(data.inflections, collect_numbered(args, "aug", { label = "指大詞" }))
	add_if_nonempty(data.inflections, collect_numbered(args, "f", { label = "陰性" }))
	add_if_nonempty(data.inflections, collect_numbered(args, "m", { label = "陽性" }))

	local g, genders = args.g or args[1]
	if g then
		if noun_gender_cat[g] ~= nil then
			data.genders = { g }
			table.insert(data.categories, noun_gender_cat[g] or nil)
			if g == "m" then
				table.insert(tracking_categories, "Polish masculine nouns without animacy")
			end
		else
			data.genders = { "?" }
		end
	else
		data.genders = { "?" }
	end

	return require("Module:headword").full_headword(data) ..
		require("Module:utilities").format_categories(tracking_categories, lang, args.sort)
end

-- adjectives and adverbs

function export.show_adj_adv(frame)
	local args = frame:getParent().args
	
	local data = {lang = lang, categories = {}, sort_key = args.sort, heads = {args.head}, inflections = {}}
	local tracking_categories = {}
	
	local PAGENAME = mw.title.getCurrentTitle().text
	
	if frame.args.adverb then
		data.pos_category = "副詞"
	else
		data.pos_category = "形容詞"
	end
	
	local comparative, superlative = {}, {}
	
	local i = 1
	
	if args[1] == "-" then
		if frame.args.adverb then
			table.insert(data.categories, "波蘭語無比較級副詞")
		else
			table.insert(data.categories, "波蘭語無比較級形容詞")
		end

		if not args[2] then
			table.insert(data.inflections, { label = "無比較級" })
		else
			table.insert(data.inflections, { label = "無通用比較級" })
			i = i + 1
			while args[i] do
				local comp, super
				if frame.args.adverb and ((args[i] == "j") or (args[i] == "regular")) then
					table.insert(comparative, PAGENAME .. "j")
					table.insert(superlative, "naj" .. PAGENAME .. "j")
				elseif args[i] == "bardziej" then
					table.insert(comparative, "[[bardziej]] " .. PAGENAME)
					table.insert(superlative, "[[najbardziej]] " .. PAGENAME)
				elseif args[i] then
					table.insert(comparative, args[i])
					table.insert(superlative, "naj" .. args[i])
				end
				i = i + 1
			end

			comparative.label, comparative.accel = "比較級", {form = "比較級"}
			superlative.label, superlative.accel = "最高級", {form = "最高級"}

			table.insert(data.inflections, comparative)
			table.insert(data.inflections, superlative)
		end
	elseif args[1] then
		while args[i] do
			local comp, super
			if frame.args.adverb and ((args[i] == "j") or (args[i] == "regular")) then
				table.insert(comparative, PAGENAME .. "j")
				table.insert(superlative, "naj" .. PAGENAME .. "j")
			elseif args[i] == "bardziej" then
				table.insert(comparative, "[[bardziej]] " .. PAGENAME)
				table.insert(superlative, "[[najbardziej]] " .. PAGENAME)
			elseif args[i] then
				table.insert(comparative, args[i])
				table.insert(superlative, "naj" .. args[i])
			end
			i = i + 1
		end

		comparative.label, comparative.accel = "比較級", {form = "比較級"}
		superlative.label, superlative.accel = "最高級", {form = "最高級"}
		
		table.insert(data.inflections, comparative)
		table.insert(data.inflections, superlative)
	else
		table.insert(tracking_categories, "缺少比較級形式的波蘭語形容詞或副詞")
	end

	if not frame.args.adverb then
		if args.adv then
			if args.adv ~= '-' then
				add_if_nonempty(data.inflections, collect_numbered(args, "adv", { label = "副詞" }))
			end
		else
			table.insert(tracking_categories, "缺少對應副詞形的波蘭語形容詞")
		end
	end

	add_if_nonempty(data.inflections, collect_numbered(args, "abbr", { label = "縮寫" }))

	return require("Module:headword").full_headword(data) ..
		require("Module:utilities").format_categories(tracking_categories, lang, args.sort)
end

return export