Module:MiniEvent

--[[ LUA script to get Mini Event information

getME: get Mini Event based on input parameters Parameters: 1 = name    : Mini Event name 2 = number  : Event Number (multiple Numbers can be specified                    separated with ",") 3 = options : One or more Output Options Some Options are mutually exclusive --- Fixed Formats The priority is in the below order Only the first ME number will be displayed C = Fixed Text ME Count Till Today E = Event Page Format (Table Row: Columns= Icon, Short Name, Dates w/o year) T = Timeline Page Format (Table Columns: Icon + Short Name #, Dates with year) (Parameter 4 can be  for italic and ' for bold') K = toKen Page Format (Table Row: Columns= Icon, Full Name Sub Text) (sub text in parameter 4) L = Latest Info Main Page Format (Short Name with small Date) --- Customizable Display Format Options F = Full ME Name 9 = Include ME Number S = Short ME Name D = Append ME Date(s) A = Alternate Text (parameter 4) 4 = alt text : Alternate Text / Sub Text / bold-italic

getLastME: get Last X Events (formatted for main page) Output Format is Fixed ("L") Parameter: 1 = Count : return Last Count Events (default 5)

getEventME: get Mini Events during LTE Output Format is Fixed ("E") Parameters: 1 = LTE name : Limited Time Event Name

getMEList: get a List of Mini Events in horizontal format Output Format is Fixed ("H") Parameters: None

getMETimeLine: get a Complete Time Line of Mini Events Output Format is Fixed ("T") Parameters: None

getMENPCUsage: get a list of Mini Events using NPC Output is Fixed Text By default only Lists ME with special NPC version or during LTE Parameters: 1 = NPC Name : Non-Playable Character name 2 = All : If present include all NPC usage --]]

local p = {}

local meData = mw.loadData( 'Module:MiniEvent/data' ) local npc = require('Module:NPC').getNPC local date = require('Dev:Date')

-- local functions -- local function getArgs(frame) local tArgs

if frame == mw.getCurrentFrame then fParent = frame:getParent tArgs = fParent.args

if (fParent.args[1]) then tArgs = fParent.args else tArgs = frame.args end else tArgs = frame end

return tArgs end

-- Date Functions local function ordinal(n) local numb = tonumber(n) or 0 local idd = math.fmod(numb,10) return (idd==1 and numb~=11 and "st") or (idd==2 and numb~=12 and "nd") or (idd==3 and numb~=13 and "rd") or (numb~=0 and "th") or "" end

local function strDate(sDate, days, noYear, sNextDate, cardinal) local theDate = date(sDate)

local nextDate local chkDate = false

if sNextDate then nextDate = date(sNextDate) chkDate = true end

local nDays = tonumber(days) or 0

local day = "" local retText = {}

if nDays > 0 then local start = theDate:copy theDate:adddays(nDays) local day = string.gsub(start:fmt("%d"), "^0", "") table.insert(retText, day)

if not cardinal then table.insert(retText, ordinal(day)) end

if start:getmonth ~= theDate:getmonth then table.insert(retText, start:fmt(" %B")) end

if not noYear and start:getyear ~= theDate:getyear then table.insert(retText, start:fmt(" %Y")) end

table.insert(retText, " - ") end

day = string.gsub(theDate:fmt("%d"), "^0", "") table.insert(retText, day)

if not cardinal then table.insert(retText, ordinal(day)) end table.insert(retText, theDate:fmt(" %B"))

if not noYear and (not chkDate or theDate:getyear ~= nextDate:getyear) then table.insert(retText, theDate:fmt(" %Y")) end

return table.concat(retText) end

local function strMultiDate(tDates, cardinal) table.sort(tDates)

local multiDates = {}

local prevDate = nil

local monAnd = nil local yearAnd = nil

for i = #tDates, 1, -1 do       local curDate = date(tDates[i]) local day = string.gsub(curDate:fmt("%d"), "^0", "") local parts = { day, (not cardinal and ordinal(day) or ""), "", "", ""}

if prevDate then if prevDate:getyear ~= curDate:getyear then parts[3] = curDate:fmt(" %B") parts[4] = curDate:fmt(" %Y") if yearAnd then parts[5] = ", " else if monAnd then parts[5] = "; and " else parts[5] = " and " yearAnd = true end end monAnd = nil else if prevDate:getmonth ~= curDate:getmonth then parts[3] = curDate:fmt(" %B")

if monAnd then parts[5] = "; and " else parts[5] = " and " end monAnd = nil else if monAnd then parts[5] = ", " else parts[5] = " and " monAnd = true end end end else parts[3] = curDate:fmt(" %B") parts[4] = curDate:fmt(" %Y") end

multiDates[i] = parts prevDate = curDate end

prevDate = nil

monAnd = nil yearAnd = nil

for i = 1, #tDates do       local curDate = date(tDates[i])

if prevDate then if prevDate:getyear == curDate:getyear then if prevDate:getmonth == curDate:getmonth then if multiDates[i][5] == " and " and monAnd then multiDates[i][5] = "; and " end end end end

if multiDates[i][5] == " and " then monAnd = true end prevDate = curDate end

local retText = {}

for _, sDate in ipairs(multiDates) do       table.insert(retText, table.concat(sDate)) end

return table.concat(retText) end

-- function get single event local function singleEvent(sName, sNumber, sOption, altText) if sName == "ttc" then return "" end

local eventName = "Unknown Event" local eventNumb = {} local eventDate = "" local eventDays = nil local eventMain = nil local npcVer   = nil

local linkText = "Work In Progress" local linkHash = ""

local outText = ""

if meData[sName] then -- ME-Type Data table local tEvent = meData[sName]

eventName = tEvent["Name"] or "Data Error" imageNPC = tEvent["Icon"] or "" nameNPC  = tEvent["NPC"] or "" linkText = eventName .. " Mini Event"

for n in string.gmatch(sNumber, "%d+") do           if tEvent[tonumber(n)] then table.insert(eventNumb, tonumber(n)) end end

if tEvent["Disp"] then eventName = tEvent["Disp"] end

if eventNumb[1] then -- ME# data table local tNumb = tEvent[eventNumb[1]] eventDate = tNumb[1] or "" eventDays = tNumb[2] or 0 eventMain = tNumb[3] npcVer   = tNumb[4]

-- check if npcVer is "none", in case LTE but no special version if npcVer and npcVer == "none" then npcVer = nil end

-- check for alternate NPC if tNumb[5] and tEvent["NPC" .. tNumb[5]] then nameNPC = tEvent["NPC".. tNumb[5]] end

linkHash = "#" .. eventName .. " #"               .. tostring(eventNumb[1]) end end

if string.match(sOption, "[ETKL]") then -- Fixed Formats if string.match(sOption, "E") then -- Event Page Format outText = "|-\n| align=\"center\" width=\"1\"|"

if nameNPC ~= "" then outText = outText .. npc({nameNPC, nil, nil, nil, eventMain or npcVer}) end

outText = outText .. "\n|"                   .. eventName                    .. "\n|"

if eventDate ~= "" then outText = outText .. strDate(eventDate, eventDays, true) end outText = outText .. "\n"

elseif string.match(sOption, "T") then -- TimeLine Page Format outText = "|"

if nameNPC ~= "" then outText = outText .. npc({nameNPC, nil, nil, nil, eventMain or npcVer}) end

outText = outText .. " "                   .. altText .. "[["                   .. linkText                    .. linkHash                    .. "|"                    .. eventName

if eventNumb[1] and eventDate ~= "" then outText = outText .. " #" .. eventNumb[1] end

outText = outText .. "]]"                   .. altText .. "\n|"

if eventDate ~= "" then outText = outText .. strDate(eventDate, eventDays) end outText = outText .. "\n"

elseif string.match(sOption, "K") then -- Token Page Activity Format outText = "|-\n|"

if nameNPC ~= "" then outText = outText .. npc({nameNPC, nil, nil, 50, eventMain or npcVer}) end

outText = outText .. "\n| colspan=\"4\" align=\"left\"|"                   .. eventName                    .. " Mini Event ''* " .. altText .. "'' "

elseif string.match(sOption, "L") then -- Latest Info Main Page Format outText = "* "                   .. eventName                    .. ""

if eventDate ~= "" then outText = outText .. " "                       .. strDate(eventDate, 0, nil, nil, true) .. " "           end outText = outText .. "\n" end else -- Normal Mode outText = "[[" .. linkText .. linkHash

if eventNumb[1] and eventDate ~= "" then outText = outText .. "|"           if string.match(sOption, "9") then outText = outText .. eventName if not string.match(sOption, "S") then outText = outText .. " Mini Event" end local sNo = "" local sep = false

for _, n in ipairs(eventNumb) do                   if sNo ~= "" then outText = outText .. (sep and "," or "") .. " #" .. sNo sep = true end sNo = tostring(n) end outText = outText .. (sep and " and" or "") .. " #" .. sNo

elseif string.match(sOption, "A") and altText ~= "" then outText = outText .. altText else outText = outText .. eventName if not string.match(sOption, "S") then outText = outText .. " Mini Event" end end else if string.match(sOption, "A") and altText ~= "" then outText = outText .. "|" .. altText elseif string.match(sOption, "S") then outText = outText .. "|" .. eventName else outText = outText .. "|" .. eventName .. " Mini Event" end end

outText = outText .. "]]"

if string.match(sOption, "D") then outText = outText .. " on the "

local tDates = {}

for _, n in ipairs(eventNumb) do               table.insert(tDates, meData[sName][n][1]) end

outText = outText .. strMultiDate(tDates) end end

return outText end

local function getCount(sName) local nCount = 0

local today = os.date("%Y/%m/%d") if meData[sName] then for idx, data in ipairs(meData[sName]) do           if today >= data[1] then nCount = nCount + 1 end end end

return "This mini event has occurred " .. nCount .. " time" .. ((nCount == 1) and "" or "s")

end

-- Public Functions --

-- function get Mini Event function p.getME(frame) local tArgs = getArgs(frame)

local sName   = tArgs[1] or "" local sNumber = tArgs[2] or "" local sOption = tArgs[3] or "" local altText = tArgs[4] or ""

sName = sName:lower:gsub("^%s*(.-)%s*$", "%1") or '' sName = sName:gsub(" mini event$", "") sName = mw.text.decode(sName, false)

sOption = tostring(sOption) sOption = sOption:upper

if sOption == "" then if sNumber ~= "" then sOption = "S" end end

if string.match(sOption, "C") then return getCount(sName) else return singleEvent(sName, sNumber, sOption, altText) end end

-- return last X versions formatted for Main Page function p.getLastME(frame) local tArgs = getArgs(frame)

local count = tArgs[1] or 5 count = tonumber(count) or 5

local tIndex = {}

local today = os.date("%Y/%m/%d") for k, v in pairs(meData) do       if k ~= "ttc" then for idx, data in ipairs(v) do               if today >= data[1] then local t = {data[1], k, idx} table.insert(tIndex, t)               end end end end

table.sort(tIndex, function(a,b) return a[1] > b[1] end)

local outText = ""

for i, ver in ipairs(tIndex) do       outText = outText .. singleEvent(ver[2], ver[3], "L", "") if i >= count then break end end

return outText end

-- return all ME during Limited Time Event formatted for Event Page function p.getEventME(frame) local tArgs = getArgs(frame)

local event = tArgs[1] or ""

-- Does not make sense if no Event specified so return blank if event == "" then return "" end

event = string.lower(event)

local tIndex = {}

for k, v in pairs(meData) do       if k ~= "ttc" then for idx, data in ipairs(v) do               if event == data[3] then local t = {data[1], k, idx} table.insert(tIndex, t)               end end end end

table.sort(tIndex, function(a,b) return a[1] < b[1] end)

local outText = ""

for i, ver in ipairs(tIndex) do       outText = outText .. singleEvent(ver[2], ver[3], "E", "") end

return outText end

-- return all ME Types in horizontal format

function p.getMEList local tIndex = {}

for k, v in pairs(meData) do       if k ~= "ttc" then local t = {k, k, 0} table.insert(tIndex, t)       end end

table.sort(tIndex, function(a,b) return a[1] < b[1] end)

local outText = ""

local space = ""

for i, ver in ipairs(tIndex) do       outText = outText .. space .. singleEvent(ver[2], 0, "S", "")

space = " · " end

return outText end

function p.getMETimeLine local tIndex = {} local tMainEvents = {} local nCount = 0 local nEventCount = 0

local lte = require("Module:LTEvent").getLTE

for k, v in pairs(meData) do       for idx, data in ipairs(v) do            local t = {data[1], k, idx, data[3], nil} table.insert(tIndex, t)

if k ~= "ttc" and data[3] then if not tMainEvents[data[3]] then tMainEvents[data[3]] = 0 end tMainEvents[data[3]] = tMainEvents[data[3]] + 1 end if k ~= "ttc" then nCount = nCount + 1 if data[3] then nEventCount = nEventCount + 1 end end end tIndex[#tIndex][5] = true end

table.sort(tIndex, function(a,b) return a[1] < b[1] end)

local prevMain = ""

local outText = "{| class=\"article-table\" style=\"width: 100%;\"\n" .. "! width=\"27%\"|Major Event\n" .. "! width=\"33%\"|Mini Event\n" .. "! width=\"40%\"|Date(s)\n"

for i, data in ipairs(tIndex) do       outText = outText .. "|-\n"

if data[2] == "ttc" then outText = outText .. "| colspan=\"3\"|" .. lte({data[4], "T", "P"}) .. "\n" prevMain = "" else if data[4] then if data[4] ~= prevMain and tMainEvents[data[4]] then outText = outText .. "| rowspan=\""                           .. tMainEvents[data[4]]                            .. "\" |" .. lte({data[4], "N", "P"}) .. "\n" prevMain = data[4] end else outText = outText .. "|\n" prevMain = "" end

if data[5] then outText = outText .. singleEvent(data[2], data[3], "T", "'''") else outText = outText .. singleEvent(data[2], data[3], "T", "") end end end

outText = outText .. "|}\n\n==Trivia==\n" .. "* There has been a total of " .. nCount .. " mini events, out of which " .. nEventCount .. " occurred during a major event, while " .. (nCount - nEventCount) .. " occured separately.\n"

return outText end

function p.getMENPCUsage(frame) local tArgs = getArgs(frame)

local npcName = tArgs[1] or "" local allVers = (tArgs[2] and tArgs[2] ~= "")

-- return if no NPC specified if npcName == "" then return "No NPC Name Specified\n" end

local tIndex = {} local tMainEvents = {}

local nCount = 0 local nEventCount = 0 local npcThemed = nil

local lte = require("Module:LTEvent").getLTE

for k, v in pairs(meData) do       if k ~= "ttc" then npcThemed = (v["Themed"] ~= nil)

for idx, data in ipairs(v) do               local meNPC = v["NPC"]

-- check alternate NPC if data[5] and v["NPC" .. data[5]] then meNPC = v["NPC" .. data[5]] end

if npcName == meNPC then local vers = data[3] or data[4]

if vers or allVers then local themed = npcThemed and (data[3] ~= nil or data[4] ~= nil) if (data[4] and data[4]=="none") then themed = nil end

local t = {data[1], k, idx, data[3], vers, themed} table.insert(tIndex, t)                   end end end end end

table.sort(tIndex, function(a,b) return a[1] < b[1] end)

local prevData = tIndex[1] local prevNumb = ""

local outText = ""

for i, data in ipairs(tIndex) do       if data[4] ~= prevData[4] or data[5] ~= prevData[5] then local themed = prevData[5] and prevData[6]

outText = outText .. "* The " .. npc({npcName, "D", nil, nil, prevData[5]}) .. " were used for the "

if prevData[4] then outText = outText .. singleEvent(prevData[2], prevNumb, "F", "") .. "s during the " .. lte({prevData[4], "N", "D"}) .. ".\n" else outText = outText .. singleEvent(prevData[2], prevNumb, "FD", "") .. ".\n" end prevNumb = "" end prevData = data prevNumb = prevNumb .. " " .. data[3] end

if prevNumb ~= "" then local themed = prevData[5] and prevData[6]

outText = outText .. "* The " .. npc({npcName, "D", nil, nil, prevData[5]}) .. " were used for the "

if prevData[4] then outText = outText .. singleEvent(prevData[2], prevNumb, "F", "") .. "s during the " .. lte({prevData[4], "N", "D"}) .. ".\n" else outText = outText .. singleEvent(prevData[2], prevNumb, "FD", "") .. ".\n" end end

return outText end

return p

-- --