2019-06-04 23:04:20 +08:00
|
|
|
import solver from 'javascript-lp-solver';
|
2019-06-20 04:14:48 +08:00
|
|
|
import { autorun, IObservableArray, observable, computed } from "mobx";
|
2019-06-22 06:27:04 +08:00
|
|
|
import { Difficulties, Difficulty, HuntMethod, ItemType, KONDRIEU_PROB, NpcType, RARE_ENEMY_PROB, SectionId, SectionIds, Server, Episode } from "../domain";
|
2019-06-11 19:13:54 +08:00
|
|
|
import { applicationStore } from './ApplicationStore';
|
2019-06-04 03:41:18 +08:00
|
|
|
import { huntMethodStore } from "./HuntMethodStore";
|
2019-06-20 04:14:48 +08:00
|
|
|
import { itemDropStores } from './ItemDropStore';
|
2019-06-20 20:06:11 +08:00
|
|
|
import { itemTypeStores } from './ItemTypeStore';
|
2019-06-22 02:06:55 +08:00
|
|
|
import Logger from 'js-logger';
|
|
|
|
|
|
|
|
const logger = Logger.get('stores/HuntOptimizerStore');
|
2019-06-04 03:41:18 +08:00
|
|
|
|
|
|
|
export class WantedItem {
|
2019-06-20 20:06:11 +08:00
|
|
|
@observable readonly itemType: ItemType;
|
2019-06-04 03:41:18 +08:00
|
|
|
@observable amount: number;
|
|
|
|
|
2019-06-20 20:06:11 +08:00
|
|
|
constructor(itemType: ItemType, amount: number) {
|
|
|
|
this.itemType = itemType;
|
2019-06-04 03:41:18 +08:00
|
|
|
this.amount = amount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
export class OptimalResult {
|
|
|
|
constructor(
|
2019-06-20 20:06:11 +08:00
|
|
|
readonly wantedItems: Array<ItemType>,
|
2019-06-16 17:43:10 +08:00
|
|
|
readonly optimalMethods: Array<OptimalMethod>
|
|
|
|
) { }
|
|
|
|
}
|
|
|
|
|
|
|
|
export class OptimalMethod {
|
|
|
|
readonly totalTime: number;
|
2019-06-05 22:49:00 +08:00
|
|
|
|
|
|
|
constructor(
|
2019-06-16 17:43:10 +08:00
|
|
|
readonly difficulty: Difficulty,
|
|
|
|
readonly sectionIds: Array<SectionId>,
|
|
|
|
readonly methodName: string,
|
2019-06-22 06:27:04 +08:00
|
|
|
readonly methodEpisode: Episode,
|
2019-06-16 17:43:10 +08:00
|
|
|
readonly methodTime: number,
|
|
|
|
readonly runs: number,
|
2019-06-20 20:06:11 +08:00
|
|
|
readonly itemCounts: Map<ItemType, number>
|
2019-06-05 22:49:00 +08:00
|
|
|
) {
|
|
|
|
this.totalTime = runs * methodTime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-22 20:58:57 +08:00
|
|
|
// TODO: take into account mothmants spawned from mothverts.
|
|
|
|
// TODO: take into account split slimes.
|
2019-06-07 02:30:14 +08:00
|
|
|
// TODO: Prefer methods that don't split pan arms over methods that do.
|
2019-06-07 03:40:12 +08:00
|
|
|
// For some reason this doesn't actually seem to be a problem, should probably investigate.
|
2019-06-16 05:00:46 +08:00
|
|
|
// TODO: Show expected value or probability per item per method.
|
2019-06-16 19:02:40 +08:00
|
|
|
// Can be useful when deciding which item to hunt first.
|
2019-06-06 21:42:02 +08:00
|
|
|
// TODO: boxes.
|
2019-06-04 03:41:18 +08:00
|
|
|
class HuntOptimizerStore {
|
2019-06-20 20:06:11 +08:00
|
|
|
@computed get huntableItemTypes(): Array<ItemType> {
|
2019-06-20 04:14:48 +08:00
|
|
|
const itemDropStore = itemDropStores.current.value;
|
2019-06-20 20:06:11 +08:00
|
|
|
return itemTypeStores.current.value.itemTypes.filter(i =>
|
|
|
|
itemDropStore.enemyDrops.getDropsForItemType(i.id).length
|
2019-06-20 04:14:48 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-22 07:12:39 +08:00
|
|
|
// TODO: wanted items per server.
|
2019-06-11 19:13:54 +08:00
|
|
|
@observable readonly wantedItems: IObservableArray<WantedItem> = observable.array();
|
2019-06-16 17:43:10 +08:00
|
|
|
@observable result?: OptimalResult;
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-11 19:13:54 +08:00
|
|
|
constructor() {
|
|
|
|
this.initialize();
|
|
|
|
}
|
|
|
|
|
2019-06-04 03:41:18 +08:00
|
|
|
optimize = async () => {
|
2019-06-06 08:24:21 +08:00
|
|
|
if (!this.wantedItems.length) {
|
2019-06-16 17:43:10 +08:00
|
|
|
this.result = undefined;
|
2019-06-06 08:24:21 +08:00
|
|
|
return;
|
|
|
|
}
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
// Initialize this set before awaiting data, so user changes don't affect this optimization
|
|
|
|
// run from this point on.
|
2019-06-20 20:06:11 +08:00
|
|
|
const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.itemType));
|
2019-06-16 17:43:10 +08:00
|
|
|
|
2019-06-04 23:01:51 +08:00
|
|
|
const methods = await huntMethodStore.methods.current.promise;
|
2019-06-20 04:14:48 +08:00
|
|
|
const dropTable = (await itemDropStores.current.promise).enemyDrops;
|
2019-06-04 23:01:51 +08:00
|
|
|
|
2019-06-05 22:49:00 +08:00
|
|
|
// Add a constraint per wanted item.
|
2019-06-04 03:41:18 +08:00
|
|
|
const constraints: { [itemName: string]: { min: number } } = {};
|
|
|
|
|
2019-06-04 23:01:51 +08:00
|
|
|
for (const wanted of this.wantedItems) {
|
2019-06-20 20:06:11 +08:00
|
|
|
constraints[wanted.itemType.name] = { min: wanted.amount };
|
2019-06-04 03:41:18 +08:00
|
|
|
}
|
|
|
|
|
2019-06-05 22:49:00 +08:00
|
|
|
// Add a variable to the LP model per method per difficulty per section ID.
|
2019-06-07 02:30:14 +08:00
|
|
|
// When a method with pan arms is encountered, two variables are added. One for the method
|
|
|
|
// with migiums and hidooms and one with pan arms.
|
2019-06-05 22:49:00 +08:00
|
|
|
// 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,
|
2019-06-07 02:30:14 +08:00
|
|
|
[itemName: string]: number,
|
2019-06-05 22:49:00 +08:00
|
|
|
}
|
|
|
|
const variables: { [methodName: string]: Variable } = {};
|
|
|
|
|
2019-06-07 02:30:14 +08:00
|
|
|
type VariableDetails = {
|
|
|
|
method: HuntMethod,
|
|
|
|
difficulty: Difficulty,
|
|
|
|
sectionId: SectionId,
|
|
|
|
splitPanArms: boolean,
|
|
|
|
}
|
|
|
|
const variableDetails: Map<string, VariableDetails> = new Map();
|
|
|
|
|
2019-06-04 23:01:51 +08:00
|
|
|
for (const method of methods) {
|
2019-06-06 21:42:02 +08:00
|
|
|
// Counts include rare enemies, so they are fractional.
|
2019-06-04 23:01:51 +08:00
|
|
|
const counts = new Map<NpcType, number>();
|
|
|
|
|
2019-06-26 23:47:53 +08:00
|
|
|
for (const [enemy, count] of method.enemy_counts.entries()) {
|
2019-06-22 06:27:04 +08:00
|
|
|
const oldCount = counts.get(enemy) || 0;
|
2019-06-06 21:42:02 +08:00
|
|
|
|
2019-06-22 06:27:04 +08:00
|
|
|
if (enemy.rareType == null) {
|
|
|
|
counts.set(enemy, oldCount + count);
|
2019-06-06 21:42:02 +08:00
|
|
|
} else {
|
|
|
|
let rate, rareRate;
|
|
|
|
|
2019-06-22 06:27:04 +08:00
|
|
|
if (enemy.rareType === NpcType.Kondrieu) {
|
2019-06-06 21:42:02 +08:00
|
|
|
rate = 1 - KONDRIEU_PROB;
|
|
|
|
rareRate = KONDRIEU_PROB;
|
|
|
|
} else {
|
|
|
|
rate = 1 - RARE_ENEMY_PROB;
|
|
|
|
rareRate = RARE_ENEMY_PROB;
|
|
|
|
}
|
|
|
|
|
2019-06-22 06:27:04 +08:00
|
|
|
counts.set(enemy, oldCount + count * rate);
|
|
|
|
counts.set(
|
|
|
|
enemy.rareType,
|
|
|
|
(counts.get(enemy.rareType) || 0) + count * rareRate
|
|
|
|
);
|
2019-06-06 21:42:02 +08:00
|
|
|
}
|
2019-06-04 23:01:51 +08:00
|
|
|
}
|
|
|
|
|
2019-06-07 02:30:14 +08:00
|
|
|
// Create a secondary counts map if there are any pan arms that can be split into
|
|
|
|
// migiums and hidooms.
|
|
|
|
const countsList: Array<Map<NpcType, number>> = [counts];
|
|
|
|
const panArmsCount = counts.get(NpcType.PanArms);
|
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
if (panArmsCount) {
|
2019-06-07 02:30:14 +08:00
|
|
|
const splitCounts = new Map(counts);
|
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
splitCounts.delete(NpcType.PanArms);
|
|
|
|
splitCounts.set(NpcType.Migium, panArmsCount);
|
|
|
|
splitCounts.set(NpcType.Hidoom, panArmsCount);
|
2019-06-07 02:30:14 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
countsList.push(splitCounts);
|
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
2019-06-07 02:30:14 +08:00
|
|
|
|
|
|
|
countsList.push(splitCounts);
|
|
|
|
}
|
2019-06-04 23:01:51 +08:00
|
|
|
|
2019-06-07 02:30:14 +08:00
|
|
|
for (let i = 0; i < countsList.length; i++) {
|
|
|
|
const counts = countsList[i];
|
|
|
|
const splitPanArms = i === 1;
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-07 02:30:14 +08:00
|
|
|
for (const diff of Difficulties) {
|
|
|
|
for (const sectionId of SectionIds) {
|
2019-06-16 17:43:10 +08:00
|
|
|
// Will contain an entry per wanted item dropped by enemies in this method/
|
|
|
|
// difficulty/section ID combo.
|
2019-06-07 02:30:14 +08:00
|
|
|
const variable: Variable = {
|
|
|
|
time: method.time
|
|
|
|
};
|
2019-06-16 17:43:10 +08:00
|
|
|
// Only add the variable if the method provides at least 1 item we want.
|
2019-06-07 02:30:14 +08:00
|
|
|
let addVariable = false;
|
|
|
|
|
|
|
|
for (const [npcType, count] of counts.entries()) {
|
|
|
|
const drop = dropTable.getDrop(diff, sectionId, npcType);
|
|
|
|
|
2019-06-26 23:47:53 +08:00
|
|
|
if (drop && wantedItems.has(drop.item_type)) {
|
|
|
|
const value = variable[drop.item_type.name] || 0;
|
|
|
|
variable[drop.item_type.name] = value + count * drop.rate;
|
2019-06-07 02:30:14 +08:00
|
|
|
addVariable = true;
|
|
|
|
}
|
2019-06-04 23:01:51 +08:00
|
|
|
}
|
2019-06-04 03:41:18 +08:00
|
|
|
|
2019-06-07 02:30:14 +08:00
|
|
|
if (addVariable) {
|
2019-06-16 05:00:46 +08:00
|
|
|
const name = this.fullMethodName(
|
|
|
|
diff, sectionId, method, splitPanArms
|
|
|
|
);
|
2019-06-07 02:30:14 +08:00
|
|
|
variables[name] = variable;
|
|
|
|
variableDetails.set(name, {
|
|
|
|
method,
|
|
|
|
difficulty: diff,
|
|
|
|
sectionId,
|
|
|
|
splitPanArms
|
|
|
|
});
|
|
|
|
}
|
2019-06-04 23:01:51 +08:00
|
|
|
}
|
|
|
|
}
|
2019-06-04 03:41:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-05 22:49:00 +08:00
|
|
|
const result: {
|
|
|
|
feasible: boolean,
|
|
|
|
bounded: boolean,
|
|
|
|
result: number,
|
2019-06-16 17:43:10 +08:00
|
|
|
/**
|
|
|
|
* Value will always be a number if result is indexed with an actual method name.
|
|
|
|
*/
|
2019-06-05 22:49:00 +08:00
|
|
|
[method: string]: number | boolean
|
|
|
|
} = solver.Solve({
|
2019-06-04 23:01:51 +08:00
|
|
|
optimize: 'time',
|
2019-06-04 03:41:18 +08:00
|
|
|
opType: 'min',
|
|
|
|
constraints,
|
|
|
|
variables
|
|
|
|
});
|
2019-06-04 23:01:51 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
if (!result.feasible) {
|
|
|
|
this.result = undefined;
|
|
|
|
return;
|
|
|
|
}
|
2019-06-05 22:49:00 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
const optimalMethods: Array<OptimalMethod> = [];
|
2019-06-06 08:24:21 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
// Loop over the entries in result, ignore standard properties that aren't variables.
|
|
|
|
for (const [variableName, runsOrOther] of Object.entries(result)) {
|
|
|
|
const details = variableDetails.get(variableName);
|
2019-06-05 22:49:00 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
if (details) {
|
|
|
|
const { method, difficulty, sectionId, splitPanArms } = details;
|
|
|
|
const runs = runsOrOther as number;
|
|
|
|
const variable = variables[variableName];
|
2019-06-05 22:49:00 +08:00
|
|
|
|
2019-06-20 20:06:11 +08:00
|
|
|
const items = new Map<ItemType, number>();
|
2019-06-05 22:49:00 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
for (const [itemName, expectedAmount] of Object.entries(variable)) {
|
|
|
|
for (const item of wantedItems) {
|
|
|
|
if (itemName === item.name) {
|
|
|
|
items.set(item, runs * expectedAmount);
|
|
|
|
break;
|
2019-06-05 22:49:00 +08:00
|
|
|
}
|
|
|
|
}
|
2019-06-16 17:43:10 +08:00
|
|
|
}
|
2019-06-05 22:49:00 +08:00
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
// 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
|
|
|
|
// purplenum or yellowboze will give you the exact same probabilities.
|
|
|
|
const sectionIds: Array<SectionId> = [];
|
|
|
|
|
|
|
|
for (const sid of SectionIds) {
|
|
|
|
let matchFound = true;
|
|
|
|
|
|
|
|
if (sid !== sectionId) {
|
|
|
|
const v = variables[
|
|
|
|
this.fullMethodName(difficulty, sid, method, splitPanArms)
|
|
|
|
];
|
|
|
|
|
|
|
|
if (!v) {
|
|
|
|
matchFound = false;
|
|
|
|
} else {
|
|
|
|
for (const itemName of Object.keys(variable)) {
|
|
|
|
if (variable[itemName] !== v[itemName]) {
|
|
|
|
matchFound = false;
|
|
|
|
break;
|
2019-06-16 05:00:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-16 17:43:10 +08:00
|
|
|
if (matchFound) {
|
|
|
|
sectionIds.push(sid);
|
|
|
|
}
|
2019-06-05 22:49:00 +08:00
|
|
|
}
|
2019-06-16 17:43:10 +08:00
|
|
|
|
|
|
|
optimalMethods.push(new OptimalMethod(
|
|
|
|
difficulty,
|
|
|
|
sectionIds,
|
|
|
|
method.name + (splitPanArms ? ' (Split Pan Arms)' : ''),
|
2019-06-22 06:27:04 +08:00
|
|
|
method.episode,
|
2019-06-16 17:43:10 +08:00
|
|
|
method.time,
|
|
|
|
runs,
|
|
|
|
items
|
|
|
|
));
|
2019-06-05 22:49:00 +08:00
|
|
|
}
|
2019-06-16 17:43:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
this.result = new OptimalResult(
|
|
|
|
[...wantedItems],
|
|
|
|
optimalMethods
|
|
|
|
);
|
2019-06-04 03:41:18 +08:00
|
|
|
}
|
2019-06-16 05:00:46 +08:00
|
|
|
|
|
|
|
private fullMethodName(
|
|
|
|
difficulty: Difficulty,
|
|
|
|
sectionId: SectionId,
|
|
|
|
method: HuntMethod,
|
|
|
|
splitPanArms: boolean
|
|
|
|
): string {
|
2019-06-20 04:14:48 +08:00
|
|
|
let name = `${difficulty}\t${sectionId}\t${method.id}`;
|
|
|
|
if (splitPanArms) name += '\tspa';
|
2019-06-16 05:00:46 +08:00
|
|
|
return name;
|
|
|
|
}
|
2019-06-22 07:12:39 +08:00
|
|
|
|
|
|
|
private initialize = async () => {
|
|
|
|
try {
|
|
|
|
await this.loadFromLocalStorage();
|
|
|
|
autorun(this.storeInLocalStorage);
|
|
|
|
} catch (e) {
|
|
|
|
logger.error(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private loadFromLocalStorage = async () => {
|
|
|
|
const wantedItemsJson = localStorage.getItem(
|
|
|
|
`HuntOptimizerStore.wantedItems.${Server[applicationStore.currentServer]}`
|
|
|
|
);
|
|
|
|
|
|
|
|
if (wantedItemsJson) {
|
|
|
|
const itemStore = await itemTypeStores.current.promise;
|
|
|
|
const wi = JSON.parse(wantedItemsJson);
|
|
|
|
|
|
|
|
const wantedItems: WantedItem[] = [];
|
|
|
|
|
|
|
|
for (const { itemTypeId, itemKindId, amount } of wi) {
|
|
|
|
const item = itemTypeId != null
|
|
|
|
? itemStore.getById(itemTypeId)
|
|
|
|
: itemStore.getById(itemKindId); // Legacy name.
|
|
|
|
|
|
|
|
if (item) {
|
|
|
|
wantedItems.push(new WantedItem(item, amount));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.wantedItems.replace(wantedItems);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private storeInLocalStorage = () => {
|
|
|
|
try {
|
|
|
|
localStorage.setItem(
|
|
|
|
`HuntOptimizerStore.wantedItems.${Server[applicationStore.currentServer]}`,
|
|
|
|
JSON.stringify(
|
|
|
|
this.wantedItems.map(({ itemType, amount }) => ({
|
|
|
|
itemTypeId: itemType.id,
|
|
|
|
amount
|
|
|
|
}))
|
|
|
|
)
|
|
|
|
);
|
|
|
|
} catch (e) {
|
|
|
|
logger.error(e);
|
|
|
|
}
|
|
|
|
}
|
2019-06-04 03:41:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export const huntOptimizerStore = new HuntOptimizerStore();
|