Module:ArtistSong:修订间差异

来自Arcaea中文维基
(使用.sortName替代.enHideName进行排序)
(mw、html)
第1行: 第1行:
p={}
p = {}
local CALv2 = mw.loadJsonData 'Template:ComplexArtistsList.json'
local CALv2 = mw.loadJsonData 'Template:ComplexArtistsList.json'
local multiId = mw.loadJsonData 'Template:Transition.json'.multiId
local trans = mw.loadJsonData 'Template:Transition.json'
local trans = mw.loadJsonData 'Template:Transition.json'
local SL = mw.loadJsonData 'Template:Song Length.json'
local SL = mw.loadJsonData 'Template:Song Length.json'


local mad = require 'Module:AnotherData'
local mad = require 'Module:AnotherData'
local data = require "Module:Arcaea Data"
local data = require 'Module:Arcaea Data'
local CL = require 'Module:ConvertLink'
local CL = require 'Module:ConvertLink'
local tt = require 'Module:TableTools'


local queryMO = data.allSongInformation("id", "mobile")
local lang = mw.language.getContentLanguage()
local queryNS = data.allSongInformation("id", "ns")
 
local queryMO = data.allSongInformation('id', 'mobile')
local queryNS = data.allSongInformation('id', 'ns')


-- byd?=>id=>c-artist
local function readList()
local function readList()
   local pickList = {}
local pickList = {}
   local pickListBYD = {}
local pickListBYD = {}
   for _, song in ipairs(mad.listOf('songs', 'mobile')) do
for _, song in ipairs(mad.listOf('songs', 'mobile')) do
   if not song.deleted then
if not song.deleted then
   local artist = song.artist
local artist = song.artist
     pickList[song.id] = song.artist
pickList[song.id] = song.artist
     if song.difficulties[4] and (song.difficulties[4].artist or song.difficulties[4].title_localized) then
if song.difficulties[4] and (song.difficulties[4].artist or song.difficulties[4].title_localized) then
     local bydArtist = song['difficulties'][4]['artist'] or song.artist
local bydArtist = song['difficulties'][4]['artist'] or song.artist
       pickListBYD[song['id']] = bydArtist
pickListBYD[song['id']] = bydArtist
     end
end
   end
end
   end
end
  
 
   for _, song in ipairs(mad.listOf('songs', 'ns')) do
for _, song in ipairs(mad.listOf('songs', 'ns')) do
     if not pickList[song.id] and not multiId[song.id] then
if not pickList[song.id] and not trans.multiId[song.id] then
pickList[song.id] = song.artist
pickList[song.id] = song.artist
if song.difficulties[4] and song.difficulties[4].artist or song.difficulties[4].title_localized then
if song.difficulties[4] and song.difficulties[4].artist or song.difficulties[4].title_localized then
     local bydArtist = song['difficulties'][4]['artist'] or song.artist
local bydArtist = song['difficulties'][4]['artist'] or song.artist
pickListBYD[song['id']] = bydArtist
pickListBYD[song['id']] = bydArtist
end
end
第36行: 第39行:
end
end


   return pickList, pickListBYD
return pickList, pickListBYD
end
end


-- groupby: c-artist=>byd?=>[id]
local function convertPick(pickList, pickListBYD)
local function convertPick(pickList, pickListBYD)
   convertList = {}
convertList = {}
   for _id,_ in pairs(pickList) do
for id, artist in pairs(pickList) do
     local artist = pickList[_id]
if not convertList[artist] then convertList[artist] = {} end
     if not convertList[artist] then convertList[artist] = {} end
if not convertList[artist].normal then convertList[artist]['normal'] = {} end
     if not convertList[artist].normal then convertList[artist]['normal'] = {} end
table.insert(convertList[artist]['normal'], id)
     table.insert(convertList[artist]['normal'], _id)
end
   end


   for _id in pairs(pickListBYD) do
for id, artist in pairs(pickListBYD) do
   local artist = pickListBYD[_id]
if not convertList[artist] then convertList[artist] = {} end
   if not convertList[artist] then convertList[artist] = {} end
if not convertList[artist].beyond then convertList[artist]['beyond'] = {} end
   if not convertList[artist].beyond then convertList[artist]['beyond'] = {} end
table.insert(convertList[artist]['beyond'], id)
   table.insert(convertList[artist]['beyond'], _id)
end
   end


   return convertList
return convertList
end
end


-- split-flatten: s-artist=>byd?=>[id]
local zhName = {}
local zhName = {}
local function genList(convertList)
local function genList(convertList)
第63行: 第66行:
local artist = artist or cArtist
local artist = artist or cArtist
if not singleList[artist] then singleList[artist] = {} end
if not singleList[artist] then singleList[artist] = {} end
     if convertList[cArtist].beyond then
if convertList[cArtist].beyond then
       if not singleList[artist].beyond then singleList[artist]['beyond'] = {} end
if not singleList[artist].beyond then singleList[artist]['beyond'] = {} end
       for _,i in pairs(convertList[cArtist]['beyond']) do
tt.extend(singleList[artist]['beyond'], convertList[cArtist]['beyond'])
       table.insert(singleList[artist]['beyond'],i)
end
       end
if convertList[cArtist].normal then
     end
if not singleList[artist].normal then singleList[artist]['normal'] = {} end
     if convertList[cArtist].normal then
tt.extend(singleList[artist]['normal'], convertList[cArtist]['normal'])
     if not singleList[artist].normal then singleList[artist]['normal'] = {} end
end
       for _,i in pairs(convertList[cArtist]['normal']) do
return singleList
       table.insert(singleList[artist]['normal'],i)
end
       end
 
     end
 
     return singleList
local singleList = {}
   end
for cArtist in pairs(convertList) do
local artistlist = {}
if CALv2[cArtist] then
   local singleList = {}
   for cArtist in pairs(convertList) do
   local artistlist={}
     if CALv2[cArtist] then
if CALv2[cArtist].__FullData__ then
if CALv2[cArtist].__FullData__ then
for _,text in pairs(CALv2[cArtist].__FullData__) do
for _, text in pairs(CALv2[cArtist].__FullData__) do
if text.link then
if text.link then
local artist = text.link
local artist = text.link
if not artistlist[artist] then
if not artistlist[artist] then
singleList = cate(singleList, cArtist, artist)
singleList = cate(singleList, cArtist, artist)
artistlist[artist]=1
artistlist[artist] = 1
if text.zh then
if text.zh then
zhName[artist] = true
zhName[artist] = true
第98行: 第97行:
else
else
for link in pairs(CALv2[cArtist]) do
for link in pairs(CALv2[cArtist]) do
           local artist = link
local artist = link
if not artistlist[artist] then
if not artistlist[artist] then
singleList = cate(singleList, cArtist, artist)
singleList = cate(singleList, cArtist, artist)
artistlist[artist]=1
artistlist[artist] = 1
end
end
end
end
end
end
     else
else
       singleList = cate(singleList, cArtist)
singleList = cate(singleList, cArtist)
     end
end
end
end
   return singleList
return singleList
end
end


local function isLetter(char)
local function isLetter(char)
return char:match("%a") ~= nil
return char:match '%a' ~= nil
end
end


-- 排序
-- 排序
local function customSort(a, b)
local function customSort(a, b)
  
local a1 = a:sub(1, 1)
local a1 = a:sub(1, 1)
local b1 = b:sub(1, 1)
local b1 = b:sub(1, 1)
   local la1 = isLetter(a:sub(1, 1))
local la1 = isLetter(a:sub(1, 1))
   local lb1 = isLetter(b:sub(1, 1))
local lb1 = isLetter(b:sub(1, 1))


   if la1 and not lb1 then
if la1 and not lb1 then
     return true
return true
   elseif lb1 and not la1 then
elseif lb1 and not la1 then
     return false
return false
   elseif string.lower(a1) == string.lower(b1) then
elseif string.lower(a1) == string.lower(b1) then
   local la2 = isLetter(a:sub(2, 2))
local la2 = isLetter(a:sub(2, 2))
   local lb2 = isLetter(b:sub(2, 2))
local lb2 = isLetter(b:sub(2, 2))
   if la2 and not lb2 then
if la2 and not lb2 then
     return true
return true
   elseif lb2 and not la2 then
elseif lb2 and not la2 then
     return false
return false
   else
else
   return string.lower(a) < string.lower(b)
return string.lower(a) < string.lower(b)
   end
end
   else
else
     return string.lower(a) < string.lower(b)
return string.lower(a) < string.lower(b)
   end
end


   return string.lower(a) < string.lower(b)
return string.lower(a) < string.lower(b)
end
end


local function songTitle(id,byd)
local function songTitle(id, byd)
local display
local display
local title = queryMO(id, 'title') or queryNS(id, 'title')
local title = queryMO(id, 'title') or queryNS(id, 'title')
if byd then
if byd then
display = queryMO(id, 'bydSongName') or queryNS(id, 'bydSongName') or 'cant find BYD'
display = queryMO(id, 'bydSongName') or queryNS(id, 'bydSongName') or 'cant find BYD'
else 
else
display = title
display = title
end
end
local res = trans.songNameToDisplayName[title]
local res = trans.songNameToDisplayName[title]
if res then return res..'|'..display end
if res then return res .. '|' .. display end
res = trans.sameName[title]
res = trans.sameName[title]
if res then return res[id]..'|'..display end
if res then return res[id] .. '|' .. display end
if title then return title..'|'..display
if title then
else return id end
return title .. '|' .. display
else
return id
end
end
end


第168行: 第169行:
end
end
bpm = queryMO(id, 'bpm') or queryNS(id, 'bpm')
bpm = queryMO(id, 'bpm') or queryNS(id, 'bpm')
if bpm then return bpm 
if bpm then
else return nil end
return bpm
else
return nil
end
end
end


local lang = mw.language.getContentLanguage()
local function songDate(id, byd)
local function songDate(id, byd)
local date, ver
local date, ver
第194行: 第197行:


local function songPack(id)
local function songPack(id)
local packid,plat,isNS
local packid, plat, isNS
-- if queryNS(id, 'set') then
-- if queryNS(id, 'set') then
-- packid = queryNS(id, 'set')
-- packid = queryNS(id, 'set')
第205行: 第208行:
end
end
local packName = trans.packName[packid] or mad.packQueryWrap(packid, plat).name
local packName = trans.packName[packid] or mad.packQueryWrap(packid, plat).name
local packLink = CL.packLink(packName,1,isNS)
local packLink = CL.packLink(packName, 1, isNS)
return packLink
return packLink
end
end


local function songLength(id,byd)
local function songLength(id, byd)
if byd then
if byd then
return SL.beyond[id]
return SL.beyond[id]
第217行: 第220行:
end
end


 
 


function p.main(frame)
function p.main(frame)
local parentArgs = frame:getParent().args
local parentArgs = frame:getParent().args
local Args = frame.args
local Args = frame.args
 
   local rl, rlb = readList()
local rl, rlb = readList()
   local cl = convertPick(rl, rlb)
local cl = convertPick(rl, rlb)
   local gl = genList(cl)
local gl = genList(cl)
   rl,rlb,cl = nil
rl, rlb, cl = nil
  
 
   local appendList = Args['appendList']
local appendList = Args['appendList']
if appendList then
if appendList then
local list = mw.text.split(appendList,'/')
local list = mw.text.split(appendList, '/')
for _,id in pairs(list) do
for _, id in pairs(list) do
local artists = Args[id .. '.'..'artist']
local artists = Args[id .. '.' .. 'artist']
local artistList = mw.text.split(artists,'/')
local artistList = mw.text.split(artists, '/')
for _, artist in pairs(artistList) do
for _, artist in pairs(artistList) do
if not gl[artist] then gl[artist]={} end
if not gl[artist] then gl[artist] = {} end
if not gl[artist].normal then gl[artist].normal={} end
if not gl[artist].normal then gl[artist].normal = {} end
table.insert(gl[artist].normal, id)
table.insert(gl[artist].normal, id)
end
end
end
end
end
end
  
   local wikiText = ""


   local a2z = {}
local render = mw.html.create()
   local gl2 = {}
 
local a2z = {}
local gl2 = {}
for name in pairs(gl) do
for name in pairs(gl) do
local enName
local enName
if mw.ustring.match(name,'[一-龠ぁ-ゔァ-ヴー々〆〤ヶ]+') then
if mw.ustring.match(name, '[一-龠ぁ-ゔァ-ヴー々〆〤ヶ]+') then
-- enName = parentArgs[name .. '.enName'] or name--nil
-- enName = parentArgs[name .. '.enName'] or name--nil
local enHide = parentArgs[name .. '.sortName'] or parentArgs[name .. '.enHideName']
local enHide = parentArgs[name .. '.sortName'] or parentArgs[name .. '.enHideName']
local notJaName = parentArgs[name .. '.notJaName'] or zhName[name]
local notJaName = parentArgs[name .. '.notJaName'] or zhName[name]
enName = enHide or parentArgs[name .. '.enName'] or name--nil
enName = enHide or parentArgs[name .. '.enName'] or name --nil
gl2[enName] = gl[name]
gl2[enName] = gl[name]
gl2[enName].localName = name
gl2[enName].localName = name
第267行: 第270行:
gl2[enName].otherName = parentArgs[name .. '.otherName'] or nil
gl2[enName].otherName = parentArgs[name .. '.otherName'] or nil
gl2[enName].description = parentArgs[name .. '.description'] or nil
gl2[enName].description = parentArgs[name .. '.description'] or nil
table.insert(a2z,enName) 
table.insert(a2z, enName)
end
end
table.sort(a2z, customSort)
table.sort(a2z, customSort)


-- 生成页首
-- 生成页首
local menuName={nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil,nil}
local menuName = {} --长27
for _, name in ipairs(a2z) do
for _, name in ipairs(a2z) do
local catId=27
local catId = 27
local first=string.lower(name:sub(1, 1))
local first = string.lower(name:sub(1, 1))
if isLetter(first) then
if isLetter(first) then
catId=string.byte(first)-string.byte("a")+1
catId = string.byte(first) - string.byte 'a' + 1
end
end
if menuName[catId]==nil then 
if menuName[catId] == nil then
menuName[catId]=name
menuName[catId] = name
end
end
end
end
if menuName[27]==nil then
if menuName[27] == nil then
menuName[27]=""
menuName[27] = ''
end
end
for i=26,1,-1 do
for i = 26, 1, -1 do
if menuName[i]==nil then
if menuName[i] == nil then
menuName[i]=menuName[i+1]
menuName[i] = menuName[i + 1]
end
end
end
end


local menu={}
local menu = {}
for _, name in ipairs(menuName) do
for _, name in ipairs(menuName) do
if gl2[name] then
if gl2[name] then
table.insert(menu,gl2[name].localName or name) 
table.insert(menu, gl2[name].localName or name)
else
else
table.insert(menu,name) 
table.insert(menu, name)
end
end
end
end


wikiText = wikiText .. frame:expandTemplate{ title = " 目录", args = menu } .. "\n\n"
render:wikitext(frame:expandTemplate {title = ' 目录', args = menu})
  
:newline()
   local function tableAppend(line,newStr) return line..'|'..newStr..'\n' end
 
  
local function songArg(artist, id, byd, arg)
   local function songArg(artist,id,byd,arg)
arg = '.' .. arg
   arg = '.'..arg
artist = artist .. '.'
   artist = artist .. '.'
local function fun(a) return a and artist .. a .. arg end
   local function fun(a) return a and artist..a..arg end
if byd then
if byd then 
return parentArgs[fun(queryMO(id, 'bydSongName'))] or parentArgs[fun(id .. '.byd')]
return parentArgs[fun(queryMO(id, 'bydSongName'))] or parentArgs[fun(id..'.byd')]
else
else
return parentArgs[fun(queryMO(id, 'title'))] or parentArgs[fun(id)]
return parentArgs[fun(queryMO(id, 'title'))] or parentArgs[fun(id)]
end
end
end
end
  
 
   local function aslAppend(artist, songid, date, byd) 
local function aslAppend(artist, songid, date, byd)
local theList={}
local theList = {}
local function Input(arg) return songid.. ( byd and '.byd' or '') ..(arg and '.'..arg or '') end
local function Input(arg) return songid .. (byd and '.byd' or '') .. (arg and '.' .. arg or '') end
local function aInput(arg) return artist ..'.' .. Input(arg) end
local function aInput(arg) return artist .. '.' .. Input(arg) end
local function none(arg) return '|<span title="'.. aInput(arg) .. '">' .. '--'..'</span>' end
local function none(arg) return '<span title="' .. aInput(arg) .. '">' .. '--' .. '</span>' end
theList.title = ("[[" .. (Args[Input('title')] or songTitle(songid,byd)) .. "]]") or none()
theList.title = ('[[' .. (Args[Input 'title'] or songTitle(songid, byd)) .. ']]') or none()
theList.length = Args[Input('length')] or songLength(songid,byd) or none()
theList.length = Args[Input 'length'] or songLength(songid, byd) or none()
theList.bpm = Args[Input('bpm')] or songBPM(songid,byd) or none()
theList.bpm = Args[Input 'bpm'] or songBPM(songid, byd) or none()
theList.pack = Args[Input('pack')] or songPack(songid) or none()
theList.pack = Args[Input 'pack'] or songPack(songid) or none()
local releaseVer, releaseDate = CL.fullVersionNum(songVer(songid,byd), date)
local releaseVer, releaseDate = CL.fullVersionNum(songVer(songid, byd), date)
local ver = Args[Input('ver')] or releaseVer or none()
local ver = Args[Input 'ver'] or releaseVer or none()
theList.date = Args[Input('date')] or releaseDate
theList.date = Args[Input 'date'] or releaseDate
local AprilFoolVer = Args[Input('AprilFoolVer')]
local AprilFoolVer = Args[Input 'AprilFoolVer']
theList.AprilFoolDate = Args[Input('AprilFoolDate')]
theList.AprilFoolDate = Args[Input 'AprilFoolDate']
local afVer
local afVer
if AprilFoolVer and theList.AprilFoolDate then
if AprilFoolVer and theList.AprilFoolDate then
afVer = ("'''愚人节'''<br>"..'v'..AprilFoolVer .. '<br />('.. theList.AprilFoolDate .. ")<br>'''常驻'''<br>") or ''
afVer = ("'''愚人节'''<br>" .. 'v' .. AprilFoolVer .. '<br />(' .. theList.AprilFoolDate .. ")<br>'''常驻'''<br>") or ''
end
end
theList.displayVer = (afVer or '')..'v'..ver .. '<br />('.. theList.date .. ')' or none()
theList.displayVer = (afVer or '') .. 'v' .. ver .. '<br />(' .. theList.date .. ')' or none()
theList.text = songArg(artist,songid,byd,'text') or none('text')
theList.text = songArg(artist, songid, byd, 'text') or none 'text'
return theList
return theList
end
end
  
 
   for _, name in ipairs(a2z) do
for _, name in ipairs(a2z) do
   local item = gl2[name]
local item = gl2[name]
   local displayName, usingName,other
local displayName, usingName, other
   if item.otherName then
if item.otherName then
   other = (' ('..item.otherName..')') or nil
other = (' (' .. item.otherName .. ')') or nil
   end
end
   if item.localName then
if item.localName then
   usingName = item.localName
usingName = item.localName
   enDisplay = item.enHide and '' or ' (' .. name ..')'
enDisplay = item.enHide and '' or ' (' .. name .. ')'
   if item.notJa then
if item.notJa then
   displayName = usingName .. (other or '') .. enDisplay
displayName = usingName .. (other or '') .. enDisplay
   else
else
   displayName = '-{<span lang="ja">' .. usingName .. '</span>}-' .. (other or '') .. enDisplay
displayName = '-{<span lang="ja">' .. usingName .. '</span>}-' .. (other or '') .. enDisplay
   end
end
   else
else
   usingName = name
usingName = name
   displayName = name .. (other or '')
displayName = name .. (other or '')
   end
end
   -- wikiText = wikiText .. '*' .. usingName .. '\n'
 
  
render:wikitext(('<span id="%s"></span><h2>%s</h2>'):format(usingName, displayName))
   wikiText = wikiText .. '<span id="' .. usingName .. '" />\n'
:newline()
   wikiText = wikiText .. '<h2><span title="'.. usingName .. '">' .. displayName..'</span></h2>\n'
:wikitext(item.description)
   -- wikiText = wikiText .. 'Span id: ' .. usingName .."\n"
:newline()
   wikiText = wikiText .. (item.description or '') .. '\n'
local wt = render:tag 'table'
   wikiText = wikiText .. '{| class="wikitable" border="1" cellspacing="1" cellpadding="5" style="text-align:center;width: 100%"\n'
wt:attr {
wikiText = wikiText .. '! style="width: 30.5%;"| 曲目名称\n'
class = 'wikitable', border = '1', cellspacing = '1', cellpadding = '5', style = 'text-align:center;width: 100%'
wikiText = wikiText .. '! style="width: 6%;"| 时长\n'
}:tag 'tr'
wikiText = wikiText .. '! style="width: 10%;"|BPM\n'
:tag 'th':attr {style = 'width: 30.5%;'}:wikitext ' 曲目名称':done()
wikiText = wikiText .. '! style="width: 18.5%;"| 曲包\n'
:tag 'th':attr {style = 'width: 6%;'}:wikitext ' 时长':done()
wikiText = wikiText .. '! style="width: 15%;"| 发布版本\n'
:tag 'th':attr {style = 'width: 10%;'}:wikitext 'BPM':done()
wikiText = wikiText .. '! style="width: 20%;"| 备注\n'
:tag 'th':attr {style = 'width: 18.5%;'}:wikitext ' 曲包':done()
  
:tag 'th':attr {style = 'width: 15%;'}:wikitext ' 发布版本':done()
   local artistSongList = {}
:tag 'th':attr {style = 'width: 20%;'}:wikitext ' 备注':done()
   local listSort = {}
 
   if item.normal then
local artistSongList = {}
   for _,songid in ipairs(item.normal) do
local listSort = {}
   local date = songDate(songid)
if item.normal then
   local theList=aslAppend(usingName, songid, date)
for _, songid in ipairs(item.normal) do
   local aslId = (theList.AprilFoolDate or theList.date)..songid
local date = songDate(songid)
   artistSongList[aslId] = theList
local theList = aslAppend(usingName, songid, date)
local aslId = (theList.AprilFoolDate or theList.date) .. songid
artistSongList[aslId] = theList
table.insert(listSort, aslId)
table.insert(listSort, aslId)
   end
end
   end
end
   if item.beyond then
if item.beyond then
   for _,songid in ipairs(item.beyond) do
for _, songid in ipairs(item.beyond) do
   local date = songDate(songid, 1)
local date = songDate(songid, 1)
   local theList = aslAppend(usingName, songid, date, 1)
local theList = aslAppend(usingName, songid, date, 1)
   local aslId = (theList.AprilFoolDate or theList.date)..songid..'byd'
local aslId = (theList.AprilFoolDate or theList.date) .. songid .. 'byd'
   artistSongList[aslId] = theList
artistSongList[aslId] = theList
table.insert(listSort, aslId)
table.insert(listSort, aslId)
   end
end
   end
end
   table.sort(listSort)
table.sort(listSort)
   for _,id in ipairs(listSort) do
for _, id in ipairs(listSort) do
   wikiText = tableAppend(wikiText, '-')
local theList = artistSongList[id]
   local theList = artistSongList[id]
wt:tag 'tr'
   wikiText = tableAppend(wikiText, theList.title)
:tag 'td':wikitext(theList.title):done()
   wikiText = tableAppend(wikiText, theList.length)
:tag 'td':wikitext(theList.length):done()
   wikiText = tableAppend(wikiText, theList.bpm)
:tag 'td':wikitext(theList.bpm):done()
   wikiText = tableAppend(wikiText, theList.pack)
:tag 'td':wikitext(theList.pack):done()
   wikiText = tableAppend(wikiText, theList.displayVer)
:tag 'td':wikitext(theList.displayVer):done()
   wikiText = tableAppend(wikiText, theList.text)
:tag 'td':wikitext(theList.text):done()
   end
end
     wikiText = wikiText .. "|}\n"
end
   end
return tostring(render)
   return wikiText
end
end


return p
return p

2024年10月20日 (日) 19:23的版本

可在Module:ArtistSong/doc创建此模块的帮助文档

p = {}
local CALv2 = mw.loadJsonData 'Template:ComplexArtistsList.json'
local trans = mw.loadJsonData 'Template:Transition.json'
local SL = mw.loadJsonData 'Template:Song Length.json'

local mad = require 'Module:AnotherData'
local data = require 'Module:Arcaea Data'
local CL = require 'Module:ConvertLink'
local tt = require 'Module:TableTools'

local lang = mw.language.getContentLanguage()

local queryMO = data.allSongInformation('id', 'mobile')
local queryNS = data.allSongInformation('id', 'ns')

-- byd?=>id=>c-artist
local function readList()
	local pickList = {}
	local pickListBYD = {}
	for _, song in ipairs(mad.listOf('songs', 'mobile')) do
		if not song.deleted then
			local artist = song.artist
			pickList[song.id] = song.artist
			if song.difficulties[4] and (song.difficulties[4].artist or song.difficulties[4].title_localized) then
				local bydArtist = song['difficulties'][4]['artist'] or song.artist
				pickListBYD[song['id']] = bydArtist
			end
		end
	end

	for _, song in ipairs(mad.listOf('songs', 'ns')) do
		if not pickList[song.id] and not trans.multiId[song.id] then
			pickList[song.id] = song.artist
			if song.difficulties[4] and song.difficulties[4].artist or song.difficulties[4].title_localized then
				local bydArtist = song['difficulties'][4]['artist'] or song.artist
				pickListBYD[song['id']] = bydArtist
			end
		end
	end

	return pickList, pickListBYD
end

-- groupby: c-artist=>byd?=>[id]
local function convertPick(pickList, pickListBYD)
	convertList = {}
	for id, artist in pairs(pickList) do
		if not convertList[artist] then convertList[artist] = {} end
		if not convertList[artist].normal then convertList[artist]['normal'] = {} end
		table.insert(convertList[artist]['normal'], id)
	end

	for id, artist in pairs(pickListBYD) do
		if not convertList[artist] then convertList[artist] = {} end
		if not convertList[artist].beyond then convertList[artist]['beyond'] = {} end
		table.insert(convertList[artist]['beyond'], id)
	end

	return convertList
end

-- split-flatten: s-artist=>byd?=>[id]
local zhName = {}
local function genList(convertList)
	local function cate(singleList, cArtist, artist)
		local artist = artist or cArtist
		if not singleList[artist] then singleList[artist] = {} end
		if convertList[cArtist].beyond then
			if not singleList[artist].beyond then singleList[artist]['beyond'] = {} end
			tt.extend(singleList[artist]['beyond'], convertList[cArtist]['beyond'])
		end
		if convertList[cArtist].normal then
			if not singleList[artist].normal then singleList[artist]['normal'] = {} end
			tt.extend(singleList[artist]['normal'], convertList[cArtist]['normal'])
		end
		return singleList
	end


	local singleList = {}
	for cArtist in pairs(convertList) do
		local artistlist = {}
		if CALv2[cArtist] then
			if CALv2[cArtist].__FullData__ then
				for _, text in pairs(CALv2[cArtist].__FullData__) do
					if text.link then
						local artist = text.link
						if not artistlist[artist] then
							singleList = cate(singleList, cArtist, artist)
							artistlist[artist] = 1
							if text.zh then
								zhName[artist] = true
							end
						end
					end
				end
			else
				for link in pairs(CALv2[cArtist]) do
					local artist = link
					if not artistlist[artist] then
						singleList = cate(singleList, cArtist, artist)
						artistlist[artist] = 1
					end
				end
			end
		else
			singleList = cate(singleList, cArtist)
		end
	end
	return singleList
end

local function isLetter(char)
	return char:match '%a' ~= nil
end

-- 排序
local function customSort(a, b)
	local a1 = a:sub(1, 1)
	local b1 = b:sub(1, 1)
	local la1 = isLetter(a:sub(1, 1))
	local lb1 = isLetter(b:sub(1, 1))

	if la1 and not lb1 then
		return true
	elseif lb1 and not la1 then
		return false
	elseif string.lower(a1) == string.lower(b1) then
		local la2 = isLetter(a:sub(2, 2))
		local lb2 = isLetter(b:sub(2, 2))
		if la2 and not lb2 then
			return true
		elseif lb2 and not la2 then
			return false
		else
			return string.lower(a) < string.lower(b)
		end
	else
		return string.lower(a) < string.lower(b)
	end

	return string.lower(a) < string.lower(b)
end

local function songTitle(id, byd)
	local display
	local title = queryMO(id, 'title') or queryNS(id, 'title')
	if byd then
		display = queryMO(id, 'bydSongName') or queryNS(id, 'bydSongName') or 'cant find BYD'
	else
		display = title
	end
	local res = trans.songNameToDisplayName[title]
	if res then return res .. '|' .. display end
	res = trans.sameName[title]
	if res then return res[id] .. '|' .. display end
	if title then
		return title .. '|' .. display
	else
		return id
	end
end

local function songBPM(id, byd)
	local bpm
	if byd then
		bpm = queryMO(id, 'bydBPM') or queryNS(id, 'bydBPM')
		if bpm then return bpm end
	end
	bpm = queryMO(id, 'bpm') or queryNS(id, 'bpm')
	if bpm then
		return bpm
	else
		return nil
	end
end

local function songDate(id, byd)
	local date, ver
	if byd then
		date = queryMO(id, 'bydDate') or queryNS(id, 'bydDate')
		if date then return date end
	end
	date = queryMO(id, 'date') or queryNS(id, 'date')
	if date then return date else return 0 end
end

local function songVer(id, byd)
	local date, ver
	if byd then
		ver = queryMO(id, 'bydVersion') or queryNS(id, 'bydVersion')
		if ver then return ver end
	end
	ver = queryMO(id, 'version') or queryNS(id, 'version')
	if ver then return ver else return id end
end

local function songPack(id)
	local packid, plat, isNS
	-- if queryNS(id, 'set') then
	-- 	packid = queryNS(id, 'set')
	-- 	plat = 'ns'
	-- 	isNS = 1
	if queryMO(id, 'set') then
		packid = queryMO(id, 'set')
	else
		return nil
	end
	local packName = trans.packName[packid] or mad.packQueryWrap(packid, plat).name
	local packLink = CL.packLink(packName, 1, isNS)
	return packLink
end

local function songLength(id, byd)
	if byd then
		return SL.beyond[id]
	else
		return SL.normal[id]
	end
end



function p.main(frame)
	local parentArgs = frame:getParent().args
	local Args = frame.args

	local rl, rlb = readList()
	local cl = convertPick(rl, rlb)
	local gl = genList(cl)
	rl, rlb, cl = nil

	local appendList = Args['appendList']
	if appendList then
		local list = mw.text.split(appendList, '/')
		for _, id in pairs(list) do
			local artists = Args[id .. '.' .. 'artist']
			local artistList = mw.text.split(artists, '/')
			for _, artist in pairs(artistList) do
				if not gl[artist] then gl[artist] = {} end
				if not gl[artist].normal then gl[artist].normal = {} end
				table.insert(gl[artist].normal, id)
			end
		end
	end

	local render = mw.html.create()

	local a2z = {}
	local gl2 = {}
	for name in pairs(gl) do
		local enName
		if mw.ustring.match(name, '[一-龠ぁ-ゔァ-ヴー々〆〤ヶ]+') then
			-- enName = parentArgs[name .. '.enName'] or name--nil
			local enHide = parentArgs[name .. '.sortName'] or parentArgs[name .. '.enHideName']
			local notJaName = parentArgs[name .. '.notJaName'] or zhName[name]
			enName = enHide or parentArgs[name .. '.enName'] or name --nil
			gl2[enName] = gl[name]
			gl2[enName].localName = name
			if enHide then
				gl2[enName].enHide = 1
			end
			if notJaName then
				gl2[enName].notJa = 1
			end
		else
			enName = name
			gl2[enName] = gl[name]
		end
		gl2[enName].otherName = parentArgs[name .. '.otherName'] or nil
		gl2[enName].description = parentArgs[name .. '.description'] or nil
		table.insert(a2z, enName)
	end
	table.sort(a2z, customSort)

	-- 生成页首
	local menuName = {} --长27
	for _, name in ipairs(a2z) do
		local catId = 27
		local first = string.lower(name:sub(1, 1))
		if isLetter(first) then
			catId = string.byte(first) - string.byte 'a' + 1
		end
		if menuName[catId] == nil then
			menuName[catId] = name
		end
	end
	if menuName[27] == nil then
		menuName[27] = ''
	end
	for i = 26, 1, -1 do
		if menuName[i] == nil then
			menuName[i] = menuName[i + 1]
		end
	end

	local menu = {}
	for _, name in ipairs(menuName) do
		if gl2[name] then
			table.insert(menu, gl2[name].localName or name)
		else
			table.insert(menu, name)
		end
	end

	render:wikitext(frame:expandTemplate {title = '目录', args = menu})
		:newline()

	local function songArg(artist, id, byd, arg)
		arg = '.' .. arg
		artist = artist .. '.'
		local function fun(a) return a and artist .. a .. arg end
		if byd then
			return parentArgs[fun(queryMO(id, 'bydSongName'))] or parentArgs[fun(id .. '.byd')]
		else
			return parentArgs[fun(queryMO(id, 'title'))] or parentArgs[fun(id)]
		end
	end

	local function aslAppend(artist, songid, date, byd)
		local theList = {}
		local function Input(arg) return songid .. (byd and '.byd' or '') .. (arg and '.' .. arg or '') end
		local function aInput(arg) return artist .. '.' .. Input(arg) end
		local function none(arg) return '<span title="' .. aInput(arg) .. '">' .. '--' .. '</span>' end
		theList.title = ('[[' .. (Args[Input 'title'] or songTitle(songid, byd)) .. ']]') or none()
		theList.length = Args[Input 'length'] or songLength(songid, byd) or none()
		theList.bpm = Args[Input 'bpm'] or songBPM(songid, byd) or none()
		theList.pack = Args[Input 'pack'] or songPack(songid) or none()
		local releaseVer, releaseDate = CL.fullVersionNum(songVer(songid, byd), date)
		local ver = Args[Input 'ver'] or releaseVer or none()
		theList.date = Args[Input 'date'] or releaseDate
		local AprilFoolVer = Args[Input 'AprilFoolVer']
		theList.AprilFoolDate = Args[Input 'AprilFoolDate']
		local afVer
		if AprilFoolVer and theList.AprilFoolDate then
			afVer = ("'''愚人节'''<br>" .. 'v' .. AprilFoolVer .. '<br />(' .. theList.AprilFoolDate .. ")<br>'''常驻'''<br>") or ''
		end
		theList.displayVer = (afVer or '') .. 'v' .. ver .. '<br />(' .. theList.date .. ')' or none()
		theList.text = songArg(artist, songid, byd, 'text') or none 'text'
		return theList
	end

	for _, name in ipairs(a2z) do
		local item = gl2[name]
		local displayName, usingName, other
		if item.otherName then
			other = (' (' .. item.otherName .. ')') or nil
		end
		if item.localName then
			usingName = item.localName
			enDisplay = item.enHide and '' or ' (' .. name .. ')'
			if item.notJa then
				displayName = usingName .. (other or '') .. enDisplay
			else
				displayName = '-{<span lang="ja">' .. usingName .. '</span>}-' .. (other or '') .. enDisplay
			end
		else
			usingName = name
			displayName = name .. (other or '')
		end

		render:wikitext(('<span id="%s"></span><h2>%s</h2>'):format(usingName, displayName))
			:newline()
			:wikitext(item.description)
			:newline()
		local wt = render:tag 'table'
		wt:attr {
			class = 'wikitable', border = '1', cellspacing = '1', cellpadding = '5', style = 'text-align:center;width: 100%'
		}:tag 'tr'
			:tag 'th':attr {style = 'width: 30.5%;'}:wikitext '曲目名称':done()
			:tag 'th':attr {style = 'width: 6%;'}:wikitext '时长':done()
			:tag 'th':attr {style = 'width: 10%;'}:wikitext 'BPM':done()
			:tag 'th':attr {style = 'width: 18.5%;'}:wikitext '曲包':done()
			:tag 'th':attr {style = 'width: 15%;'}:wikitext '发布版本':done()
			:tag 'th':attr {style = 'width: 20%;'}:wikitext '备注':done()

		local artistSongList = {}
		local listSort = {}
		if item.normal then
			for _, songid in ipairs(item.normal) do
				local date = songDate(songid)
				local theList = aslAppend(usingName, songid, date)
				local aslId = (theList.AprilFoolDate or theList.date) .. songid
				artistSongList[aslId] = theList
				table.insert(listSort, aslId)
			end
		end
		if item.beyond then
			for _, songid in ipairs(item.beyond) do
				local date = songDate(songid, 1)
				local theList = aslAppend(usingName, songid, date, 1)
				local aslId = (theList.AprilFoolDate or theList.date) .. songid .. 'byd'
				artistSongList[aslId] = theList
				table.insert(listSort, aslId)
			end
		end
		table.sort(listSort)
		for _, id in ipairs(listSort) do
			local theList = artistSongList[id]
			wt:tag 'tr'
				:tag 'td':wikitext(theList.title):done()
				:tag 'td':wikitext(theList.length):done()
				:tag 'td':wikitext(theList.bpm):done()
				:tag 'td':wikitext(theList.pack):done()
				:tag 'td':wikitext(theList.displayVer):done()
				:tag 'td':wikitext(theList.text):done()
		end
	end
	return tostring(render)
end

return p