diff --git a/src/domain/index.ts b/src/domain/index.ts index 2577803e..8ba6a48f 100644 --- a/src/domain/index.ts +++ b/src/domain/index.ts @@ -324,6 +324,10 @@ export class EnemyDrop implements ItemDrop { export class HuntMethod { constructor( + /** + * Time taken in hours. + */ + public time: number, public quest: SimpleQuest ) { } } diff --git a/src/enums.ts b/src/enums.ts index fcc03916..ba634ed7 100644 --- a/src/enums.ts +++ b/src/enums.ts @@ -20,17 +20,11 @@ export class EnumMap { private keys: K[]; private values = new Map(); - constructor(enum_: any, initialValue: V | ((key: K) => V)) { + constructor(enum_: any, initialValue: (key: K) => V) { this.keys = enumValues(enum_); - if (!(initialValue instanceof Function)) { - for (const key of this.keys) { - this.values.set(key, initialValue); - } - } else { - for (const key of this.keys) { - this.values.set(key, initialValue(key)); - } + for (const key of this.keys) { + this.values.set(key, initialValue(key)); } } diff --git a/src/stores/HuntMethodStore.ts b/src/stores/HuntMethodStore.ts index c80484c0..1b3b997d 100644 --- a/src/stores/HuntMethodStore.ts +++ b/src/stores/HuntMethodStore.ts @@ -20,30 +20,46 @@ class HuntMethodStore { return NpcType.byNameAndEpisode(enemy, parseInt(episode, 10))!; }); - return rows.slice(2).map(row => { - const questName = row[0]; - - 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)); - } + 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]); - return enemies; + 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 + ) + ); }); - - return new HuntMethod( - new SimpleQuest( - questName, - npcs - ) - ); - }); } } diff --git a/src/stores/HuntOptimizerStore.ts b/src/stores/HuntOptimizerStore.ts index f8ebdc76..543a0604 100644 --- a/src/stores/HuntOptimizerStore.ts +++ b/src/stores/HuntOptimizerStore.ts @@ -1,11 +1,11 @@ import solver from 'javascript-lp-solver'; -import { observable } from "mobx"; -import { Difficulties, Item, NpcType, SectionIds } from "../domain"; +import { IObservableArray, observable, runInAction } from "mobx"; +import { Difficulties, Difficulty, Item, NpcType, SectionId, SectionIds } from "../domain"; import { huntMethodStore } from "./HuntMethodStore"; import { itemDropStore } from './ItemDropStore'; export class WantedItem { - @observable item: Item; + @observable readonly item: Item; @observable amount: number; constructor(item: Item, amount: number) { @@ -14,8 +14,27 @@ export class WantedItem { } } +export class OptimizationResult { + public readonly totalTime: number; + + constructor( + public readonly difficulty: Difficulty, + public readonly sectionId: SectionId, + public readonly methodName: string, + public readonly methodTime: number, + public readonly runs: number, + public readonly itemCounts: Map + ) { + this.totalTime = runs * methodTime; + } +} + +// TODO: group similar methods (e.g. same difficulty, same quest and similar ID). +// This way people can choose their preferred section ID. +// TODO: Cutter doesn't seem to work. class HuntOptimizerStore { - @observable wantedItems: Array = []; + @observable readonly wantedItems: Array = []; + @observable readonly result: IObservableArray = observable.array(); optimize = async () => { if (!this.wantedItems.length) return; @@ -23,14 +42,23 @@ class HuntOptimizerStore { const methods = await huntMethodStore.methods.current.promise; const dropTable = await itemDropStore.enemyDrops.current.promise; + // Add a constraint per wanted item. const constraints: { [itemName: string]: { min: number } } = {}; for (const wanted of this.wantedItems) { constraints[wanted.item.name] = { min: wanted.amount }; } - const items = new Set(this.wantedItems.map(i => i.item)); - const variables: { [methodName: string]: { [itemName: string]: number } } = {}; + // Add a variable to the LP model per method per difficulty per section ID. + // Each variable has a time property to minimize and a property per item with the number + // of enemies that drop the item multiplied by the corresponding drop rate as its value. + type Variable = { + time: number, + [itemName: string]: number + } + const variables: { [methodName: string]: Variable } = {}; + + const wantedItems = new Set(this.wantedItems.map(i => i.item)); for (const method of methods) { const counts = new Map(); @@ -42,40 +70,75 @@ class HuntOptimizerStore { for (const diff of Difficulties) { for (const sectionId of SectionIds) { - const variable: { [itemName: string]: number } = { - time: 0.5 + const variable: Variable = { + time: method.time }; + let addVariable = false; for (const [npcType, count] of counts.entries()) { const drop = dropTable.getDrop(diff, sectionId, npcType); - if (drop && items.has(drop.item)) { - variable[drop.item.name] = count * drop.rate; + if (drop && wantedItems.has(drop.item)) { + const value = variable[drop.item.name] || 0; + variable[drop.item.name] = value + count * drop.rate; + addVariable = true; } } - if (Object.keys(variable).length) { - variables[`${diff} ${sectionId} ${method.quest.name}`] = variable; + if (addVariable) { + variables[`${diff}\t${sectionId}\t${method.quest.name}`] = variable; } } } } - const result = solver.Solve({ + const result: { + feasible: boolean, + bounded: boolean, + result: number, + [method: string]: number | boolean + } = solver.Solve({ optimize: 'time', opType: 'min', constraints, variables }); - console.log(result); + runInAction(() => { + this.result.splice(0); + + for (const [method, runsOrOther] of Object.entries(result)) { + const [diffStr, sIdStr, methodName] = method.split('\t', 3); + + if (sIdStr && methodName) { + const runs = runsOrOther as number; + const variable = variables[method]; + const diff = (Difficulty as any)[diffStr]; + const sectionId = (SectionId as any)[sIdStr]; + + const items = new Map(); + + for (const [itemName, expectedValue] of Object.entries(variable)) { + for (const item of wantedItems) { + if (itemName === item.name) { + items.set(item, runs * expectedValue); + break; + } + } + } + + this.result.push(new OptimizationResult( + diff, + sectionId, + methodName, + 0.5, + runs, + items + )); + } + } + }); } } export const huntOptimizerStore = new HuntOptimizerStore(); - -type MethodWithDropRates = { - name: string - time: number - [itemName: string]: any -} diff --git a/src/stores/ItemDropStore.ts b/src/stores/ItemDropStore.ts index 5fd99430..c5af1713 100644 --- a/src/stores/ItemDropStore.ts +++ b/src/stores/ItemDropStore.ts @@ -7,7 +7,7 @@ import { ServerMap } from "./ServerMap"; class EnemyDropTable { private map: EnumMap>> = - new EnumMap(Difficulty, new EnumMap(SectionId, new Map())); + new EnumMap(Difficulty, () => new EnumMap(SectionId, () => new Map())); getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop | undefined { return this.map.get(difficulty).get(sectionId).get(npcType); @@ -74,7 +74,7 @@ class ItemDropStore { continue; } - const rareRate = parseFloat(cells[5]); + const rareRate = parseFloat(cells[6]); if (!rareRate) { console.error(`Couldn't parse rare_rate for line ${lineNo}.`); diff --git a/src/stores/ItemStore.ts b/src/stores/ItemStore.ts index 63ad3eba..7e0e68c2 100644 --- a/src/stores/ItemStore.ts +++ b/src/stores/ItemStore.ts @@ -10,7 +10,7 @@ class ItemStore { new Loadable([], () => this.loadItems(server)) ); - dedupItem(name: string): Item { + dedupItem = (name: string): Item => { let item = this.itemMap.get(name); if (!item) { @@ -25,7 +25,7 @@ class ItemStore { `${process.env.PUBLIC_URL}/items.${Server[server].toLowerCase()}.tsv` ); const data = await response.text(); - return data.split('\n').slice(1).map(name => new Item(name)); + return data.split('\n').slice(1).map(name => this.dedupItem(name)); } } diff --git a/src/stores/ServerMap.ts b/src/stores/ServerMap.ts index fa10b5e3..d4f444bb 100644 --- a/src/stores/ServerMap.ts +++ b/src/stores/ServerMap.ts @@ -4,7 +4,7 @@ import { applicationStore } from "./ApplicationStore"; import { EnumMap } from "../enums"; export class ServerMap extends EnumMap { - constructor(initialValue: V | ((server: Server) => V)) { + constructor(initialValue: (server: Server) => V) { super(Server, initialValue) } diff --git a/src/ui/ApplicationComponent.tsx b/src/ui/ApplicationComponent.tsx index 77b6a31e..dc673996 100644 --- a/src/ui/ApplicationComponent.tsx +++ b/src/ui/ApplicationComponent.tsx @@ -42,7 +42,7 @@ export class ApplicationComponent extends React.Component { Hunt Optimizer - +
diff --git a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx b/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx index c1401ae2..2cd9ebf9 100644 --- a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx +++ b/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx @@ -1,11 +1,13 @@ import React from "react"; import './HuntOptimizerComponent.css'; import { WantedItemsComponent } from "./WantedItemsComponent"; +import { OptimizationResultComponent } from "./OptimizationResultComponent"; export function HuntOptimizerComponent() { return (
+
); } diff --git a/src/ui/hunt-optimizer/OptimizationResultComponent.tsx b/src/ui/hunt-optimizer/OptimizationResultComponent.tsx new file mode 100644 index 00000000..5a792843 --- /dev/null +++ b/src/ui/hunt-optimizer/OptimizationResultComponent.tsx @@ -0,0 +1,52 @@ +import { Table } from "antd"; +import { observer } from "mobx-react"; +import React from "react"; +import { Item } from "../../domain"; +import { huntOptimizerStore, OptimizationResult } from "../../stores/HuntOptimizerStore"; + +@observer +export class OptimizationResultComponent extends React.Component { + render() { + const items = new Set(); + + for (const r of huntOptimizerStore.result) { + for (const i of r.itemCounts.keys()) { + items.add(i); + } + } + + return ( +
+

Optimization Result

+ index.toString()} + size="small" + scroll={{ x: true, y: true }} + > + + + + + + + {[...items].map(item => + + title={item.name} + key={item.name} + render={(_, result) => { + const count = result.itemCounts.get(item); + return count && count.toFixed(2); + }} + /> + )} +
+
+ ); + } + + private fixed1(time: number): string { + return time.toFixed(1); + } +}