mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Started working on DPS calculator.
This commit is contained in:
parent
d6f422335d
commit
d35b2356e7
125
src/stores/DpsCalcStore.ts
Normal file
125
src/stores/DpsCalcStore.ts
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
import { observable, IObservableArray, computed } from "mobx";
|
||||||
|
|
||||||
|
const NORMAL_DAMAGE_FACTOR = 0.2 * 0.9;
|
||||||
|
const HEAVY_DAMAGE_FACTOR = NORMAL_DAMAGE_FACTOR * 1.89;
|
||||||
|
const SAC_DAMAGE_FACTOR = NORMAL_DAMAGE_FACTOR * 3.32;
|
||||||
|
const VJAYA_DAMAGE_FACTOR = NORMAL_DAMAGE_FACTOR * 5.56;
|
||||||
|
const CRIT_FACTOR = 1.5;
|
||||||
|
|
||||||
|
class WeaponType {
|
||||||
|
constructor(
|
||||||
|
readonly minAtp: number,
|
||||||
|
readonly maxAtp: number
|
||||||
|
) { }
|
||||||
|
}
|
||||||
|
|
||||||
|
class Weapon {
|
||||||
|
readonly type: WeaponType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Integer from 0 to 100.
|
||||||
|
*/
|
||||||
|
@observable attributePercentage: number = 0;
|
||||||
|
@observable grind: number = 0;
|
||||||
|
|
||||||
|
@computed get shiftaAtp(): number {
|
||||||
|
if (this.type.minAtp === this.type.maxAtp) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return this.type.maxAtp * this.store.shiftaFactor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get grindAtp(): number {
|
||||||
|
return 2 * this.grind;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get minAtp(): number {
|
||||||
|
return this.type.minAtp + this.grindAtp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get maxAtp(): number {
|
||||||
|
return this.type.maxAtp + this.grindAtp + this.shiftaAtp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get finalMinAtp(): number {
|
||||||
|
return this.minAtp
|
||||||
|
+ this.store.armorAtp
|
||||||
|
+ this.store.shieldAtp
|
||||||
|
+ this.store.baseAtp
|
||||||
|
+ this.store.baseShiftaAtp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get finalMaxAtp(): number {
|
||||||
|
return this.maxAtp
|
||||||
|
+ this.store.armorAtp
|
||||||
|
+ this.store.shieldAtp
|
||||||
|
+ this.store.baseAtp
|
||||||
|
+ this.store.baseShiftaAtp;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get minNormalDamage(): number {
|
||||||
|
return (this.finalMinAtp - this.store.enemyDfp) * NORMAL_DAMAGE_FACTOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get maxNormalDamage(): number {
|
||||||
|
return (this.finalMaxAtp - this.store.enemyDfp) * NORMAL_DAMAGE_FACTOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get avgNormalDamage(): number {
|
||||||
|
return (this.minNormalDamage + this.maxNormalDamage) / 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get minHeavyDamage(): number {
|
||||||
|
return (this.finalMinAtp - this.store.enemyDfp) * HEAVY_DAMAGE_FACTOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get maxHeavyDamage(): number {
|
||||||
|
return (this.finalMaxAtp - this.store.enemyDfp) * HEAVY_DAMAGE_FACTOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get avgHeavyDamage(): number {
|
||||||
|
return (this.minHeavyDamage + this.maxHeavyDamage) / 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private store: DpsCalcStore,
|
||||||
|
type: WeaponType,
|
||||||
|
) {
|
||||||
|
this.type = type;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class DpsCalcStore {
|
||||||
|
//
|
||||||
|
// Character Details
|
||||||
|
//
|
||||||
|
|
||||||
|
@observable charAtp: number = 0;
|
||||||
|
@observable magPow: number = 0;
|
||||||
|
@observable armorAtp: number = 0;
|
||||||
|
@observable shieldAtp: number = 0;
|
||||||
|
@observable shiftaLvl: number = 0;
|
||||||
|
|
||||||
|
@computed get baseAtp(): number {
|
||||||
|
return this.charAtp + 2 * this.magPow;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get shiftaFactor(): number {
|
||||||
|
return this.shiftaLvl ? 0.013 * (this.shiftaLvl - 1) + 0.1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get baseShiftaAtp(): number {
|
||||||
|
return this.baseAtp * this.shiftaFactor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable readonly weapons: IObservableArray<Weapon> = observable.array();
|
||||||
|
|
||||||
|
//
|
||||||
|
// Enemy Details
|
||||||
|
//
|
||||||
|
|
||||||
|
@observable enemyDfp: number = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const dpsCalcStore = new DpsCalcStore();
|
@ -6,13 +6,15 @@ import './ApplicationComponent.css';
|
|||||||
import { withErrorBoundary } from './ErrorBoundary';
|
import { withErrorBoundary } from './ErrorBoundary';
|
||||||
import { HuntOptimizerComponent } from './hunt-optimizer/HuntOptimizerComponent';
|
import { HuntOptimizerComponent } from './hunt-optimizer/HuntOptimizerComponent';
|
||||||
import { QuestEditorComponent } from './quest-editor/QuestEditorComponent';
|
import { QuestEditorComponent } from './quest-editor/QuestEditorComponent';
|
||||||
|
import { DpsCalcComponent } from './dps-calc/DpsCalcComponent';
|
||||||
|
|
||||||
const QuestEditor = withErrorBoundary(QuestEditorComponent);
|
const QuestEditor = withErrorBoundary(QuestEditorComponent);
|
||||||
const HuntOptimizer = withErrorBoundary(HuntOptimizerComponent);
|
const HuntOptimizer = withErrorBoundary(HuntOptimizerComponent);
|
||||||
|
const DpsCalc = withErrorBoundary(DpsCalcComponent);
|
||||||
|
|
||||||
@observer
|
@observer
|
||||||
export class ApplicationComponent extends React.Component {
|
export class ApplicationComponent extends React.Component {
|
||||||
state = { tool: 'questEditor' }
|
state = { tool: this.initTool() }
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
let toolComponent;
|
let toolComponent;
|
||||||
@ -24,6 +26,9 @@ export class ApplicationComponent extends React.Component {
|
|||||||
case 'huntOptimizer':
|
case 'huntOptimizer':
|
||||||
toolComponent = <HuntOptimizer />;
|
toolComponent = <HuntOptimizer />;
|
||||||
break;
|
break;
|
||||||
|
case 'dpsCalc':
|
||||||
|
toolComponent = <DpsCalc />;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -43,6 +48,9 @@ export class ApplicationComponent extends React.Component {
|
|||||||
<Menu.Item key="huntOptimizer">
|
<Menu.Item key="huntOptimizer">
|
||||||
Hunt Optimizer
|
Hunt Optimizer
|
||||||
</Menu.Item>
|
</Menu.Item>
|
||||||
|
{/* <Menu.Item key="dpsCalc">
|
||||||
|
DPS Calculator
|
||||||
|
</Menu.Item> */}
|
||||||
</Menu>
|
</Menu>
|
||||||
</div>
|
</div>
|
||||||
<div className="ApplicationComponent-main">
|
<div className="ApplicationComponent-main">
|
||||||
@ -55,4 +63,9 @@ export class ApplicationComponent extends React.Component {
|
|||||||
private menuClicked = (e: ClickParam) => {
|
private menuClicked = (e: ClickParam) => {
|
||||||
this.setState({ tool: e.key });
|
this.setState({ tool: e.key });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private initTool(): string {
|
||||||
|
const param = window.location.search.slice(1).split('&').find(p => p.startsWith('tool='));
|
||||||
|
return param ? param.slice(5) : 'questEditor';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
57
src/ui/dps-calc/DpsCalcComponent.tsx
Normal file
57
src/ui/dps-calc/DpsCalcComponent.tsx
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
import { InputNumber } from "antd";
|
||||||
|
import { observer } from "mobx-react";
|
||||||
|
import React from "react";
|
||||||
|
import { dpsCalcStore } from "../../stores/DpsCalcStore";
|
||||||
|
|
||||||
|
@observer
|
||||||
|
export class DpsCalcComponent extends React.Component {
|
||||||
|
render() {
|
||||||
|
return (
|
||||||
|
<section>
|
||||||
|
<section>
|
||||||
|
<div>Character ATP:</div>
|
||||||
|
<InputNumber
|
||||||
|
value={dpsCalcStore.charAtp}
|
||||||
|
min={0}
|
||||||
|
step={1}
|
||||||
|
onChange={(value) => dpsCalcStore.charAtp = value || 0}
|
||||||
|
/>
|
||||||
|
<div>MAG POW:</div>
|
||||||
|
<InputNumber
|
||||||
|
value={dpsCalcStore.magPow}
|
||||||
|
min={0}
|
||||||
|
max={200}
|
||||||
|
step={1}
|
||||||
|
onChange={(value) => dpsCalcStore.magPow = value || 0}
|
||||||
|
/>
|
||||||
|
<div>Armor ATP:</div>
|
||||||
|
<InputNumber
|
||||||
|
value={dpsCalcStore.armorAtp}
|
||||||
|
min={0}
|
||||||
|
step={1}
|
||||||
|
onChange={(value) => dpsCalcStore.armorAtp = value || 0}
|
||||||
|
/>
|
||||||
|
<div>Shield ATP:</div>
|
||||||
|
<InputNumber
|
||||||
|
value={dpsCalcStore.shieldAtp}
|
||||||
|
min={0}
|
||||||
|
step={1}
|
||||||
|
onChange={(value) => dpsCalcStore.shieldAtp = value || 0}
|
||||||
|
/>
|
||||||
|
<div>Shifta level:</div>
|
||||||
|
<InputNumber
|
||||||
|
value={dpsCalcStore.shiftaLvl}
|
||||||
|
min={0}
|
||||||
|
max={30}
|
||||||
|
step={1}
|
||||||
|
onChange={(value) => dpsCalcStore.shiftaLvl = value || 0}
|
||||||
|
/>
|
||||||
|
<div>Shifta factor:</div>
|
||||||
|
<div>{dpsCalcStore.shiftaFactor.toFixed(3)}</div>
|
||||||
|
<div>Base shifta ATP:</div>
|
||||||
|
<div>{dpsCalcStore.baseShiftaAtp.toFixed(2)}</div>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user