prototype for writing a save Merge branch 'main' into savesystem violently read player from file oops, missed revisions in StateManager create StateManager from file autoformat the world oops, forgot to save the split-up of save.ts Save on end-of-day, or after endgame. Putting it here avoids a circular reference problem Merge branch 'main' into savesystem Integrate save system Deal with save corruption correctly Co-authored-by: Kistaro Windrider <kistaro@gmail.com> Reviewed-on: #42 Co-authored-by: Nyeogmi <economicsbat@gmail.com> Co-committed-by: Nyeogmi <economicsbat@gmail.com>
396 lines
9.7 KiB
TypeScript
396 lines
9.7 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";
|
|
import { SaveFileV1, mustBeSaveFileV1 } from "./saveformat.ts";
|
|
|
|
interface NewRoundConfig {
|
|
asSuccessor: SuccessorOption;
|
|
withWish: Wish | null;
|
|
}
|
|
|
|
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(args: NewRoundConfig | SaveFileV1) {
|
|
if ("asSuccessor" in args) {
|
|
//asSuccessor: SuccessorOption, withWish: Wish | null) {
|
|
const config = args as NewRoundConfig;
|
|
const asSuccessor = config.asSuccessor;
|
|
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 = config.withWish;
|
|
this.#exp = 0;
|
|
this.#blood = 0;
|
|
this.#itemsPurloined = 0;
|
|
this.#skillsLearned = [];
|
|
this.#untrimmedSkillsAvailable = [];
|
|
this.#thrallsUnlocked = [];
|
|
this.#thrallDamage = {};
|
|
this.#thrallsObtainedItem = [];
|
|
this.#thrallsDeliveredItem = [];
|
|
|
|
this.refill();
|
|
} else {
|
|
const file = mustBeSaveFileV1(args);
|
|
this.#name = file.name;
|
|
this.#thrallTemplate = file.thrallTemplateId;
|
|
this.#nImprovements = file.nImprovements;
|
|
this.#stats = {
|
|
AGI: file.stats.agi,
|
|
INT: file.stats.int,
|
|
CHA: file.stats.cha,
|
|
PSI: file.stats.psi,
|
|
};
|
|
this.#talents = {
|
|
AGI: file.talents.agi,
|
|
INT: file.talents.int,
|
|
CHA: file.talents.cha,
|
|
PSI: file.talents.psi,
|
|
};
|
|
(this.#isInPenance = file.isInPenance),
|
|
(this.#wish = file.wishId >= 0 ? { id: file.wishId } : null);
|
|
this.#exp = file.exp;
|
|
this.#blood = file.blood;
|
|
this.#itemsPurloined = file.itemsPurloined;
|
|
this.#skillsLearned = file.skillsLearned;
|
|
this.#untrimmedSkillsAvailable = file.untrimmedSkillsAvailableIds.map(
|
|
(id) => {
|
|
return { id: id };
|
|
},
|
|
);
|
|
this.#thrallsUnlocked = file.thrallsUnlocked;
|
|
this.#thrallDamage = {};
|
|
for (let i = 0; i < file.thrallDamage.length; ++i) {
|
|
this.#thrallDamage[i] = file.thrallDamage[i];
|
|
}
|
|
this.#thrallsObtainedItem = file.thrallsObtainedItem;
|
|
this.#thrallsDeliveredItem = file.thrallsDeliveredItem;
|
|
}
|
|
}
|
|
|
|
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: asSuccessor, withWish: withWish });
|
|
}
|
|
|
|
export function rehydratePlayerProgress(savefile: SaveFileV1) {
|
|
active = new PlayerProgress(savefile);
|
|
}
|
|
|
|
export function getPlayerProgress(): PlayerProgress {
|
|
if (active == null) {
|
|
throw new Error(
|
|
`trying to get player progress before it has been initialized`,
|
|
);
|
|
}
|
|
return active;
|
|
}
|