diff --git a/src/Loadable.ts b/src/Loadable.ts index 2d17b3e9..e15b75a3 100644 --- a/src/Loadable.ts +++ b/src/Loadable.ts @@ -59,7 +59,7 @@ export class Loadable { } /** - * This method returns valid data as soon as possible. + * This property returns valid data as soon as possible. * If the Loadable is uninitialized a data load will be triggered, otherwise the current value will be returned. */ get promise(): Promise { diff --git a/src/domain/NpcType.ts b/src/domain/NpcType.ts index 07dc684b..6f50028b 100644 --- a/src/domain/NpcType.ts +++ b/src/domain/NpcType.ts @@ -447,3 +447,190 @@ export class NpcType { NpcType.SaintMilion.rareType = NpcType.Kondrieu; NpcType.Shambertin.rareType = NpcType.Kondrieu; }()); + +export const NpcTypes: Array = [ + + // + // Unknown NPCs + // + + NpcType.Unknown, + + // + // Friendly NPCs + // + + NpcType.FemaleFat, + NpcType.FemaleMacho, + NpcType.FemaleTall, + NpcType.MaleDwarf, + NpcType.MaleFat, + NpcType.MaleMacho, + NpcType.MaleOld, + NpcType.BlueSoldier, + NpcType.RedSoldier, + NpcType.Principal, + NpcType.Tekker, + NpcType.GuildLady, + NpcType.Scientist, + NpcType.Nurse, + NpcType.Irene, + NpcType.ItemShop, + NpcType.Nurse2, + + // + // Enemy NPCs + // + + // Episode I Forest + + NpcType.Hildebear, + NpcType.Hildeblue, + NpcType.RagRappy, + NpcType.AlRappy, + NpcType.Monest, + NpcType.Mothmant, + NpcType.SavageWolf, + NpcType.BarbarousWolf, + NpcType.Booma, + NpcType.Gobooma, + NpcType.Gigobooma, + NpcType.Dragon, + + // Episode I Caves + + NpcType.GrassAssassin, + NpcType.PoisonLily, + NpcType.NarLily, + NpcType.NanoDragon, + NpcType.EvilShark, + NpcType.PalShark, + NpcType.GuilShark, + NpcType.PofuillySlime, + NpcType.PouillySlime, + NpcType.PanArms, + NpcType.Migium, + NpcType.Hidoom, + NpcType.DeRolLe, + + // Episode I Mines + + NpcType.Dubchic, + NpcType.Gilchic, + NpcType.Garanz, + NpcType.SinowBeat, + NpcType.SinowGold, + NpcType.Canadine, + NpcType.Canane, + NpcType.Dubswitch, + NpcType.VolOpt, + + // Episode I Ruins + + NpcType.Delsaber, + NpcType.ChaosSorcerer, + NpcType.DarkGunner, + NpcType.DeathGunner, + NpcType.ChaosBringer, + NpcType.DarkBelra, + NpcType.Dimenian, + NpcType.LaDimenian, + NpcType.SoDimenian, + NpcType.Bulclaw, + NpcType.Bulk, + NpcType.Claw, + NpcType.DarkFalz, + + // Episode II VR Temple + + NpcType.Hildebear2, + NpcType.Hildeblue2, + NpcType.RagRappy2, + NpcType.LoveRappy, + NpcType.StRappy, + NpcType.HalloRappy, + NpcType.EggRappy, + NpcType.Monest2, + NpcType.Mothmant2, + NpcType.PoisonLily2, + NpcType.NarLily2, + NpcType.GrassAssassin2, + NpcType.Dimenian2, + NpcType.LaDimenian2, + NpcType.SoDimenian2, + NpcType.DarkBelra2, + NpcType.BarbaRay, + + // Episode II VR Spaceship + + NpcType.SavageWolf2, + NpcType.BarbarousWolf2, + NpcType.PanArms2, + NpcType.Migium2, + NpcType.Hidoom2, + NpcType.Dubchic2, + NpcType.Gilchic2, + NpcType.Garanz2, + NpcType.Dubswitch2, + NpcType.Delsaber2, + NpcType.ChaosSorcerer2, + NpcType.GolDragon, + + // Episode II Central Control Area + + NpcType.SinowBerill, + NpcType.SinowSpigell, + NpcType.Merillia, + NpcType.Meriltas, + NpcType.Mericarol, + NpcType.Mericus, + NpcType.Merikle, + NpcType.UlGibbon, + NpcType.ZolGibbon, + NpcType.Gibbles, + NpcType.Gee, + NpcType.GiGue, + NpcType.GalGryphon, + + // Episode II Seabed + + NpcType.Deldepth, + NpcType.Delbiter, + NpcType.Dolmolm, + NpcType.Dolmdarl, + NpcType.Morfos, + NpcType.Recobox, + NpcType.Recon, + NpcType.Epsilon, + NpcType.SinowZoa, + NpcType.SinowZele, + NpcType.IllGill, + NpcType.DelLily, + NpcType.OlgaFlow, + + // Episode IV + + NpcType.SandRappy, + NpcType.DelRappy, + NpcType.Astark, + NpcType.SatelliteLizard, + NpcType.Yowie, + NpcType.MerissaA, + NpcType.MerissaAA, + NpcType.Girtablulu, + NpcType.Zu, + NpcType.Pazuzu, + NpcType.Boota, + NpcType.ZeBoota, + NpcType.BaBoota, + NpcType.Dorphon, + NpcType.DorphonEclair, + NpcType.Goran, + NpcType.PyroGoran, + NpcType.GoranDetonator, + NpcType.SaintMilion, + NpcType.Shambertin, + NpcType.Kondrieu, +]; + +export const EnemyNpcTypes = NpcTypes.filter(type => type.enemy); diff --git a/src/domain/index.ts b/src/domain/index.ts index 42c7526c..670fa845 100644 --- a/src/domain/index.ts +++ b/src/domain/index.ts @@ -338,27 +338,37 @@ export class EnemyDrop implements ItemDrop { } export class HuntMethod { + readonly npcs: Array; + readonly enemies: Array; + readonly enemyCounts: Map; + constructor( /** * The time it takes to complete the quest in hours. */ - public time: number, - public name: string, - public quest: SimpleQuest - ) { } + public readonly time: number, + public readonly name: string, + public readonly quest: SimpleQuest + ) { + if (time <= 0) throw new Error('time must be greater than zero.'); + + this.npcs = this.quest.npcs; + this.enemies = this.npcs.filter(npc => npc.type.enemy); + this.enemyCounts = new Map(); + + for (const npc of this.enemies) { + this.enemyCounts.set(npc.type, (this.enemyCounts.get(npc.type) || 0) + 1); + } + } } export class SimpleQuest { - enemies: SimpleNpc[]; - constructor( - public name: string, - public npcs: SimpleNpc[] + public readonly name: string, + public readonly npcs: SimpleNpc[] ) { if (!name) throw new Error('name is required.'); if (!npcs) throw new Error('npcs is required.'); - - this.enemies = npcs.filter(npc => npc.type.enemy); } } diff --git a/src/stores/HuntOptimizerStore.ts b/src/stores/HuntOptimizerStore.ts index 08a524ac..8b12f076 100644 --- a/src/stores/HuntOptimizerStore.ts +++ b/src/stores/HuntOptimizerStore.ts @@ -131,7 +131,7 @@ class HuntOptimizerStore { // Counts include rare enemies, so they are fractional. const counts = new Map(); - for (const enemy of method.quest.enemies) { + for (const enemy of method.enemies) { const count = counts.get(enemy.type); if (enemy.type.rareType == null) { diff --git a/src/ui/hunt-optimizer/HuntOptimizerComponent.css b/src/ui/hunt-optimizer/HuntOptimizerComponent.css index 8767cb37..2dc3adbf 100644 --- a/src/ui/hunt-optimizer/HuntOptimizerComponent.css +++ b/src/ui/hunt-optimizer/HuntOptimizerComponent.css @@ -5,7 +5,23 @@ margin-top: 10px; } -.ho-HuntOptimizerComponent > *:nth-child(2) { - flex-grow: 1; - overflow: hidden; -} \ No newline at end of file +.ho-HuntOptimizerComponent > .ant-tabs { + flex: 1; + display: flex; + flex-direction: column; + align-items: stretch; +} + +.ho-HuntOptimizerComponent > .ant-tabs > .ant-tabs-content { + flex: 1; + display: flex; + flex-direction: column; + align-items: stretch; +} + +.ho-HuntOptimizerComponent > .ant-tabs > .ant-tabs-content > .ant-tabs-tabpane-active { + flex: 1; + display: flex; + flex-direction: column; + align-items: stretch; +} diff --git a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx b/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx index 2cd9ebf9..e4ea802c 100644 --- a/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx +++ b/src/ui/hunt-optimizer/HuntOptimizerComponent.tsx @@ -1,13 +1,22 @@ +import { Tabs } from "antd"; import React from "react"; import './HuntOptimizerComponent.css'; -import { WantedItemsComponent } from "./WantedItemsComponent"; -import { OptimizationResultComponent } from "./OptimizationResultComponent"; +import { OptimizerComponent } from "./OptimizerComponent"; +import { MethodsComponent } from "./MethodsComponent"; + +const TabPane = Tabs.TabPane; export function HuntOptimizerComponent() { return (
- - + + + + + + + +
); } diff --git a/src/ui/hunt-optimizer/MethodsComponent.css b/src/ui/hunt-optimizer/MethodsComponent.css new file mode 100644 index 00000000..9c441aae --- /dev/null +++ b/src/ui/hunt-optimizer/MethodsComponent.css @@ -0,0 +1,3 @@ +.ho-MethodsComponent { + flex: 1; +} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/MethodsComponent.tsx b/src/ui/hunt-optimizer/MethodsComponent.tsx new file mode 100644 index 00000000..ec22062c --- /dev/null +++ b/src/ui/hunt-optimizer/MethodsComponent.tsx @@ -0,0 +1,110 @@ +import { observer } from "mobx-react"; +import React from "react"; +import { AutoSizer, GridCellRenderer, MultiGrid, Index } from "react-virtualized"; +import { HuntMethod } from "../../domain"; +import { EnemyNpcTypes } from "../../domain/NpcType"; +import { huntMethodStore } from "../../stores/HuntMethodStore"; +import "./MethodsComponent.css"; + +type Column = { + name: string, + width: number, + cellValue: (method: HuntMethod) => string, + className?: string +} + +@observer +export class MethodsComponent extends React.Component { + static columns: Array = (() => { + // Standard columns. + const columns: Column[] = [ + { + name: 'Method', + width: 250, + cellValue: (method) => method.name, + }, + { + name: 'Hours', + width: 50, + cellValue: (method) => method.time.toString(), + }, + ]; + + // One column per enemy type. + for (const enemy of EnemyNpcTypes) { + columns.push({ + name: enemy.name, + width: 50, + cellValue: (method) => { + const count = method.enemyCounts.get(enemy); + return count == null ? '' : count.toString(); + }, + className: 'number', + }); + } + + return columns; + })(); + + render() { + const methods = huntMethodStore.methods.current.value; + + return ( +
+ + {({ width, height }) => ( + + )} + +
+ ); + } + + private columnWidth = ({ index }: Index): number => { + return MethodsComponent.columns[index].width; + } + + private cellRenderer: GridCellRenderer = ({ columnIndex, rowIndex, style }) => { + const column = MethodsComponent.columns[columnIndex]; + let text: string; + const classes = []; + + if (columnIndex === MethodsComponent.columns.length - 1) { + classes.push('last-in-row'); + } + + if (rowIndex === 0) { + // Header row + text = column.name; + } else { + // Method row + if (column.className) { + classes.push(column.className); + } + + const method = huntMethodStore.methods.current.value[rowIndex - 1]; + + text = column.cellValue(method); + } + + return ( +
+ {text} +
+ ); + } +} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/OptimizerComponent.css b/src/ui/hunt-optimizer/OptimizerComponent.css new file mode 100644 index 00000000..1c53b948 --- /dev/null +++ b/src/ui/hunt-optimizer/OptimizerComponent.css @@ -0,0 +1,10 @@ +.ho-OptimizerComponent { + flex: 1; + display: flex; + align-items: stretch; +} + +.ho-OptimizerComponent > *:nth-child(2) { + flex: 1; + overflow: hidden; +} \ No newline at end of file diff --git a/src/ui/hunt-optimizer/OptimizerComponent.tsx b/src/ui/hunt-optimizer/OptimizerComponent.tsx new file mode 100644 index 00000000..994ebf74 --- /dev/null +++ b/src/ui/hunt-optimizer/OptimizerComponent.tsx @@ -0,0 +1,13 @@ +import React from "react"; +import { WantedItemsComponent } from "./WantedItemsComponent"; +import { OptimizationResultComponent } from "./OptimizationResultComponent"; +import "./OptimizerComponent.css"; + +export function OptimizerComponent() { + return ( +
+ + +
+ ); +}