phantasmal-world/src/stores/HuntMethodStore.ts

67 lines
2.3 KiB
TypeScript
Raw Normal View History

import { observable } from "mobx";
import { HuntMethod, NpcType, Server, SimpleNpc, SimpleQuest } from "../domain";
import { Loadable } from "../Loadable";
import { ServerMap } from "./ServerMap";
class HuntMethodStore {
@observable methods: ServerMap<Loadable<Array<HuntMethod>>> = new ServerMap(server =>
new Loadable([], () => this.loadHuntMethods(server))
);
private async loadHuntMethods(server: Server): Promise<HuntMethod[]> {
const response = await fetch(
`${process.env.PUBLIC_URL}/quests.${Server[server].toLowerCase()}.tsv`
);
const data = await response.text();
const rows = data.split('\n').map(line => line.split('\t'));
const npcTypeByIndex = rows[0].slice(2, -2).map((episode, i) => {
const enemy = rows[1][i + 2];
return NpcType.byNameAndEpisode(enemy, parseInt(episode, 10))!;
});
return rows.slice(2)
.filter(row => {
const questName = row[0];
// TODO: let's not hard code this...
switch (questName) {
case 'MAXIMUM ATTACK 3 Ver2':
case 'LOGiN presents 勇場のマッチレース':
return false;
default:
return true;
}
})
.map(row => {
const questName = row[0];
const time = parseFloat(row[1]);
const npcs = row.slice(2, -2).flatMap((cell, cellI) => {
const amount = parseInt(cell, 10);
const type = npcTypeByIndex[cellI];
const enemies = [];
if (type) {
for (let i = 0; i < amount; i++) {
enemies.push(new SimpleNpc(type));
}
} else {
console.error(`Couldn't get type for cellI ${cellI}.`);
}
return enemies;
});
return new HuntMethod(
time,
new SimpleQuest(
questName,
npcs
)
);
});
}
}
export const huntMethodStore = new HuntMethodStore();