mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 22:58:29 +08:00
Renamed ItemKind to ItemType for consistency with NpcType.
This commit is contained in:
parent
4a3f5991ed
commit
a684bb65c8
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -315,18 +315,18 @@ export class AreaVariant {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Abstract base class of all item kinds.
|
// Abstract base class of all item kinds.
|
||||||
export class ItemKind {
|
export class ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
readonly id: number,
|
readonly id: number,
|
||||||
readonly name: string
|
readonly name: string
|
||||||
) {
|
) {
|
||||||
|
|
||||||
if (Object.getPrototypeOf(this) === Object.getPrototypeOf(ItemKind))
|
if (Object.getPrototypeOf(this) === Object.getPrototypeOf(ItemType))
|
||||||
throw new Error('Abstract class should not be instantiated directly.');
|
throw new Error('Abstract class should not be instantiated directly.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class WeaponItemKind extends ItemKind {
|
export class WeaponItemType extends ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
@ -340,7 +340,7 @@ export class WeaponItemKind extends ItemKind {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ArmorItemKind extends ItemKind {
|
export class ArmorItemType extends ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
@ -349,7 +349,7 @@ export class ArmorItemKind extends ItemKind {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ShieldItemKind extends ItemKind {
|
export class ShieldItemType extends ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
@ -358,7 +358,7 @@ export class ShieldItemKind extends ItemKind {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class UnitItemKind extends ItemKind {
|
export class UnitItemType extends ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
@ -367,7 +367,7 @@ export class UnitItemKind extends ItemKind {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ToolItemKind extends ItemKind {
|
export class ToolItemType extends ItemType {
|
||||||
constructor(
|
constructor(
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
@ -377,7 +377,7 @@ export class ToolItemKind extends ItemKind {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type ItemDrop = {
|
type ItemDrop = {
|
||||||
item: ItemKind,
|
itemType: ItemType,
|
||||||
anythingRate: number,
|
anythingRate: number,
|
||||||
rareRate: number
|
rareRate: number
|
||||||
}
|
}
|
||||||
@ -389,7 +389,7 @@ export class EnemyDrop implements ItemDrop {
|
|||||||
readonly difficulty: Difficulty,
|
readonly difficulty: Difficulty,
|
||||||
readonly sectionId: SectionId,
|
readonly sectionId: SectionId,
|
||||||
readonly npcType: NpcType,
|
readonly npcType: NpcType,
|
||||||
readonly item: ItemKind,
|
readonly itemType: ItemType,
|
||||||
readonly anythingRate: number,
|
readonly anythingRate: number,
|
||||||
readonly rareRate: number
|
readonly rareRate: number
|
||||||
) {
|
) {
|
||||||
|
34
src/dto.ts
34
src/dto.ts
@ -1,11 +1,11 @@
|
|||||||
export type ItemKindDto = WeaponItemKindDto
|
export type ItemTypeDto = WeaponItemTypeDto
|
||||||
| ArmorItemKindDto
|
| ArmorItemTypeDto
|
||||||
| ShieldItemKindDto
|
| ShieldItemTypeDto
|
||||||
| UnitItemKindDto
|
| UnitItemTypeDto
|
||||||
| ToolItemKindDto
|
| ToolItemTypeDto
|
||||||
|
|
||||||
export type WeaponItemKindDto = {
|
export type WeaponItemTypeDto = {
|
||||||
type: 'weapon',
|
class: 'weapon',
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
minAtp: number,
|
minAtp: number,
|
||||||
@ -15,26 +15,26 @@ export type WeaponItemKindDto = {
|
|||||||
requiredAtp: number,
|
requiredAtp: number,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ArmorItemKindDto = {
|
export type ArmorItemTypeDto = {
|
||||||
type: 'armor',
|
class: 'armor',
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ShieldItemKindDto = {
|
export type ShieldItemTypeDto = {
|
||||||
type: 'shield',
|
class: 'shield',
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type UnitItemKindDto = {
|
export type UnitItemTypeDto = {
|
||||||
type: 'unit',
|
class: 'unit',
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type ToolItemKindDto = {
|
export type ToolItemTypeDto = {
|
||||||
type: 'tool',
|
class: 'tool',
|
||||||
id: number,
|
id: number,
|
||||||
name: string,
|
name: string,
|
||||||
}
|
}
|
||||||
@ -44,7 +44,7 @@ export type EnemyDropDto = {
|
|||||||
episode: number,
|
episode: number,
|
||||||
sectionId: string,
|
sectionId: string,
|
||||||
enemy: string,
|
enemy: string,
|
||||||
itemKindId: number,
|
itemTypeId: number,
|
||||||
dropRate: number,
|
dropRate: number,
|
||||||
rareRate: number,
|
rareRate: number,
|
||||||
}
|
}
|
||||||
@ -54,6 +54,6 @@ export type BoxDropDto = {
|
|||||||
episode: number,
|
episode: number,
|
||||||
sectionId: string,
|
sectionId: string,
|
||||||
areaId: number,
|
areaId: number,
|
||||||
itemKindId: number,
|
itemTypeId: number,
|
||||||
dropRate: number,
|
dropRate: number,
|
||||||
}
|
}
|
||||||
|
@ -1,24 +1,24 @@
|
|||||||
import solver from 'javascript-lp-solver';
|
import solver from 'javascript-lp-solver';
|
||||||
import { autorun, IObservableArray, observable, computed } from "mobx";
|
import { autorun, IObservableArray, observable, computed } from "mobx";
|
||||||
import { Difficulties, Difficulty, HuntMethod, ItemKind, KONDRIEU_PROB, NpcType, RARE_ENEMY_PROB, SectionId, SectionIds, Server } from "../domain";
|
import { Difficulties, Difficulty, HuntMethod, ItemType, KONDRIEU_PROB, NpcType, RARE_ENEMY_PROB, SectionId, SectionIds, Server } from "../domain";
|
||||||
import { applicationStore } from './ApplicationStore';
|
import { applicationStore } from './ApplicationStore';
|
||||||
import { huntMethodStore } from "./HuntMethodStore";
|
import { huntMethodStore } from "./HuntMethodStore";
|
||||||
import { itemDropStores } from './ItemDropStore';
|
import { itemDropStores } from './ItemDropStore';
|
||||||
import { itemKindStores } from './ItemKindStore';
|
import { itemTypeStores } from './ItemTypeStore';
|
||||||
|
|
||||||
export class WantedItem {
|
export class WantedItem {
|
||||||
@observable readonly itemKind: ItemKind;
|
@observable readonly itemType: ItemType;
|
||||||
@observable amount: number;
|
@observable amount: number;
|
||||||
|
|
||||||
constructor(itemKind: ItemKind, amount: number) {
|
constructor(itemType: ItemType, amount: number) {
|
||||||
this.itemKind = itemKind;
|
this.itemType = itemType;
|
||||||
this.amount = amount;
|
this.amount = amount;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class OptimalResult {
|
export class OptimalResult {
|
||||||
constructor(
|
constructor(
|
||||||
readonly wantedItems: Array<ItemKind>,
|
readonly wantedItems: Array<ItemType>,
|
||||||
readonly optimalMethods: Array<OptimalMethod>
|
readonly optimalMethods: Array<OptimalMethod>
|
||||||
) { }
|
) { }
|
||||||
}
|
}
|
||||||
@ -32,7 +32,7 @@ export class OptimalMethod {
|
|||||||
readonly methodName: string,
|
readonly methodName: string,
|
||||||
readonly methodTime: number,
|
readonly methodTime: number,
|
||||||
readonly runs: number,
|
readonly runs: number,
|
||||||
readonly itemCounts: Map<ItemKind, number>
|
readonly itemCounts: Map<ItemType, number>
|
||||||
) {
|
) {
|
||||||
this.totalTime = runs * methodTime;
|
this.totalTime = runs * methodTime;
|
||||||
}
|
}
|
||||||
@ -44,10 +44,10 @@ export class OptimalMethod {
|
|||||||
// Can be useful when deciding which item to hunt first.
|
// Can be useful when deciding which item to hunt first.
|
||||||
// TODO: boxes.
|
// TODO: boxes.
|
||||||
class HuntOptimizerStore {
|
class HuntOptimizerStore {
|
||||||
@computed get huntableItems(): Array<ItemKind> {
|
@computed get huntableItemTypes(): Array<ItemType> {
|
||||||
const itemDropStore = itemDropStores.current.value;
|
const itemDropStore = itemDropStores.current.value;
|
||||||
return itemKindStores.current.value.itemKinds.filter(i =>
|
return itemTypeStores.current.value.itemTypes.filter(i =>
|
||||||
itemDropStore.enemyDrops.getDropsForItemKind(i.id).length
|
itemDropStore.enemyDrops.getDropsForItemType(i.id).length
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -73,13 +73,15 @@ class HuntOptimizerStore {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (wantedItemsJson) {
|
if (wantedItemsJson) {
|
||||||
const itemStore = await itemKindStores.current.promise;
|
const itemStore = await itemTypeStores.current.promise;
|
||||||
const wi = JSON.parse(wantedItemsJson);
|
const wi = JSON.parse(wantedItemsJson);
|
||||||
|
|
||||||
const wantedItems: WantedItem[] = [];
|
const wantedItems: WantedItem[] = [];
|
||||||
|
|
||||||
for (const { itemKindId, amount } of wi) {
|
for (const { itemTypeId, itemKindId, amount } of wi) {
|
||||||
const item = itemStore.getById(itemKindId);
|
const item = itemTypeId != null
|
||||||
|
? itemStore.getById(itemTypeId)
|
||||||
|
: itemStore.getById(itemKindId); // Legacy name.
|
||||||
|
|
||||||
if (item) {
|
if (item) {
|
||||||
wantedItems.push(new WantedItem(item, amount));
|
wantedItems.push(new WantedItem(item, amount));
|
||||||
@ -95,8 +97,8 @@ class HuntOptimizerStore {
|
|||||||
localStorage.setItem(
|
localStorage.setItem(
|
||||||
`HuntOptimizerStore.wantedItems.${Server[applicationStore.currentServer]}`,
|
`HuntOptimizerStore.wantedItems.${Server[applicationStore.currentServer]}`,
|
||||||
JSON.stringify(
|
JSON.stringify(
|
||||||
this.wantedItems.map(({ itemKind, amount }) => ({
|
this.wantedItems.map(({ itemType, amount }) => ({
|
||||||
itemKindId: itemKind.id,
|
itemTypeId: itemType.id,
|
||||||
amount
|
amount
|
||||||
}))
|
}))
|
||||||
)
|
)
|
||||||
@ -114,7 +116,7 @@ class HuntOptimizerStore {
|
|||||||
|
|
||||||
// Initialize this set before awaiting data, so user changes don't affect this optimization
|
// Initialize this set before awaiting data, so user changes don't affect this optimization
|
||||||
// run from this point on.
|
// run from this point on.
|
||||||
const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.itemKind));
|
const wantedItems = new Set(this.wantedItems.filter(w => w.amount > 0).map(w => w.itemType));
|
||||||
|
|
||||||
const methods = await huntMethodStore.methods.current.promise;
|
const methods = await huntMethodStore.methods.current.promise;
|
||||||
const dropTable = (await itemDropStores.current.promise).enemyDrops;
|
const dropTable = (await itemDropStores.current.promise).enemyDrops;
|
||||||
@ -123,7 +125,7 @@ class HuntOptimizerStore {
|
|||||||
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.itemKind.name] = { min: wanted.amount };
|
constraints[wanted.itemType.name] = { min: wanted.amount };
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a variable to the LP model per method per difficulty per section ID.
|
// Add a variable to the LP model per method per difficulty per section ID.
|
||||||
@ -216,9 +218,9 @@ class HuntOptimizerStore {
|
|||||||
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 && wantedItems.has(drop.item)) {
|
if (drop && wantedItems.has(drop.itemType)) {
|
||||||
const value = variable[drop.item.name] || 0;
|
const value = variable[drop.itemType.name] || 0;
|
||||||
variable[drop.item.name] = value + count * drop.rate;
|
variable[drop.itemType.name] = value + count * drop.rate;
|
||||||
addVariable = true;
|
addVariable = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -271,7 +273,7 @@ class HuntOptimizerStore {
|
|||||||
const runs = runsOrOther as number;
|
const runs = runsOrOther as number;
|
||||||
const variable = variables[variableName];
|
const variable = variables[variableName];
|
||||||
|
|
||||||
const items = new Map<ItemKind, number>();
|
const items = new Map<ItemType, 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) {
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import { observable } from "mobx";
|
import { observable } from "mobx";
|
||||||
import { Difficulties, Difficulty, EnemyDrop, NpcType, SectionId, SectionIds, Server, ItemKind } from "../domain";
|
import { Difficulties, Difficulty, EnemyDrop, NpcType, SectionId, SectionIds, Server, ItemType } from "../domain";
|
||||||
import { NpcTypes } from "../domain/NpcType";
|
import { NpcTypes } from "../domain/NpcType";
|
||||||
import { EnemyDropDto } from "../dto";
|
import { EnemyDropDto } from "../dto";
|
||||||
import { Loadable } from "../Loadable";
|
import { Loadable } from "../Loadable";
|
||||||
import { itemKindStores } from "./ItemKindStore";
|
import { itemTypeStores } from "./ItemTypeStore";
|
||||||
import { ServerMap } from "./ServerMap";
|
import { ServerMap } from "./ServerMap";
|
||||||
|
|
||||||
class EnemyDropTable {
|
class EnemyDropTable {
|
||||||
@ -11,8 +11,8 @@ class EnemyDropTable {
|
|||||||
private table: Array<EnemyDrop> =
|
private table: Array<EnemyDrop> =
|
||||||
new Array(Difficulties.length * SectionIds.length * NpcTypes.length);
|
new Array(Difficulties.length * SectionIds.length * NpcTypes.length);
|
||||||
|
|
||||||
// Mapping of ItemKind ids to EnemyDrops.
|
// Mapping of ItemType ids to EnemyDrops.
|
||||||
private itemKindToDrops: Array<Array<EnemyDrop>> = new Array();
|
private itemTypeToDrops: Array<Array<EnemyDrop>> = new Array();
|
||||||
|
|
||||||
getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop | undefined {
|
getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop | undefined {
|
||||||
return this.table[
|
return this.table[
|
||||||
@ -29,18 +29,18 @@ class EnemyDropTable {
|
|||||||
+ npcType.id
|
+ npcType.id
|
||||||
] = drop;
|
] = drop;
|
||||||
|
|
||||||
let drops = this.itemKindToDrops[drop.item.id];
|
let drops = this.itemTypeToDrops[drop.itemType.id];
|
||||||
|
|
||||||
if (!drops) {
|
if (!drops) {
|
||||||
drops = [];
|
drops = [];
|
||||||
this.itemKindToDrops[drop.item.id] = drops;
|
this.itemTypeToDrops[drop.itemType.id] = drops;
|
||||||
}
|
}
|
||||||
|
|
||||||
drops.push(drop);
|
drops.push(drop);
|
||||||
}
|
}
|
||||||
|
|
||||||
getDropsForItemKind(itemKindId: number): Array<EnemyDrop> {
|
getDropsForItemType(itemTypeId: number): Array<EnemyDrop> {
|
||||||
return this.itemKindToDrops[itemKindId] || [];
|
return this.itemTypeToDrops[itemTypeId] || [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -48,7 +48,7 @@ class ItemDropStore {
|
|||||||
@observable enemyDrops: EnemyDropTable = new EnemyDropTable();
|
@observable enemyDrops: EnemyDropTable = new EnemyDropTable();
|
||||||
|
|
||||||
load = async (server: Server): Promise<ItemDropStore> => {
|
load = async (server: Server): Promise<ItemDropStore> => {
|
||||||
const itemKindStore = await itemKindStores.current.promise;
|
const itemTypeStore = await itemTypeStores.current.promise;
|
||||||
const response = await fetch(
|
const response = await fetch(
|
||||||
`${process.env.PUBLIC_URL}/enemyDrops.${Server[server].toLowerCase()}.json`
|
`${process.env.PUBLIC_URL}/enemyDrops.${Server[server].toLowerCase()}.json`
|
||||||
);
|
);
|
||||||
@ -65,10 +65,10 @@ class ItemDropStore {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const difficulty = (Difficulty as any)[dropDto.difficulty];
|
const difficulty = (Difficulty as any)[dropDto.difficulty];
|
||||||
const itemKind = itemKindStore.getById(dropDto.itemKindId);
|
const itemType = itemTypeStore.getById(dropDto.itemTypeId);
|
||||||
|
|
||||||
if (!itemKind) {
|
if (!itemType) {
|
||||||
console.warn(`Couldn't find item kind ${dropDto.itemKindId}.`);
|
console.warn(`Couldn't find item kind ${dropDto.itemTypeId}.`);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,7 +83,7 @@ class ItemDropStore {
|
|||||||
difficulty,
|
difficulty,
|
||||||
sectionId,
|
sectionId,
|
||||||
npcType,
|
npcType,
|
||||||
itemKind,
|
itemType,
|
||||||
dropDto.dropRate,
|
dropDto.dropRate,
|
||||||
dropDto.rareRate
|
dropDto.rareRate
|
||||||
));
|
));
|
||||||
|
@ -1,80 +0,0 @@
|
|||||||
import { observable } from "mobx";
|
|
||||||
import { ItemKind, Server, WeaponItemKind, ArmorItemKind, ShieldItemKind, ToolItemKind, UnitItemKind } from "../domain";
|
|
||||||
import { Loadable } from "../Loadable";
|
|
||||||
import { ServerMap } from "./ServerMap";
|
|
||||||
import { ItemKindDto } from "../dto";
|
|
||||||
|
|
||||||
class ItemKindStore {
|
|
||||||
private idToItemKind: Array<ItemKind> = [];
|
|
||||||
|
|
||||||
@observable itemKinds: Array<ItemKind> = [];
|
|
||||||
|
|
||||||
getById(id: number): ItemKind | undefined {
|
|
||||||
return this.idToItemKind[id];
|
|
||||||
}
|
|
||||||
|
|
||||||
load = async (server: Server): Promise<ItemKindStore> => {
|
|
||||||
const response = await fetch(
|
|
||||||
`${process.env.PUBLIC_URL}/itemKinds.${Server[server].toLowerCase()}.json`
|
|
||||||
);
|
|
||||||
const data: Array<ItemKindDto> = await response.json();
|
|
||||||
|
|
||||||
const itemKinds = new Array<ItemKind>();
|
|
||||||
|
|
||||||
for (const itemKindDto of data) {
|
|
||||||
let itemKind: ItemKind;
|
|
||||||
|
|
||||||
switch (itemKindDto.type) {
|
|
||||||
case 'weapon':
|
|
||||||
itemKind = new WeaponItemKind(
|
|
||||||
itemKindDto.id,
|
|
||||||
itemKindDto.name,
|
|
||||||
itemKindDto.minAtp,
|
|
||||||
itemKindDto.maxAtp,
|
|
||||||
itemKindDto.ata,
|
|
||||||
itemKindDto.maxGrind,
|
|
||||||
itemKindDto.requiredAtp,
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case 'armor':
|
|
||||||
itemKind = new ArmorItemKind(
|
|
||||||
itemKindDto.id,
|
|
||||||
itemKindDto.name,
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case 'shield':
|
|
||||||
itemKind = new ShieldItemKind(
|
|
||||||
itemKindDto.id,
|
|
||||||
itemKindDto.name,
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case 'unit':
|
|
||||||
itemKind = new UnitItemKind(
|
|
||||||
itemKindDto.id,
|
|
||||||
itemKindDto.name,
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case 'tool':
|
|
||||||
itemKind = new ToolItemKind(
|
|
||||||
itemKindDto.id,
|
|
||||||
itemKindDto.name,
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.idToItemKind[itemKind.id] = itemKind;
|
|
||||||
itemKinds.push(itemKind);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.itemKinds = itemKinds;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export const itemKindStores: ServerMap<Loadable<ItemKindStore>> = new ServerMap(server => {
|
|
||||||
const store = new ItemKindStore();
|
|
||||||
return new Loadable(store, () => store.load(server));
|
|
||||||
});
|
|
80
src/stores/ItemTypeStore.ts
Normal file
80
src/stores/ItemTypeStore.ts
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
import { observable } from "mobx";
|
||||||
|
import { ItemType, Server, WeaponItemType, ArmorItemType, ShieldItemType, ToolItemType, UnitItemType } from "../domain";
|
||||||
|
import { Loadable } from "../Loadable";
|
||||||
|
import { ServerMap } from "./ServerMap";
|
||||||
|
import { ItemTypeDto } from "../dto";
|
||||||
|
|
||||||
|
class ItemTypeStore {
|
||||||
|
private idToItemType: Array<ItemType> = [];
|
||||||
|
|
||||||
|
@observable itemTypes: Array<ItemType> = [];
|
||||||
|
|
||||||
|
getById(id: number): ItemType | undefined {
|
||||||
|
return this.idToItemType[id];
|
||||||
|
}
|
||||||
|
|
||||||
|
load = async (server: Server): Promise<ItemTypeStore> => {
|
||||||
|
const response = await fetch(
|
||||||
|
`${process.env.PUBLIC_URL}/itemTypes.${Server[server].toLowerCase()}.json`
|
||||||
|
);
|
||||||
|
const data: Array<ItemTypeDto> = await response.json();
|
||||||
|
|
||||||
|
const itemTypes = new Array<ItemType>();
|
||||||
|
|
||||||
|
for (const itemTypeDto of data) {
|
||||||
|
let itemType: ItemType;
|
||||||
|
|
||||||
|
switch (itemTypeDto.class) {
|
||||||
|
case 'weapon':
|
||||||
|
itemType = new WeaponItemType(
|
||||||
|
itemTypeDto.id,
|
||||||
|
itemTypeDto.name,
|
||||||
|
itemTypeDto.minAtp,
|
||||||
|
itemTypeDto.maxAtp,
|
||||||
|
itemTypeDto.ata,
|
||||||
|
itemTypeDto.maxGrind,
|
||||||
|
itemTypeDto.requiredAtp,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'armor':
|
||||||
|
itemType = new ArmorItemType(
|
||||||
|
itemTypeDto.id,
|
||||||
|
itemTypeDto.name,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'shield':
|
||||||
|
itemType = new ShieldItemType(
|
||||||
|
itemTypeDto.id,
|
||||||
|
itemTypeDto.name,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'unit':
|
||||||
|
itemType = new UnitItemType(
|
||||||
|
itemTypeDto.id,
|
||||||
|
itemTypeDto.name,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case 'tool':
|
||||||
|
itemType = new ToolItemType(
|
||||||
|
itemTypeDto.id,
|
||||||
|
itemTypeDto.name,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.idToItemType[itemType.id] = itemType;
|
||||||
|
itemTypes.push(itemType);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.itemTypes = itemTypes;
|
||||||
|
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const itemTypeStores: ServerMap<Loadable<ItemTypeStore>> = new ServerMap(server => {
|
||||||
|
const store = new ItemTypeStore();
|
||||||
|
return new Loadable(store, () => store.load(server));
|
||||||
|
});
|
@ -3,7 +3,7 @@ import { observer } from "mobx-react";
|
|||||||
import React from "react";
|
import React from "react";
|
||||||
import { AutoSizer, Column, Table, TableCellRenderer } from "react-virtualized";
|
import { AutoSizer, Column, Table, TableCellRenderer } from "react-virtualized";
|
||||||
import { huntOptimizerStore, WantedItem } from "../../stores/HuntOptimizerStore";
|
import { huntOptimizerStore, WantedItem } from "../../stores/HuntOptimizerStore";
|
||||||
import { itemKindStores } from "../../stores/ItemKindStore";
|
import { itemTypeStores } from "../../stores/ItemTypeStore";
|
||||||
import { BigSelect } from "../BigSelect";
|
import { BigSelect } from "../BigSelect";
|
||||||
import './WantedItemsComponent.less';
|
import './WantedItemsComponent.less';
|
||||||
|
|
||||||
@ -35,9 +35,9 @@ export class WantedItemsComponent extends React.Component {
|
|||||||
placeholder="Add an item"
|
placeholder="Add an item"
|
||||||
value={undefined}
|
value={undefined}
|
||||||
style={{ width: 200 }}
|
style={{ width: 200 }}
|
||||||
options={huntOptimizerStore.huntableItems.map(itemKind => ({
|
options={huntOptimizerStore.huntableItemTypes.map(itemType => ({
|
||||||
label: itemKind.name,
|
label: itemType.name,
|
||||||
value: itemKind.id
|
value: itemType.id
|
||||||
}))}
|
}))}
|
||||||
onChange={this.addWanted}
|
onChange={this.addWanted}
|
||||||
/>
|
/>
|
||||||
@ -74,7 +74,7 @@ export class WantedItemsComponent extends React.Component {
|
|||||||
width={150}
|
width={150}
|
||||||
flexGrow={1}
|
flexGrow={1}
|
||||||
cellDataGetter={({ rowData }) =>
|
cellDataGetter={({ rowData }) =>
|
||||||
(rowData as WantedItem).itemKind.name
|
(rowData as WantedItem).itemType.name
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<Column
|
<Column
|
||||||
@ -92,11 +92,11 @@ export class WantedItemsComponent extends React.Component {
|
|||||||
|
|
||||||
private addWanted = (selected: any) => {
|
private addWanted = (selected: any) => {
|
||||||
if (selected) {
|
if (selected) {
|
||||||
let added = huntOptimizerStore.wantedItems.find(w => w.itemKind.id === selected.value);
|
let added = huntOptimizerStore.wantedItems.find(w => w.itemType.id === selected.value);
|
||||||
|
|
||||||
if (!added) {
|
if (!added) {
|
||||||
const itemKind = itemKindStores.current.value.getById(selected.value)!;
|
const itemType = itemTypeStores.current.value.getById(selected.value)!;
|
||||||
huntOptimizerStore.wantedItems.push(new WantedItem(itemKind, 1));
|
huntOptimizerStore.wantedItems.push(new WantedItem(itemType, 1));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,13 +2,13 @@ import cheerio from 'cheerio';
|
|||||||
import fs from 'fs';
|
import fs from 'fs';
|
||||||
import 'isomorphic-fetch';
|
import 'isomorphic-fetch';
|
||||||
import { Difficulty, NpcType, SectionId, SectionIds } from '../src/domain';
|
import { Difficulty, NpcType, SectionId, SectionIds } from '../src/domain';
|
||||||
import { BoxDropDto, EnemyDropDto, ItemKindDto } from '../src/dto';
|
import { BoxDropDto, EnemyDropDto, ItemTypeDto } from '../src/dto';
|
||||||
|
|
||||||
export async function updateDropsFromWebsite(items: ItemKindDto[]) {
|
export async function updateDropsFromWebsite(itemTypes: ItemTypeDto[]) {
|
||||||
const normal = await download(items, Difficulty.Normal);
|
const normal = await download(itemTypes, Difficulty.Normal);
|
||||||
const hard = await download(items, Difficulty.Hard);
|
const hard = await download(itemTypes, Difficulty.Hard);
|
||||||
const vhard = await download(items, Difficulty.VHard, 'very-hard');
|
const vhard = await download(itemTypes, Difficulty.VHard, 'very-hard');
|
||||||
const ultimate = await download(items, Difficulty.Ultimate);
|
const ultimate = await download(itemTypes, Difficulty.Ultimate);
|
||||||
|
|
||||||
const enemyJson = JSON.stringify([
|
const enemyJson = JSON.stringify([
|
||||||
...normal.enemyDrops,
|
...normal.enemyDrops,
|
||||||
@ -30,7 +30,7 @@ export async function updateDropsFromWebsite(items: ItemKindDto[]) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function download(
|
async function download(
|
||||||
items: ItemKindDto[],
|
itemTypes: ItemTypeDto[],
|
||||||
difficulty: Difficulty,
|
difficulty: Difficulty,
|
||||||
difficultyUrl: string = Difficulty[difficulty].toLowerCase()
|
difficultyUrl: string = Difficulty[difficulty].toLowerCase()
|
||||||
) {
|
) {
|
||||||
@ -109,10 +109,10 @@ async function download(
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const itemKind = items.find(i => i.name === item);
|
const itemType = itemTypes.find(i => i.name === item);
|
||||||
|
|
||||||
if (!itemKind) {
|
if (!itemType) {
|
||||||
throw new Error(`No item kind found with name "${item}".`)
|
throw new Error(`No item type found with name "${item}".`)
|
||||||
}
|
}
|
||||||
|
|
||||||
const npcType = NpcType.byNameAndEpisode(enemyOrBox, episode);
|
const npcType = NpcType.byNameAndEpisode(enemyOrBox, episode);
|
||||||
@ -132,7 +132,7 @@ async function download(
|
|||||||
episode,
|
episode,
|
||||||
sectionId: SectionId[sectionId],
|
sectionId: SectionId[sectionId],
|
||||||
enemy: npcType.code,
|
enemy: npcType.code,
|
||||||
itemKindId: itemKind.id,
|
itemTypeId: itemType.id,
|
||||||
dropRate: dropRateNum / dropRateDenom,
|
dropRate: dropRateNum / dropRateDenom,
|
||||||
rareRate: rareRateNum / rareRateDenom,
|
rareRate: rareRateNum / rareRateDenom,
|
||||||
});
|
});
|
||||||
|
@ -4,7 +4,7 @@ import { parseItemPmt } from '../src/bin-data/parsing/itempmt';
|
|||||||
import { parseUnitxt, Unitxt } from '../src/bin-data/parsing/unitxt';
|
import { parseUnitxt, Unitxt } from '../src/bin-data/parsing/unitxt';
|
||||||
import { Difficulties, Difficulty, Episode, Episodes, NpcType, SectionId, SectionIds } from '../src/domain';
|
import { Difficulties, Difficulty, Episode, Episodes, NpcType, SectionId, SectionIds } from '../src/domain';
|
||||||
import { NpcTypes } from '../src/domain/NpcType';
|
import { NpcTypes } from '../src/domain/NpcType';
|
||||||
import { BoxDropDto, EnemyDropDto, ItemKindDto } from '../src/dto';
|
import { BoxDropDto, EnemyDropDto, ItemTypeDto } from '../src/dto';
|
||||||
import { updateDropsFromWebsite } from './updateDropsEphinea';
|
import { updateDropsFromWebsite } from './updateDropsEphinea';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -48,13 +48,13 @@ async function loadUnitxt(): Promise<Unitxt> {
|
|||||||
return unitxt;
|
return unitxt;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
async function updateItems(itemNames: Array<string>): Promise<ItemTypeDto[]> {
|
||||||
const buf = await fs.promises.readFile(
|
const buf = await fs.promises.readFile(
|
||||||
`${RESOURCE_DIR}/ship-config/param/ItemPMT.bin`
|
`${RESOURCE_DIR}/ship-config/param/ItemPMT.bin`
|
||||||
);
|
);
|
||||||
|
|
||||||
const itemPmt = parseItemPmt(new ArrayBufferCursor(buf.buffer, true));
|
const itemPmt = parseItemPmt(new ArrayBufferCursor(buf.buffer, true));
|
||||||
const items = new Array<ItemKindDto>();
|
const itemTypes = new Array<ItemTypeDto>();
|
||||||
const ids = new Set<number>();
|
const ids = new Set<number>();
|
||||||
|
|
||||||
itemPmt.weapons.forEach((category, categoryI) => {
|
itemPmt.weapons.forEach((category, categoryI) => {
|
||||||
@ -63,8 +63,8 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
|
|
||||||
if (!ids.has(id)) {
|
if (!ids.has(id)) {
|
||||||
ids.add(id);
|
ids.add(id);
|
||||||
items.push({
|
itemTypes.push({
|
||||||
type: 'weapon',
|
class: 'weapon',
|
||||||
id,
|
id,
|
||||||
name: itemNames[weapon.id],
|
name: itemNames[weapon.id],
|
||||||
minAtp: weapon.minAtp,
|
minAtp: weapon.minAtp,
|
||||||
@ -82,8 +82,8 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
|
|
||||||
if (!ids.has(id)) {
|
if (!ids.has(id)) {
|
||||||
ids.add(id);
|
ids.add(id);
|
||||||
items.push({
|
itemTypes.push({
|
||||||
type: 'armor',
|
class: 'armor',
|
||||||
id,
|
id,
|
||||||
name: itemNames[armor.id],
|
name: itemNames[armor.id],
|
||||||
});
|
});
|
||||||
@ -95,8 +95,8 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
|
|
||||||
if (!ids.has(id)) {
|
if (!ids.has(id)) {
|
||||||
ids.add(id);
|
ids.add(id);
|
||||||
items.push({
|
itemTypes.push({
|
||||||
type: 'shield',
|
class: 'shield',
|
||||||
id,
|
id,
|
||||||
name: itemNames[shield.id],
|
name: itemNames[shield.id],
|
||||||
});
|
});
|
||||||
@ -108,8 +108,8 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
|
|
||||||
if (!ids.has(id)) {
|
if (!ids.has(id)) {
|
||||||
ids.add(id);
|
ids.add(id);
|
||||||
items.push({
|
itemTypes.push({
|
||||||
type: 'unit',
|
class: 'unit',
|
||||||
id,
|
id,
|
||||||
name: itemNames[unit.id],
|
name: itemNames[unit.id],
|
||||||
});
|
});
|
||||||
@ -122,8 +122,8 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
|
|
||||||
if (!ids.has(id)) {
|
if (!ids.has(id)) {
|
||||||
ids.add(id);
|
ids.add(id);
|
||||||
items.push({
|
itemTypes.push({
|
||||||
type: 'tool',
|
class: 'tool',
|
||||||
id,
|
id,
|
||||||
name: itemNames[tool.id],
|
name: itemNames[tool.id],
|
||||||
});
|
});
|
||||||
@ -132,11 +132,11 @@ async function updateItems(itemNames: Array<string>): Promise<ItemKindDto[]> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await fs.promises.writeFile(
|
await fs.promises.writeFile(
|
||||||
`${PUBLIC_DIR}/itemKinds.ephinea.json`,
|
`${PUBLIC_DIR}/itemTypes.ephinea.json`,
|
||||||
JSON.stringify(items, null, 4)
|
JSON.stringify(itemTypes, null, 4)
|
||||||
);
|
);
|
||||||
|
|
||||||
return items;
|
return itemTypes;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function updateDrops(itemPt: ItemPt) {
|
async function updateDrops(itemPt: ItemPt) {
|
||||||
@ -408,18 +408,18 @@ async function loadEnemyDrops(
|
|||||||
|
|
||||||
if (enemy) {
|
if (enemy) {
|
||||||
const rareRate = expandDropRate(parseInt(prevLine, 10));
|
const rareRate = expandDropRate(parseInt(prevLine, 10));
|
||||||
const itemId = parseInt(trimmed, 16);
|
const itemTypeId = parseInt(trimmed, 16);
|
||||||
const dar = itemPt[episode][difficulty][sectionId].darTable.get(enemy);
|
const dar = itemPt[episode][difficulty][sectionId].darTable.get(enemy);
|
||||||
|
|
||||||
if (dar == null) {
|
if (dar == null) {
|
||||||
console.error(`No DAR found for ${enemy.name}.`);
|
console.error(`No DAR found for ${enemy.name}.`);
|
||||||
} else if (rareRate > 0 && itemId) {
|
} else if (rareRate > 0 && itemTypeId) {
|
||||||
drops.push({
|
drops.push({
|
||||||
difficulty: Difficulty[difficulty],
|
difficulty: Difficulty[difficulty],
|
||||||
episode: episode,
|
episode: episode,
|
||||||
sectionId: SectionId[sectionId],
|
sectionId: SectionId[sectionId],
|
||||||
enemy: enemy.code,
|
enemy: enemy.code,
|
||||||
itemKindId: itemId,
|
itemTypeId,
|
||||||
dropRate: dar,
|
dropRate: dar,
|
||||||
rareRate,
|
rareRate,
|
||||||
});
|
});
|
||||||
@ -455,15 +455,15 @@ async function loadBoxDrops(
|
|||||||
if (lineNo % 3 == 2) {
|
if (lineNo % 3 == 2) {
|
||||||
const areaId = parseInt(prevPrevLine, 10);
|
const areaId = parseInt(prevPrevLine, 10);
|
||||||
const dropRate = expandDropRate(parseInt(prevLine, 10));
|
const dropRate = expandDropRate(parseInt(prevLine, 10));
|
||||||
const itemId = parseInt(trimmed, 16);
|
const itemTypeId = parseInt(trimmed, 16);
|
||||||
|
|
||||||
if (dropRate > 0 && itemId) {
|
if (dropRate > 0 && itemTypeId) {
|
||||||
drops.push({
|
drops.push({
|
||||||
difficulty: Difficulty[difficulty],
|
difficulty: Difficulty[difficulty],
|
||||||
episode: episode,
|
episode: episode,
|
||||||
sectionId: SectionId[sectionId],
|
sectionId: SectionId[sectionId],
|
||||||
areaId,
|
areaId,
|
||||||
itemKindId: itemId,
|
itemTypeId,
|
||||||
dropRate,
|
dropRate,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user