Items in the optimization result are now shown in the same order as the wanted list.

This commit is contained in:
Daan Vanden Bosch 2019-06-16 11:43:10 +02:00
parent 583ef859a3
commit 30464680cb
3 changed files with 136 additions and 114 deletions

3
.gitignore vendored
View File

@ -1,5 +1,8 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
# Editors
.vscode
# dependencies # dependencies
/node_modules /node_modules
/.pnp /.pnp

View File

@ -16,16 +16,23 @@ export class WantedItem {
} }
} }
export class OptimizationResult { export class OptimalResult {
public readonly totalTime: number; constructor(
readonly wantedItems: Array<Item>,
readonly optimalMethods: Array<OptimalMethod>
) { }
}
export class OptimalMethod {
readonly totalTime: number;
constructor( constructor(
public readonly difficulty: Difficulty, readonly difficulty: Difficulty,
public readonly sectionIds: Array<SectionId>, readonly sectionIds: Array<SectionId>,
public readonly methodName: string, readonly methodName: string,
public readonly methodTime: number, readonly methodTime: number,
public readonly runs: number, readonly runs: number,
public readonly itemCounts: Map<Item, number> readonly itemCounts: Map<Item, number>
) { ) {
this.totalTime = runs * methodTime; this.totalTime = runs * methodTime;
} }
@ -39,7 +46,7 @@ export class OptimizationResult {
// TODO: boxes. // TODO: boxes.
class HuntOptimizerStore { class HuntOptimizerStore {
@observable readonly wantedItems: IObservableArray<WantedItem> = observable.array(); @observable readonly wantedItems: IObservableArray<WantedItem> = observable.array();
@observable readonly results: IObservableArray<OptimizationResult> = observable.array(); @observable result?: OptimalResult;
constructor() { constructor() {
this.initialize(); this.initialize();
@ -92,10 +99,14 @@ class HuntOptimizerStore {
optimize = async () => { optimize = async () => {
if (!this.wantedItems.length) { if (!this.wantedItems.length) {
this.results.splice(0); this.result = undefined;
return; return;
} }
// Initialize this set before awaiting data, so user changes don't affect this optimization
// run from this point on.
const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.item));
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;
@ -125,8 +136,6 @@ class HuntOptimizerStore {
} }
const variableDetails: Map<string, VariableDetails> = new Map(); const variableDetails: Map<string, VariableDetails> = new Map();
const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.item));
for (const method of methods) { for (const method of methods) {
// Counts include rare enemies, so they are fractional. // Counts include rare enemies, so they are fractional.
const counts = new Map<NpcType, number>(); const counts = new Map<NpcType, number>();
@ -158,22 +167,25 @@ class HuntOptimizerStore {
// migiums and hidooms. // migiums and hidooms.
const countsList: Array<Map<NpcType, number>> = [counts]; const countsList: Array<Map<NpcType, number>> = [counts];
const panArmsCount = counts.get(NpcType.PanArms); const panArmsCount = counts.get(NpcType.PanArms);
const panArms2Count = counts.get(NpcType.PanArms2);
if (panArmsCount || panArms2Count) { if (panArmsCount) {
const splitCounts = new Map(counts); const splitCounts = new Map(counts);
if (panArmsCount) { splitCounts.delete(NpcType.PanArms);
splitCounts.delete(NpcType.PanArms); splitCounts.set(NpcType.Migium, panArmsCount);
splitCounts.set(NpcType.Migium, panArmsCount); splitCounts.set(NpcType.Hidoom, panArmsCount);
splitCounts.set(NpcType.Hidoom, panArmsCount);
}
if (panArms2Count) { countsList.push(splitCounts);
splitCounts.delete(NpcType.PanArms2); }
splitCounts.set(NpcType.Migium2, panArms2Count);
splitCounts.set(NpcType.Hidoom2, panArms2Count); const panArms2Count = counts.get(NpcType.PanArms2);
}
if (panArms2Count) {
const splitCounts = new Map(counts);
splitCounts.delete(NpcType.PanArms2);
splitCounts.set(NpcType.Migium2, panArms2Count);
splitCounts.set(NpcType.Hidoom2, panArms2Count);
countsList.push(splitCounts); countsList.push(splitCounts);
} }
@ -184,9 +196,12 @@ class HuntOptimizerStore {
for (const diff of Difficulties) { for (const diff of Difficulties) {
for (const sectionId of SectionIds) { for (const sectionId of SectionIds) {
// Will contain an entry per wanted item dropped by enemies in this method/
// difficulty/section ID combo.
const variable: Variable = { const variable: Variable = {
time: method.time time: method.time
}; };
// Only add the variable if the method provides at least 1 item we want.
let addVariable = false; let addVariable = false;
for (const [npcType, count] of counts.entries()) { for (const [npcType, count] of counts.entries()) {
@ -220,6 +235,9 @@ class HuntOptimizerStore {
feasible: boolean, feasible: boolean,
bounded: boolean, bounded: boolean,
result: number, result: number,
/**
* Value will always be a number if result is indexed with an actual method name.
*/
[method: string]: number | boolean [method: string]: number | boolean
} = solver.Solve({ } = solver.Solve({
optimize: 'time', optimize: 'time',
@ -228,73 +246,78 @@ class HuntOptimizerStore {
variables variables
}); });
runInAction(() => { if (!result.feasible) {
this.results.splice(0); this.result = undefined;
return;
}
if (!result.feasible) { const optimalMethods: Array<OptimalMethod> = [];
return;
}
for (const [variableName, runsOrOther] of Object.entries(result)) { // Loop over the entries in result, ignore standard properties that aren't variables.
const details = variableDetails.get(variableName); for (const [variableName, runsOrOther] of Object.entries(result)) {
const details = variableDetails.get(variableName);
if (details) { if (details) {
const { method, difficulty, sectionId, splitPanArms } = details; const { method, difficulty, sectionId, splitPanArms } = details;
const runs = runsOrOther as number; const runs = runsOrOther as number;
const variable = variables[variableName]; const variable = variables[variableName];
const items = new Map<Item, number>(); const items = new Map<Item, number>();
for (const [itemName, expectedAmount] of Object.entries(variable)) { for (const [itemName, expectedAmount] of Object.entries(variable)) {
for (const item of wantedItems) { for (const item of wantedItems) {
if (itemName === item.name) { if (itemName === item.name) {
items.set(item, runs * expectedAmount); items.set(item, runs * expectedAmount);
break; break;
}
} }
} }
}
// Find all section IDs that provide the same items with the same expected amount. // Find all section IDs that provide the same items with the same expected amount.
// E.g. if you need a spread needle and a bringer's right arm, using either // E.g. if you need a spread needle and a bringer's right arm, using either
// purplenum or yellowboze will give you the exact same probabilities. // purplenum or yellowboze will give you the exact same probabilities.
const sectionIds: Array<SectionId> = []; const sectionIds: Array<SectionId> = [];
for (const sid of SectionIds) { for (const sid of SectionIds) {
let matchFound = true; let matchFound = true;
if (sid !== sectionId) { if (sid !== sectionId) {
const v = variables[ const v = variables[
this.fullMethodName(difficulty, sid, method, splitPanArms) this.fullMethodName(difficulty, sid, method, splitPanArms)
]; ];
if (!v) { if (!v) {
matchFound = false; matchFound = false;
} else { } else {
for (const itemName of Object.keys(variable)) { for (const itemName of Object.keys(variable)) {
if (variable[itemName] !== v[itemName]) { if (variable[itemName] !== v[itemName]) {
matchFound = false; matchFound = false;
break; break;
}
} }
} }
} }
if (matchFound) {
sectionIds.push(sid);
}
} }
this.results.push(new OptimizationResult( if (matchFound) {
difficulty, sectionIds.push(sid);
sectionIds, }
method.name + (splitPanArms ? ' (Split Pan Arms)' : ''),
method.time,
runs,
items
));
} }
optimalMethods.push(new OptimalMethod(
difficulty,
sectionIds,
method.name + (splitPanArms ? ' (Split Pan Arms)' : ''),
method.time,
runs,
items
));
} }
}); }
this.result = new OptimalResult(
[...wantedItems],
optimalMethods
);
} }
private fullMethodName( private fullMethodName(

View File

@ -2,26 +2,26 @@ import { computed } from "mobx";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import React from "react"; import React from "react";
import { AutoSizer, Index } from "react-virtualized"; import { AutoSizer, Index } from "react-virtualized";
import { Item } from "../../domain"; import { huntOptimizerStore, OptimalMethod } from "../../stores/HuntOptimizerStore";
import { huntOptimizerStore, OptimizationResult } from "../../stores/HuntOptimizerStore";
import { Column, DataTable } from "../dataTable"; import { Column, DataTable } from "../dataTable";
import "./OptimizationResultComponent.less";
import { hoursToString } from "../time"; import { hoursToString } from "../time";
import "./OptimizationResultComponent.less";
@observer @observer
export class OptimizationResultComponent extends React.Component { export class OptimizationResultComponent extends React.Component {
@computed private get columns(): Column<OptimizationResult>[] { @computed private get columns(): Column<OptimalMethod>[] {
// Standard columns. // Standard columns.
const results = huntOptimizerStore.results; const result = huntOptimizerStore.result;
const optimalMethods = result ? result.optimalMethods : [];
let totalRuns = 0; let totalRuns = 0;
let totalTime = 0; let totalTime = 0;
for (const result of results) { for (const method of optimalMethods) {
totalRuns += result.runs; totalRuns += method.runs;
totalTime += result.totalTime; totalTime += method.totalTime;
} }
const columns: Column<OptimizationResult>[] = [ const columns: Column<OptimalMethod>[] = [
{ {
name: 'Difficulty', name: 'Difficulty',
width: 75, width: 75,
@ -67,47 +67,43 @@ export class OptimizationResultComponent extends React.Component {
]; ];
// Add one column per item. // Add one column per item.
const items = new Set<Item>(); if (result) {
for (const item of result.wantedItems) {
let totalCount = 0;
for (const r of results) { for (const method of optimalMethods) {
for (const i of r.itemCounts.keys()) { totalCount += method.itemCounts.get(item) || 0;
items.add(i); }
columns.push({
name: item.name,
width: 80,
cellRenderer: (result) => {
const count = result.itemCounts.get(item);
return count ? count.toFixed(2) : '';
},
tooltip: (result) => {
const count = result.itemCounts.get(item);
return count ? count.toString() : '';
},
className: 'number',
footerValue: totalCount.toFixed(2),
footerTooltip: totalCount.toString()
});
} }
} }
for (const item of items) {
const totalCount = results.reduce(
(acc, r) => acc + (r.itemCounts.get(item) || 0),
0
);
columns.push({
name: item.name,
width: 80,
cellRenderer: (result) => {
const count = result.itemCounts.get(item);
return count ? count.toFixed(2) : '';
},
tooltip: (result) => {
const count = result.itemCounts.get(item);
return count ? count.toString() : '';
},
className: 'number',
footerValue: totalCount.toFixed(2),
footerTooltip: totalCount.toString()
});
}
return columns; return columns;
} }
// Make sure render is called when result changes. // Make sure render is called when result changes.
@computed private get updateTrigger() { @computed private get updateTrigger() {
return huntOptimizerStore.results.slice(0, 0); return huntOptimizerStore.result != null;
} }
render() { render() {
this.updateTrigger; // eslint-disable-line this.updateTrigger; // eslint-disable-line
const result = huntOptimizerStore.result;
return ( return (
<section className="ho-OptimizationResultComponent"> <section className="ho-OptimizationResultComponent">
@ -118,11 +114,11 @@ export class OptimizationResultComponent extends React.Component {
<DataTable <DataTable
width={width} width={width}
height={height} height={height}
rowCount={huntOptimizerStore.results.length} rowCount={result ? result.optimalMethods.length : 0}
columns={this.columns} columns={this.columns}
fixedColumnCount={3} fixedColumnCount={3}
record={this.record} record={this.record}
footer={huntOptimizerStore.results.length > 0} footer={result != null}
updateTrigger={this.updateTrigger} updateTrigger={this.updateTrigger}
/> />
} }
@ -132,7 +128,7 @@ export class OptimizationResultComponent extends React.Component {
); );
} }
private record = ({ index }: Index): OptimizationResult => { private record = ({ index }: Index): OptimalMethod => {
return huntOptimizerStore.results[index]; return huntOptimizerStore.result!.optimalMethods[index];
} }
} }