Documentation for this module may be created at မေႃႇၵျူး:pi-conj/verb/doc

local export = {}

local links = require("Module:links")
local lang = require("Module:languages").getByCode("pi")
local m_parameters = require("Module:parameters")
local m_noun =       require("Module:pi-decl/noun")
local m_scripts =    require("Module:scripts")
local to_script =    require("Module:pi-Latn-translit").tr
local debug_outer = ''
local gsub = mw.ustring.gsub
local match = mw.ustring.match
local sub = mw.ustring.sub
local asub = string.sub   -- quicker for ASCII, but works on bytes.
local u = mw.ustring.char -- uses this for readability.
local ti = table.insert
-- require "Module:log globals" -- Check for abuse of globals.  Only require for testing! 

local function dc(x) -- Use this to make marks legible.  The name 'dc' means 'drop carrier'.
	return gsub(x, "[𑀓कকකကกᨠកກ𑄇]", "") -- These are the letter ka in the 10 supported Indic scripts.
end

local endings = {
-- pa endings
	["ati"] = "ati",		["ติ"] = "ati",			["ति"] = "ati",			["তি"] = "ati",
	["တိ"] = "ati",									["ᨲᩥ"] = "ati",
	["តិ"] = "ati",			["ති"] = "ati",			["𑀢𑀺"] = "ati",  	    ["ຕິ"] = "ati",
	["𑄖𑄨"] = "ati",

	["ayati"] = "ayati",	["ยติ"] = "ayati",		["यति"] = "ayati",			["যতি"] = "ayati",
	["ယတိ"] = "ayati",		["ะยะติ"] = "ayati",		["ᨿᨲᩥ"] = "ayati",	["ຢຕິ"] = "ayati",
	["យតិ"] = "ayati",		["යති"] = "ayati",		["𑀬𑀢𑀺"] = "ayati",	["ະຢະຕິ"] = "ayati",
	["ຍຕິ"] = "ayati",		["ະຍະຕິ"] = "ayati",		["𑄠𑄖𑄨"] = "ayati",

	["āti"] = "āti",		["าติ"] = "āti",			[dc("काति")] = "āti",	[dc("কাতি")] = "āti",
	["ာတိ"] = "āti",		["ါတိ"] = "āti",		["ᩣᨲᩥ"] = "āti",		["ᩤᨲᩥ"] = "āti",
	[dc("កាតិ")] = "āti",	[dc("කාති")] = "āti",	[dc("𑀓𑀸𑀢𑀺")] = "āti",		["າຕິ"] = "āti",
	[dc("𑄇𑄂𑄖𑄨")] = "āti",

	["eti"] = "eti",								[dc("केति")] = "eti",	[dc("কেতি")] = "eti",
	["ေတိ"] = "eti",								[dc("ᨠᩮᨲᩥ")] = "eti",
	[dc("កេតិ")] = "eti",	[dc("කෙති")] = "eti",	[dc("𑀓𑁂𑀢𑀺")] = "eti",		[dc("𑄇𑄬𑄖𑄨")] = "eti",

	["oti"] = "oti",								[dc("कोति")] = "oti",	[dc("কোতি")] = "oti",
	["ောတိ"] = "oti",		["ေါတိ"] = "oti",		["ᩮᩣᨲᩥ"] = "oti",		["ᩮᩤᨲᩥ"] = "oti",
	[dc("កោតិ")] = "oti",	[dc("කොති")] = "oti",	[dc("𑀓𑁄𑀢𑀺")] = "oti",		[dc("𑄇𑄮𑄖𑄨")] = "oti",
-- pm endings
	["ate"] = "ate",		["เต"] = "ate",			["ते"] = "ate",			["তে"] = "ate",
	["တေ"] = "ate",									["ᨲᩮ"] = "ate",
	["តេ"] = "ate",			["තෙ"] = "ate",			["𑀢𑁂"] = "ate",   ["ເຕ"] = "ate",
	["𑄖𑄬"] = "ate",

	["ayate"] = "ayate",	["ยเต"] = "ayate",		["यते"] = "ayate",			["যতে"] = "ayate",
	["ယတေ"] = "ayate",		["ะยะเต"] = "ayate",		["ᨿᨲᩮ"] = "ayate",	["ຢເຕ"] = "ayate",
	["យតេ"] = "ayate",		["යතෙ"] = "ayate",		["𑀬𑀢𑁂"] = "ayate",	["ະຢະເຕ"] = "ayate",
	["ຍເຕ"] = "ayate",		["ະຍະເຕ"] = "ayate",	["𑄠𑄖𑄬"] = "ayate",

	["āte"] = "āte",		["าเต"] = "āte",		[dc("काते")] = "āte",	[dc("কাতে")] = "āte",
	["ာတေ"] = "āte",		["ါတေ"] = "āte",		["ᩣᨲᩮ"] = "āte",		["ᩤᨲᩮ"] = "āte",
	[dc("កាតេ")] = "āte",	[dc("කාතෙ")] = "āte",	[dc("𑀓𑀸𑀢𑁂")] = "āte",  ["າເຕ"] = "āte",
	[dc("𑄇𑄂𑄖𑄬")] = "āte",

	["ete"] = "ete",								[dc("केते")] = "ete",	[dc("কেতে")] = "ete",
	["ေတေ"] = "ete",								[dc("ᨠᩮᨲᩮ")] = "ete",
	[dc("កេតេ")] = "ete",	[dc("කෙතෙ")] = "ete",	[dc("𑀓𑁂𑀢𑁂")] = "ete",	[dc("𑄇𑄬𑄖𑄬")] = "ete",

	["ute"] = "ute",		[dc("กุเต")] = "ute",		[dc("कुते")] = "ute",	[dc("কুতে")] = "ute",
	[dc("ကုတေ")] = "ute",							[dc("ᨠᩩᨲᩮ")] = "ute",	
	[dc("កុតេ")] = "ute",	[dc("කුතෙ")] = "ute",	[dc("𑀓𑀼𑀢𑁂")] = "ute",  [dc("ກຸເຕ")] = "ute",
	[dc("𑄇𑄪𑄖𑄬")] = "ute",
-- optative endings
	["eyya"] = "eyya",		["ยฺย"] = "eyya",		[dc("केय्य")] = "eyya",	[dc("কেয্য")] = "eyya",
	["ေယျ"] = "eyya",		["ยยะ"] = "eyya",		[dc("ᨠᩮᨿ᩠ᨿ")] = "eyya",	["ຢຢະ"] = "eyya",
	[dc("កេយ្យ")] = "eyya",	[dc("කෙය්‍ය")] = "eyya",	[dc("𑀓𑁂𑀬𑁆𑀬")] = "eyya",	["ຢ຺ຢ"] = "eyya",
	["ຍ຺ຍ"] = "eyya",		["ຍຍະ"] = "eyya",		[dc("𑄇𑄬𑄠𑄳𑄠")] = "eyya",

	["ā"] = "ā",			["า"] = "ā",			[dc("का")] = "ā",	[dc("কা")] = "ā",
	["ာ"] = "ā",			["ါ"] = "ā",			["ᩣ"] = "ā",		["ᩤ"] = "ā",
	[dc("កា")] = "ā",		[dc("කා")] = "ā",		[dc("𑀓𑀸")] = "ā",    ["າ"] = "ā",
	[dc("𑄇𑄂")] = "ā",

	["etha"] = "etha",		["ถ"] = "etha",			[dc("केथ")] = "etha",	[dc("কেথ")] = "etha",
	["ေထ"] = "etha",		["ถะ"] = "etha",		[dc("ᨠᩮᨳ")] = "etha",	["ຖ"] = "etha",
	[dc("កេថ")] = "etha",	[dc("කෙථ")] = "etha",	[dc("𑀓𑁂𑀣")] = "etha",	["ຖະ"] = "etha",
	[dc("𑄇𑄬𑄗")] = "etha",
-- imperfect endings
	["attha"] = "attha",	["ตฺถ"] = "attha",		["त्थ"] = "attha",	["ত্থ"] = "attha",
	["တ္ထ"] = "attha",		["ᨲ᩠ᨳ"] = "attha",		["ตถะ"] = "attha",	["ຕຖະ"] = "attha",
	["ត្ថ"] = "attha",		["ත්‍ථ"] = "attha",		["𑀢𑁆𑀣"] = "attha",  ["ຕ຺ຖ"] = "attha",
	["𑄖𑄴𑄗"] = "attha",
	["a"] = "a",          ["ะ"] = "a",             ["ະ"] = "a", -- Only good for alphabets!
}

local aorist_endings = {
-- aorist endings.  Only active seems predictable enough.
	["i"] = "i",		[dc("กิ")] = "i",		 [dc("कि")] = "i",		 [dc("কি")] = "i",
	[dc("ကိ")] = "i",	[dc("ᨠᩥ")] = "i",
	[dc("កិ")] = "i",	[dc("කි")] = "i",		 [dc("𑀓𑀺")] = "i",	 [dc("ກິ")] = "i",
	[dc("𑄇𑄨")] = "i",

	["esi"] = "esi",		["สิ"] = "esi",		[dc("केसि")] = "esi",	[dc("কেসি")] = "esi",
	[dc("ကေသိ")] = "esi",	 [dc("ᨠᩮᩈᩥ")] = "esi",
	[dc("កេសិ")] = "esi",	[dc("කෙසි")] = "esi",	[dc("𑀓𑁂𑀲𑀺")] = "esi",   ["ສິ"] = "esi",
	[dc("𑄇𑄬𑄥𑄨")] = "esi",

	["āsi"] = "āsi",		["าสิ"] = "āsi",	[dc("कासि")] = "āsi",	 [dc("কাসি")] = "āsi",
	[dc("ကာသိ")] = "āsi", [dc("ကါသိ")] = "āsi", [dc("ᨠᩣᩈᩥ")] = "āsi", [dc("ᨠᩤᩈᩥ")] = "āsi",
	[dc("កាសិ")] = "āsi",	[dc("කාසි")] = "āsi",	[dc("𑀓𑀸𑀲𑀺")] = "āsi",	[dc("ກາສິ")] = "āsi",
	[dc("𑄇𑄂𑄖𑄨")] = "āsi",

-- Purpose of "osi" is to bleed "āsi" when the vowel is written as two characters in from NFC
	[dc("ကောသိ")] = "osi", [dc("ကေါသိ")] = "osi", [dc("ᨠᩮᩣᩈᩥ")] = "osi", [dc("ᨠᩮᩤᩈᩥ")] = "osi",


	["ayi"] = "ayi",		["ยิ"] = "ayi",	["यि"] = "ayi",	 ["যি"] = "ayi",
	["ယိ"] = "ayi", ["ᨿᩥ"] = "ayi",	["ะยิ"] = "ayi",	["ະຍິ"] = "ayi",	["ະຢິ"] = "ayi",
	["យិ"] = "ayi",	["යි"] = "ayi",	["𑀬𑀺"] = "ayi",	["ຍິ"] = "ayi",	["ຢິ"] = "ayi",
	["𑄠𑄨"] = "ayi",
}

local future_endings = {
-- fa endings
	["ati"] = "ati",		["ติ"] = "ati",			["ति"] = "ati",			["তি"] = "ati",
	["တိ"] = "ati",									["ᨲᩥ"] = "ati",
	["តិ"] = "ati",			["ති"] = "ati",			["𑀢𑀺"] = "ati",  	["ຕິ"] = "ati",
	["𑄖𑄨"] = "ati",

	["iti"] = "iti",		[dc("กิติ")]="iti",			[dc("किति")]="iti",		[dc("কিতি")]="iti",
	[dc("ကိတိ")]="iti",									[dc("ᨠᩥᨲᩥ")]="iti",
	[dc("កិតិ")]="iti",		[dc("කිති")]="iti",			[dc("𑀓𑀺𑀢𑀺")]="iti",		[dc("ກິຕິ")]="iti",
	[dc("𑄇𑄨𑄖𑄨")]="iti",

	["heti"] = "heti",		["เหติ"] = "heti",			["हेति"] = "heti",		["হেতি"] = "heti",
	["ဟေတိ"] = "heti",									["ᩉᩮᨲᩥ"] = "heti",
	["ហេតិ"] = "heti",		["හෙති"] = "heti",			["𑀳𑁂𑀢𑀺"] = "heti",		["ເຫຕິ"] = "heti",
	["𑄦𑄬𑄖𑄨"] = "heti",
	
	["kāhiti"] = "kāhiti",	["กาหิติ"] = "kāhiti",		["काहिति"] = "kāhiti",	["কাহিতি"] = "kāhiti",
	["ကာဟိတိ"] = "kāhiti",							["ᨠᩣᩉᩥᨲᩥ"] = "kāhiti",
	["កាហិតិ"] = "kāhiti",	["කාහිති"] = "kāhiti",		["𑀓𑀸𑀳𑀺𑀢𑀺"] = "kāhiti",	["ກາຫິຕິ"] = "kāhiti",
	["𑄇𑄂𑄦𑄨𑄖𑄨"] = "kāhiti",

-- fm endings
	["ate"] = "ate",		["เต"] = "ate",			["ते"] = "ate",			["তে"] = "ate",
	["တေ"] = "ate",									["ᨲᩮ"] = "ate",
	["តេ"] = "ate",			["තෙ"] = "ate",			["𑀢𑁂"] = "ate",   ["ເຕ"] = "ate",
	["𑄖𑄬"] = "ate",

-- ca ending.
	["ā"] = "ā",			["า"] = "ā",			[dc("का")] = "ā",	[dc("কা")] = "ā",
	["ာ"] = "ā",			["ါ"] = "ā",			["ᩣ"] = "ā",		["ᩤ"] = "ā",
	[dc("កា")] = "ā",		[dc("කා")] = "ā",		[dc("𑀓𑀸")] = "ā",    ["າ"] = "ā",
	[dc("𑄇𑄂")] = "ā",

	["a"] = "a",
-- cm ending
	["atha"] = "atha",		["ถ"] = "atha",			["थ"] = "atha",	["থ"] = "atha",
	["ထ"] = "atha",		["ถะ"] = "atha",		["ᨳ"] = "atha",	["ຖ"] = "atha",
	["ថ"] = "atha",	["ථ"] = "atha",	["𑀣"] = "atha",	["ຖະ"] = "atha",
	["𑄗"] = "atha",
}

local cons = "^[ก-ฮअक-हঅক-হက-ဢᨠ-ᩌᩔ-ᩖᩚ-ᩞក-អඅක-ෆ𑀅𑀓-𑀷ກ-ຮໜ-ໟ𑄇-𑄦𑅇]"

local detectEndingFromList = function(cite3s, endings, latinise, flags)
	if latinise then
		local cfin = cite3s		                     -- In general, this logic would need the
		cite3s = (lang:transliterate(cfin, latinise))  -- trwo() function from the transliterator.
	end
	local len
	local ending
	local wordEnd
	for len = 5, 1, -1 do
		wordEnd = sub(cite3s, -len)
		ending = endings[wordEnd]
		if not ending then
			-- Try again
		elseif wordEnd == "ติ" then
			if match(cite3s, 'เ.ติ$') then
				return "eti"
			elseif match(cite3s, 'โ.ติ$') then
				return "oti"
			elseif match(cite3s, dc('เ.กฺ.ติ$')) then
				return "eti"
			elseif match(cite3s, dc('โ.กฺ.ติ$')) then
				return "oti"
			else
				return ending
			end
		elseif wordEnd == "เต" then
			if match(cite3s, 'เ.เต$') then
				return "ete"
			elseif match(cite3s, dc('เ.กฺ.เต$')) then
				return "ete"
			else
				return ending
			end
		elseif wordEnd == "ຕິ" then
			if match(cite3s, 'ເ.ຕິ$') then
				return "eti"
			elseif match(cite3s, 'ໂ.ຕິ$') then
				return "oti"
			elseif match(cite3s, dc('ເ.ກ຺.ຕິ$')) then
				return "eti"
			elseif match(cite3s, dc('ໂ.ກ຺.ຕິ$')) then
				return "oti"
			else
				return ending
			end
		elseif wordEnd == "ເຕ" then
			if match(cite3s, 'ເ.ເຕ$') then
				return "ete"
			elseif match(cite3s, dc('ເ.ກ຺.ເຕ$')) then
				return "ete"
			else
				return ending
			end
		elseif wordEnd == "สิ" then
			if match(cite3s, 'เ.สิ$') then
				return "esi"
			elseif match(cite3s, dc('เ.กฺ.สิ$')) then
				return "esi"
			else
				-- Try again
			end
		elseif wordEnd == "ສິ" then
			if match(cite3s, 'ເ.ສິ$') then
				return "esi"
			elseif match(cite3s, dc('ເ.ກ຺.ສິ$')) then
				return "esi"
			else
				-- Try again
			end
		elseif ending == 'ayati' and len == 3 then
			if match(sub(cite3s,-4,-4), cons) then
				return ending
			else
				return "ati"
			end
		elseif ending == 'ayate' and len == 3 then
			if match(sub(cite3s,-4,-4), cons) then
				return ending
			else
				return "ate"
			end
		elseif ending == 'ayi' and len == 2 then
			if match(sub(cite3s,-3,-3), cons) then
				return ending
			else
				return "i"
			end
		elseif ending == 'āsi' and flags then
			if flags.aasi == '1' then
				return ending
			else
				-- Try again
			end
		elseif ending == 'osi' then
			return "i" -- Avoid misrecognising as 'āsi' in Burmese and similar.
		else
--			if ending == "i" and not allowi then
--				ending = "ati" -- For brūti and similar
--			end
-- debug_outer = debug_outer..'\n'..cite3s..'='..ending..'\n'
			return ending
		end
	end
	if match(sub(cite3s, -1, -1), cons) then
-- debug_outer = debug_outer..'\n'..cite3s..'='.."a"..'\n'
		return "a"
	else
-- debug_outer = debug_outer..'\n'..cite3s..'='.."ati"..'\n'
		return "ati"
	end
end

export.detectEnding = function(cite3s, latinise)
	return detectEndingFromList(cite3s, endings, latinise)
end

export.detectAoristEnding = function(cite3s, latinise, flags)
	return detectEndingFromList(cite3s, aorist_endings, latinise, flags)
end


local active_optative = {
	{"⌫⌫⌫eyyāmi", "⌫⌫⌫eyyaṃ", "⌫⌫⌫e"}, {"⌫⌫⌫eyyāma"},
	{"⌫⌫⌫eyyāsi", "⌫⌫⌫e"},             {"⌫⌫⌫eyyātha"},
	{"⌫⌫⌫eyya",   "⌫⌫⌫e"},             {"⌫⌫⌫eyyuṃ"}
}
local active_optative_ati  = mw.clone(active_optative)
local active_optative_aati = mw.clone(active_optative)
active_optative_aati[1] = {"⌫⌫⌫eyyāmi", "⌫⌫⌫eyyaṃ"}
active_optative_aati[3] = {"⌫⌫⌫eyyāsi"}
local active_optative_eti  = mw.clone(active_optative_aati)
active_optative_eti[5] = {"⌫⌫⌫eyya"}
local active_optative_oti  = mw.clone(active_optative_eti)
local active_optative_ute  = mw.clone(active_optative_eti)
local middle_optative = {
	{"⌫⌫⌫eyyaṃ"}, {"⌫⌫⌫eyyāmhe"},
	{"⌫⌫⌫etho"},  {"⌫⌫⌫eyyavho"},
	{"⌫⌫⌫etha"},  {"⌫⌫⌫eraṃ"}
}
local middle_optative_ati  = mw.clone(middle_optative)
local middle_optative_aati = mw.clone(middle_optative)
local middle_optative_eti  = mw.clone(middle_optative)
local middle_optative_oti  = mw.clone(middle_optative)
local middle_optative_ute  = mw.clone(middle_optative)

local empty = {{}, {}, {}, {}, {}, {}}

local pattern = {
	["ati"] = {
		{"⌫⌫⌫āmi"}, {"⌫⌫⌫āma"},   {"⌫⌫si"}, {"⌫ha"},    {""}, {"⌫⌫nti"},
		mid = "ate",
		imperfect = {
			{"⌫⌫", "⌫⌫ṃ"},	{"⌫⌫mhā"},
			{"⌫⌫⌫o"},		{"⌫tha"},
			{"⌫⌫", "⌫⌫⌫ā"},	{"⌫⌫⌫u", "⌫⌫⌫ū"},
		},
		imperative = {
			{"⌫⌫⌫āmi"}, {"⌫⌫⌫āma"}, {"⌫⌫", "⌫⌫⌫āhi"}, {"⌫ha"}, {"⌫u"}, {"⌫⌫ntu"}
		},
		optative_active = active_optative_ati,
		optative_middle = middle_optative_ati,
		participle = {{"⌫⌫nt"}},
	},
	["āti"] = { 
		{"⌫⌫mi"}, {"⌫⌫ma"},   {"⌫⌫si"}, {"⌫ha"},    {""}, {"⌫⌫⌫anti"},
		mid = "āte",
		imperfect = {
			{"⌫⌫⌫a", "⌫⌫⌫aṃ"},	{"⌫⌫⌫amhā"},
			{"⌫⌫⌫o"},			{"⌫⌫⌫attha"},
			{"⌫⌫⌫a", "⌫⌫"},		{"⌫⌫⌫u", "⌫⌫⌫ū"},
		},
		imperative = {
			{"⌫⌫mi"}, {"⌫⌫ma"},  {"⌫⌫", "⌫⌫hi"}, {"⌫ha"},  {"⌫u"}, {"⌫⌫⌫antu"},
		},
		optative_active = active_optative_aati,
		optative_middle = middle_optative_aati,
		participle = {{"⌫⌫⌫ant"}},
	},
	["eti"] = {
		{"⌫⌫mi"}, {"⌫⌫ma"},   {"⌫⌫si"}, {"⌫ha"},    {""}, {"⌫⌫nti"},
		mid = nil, -- "ete",
		imperfect = empty,
		imperative = {
			{"⌫⌫mi"}, {"⌫⌫ma"},   {"⌫⌫hi"}, {"⌫ha"},    {"⌫u"}, {"⌫⌫ntu"},
		},
		optative_active = active_optative_eti,
		optative_middle = middle_optative_eti,
		participle = {{"⌫⌫nt"}},
	},
	["oti"] = {
		{"⌫⌫mi"}, {"⌫⌫ma"},   {"⌫⌫si"}, {"⌫ha"},    {""}, {"⌫⌫nti"},
		mid = "oti_ute",
		imperfect = empty,
		imperative = {
			{"⌫⌫mi"}, {"⌫⌫ma"},   {"⌫⌫hi"}, {"⌫ha"},    {"⌫u"}, {"⌫⌫ntu"},
		},
		optative_active = active_optative_oti,
		optative_middle = middle_optative_oti,
		participle = {{"⌫⌫nt"}},
	},
	["ate"] = { -- Also services "ati"
		{"⌫⌫⌫e"}, {"⌫⌫⌫āmhe"},  {"⌫⌫se"}, {"⌫⌫vhe"},  {"⌫e"}, {"⌫⌫nte", "⌫⌫re"},
		mid = "ate",
		imperfect = {
			{"⌫⌫⌫iṃ"},	{"⌫⌫⌫āmhase", "⌫⌫mhase"},
			{"⌫⌫se"},	{"⌫⌫vhaṃ"},
			{"⌫tha"},	{"⌫thuṃ"},
		},
		imperative = {
			{"⌫⌫⌫e"}, {"⌫⌫⌫āmase"},  {"⌫⌫ssu"}, {"⌫⌫vho"},  {"⌫aṃ"}, {"⌫⌫ntaṃ"},
		},
		optative_active = active_optative_ati,
		optative_middle = middle_optative_ati,
		participle = {{"⌫⌫māna"}},
	},
	["āte"] = { -- Also services "āti"
		{"⌫⌫⌫e"},	{"⌫⌫mhe"},
		{"⌫⌫se"},	{"⌫⌫⌫avhe"},
		{"⌫e"},		{"⌫⌫⌫ante", "⌫⌫re"},
		mid = "āte",
		imperfect = empty,
		optative_active = active_optative_aati,
		optative_middle = middle_optative_aati,
		participle = {{"⌫⌫⌫amāna"}},
	},
	["ete"] = { -- Also services "eti"
		{"⌫⌫⌫aye"},		{"⌫⌫⌫ayāmhe"},
		{"⌫⌫⌫ayase"},	{"⌫⌫⌫ayavhe"},
		{"⌫⌫⌫ayate"},	{"⌫⌫⌫ayante", "⌫⌫⌫ayare"},
		mid = "ete",
		imperfect = empty,
		imperative = {
			{"⌫⌫⌫aye"},		{"⌫⌫⌫ayāmase"},
			{"⌫⌫⌫ayassu"},	{"⌫⌫⌫ayavho"},
			{}, -- {"⌫⌫⌫ayataṃ", "⌫aṃ"}, -- 3s needs review
			{"⌫⌫⌫ayantaṃ"}
		},
		optative_active = active_optative_eti,
		optative_middle = middle_optative_eti,
		participle = {{"⌫⌫⌫ayamāna"}},
	},
	["ute"] = { -- Actual ending is "oti" when copied to "oti_ute".
		{"⌫⌫⌫ve"},	{"⌫⌫⌫umhe"},
		{"⌫⌫⌫use"},	{"⌫⌫⌫uvhe"},
		{"⌫⌫⌫ute"},	{"⌫⌫⌫unte", "⌫⌫⌫ure"},
		mid = "ute",
		imperfect = empty,
		imperative = {
			{"⌫⌫⌫ve"},		{"⌫⌫⌫vāmase"},
			{"⌫⌫⌫ussu"},	{"⌫⌫⌫uvho"},
			{"⌫⌫⌫utaṃ"},	{"⌫⌫⌫untaṃ"}
			-- 1s, 1pl and 3pl will often need manual repair.
		},
		optative_active = active_optative_ute,
		optative_middle = middle_optative_ute,
	},
	["eyya"] = {
		optative_active = { -- '-e' is not included!
			{"⌫āmi", "⌫aṃ"}, {"⌫āma"},
			{"⌫āsi"},        {"⌫ātha"},
			{""},           {"⌫uṃ"}
		},
		optative_middle = {
			{"ṃ"},       {"⌫āmhe"},
			{"⌫⌫⌫tho"},  {"vho"},
			{"⌫⌫⌫tha"},  {"⌫⌫⌫raṃ"}
		},
	},
	["ā"] = {
		optative_active = {
			{"mi"}, {"ma"},  {"si"}, {"tha"},  {""}, {"⌫uṃ"}
		}
	},
	["etha"] = {
		mid = "etha",
		optative_middle = {
			{"⌫⌫⌫yyaṃ"}, {"⌫⌫⌫yyāmhe"},
			{"⌫o"},      {"⌫⌫⌫yyavho"},
			{""},        {"⌫⌫⌫raṃ"}
		},
	},
	["a"] = {
		mid="a_attha",
		imperfect = {
			{"", "ṃ"},	{"mhā"},
			{"⌫o"},		{"ttha"},
			{"", "⌫ā"},	{"⌫u", "⌫ū"},
		}
	},
	["ā2"] = { -- imperfect form
		mid="ā_attha",
		imperfect = {
			{"⌫a", "⌫aṃ"},	{"⌫amhā"},
			{"⌫o"},			{"⌫attha"},
			{"⌫a", ""},		{"⌫u", "⌫ū"},
		}
	},
	["attha"] = {
		mid = "attha",
		imperfect = {
			{"⌫⌫⌫⌫⌫iṃ"}, {"⌫⌫⌫⌫⌫āmhase", "⌫⌫⌫⌫mhase"},
            {"⌫⌫⌫⌫se"}, {"⌫⌫⌫⌫vhaṃ"}, {""}, {"⌫uṃ"}
		}
	},
	["a_attha"] = {
		mid = "a_attha",
		imperfect = {
			{"⌫iṃ"}, {"⌫āmhase", "mhase"},
            {"se"}, {"vhaṃ"}, {"ttha"}, {"tthuṃ"}
		}
	},
	["ā_attha"] = {
		mid = "ā_attha",
		imperfect = {
			{"⌫iṃ"}, {"mhase", "⌫amhase"},
            {"⌫ase"}, {"⌫avhaṃ"}, {"⌫attha"}, {"⌫atthuṃ"}
		}
	},
	["i"] = {
		mid = nil,
		aorist = {{"ṃ"}, {"mha"}, {""}, {"ttha"}, {""}, {"ṃsu"}} -- 3p might be wrong too often
		-- However, -ayi seems to consistently yield -ayiṃsu rather than ayuṃ.
	},
	["esi"] = {
		mid = nil,
		aorist = {{"ṃ"}, {}, {""}, {}, {""}, {"⌫uṃ"}} -- 3p might be wrong too often
	},
	["āsi"] = { -- A common mix of sigmatic and thematic aorists!
		mid = nil,
		aorist = {{"ṃ"}, {"⌫⌫⌫amha", "⌫⌫⌫amhā"}, {""}, {"⌫⌫⌫attha"}, {""}, {"⌫⌫⌫aṃsu"}}
	},
}

-- -oti is too different from -ute in some scripts.
pattern.oti_ute = mw.clone(pattern.ute)
pattern.oti_ute.mid = "oti_ute"

pattern.ayati = mw.clone(pattern.ati)
pattern.ayati.mid = "ayate"
-- Lose some of the shorter forms.
pattern.ayati.imperative[3] = {"⌫⌫⌫āhi"}
pattern.ayati.optative_active[1] = {"⌫⌫⌫eyyāmi", "⌫⌫⌫eyyaṃ"}
pattern.ayati.optative_active[3] = {"⌫⌫⌫eyyāsi"}
pattern.ayate = mw.clone(pattern.ate)
pattern.ayate.mid = "ayate"
pattern.ayate.imperative[5] = {} -- {"⌫aṃ", "⌫⌫⌫⌫⌫etaṃ"} -- 3s needs review
pattern.ayate.optative_active = pattern.ayati.optative_active

pattern.ayi = mw.clone(pattern.i)
pattern.ayi.aorist = mw.clone(pattern.i.aorist)
pattern.ayi.aorist[6] = {"ṃsu"}

-- convert Latin script inflections to another script
-- Ⓓ deletes last preposed vowel (for Thai and Lao)
-- Ⓣ toggles last preposed vowel between 'e' and 'o' (for Thai and Lao)
local convert_suffixes = function(stem, nstrip, suffixes, sc, impl, yopt)
	local form, pre, stem_pre
	local yung, yaa
	local xlitend = {}
	local strip = string.rep("⌫", nstrip)
	local altstem, as1
	local option = {impl = impl}
	if sc == 'Thai' or sc == 'Laoo' then
		altstem = to_script(stem, sc)
		stem_pre = match(altstem, "^[เโເໂ]")
		as1 = sub(altstem, 1, 1)
		if yopt and sc == 'Laoo' then
			if yopt == 'yung' or yopt == 'ຍ' then
				yung = 1
			elseif yopt == 'yaa' or yopt == 'ຢ' then
				yaa = 1
			elseif yopt == 'both' then
				yung = 1 yaa = 1
			else
				error('Parameter yaa has invalid value "'..yopt..'"')
			end
		end
	end
	for k = 1, #suffixes do
		xlitend[k] = {}
		form = export.joinSuffix('Latn', stem, suffixes[k])
		local altforms = {}
		for ia, va in pairs(form) do
			altform = to_script(va, sc, option)
			if yung or yaa then
				local y1, y2
				if yung then y1 = gsub(altform, 'ຢ', 'ຍ') end
				if yaa  then y2 = gsub(altform, 'ຍ', 'ຢ') end
				if y1 then
					table.insert(altforms, y1)
					if y2 then
						if y1 ~= y2 then table.insert(altforms, y2) end
					end
				else
					table.insert(altforms, y2)
				end
			else
				table.insert(altforms, altform)
			end
		end
		local ib = 1 -- Conversion is not one-to-one!
		for ia, altform in pairs(altforms) do
			local ibin = ib
-- Special handling is needed for a preposed vowel.
			pre = match(altform, "^[เโເໂ]")
			if pre then
				if stem_pre then
					if sub(altform, 1, 1) == as1 then
						xlitend[k][ib] = sub(strip, 2) .. sub(altform, 3)
						ib = ib + 1
					else
--						mw.addWarning("Cannot abstract derivation "..altstem.." > "..altform)
						xlitend[k].voi = true
						xlitend[k][ib] = sub(strip, 2) .. 'Ⓣ' .. sub(altform, 3)
						ib = ib + 1
					end -- Don't generate gibberish
				else
					xlitend[k][ib] = strip .. "↶" .. pre .. sub(altform, 3)
					ib = ib + 1
				end
			else
				if stem_pre then
					xlitend[k].voi = true
					xlitend[k][ib] = sub(strip, 2) .. 'Ⓓ' .. sub(altform, 2)
					ib = ib + 1
				else
					xlitend[k][ib] = strip .. sub(altform, 2)
					ib = ib + 1
				end
			end
-- Add alternative Thai/Lao endings
-- A lot of data is needed to decide whether the disabled logic is needed.
-- Any dead code here must be kept around in case facts later show that it is
-- needed for some spelling variant.
			if (sc == 'Thai' or sc == 'Laoo') and impl == 'yes' and ib > ibin then
				local altend
				if impl=='yes' then
					altend = gsub(xlitend[k][ibin], dc("([มวມວ][กฺກ຺])([เโເໂ])([หຫ])"), "%2%1%3")
				else
					xlitend[k][ibin] = gsub(xlitend[k][ibin], "ัว", "ะว") -- TBC!
					altend = gsub(xlitend[k][ibin], dc("([มวມວ])([เโເໂ])([หຫ])"), "%2%1%3")
				end
				if altend ~= xlitend[k][ibin] then
					xlitend[k][ib] = xlitend[k][ibin]
					xlitend[k][ibin] = altend
					ib = ib + 1
				end
			end
		end
	end
	return xlitend
end

function export.getSuffixes(sc, ending, tense, option)
	local impl = option and option.impl or 'yes'
	local yopt = option and option.y
	local pat
	if not pattern[ending] then
		error('No verb pattern for ending "'..ending..'".')
	end
	if tense then
		pat = pattern[ending][tense]
	else
		pat = pattern[ending]
	end
	if 'Latn' == sc then
		return pat
	elseif not pat then
		return nil
	elseif pat[sc] then
		return pat[sc]
	else
		local olen = {Mymr = 2, Lana = 2} -- Assume NFC
		local attha_len = {Latn = 5, Sinh = 4}
		eyya_len = {Mymr = 3, Sinh = 5}
		local alen
		if impl == 'no' and (sc == 'Thai' or sc == 'Laoo') then
			alen = 1
		else
			alen = 0
		end
		local way = {
			["ati"]  = {pseudostem = "kati", ndel = 2 + alen},
			["ayati"]  = {pseudostem = "kayati", ndel = 3 + 2 * alen},
			["āti"]  = {pseudostem = "kāti", ndel = 3},
			["eti"]  = {pseudostem = "keti", ndel = 3},
			["oti"]  = {pseudostem = "soti", ndel = 2 + (olen[sc] or 1)},
			["oti_ute"]  = {pseudostem = "soti", ndel = 2 + (olen[sc] or 1)},
			["ate"]  = {pseudostem = "kate", ndel = 2 + alen},
			["ayate"]  = {pseudostem = "kayate", ndel = 3 + 2 * alen},
			["āte"]  = {pseudostem = "kāte", ndel = 3},
			["ete"]  = {pseudostem = "kete", ndel = 3},
			["ute"]  = {pseudostem = "sute", ndel = 3},
			["eyya"] = {pseudostem = "keyya", ndel = eyya_len[sc] or 4},
			["ā"]    = {pseudostem = "kā", ndel = 1},
			["ā2"]   = {pseudostem = "kā", ndel = 1},
			["a"]    = {pseudostem = "ka", ndel = alen},
			["a_attha"] = {pseudostem = "ka", ndel = alen},
			["ā_attha"] = {pseudostem = "kā", ndel = 1},
			["etha"] = {pseudostem = "ketha", ndel = 2 + alen},
			["attha"] = {pseudostem = "kattha", ndel = (attha_len[sc] or 3) + alen},
			["i"]     = {pseudostem = "ki", ndel = 1},
			["esi"]   = {pseudostem = "kesi", ndel = 3},
			["āsi"]   = {pseudostem = "kāsi", ndel = 3},
			["ayi"]   = {pseudostem = "kayi", ndel = 2 + alen},
		}
		way = way[ending]
		if way then
			pat[sc] = convert_suffixes(way.pseudostem, way.ndel, pat, sc,
									   impl, yopt)
			if ending == 'etha' then
				debug_outer=debug_outer..'tr:'..pat[sc][1][1]
			end
		else
			error('Cannot convert derivations from suffix '..ending..' for '..sc)
		end
		return pat[sc]
	end
end

local function dump_table(name, futargs) -- For debugging only!
	for k, v in pairs(futargs) do
		local t = type(v)
		if t == 'string' then
			mw.addWarning(name..'.'..k..'='..v)
		elseif t == 'table' then
			mw.addWarning(name..'.'..k..' is a table of '..#v)
		else
			mw.addWarning(name..'.'..k..' is a '..type(v))
		end
	end
end

local function define_overrides(params, prefix, mod_default)
	for _, pn in pairs({"1s", "2s", "3s", "1p", "2p", "3p"}) do
		params[prefix.."a"..pn] = {list = true}
		params[prefix.."m"..pn] = {list = true}
		params[prefix.."a"..pn.."_mod"] = {default = mod_default}
		params[prefix.."m"..pn.."_mod"] = {default = mod_default}
	end
end

local function addFutureParams(params, mod_default) -- i.e. future specific
	params.cond = {list = true} -- Only forms including augment
	params.cond_midl = {list = true} -- Only forms including augment
	params.cond_voice = {} -- defaults to (corrected) value of voice
	params.fap_mod = {default = mod_default}
	params.fap = {list = true}
	params.fmp_mod = {default = mod_default}
	params.fmp = {list = true}
	define_overrides(params, 'cond_', mod_default)
end

function export.parse(args)
	local mod_default = 'after'
	if args.impf_midl and not args.midl then
		error("Argument impf_midl may only be used if argument midl is.")
	end
	local params = {
		[1] = {}, -- Just ignore
		cite = {list = true},
		midl = {list = true},
		label = {},
		impl = {},
		y = {},
		aa = {},
		showtr = {},
		subst = {},
		sc = {},
   		voice = {default = 'both'}, -- Overridden by cite and midl
		impf_voice = {default = 'none'},
		augment = {default = 'both_made'},
		impf = {list = true}, -- Only forms including augment
		impf_midl = {list = true}, -- Only forms including augment
		impr_voice = {}, -- defaults to (corrected) value of voice
		opta_voice = {}, -- defaults to (corrected) value of voice
		opta_mod = {default = mod_default},
		opta = {list = true},
		opta_midl = {list = true},
		aori = {list = true},
		aor2 = {list = true},
		aori_voice = {default = 'act'}, -- paradigms only generate active forms!
		aori_augment = {}, -- No default!
		aasi = {}, -- Set to 1 to recognise "āsi" as a special ending.
		perf = {list = true},
		perf_voice = {default = 'both'},
		part_voice = {}, -- defaults to (corrected) value of voice
		pap_mod = {default = mod_default},
		pap = {list = true},
		pmp_mod = {default = mod_default},
		pmp = {list = true},
-- Options for future system:
		futu = {list = true},
		futu_midl = {list = true},
		futu_voice = {},
		cond_augment = {},
		futu_part_voice = {},
	}
	addFutureParams(params, mod_default)
	define_overrides(params, '',      mod_default)
	define_overrides(params, 'impf_', mod_default)
	define_overrides(params, 'impr_', mod_default)
	define_overrides(params, 'opta_', mod_default)
	define_overrides(params, 'aori_', mod_default)
	define_overrides(params, 'perf_', mod_default)
-- Options for future system:
	define_overrides(params, 'futu_', mod_default)
	define_overrides(params, 'cond_', mod_default)
	return m_parameters.process(args, params)
end

function export.futureParse(args, subset)
	local mod_default = 'after'
	if args.cond_midl and not args.midl then
		error("Argument cond_midl may only be used if argument midl is.")
	end
	local params = {
		[1] = {}, -- Just ignore
		cite = {list = true},
		midl = {list = true},
		label = {},
		impl = {},
		y = {},
		aa = {},
		showtr = {},
		subst = {},
		sc = {},
   		voice = {default = 'both'}, -- Overridden by cite and midl
		augment = {default = 'with_made'},
		part_voice = {} -- defaults to (corrected) value of voice
	}
	addFutureParams(params, mod_default)
	define_overrides(params, '',      mod_default)
	if subset then -- Lists have already been formed.
		define_overrides(params, 'futu_', mod_default)
		local futargs = {}
		local replaced = {
			[1] = 1, cite = 1, midl = 1, voice = 1, augment = 1, part_voice = 1
		}
		for k, v in pairs(args) do
			if params[k] and not replaced[k] then
				if type(k) ~= 'string' then
					-- Weird!  Just skip.
				elseif match(k,  '^futu_[am][123]') then
					futargs[sub(k, 6)] = args[k]
				elseif match(k, '^[am][123][sp]') then
					-- Don't copy
				else
					futargs[k] = args[k]
				end
			end
		end
		futargs.cite		= args.futu
		futargs.midl		= args.futu_midl
		futargs.voice		= args.futu_voice or 'both'
		futargs.augment		= args.cond_augment or
			match(args.augment, '^with') and args.augment or 'with_made' 
		futargs.part_voice	= args.futu_part_voice or args.part_voice
--		dump_table('futargs', futargs)
		return futargs
	else
		return m_parameters.process(args, params)
	end
end

local arrcat_nodup = m_noun.arrcat_nodup

local function kill_final_vole(stem)
	return gsub(stem, "([เโເໂ])([^เโເໂ]*)$", "%2")
end

local function toggle_vo(stem)
	local toggle = {["เ"]="โ", ["โ"]="เ", ["ເ"]="ໂ", ["ໂ"]="ເ"}
	local function f_toggle(m1, m2) return toggle[m1]..m2 end
	return gsub(stem, "([เโເໂ])([^เโເໂ]*)$", f_toggle)
end

function export.joinSuffix(sc, base, pattern, option)
	if pattern.voi then
		local res = {}
		for _,v in ipairs(pattern) do
			local forms, rest
			local base2 = nil
			if match(v, '⌫⌫⌫⌫⌫Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -6))
				rest = sub(v,7)
			elseif match(v, '⌫⌫⌫⌫Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -5))
				rest = sub(v,6)
			elseif match(v, '⌫⌫⌫Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -4))
				rest = sub(v,5)
			elseif match(v, '⌫⌫Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -3))
				rest = sub(v,4)
			elseif match(v, '⌫Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -2))
				rest = sub(v,3)
			elseif match(v, 'Ⓓ') then
				base2 = kill_final_vole(sub(base, 1, -1))
				rest = sub(v,2)
			elseif match(v, '⌫⌫⌫⌫⌫Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -6))
				rest = sub(v,7)
			elseif match(v, '⌫⌫⌫⌫Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -5))
				rest = sub(v,6)
			elseif match(v, '⌫⌫⌫Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -4))
				rest = sub(v,5)
			elseif match(v, '⌫⌫Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -3))
				rest = sub(v,4)
			elseif match(v, '⌫Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -2))
				rest = sub(v,3)
			elseif match(v, 'Ⓣ') then
				base2 = toggle_vo(sub(base, 1, -1))
				rest = sub(v,2)
			end
			if base2 then
				forms = m_noun.joinSuffix(sc, base2, {rest})
			else
				forms = m_noun.joinSuffix(sc, base, {v})
			end
			res = arrcat_nodup(res, forms)
		end
		return res
	elseif pattern.kill_final_preposed then
		return m_noun.joinSuffix(sc, kill_final_vole(base), pattern, option)
	else
		return m_noun.joinSuffix(sc, base, pattern, option)
	end
end

function export.joinSuffixes(scriptCode, stem, pattern, option)
	local forms = {}
	for i = 1, 6 do
		forms[i] = export.joinSuffix(scriptCode, stem, pattern[i], option)
	end
	return forms
end

function export.modify_voice(forms, prefix, args)
	-- Only null action is implemented.
		for i, pn in pairs({"1s", "1p", "2s", "2p", "3s", "3p"}) do
			local flag = pn..'_mod'
			if #args[pn] > 0 then
				if args[flag] == 'after' then
					forms[i] = arrcat_nodup(forms[i], args[pn])
					forms.isnil = false
				elseif args[flag] == 'replace' then
					forms[i] = args[pn]
					forms.isnil = false
				elseif args[flag] == 'before' then
					forms[i] = arrcat_nodup(args[pn], forms[i])
					forms.isnil = false
				elseif args[flag] == 'blank' then
					forms[i] = {}
				else
					error('Argument '..prefix..pn..' has invalid value '..args[flag])
				end
			elseif args[flag] == 'blank' then
				forms[i] = {}
			end
		end
end

function export.modify_voice(forms, prefix, args)
		for i, pn in pairs({"1s", "1p", "2s", "2p", "3s", "3p"}) do
			local flag = pn..'_mod'
			local added = nil;
			if args[pn] and #args[pn] > 0 then
				added = true;
				if args[flag] == 'after' then
					forms[i] = arrcat_nodup(forms[i], args[pn])
					forms.isnil = false
				elseif args[flag] == 'replace' then
					forms[i] = args[pn]
					forms.isnil = false
				elseif args[flag] == 'before' then
					forms[i] = arrcat_nodup(args[pn], forms[i])
					forms.isnil = false
				elseif args[flag] == 'blank' then
					forms[i] = {}
					added = nil; 
				else
					error('Argument '..prefix..pn..' has invalid value '..args[flag])
				end
				if added and (i == 2) then forms.add1p = true end
			elseif args[flag] == 'blank' then
				forms[i] = {}
			end
		end
end

function export.present(cite, tense, formsa, formsm)
	local rows = {"1st", "2nd", "3rd"}
	local forms1
	local output = {}
	ti(output, '<div class="NavFrame" style="min-width:30%">'..
				'<div class="NavHead" style="background:#d9ebff">'..tense..
				' forms of "' .. cite .. '"' ..'</div><div class="NavContent">')
	ti(output,
	'<table class="inflection-table" style="background:#F9F9F9;'..
		'text-align:center;width:100%">')

	if formsa and formsm then
		ti(output, '<tr style="background:#eff7ff"><th></th><th colspan=2>Active</th><th colspan=2>Middle</th></tr>')
		ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th><th>Singular</th><th>Plural</th></tr>')
		for i,v in ipairs(rows) do
			if #formsa[2*i-1] > 0 or #formsa[2*i] > 0 or #formsm[2*i-1] > 0 or #formsm[2*i] > 0 then
				ti(output, "<tr><th style=\"background-color:#eff7ff;\">" .. v .. "</th>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, formsa[2 * i - 1]) .. "</td>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, formsa[2 * i]) .. "</td>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, formsm[2 * i - 1]) .. "</td>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, formsm[2 * i]) .. "</td>")
				ti(output, "</tr>")
			end
		end
	else
		if formsa then
			forms1 = formsa
		else
			forms1 = formsm
		end
		ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th></tr>')
		for i,v in ipairs(rows) do
			if #forms1[2*i-1] > 0 or #forms1[2*i] > 0 then
				ti(output, "<tr><td style=\"background-color:#eff7ff;\">" .. v .. "</td>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, forms1[2 * i - 1]) .. "</td>")
				ti(output, "<td>" .. m_noun.orJoin(currentScript, forms1[2 * i]) .. "</td>")
				ti(output, "</tr>")
			end
		end
	end
	ti(output, "</table></div></div>")
	return table.concat(output)
end

-- Basic form of augment:
local augment = {
	Latn="a",
	Thai="อ",
	Deva="अ",
	Beng="অ",
	Mymr="အ",
	Lana="ᩋ",
	Khmr="អ",
	Sinh="අ",
	Brah="𑀅",
	Laoo="ອ",
	Cakm="𑄃",
}
local augment_alph= {
	Thai = "อะ",
	Laoo = "ອະ"
}

local enda3s = {
	Latn="ti",
	Thai="ติ",
	Deva="ति",
	Beng="তি",
	Mymr="တိ",
	Lana="ᨲᩥ",
	Khmr="តិ",
	Sinh="ති",
	Brah="𑀢𑀺",
	Laoo="ຕິ",
	Cakm="𑄖𑄨",
}

local function select_args(old, prefix) -- prefix must be in ASCII
	local new = {}
	local pattern = '^'..prefix
	local follow = string.len(pattern)
	for i, v in pairs(old) do
		if string.match(i,pattern) then
			new[asub(i, follow)] = v
		end
	end
	return new
end

function export.modify(formsa, formsm, prefix, args)
	local subprefix
	if not formsa then formsa = {{}, {}, {}, {}, {}, {}, isnil=true} end
	subprefix = prefix..'a'
	export.modify_voice(formsa, subprefix, select_args(args,subprefix))
	if formsa.isnil then formsa = nil end
	if not formsm then formsm = {{}, {}, {}, {}, {}, {}, isnil=true} end
	subprefix = prefix..'m'
	export.modify_voice(formsm, subprefix, select_args(args,subprefix))
	if formsm.isnil then formsm = nil end
	return formsa, formsm
end

local hike = true -- Whether heading for voices and numbers should be in common for tables. 
local function rows2v(currentScript, tense, formsa, formsm, x0)
	local rows = {"1st", "2nd", "3rd"}
	local output = {}

    formsa = formsa or empty
    formsm = formsm or empty
	ti(output, '<tr style="background:#d9ebff"><th colspan=5>'..tense..'</th></tr>')
	if not hike then
		ti(output, '<tr style="background:#eff7ff"><th></th><th colspan=2>Active</th><th colspan=2>Middle</th></tr>')
		ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th><th>Singular</th><th>Plural</th></tr>')
	end	
	for i,v in ipairs(rows) do
		if #formsa[2*i-1] > 0 or #formsa[2*i] > 0 or #formsm[2*i-1] > 0 or #formsm[2*i] > 0 then
			ti(output, "<tr><th style=\"background-color:#eff7ff;\">" .. v .. "</th>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, formsa[2 * i - 1], x0))
			ti(output, "</td>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, formsa[2 * i], x0))
			ti(output, "</td>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, formsm[2 * i - 1], x0))
			ti(output, "</td>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, formsm[2 * i], x0))
			ti(output, "</td>")
			ti(output, "</tr>")
		end
	end
	return table.concat(output)
end

local function rows1v(currentScript, tense, formsa, formsm, xo)
	local rows = {"1st", "2nd", "3rd"}
	local output = {}

	local forms1 = formsa or formsm or empty
	ti(output, '<tr style="background:#d9ebff"><th colspan=3>'..tense..'</th></tr>')
	if not hike then
		ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th></tr>')
	end
	for i,v in ipairs(rows) do
		if #forms1[2*i-1] > 0 or #forms1[2*i] > 0 then
			ti(output, "<tr><td style=\"background-color:#eff7ff;\">")
			ti(output, v)
			ti(output, "</td>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, forms1[2 * i - 1], xo))
			ti(output, "</td>")
			ti(output, "<td>")
			ti(output, m_noun.orJoin(currentScript, forms1[2 * i], xo))
			ti(output, "</td>")
			ti(output, "</tr>")
		end
	end
	return table.concat(output)
end

function export.present_conj(label, currentScript, parts, xo)
	local output = {}
	parts.cat = parts.cat or {}
	local havea = parts.presa or parts.impra or parts.impfa or parts.optaa
	local havea = havea or parts.aoria or parts.futua or parts.conda or parts.perfa
	local havem = parts.presm or parts.imprm or parts.impfm or parts.optam
	havem = havem or parts.aorim or parts.futum or parts.condm or parts.perfa
	local have2 = parts.presa and parts.presm  or  parts.impra and parts.imprm or
				parts.impfa and parts.impfm  or  parts.optaa and parts.optam or
				parts.aoria and parts.aorim  or  parts.futua and parts.futum or
				parts.conda and parts.condm  or  parts.perfa and parts.perfm
	ti(output, '<div class="NavFrame" style="min-width:30%">'..
				'<div class="NavHead" style="background:#d9ebff">'..
				'Conjugation of "' .. label .. '"</div><div class="NavContent">')
	ti(output, '<table class="inflection-table" style="background:#F9F9F9;'..
		'text-align:center;width:100%">')
	if have2 then
		if hike then
			ti(output, '<tr style="background:#eff7ff"><th></th><th colspan=2>Active</th><th colspan=2>Middle</th></tr>')
			ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th><th>Singular</th><th>Plural</th></tr>')
		end	
		if parts.presa or parts.presm then
			ti(output, rows2v(currentScript, 'Present', parts.presa, parts.presm, xo))
		end
		if parts.impfa or parts.impfm then
			ti(output, rows2v(currentScript, 'Imperfect', parts.impfa, parts.impfm, xo))
		end
		if parts.impra or parts.imprm then
			ti(output, rows2v(currentScript, 'Imperative', parts.impra, parts.imprm, xo))
		end
		if parts.optaa or parts.optam then
			ti(output, rows2v(currentScript, 'Optative', parts.optaa, parts.optam, xo))
		end
		if parts.aoria or parts.aorim then
			ti(output, rows2v(currentScript, 'Aorist', parts.aoria, parts.aorim, xo))
		end
		if parts.futua or parts.futum then
			ti(output, rows2v(currentScript, 'Future', parts.futua, parts.futum, xo))
		end
		if parts.conda or parts.condm then
			ti(output, rows2v(currentScript, 'Conditional', parts.conda, parts.condm, xo))
		end
		if parts.perfa or parts.perfm then
			ti(output, rows2v(currentScript, 'Perfect', parts.perfa, parts.perfm, xo))
		end
	else
		if hike then
			ti(output, '<tr style="background:#eff7ff"><th></th><th>Singular</th><th>Plural</th></tr>')
		end
		if parts.presa or parts.presm then
			ti(output, rows1v(currentScript, 'Present', parts.presa, parts.presm, xo))
		end
		if parts.impfa or parts.impfm then
			ti(output, rows1v(currentScript, 'Imperfect', parts.impfa, parts.impfm, xo))
		end
		if parts.impra or parts.imprm then
			ti(output, rows1v(currentScript, 'Imperative', parts.impra, parts.imprm, xo))
		end
		if parts.optaa or parts.optam then
			ti(output, rows1v(currentScript, 'Optative', parts.optaa, parts.optam, xo))
		end
		if parts.aoria or parts.aorim then
			ti(output, rows1v(currentScript, 'Aorist', parts.aoria, parts.aorim, xo))
		end
		if parts.futua or parts.futum then
			ti(output, rows1v(currentScript, 'Future', parts.futua, parts.futum, xo))
		end
		if parts.conda or parts.condm then
			ti(output, rows1v(currentScript, 'Conditional', parts.conda, parts.condm, xo))
		end
		if parts.perfa or parts.perfm then
			ti(output, rows1v(currentScript, 'Perfect', parts.perfa, parts.perfm, xo))
		end
	end
	if parts.conda or parts.condm then
		ti(parts.cat, "[[category:Pali verbs exhibiting conditional mood]]")
		if parts.conda and parts.conda.add1p then
			ti(parts.cat, "[[category:Pali verbs with manual 1p conditional]]")
		end
	end
	ti(output, "</table></div></div>")
	if parts.pap and #parts.pap > 0 then
		table.insert(output, '\n* Present active participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.pap, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.pmp and #parts.pmp > 0 then
		table.insert(output, '\n* Present middle participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.pmp, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.fap and #parts.fap > 0 then
		table.insert(output, '\n* Future active participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.fap, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.fmp and #parts.fmp > 0 then
		table.insert(output, '\n* Future middle participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.fmp, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.debug then table.insert(output, parts.debug) end
	if parts.cat then ti(output, table.concat(parts.cat)) end
	return table.concat(output)
end

function export.present_special_oldway(label, currentScript, parts, xo)
	output = {}
	parts.cat = parts.cat or {}
	if parts.presa or parts.presm then
		table.insert(output, export.present(label, 'Present', parts.presa, parts.presm, xo))
	end
	if parts.impfa or parts.impfm then
		table.insert(output, export.present(label, 'Imperfect', parts.impfa, parts.impfm, xo))
	end
	if parts.impra or parts.imprm then
		table.insert(output, export.present(label, 'Imperative', parts.impra, parts.imprm, xo))
	end
	if parts.optaa or parts.optam then
		table.insert(output, export.present(label, 'Optative', parts.optaa, parts.optam, xo))
	end
	if parts.futua or parts.futum then
		ti(output, export.present(label, 'Future', parts.futua, parts.futum, xo))
	end
	if parts.conda or parts.condm then
		ti(output, export.present(label, 'Conditional', parts.conda, parts.condm, xo))
		ti(parts.cat, "[[category:Pali verbs exhibiting conditional mood]]")
		if parts.conda and parts.conda.add1p then
			ti(parts.cat, "[[category:Pali verbs with manual 1p conditional]]")
		end
	end
	if parts.pap and #parts.pap > 0 then
		table.insert(output, '\n* Present active participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.pap, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.pmp and #parts.pmp > 0 then
		table.insert(output, '\n* Present middle participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.pmp, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.fap and #parts.fap > 0 then
		table.insert(output, '\n* Future active participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.fap, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.fmp and #parts.fmp > 0 then
		table.insert(output, '\n* Future middle participle: ')
		table.insert(output, m_noun.orJoin(currentScript, parts.fmp, xo))
		table.insert(output, ', which see for forms and usage')
	end
	if parts.debug then table.insert(output, parts.debug) end
	if parts.cat then ti(output, table.concat(parts.cat)) end
	return table.concat(output)
end

function export.present_future(label, currentScript, parts, xo)
	if true then -- Newer table format
	return export.present_conj(label, currentScript, parts, xo)
	else
	return export.present_special_oldway(label, currentScript, parts, xo)
	end
end

function export.present_special(label, currentScript, parts, xo)
	if true then -- Newer table format
	return export.present_conj(label, currentScript, parts, xo)
	else
	return export.present_special_oldway(label, currentScript, parts, xo)
	end
end

function export.special_forms(cite, args)
	local currentScript = lang:findBestScript(cite)
	local scriptCode = currentScript:getCode()
	local latinise = nil
	if scriptCode == "None" and args.sc then -- Resort to sc
		scriptCode = args["sc"]
		currentScript = m_scripts.getByCode(scriptCode, "No such script as ".. scriptCode .. ".")
		latinise = currentScript
		augment[scriptCode] = to_script("a", scriptCode)
	end
	args = export.parse(args)
	args.cite = arrcat_nodup({cite}, args.cite)
	local debug = '\n\nAt (1), args.cite = '..table.concat(args.cite, ' ')
	local patpa, patpm, basepa, basepm
	local parts = {}
	local option = {}
	option.aa = args.aa

-- Handle present tense.  This forms a base for others.
	local ending = export.detectEnding(cite, latinise)
--	mw.addWarning('ending='..ending)
	patpa = export.getSuffixes(scriptCode, ending, nil, args)
	basepa = cite
	local mset = nil
	local forBoth = false
	local aset = args.cite
	local voice = "both"
	if 'e' == asub(ending, -1) then -- No active inflection
		patpm = patpa
		patpa = nil
		basepm = cite
		aset = nil
		mset = args.cite
		voice = "midl"
	elseif args.midl and args.midl[1] then
		basepm = args.midl[1]
		mset = args.midl
		local midend = export.detectEnding(basepm, latinise)
		patpm = pattern[midend].mid and
		        export.getSuffixes(scriptCode, pattern[midend].mid, nil, args)
	elseif args.voice == 'act' then
		patpm = nil
		voice = "act"
	else
		patpm = pattern[ending].mid and
				export.getSuffixes(scriptCode, pattern[ending].mid, nil, args)
		basepm = cite
		forBoth = true
		mset = args.cite
	end
	local formsa = nil
	local formsm = nil
	if patpa then
		formsa = export.joinSuffixes(scriptCode, basepa, patpa, option)
		for k = 2, #aset do
			basepa = aset[k]
			patpa = export.getSuffixes(
				scriptCode, export.detectEnding(basepa, latinise), nil, args)
			local extras = export.joinSuffixes(scriptCode, basepa, patpa, option)
			for ip = 1, 6 do
				formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
			end
		end
	end
	if patpm or forBoth then
		if patpm then
			formsm = export.joinSuffixes(scriptCode, basepm, patpm, option)
		else
			formsm = {{}, {}, {}, {}, {}, {}}
		end
		for k = 2, #mset do
			basepm = mset[k]
			local kend = export.detectEnding(basepm, latinise)
			kend = pattern[kend].mid
			if kend then
				patpm = export.getSuffixes(scriptCode, kend, nil, args)
				local extras = export.joinSuffixes(scriptCode, basepm, patpm, option)
				for ip = 1, 6 do
					formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
				end
			end
		end
	end
	if args.voice == "act" or args.voice == "both" then
		-- handled already
	elseif args.voice == "none" then
		formsa = nil
		formsm = nil
		voice = "none"
	elseif args.voice then
		error('Argument voice has bad value "'..args.voice..'.')
	end
	formsa, formsm = export.modify(formsa, formsm, '', args)

	parts.presa = formsa
	parts.presm = formsm

-- Form imperative from present
    formsa = nil
    formsm = nil
    local imperativeVoice = args.impr_voice or voice
    if imperativeVoice == 'act' or imperativeVoice == 'both' then
    	local patma, kend
    	formsa = {{}, {}, {}, {}, {}, {}}
    	for k = 1, #aset do
    		kend = export.detectEnding(aset[k], latinise)
    		patma = export.getSuffixes(scriptCode, kend, 'imperative', args)
    		if patma then 
	   			local extras = export.joinSuffixes(scriptCode, aset[k], patma, option)
    			for ip = 1, 6 do
    				formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
    			end
    		end
    	end
    end
    if imperativeVoice == 'midl' or imperativeVoice == 'both' then
    	local patmm
    	formsm = {{}, {}, {}, {}, {}, {}}
    	local nstem, form3s
    	if forBoth then
    		nstem = #aset
    	else
    		nstem = #mset
    	end
    	for k = 1, nstem do
   			form3s = mset[k]
    		local kend = export.detectEnding(form3s, latinise)
    		if pattern[kend].mid then
    			kend = pattern[kend].mid
    		elseif kend == "eti" then
    			kend = nil
    		else
    			error('No middle endings for ending "'..kend..'".')
    			kend = nil
    		end
    		patmm = kend and export.getSuffixes(scriptCode, kend, 'imperative', args)
    		if patmm then 
	   			local extras = export.joinSuffixes(scriptCode, form3s, patmm, option)
    			for ip = 1, 6 do
    				formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
    			end
    		end
    	end
    end
    debug = debug..'\n\nAt (8), formsa='..tostring(formsa)
	formsa, formsm = export.modify(formsa, formsm, 'impr_', args)
	if formsa then
		debug = debug..'\n\nAt (9), formsa='..tostring(formsa)..
				' formsa.isnil='..tostring(formsa.isnil)
	else
		debug = debug..'\n\nAt (9), formsa='..tostring(formsa)
	end
	parts.impra = formsa
	parts.imprm = formsm

-- Form imperfect from present.  Existence defaults to false for both voices.
	local aug
	if args["impl"] == 'no' then
		aug = augment_alph[scriptCode] or augment[scriptCode]
	else
		aug = augment[scriptCode]
	end
--	error("impl="..tostring(args["impl"]).." aug="..aug)
	local impf = {}
	if (args.impf_voice ~= 'none') then
		if args.augment == 'without' then
			impf = aset or {}
		elseif args.augment == 'with_made' then
			if aset then
				for k = 1, #aset do
					impf[k] = aug .. aset[k]
				end
			end
		elseif args.augment == 'with_given' then
			impf = args.impf
		elseif args.augment == 'both_made' then
			if aset then
				for k = 1, #aset do
					impf[k] = aug .. aset[k]
				end
				for k = 1, #aset do
					impf[#aset+k] = aset[k]
				end
			end
		elseif args.augment == 'both_given' then
			impf = arrcat_nodup(args.impf or {}, aset or {})
		else
			error('Argument augment has bad value "'..args.augment..'".')
		end
	end
	if (args.impf_voice == 'both' or args.impf_voice == 'act')
--	and (voice == 'both' or voice == 'act')
	then
		formsa = {{}, {}, {}, {}, {}, {}}
		for k = 1, #impf do
			kend = export.detectEnding(impf[k], latinise)
			if kend == "ā" then kend = "ā2" end
			patma = export.getSuffixes(scriptCode, kend, 'imperfect', args)
			if true then -- Make false for debugging.
			elseif patma == nil or patma[5] == nil or patma[5][1] == nil then
				mw.addWarning('impf['..k..']='..impf[k]..' kend='..kend..' patma[5][1]=nil')
			else
				mw.addWarning('impf['..k..']='..impf[k]..' kend='..kend..' patma[5][1]='..patma[5][1])
			end
			if patma then
				local extras = export.joinSuffixes(scriptCode, impf[k], patma, option)
				for ip = 1, 6 do
					formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
				end
			end
		end
	elseif (args.impf_voice == 'none' or args.impf_voice == 'midl') then
		formsa = nil
	else
		error('Argument impf_voice has bad value "'..args.impf_voice..'".')
	end
	if (args.impf_voice == 'both' or args.impf_voice == 'midl')
--	and (voice == 'both' or voice == 'midl')
	then
		formsm = {{}, {}, {}, {}, {}, {}}
		if not forBoth then
			impf = {}
			if args.augment == 'without' then
				impf = mset
			elseif args.augment == 'with_made' then
				for k = 1, #mset do
					impf[k] = aug .. mset[k]
				end
			elseif args.augment == 'with_given' then
				impf = args.midl and args.midl[1] and args.impf_midl or args.midl
			elseif args.augment == 'both_made' then
				for k = 1, #mset do
					impf[k] = aug .. mset[k]
				end
				for k = 1, #mset do
					impf[#mset+k] = mset[k]
				end
			elseif args.augment == 'both_given' then
				impf = arrcat_nodup(
					args.midl and args.midl[1] and (args.impf_midl or {}) or
					args.impf or {},
					mset or {})
			else
				error('Argument augment has bad value "'..args.augment..'".')
			end
		end
		for k = 1, #impf do
			kend = export.detectEnding(impf[k], latinise)
			debug=debug..'\n\nAt (7) impf[k]='..impf[k]
			if kend == "ā" then kend = "ā2" end
			if forBoth then
	    		if pattern[kend].mid then
	    			kend = pattern[kend].mid
	    		elseif kend == "eti" then
	    			kend = nil
				else
	    			error('No middle endings for ending "'..kend..'".')
	    			kend = nil
	    		end
			end
			local pat = kend and export.getSuffixes(scriptCode, kend, 'imperfect', args)
			if not kend then
				-- nothng to do
			elseif pat then
				local extras = export.joinSuffixes(scriptCode, impf[k], pat, option)
				for ip = 1, 6 do
					formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
				end
			else
				mw.addWarning('No imperfect middle suffixes for '..impf[k]..
					' ('..kend..')')
			end
		end
	else
		formsm = nil
	end
	formsa, formsm = export.modify(formsa, formsm, 'impf_', args)
	parts.impfa = formsa
	parts.impfm = formsm
-- Form optative from present citation forms.
	args.opta_voice = args.opta_voice or voice
	debug = debug .. '\n\nAt (2), args.cite = '..table.concat(args.cite, ' ')
	local opta = args.cite
	debug = debug .. '\n\nAt (3), opta = '..table.concat(opta, ' ')
	if args.opta then
		debug = debug..'\n\nAt (4), args.opta='..table.concat(args.opta)..', args.opta_mod='..args.opta_mod
		if args.opta_mod == 'replace' then
			opta = args.opta
		elseif args.opta_mod == 'after' then
		opta = arrcat_nodup(opta, args.opta)
		elseif args.opta_mod == 'before' then
			opta = arrcat_nodup(args.opta, opta)
		elseif args.opta_mod == 'blank' then
			args.opta_voice = 'none'
		else
			error('Argument opta_mod has bad value "'..args.opta_mode..'".')
		end
	end
	debug = debug .. '\n\nAt (5), opta = '..table.concat(opta, ' ')..
			', args.opta_voice='..args.opta_voice
	if args.opta_voice == 'both' or args.opta_voice == 'act' then
		formsa = {{}, {}, {}, {}, {}, {}}
		local kend, patma
		for k = 1, #opta do
			kend = export.detectEnding(opta[k], latinise)
			patma = export.getSuffixes(scriptCode, kend, 'optative_active', args)
			if true then -- Make false for debugging.
			elseif patma == nil or patma[5] == nil or patma[5][1] == nil then
				mw.addWarning('opta['..k..']='..opta[k]..' kend='..kend..' patma[5][1]=nil')
			else
				mw.addWarning('opta['..k..']='..opta[k]..' kend='..kend..' patma[5][1]='..patma[5][1])
			end
			if patma then
				local extras = export.joinSuffixes(scriptCode, opta[k], patma, option)
				for ip = 1, 6 do
					formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
				end
			end
		end
	elseif args.opta_voice == 'midl' or args.opta_voice == 'none' then
		formsa = nil
	else
		error('Argument opta_voice has bad value "'..args.opta_voice..'".')
	end
	if #args.opta_midl > 0 then opta = args.opta_midl end
	formsm = nil

	if args.opta_voice == 'both' or args.opta_voice == 'midl' then
		formsm = {{}, {}, {}, {}, {}, {}}
		local kend, patma
		debug=debug..'\n\nAt (6), opta='..table.concat(opta)
		for k = 1, #opta do
			kend = export.detectEnding(opta[k], latinise)
			patma = export.getSuffixes(scriptCode, kend, 'optative_middle', args)
			if patma then
				local extras = export.joinSuffixes(scriptCode, opta[k], patma, option)
				for ip = 1, 6 do
					formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
				end
			else
				debug = debug.."\n\nNo "..kend..".optative_middle"
			end
		end
	else
		formsm = nil
	end

	formsa, formsm = export.modify(formsa, formsm, 'opta_', args)
		parts.optaa = formsa
	parts.optam = formsm

-- Generate aorist forms - currently only from aorist specific parameters.
	formsa = nil
	formsm = nil
	if args.aori and #args.aori > 0 or args.aor2 and #args.aor2 > 0 then
		local flags = {}
		flags.aasi = args.aasi
		formsa = { {}, {}, {}, {}, {}, {} }
		local aori = {} -- List of *aorist* 3s.  Cannnot tell -ati from -i!
		for k = 1, #(args.aor2 or {}) do
			aori[k] = aug .. args.aor2[k]
		end
		aori = arrcat_nodup(aori, args.aor2 or {})
		aori = arrcat_nodup(aori, args.aori)
		for k = 1, #aori do
			local kend = export.detectAoristEnding(aori[k], latinise, flags)
			if kend == "āsi" then
				parts.cat = parts.cat or {}
				ti(parts.cat, "[[category:Pali verbs with aorist in -āsi/-a]]")
			end
			local patma = export.getSuffixes(scriptCode, kend, 'aorist', args)
			if patma then
				local extras = export.joinSuffixes(scriptCode, aori[k], patma, option)
				for ip = 1, 6 do
					formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
				end
			else
				debug = debug.."\n\nNo "..kend..".aorist"
			end
		end
		if #formsa[1] == 0 and #formsa[2] == 0 and #formsa[3] == 0 and
		   #formsa[4] == 0 and #formsa[5] == 0 and #formsa[6] == 0 then 	
			formsa = nil
		end
	end
	parts.aoria, parts.aorim = export.modify(formsa, formsm, 'aori_', args)

-- Generate perfect tense by write in
	formsa = nil
	formsm = nil
	parts.perfa, parts.perfm = export.modify(formsa, formsm, 'perf_', args)
	
--	List present participles.
	local pap, pmp = {}, {}
	args.part_voice = args.part_voice or voice
	if args.part_voice == 'both' or args.part_voice == 'act' then
		if voice == 'both' or voice == 'act' then
			for k = 1, aset and #aset or 0 do
				local kend = export.detectEnding(aset[k], latinise)
				local pat = kend and export.getSuffixes(scriptCode, kend, "participle", args)
				if pat then
					local extra = export.joinSuffix(scriptCode, aset[k], pat[1])
					pap = arrcat_nodup(pap, extra)
				end
			end
		end
		if args.pap_mod == 'blank' then
			pap = {}
		elseif args.pap_mod == 'after' then
			pap = arrcat_nodup(pap, args.pap)
		elseif args.pap_mod == 'before' then
			pap = arrcat_nodup(args.pap, pap)
		elseif args.pap_mod == 'replace' then
			pap = args.pap
		else
			error('Argument pap_mod has invalid value '..args.pap_mod)
		end
		parts.pap = pap
	elseif args.part_voice == 'midl' or args.part_voice == 'none' then
		-- permissible
	else
		error('Argument part_voice has invalid value '..args.part_voice)
	end
	if args.part_voice == 'both' or args.part_voice == 'midl' then
		local list
		local from_active = forBoth or voice == 'act'
		if from_active then
			list = aset
		else
			list = mset
		end
		for k = 1, list and #list or 0 do
			local kend = export.detectEnding(list[k], latinise)
			if from_active then kend = pattern[kend].mid end
			local pat = kend and export.getSuffixes(scriptCode, kend, "participle", args)
			if pat then
				local extra = export.joinSuffix(scriptCode, list[k], pat[1])
				pmp = arrcat_nodup(pmp, extra)
			end
		end
		if args.pmp_mod == 'blank' then
			pmp = {}
		elseif args.pmp_mod == 'after' then
			pmp = arrcat_nodup(pmp, args.pmp)
		elseif args.pmp_mod == 'before' then
			pmp = arrcat_nodup(args.pmp, pmp)
		elseif args.pmp_mod == 'replace' then
			pmp = args.pmp
		else
			error('Argument pmp_mod has invalid value '..args.pmp_mod)
		end
		parts.pmp = pmp
	end
	
	-- parts.debug = debug..'\n\n'..debug_outer
	return parts
end

function export.show(frame)
	local args = mw.clone(frame:getParent().args)
	local cite = args[1] or args["cite"] or mw.title.getCurrentTitle().text
	local currentScript = lang:findBestScript(cite)
	local xo = {impl = args.impl, y = args.y, showtr = args.showtr, subst = args.subst}
	local parts = export.special_forms(cite, args)
	parts.cat = parts.cat or {}
	if args.futu then
		local fuparts = export.future_forms(args.futu, args, true)
		fuparts.cat = fuparts.cat or {}
		for k, v in pairs(fuparts) do
			if k == 'cat' then
				parts.cat = arrcat_nodup(parts.cat, fuparts.cat)
			else
				parts[k] = fuparts[k]
			end
		end
--		dump_table('parts', parts)
	end
	local label = args.label or cite
	return export.present_special(label, currentScript, parts, xo)
end

function export.detectFutureEnding(cite3s, latinise)
	if latinise then
		local cfin = cite3s
		cite3s = (lang:transliterate(cfin, latinise))
	end
	local len, wordEnd
	local ending
	for len = 6, 1, -1 do
		wordEnd = sub(cite3s, -len)
		ending = future_endings[wordEnd]
		if not ending then
			-- Try again
		else
-- debug_outer = debug_outer..'\n'..cite3s..'='..ending..'\n'
			return ending
		end
	end
-- debug_outer = debug_outer..'\n'..cite3s..'='.."a"..'\n'
	return "a"
end

local future_pattern = {
	["ati"] = {
		{"⌫⌫⌫āmi"}, {"⌫⌫⌫āma"},   {"⌫⌫si"}, {"⌫ha"},    {""}, {"⌫⌫nti"},
		mid = "ate",
		conditional = {
			{"⌫⌫ṃ"},  -- Collins also has 1s {"⌫⌫⌫āmi"}
							{},					 -- Kaccayano, Aggavamsa and Duroiselle have -amhā
												 -- while Warder and Collins have -āma.
--	                        {"⌫⌫mhā", "⌫⌫⌫āma"}, -- We need to gather evidence and then decide.  
			{"⌫⌫", "⌫⌫⌫i"},	{"⌫ha"},             -- Duroiselle also has 2s "⌫⌫⌫e", which Collins says is middle.
			{"⌫⌫⌫ā", "⌫⌫"},	{"⌫⌫ṃsu"},           -- Duroiselle also has 3s as in future.
		},
		participle = {{"⌫⌫nt"}},
	},
	["ate"] = { -- Also services "ati"
		{"⌫⌫ṃ"}, {"⌫⌫⌫āmhe"},  {"⌫⌫se"}, {"⌫⌫vhe"},  {"⌫e"}, {"⌫⌫nte"},
		conditional = {
			{"⌫⌫ṃ"},	        {"⌫⌫⌫āmhase"},
			{"⌫⌫⌫e", "⌫⌫se"},	{"⌫⌫vhe"}, -- Duroiselle has first 2s as active.
			{"⌫ha"},	        {"⌫⌫⌫iṃsu"},
		},
		participle = {{"⌫⌫māna", "⌫⌫⌫āna"}},
		mid = "ate"
	},
	["ā"] = {
		conditional = {
			{"⌫aṃ"},  -- Collins also has 1s {"mi"}
	        {}, --          {"⌫amhā", "ma"},  -- See above for controversy.  
			{"⌫a", "⌫i"},	{"⌫atha"},        -- Duroiselle also has 2s "⌫⌫⌫e", which Collins says is middle.
			{"", "⌫a"},  	{"⌫aṃsu"},        -- Duroiselle also has 3s as in future.
		},
        mid = "atha_for_aa",
	},
	["a"] = {
		conditional = {
			{"ṃ"},  -- Collins also has 1s {"āmi"}
	        {}, --         {"mhā", "⌫āma"}, -- See above for controversy.  
			{"", "⌫i"},	    {"tha"},        -- Duroiselle also has 2s "⌫⌫⌫e", which Collins says is middle.
			{"⌫ā", ""},  	{"ṃsu"},        -- Duroiselle also has 3s as in future.
		},
        mid = "atha_for_a",
	},
	["atha"] = {
		conditional = {
			{"⌫⌫⌫ṃ"},	        {"⌫⌫⌫⌫āmhase"},
			{"⌫⌫⌫⌫e", "⌫⌫⌫se"},	{"⌫⌫⌫vhe"}, -- Duroiselle has first 2s as active.
			{""},	            {"⌫⌫⌫⌫iṃsu"},
		},
		mid = "atha"
	},
    ["atha_for_a"] = {
       	conditional = {
			{"ṃ"},	        {"⌫āmhase"},
			{"⌫e", "se"},	{"vhe"}, -- Duroiselle has first 2s as active.
			{"tha"},	    {"⌫iṃsu"},
		},
		mid = "atha_for_a"
	},
	atha_for_aa = {
		conditional = {
			{"⌫aṃ"},	        {"mhase"},
			{"⌫e", "⌫ase"},	{"⌫avhe"}, -- Duroiselle has first 2s as active.
			{"⌫atha"},	    {"⌫iṃsu"},
		},
		mid = "atha_for_aa"
	},
	iti = {
		{"⌫⌫⌫āmi"}, {"⌫⌫⌫āma"}, {"⌫⌫si"}, {"⌫ha"}, {""}, {"⌫⌫nti"},
		mid = "void"
	},
	heti = {
		{"⌫⌫mi"}, {"⌫⌫ma"}, {"⌫⌫si"}, {"⌫ha"}, {""}, {"⌫⌫nti"},
		mid = "void"
	},
	["kāhiti"] = {
		{"⌫⌫⌫āmi"}, {"⌫⌫⌫āma"}, {"⌫⌫si"}, {}, {""}, {"⌫⌫nti"},
		mid = "void"
	},
	void = { {}, {}, {}, {}, {}, {}, mid = "void"}
}

-- convert Latin script inflections to another script
-- ⒹⓉ

function export.getFutureSuffixes(sc, ending, tense, option)
	local impl = option and option.impl or 'yes'
	local yopt = option and option.y
	local pat
	if not future_pattern[ending] then
		error('No verb pattern for ending "'..ending..'".')
	end
	if tense then
		pat = future_pattern[ending][tense]
	else
		pat = future_pattern[ending]
	end
	if 'Latn' == sc then
		return pat
	elseif not pat then
		return nil
	elseif pat[sc] then
		return pat[sc]
	else
		local olen = {Mymr = 2, Lana = 2} -- Assume NFC
		local attha_len = {Latn = 5, Sinh = 4}
		eyya_len = {Mymr = 3, Sinh = 5}
		local alen
		if impl == 'no' and (sc == 'Thai' or sc == 'Laoo') then
			alen = 1
		else
			alen = 0
		end
		local way = {
			["ati"]  = {pseudostem = "kati", ndel = 2 + alen},
			["heti"]  = {pseudostem = "heti", ndel = 3}, -- Tricky code
			["iti"]  = {pseudostem = "kiti", ndel = 3},
			["ate"]  = {pseudostem = "kate", ndel = 2 + alen},
			["ā"]    = {pseudostem = "kā", ndel = 1},
			["a"]    = {pseudostem = "ka", ndel = alen},
			["atha"] = {pseudostem = "katha", ndel = 1 + 2 * alen},
			atha_for_a = {pseudostem = "ka", ndel = alen},
			atha_for_aa = {pseudostem = "kā", ndel = 1},
			["kāhiti"]  = {pseudostem = "kāhiti", ndel = 5}, --Tricky code
			void = {pseudostem = "ku", ndel = 1}, -- rhubarb
		}
		way = way[ending]
		if way then
			pat[sc] = convert_suffixes(way.pseudostem, way.ndel, pat, sc,
									   impl, yopt)
		else
			error('Cannot convert derivations from suffix '..ending..' for '..sc)
		end
		return pat[sc]
	end
end

function export.future_forms(cite, args, subset)
	local currentScript = lang:findBestScript(cite)
	local scriptCode = currentScript:getCode()
	local latinise = nil
	if scriptCode == "None" and args.sc then -- Resort to sc
		scriptCode = args["sc"]
		currentScript = m_scripts.getByCode(scriptCode, "No such script as ".. scriptCode .. ".")
		latinise = currentScript
		augment[scriptCode] = to_script("a", scriptCode)
	end
	if subset then args = export.parse(args) end
	args = export.futureParse(args, subset)
	args.cite = arrcat_nodup({cite}, args.cite)
	local debug = '\n\nAt (1), args.cite = '..table.concat(args.cite, ' ')
	local patpa, patpm, basepa, basepm
	local parts = {}
	local option = {}
	option.aa = args.aa

-- Handle future tense.  This forms a base for others.
	local ending = export.detectFutureEnding(cite, latinise)
	patpa = export.getFutureSuffixes(scriptCode, ending, nil, args)
	basepa = cite
	local mset = nil
	local forBoth = false
	local aset = args.cite
	local voice = "both"
	if 'e' == asub(ending, -1) then -- No active inflection
		patpm = patpa
		patpa = nil
		basepm = cite
		aset = nil
		mset = args.cite
		voice = "midl"
	elseif args.midl and args.midl[1] then
		basepm = args.midl[1]
		mset = args.midl
		local midend = export.detectFutureEnding(basepm, latinise)
		patpm = future_pattern[midend].mid and
				export.getFutureSuffixes(scriptCode, future_pattern[midend].mid, nil, args)
	elseif args.voice == 'act' then
		patpm = nil
		voice = "act"
	else
		patpm = future_pattern[ending].mid and
				export.getFutureSuffixes(scriptCode, future_pattern[ending].mid, nil, args)
		basepm = cite
		forBoth = true
		mset = args.cite
	end
	local formsa = nil
	local formsm = nil
	if patpa then
		formsa = export.joinSuffixes(scriptCode, basepa, patpa, option)
		for k = 2, #aset do
			basepa = aset[k]
			patpa = export.getFutureSuffixes(
				scriptCode, export.detectFutureEnding(basepa, latinise), nil, args)
			local extras = export.joinSuffixes(scriptCode, basepa, patpa, option)
			for ip = 1, 6 do
				formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
			end
		end
	end
	if patpm or forBoth then
		if patpm then
			formsm = export.joinSuffixes(scriptCode, basepm, patpm, option)
		else
			formsm = {{}, {}, {}, {}, {}, {}}
		end
		for k = 2, #mset do
			basepm = mset[k]
			local kend = export.detectFutureEnding(basepm, latinise)
			kend = future_pattern[kend].mid
			if kend then
				patpm = export.getFutureSuffixes(scriptCode, kend, nil, args)
				local extras = export.joinSuffixes(scriptCode, basepm, patpm, option)
				for ip = 1, 6 do
					formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
				end
			end
		end
	end
	if args.voice == "act" or args.voice == "both" then
		-- handled already
	elseif args.voice == "none" then
		formsa = nil
		formsm = nil
		voice = "none"
	elseif args.voice then
		error('Argument voice has bad value "'..args.voice..'.')
	end
	formsa, formsm = export.modify(formsa, formsm, '', args)

	parts.futua = formsa
	parts.futum = formsm

-- Form conditional from future.
	local aug
	if args["impl"] == 'no' then
		aug = augment_alph[scriptCode] or augment[scriptCode]
	else
		aug = augment[scriptCode]
	end
--	error("impl="..tostring(args["impl"]).." aug="..aug)
--	error("impl="..tostring(args["impl"]).." aug="..aug.." aset="..tostring(aset[1])..
--		" args.augment="..args.augment)
	args.cond_voice = args.cond_voice or voice
--	error("args.cond_voice="..tostring(args.cond_voice))
	local cond = {}
	if (args.cond_voice ~= 'none') then
		if args.augment == 'without' then
			if aset then cond = aset end
		elseif args.augment == 'with_made' then
			if aset then
				for k = 1, #aset do
					cond[k] = aug .. aset[k]
				end
			end
		elseif args.augment == 'with_given' then
			cond = args.cond
		elseif args.augment == 'both_made' then
			if aset then
				for k = 1, #aset do
					cond[k] = aug .. aset[k]
				end
				for k = 1, #aset do
					cond[#aset+k] = aset[k]
				end
			end
		elseif args.augment == 'both_given' then
			cond = arrcat_nodup(args.cond or {}, aset or {})
		else
			error('Argument augment has bad value "'..args.augment..'".')
		end
	end
--	mw.addWarning('args.cite[1]='..args.cite[1]..', args.cond_voice='..args.cond_voice)
	if (args.cond_voice == 'both' or args.cond_voice == 'act')
	then
		formsa = {{}, {}, {}, {}, {}, {}}
		for k = 1, #cond do
			local kend = export.detectFutureEnding(cond[k], latinise)
			local patma = export.getFutureSuffixes(scriptCode, kend, 'conditional', args)
			if patma then
				local extras = export.joinSuffixes(scriptCode, cond[k], patma, option)
				for ip = 1, 6 do
					formsa[ip] = arrcat_nodup(formsa[ip], extras[ip])
				end
			end
		end
	elseif (args.cond_voice == 'none' or args.cond_voice == 'midl') then
		formsa = nil
	else
		error('Argument cond_voice has bad value "'..args.cond_voice..'".')
	end
	if (args.cond_voice == 'both' or args.cond_voice == 'midl')
	then
		formsm = {{}, {}, {}, {}, {}, {}}
		if not forBoth then
			cond = {}
			if args.augment == 'without' then
				cond = mset
			elseif args.augment == 'with_made' then
				for k = 1, #mset do
					cond[k] = aug .. mset[k]
				end
			elseif args.augment == 'with_given' then
				cond = args.midl and args.midl[1] and args.cond_midl or args.cond
			elseif args.augment == 'both_made' then
				for k = 1, #mset do
					cond[k] = aug .. mset[k]
				end
				for k = 1, #mset do
					cond[#mset+k] = mset[k]
				end
			elseif args.augment == 'both_given' then
				cond = arrcat_nodup(
					args.midl and args.midl[1] and (args.cond_midl or {}) or
					args.cond or {},
					mset or {})
			else
				error('Argument augment has bad value "'..args.augment..'".')
			end
		end
		for k = 1, #cond do
			local kend = export.detectFutureEnding(cond[k], latinise)
			debug=debug..'\n\nAt (7) cond[k]='..cond[k]
			if forBoth then
	    		if future_pattern[kend].mid then
	    			kend = future_pattern[kend].mid
	    		elseif kend == "eti" then
	    			kend = nil
				else
					mw.addWarning('No middle endings for ending "'..kend..'".')
	    			kend = nil
	    		end
			end
			local pat = kend and export.getFutureSuffixes(scriptCode, kend, 'conditional', args)
			if not kend then
				-- nothing to do
			elseif pat then
				local extras = export.joinSuffixes(scriptCode, cond[k], pat, option)
				for ip = 1, 6 do
					formsm[ip] = arrcat_nodup(formsm[ip], extras[ip])
				end
			else
				mw.addWarning('No conditional middle suffixes for '..cond[k]..
					' ('..kend..')')
			end
		end
	else
		formsm = nil
	end
	formsa, formsm = export.modify(formsa, formsm, 'cond_', args)
	parts.conda = formsa
	parts.condm = formsm

--	List present participles.
	local pap, pmp = {}, {}
	args.part_voice = args.part_voice or voice
	if args.part_voice == 'both' or args.part_voice == 'act' then
		if voice == 'both' or voice == 'act' then
			for k = 1, aset and #aset or 0 do
				local kend = export.detectFutureEnding(aset[k], latinise)
				local pat = kend and export.getFutureSuffixes(scriptCode, kend, "participle", args)
				if pat then
					local extra = export.joinSuffix(scriptCode, aset[k], pat[1])
					pap = arrcat_nodup(pap, extra)
				end
			end
		end
		if args.fap_mod == 'blank' then
			pap = {}
		elseif args.fap_mod == 'after' then
			pap = arrcat_nodup(pap, args.fap)
		elseif args.fap_mod == 'before' then
			pap = arrcat_nodup(args.fap, pap)
		elseif args.fap_mod == 'replace' then
			pap = args.fap
		else
			error('Argument fap_mod has invalid value '..args.fap_mod)
		end
		parts.fap = pap
	elseif args.part_voice == 'midl' or args.part_voice == 'none' then
		-- permissible
	else
		error('Argument part_voice has invalid value '..tostring(args.part_voice))
	end
	if args.part_voice == 'both' or args.part_voice == 'midl' then
		local list
		local from_active = forBoth or voice == 'act'
		if from_active then
			list = aset
		else
			list = mset
		end
		for k = 1, list and #list or 0 do
			local kend = export.detectFutureEnding(list[k], latinise)
			if from_active then kend = future_pattern[kend].mid end
			local pat = kend and export.getFutureSuffixes(scriptCode, kend, "participle", args)
			if pat then
--				if 2 == k then
--					error("k="..k.."list[k]="..list[k])
--				end
				local extra = export.joinSuffix(scriptCode, list[k], pat[1])
				pmp = arrcat_nodup(pmp, extra)
			end
		end
		if args.fmp_mod == 'blank' then
			pmp = {}
		elseif args.fmp_mod == 'after' then
			pmp = arrcat_nodup(pmp, args.fmp)
		elseif args.fmp_mod == 'before' then
			pmp = arrcat_nodup(args.fmp, pmp)
		elseif args.fmp_mod == 'replace' then
			pmp = args.fmp
		else
			error('Argument fmp_mod has invalid value '..args.fmp_mod)
		end
		parts.fmp = pmp
	end

	-- parts.debug = debug..'\n\n'..debug_outer
	return parts
end

function export.show_future(frame)
	local args = mw.clone(frame:getParent().args)
	local cite = args[1] or args["cite"] -- or mw.title.getCurrentTitle().text
	local xo = {impl = args.impl, y = args.y, showtr = args.showtr, subst = args.subst}
	local parts = export.future_forms(cite, args, false)
	local label = args.label or cite
	local currentScript = lang:findBestScript(cite)
	return export.present_future(label, currentScript, parts, xo)
end

return export