taiko-web/public/src/js/importsongs.js

596 lines
17 KiB
JavaScript
Raw Normal View History

2019-01-05 08:44:28 +01:00
class ImportSongs{
constructor(limited, otherFiles){
this.limited = limited
2019-01-05 08:44:28 +01:00
this.tjaFiles = []
this.osuFiles = []
2019-02-03 13:04:25 +01:00
this.assetFiles = {}
this.otherFiles = otherFiles || {}
2019-01-05 08:44:28 +01:00
this.songs = []
2019-02-03 13:04:25 +01:00
this.stylesheet = []
this.songTitle = this.otherFiles.songTitle || {}
this.uraRegex = /\s*[\(]裏[\)]$/
2019-01-05 08:44:28 +01:00
this.courseTypes = {
"easy": 0,
"normal": 1,
"hard": 2,
"oni": 3,
"ura": 4
}
this.categoryAliases = {}
assets.categories.forEach(cat => {
2020-04-27 17:47:55 +02:00
this.categoryAliases[cat.title.toLowerCase()] = cat.id
if(cat.aliases){
cat.aliases.forEach(alias => {
2020-04-27 17:47:55 +02:00
this.categoryAliases[alias.toLowerCase()] = cat.id
})
}
if(cat.title_lang){
for(var i in cat.title_lang){
this.categoryAliases[cat.title_lang[i].toLowerCase()] = cat.id
}
2019-01-25 02:42:05 +01:00
}
2020-04-27 17:47:55 +02:00
})
2019-02-03 13:04:25 +01:00
this.assetSelectors = {
"bg-pattern-1": ".pattern-bg",
"bg_genre_0": "#song-select",
"title-screen": "#title-screen",
"dancing-don": "#loading-don",
"touch_drum": "#touch-drum-img",
"touch_fullscreen": "#touch-full-btn",
"touch_pause": "#touch-pause-btn",
"bg_stage_1": ".song-stage-1",
"bg_stage_2": ".song-stage-2",
"bg_stage_3": ".song-stage-3"
}
this.comboVoices = ["v_combo_50"].concat(Array.from(Array(50), (d, i) => "v_combo_" + ((i + 1) * 100)))
}
load(files){
var extensionRegex = /\.[^\/]+$/
files.sort((a, b) => {
var path1 = a.path.replace(extensionRegex, "")
var path2 = b.path.replace(extensionRegex, "")
return path1 > path2 ? 1 : -1
})
2019-01-05 08:44:28 +01:00
var metaFiles = []
2019-01-05 08:44:28 +01:00
for(var i = 0; i < files.length; i++){
var file = files[i]
var name = file.name.toLowerCase()
var path = file.path.toLowerCase()
2019-01-05 08:44:28 +01:00
if(name.endsWith(".tja")){
this.tjaFiles.push({
file: file,
index: i
})
}else if(name.endsWith(".osu")){
this.osuFiles.push({
file: file,
index: i
})
}else if(!this.limited && (name === "genre.ini" || name === "box.def") || name === "songtitle.txt"){
var level = (file.path.match(/\//g) || []).length
2019-01-05 08:44:28 +01:00
metaFiles.push({
file: file,
level: (level * 2) + (name === "genre.ini" ? 1 : 0)
})
}else if(!this.limited && (path.indexOf("/taiko-web assets/") !== -1 || path.indexOf("taiko-web assets/") === 0)){
2019-02-03 13:04:25 +01:00
if(!(name in this.assetFiles)){
this.assetFiles[name] = file
}
2019-01-05 08:44:28 +01:00
}else{
2019-02-03 13:04:25 +01:00
this.otherFiles[path] = file
2019-01-05 08:44:28 +01:00
}
}
var metaPromises = []
metaFiles.forEach(fileObj => {
metaPromises.push(this.addMeta(fileObj))
})
return Promise.all(metaPromises).then(() => {
2019-01-05 08:44:28 +01:00
var songPromises = []
this.tjaFiles.forEach(fileObj => {
songPromises.push(this.addTja(fileObj).catch(e => console.warn(e)))
2019-01-05 08:44:28 +01:00
})
this.osuFiles.forEach(fileObj => {
songPromises.push(this.addOsu(fileObj).catch(e => console.warn(e)))
2019-01-05 08:44:28 +01:00
})
2019-02-03 13:04:25 +01:00
songPromises.push(this.addAssets())
return Promise.all(songPromises)
}).then(this.loaded.bind(this))
2019-01-05 08:44:28 +01:00
}
addMeta(fileObj){
var file = fileObj.file
var level = fileObj.level
var name = file.name.toLowerCase()
return file.read(name === "songtitle.txt" ? undefined : "sjis").then(data => {
var data = data.replace(/\0/g, "").split("\n")
2019-01-05 08:44:28 +01:00
var category
if(name === "genre.ini"){
var key
for(var i = 0; i < data.length; i++){
var line = data[i].trim().toLowerCase()
if(line.startsWith("[") && line.endsWith("]")){
key = line.slice(1, -1)
}else if(key === "genre"){
var equalsPos = line.indexOf("=")
if(equalsPos !== -1 && line.slice(0, equalsPos).trim() === "genrename"){
var value = line.slice(equalsPos + 1).trim()
2020-04-27 17:47:55 +02:00
if(value.toLowerCase() in this.categoryAliases){
category = value
}else{
category = data[i].trim().slice(equalsPos + 1).trim()
}
2019-01-05 08:44:28 +01:00
break
}
}
}
}else if(name === "box.def"){
for(var i = 0; i < data.length; i++){
var line = data[i].trim().toLowerCase()
if(line.startsWith("#title:")){
var value = line.slice(7).trim()
2020-04-27 17:47:55 +02:00
if(value.toLowerCase() in this.categoryAliases){
category = value
2019-01-05 08:44:28 +01:00
}
}else if(line.startsWith("#genre:")){
var value = line.slice(7).trim()
2020-04-27 17:47:55 +02:00
if(value.toLowerCase() in this.categoryAliases){
category = value
}else{
category = data[i].trim().slice(7).trim()
}
2019-01-05 08:44:28 +01:00
break
}
}
}else if(name === "songtitle.txt"){
var lastTitle
for(var i = 0; i < data.length; i++){
var line = data[i].trim()
if(line){
var lang = line.slice(0, 2)
if(line.charAt(2) !== " " || !(lang in allStrings)){
this.songTitle[line] = {}
lastTitle = line
}else if(lastTitle){
this.songTitle[lastTitle][lang] = line.slice(3).trim()
}
}
}
2019-01-05 08:44:28 +01:00
}
if(category){
var metaPath = file.path.toLowerCase().slice(0, file.name.length * -1)
var filesLoop = fileObj => {
var tjaPath = fileObj.file.path.toLowerCase().slice(0, fileObj.file.name.length * -1)
2019-01-05 08:44:28 +01:00
if(tjaPath.startsWith(metaPath) && (!("categoryLevel" in fileObj) || fileObj.categoryLevel < level)){
2020-04-27 17:47:55 +02:00
if(category.toLowerCase() in this.categoryAliases){
fileObj.category_id = this.categoryAliases[category.toLowerCase()]
}else{
fileObj.category = category
}
2019-01-05 08:44:28 +01:00
fileObj.categoryLevel = level
}
}
this.tjaFiles.forEach(filesLoop)
this.osuFiles.forEach(filesLoop)
2019-01-05 08:44:28 +01:00
}
}).catch(e => {
console.warn(e)
})
2019-01-05 08:44:28 +01:00
}
addTja(fileObj){
var file = fileObj.file
var index = fileObj.index
var category = fileObj.category
2020-04-27 17:47:55 +02:00
var category_id = fileObj.category_id
if(!this.limited){
var filePromise = file.read("sjis")
}else{
var filePromise = Promise.resolve()
}
return filePromise.then(dataRaw => {
var data = dataRaw ? dataRaw.replace(/\0/g, "").split("\n") : []
var tja = new ParseTja(data, "oni", 0, 0, true)
2019-01-05 08:44:28 +01:00
var songObj = {
id: index + 1,
2020-03-13 03:34:54 +01:00
order: index + 1,
title: file.name.slice(0, file.name.lastIndexOf(".")),
2019-01-05 08:44:28 +01:00
type: "tja",
chart: file,
2020-03-14 10:44:29 +01:00
courses: {},
music: "muted",
custom: true
}
if(this.limited){
songObj.unloaded = true
2019-01-05 08:44:28 +01:00
}
2020-03-13 03:34:54 +01:00
var coursesAdded = false
var titleLang = {}
2020-03-13 03:34:54 +01:00
var titleLangAdded = false
var subtitleLangAdded = false
var subtitleLang = {}
var dir = file.path.toLowerCase()
2019-01-05 08:44:28 +01:00
dir = dir.slice(0, dir.lastIndexOf("/") + 1)
for(var diff in tja.metadata){
var meta = tja.metadata[diff]
songObj.title = meta.title || file.name.slice(0, file.name.lastIndexOf("."))
2019-01-05 08:44:28 +01:00
var subtitle = meta.subtitle || ""
if(subtitle.startsWith("--") || subtitle.startsWith("++")){
subtitle = subtitle.slice(2).trim()
2019-01-05 08:44:28 +01:00
}
songObj.subtitle = subtitle
2019-02-20 23:42:18 +01:00
songObj.preview = meta.demostart || 0
2020-03-13 03:34:54 +01:00
songObj.courses[diff] = {
stars: meta.level || 0,
branch: !!meta.branch
}
coursesAdded = true
2019-01-05 08:44:28 +01:00
if(meta.wave){
songObj.music = this.otherFiles[dir + meta.wave.toLowerCase()] || songObj.music
2019-01-05 08:44:28 +01:00
}
if(meta.genre){
2020-04-27 17:47:55 +02:00
if(meta.genre.toLowerCase() in this.categoryAliases){
songObj.category_id = this.categoryAliases[meta.genre.toLowerCase()]
}else{
songObj.category = meta.genre
}
2019-01-05 08:44:28 +01:00
}
2019-02-03 13:04:25 +01:00
if(meta.taikowebskin){
songObj.song_skin = this.getSkin(dir, meta.taikowebskin)
}
2019-11-25 02:04:59 +01:00
if(meta.maker){
var maker = meta.maker
var url = null
var gt = maker.lastIndexOf(">")
if(gt === maker.length - 1){
var lt = maker.lastIndexOf("<")
if(lt !== -1 && lt !== gt - 2){
url = maker.slice(lt + 2, gt)
if(url.startsWith("http://") || url.startsWith("https://")){
maker = maker.slice(0, lt).trim()
}else{
url = null
}
}
}
songObj.maker = {
name: maker,
url: url,
id: 1
}
2019-11-25 02:04:59 +01:00
}
if(meta.lyrics){
var lyricsFile = this.normPath(this.joinPath(dir, meta.lyrics))
if(lyricsFile in this.otherFiles){
songObj.lyrics = true
songObj.lyricsFile = this.otherFiles[lyricsFile]
}
}else if(meta.inlineLyrics){
songObj.lyrics = true
}
for(var id in allStrings){
var songTitle = songObj.title
var ura = ""
if(songTitle){
var uraPos = songTitle.search(this.uraRegex)
if(uraPos !== -1){
ura = songTitle.slice(uraPos)
songTitle = songTitle.slice(0, uraPos)
}
}
if(meta["title" + id]){
titleLang[id] = meta["title" + id]
2020-03-13 03:34:54 +01:00
titleLangAdded = true
}else if(songTitle in this.songTitle && this.songTitle[songTitle][id]){
titleLang[id] = this.songTitle[songTitle][id] + ura
2020-03-13 03:34:54 +01:00
titleLangAdded = true
}
if(meta["subtitle" + id]){
subtitleLang[id] = meta["subtitle" + id]
2020-03-13 03:34:54 +01:00
subtitleLangAdded = true
}
}
}
2020-03-13 03:34:54 +01:00
if(titleLangAdded){
songObj.title_lang = titleLang
}
2020-03-13 03:34:54 +01:00
if(subtitleLangAdded){
songObj.subtitle_lang = subtitleLang
2019-01-05 08:44:28 +01:00
}
2020-04-27 17:47:55 +02:00
if(!songObj.category_id && !songObj.category){
if(!category && category_id === undefined){
songObj.category_id = this.getCategory(file, [songTitle || songObj.title, file.name.slice(0, file.name.lastIndexOf("."))])
}else if(category){
songObj.category = category
songObj.orginalCategory = category
}else{
songObj.category_id = category_id
}
2019-01-05 08:44:28 +01:00
}
if(coursesAdded || songObj.unloaded){
2019-01-05 08:44:28 +01:00
this.songs[index] = songObj
}
if(!this.limited){
var hash = md5.base64(dataRaw).slice(0, -2)
songObj.hash = hash
scoreStorage.songTitles[songObj.title] = hash
var score = scoreStorage.get(hash, false, true)
if(score){
score.title = songObj.title
}
}
})
2019-01-05 08:44:28 +01:00
}
addOsu(fileObj){
var file = fileObj.file
var index = fileObj.index
var category = fileObj.category
2020-04-27 17:47:55 +02:00
var category_id = fileObj.category_id
if(!this.limited){
var filePromise = file.read()
}else{
var filePromise = Promise.resolve()
}
return filePromise.then(dataRaw => {
var data = dataRaw ? dataRaw.replace(/\0/g, "").split("\n") : []
var osu = new ParseOsu(data, "oni", 0, 0, true)
var dir = file.path.toLowerCase()
2019-01-05 08:44:28 +01:00
dir = dir.slice(0, dir.lastIndexOf("/") + 1)
var songObj = {
id: index + 1,
2020-03-13 03:34:54 +01:00
order: index + 1,
2019-01-05 08:44:28 +01:00
type: "osu",
chart: file,
2019-01-05 08:44:28 +01:00
subtitle: osu.metadata.ArtistUnicode || osu.metadata.Artist,
2020-03-13 03:34:54 +01:00
subtitle_lang: {
en: osu.metadata.Artist || osu.metadata.ArtistUnicode
},
preview: osu.generalInfo.PreviewTime ? osu.generalInfo.PreviewTime / 1000 : 0,
courses: {},
music: (osu.generalInfo.AudioFilename ? this.otherFiles[dir + osu.generalInfo.AudioFilename.toLowerCase()] : "") || "muted"
}
if(!this.limited){
songObj.courses.oni = {
stars: parseInt(osu.difficulty.overallDifficulty) || 0,
branch: false
}
}else{
songObj.unloaded = true
2019-01-05 08:44:28 +01:00
}
var filename = file.name.slice(0, file.name.lastIndexOf("."))
var title = osu.metadata.TitleUnicode || osu.metadata.Title || file.name.slice(0, file.name.lastIndexOf("."))
2019-01-05 08:44:28 +01:00
if(title){
var suffix = ""
var matches = filename.match(/\[.+?\]$/)
if(matches){
suffix = " " + matches[0]
}
songObj.title = title + suffix
2020-03-13 03:34:54 +01:00
songObj.title_lang = {
en: (osu.metadata.Title || osu.metadata.TitleUnicode || title) + suffix
2020-03-13 03:34:54 +01:00
}
2019-01-05 08:44:28 +01:00
}else{
songObj.title = filename
}
this.songs[index] = songObj
2020-04-27 17:47:55 +02:00
if(!category && category_id === undefined){
songObj.category_id = this.getCategory(file, [osu.metadata.TitleUnicode, osu.metadata.Title, file.name.slice(0, file.name.lastIndexOf("."))])
}else if(category){
songObj.category = category
songObj.orginalCategory = category
}else{
songObj.category_id = category_id
}
if(!this.limited){
var hash = md5.base64(dataRaw).slice(0, -2)
songObj.hash = hash
scoreStorage.songTitles[songObj.title] = hash
var score = scoreStorage.get(hash, false, true)
if(score){
score.title = songObj.title
}
}
})
2019-01-05 08:44:28 +01:00
}
2019-02-03 13:04:25 +01:00
addAssets(){
var promises = []
for(let name in this.assetFiles){
let id = this.getFilename(name)
var file = this.assetFiles[name]
var index = name.lastIndexOf(".")
if(name === "vectors.json"){
promises.push(file.read().then(response => {
vectors = JSON.parse(response)
}))
2019-02-03 13:04:25 +01:00
}
if(name.endsWith(".png")){
let image = document.createElement("img")
promises.push(pageEvents.load(image).then(() => {
if(id in this.assetSelectors){
var selector = this.assetSelectors[id]
this.stylesheet.push(selector + '{background-image:url("' + image.src + '")}')
}
}))
image.id = name
promises.push(file.blob().then(blob => {
image.src = URL.createObjectURL(blob)
}))
loader.assetsDiv.appendChild(image)
var oldImage = assets.image[id]
if(oldImage && oldImage.parentNode){
URL.revokeObjectURL(oldImage.src)
oldImage.parentNode.removeChild(oldImage)
}
assets.image[id] = image
}
if(assets.audioSfx.indexOf(name) !== -1){
assets.sounds[id].clean()
promises.push(this.loadSound(file, name, snd.sfxGain))
}
if(assets.audioMusic.indexOf(name) !== -1){
assets.sounds[id].clean()
promises.push(this.loadSound(file, name, snd.musicGain))
}
if(assets.audioSfxLR.indexOf(name) !== -1){
assets.sounds[id + "_p1"].clean()
assets.sounds[id + "_p2"].clean()
promises.push(this.loadSound(file, name, snd.sfxGain).then(sound => {
assets.sounds[id + "_p1"] = assets.sounds[id].copy(snd.sfxGainL)
assets.sounds[id + "_p2"] = assets.sounds[id].copy(snd.sfxGainR)
}))
}
if(assets.audioSfxLoud.indexOf(name) !== -1){
assets.sounds[id].clean()
promises.push(this.loadSound(file, name, snd.sfxLoudGain))
}
if(this.comboVoices.indexOf(id) !== -1){
promises.push(snd.sfxGain.load(file).then(sound => {
assets.sounds[id] = sound
assets.sounds[id + "_p1"] = assets.sounds[id].copy(snd.sfxGainL)
assets.sounds[id + "_p2"] = assets.sounds[id].copy(snd.sfxGainR)
}))
}
}
return Promise.all(promises)
2019-02-03 13:04:25 +01:00
}
loadSound(file, name, gain){
var id = this.getFilename(name)
return gain.load(file).then(sound => {
2019-02-03 13:04:25 +01:00
assets.sounds[id] = sound
})
}
getFilename(name){
return name.slice(0, name.lastIndexOf("."))
}
getCategory(file, exclude){
var path = file.path.toLowerCase().split("/")
2019-01-05 08:44:28 +01:00
for(var i = path.length - 2; i >= 0; i--){
var hasTitle = false
for(var j in exclude){
2020-03-25 22:56:49 +01:00
if(exclude[j] && path[i].indexOf(exclude[j].toLowerCase()) !== -1){
hasTitle = true
break
}
}
if(!hasTitle){
for(var cat in this.categoryAliases){
if(path[i].indexOf(cat) !== -1){
return this.categoryAliases[cat]
}
2019-01-05 08:44:28 +01:00
}
}
}
}
2019-02-03 13:04:25 +01:00
getSkin(dir, config){
var configArray = config.toLowerCase().split(",")
var configObj = {}
for(var i in configArray){
var string = configArray[i].trim()
var space = string.indexOf(" ")
if(space !== -1){
configObj[string.slice(0, space).trim()] = string.slice(space + 1).trim()
}
}
if(!configObj.dir){
configObj.dir = ""
}
configObj.prefix = "custom "
var skinnable = ["song", "stage", "don"]
for(var i in skinnable){
var skinName = skinnable[i]
var skinValue = configObj[skinName]
if(skinValue && skinValue !== "none"){
var fileName = "bg_" + skinName + "_" + configObj.name
var skinPath = this.joinPath(dir, configObj.dir, fileName)
for(var j = 0; j < 2; j++){
if(skinValue !== "static"){
var suffix = (j === 0 ? "_a" : "_b") + ".png"
}else{
var suffix = ".png"
}
var skinFull = this.normPath(skinPath + suffix)
if(skinFull in this.otherFiles){
configObj[fileName + suffix] = this.otherFiles[skinFull]
}else{
configObj[skinName] = null
}
if(skinValue === "static"){
break
}
}
}
}
return configObj
}
2019-01-05 08:44:28 +01:00
loaded(){
this.songs = this.songs.filter(song => typeof song !== "undefined")
2019-02-03 13:04:25 +01:00
if(this.stylesheet.length){
var style = document.createElement("style")
style.appendChild(document.createTextNode(this.stylesheet.join("\n")))
document.head.appendChild(style)
}
2019-01-05 08:44:28 +01:00
if(this.songs.length){
if(this.limited){
assets.otherFiles = this.otherFiles
assets.otherFiles.songTitle = this.songTitle
}
return Promise.resolve(this.songs)
}else if(Object.keys(this.assetFiles).length){
return Promise.resolve()
2019-01-05 08:44:28 +01:00
}else{
return Promise.reject("nosongs")
2019-01-05 08:44:28 +01:00
}
this.clean()
2019-01-05 08:44:28 +01:00
}
2019-02-03 13:04:25 +01:00
joinPath(){
var resultPath = arguments[0]
for(var i = 1; i < arguments.length; i++){
var pPath = arguments[i]
if(pPath && (pPath[0] === "/" || pPath[0] === "\\")){
resultPath = pPath
}else{
var lastChar = resultPath.slice(-1)
if(resultPath && (lastChar !== "/" || lastChar !== "\\")){
resultPath = resultPath + "/"
}
resultPath = resultPath + pPath
}
}
return resultPath
}
normPath(path){
path = path.replace(/\\/g, "/").toLowerCase()
while(path[0] === "/"){
path = path.slice(1)
}
var comps = path.split("/")
for(var i = 0; i < comps.length; i++){
if(comps[i] === "." || comps[i] === ""){
comps.splice(i, 1)
i--
}else if(i !== 0 && comps[i] === ".." && comps[i - 1] !== ".."){
comps.splice(i - 1, 2)
i -= 2
}
}
return comps.join("/")
}
2019-01-05 08:44:28 +01:00
clean(){
delete this.songs
delete this.tjaFiles
delete this.osuFiles
delete this.assetFiles
2019-01-05 08:44:28 +01:00
delete this.otherFiles
}
}