mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
Hunt optimizer is working but isn't very user-friendly yet.
This commit is contained in:
parent
63a703e708
commit
45c9df039a
@ -324,6 +324,10 @@ export class EnemyDrop implements ItemDrop {
|
|||||||
|
|
||||||
export class HuntMethod {
|
export class HuntMethod {
|
||||||
constructor(
|
constructor(
|
||||||
|
/**
|
||||||
|
* Time taken in hours.
|
||||||
|
*/
|
||||||
|
public time: number,
|
||||||
public quest: SimpleQuest
|
public quest: SimpleQuest
|
||||||
) { }
|
) { }
|
||||||
}
|
}
|
||||||
|
@ -20,19 +20,13 @@ export class EnumMap<K, V> {
|
|||||||
private keys: K[];
|
private keys: K[];
|
||||||
private values = new Map<K, V>();
|
private values = new Map<K, V>();
|
||||||
|
|
||||||
constructor(enum_: any, initialValue: V | ((key: K) => V)) {
|
constructor(enum_: any, initialValue: (key: K) => V) {
|
||||||
this.keys = enumValues(enum_);
|
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) {
|
for (const key of this.keys) {
|
||||||
this.values.set(key, initialValue(key));
|
this.values.set(key, initialValue(key));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
get(key: K): V {
|
get(key: K): V {
|
||||||
return this.values.get(key)!;
|
return this.values.get(key)!;
|
||||||
|
@ -20,8 +20,21 @@ class HuntMethodStore {
|
|||||||
return NpcType.byNameAndEpisode(enemy, parseInt(episode, 10))!;
|
return NpcType.byNameAndEpisode(enemy, parseInt(episode, 10))!;
|
||||||
});
|
});
|
||||||
|
|
||||||
return rows.slice(2).map(row => {
|
return rows.slice(2)
|
||||||
|
.filter(row => {
|
||||||
const questName = row[0];
|
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 npcs = row.slice(2, -2).flatMap((cell, cellI) => {
|
||||||
const amount = parseInt(cell, 10);
|
const amount = parseInt(cell, 10);
|
||||||
@ -32,12 +45,15 @@ class HuntMethodStore {
|
|||||||
for (let i = 0; i < amount; i++) {
|
for (let i = 0; i < amount; i++) {
|
||||||
enemies.push(new SimpleNpc(type));
|
enemies.push(new SimpleNpc(type));
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
console.error(`Couldn't get type for cellI ${cellI}.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return enemies;
|
return enemies;
|
||||||
});
|
});
|
||||||
|
|
||||||
return new HuntMethod(
|
return new HuntMethod(
|
||||||
|
time,
|
||||||
new SimpleQuest(
|
new SimpleQuest(
|
||||||
questName,
|
questName,
|
||||||
npcs
|
npcs
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
import solver from 'javascript-lp-solver';
|
import solver from 'javascript-lp-solver';
|
||||||
import { observable } from "mobx";
|
import { IObservableArray, observable, runInAction } from "mobx";
|
||||||
import { Difficulties, Item, NpcType, SectionIds } from "../domain";
|
import { Difficulties, Difficulty, Item, NpcType, SectionId, SectionIds } from "../domain";
|
||||||
import { huntMethodStore } from "./HuntMethodStore";
|
import { huntMethodStore } from "./HuntMethodStore";
|
||||||
import { itemDropStore } from './ItemDropStore';
|
import { itemDropStore } from './ItemDropStore';
|
||||||
|
|
||||||
export class WantedItem {
|
export class WantedItem {
|
||||||
@observable item: Item;
|
@observable readonly item: Item;
|
||||||
@observable amount: number;
|
@observable amount: number;
|
||||||
|
|
||||||
constructor(item: Item, 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<Item, number>
|
||||||
|
) {
|
||||||
|
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 {
|
class HuntOptimizerStore {
|
||||||
@observable wantedItems: Array<WantedItem> = [];
|
@observable readonly wantedItems: Array<WantedItem> = [];
|
||||||
|
@observable readonly result: IObservableArray<OptimizationResult> = observable.array();
|
||||||
|
|
||||||
optimize = async () => {
|
optimize = async () => {
|
||||||
if (!this.wantedItems.length) return;
|
if (!this.wantedItems.length) return;
|
||||||
@ -23,14 +42,23 @@ class HuntOptimizerStore {
|
|||||||
const methods = await huntMethodStore.methods.current.promise;
|
const methods = await huntMethodStore.methods.current.promise;
|
||||||
const dropTable = await itemDropStore.enemyDrops.current.promise;
|
const dropTable = await itemDropStore.enemyDrops.current.promise;
|
||||||
|
|
||||||
|
// Add a constraint per wanted item.
|
||||||
const constraints: { [itemName: string]: { min: number } } = {};
|
const constraints: { [itemName: string]: { min: number } } = {};
|
||||||
|
|
||||||
for (const wanted of this.wantedItems) {
|
for (const wanted of this.wantedItems) {
|
||||||
constraints[wanted.item.name] = { min: wanted.amount };
|
constraints[wanted.item.name] = { min: wanted.amount };
|
||||||
}
|
}
|
||||||
|
|
||||||
const items = new Set(this.wantedItems.map(i => i.item));
|
// Add a variable to the LP model per method per difficulty per section ID.
|
||||||
const variables: { [methodName: string]: { [itemName: string]: number } } = {};
|
// 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) {
|
for (const method of methods) {
|
||||||
const counts = new Map<NpcType, number>();
|
const counts = new Map<NpcType, number>();
|
||||||
@ -42,40 +70,75 @@ class HuntOptimizerStore {
|
|||||||
|
|
||||||
for (const diff of Difficulties) {
|
for (const diff of Difficulties) {
|
||||||
for (const sectionId of SectionIds) {
|
for (const sectionId of SectionIds) {
|
||||||
const variable: { [itemName: string]: number } = {
|
const variable: Variable = {
|
||||||
time: 0.5
|
time: method.time
|
||||||
};
|
};
|
||||||
|
let addVariable = false;
|
||||||
|
|
||||||
for (const [npcType, count] of counts.entries()) {
|
for (const [npcType, count] of counts.entries()) {
|
||||||
const drop = dropTable.getDrop(diff, sectionId, npcType);
|
const drop = dropTable.getDrop(diff, sectionId, npcType);
|
||||||
|
|
||||||
if (drop && items.has(drop.item)) {
|
if (drop && wantedItems.has(drop.item)) {
|
||||||
variable[drop.item.name] = count * drop.rate;
|
const value = variable[drop.item.name] || 0;
|
||||||
|
variable[drop.item.name] = value + count * drop.rate;
|
||||||
|
addVariable = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Object.keys(variable).length) {
|
if (addVariable) {
|
||||||
variables[`${diff} ${sectionId} ${method.quest.name}`] = variable;
|
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',
|
optimize: 'time',
|
||||||
opType: 'min',
|
opType: 'min',
|
||||||
constraints,
|
constraints,
|
||||||
variables
|
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<Item, number>();
|
||||||
|
|
||||||
|
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();
|
export const huntOptimizerStore = new HuntOptimizerStore();
|
||||||
|
|
||||||
type MethodWithDropRates = {
|
|
||||||
name: string
|
|
||||||
time: number
|
|
||||||
[itemName: string]: any
|
|
||||||
}
|
|
||||||
|
@ -7,7 +7,7 @@ import { ServerMap } from "./ServerMap";
|
|||||||
|
|
||||||
class EnemyDropTable {
|
class EnemyDropTable {
|
||||||
private map: EnumMap<Difficulty, EnumMap<SectionId, Map<NpcType, EnemyDrop>>> =
|
private map: EnumMap<Difficulty, EnumMap<SectionId, Map<NpcType, EnemyDrop>>> =
|
||||||
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 {
|
getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop | undefined {
|
||||||
return this.map.get(difficulty).get(sectionId).get(npcType);
|
return this.map.get(difficulty).get(sectionId).get(npcType);
|
||||||
@ -74,7 +74,7 @@ class ItemDropStore {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const rareRate = parseFloat(cells[5]);
|
const rareRate = parseFloat(cells[6]);
|
||||||
|
|
||||||
if (!rareRate) {
|
if (!rareRate) {
|
||||||
console.error(`Couldn't parse rare_rate for line ${lineNo}.`);
|
console.error(`Couldn't parse rare_rate for line ${lineNo}.`);
|
||||||
|
@ -10,7 +10,7 @@ class ItemStore {
|
|||||||
new Loadable([], () => this.loadItems(server))
|
new Loadable([], () => this.loadItems(server))
|
||||||
);
|
);
|
||||||
|
|
||||||
dedupItem(name: string): Item {
|
dedupItem = (name: string): Item => {
|
||||||
let item = this.itemMap.get(name);
|
let item = this.itemMap.get(name);
|
||||||
|
|
||||||
if (!item) {
|
if (!item) {
|
||||||
@ -25,7 +25,7 @@ class ItemStore {
|
|||||||
`${process.env.PUBLIC_URL}/items.${Server[server].toLowerCase()}.tsv`
|
`${process.env.PUBLIC_URL}/items.${Server[server].toLowerCase()}.tsv`
|
||||||
);
|
);
|
||||||
const data = await response.text();
|
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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@ import { applicationStore } from "./ApplicationStore";
|
|||||||
import { EnumMap } from "../enums";
|
import { EnumMap } from "../enums";
|
||||||
|
|
||||||
export class ServerMap<V> extends EnumMap<Server, V> {
|
export class ServerMap<V> extends EnumMap<Server, V> {
|
||||||
constructor(initialValue: V | ((server: Server) => V)) {
|
constructor(initialValue: (server: Server) => V) {
|
||||||
super(Server, initialValue)
|
super(Server, initialValue)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,11 +1,13 @@
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import './HuntOptimizerComponent.css';
|
import './HuntOptimizerComponent.css';
|
||||||
import { WantedItemsComponent } from "./WantedItemsComponent";
|
import { WantedItemsComponent } from "./WantedItemsComponent";
|
||||||
|
import { OptimizationResultComponent } from "./OptimizationResultComponent";
|
||||||
|
|
||||||
export function HuntOptimizerComponent() {
|
export function HuntOptimizerComponent() {
|
||||||
return (
|
return (
|
||||||
<section className="ho-HuntOptimizerComponent">
|
<section className="ho-HuntOptimizerComponent">
|
||||||
<WantedItemsComponent />
|
<WantedItemsComponent />
|
||||||
|
<OptimizationResultComponent />
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
52
src/ui/hunt-optimizer/OptimizationResultComponent.tsx
Normal file
52
src/ui/hunt-optimizer/OptimizationResultComponent.tsx
Normal file
@ -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<Item>();
|
||||||
|
|
||||||
|
for (const r of huntOptimizerStore.result) {
|
||||||
|
for (const i of r.itemCounts.keys()) {
|
||||||
|
items.add(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<section>
|
||||||
|
<h2>Optimization Result</h2>
|
||||||
|
<Table
|
||||||
|
dataSource={huntOptimizerStore.result}
|
||||||
|
pagination={false}
|
||||||
|
rowKey={(_, index) => index.toString()}
|
||||||
|
size="small"
|
||||||
|
scroll={{ x: true, y: true }}
|
||||||
|
>
|
||||||
|
<Table.Column title="Difficulty" dataIndex="difficulty" />
|
||||||
|
<Table.Column title="Method" dataIndex="methodName" />
|
||||||
|
<Table.Column title="Section ID" dataIndex="sectionId" />
|
||||||
|
<Table.Column title="Hours/Run" dataIndex="methodTime" render={this.fixed1} />
|
||||||
|
<Table.Column title="Runs" dataIndex="runs" render={this.fixed1} />
|
||||||
|
<Table.Column title="Total Hours" dataIndex="totalTime" render={this.fixed1} />
|
||||||
|
{[...items].map(item =>
|
||||||
|
<Table.Column<OptimizationResult>
|
||||||
|
title={item.name}
|
||||||
|
key={item.name}
|
||||||
|
render={(_, result) => {
|
||||||
|
const count = result.itemCounts.get(item);
|
||||||
|
return count && count.toFixed(2);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Table>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private fixed1(time: number): string {
|
||||||
|
return time.toFixed(1);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user