346 lines
8.0 KiB
TypeScript
346 lines
8.0 KiB
TypeScript
import { ALL_STATS, Skill, Stat, SuccessorOption, Wish } from "./datatypes.ts";
|
|
import { getSkills } from "./skills.ts";
|
|
import { getThralls, ItemStage, LifeStage, Thrall } from "./thralls.ts";
|
|
|
|
export class PlayerProgress {
|
|
#name: string;
|
|
#thrallTemplate: number;
|
|
#nImprovements: number;
|
|
#stats: Record<Stat, number>;
|
|
#talents: Record<Stat, number>;
|
|
#isInPenance: boolean;
|
|
#wish: Wish | null;
|
|
#exp: number;
|
|
#blood: number;
|
|
#itemsPurloined: number;
|
|
#skillsLearned: number[]; // use the raw ID representation for indexOf
|
|
#untrimmedSkillsAvailable: Skill[];
|
|
#thrallsUnlocked: number[];
|
|
#thrallDamage: Record<number, number>;
|
|
#thrallsObtainedItem: number[];
|
|
#thrallsDeliveredItem: number[];
|
|
|
|
constructor(asSuccessor: SuccessorOption, withWish: Wish | null) {
|
|
this.#name = asSuccessor.name;
|
|
this.#thrallTemplate = asSuccessor.template.id;
|
|
this.#nImprovements = asSuccessor.nImprovements;
|
|
this.#stats = { ...asSuccessor.stats };
|
|
this.#talents = { ...asSuccessor.talents };
|
|
this.#isInPenance = asSuccessor.inPenance;
|
|
this.#wish = withWish;
|
|
this.#exp = 0;
|
|
this.#blood = 0;
|
|
this.#itemsPurloined = 0;
|
|
this.#skillsLearned = [];
|
|
this.#untrimmedSkillsAvailable = [];
|
|
this.#thrallsUnlocked = [];
|
|
this.#thrallDamage = {};
|
|
this.#thrallsObtainedItem = [];
|
|
this.#thrallsDeliveredItem = [];
|
|
|
|
this.refill();
|
|
}
|
|
|
|
applyEndOfTurn() {
|
|
for (let stat of ALL_STATS.values()) {
|
|
this.add(stat, this.#talents[stat]);
|
|
}
|
|
}
|
|
|
|
get name(): string {
|
|
return this.#name;
|
|
}
|
|
|
|
get template(): Thrall {
|
|
return { id: this.#thrallTemplate };
|
|
}
|
|
|
|
get nImprovements(): number {
|
|
return this.#nImprovements;
|
|
}
|
|
|
|
get isInPenance(): boolean {
|
|
return this.#isInPenance;
|
|
}
|
|
|
|
refill() {
|
|
this.#blood = 1000;
|
|
|
|
let learnableSkills = []; // TODO: Also include costing info
|
|
for (let skill of getSkills()
|
|
.getAvailableSkills(this.#isInPenance)
|
|
.values()) {
|
|
if (this.#canBeAvailable(skill)) {
|
|
learnableSkills.push(skill);
|
|
}
|
|
}
|
|
|
|
for (let thrall of getThralls().getAll()) {
|
|
let stage = this.getThrallLifeStage(thrall);
|
|
if (stage == LifeStage.Vampirized || stage == LifeStage.Dead) {
|
|
continue;
|
|
}
|
|
this.#thrallDamage[thrall.id] = Math.max(
|
|
this.#thrallDamage[thrall.id] ?? 0 - 0.2,
|
|
0.0,
|
|
);
|
|
}
|
|
|
|
this.#untrimmedSkillsAvailable = learnableSkills;
|
|
}
|
|
|
|
hasLearned(skill: Skill) {
|
|
return this.#skillsLearned.indexOf(skill.id) !== -1;
|
|
}
|
|
|
|
learnSkill(skill: Skill) {
|
|
if (this.#skillsLearned.indexOf(skill.id) != -1) {
|
|
return;
|
|
}
|
|
this.#skillsLearned.push(skill.id);
|
|
|
|
// remove entries for that skill
|
|
let skills2 = [];
|
|
for (let entry of this.#untrimmedSkillsAvailable.values()) {
|
|
if (entry.id == skill.id) {
|
|
continue;
|
|
}
|
|
skills2.push(entry);
|
|
}
|
|
this.#untrimmedSkillsAvailable = skills2;
|
|
}
|
|
|
|
#canBeAvailable(skill: Skill) {
|
|
// make sure we haven't learned this skill already
|
|
if (this.hasLearned(skill)) {
|
|
return false;
|
|
}
|
|
|
|
let data = getSkills().get(skill);
|
|
|
|
// make sure the prereqs are met
|
|
for (let prereq of data.prereqs.values()) {
|
|
if (!this.hasLearned(prereq)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// ok, we're good!!
|
|
return true;
|
|
}
|
|
|
|
purloinItem() {
|
|
this.#itemsPurloined += 1;
|
|
}
|
|
|
|
getItemsPurloined() {
|
|
return this.#itemsPurloined;
|
|
}
|
|
|
|
add(stat: Stat, amount: number) {
|
|
if (amount != Math.floor(amount)) {
|
|
throw `stat increment must be integer: ${amount}`;
|
|
}
|
|
this.#stats[stat] += amount;
|
|
this.#stats[stat] = Math.min(Math.max(this.#stats[stat], -99), 999);
|
|
}
|
|
|
|
addExperience(amt: number) {
|
|
this.#exp += amt;
|
|
}
|
|
|
|
getExperience(): number {
|
|
return this.#exp;
|
|
}
|
|
|
|
spendExperience(cost: number) {
|
|
if (this.#exp < cost) {
|
|
throw `can't spend ${cost}`;
|
|
}
|
|
this.#exp -= cost;
|
|
}
|
|
|
|
getStat(stat: Stat): number {
|
|
return this.#stats[stat];
|
|
}
|
|
|
|
getTalent(stat: Stat): number {
|
|
return this.#talents[stat];
|
|
}
|
|
|
|
getBlood(): number {
|
|
return Math.floor(Math.max(this.#blood, 0));
|
|
}
|
|
|
|
addBlood(amt: number) {
|
|
this.#blood += amt;
|
|
this.#blood = Math.min(this.#blood, 5000);
|
|
}
|
|
|
|
spendBlood(amt: number) {
|
|
this.#blood -= amt;
|
|
}
|
|
|
|
getWish(): Wish | null {
|
|
return this.#wish;
|
|
}
|
|
|
|
getAvailableSkills(): Skill[] {
|
|
// Sort by cost, then by name, then trim down to first 6
|
|
let skillsAvailable = [...this.#untrimmedSkillsAvailable];
|
|
skillsAvailable.sort((a, b) => {
|
|
let name1 = getSkills().get(a).profile.name;
|
|
let name2 = getSkills().get(b).profile.name;
|
|
|
|
if (name1 < name2) {
|
|
return -1;
|
|
}
|
|
if (name1 > name2) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
});
|
|
skillsAvailable.sort((a, b) => {
|
|
return getSkills().computeCost(a) - getSkills().computeCost(b);
|
|
});
|
|
return skillsAvailable.slice(0, 6);
|
|
}
|
|
|
|
getUntrimmedAvailableSkillIds(): number[] {
|
|
return this.#untrimmedSkillsAvailable.map((s) => s.id);
|
|
}
|
|
|
|
getLearnedSkills() : Skill[] {
|
|
let learnedSkills = [];
|
|
for (let s of this.#skillsLearned.values()) {
|
|
learnedSkills.push({ id: s });
|
|
}
|
|
return learnedSkills;
|
|
}
|
|
|
|
getRawLearnedSkills() : number[] {
|
|
return [...this.#skillsLearned];
|
|
}
|
|
|
|
getStats() {
|
|
return { ...this.#stats };
|
|
}
|
|
getTalents() {
|
|
return { ...this.#talents };
|
|
}
|
|
|
|
unlockThrall(thrall: Thrall) {
|
|
let { id } = thrall;
|
|
if (this.#thrallsUnlocked.indexOf(id) != -1) {
|
|
return;
|
|
}
|
|
this.#thrallsUnlocked.push(id);
|
|
}
|
|
|
|
isThrallUnlocked(thrall: Thrall) {
|
|
return this.#thrallsUnlocked.indexOf(thrall.id) != -1;
|
|
}
|
|
|
|
getUnlockedThrallIds() : number[] {
|
|
return [...this.#thrallsUnlocked];
|
|
}
|
|
|
|
damageThrall(thrall: Thrall, amount: number) {
|
|
if (amount <= 0.0) {
|
|
throw new Error(`damage must be some positive amount, not ${amount}`);
|
|
}
|
|
let stage = this.getThrallLifeStage(thrall);
|
|
|
|
if (stage == LifeStage.Vampirized) {
|
|
this.#thrallDamage[thrall.id] = 4.0;
|
|
}
|
|
this.#thrallDamage[thrall.id] =
|
|
(this.#thrallDamage[thrall.id] ?? 0.0) + amount;
|
|
}
|
|
|
|
getThrallDamage(thrall: Thrall): number {
|
|
return this.#thrallDamage[thrall.id] ?? 0.0;
|
|
}
|
|
|
|
getThrallLifeStage(thrall: Thrall): LifeStage {
|
|
let damage = this.#thrallDamage[thrall.id] ?? 0;
|
|
if (damage < 0.5) {
|
|
return LifeStage.Fresh;
|
|
}
|
|
if (damage < 1.75) {
|
|
return LifeStage.Average;
|
|
}
|
|
if (damage < 3.0) {
|
|
return LifeStage.Poor;
|
|
}
|
|
if (damage < 4.0) {
|
|
return LifeStage.Vampirized;
|
|
}
|
|
return LifeStage.Dead;
|
|
}
|
|
|
|
obtainThrallItem(thrall: Thrall) {
|
|
if (this.#thrallsObtainedItem.indexOf(thrall.id) != -1) {
|
|
return;
|
|
}
|
|
this.#thrallsObtainedItem.push(thrall.id);
|
|
}
|
|
|
|
getThrallObtainedItemIds() : number[] {
|
|
return [...this.#thrallsObtainedItem];
|
|
}
|
|
|
|
deliverThrallItem(thrall: Thrall) {
|
|
if (this.#thrallsDeliveredItem.indexOf(thrall.id) != -1) {
|
|
return;
|
|
}
|
|
this.#thrallsDeliveredItem.push(thrall.id);
|
|
}
|
|
|
|
getThrallDeliveredItemIds() : number[] {
|
|
return [...this.#thrallsDeliveredItem];
|
|
}
|
|
|
|
getThrallItemStage(thrall: Thrall): ItemStage {
|
|
if (this.#thrallsDeliveredItem.indexOf(thrall.id) != -1) {
|
|
return ItemStage.Delivered;
|
|
}
|
|
if (this.#thrallsObtainedItem.indexOf(thrall.id) != -1) {
|
|
return ItemStage.Obtained;
|
|
}
|
|
return ItemStage.Untouched;
|
|
}
|
|
|
|
anyAffordableSkillsAtMinimum() {
|
|
let skills = this.getAvailableSkills();
|
|
for (let skill of skills.values()) {
|
|
if (getSkills().isAtMinimum(skill)) {
|
|
if (
|
|
getPlayerProgress().getExperience() > getSkills().computeCost(skill)
|
|
) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
let active: PlayerProgress | null = null;
|
|
|
|
export function initPlayerProgress(
|
|
asSuccessor: SuccessorOption,
|
|
withWish: Wish | null,
|
|
) {
|
|
active = new PlayerProgress(asSuccessor, withWish);
|
|
}
|
|
|
|
export function getPlayerProgress(): PlayerProgress {
|
|
if (active == null) {
|
|
throw new Error(
|
|
`trying to get player progress before it has been initialized`,
|
|
);
|
|
}
|
|
return active;
|
|
}
|