phantasmal-world/src/stores/HuntOptimizerStore.ts

362 lines
13 KiB
TypeScript
Raw Normal View History

2019-06-04 23:04:20 +08:00
import solver from 'javascript-lp-solver';
import { autorun, IObservableArray, observable, computed } from "mobx";
import { Difficulties, Difficulty, HuntMethod, ItemType, KONDRIEU_PROB, NpcType, RARE_ENEMY_PROB, SectionId, SectionIds, Server, Episode } from "../domain";
2019-07-02 23:00:24 +08:00
import { application_store } from './ApplicationStore';
import { hunt_method_store } from "./HuntMethodStore";
import { item_drop_stores as item_drop_stores } from './ItemDropStore';
import { item_type_stores } from './ItemTypeStore';
2019-06-22 02:06:55 +08:00
import Logger from 'js-logger';
const logger = Logger.get('stores/HuntOptimizerStore');
export class WantedItem {
2019-07-02 23:00:24 +08:00
@observable readonly item_type: ItemType;
@observable amount: number;
2019-07-02 23:00:24 +08:00
constructor(item_type: ItemType, amount: number) {
this.item_type = item_type;
this.amount = amount;
}
}
export class OptimalResult {
constructor(
2019-07-02 23:00:24 +08:00
readonly wanted_items: Array<ItemType>,
readonly optimal_methods: Array<OptimalMethod>
) { }
}
export class OptimalMethod {
2019-07-02 23:00:24 +08:00
readonly total_time: number;
constructor(
readonly difficulty: Difficulty,
2019-07-02 23:00:24 +08:00
readonly section_ids: Array<SectionId>,
readonly method_name: string,
readonly method_episode: Episode,
readonly method_time: number,
readonly runs: number,
2019-07-02 23:00:24 +08:00
readonly item_counts: Map<ItemType, number>
) {
2019-07-02 23:00:24 +08:00
this.total_time = runs * method_time;
}
}
// TODO: take into account mothmants spawned from mothverts.
// TODO: take into account split slimes.
// TODO: Prefer methods that don't split pan arms over methods that do.
// For some reason this doesn't actually seem to be a problem, should probably investigate.
// TODO: Show expected value or probability per item per method.
// Can be useful when deciding which item to hunt first.
// TODO: boxes.
class HuntOptimizerStore {
2019-07-02 23:00:24 +08:00
@computed get huntable_item_types(): Array<ItemType> {
const item_drop_store = item_drop_stores.current.value;
return item_type_stores.current.value.item_types.filter(i =>
item_drop_store.enemy_drops.get_drops_for_item_type(i.id).length
);
}
// TODO: wanted items per server.
2019-07-02 23:00:24 +08:00
@observable readonly wanted_items: IObservableArray<WantedItem> = observable.array();
@observable result?: OptimalResult;
2019-06-11 19:13:54 +08:00
constructor() {
this.initialize();
}
optimize = async () => {
2019-07-02 23:00:24 +08:00
if (!this.wanted_items.length) {
this.result = undefined;
2019-06-06 08:24:21 +08:00
return;
}
// Initialize this set before awaiting data, so user changes don't affect this optimization
// run from this point on.
2019-07-02 23:00:24 +08:00
const wanted_items = new Set(this.wanted_items.filter(w => w.amount > 0).map(w => w.item_type));
2019-07-02 23:00:24 +08:00
const methods = await hunt_method_store.methods.current.promise;
const drop_table = (await item_drop_stores.current.promise).enemy_drops;
// Add a constraint per wanted item.
2019-07-02 23:00:24 +08:00
const constraints: { [item_name: string]: { min: number } } = {};
2019-07-02 23:00:24 +08:00
for (const wanted of this.wanted_items) {
constraints[wanted.item_type.name] = { min: wanted.amount };
}
// Add a variable to the LP model per method per difficulty per section ID.
// 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.
// 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-07-02 23:00:24 +08:00
[item_name: string]: number,
}
2019-07-02 23:00:24 +08:00
const variables: { [method_name: string]: Variable } = {};
type VariableDetails = {
method: HuntMethod,
difficulty: Difficulty,
2019-07-02 23:00:24 +08:00
section_id: SectionId,
split_pan_arms: boolean,
}
2019-07-02 23:00:24 +08:00
const variable_details: Map<string, VariableDetails> = new Map();
for (const method of methods) {
// Counts include rare enemies, so they are fractional.
const counts = new Map<NpcType, number>();
2019-06-26 23:47:53 +08:00
for (const [enemy, count] of method.enemy_counts.entries()) {
2019-07-02 23:00:24 +08:00
const old_count = counts.get(enemy) || 0;
2019-07-02 23:00:24 +08:00
if (enemy.rare_type == null) {
counts.set(enemy, old_count + count);
} else {
2019-07-02 23:00:24 +08:00
let rate, rare_rate;
2019-07-02 23:00:24 +08:00
if (enemy.rare_type === NpcType.Kondrieu) {
rate = 1 - KONDRIEU_PROB;
2019-07-02 23:00:24 +08:00
rare_rate = KONDRIEU_PROB;
} else {
rate = 1 - RARE_ENEMY_PROB;
2019-07-02 23:00:24 +08:00
rare_rate = RARE_ENEMY_PROB;
}
2019-07-02 23:00:24 +08:00
counts.set(enemy, old_count + count * rate);
counts.set(
2019-07-02 23:00:24 +08:00
enemy.rare_type,
(counts.get(enemy.rare_type) || 0) + count * rare_rate
);
}
}
// Create a secondary counts map if there are any pan arms that can be split into
// migiums and hidooms.
2019-07-02 23:00:24 +08:00
const counts_list: Array<Map<NpcType, number>> = [counts];
const pan_arms_count = counts.get(NpcType.PanArms);
2019-07-02 23:00:24 +08:00
if (pan_arms_count) {
const split_counts = new Map(counts);
2019-07-02 23:00:24 +08:00
split_counts.delete(NpcType.PanArms);
split_counts.set(NpcType.Migium, pan_arms_count);
split_counts.set(NpcType.Hidoom, pan_arms_count);
2019-07-02 23:00:24 +08:00
counts_list.push(split_counts);
}
2019-07-02 23:00:24 +08:00
const pan_arms_2_count = counts.get(NpcType.PanArms2);
2019-07-02 23:00:24 +08:00
if (pan_arms_2_count) {
const split_counts = new Map(counts);
2019-07-02 23:00:24 +08:00
split_counts.delete(NpcType.PanArms2);
split_counts.set(NpcType.Migium2, pan_arms_2_count);
split_counts.set(NpcType.Hidoom2, pan_arms_2_count);
2019-07-02 23:00:24 +08:00
counts_list.push(split_counts);
}
2019-07-02 23:00:24 +08:00
for (let i = 0; i < counts_list.length; i++) {
const counts = counts_list[i];
const split_pan_arms = i === 1;
2019-07-02 23:00:24 +08:00
for (const difficulty of Difficulties) {
for (const section_id of SectionIds) {
// Will contain an entry per wanted item dropped by enemies in this method/
// difficulty/section ID combo.
const variable: Variable = {
time: method.time
};
// Only add the variable if the method provides at least 1 item we want.
2019-07-02 23:00:24 +08:00
let add_variable = false;
2019-07-02 23:00:24 +08:00
for (const [npc_type, count] of counts.entries()) {
const drop = drop_table.get_drop(difficulty, section_id, npc_type);
2019-07-02 23:00:24 +08:00
if (drop && wanted_items.has(drop.item_type)) {
2019-06-26 23:47:53 +08:00
const value = variable[drop.item_type.name] || 0;
variable[drop.item_type.name] = value + count * drop.rate;
2019-07-02 23:00:24 +08:00
add_variable = true;
}
}
2019-07-02 23:00:24 +08:00
if (add_variable) {
const name = this.full_method_name(
difficulty, section_id, method, split_pan_arms
);
variables[name] = variable;
2019-07-02 23:00:24 +08:00
variable_details.set(name, {
method,
2019-07-02 23:00:24 +08:00
difficulty,
section_id,
split_pan_arms
});
}
}
}
}
}
const result: {
feasible: boolean,
bounded: boolean,
result: number,
/**
* Value will always be a number if result is indexed with an actual method name.
*/
[method: string]: number | boolean
} = solver.Solve({
optimize: 'time',
opType: 'min',
constraints,
variables
});
if (!result.feasible) {
this.result = undefined;
return;
}
2019-07-02 23:00:24 +08:00
const optimal_methods: Array<OptimalMethod> = [];
2019-06-06 08:24:21 +08:00
// Loop over the entries in result, ignore standard properties that aren't variables.
2019-07-02 23:00:24 +08:00
for (const [variable_name, runs_or_other] of Object.entries(result)) {
const details = variable_details.get(variable_name);
if (details) {
2019-07-02 23:00:24 +08:00
const { method, difficulty, section_id, split_pan_arms } = details;
const runs = runs_or_other as number;
const variable = variables[variable_name];
const items = new Map<ItemType, number>();
2019-07-02 23:00:24 +08:00
for (const [item_name, expected_amount] of Object.entries(variable)) {
for (const item of wanted_items) {
if (item_name === item.name) {
items.set(item, runs * expected_amount);
break;
}
}
}
// 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.
2019-07-02 23:00:24 +08:00
const section_ids: Array<SectionId> = [];
for (const sid of SectionIds) {
2019-07-02 23:00:24 +08:00
let match_found = true;
2019-07-02 23:00:24 +08:00
if (sid !== section_id) {
const v = variables[
2019-07-02 23:00:24 +08:00
this.full_method_name(difficulty, sid, method, split_pan_arms)
];
if (!v) {
2019-07-02 23:00:24 +08:00
match_found = false;
} else {
2019-07-02 23:00:24 +08:00
for (const item_name of Object.keys(variable)) {
if (variable[item_name] !== v[item_name]) {
match_found = false;
break;
}
}
}
}
2019-07-02 23:00:24 +08:00
if (match_found) {
section_ids.push(sid);
}
}
2019-07-02 23:00:24 +08:00
optimal_methods.push(new OptimalMethod(
difficulty,
2019-07-02 23:00:24 +08:00
section_ids,
method.name + (split_pan_arms ? ' (Split Pan Arms)' : ''),
method.episode,
method.time,
runs,
items
));
}
}
this.result = new OptimalResult(
2019-07-02 23:00:24 +08:00
[...wanted_items],
optimal_methods
);
}
2019-07-02 23:00:24 +08:00
private full_method_name(
difficulty: Difficulty,
2019-07-02 23:00:24 +08:00
section_id: SectionId,
method: HuntMethod,
2019-07-02 23:00:24 +08:00
split_pan_arms: boolean
): string {
2019-07-02 23:00:24 +08:00
let name = `${difficulty}\t${section_id}\t${method.id}`;
if (split_pan_arms) name += '\tspa';
return name;
}
private initialize = async () => {
try {
2019-07-02 23:00:24 +08:00
await this.load_from_local_storage();
autorun(this.store_in_local_storage);
} catch (e) {
logger.error(e);
}
}
2019-07-02 23:00:24 +08:00
private load_from_local_storage = async () => {
const wanted_items_json = localStorage.getItem(
`HuntOptimizerStore.wantedItems.${Server[application_store.current_server]}`
);
2019-07-02 23:00:24 +08:00
if (wanted_items_json) {
const item_store = await item_type_stores.current.promise;
const wi: StoredWantedItem[] = JSON.parse(wanted_items_json);
2019-07-02 23:00:24 +08:00
const wanted_items: WantedItem[] = [];
for (const { itemTypeId, itemKindId, amount } of wi) {
2019-07-02 23:00:24 +08:00
const item = itemTypeId != undefined
? item_store.get_by_id(itemTypeId)
: item_store.get_by_id(itemKindId!);
if (item) {
2019-07-02 23:00:24 +08:00
wanted_items.push(new WantedItem(item, amount));
}
}
2019-07-02 23:00:24 +08:00
this.wanted_items.replace(wanted_items);
}
}
2019-07-02 23:00:24 +08:00
private store_in_local_storage = () => {
try {
localStorage.setItem(
2019-07-02 23:00:24 +08:00
`HuntOptimizerStore.wantedItems.${Server[application_store.current_server]}`,
JSON.stringify(
2019-07-02 23:00:24 +08:00
this.wanted_items.map(({ item_type: itemType, amount }): StoredWantedItem => ({
itemTypeId: itemType.id,
amount
}))
)
);
} catch (e) {
logger.error(e);
}
}
}
2019-07-02 23:00:24 +08:00
type StoredWantedItem = {
itemTypeId?: number, // Should only be undefined if the legacy name is still used.
itemKindId?: number, // Legacy name.
amount: number,
};
export const hunt_optimizer_store = new HuntOptimizerStore();