2019-07-03 00:08:06 +08:00
|
|
|
import Logger from "js-logger";
|
2019-06-22 07:12:39 +08:00
|
|
|
import { autorun, IReactionDisposer, observable } from "mobx";
|
2019-06-22 06:27:04 +08:00
|
|
|
import { HuntMethod, NpcType, Server, SimpleQuest } from "../domain";
|
|
|
|
import { QuestDto } from "../dto";
|
2019-06-04 03:41:18 +08:00
|
|
|
import { Loadable } from "../Loadable";
|
2019-06-04 23:01:51 +08:00
|
|
|
import { ServerMap } from "./ServerMap";
|
2019-06-22 02:06:55 +08:00
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
const logger = Logger.get("stores/HuntMethodStore");
|
2019-06-01 22:02:06 +08:00
|
|
|
|
|
|
|
class HuntMethodStore {
|
2019-07-03 00:08:06 +08:00
|
|
|
@observable methods: ServerMap<Loadable<Array<HuntMethod>>> = new ServerMap(
|
|
|
|
server => new Loadable([], () => this.load_hunt_methods(server))
|
2019-06-04 03:41:18 +08:00
|
|
|
);
|
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
private storage_disposer?: IReactionDisposer;
|
2019-06-22 07:12:39 +08:00
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
private async load_hunt_methods(server: Server): Promise<HuntMethod[]> {
|
2019-06-04 23:01:51 +08:00
|
|
|
const response = await fetch(
|
2019-06-22 06:27:04 +08:00
|
|
|
`${process.env.PUBLIC_URL}/quests.${Server[server].toLowerCase()}.json`
|
2019-06-04 23:01:51 +08:00
|
|
|
);
|
2019-07-03 00:08:06 +08:00
|
|
|
const quests = (await response.json()) as QuestDto[];
|
2019-06-22 20:58:57 +08:00
|
|
|
const methods = new Array<HuntMethod>();
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-22 20:58:57 +08:00
|
|
|
for (const quest of quests) {
|
2019-07-02 23:00:24 +08:00
|
|
|
let total_count = 0;
|
|
|
|
const enemy_counts = new Map<NpcType, number>();
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-22 06:27:04 +08:00
|
|
|
for (const [code, count] of Object.entries(quest.enemyCounts)) {
|
2019-07-02 23:00:24 +08:00
|
|
|
const npc_type = NpcType.by_code(code);
|
2019-06-07 02:30:14 +08:00
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
if (!npc_type) {
|
2019-06-22 06:27:04 +08:00
|
|
|
logger.error(`No NpcType found for code ${code}.`);
|
2019-06-07 02:30:14 +08:00
|
|
|
} else {
|
2019-07-02 23:00:24 +08:00
|
|
|
enemy_counts.set(npc_type, count);
|
|
|
|
total_count += count;
|
2019-06-07 02:30:14 +08:00
|
|
|
}
|
2019-06-22 06:27:04 +08:00
|
|
|
}
|
2019-06-07 02:30:14 +08:00
|
|
|
|
2019-06-22 20:58:57 +08:00
|
|
|
// Filter out some quests.
|
2019-06-22 23:05:42 +08:00
|
|
|
/* eslint-disable no-fallthrough */
|
2019-06-22 20:58:57 +08:00
|
|
|
switch (quest.id) {
|
|
|
|
// The following quests are left out because their enemies don't drop anything.
|
2019-07-03 00:08:06 +08:00
|
|
|
case 31: // Black Paper's Dangerous Deal
|
|
|
|
case 34: // Black Paper's Dangerous Deal 2
|
2019-06-22 20:58:57 +08:00
|
|
|
case 1305: // Maximum Attack S (Ep. 1)
|
|
|
|
case 1306: // Maximum Attack S (Ep. 2)
|
|
|
|
case 1307: // Maximum Attack S (Ep. 4)
|
2019-07-03 00:08:06 +08:00
|
|
|
case 313: // Beyond the Horizon
|
2019-06-22 20:58:57 +08:00
|
|
|
|
|
|
|
// MAXIMUM ATTACK 3 Ver2 is filtered out because its actual enemy count depends on the path taken.
|
|
|
|
// TODO: generate a method per path.
|
|
|
|
case 314:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
methods.push(
|
|
|
|
new HuntMethod(
|
|
|
|
`q${quest.id}`,
|
2019-06-22 06:27:04 +08:00
|
|
|
quest.name,
|
2019-07-03 00:08:06 +08:00
|
|
|
new SimpleQuest(quest.id, quest.name, quest.episode, enemy_counts),
|
|
|
|
/^\d-\d.*/.test(quest.name) ? 0.75 : total_count > 400 ? 0.75 : 0.5
|
2019-06-22 20:58:57 +08:00
|
|
|
)
|
2019-06-07 02:30:14 +08:00
|
|
|
);
|
2019-06-22 20:58:57 +08:00
|
|
|
}
|
2019-06-22 07:12:39 +08:00
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
this.load_from_local_storage(methods, server);
|
2019-06-22 07:12:39 +08:00
|
|
|
return methods;
|
|
|
|
}
|
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
private load_from_local_storage = (methods: HuntMethod[], server: Server) => {
|
2019-06-22 07:12:39 +08:00
|
|
|
try {
|
2019-07-02 23:00:24 +08:00
|
|
|
const method_user_times_json = localStorage.getItem(
|
2019-06-22 07:12:39 +08:00
|
|
|
`HuntMethodStore.methodUserTimes.${Server[server]}`
|
|
|
|
);
|
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
if (method_user_times_json) {
|
|
|
|
const user_times: StoredUserTimes = JSON.parse(method_user_times_json);
|
2019-06-22 07:12:39 +08:00
|
|
|
|
|
|
|
for (const method of methods) {
|
2019-07-02 23:00:24 +08:00
|
|
|
method.user_time = user_times[method.id];
|
2019-06-22 07:12:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
if (this.storage_disposer) {
|
|
|
|
this.storage_disposer();
|
2019-06-22 07:12:39 +08:00
|
|
|
}
|
|
|
|
|
2019-07-03 00:08:06 +08:00
|
|
|
this.storage_disposer = autorun(() => this.store_in_local_storage(methods, server));
|
2019-06-22 07:12:39 +08:00
|
|
|
} catch (e) {
|
|
|
|
logger.error(e);
|
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-06-22 07:12:39 +08:00
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
private store_in_local_storage = (methods: HuntMethod[], server: Server) => {
|
2019-06-22 07:12:39 +08:00
|
|
|
try {
|
2019-07-02 23:00:24 +08:00
|
|
|
const user_times: StoredUserTimes = {};
|
2019-06-22 07:12:39 +08:00
|
|
|
|
|
|
|
for (const method of methods) {
|
2019-07-02 23:00:24 +08:00
|
|
|
if (method.user_time != undefined) {
|
|
|
|
user_times[method.id] = method.user_time;
|
2019-06-22 07:12:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
localStorage.setItem(
|
|
|
|
`HuntMethodStore.methodUserTimes.${Server[server]}`,
|
2019-07-02 23:00:24 +08:00
|
|
|
JSON.stringify(user_times)
|
2019-06-22 07:12:39 +08:00
|
|
|
);
|
|
|
|
} catch (e) {
|
|
|
|
logger.error(e);
|
|
|
|
}
|
2019-07-03 00:08:06 +08:00
|
|
|
};
|
2019-06-01 22:02:06 +08:00
|
|
|
}
|
|
|
|
|
2019-07-02 23:00:24 +08:00
|
|
|
type StoredUserTimes = { [method_id: string]: number };
|
|
|
|
|
|
|
|
export const hunt_method_store = new HuntMethodStore();
|