autoformat the world
This commit is contained in:
parent
2837461add
commit
3a968af5ca
@ -4,8 +4,8 @@ import { getThralls, ItemStage, LifeStage, Thrall } from "./thralls.ts";
|
|||||||
import { SaveFileV1, mustBeSaveFileV1 } from "./saveformat.ts";
|
import { SaveFileV1, mustBeSaveFileV1 } from "./saveformat.ts";
|
||||||
|
|
||||||
interface NewRoundConfig {
|
interface NewRoundConfig {
|
||||||
asSuccessor: SuccessorOption,
|
asSuccessor: SuccessorOption;
|
||||||
withWish: Wish | null,
|
withWish: Wish | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class PlayerProgress {
|
export class PlayerProgress {
|
||||||
@ -28,7 +28,7 @@ export class PlayerProgress {
|
|||||||
|
|
||||||
constructor(args: NewRoundConfig | SaveFileV1) {
|
constructor(args: NewRoundConfig | SaveFileV1) {
|
||||||
if ("asSuccesor" in args) {
|
if ("asSuccesor" in args) {
|
||||||
//asSuccessor: SuccessorOption, withWish: Wish | null) {
|
//asSuccessor: SuccessorOption, withWish: Wish | null) {
|
||||||
const config = args as NewRoundConfig;
|
const config = args as NewRoundConfig;
|
||||||
const asSuccessor = config.asSuccessor;
|
const asSuccessor = config.asSuccessor;
|
||||||
this.#name = asSuccessor.name;
|
this.#name = asSuccessor.name;
|
||||||
@ -59,23 +59,27 @@ export class PlayerProgress {
|
|||||||
INT: file.stats.int,
|
INT: file.stats.int,
|
||||||
CHA: file.stats.cha,
|
CHA: file.stats.cha,
|
||||||
PSI: file.stats.psi,
|
PSI: file.stats.psi,
|
||||||
}
|
};
|
||||||
this.#talents = {
|
this.#talents = {
|
||||||
AGI: file.talents.agi,
|
AGI: file.talents.agi,
|
||||||
INT: file.talents.int,
|
INT: file.talents.int,
|
||||||
CHA: file.talents.cha,
|
CHA: file.talents.cha,
|
||||||
PSI: file.talents.psi,
|
PSI: file.talents.psi,
|
||||||
}
|
};
|
||||||
this.#isInPenance = file.isInPenance,
|
(this.#isInPenance = file.isInPenance),
|
||||||
this.#wish = file.wishId >= 0 ? {id: file.wishId} : null;
|
(this.#wish = file.wishId >= 0 ? { id: file.wishId } : null);
|
||||||
this.#exp = file.exp;
|
this.#exp = file.exp;
|
||||||
this.#blood = file.blood;
|
this.#blood = file.blood;
|
||||||
this.#itemsPurloined = file.itemsPurloined;
|
this.#itemsPurloined = file.itemsPurloined;
|
||||||
this.#skillsLearned = file.skillsLearned;
|
this.#skillsLearned = file.skillsLearned;
|
||||||
this.#untrimmedSkillsAvailable = file.untrimmedSkillsAvailableIds.map((id) => {return {id: id}});
|
this.#untrimmedSkillsAvailable = file.untrimmedSkillsAvailableIds.map(
|
||||||
|
(id) => {
|
||||||
|
return { id: id };
|
||||||
|
},
|
||||||
|
);
|
||||||
this.#thrallsUnlocked = file.thrallsUnlocked;
|
this.#thrallsUnlocked = file.thrallsUnlocked;
|
||||||
this.#thrallDamage = {};
|
this.#thrallDamage = {};
|
||||||
for(let i = 0; i < file.thrallDamage.length; ++i) {
|
for (let i = 0; i < file.thrallDamage.length; ++i) {
|
||||||
this.#thrallDamage[i] = file.thrallDamage[i];
|
this.#thrallDamage[i] = file.thrallDamage[i];
|
||||||
}
|
}
|
||||||
this.#thrallsObtainedItem = file.thrallsObtainedItem;
|
this.#thrallsObtainedItem = file.thrallsObtainedItem;
|
||||||
@ -252,7 +256,7 @@ export class PlayerProgress {
|
|||||||
return this.#untrimmedSkillsAvailable.map((s) => s.id);
|
return this.#untrimmedSkillsAvailable.map((s) => s.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
getLearnedSkills() : Skill[] {
|
getLearnedSkills(): Skill[] {
|
||||||
let learnedSkills = [];
|
let learnedSkills = [];
|
||||||
for (let s of this.#skillsLearned.values()) {
|
for (let s of this.#skillsLearned.values()) {
|
||||||
learnedSkills.push({ id: s });
|
learnedSkills.push({ id: s });
|
||||||
@ -260,7 +264,7 @@ export class PlayerProgress {
|
|||||||
return learnedSkills;
|
return learnedSkills;
|
||||||
}
|
}
|
||||||
|
|
||||||
getRawLearnedSkills() : number[] {
|
getRawLearnedSkills(): number[] {
|
||||||
return [...this.#skillsLearned];
|
return [...this.#skillsLearned];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -283,7 +287,7 @@ export class PlayerProgress {
|
|||||||
return this.#thrallsUnlocked.indexOf(thrall.id) != -1;
|
return this.#thrallsUnlocked.indexOf(thrall.id) != -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
getUnlockedThrallIds() : number[] {
|
getUnlockedThrallIds(): number[] {
|
||||||
return [...this.#thrallsUnlocked];
|
return [...this.#thrallsUnlocked];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -328,7 +332,7 @@ export class PlayerProgress {
|
|||||||
this.#thrallsObtainedItem.push(thrall.id);
|
this.#thrallsObtainedItem.push(thrall.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
getThrallObtainedItemIds() : number[] {
|
getThrallObtainedItemIds(): number[] {
|
||||||
return [...this.#thrallsObtainedItem];
|
return [...this.#thrallsObtainedItem];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -339,7 +343,7 @@ export class PlayerProgress {
|
|||||||
this.#thrallsDeliveredItem.push(thrall.id);
|
this.#thrallsDeliveredItem.push(thrall.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
getThrallDeliveredItemIds() : number[] {
|
getThrallDeliveredItemIds(): number[] {
|
||||||
return [...this.#thrallsDeliveredItem];
|
return [...this.#thrallsDeliveredItem];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -374,7 +378,7 @@ export function initPlayerProgress(
|
|||||||
asSuccessor: SuccessorOption,
|
asSuccessor: SuccessorOption,
|
||||||
withWish: Wish | null,
|
withWish: Wish | null,
|
||||||
) {
|
) {
|
||||||
active = new PlayerProgress({asSuccessor:asSuccessor, withWish:withWish});
|
active = new PlayerProgress({ asSuccessor: asSuccessor, withWish: withWish });
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getPlayerProgress(): PlayerProgress {
|
export function getPlayerProgress(): PlayerProgress {
|
||||||
|
484
src/save.ts
484
src/save.ts
@ -1,11 +1,11 @@
|
|||||||
import { getPlayerProgress } from "./playerprogress"
|
import { getPlayerProgress } from "./playerprogress";
|
||||||
import { getStateManager } from "./statemanager";
|
import { getStateManager } from "./statemanager";
|
||||||
import { getThralls } from "./thralls";
|
import { getThralls } from "./thralls";
|
||||||
import {SaveFileV1, StatCounterV1} from "./saveformat";
|
import { SaveFileV1, StatCounterV1 } from "./saveformat";
|
||||||
|
|
||||||
export interface SaveFile {
|
export interface SaveFile {
|
||||||
version: string;
|
version: string;
|
||||||
revision: number;
|
revision: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
type SaveSlot = "FLEDGLING_SLOT_1" | "FLEDGLING_SLOT_2";
|
type SaveSlot = "FLEDGLING_SLOT_1" | "FLEDGLING_SLOT_2";
|
||||||
@ -13,279 +13,285 @@ type SaveSlot = "FLEDGLING_SLOT_1" | "FLEDGLING_SLOT_2";
|
|||||||
/// Checks whether obj is a valid save file, as far as we can tell, and returns
|
/// Checks whether obj is a valid save file, as far as we can tell, and returns
|
||||||
/// it unchanged if it is, or throws an error if it's not valid.
|
/// it unchanged if it is, or throws an error if it's not valid.
|
||||||
export function mustBeSaveFileV1(obj: unknown): SaveFileV1 {
|
export function mustBeSaveFileV1(obj: unknown): SaveFileV1 {
|
||||||
if (obj === undefined || obj === null) {
|
if (obj === undefined || obj === null) {
|
||||||
throw new Error("nonexistent");
|
throw new Error("nonexistent");
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
if (typeof obj !== "object") {
|
||||||
throw new Error(`not an object; was ${typeof(obj)}`);
|
throw new Error(`not an object; was ${typeof obj}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!("version" in obj)) {
|
if (!("version" in obj)) {
|
||||||
throw new Error("no magic number");
|
throw new Error("no magic number");
|
||||||
}
|
}
|
||||||
if (obj.version !== "fledgling_save_v1") {
|
if (obj.version !== "fledgling_save_v1") {
|
||||||
throw new Error(`bad magic number: ${obj.version}`);
|
throw new Error(`bad magic number: ${obj.version}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
version: "fledgling_save_v1",
|
||||||
|
revision: mustGetNumber(obj, "revision"),
|
||||||
|
turn: mustGetNumber(obj, "turn"),
|
||||||
|
name: mustGetString(obj, "name"),
|
||||||
|
thrallTemplateId: mustGetNumber(obj, "thrallTemplateId"),
|
||||||
|
nImprovements: mustGetNumber(obj, "nImprovements"),
|
||||||
|
stats: mustGetStatCounterV1(obj, "stats"),
|
||||||
|
talents: mustGetStatCounterV1(obj, "talents"),
|
||||||
|
isInPenance: mustGetBoolean(obj, "isInPenance"),
|
||||||
|
wishId: mustGetNumber(obj, "wishId"),
|
||||||
|
exp: mustGetNumber(obj, "exp"),
|
||||||
|
blood: mustGetNumber(obj, "blood"),
|
||||||
|
itemsPurloined: mustGetNumber(obj, "itemsPurloined"),
|
||||||
|
skillsLearned: mustGetNumberArray(obj, "skillsLearned"),
|
||||||
|
untrimmedSkillsAvailableIds: mustGetNumberArray(
|
||||||
|
obj,
|
||||||
|
"untrimmedSkillsAvailableIds",
|
||||||
|
),
|
||||||
|
thrallsUnlocked: mustGetNumberArray(obj, "thrallsUnlocked"),
|
||||||
|
thrallDamage: mustGetNumberArray(obj, "thrallDamage"),
|
||||||
|
thrallsObtainedItem: mustGetNumberArray(obj, "thrallsObtainedItem"),
|
||||||
|
thrallsDeliveredItem: mustGetNumberArray(obj, "thrallsDeliveredItem"),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetNumber(obj: object, key: string): number {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "number") {
|
||||||
|
throw new Error(`not a number: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetString(obj: object, key: string): string {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "string") {
|
||||||
|
throw new Error(`not a string: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetStatCounterV1(obj: object, key: string): StatCounterV1 {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "object") {
|
||||||
|
throw new Error(`not an object: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
return {
|
return {
|
||||||
version: "fledgling_save_v1",
|
agi: mustGetNumber(val, "agi"),
|
||||||
revision: mustGetNumber(obj, "revision"),
|
int: mustGetNumber(val, "int"),
|
||||||
turn: mustGetNumber(obj, "turn"),
|
cha: mustGetNumber(val, "cha"),
|
||||||
name: mustGetString(obj, "name"),
|
psi: mustGetNumber(val, "psi"),
|
||||||
thrallTemplateId: mustGetNumber(obj, "thrallTemplateId"),
|
|
||||||
nImprovements: mustGetNumber(obj, "nImprovements"),
|
|
||||||
stats: mustGetStatCounterV1(obj, "stats"),
|
|
||||||
talents: mustGetStatCounterV1(obj, "talents"),
|
|
||||||
isInPenance: mustGetBoolean(obj, "isInPenance"),
|
|
||||||
wishId: mustGetNumber(obj, "wishId"),
|
|
||||||
exp: mustGetNumber(obj, "exp"),
|
|
||||||
blood: mustGetNumber(obj, "blood"),
|
|
||||||
itemsPurloined: mustGetNumber(obj, "itemsPurloined"),
|
|
||||||
skillsLearned: mustGetNumberArray(obj, "skillsLearned"),
|
|
||||||
untrimmedSkillsAvailableIds: mustGetNumberArray(obj, "untrimmedSkillsAvailableIds"),
|
|
||||||
thrallsUnlocked: mustGetNumberArray(obj, "thrallsUnlocked"),
|
|
||||||
thrallDamage: mustGetNumberArray(obj, "thrallDamage"),
|
|
||||||
thrallsObtainedItem: mustGetNumberArray(obj, "thrallsObtainedItem"),
|
|
||||||
thrallsDeliveredItem: mustGetNumberArray(obj, "thrallsDeliveredItem"),
|
|
||||||
};
|
};
|
||||||
|
} catch (e) {
|
||||||
|
let message = "unrecognizable error";
|
||||||
|
if (e instanceof Error) {
|
||||||
|
message = e.message;
|
||||||
|
}
|
||||||
|
throw new Error(`reading ${key}: ${message}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function mustGetNumber(obj: object, key: string) : number {
|
function mustGetBoolean(obj: object, key: string): boolean {
|
||||||
if (obj === null || obj === undefined) {
|
if (obj === null || obj === undefined) {
|
||||||
throw new Error("container absent");
|
throw new Error("container absent");
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
if (typeof obj !== "object") {
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
}
|
}
|
||||||
if (!(key in obj)) {
|
if (!(key in obj)) {
|
||||||
throw new Error(`missing number: ${key}`);
|
throw new Error(`missing number: ${key}`);
|
||||||
}
|
}
|
||||||
const dict = obj as {[key:string]:any};
|
const dict = obj as { [key: string]: any };
|
||||||
const val = dict[key];
|
const val = dict[key];
|
||||||
if (typeof(val) !== "number") {
|
if (typeof val !== "boolean") {
|
||||||
throw new Error(`not a number: ${key}: ${val}`);
|
throw new Error(`not boolean: ${key}: ${val}`);
|
||||||
}
|
}
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
function mustGetString(obj: object, key: string) : string {
|
function mustGetNumberArray(obj: object, key: string): number[] {
|
||||||
if (obj === null || obj === undefined) {
|
if (obj === null || obj === undefined) {
|
||||||
throw new Error("container absent");
|
throw new Error("container absent");
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
if (typeof obj !== "object") {
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
}
|
}
|
||||||
if (!(key in obj)) {
|
if (!(key in obj)) {
|
||||||
throw new Error(`missing number: ${key}`);
|
throw new Error(`missing number: ${key}`);
|
||||||
}
|
}
|
||||||
const dict = obj as {[key:string]:any};
|
const dict = obj as { [key: string]: any };
|
||||||
const val = dict[key];
|
const val = dict[key];
|
||||||
if (typeof(val) !== "string") {
|
if (typeof val !== "object") {
|
||||||
throw new Error(`not a string: ${key}: ${val}`);
|
throw new Error(`not an object: ${key}: ${val}`);
|
||||||
}
|
}
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetStatCounterV1(obj: object, key: string) : StatCounterV1 {
|
for (const x of val) {
|
||||||
if (obj === null || obj === undefined) {
|
if (typeof x !== "number") {
|
||||||
throw new Error("container absent");
|
throw new Error(`contained non-number item in ${key}: ${val}`);
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
}
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
return val;
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "object") {
|
|
||||||
throw new Error(`not an object: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
return {
|
|
||||||
agi: mustGetNumber(val, "agi"),
|
|
||||||
int: mustGetNumber(val, "int"),
|
|
||||||
cha: mustGetNumber(val, "cha"),
|
|
||||||
psi: mustGetNumber(val, "psi"),
|
|
||||||
};
|
|
||||||
} catch (e) {
|
|
||||||
let message = "unrecognizable error";
|
|
||||||
if (e instanceof Error) {
|
|
||||||
message = e.message;
|
|
||||||
}
|
|
||||||
throw new Error(`reading ${key}: ${message}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetBoolean(obj: object, key: string) : boolean {
|
|
||||||
if (obj === null || obj === undefined) {
|
|
||||||
throw new Error("container absent");
|
|
||||||
}
|
|
||||||
if (typeof(obj) !== "object") {
|
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "boolean") {
|
|
||||||
throw new Error(`not boolean: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetNumberArray(obj: object, key: string) : number[] {
|
|
||||||
if (obj === null || obj === undefined) {
|
|
||||||
throw new Error("container absent");
|
|
||||||
}
|
|
||||||
if (typeof(obj) !== "object") {
|
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "object") {
|
|
||||||
throw new Error(`not an object: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const x of val) {
|
|
||||||
if (typeof(x) !== "number") {
|
|
||||||
throw new Error(`contained non-number item in ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return val;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The result of attempting to load a V1 save file.
|
/// The result of attempting to load a V1 save file.
|
||||||
interface SaveFileV1LoadResult {
|
interface SaveFileV1LoadResult {
|
||||||
// If present and valid, the loaded file.
|
// If present and valid, the loaded file.
|
||||||
file : SaveFileV1 | null;
|
file: SaveFileV1 | null;
|
||||||
|
|
||||||
/// A file loading error, if any. If `file` is present, this refers
|
/// A file loading error, if any. If `file` is present, this refers
|
||||||
/// to an error reading from the *other* slot.
|
/// to an error reading from the *other* slot.
|
||||||
error: string | null;
|
error: string | null;
|
||||||
|
|
||||||
/// The slot this file was loaded from, or that a load attempt failed from.
|
/// The slot this file was loaded from, or that a load attempt failed from.
|
||||||
/// If multiple load attempts failed and none succeeded, this refers to
|
/// If multiple load attempts failed and none succeeded, this refers to
|
||||||
/// any one attempted slot.
|
/// any one attempted slot.
|
||||||
slot: SaveSlot;
|
slot: SaveSlot;
|
||||||
}
|
}
|
||||||
|
|
||||||
function readFromSlot(slot: SaveSlot): SaveFileV1LoadResult {
|
function readFromSlot(slot: SaveSlot): SaveFileV1LoadResult {
|
||||||
var serialized = localStorage.getItem(slot);
|
var serialized = localStorage.getItem(slot);
|
||||||
if (serialized === null) {
|
if (serialized === null) {
|
||||||
return {
|
return {
|
||||||
file: null,
|
file: null,
|
||||||
error: null,
|
error: null,
|
||||||
slot: slot,
|
slot: slot,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
return {
|
return {
|
||||||
file: mustBeSaveFileV1(JSON.parse(serialized)),
|
file: mustBeSaveFileV1(JSON.parse(serialized)),
|
||||||
error: null,
|
error: null,
|
||||||
slot: slot,
|
slot: slot,
|
||||||
};
|
};
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
let message = "unidentifiable error";
|
let message = "unidentifiable error";
|
||||||
if (e instanceof Error) {
|
if (e instanceof Error) {
|
||||||
message = e.message;
|
message = e.message;
|
||||||
}
|
|
||||||
return {
|
|
||||||
file: null,
|
|
||||||
error: message,
|
|
||||||
slot: slot,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
return {
|
||||||
|
file: null,
|
||||||
|
error: message,
|
||||||
|
slot: slot,
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Reads the more recent valid save file, if either is valid. If no save files
|
/// Reads the more recent valid save file, if either is valid. If no save files
|
||||||
/// are present, `file` and `error` will both be absent. If an invalid save
|
/// are present, `file` and `error` will both be absent. If an invalid save
|
||||||
/// file is discovered, `error` will refer to the issue(s) detected.
|
/// file is discovered, `error` will refer to the issue(s) detected.
|
||||||
function readBestSave(): SaveFileV1LoadResult {
|
function readBestSave(): SaveFileV1LoadResult {
|
||||||
const from1 = readFromSlot("FLEDGLING_SLOT_1");
|
const from1 = readFromSlot("FLEDGLING_SLOT_1");
|
||||||
const from2 = readFromSlot("FLEDGLING_SLOT_2");
|
const from2 = readFromSlot("FLEDGLING_SLOT_2");
|
||||||
if (from1.file && from2.file) {
|
if (from1.file && from2.file) {
|
||||||
return (from1.file.revision > from2.file.revision) ? from1 : from2;
|
return from1.file.revision > from2.file.revision ? from1 : from2;
|
||||||
}
|
}
|
||||||
|
|
||||||
var errors : string[] = [];
|
var errors: string[] = [];
|
||||||
if (from1.error) {
|
if (from1.error) {
|
||||||
errors = ["slot 1 error: " + from1.error];
|
errors = ["slot 1 error: " + from1.error];
|
||||||
}
|
}
|
||||||
if (from2.error) {
|
if (from2.error) {
|
||||||
errors.push("slot 2 error: " + from2.error);
|
errors.push("slot 2 error: " + from2.error);
|
||||||
}
|
}
|
||||||
var msg : string | null = errors.length > 0 ? errors.join("\n") : null;
|
var msg: string | null = errors.length > 0 ? errors.join("\n") : null;
|
||||||
if (from1.file) {
|
if (from1.file) {
|
||||||
return {
|
|
||||||
file: from1.file,
|
|
||||||
error: msg,
|
|
||||||
slot: "FLEDGLING_SLOT_1",
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return {
|
return {
|
||||||
file: from2.file,
|
file: from1.file,
|
||||||
error: msg,
|
error: msg,
|
||||||
slot: "FLEDGLING_SLOT_2",
|
slot: "FLEDGLING_SLOT_1",
|
||||||
};
|
};
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
file: from2.file,
|
||||||
|
error: msg,
|
||||||
|
slot: "FLEDGLING_SLOT_2",
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function save() {
|
export function save() {
|
||||||
const targetSlot : SaveSlot = (readBestSave().slot === "FLEDGLING_SLOT_1") ? "FLEDGLING_SLOT_2" : "FLEDGLING_SLOT_1";
|
const targetSlot: SaveSlot =
|
||||||
return saveIntoSlot(targetSlot);
|
readBestSave().slot === "FLEDGLING_SLOT_1"
|
||||||
|
? "FLEDGLING_SLOT_2"
|
||||||
|
: "FLEDGLING_SLOT_1";
|
||||||
|
return saveIntoSlot(targetSlot);
|
||||||
}
|
}
|
||||||
|
|
||||||
function extractCurrentState() : SaveFileV1 {
|
function extractCurrentState(): SaveFileV1 {
|
||||||
const progress = getPlayerProgress();
|
const progress = getPlayerProgress();
|
||||||
const stateManager = getStateManager();
|
const stateManager = getStateManager();
|
||||||
var thrallDamage : number[] = [];
|
var thrallDamage: number[] = [];
|
||||||
const nThralls = getThralls().length;
|
const nThralls = getThralls().length;
|
||||||
for (let i = 0; i < nThralls; ++i) {
|
for (let i = 0; i < nThralls; ++i) {
|
||||||
thrallDamage.push(progress.getThrallDamage({id: i}));
|
thrallDamage.push(progress.getThrallDamage({ id: i }));
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
version: "fledgling_save_v1",
|
version: "fledgling_save_v1",
|
||||||
revision: stateManager.nextRevision(),
|
revision: stateManager.nextRevision(),
|
||||||
turn: stateManager.getTurn(),
|
turn: stateManager.getTurn(),
|
||||||
name: progress.name,
|
name: progress.name,
|
||||||
thrallTemplateId: progress.template.id,
|
thrallTemplateId: progress.template.id,
|
||||||
nImprovements: progress.nImprovements,
|
nImprovements: progress.nImprovements,
|
||||||
stats: {
|
stats: {
|
||||||
agi: progress.getStat("AGI"),
|
agi: progress.getStat("AGI"),
|
||||||
int: progress.getStat("INT"),
|
int: progress.getStat("INT"),
|
||||||
cha: progress.getStat("CHA"),
|
cha: progress.getStat("CHA"),
|
||||||
psi: progress.getStat("PSI"),
|
psi: progress.getStat("PSI"),
|
||||||
},
|
},
|
||||||
talents: {
|
talents: {
|
||||||
agi: progress.getStat("AGI"),
|
agi: progress.getStat("AGI"),
|
||||||
int: progress.getStat("INT"),
|
int: progress.getStat("INT"),
|
||||||
cha: progress.getStat("CHA"),
|
cha: progress.getStat("CHA"),
|
||||||
psi: progress.getStat("PSI"),
|
psi: progress.getStat("PSI"),
|
||||||
},
|
},
|
||||||
isInPenance: progress.isInPenance,
|
isInPenance: progress.isInPenance,
|
||||||
wishId: progress.getWish()?.id ?? -1,
|
wishId: progress.getWish()?.id ?? -1,
|
||||||
exp: progress.getExperience(),
|
exp: progress.getExperience(),
|
||||||
blood: progress.getBlood(),
|
blood: progress.getBlood(),
|
||||||
itemsPurloined: progress.getItemsPurloined(),
|
itemsPurloined: progress.getItemsPurloined(),
|
||||||
skillsLearned: progress.getRawLearnedSkills(),
|
skillsLearned: progress.getRawLearnedSkills(),
|
||||||
untrimmedSkillsAvailableIds: progress.getUntrimmedAvailableSkillIds(),
|
untrimmedSkillsAvailableIds: progress.getUntrimmedAvailableSkillIds(),
|
||||||
thrallsUnlocked: progress.getUnlockedThrallIds(),
|
thrallsUnlocked: progress.getUnlockedThrallIds(),
|
||||||
thrallDamage: thrallDamage,
|
thrallDamage: thrallDamage,
|
||||||
thrallsObtainedItem: progress.getThrallObtainedItemIds(),
|
thrallsObtainedItem: progress.getThrallObtainedItemIds(),
|
||||||
thrallsDeliveredItem: progress.getThrallDeliveredItemIds(),
|
thrallsDeliveredItem: progress.getThrallDeliveredItemIds(),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function saveIntoSlot(slot: SaveSlot) {
|
function saveIntoSlot(slot: SaveSlot) {
|
||||||
localStorage.setItem(slot, JSON.stringify(extractCurrentState()));
|
localStorage.setItem(slot, JSON.stringify(extractCurrentState()));
|
||||||
}
|
}
|
||||||
|
|
||||||
export function wipeSaves() {
|
export function wipeSaves() {
|
||||||
localStorage.removeItem("FLEDGLING_SLOT_1");
|
localStorage.removeItem("FLEDGLING_SLOT_1");
|
||||||
localStorage.removeItem("FLEDGLING_SLOT_2");
|
localStorage.removeItem("FLEDGLING_SLOT_2");
|
||||||
}
|
}
|
@ -1,180 +1,183 @@
|
|||||||
export interface StatCounterV1{
|
export interface StatCounterV1 {
|
||||||
agi: number;
|
agi: number;
|
||||||
int: number;
|
int: number;
|
||||||
cha: number;
|
cha: number;
|
||||||
psi: number;
|
psi: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SaveFileV1 {
|
export interface SaveFileV1 {
|
||||||
version: "fledgling_save_v1";
|
version: "fledgling_save_v1";
|
||||||
revision: number;
|
revision: number;
|
||||||
|
|
||||||
turn: number;
|
turn: number;
|
||||||
|
|
||||||
name: string;
|
name: string;
|
||||||
thrallTemplateId: number;
|
thrallTemplateId: number;
|
||||||
nImprovements: number;
|
nImprovements: number;
|
||||||
stats: StatCounterV1;
|
stats: StatCounterV1;
|
||||||
talents: StatCounterV1;
|
talents: StatCounterV1;
|
||||||
isInPenance: boolean;
|
isInPenance: boolean;
|
||||||
wishId: number; // negative: Wish is absent
|
wishId: number; // negative: Wish is absent
|
||||||
exp: number;
|
exp: number;
|
||||||
blood: number;
|
blood: number;
|
||||||
itemsPurloined: number;
|
itemsPurloined: number;
|
||||||
skillsLearned: number[];
|
skillsLearned: number[];
|
||||||
untrimmedSkillsAvailableIds: number[];
|
untrimmedSkillsAvailableIds: number[];
|
||||||
thrallsUnlocked: number[];
|
thrallsUnlocked: number[];
|
||||||
thrallDamage: number[]; // 0: thrall is absent or undamaged
|
thrallDamage: number[]; // 0: thrall is absent or undamaged
|
||||||
thrallsObtainedItem: number[];
|
thrallsObtainedItem: number[];
|
||||||
thrallsDeliveredItem: number[];
|
thrallsDeliveredItem: number[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks whether obj is a valid save file, as far as we can tell, and returns
|
/// Checks whether obj is a valid save file, as far as we can tell, and returns
|
||||||
/// it unchanged if it is, or throws an error if it's not valid.
|
/// it unchanged if it is, or throws an error if it's not valid.
|
||||||
export function mustBeSaveFileV1(obj: unknown): SaveFileV1 {
|
export function mustBeSaveFileV1(obj: unknown): SaveFileV1 {
|
||||||
if (obj === undefined || obj === null) {
|
if (obj === undefined || obj === null) {
|
||||||
throw new Error("nonexistent");
|
throw new Error("nonexistent");
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
if (typeof obj !== "object") {
|
||||||
throw new Error(`not an object; was ${typeof(obj)}`);
|
throw new Error(`not an object; was ${typeof obj}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!("version" in obj)) {
|
if (!("version" in obj)) {
|
||||||
throw new Error("no magic number");
|
throw new Error("no magic number");
|
||||||
}
|
}
|
||||||
if (obj.version !== "fledgling_save_v1") {
|
if (obj.version !== "fledgling_save_v1") {
|
||||||
throw new Error(`bad magic number: ${obj.version}`);
|
throw new Error(`bad magic number: ${obj.version}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
version: "fledgling_save_v1",
|
||||||
|
revision: mustGetNumber(obj, "revision"),
|
||||||
|
turn: mustGetNumber(obj, "turn"),
|
||||||
|
name: mustGetString(obj, "name"),
|
||||||
|
thrallTemplateId: mustGetNumber(obj, "thrallTemplateId"),
|
||||||
|
nImprovements: mustGetNumber(obj, "nImprovements"),
|
||||||
|
stats: mustGetStatCounterV1(obj, "stats"),
|
||||||
|
talents: mustGetStatCounterV1(obj, "talents"),
|
||||||
|
isInPenance: mustGetBoolean(obj, "isInPenance"),
|
||||||
|
wishId: mustGetNumber(obj, "wishId"),
|
||||||
|
exp: mustGetNumber(obj, "exp"),
|
||||||
|
blood: mustGetNumber(obj, "blood"),
|
||||||
|
itemsPurloined: mustGetNumber(obj, "itemsPurloined"),
|
||||||
|
skillsLearned: mustGetNumberArray(obj, "skillsLearned"),
|
||||||
|
untrimmedSkillsAvailableIds: mustGetNumberArray(
|
||||||
|
obj,
|
||||||
|
"untrimmedSkillsAvailableIds",
|
||||||
|
),
|
||||||
|
thrallsUnlocked: mustGetNumberArray(obj, "thrallsUnlocked"),
|
||||||
|
thrallDamage: mustGetNumberArray(obj, "thrallDamage"),
|
||||||
|
thrallsObtainedItem: mustGetNumberArray(obj, "thrallsObtainedItem"),
|
||||||
|
thrallsDeliveredItem: mustGetNumberArray(obj, "thrallsDeliveredItem"),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetNumber(obj: object, key: string): number {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "number") {
|
||||||
|
throw new Error(`not a number: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetString(obj: object, key: string): string {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "string") {
|
||||||
|
throw new Error(`not a string: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function mustGetStatCounterV1(obj: object, key: string): StatCounterV1 {
|
||||||
|
if (obj === null || obj === undefined) {
|
||||||
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "object") {
|
||||||
|
throw new Error(`not an object: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
return {
|
return {
|
||||||
version: "fledgling_save_v1",
|
agi: mustGetNumber(val, "agi"),
|
||||||
revision: mustGetNumber(obj, "revision"),
|
int: mustGetNumber(val, "int"),
|
||||||
turn: mustGetNumber(obj, "turn"),
|
cha: mustGetNumber(val, "cha"),
|
||||||
name: mustGetString(obj, "name"),
|
psi: mustGetNumber(val, "psi"),
|
||||||
thrallTemplateId: mustGetNumber(obj, "thrallTemplateId"),
|
|
||||||
nImprovements: mustGetNumber(obj, "nImprovements"),
|
|
||||||
stats: mustGetStatCounterV1(obj, "stats"),
|
|
||||||
talents: mustGetStatCounterV1(obj, "talents"),
|
|
||||||
isInPenance: mustGetBoolean(obj, "isInPenance"),
|
|
||||||
wishId: mustGetNumber(obj, "wishId"),
|
|
||||||
exp: mustGetNumber(obj, "exp"),
|
|
||||||
blood: mustGetNumber(obj, "blood"),
|
|
||||||
itemsPurloined: mustGetNumber(obj, "itemsPurloined"),
|
|
||||||
skillsLearned: mustGetNumberArray(obj, "skillsLearned"),
|
|
||||||
untrimmedSkillsAvailableIds: mustGetNumberArray(obj, "untrimmedSkillsAvailableIds"),
|
|
||||||
thrallsUnlocked: mustGetNumberArray(obj, "thrallsUnlocked"),
|
|
||||||
thrallDamage: mustGetNumberArray(obj, "thrallDamage"),
|
|
||||||
thrallsObtainedItem: mustGetNumberArray(obj, "thrallsObtainedItem"),
|
|
||||||
thrallsDeliveredItem: mustGetNumberArray(obj, "thrallsDeliveredItem"),
|
|
||||||
};
|
};
|
||||||
|
} catch (e) {
|
||||||
|
let message = "unrecognizable error";
|
||||||
|
if (e instanceof Error) {
|
||||||
|
message = e.message;
|
||||||
|
}
|
||||||
|
throw new Error(`reading ${key}: ${message}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function mustGetNumber(obj: object, key: string) : number {
|
function mustGetBoolean(obj: object, key: string): boolean {
|
||||||
if (obj === null || obj === undefined) {
|
if (obj === null || obj === undefined) {
|
||||||
throw new Error("container absent");
|
throw new Error("container absent");
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
if (typeof obj !== "object") {
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
}
|
}
|
||||||
if (!(key in obj)) {
|
if (!(key in obj)) {
|
||||||
throw new Error(`missing number: ${key}`);
|
throw new Error(`missing number: ${key}`);
|
||||||
}
|
}
|
||||||
const dict = obj as {[key:string]:any};
|
const dict = obj as { [key: string]: any };
|
||||||
const val = dict[key];
|
const val = dict[key];
|
||||||
if (typeof(val) !== "number") {
|
if (typeof val !== "boolean") {
|
||||||
throw new Error(`not a number: ${key}: ${val}`);
|
throw new Error(`not boolean: ${key}: ${val}`);
|
||||||
}
|
}
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
function mustGetString(obj: object, key: string) : string {
|
function mustGetNumberArray(obj: object, key: string): number[] {
|
||||||
if (obj === null || obj === undefined) {
|
if (obj === null || obj === undefined) {
|
||||||
throw new Error("container absent");
|
throw new Error("container absent");
|
||||||
|
}
|
||||||
|
if (typeof obj !== "object") {
|
||||||
|
throw new Error(`container was not an object; was ${typeof obj}`);
|
||||||
|
}
|
||||||
|
if (!(key in obj)) {
|
||||||
|
throw new Error(`missing number: ${key}`);
|
||||||
|
}
|
||||||
|
const dict = obj as { [key: string]: any };
|
||||||
|
const val = dict[key];
|
||||||
|
if (typeof val !== "object") {
|
||||||
|
throw new Error(`not an object: ${key}: ${val}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const x of val) {
|
||||||
|
if (typeof x !== "number") {
|
||||||
|
throw new Error(`contained non-number item in ${key}: ${val}`);
|
||||||
}
|
}
|
||||||
if (typeof(obj) !== "object") {
|
}
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
return val;
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "string") {
|
|
||||||
throw new Error(`not a string: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetStatCounterV1(obj: object, key: string) : StatCounterV1 {
|
|
||||||
if (obj === null || obj === undefined) {
|
|
||||||
throw new Error("container absent");
|
|
||||||
}
|
|
||||||
if (typeof(obj) !== "object") {
|
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "object") {
|
|
||||||
throw new Error(`not an object: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
return {
|
|
||||||
agi: mustGetNumber(val, "agi"),
|
|
||||||
int: mustGetNumber(val, "int"),
|
|
||||||
cha: mustGetNumber(val, "cha"),
|
|
||||||
psi: mustGetNumber(val, "psi"),
|
|
||||||
};
|
|
||||||
} catch (e) {
|
|
||||||
let message = "unrecognizable error";
|
|
||||||
if (e instanceof Error) {
|
|
||||||
message = e.message;
|
|
||||||
}
|
|
||||||
throw new Error(`reading ${key}: ${message}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetBoolean(obj: object, key: string) : boolean {
|
|
||||||
if (obj === null || obj === undefined) {
|
|
||||||
throw new Error("container absent");
|
|
||||||
}
|
|
||||||
if (typeof(obj) !== "object") {
|
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "boolean") {
|
|
||||||
throw new Error(`not boolean: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
return val;
|
|
||||||
}
|
|
||||||
|
|
||||||
function mustGetNumberArray(obj: object, key: string) : number[] {
|
|
||||||
if (obj === null || obj === undefined) {
|
|
||||||
throw new Error("container absent");
|
|
||||||
}
|
|
||||||
if (typeof(obj) !== "object") {
|
|
||||||
throw new Error(`container was not an object; was ${typeof(obj)}`);
|
|
||||||
}
|
|
||||||
if (!(key in obj)) {
|
|
||||||
throw new Error(`missing number: ${key}`);
|
|
||||||
}
|
|
||||||
const dict = obj as {[key:string]:any};
|
|
||||||
const val = dict[key];
|
|
||||||
if (typeof(val) !== "object") {
|
|
||||||
throw new Error(`not an object: ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const x of val) {
|
|
||||||
if (typeof(x) !== "number") {
|
|
||||||
throw new Error(`contained non-number item in ${key}: ${val}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return val;
|
|
||||||
}
|
}
|
@ -18,7 +18,7 @@ export class StateManager {
|
|||||||
#turn: number;
|
#turn: number;
|
||||||
#revision: number;
|
#revision: number;
|
||||||
|
|
||||||
constructor(file?:SaveFileV1) {
|
constructor(file?: SaveFileV1) {
|
||||||
this.#turn = file?.turn ?? 1;
|
this.#turn = file?.turn ?? 1;
|
||||||
this.#revision = file?.revision ?? 1;
|
this.#revision = file?.revision ?? 1;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user