local m_links = require("Module:links")
local m_strutils = require("Module:string utilities")
local ar_utilities = require("Module:ar-utilities")

local export = {}

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

local u = mw.ustring.char

-- hamza variants
local HAMZA            = u(0x0621) -- hamza on the line (stand-alone hamza) = ء
local HAMZA_ON_ALIF    = u(0x0623)
local HAMZA_ON_W       = u(0x0624)
local HAMZA_UNDER_ALIF = u(0x0625)
local HAMZA_ON_Y       = u(0x0626)
local HAMZA_ANY = "[" .. HAMZA .. HAMZA_ON_ALIF .. HAMZA_UNDER_ALIF .. HAMZA_ON_W .. HAMZA_ON_Y .. "]"
local HAMZA_PH = u(0xFFF0) -- hamza placeholder

-- diacritics
local A  = u(0x064E) -- fatḥa
local AN = u(0x064B) -- fatḥatān (fatḥa tanwīn)
local U  = u(0x064F) -- ḍamma
local UN = u(0x064C) -- ḍammatān (ḍamma tanwīn)
local I  = u(0x0650) -- kasra
local IN = u(0x064D) -- kasratān (kasra tanwīn)
local SK = u(0x0652) -- sukūn = no vowel
local SH = u(0x0651) -- šadda = gemination of consonants
local DAGGER_ALIF = u(0x0670)
local DIACRITIC_ANY_BUT_SH = "[" .. A .. I .. U .. AN .. IN .. UN .. SK .. DAGGER_ALIF .. "]"

-- various letters and signs
local ALIF   = u(0x0627) -- ʾalif = ا
local AMAQ   = u(0x0649) -- ʾalif maqṣūra = ى
local AMAD   = u(0x0622) -- ʾalif madda = آ
local TAM    = u(0x0629) -- tāʾ marbūṭa = ة
local T      = u(0x062A) -- tāʾ = ت
local HYPHEN = u(0x0640)
local N      = u(0x0646) -- nūn = ن
local W      = u(0x0648) -- wāw = و
local Y      = u(0x064A) -- yā = ي
local LRM    = u(0x200e) -- left-to-right mark

local function ine(x) -- If Not Empty
    if x == "" then
        return nil
    else
        return x
    end
end

function canon_shadda_hamza(word)
	if not word then
		return nil
	end
	-- shadda+short-vowel (including tanwīn vowels, i.e. -an -in -un) gets
	-- replaced with short-vowel+shadda during NFC normalisation, which
	-- MediaWiki does for all Unicode strings; however, it makes various
	-- processes inconvenient, so undo it.
	word = mw.ustring.gsub(word, "(" .. DIACRITIC_ANY_BUT_SH .. ")" .. SH, SH .. "%1")
	-- replace hamza with hamza placeholder; it will be fixed later by hamza_seat()
	word = mw.ustring.gsub(word, HAMZA_ANY .. "([" .. A .. I .. U .. "])$", HAMZA_PH .. "%1")
	return word
end

-- Supply the appropriate hamza seat(s) for a placeholder hamza.
function hamza_seat(word)
	-- FIXME! Allow multiple possibilities in inflection tables
	if mw.ustring.find(word, HAMZA_PH) then -- optimization to avoid many regexp substs
		return ar_utilities.process_hamza(word)[1]
	end
	return word
end

local forms = {
    ["base"] = true,
    ["1s"] = true,
    ["2ms"] = true,
    ["2fs"] = true,
    ["3ms"] = true,
    ["3fs"] = true,
    ["2d"] = true,
    ["3d"] = true,
    ["1p"] = true,
    ["2mp"] = true,
    ["2fp"] = true,
    ["3mp"] = true,
    ["3fp"] = true,
}

local function linkify(args)
    for form in pairs(forms) do
        if args[form] == "-" or not ine(args[form]) then
            args[form] = "—"
        else
        	args[form] = hamza_seat(args[form])
            args[form] = m_links.full_link({lang = lang, term = args[form], tr = ine(args[form .. "tr"]) or nil})
        end
    end
end

local attach_1s = {}

attach_1s["i"] = function(args)
    if not ine(args["1s"]) then
    	-- if ends with -y, add -ya; else, truncate final short vowel and add -ī
        if mw.ustring.find(args["stem"], "يْ?$") then
            args["1s"] = mw.ustring.gsub(args["stem"], SK .. "?$", SH .. A, 1)
            args["1str"] = args["stemtr"] and (args["stemtr"] .. "ya")
            args["1str"] = args["stemtr"] and (mw.ustring.gsub(args["1str"], "īya$", "iyya", 1))
        else
            args["1s"] = mw.ustring.gsub(args["stem"], "[" .. A .. I .. U .. SK .. "]?$", I .. Y, 1)
            args["1str"] = args["stemtr"] and (mw.ustring.gsub(args["stemtr"], "[aiu]?$", "ī", 1))
        end
    end
end

attach_1s["ni"] = function(args)
    if not ine(args["1s"]) then
    	-- if already ends in -n, double the -n then add -ī; else, add -nī
        if mw.ustring.find(args["stem"], "نْ?$") then
            args["1s"] = mw.ustring.gsub(args["stem"], SK .. "?$", SH .. I .. Y, 1)
        else
            args["1s"] = args["stem"] .. "نِي"
        end
        args["1str"] = args["stemtr"] and (args["stemtr"] .. "nī")
    end
end

local function attach_1p(args)
    if not ine(args["1p"]) then
    	-- if already ends in -n, double the -n then add -ā; else, add -nā
        if mw.ustring.find(args["stem"], "نْ?$") then
            args["1p"] = mw.ustring.gsub(args["stem"], SK .. "?$", SH .. A .. ALIF, 1)
        else
            args["1p"] = args["stem"] .. "نَا"
        end
        args["1ptr"] = args["stemtr"] and (args["stemtr"] .. "nā")
    end
end

local function attach_2(args)
    local stem2 = args["stem"] .. "ك"
    local stem2tr = args["stemtr"] and (args["stemtr"] .. "k")
    if not ine(args["2ms"]) then
        args["2ms"] = stem2 .. A
        args["2mstr"] = args["stemtr"] and (stem2tr .. "a")
    end
    if not ine(args["2fs"]) then
        args["2fs"] = stem2 .. I
        args["2fstr"] = args["stemtr"] and (stem2tr .. "i")
    end
    if not ine(args["2d"]) then
        args["2d"] = stem2 .. "ُمَا"
        args["2dtr"] = args["stemtr"] and (stem2tr .. "umā")
    end
    if not ine(args["2mp"]) then
        args["2mp"] = stem2 .. "ُمْ"
        args["2mptr"] = args["stemtr"] and (stem2tr .. "um")
    end
    if not ine(args["2fp"]) then
        args["2fp"] = stem2 .. "ُنَّ"
        args["2fptr"] = args["stemtr"] and (stem2tr .. "unna")
    end
end

local function attach_3(args)
    local stem3 = nil
    local stem3tr = nil
    if mw.ustring.find(args["stem"], "[" .. Y .. I .. "]" .. SK .. "?$") then
        stem3 = args["stem"] .. "هِ"
        stem3tr = args["stemtr"] and (args["stemtr"] .. "hi")
    else
        stem3 = args["stem"] .. "هُ"
        stem3tr = args["stemtr"] and (args["stemtr"] .. "hu")
    end
    if not ine(args["3ms"]) then
        args["3ms"] = stem3
        args["3mstr"] = args["stemtr"] and (stem3tr)
    end
    if not ine(args["3d"]) then
        args["3d"] = stem3 .. "مَا"
        args["3dtr"] = args["stemtr"] and (stem3tr .. "mā")
    end
    if not ine(args["3mp"]) then
        args["3mp"] = stem3 .. "مْ"
        args["3mptr"] = args["stemtr"] and (stem3tr .. "m")
    end
    if not ine(args["3fp"]) then
        args["3fp"] = stem3 .. "نَّ"
        args["3fptr"] = args["stemtr"] and (stem3tr .. "nna")
    end
end

local function attach_3fs(args)
    if not ine(args["3fs"]) then
        args["3fs"] = args["stem"] .. "هَا"
        args["3fstr"] = args["stemtr"] and (args["stemtr"] .. "hā")
    end
end

local function base_to_stem(args)
	-- replace final -ā with -ay (e.g. إِلَى becomes إِلَيْهِ)
    args["stem"] = mw.ustring.gsub(args["base"], "[" .. A .. DAGGER_ALIF .. "]*" .. AMAQ .. DAGGER_ALIF .. "?$", A .. Y .. SK, 1)
    -- replace final tāʾ marbūṭa with regular t (e.g. حَالَةَ becomes حَالَتَهُ)
    args["stem"] = mw.ustring.gsub(args["stem"], TAM .. "([" .. A .. I .. U .. "])$", T .. "%1", 1) 
    if not ine(args["stemtr"]) then
        if args["stem"] ~= args["base"] then
            args["stemtr"] = args["basetr"] and mw.ustring.gsub(args["basetr"], "ā$", "ay", 1)
        else
            args["stemtr"] = args["basetr"]
        end
    end
    args["stem"] = mw.ustring.gsub(args["stem"], "(.)" .. HYPHEN .. "$", "%1", 1)
    args["stemtr"] = args["stemtr"] and (mw.ustring.gsub(args["stemtr"], "(.)%-$", "%1", 1))
end

local template = [===[
<div class="NavFrame">
<div class="NavHead" align=left>&nbsp; &nbsp; {heading}</div>
<div class="NavContent" style="text-align: center;">
{\op}| style="margin:1em; margin-left: 0; border: 1px solid #AAAAAA; border-collapse:collapse; text-align: center;" cellpadding="4" rules="all"
|-
! style="background:#E2E4C0;" colspan="5" |基本形
|<big>{base}</big>
|-
! style="background:#E2E4C0;" rowspan="2"| 帶人稱形
! style="background:#C0CFE4;" colspan="2" | 單數
! style="background:#C0CFE4;" | 雙數
! style="background:#C0CFE4;" colspan="2" | 複數
|-
! style="background:#C0CFE4;" | 陽性
! style="background:#C0CFE4;" | 陰性
! style="background:#C0CFE4;" | 通性
! style="background:#C0CFE4;" | 陽性
! style="background:#C0CFE4;" | 陰性
|-
! style="background:#C0CFE4; text-align: right;"| 第一人稱
| colspan="2" | {1s}
|
| colspan="2" | {1p}
|-
! style="background:#C0CFE4; text-align: right;"| 第二人稱
| {2ms}
| {2fs}
| {2d}
| {2mp}
| {2fp}
|-
! style="background:#C0CFE4; text-align: right;"| 第三人稱
| {3ms}
| {3fs}
| {3d}
| {3mp}
| {3fp}
|{\cl}</div></div>]===]

local function make_table(args)
    return m_strutils.format(template, args)
end

function export.inflect(frame)
    local args = frame:getParent().args or {}
    local ni = frame.args[1]
    if args["ni"] and args["ni"] ~= "" and args["ni"] ~= "-" then
        ni = "ni"
    elseif ni ~= "ni" then
        ni = "i"
    end
    PAGENAME = mw.title.getCurrentTitle().text
    SUBPAGENAME = mw.title.getCurrentTitle().subpageText
    NAMESPACE = mw.title.getCurrentTitle().nsText

    args["base"] = canon_shadda_hamza(ine(args["base"]) or ine(args[1]) or SUBPAGENAME)
    args["basetr"] = ine(args["basetr"]) or ine(args[2])
    if NAMESPACE == "Template" and not args["base"] then
        args["base"] = "ـ"
        args["basetr"] = "-"
    end
    args["stem"] = canon_shadda_hamza(ine(args["stem"]) or ine(args[3]))
    if not args["stem"] then
        base_to_stem(args)
    end
    args["stemtr"] = ine(args["stemtr"]) or ine(args[4]) or args["stem"] == args["base"] and args["basetr"] or nil

    attach_1s[ni](args)
    attach_1p(args)
    attach_2(args)
    attach_3(args)
    attach_3fs(args)

    linkify(args)

    args["heading"] = ine(args["heading"]) or "變格形"

    return make_table(args)
end

return export