Penance cycle
This commit is contained in:
parent
8c917df618
commit
2361b880eb
@ -14,6 +14,7 @@ export type SkillGoverning = {
|
|||||||
note: string,
|
note: string,
|
||||||
scoring: SkillScoring,
|
scoring: SkillScoring,
|
||||||
mortalServantValue: number,
|
mortalServantValue: number,
|
||||||
|
flipped: boolean,
|
||||||
};
|
};
|
||||||
export type SkillProfile = {
|
export type SkillProfile = {
|
||||||
name: string,
|
name: string,
|
||||||
@ -21,6 +22,7 @@ export type SkillProfile = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type SkillData = {
|
export type SkillData = {
|
||||||
|
isDegrading?: boolean;
|
||||||
governing: SkillGoverning,
|
governing: SkillGoverning,
|
||||||
profile: SkillProfile,
|
profile: SkillProfile,
|
||||||
prereqs: Skill[]
|
prereqs: Skill[]
|
||||||
@ -35,11 +37,25 @@ export type Skill = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type WishData = {
|
export type WishData = {
|
||||||
profile: {name: string},
|
profile: {
|
||||||
|
name: string,
|
||||||
|
note: string,
|
||||||
|
domicile: string,
|
||||||
|
reignSentence: string;
|
||||||
|
failureName: string,
|
||||||
|
failureDomicile: string,
|
||||||
|
failureReignSentence: string,
|
||||||
|
failureSuccessorVerb: string;
|
||||||
|
},
|
||||||
|
isRandomlyAvailable: boolean,
|
||||||
|
isCompulsory: boolean;
|
||||||
bannedSkills: () => Skill[],
|
bannedSkills: () => Skill[],
|
||||||
discouragedSkills: () => Skill[],
|
discouragedSkills: () => Skill[],
|
||||||
encouragedSkills: () => Skill[],
|
encouragedSkills: () => Skill[],
|
||||||
requiredSkills: () => Skill[]
|
requiredSkills: () => Skill[]
|
||||||
|
prologue: VNScene,
|
||||||
|
onVictory: VNScene,
|
||||||
|
onFailure: VNScene,
|
||||||
}
|
}
|
||||||
export type Wish = {
|
export type Wish = {
|
||||||
id: number
|
id: number
|
||||||
@ -61,6 +77,9 @@ export type Ending = {
|
|||||||
export type EndingPersonal = {
|
export type EndingPersonal = {
|
||||||
rank: string,
|
rank: string,
|
||||||
domicile: string,
|
domicile: string,
|
||||||
|
reignSentence: string,
|
||||||
|
successorVerb: string,
|
||||||
|
progenerateVerb: string,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type EndingAnalytics = {
|
export type EndingAnalytics = {
|
||||||
@ -74,6 +93,9 @@ export type SuccessorOption = {
|
|||||||
title: string,
|
title: string,
|
||||||
note: string | null, // ex "already a vampire"
|
note: string | null, // ex "already a vampire"
|
||||||
stats: Record<Stat, number>,
|
stats: Record<Stat, number>,
|
||||||
talents: Record<Stat, number>
|
talents: Record<Stat, number>,
|
||||||
|
skills: Skill[],
|
||||||
|
inPenance: boolean;
|
||||||
|
isCompulsory: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,19 +50,23 @@ export class EndgameModal {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get #canProgenerate(): boolean {
|
get #canProgenerate(): boolean {
|
||||||
return this.#selectedSuccessor != null && this.#selectedWish != null;
|
return this.#selectedSuccessor != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
#progenerate() {
|
#progenerate() {
|
||||||
let successor =
|
let successor =
|
||||||
this.#ending!.successorOptions[this.#selectedSuccessor!];
|
this.#ending!.successorOptions[this.#selectedSuccessor!];
|
||||||
let wish =
|
let wish =
|
||||||
this.#ending!.wishOptions[this.#selectedWish!];
|
this.#selectedWish != null
|
||||||
|
? this.#ending!.wishOptions[this.#selectedWish!]
|
||||||
|
: null;
|
||||||
this.#ending = null;
|
this.#ending = null;
|
||||||
getStateManager().startGame(successor, wish);
|
getStateManager().startGame(successor, wish);
|
||||||
}
|
}
|
||||||
|
|
||||||
#update() {
|
#update() {
|
||||||
|
this.#fixCompulsory();
|
||||||
|
|
||||||
this.#drawpile.clear();
|
this.#drawpile.clear();
|
||||||
if (this.#page == 0) {
|
if (this.#page == 0) {
|
||||||
let analytics = this.#ending?.analytics;
|
let analytics = this.#ending?.analytics;
|
||||||
@ -81,7 +85,7 @@ export class EndgameModal {
|
|||||||
let whereLabel =
|
let whereLabel =
|
||||||
mortalServants >= 25 ? "where you live with many friends." :
|
mortalServants >= 25 ? "where you live with many friends." :
|
||||||
mortalServants >= 1 ? "where you live with a couple of friends." :
|
mortalServants >= 1 ? "where you live with a couple of friends." :
|
||||||
"where you live completely alone.";
|
"where you live without friends.";
|
||||||
D.drawText(whereLabel, new Point(0, 160), FG_TEXT)
|
D.drawText(whereLabel, new Point(0, 160), FG_TEXT)
|
||||||
D.drawText("You have achieved:", new Point(0, 192), FG_TEXT)
|
D.drawText("You have achieved:", new Point(0, 192), FG_TEXT)
|
||||||
let itemsPurloinedText = itemsPurloined == 1 ? "item purloined" : "items purloined";
|
let itemsPurloinedText = itemsPurloined == 1 ? "item purloined" : "items purloined";
|
||||||
@ -107,11 +111,12 @@ export class EndgameModal {
|
|||||||
msg = "That feels like a lot!"
|
msg = "That feels like a lot!"
|
||||||
}
|
}
|
||||||
D.drawText(msg, new Point(0, 288), FG_TEXT)
|
D.drawText(msg, new Point(0, 288), FG_TEXT)
|
||||||
D.drawText("Your reign continues unimpeded from the shadows. It is now time to", new Point(0, 320), FG_TEXT, {forceWidth: WIDTH})
|
let reignSentence = this.#ending?.personal?.reignSentence ?? "Your reign is in an unknown state.";
|
||||||
|
D.drawText(`${reignSentence} It is now time to`, new Point(0, 320), FG_TEXT, {forceWidth: WIDTH})
|
||||||
})
|
})
|
||||||
addButton(
|
addButton(
|
||||||
this.#drawpile,
|
this.#drawpile,
|
||||||
"Appoint a Successor",
|
this.#ending?.personal?.successorVerb ?? "Do Unknown Things",
|
||||||
new Rect(
|
new Rect(
|
||||||
new Point(0, HEIGHT - 32), new Size(WIDTH, 32)
|
new Point(0, HEIGHT - 32), new Size(WIDTH, 32)
|
||||||
),
|
),
|
||||||
@ -126,17 +131,21 @@ export class EndgameModal {
|
|||||||
D.drawText("Choose your successor:", new Point(0, 0), FG_TEXT);
|
D.drawText("Choose your successor:", new Point(0, 0), FG_TEXT);
|
||||||
})
|
})
|
||||||
|
|
||||||
this.#addCandidate(0, new Point(0, 32))
|
this.#addCandidate(0, new Point(0, 16))
|
||||||
this.#addCandidate(1, new Point(0, 96))
|
this.#addCandidate(1, new Point(0, 80))
|
||||||
this.#addCandidate(2, new Point(0, 160))
|
this.#addCandidate(2, new Point(0, 144))
|
||||||
|
|
||||||
|
let optionalNote = " (optional, punishes failure)";
|
||||||
|
if (this.#hasCompulsoryWish) {
|
||||||
|
optionalNote = "";
|
||||||
|
}
|
||||||
this.#drawpile.add(0, () => {
|
this.#drawpile.add(0, () => {
|
||||||
D.drawText("Plan their destiny:", new Point(0, 240), FG_TEXT);
|
D.drawText(`Plan their destiny:${optionalNote}`, new Point(0, 224), FG_TEXT);
|
||||||
})
|
})
|
||||||
|
|
||||||
this.#addWish(0, new Point(0, 272))
|
this.#addWish(1, new Point(0, 240))
|
||||||
this.#addWish(1, new Point(128, 272))
|
this.#addWish(0, new Point(128, 240))
|
||||||
this.#addWish(2, new Point(256, 272))
|
this.#addWish(2, new Point(256, 240))
|
||||||
|
|
||||||
addButton(
|
addButton(
|
||||||
this.#drawpile,
|
this.#drawpile,
|
||||||
@ -151,7 +160,7 @@ export class EndgameModal {
|
|||||||
)
|
)
|
||||||
addButton(
|
addButton(
|
||||||
this.#drawpile,
|
this.#drawpile,
|
||||||
"Progenerate",
|
this.#ending?.personal.progenerateVerb ?? "Unknown Action",
|
||||||
new Rect(
|
new Rect(
|
||||||
new Point(WIDTH/3, HEIGHT - 32), new Size(WIDTH - WIDTH / 3, 32)
|
new Point(WIDTH/3, HEIGHT - 32), new Size(WIDTH - WIDTH / 3, 32)
|
||||||
),
|
),
|
||||||
@ -163,6 +172,61 @@ export class EndgameModal {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this.#drawpile.executeOnClick();
|
this.#drawpile.executeOnClick();
|
||||||
|
this.#fixCompulsory();
|
||||||
|
}
|
||||||
|
|
||||||
|
#fixCompulsory() {
|
||||||
|
// allow player to select freely between compulsory options
|
||||||
|
{
|
||||||
|
let candidates = this.#ending?.successorOptions ?? [];
|
||||||
|
let selectedSuccessor = this.#selectedSuccessor;
|
||||||
|
let compulsorySelected = false;
|
||||||
|
if (selectedSuccessor) {
|
||||||
|
compulsorySelected = candidates[selectedSuccessor]?.isCompulsory;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!compulsorySelected) {
|
||||||
|
for (let c = 0; c < candidates.length; c++) {
|
||||||
|
if (candidates[c]?.isCompulsory) {
|
||||||
|
this.#selectedSuccessor = c;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
let wishes = this.#ending?.wishOptions ?? [];
|
||||||
|
let selectedWish = this.#selectedWish;
|
||||||
|
let compulsorySelected = false;
|
||||||
|
if (selectedWish) {
|
||||||
|
let wish = wishes[selectedWish];
|
||||||
|
if (wish) {
|
||||||
|
let data = getWishes().get(wish);
|
||||||
|
compulsorySelected = data.isCompulsory;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!compulsorySelected) {
|
||||||
|
for (let w = 0; w < wishes.length; w++) {
|
||||||
|
if (getWishes().get(wishes[w]).isCompulsory) {
|
||||||
|
this.#selectedWish = w;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get #hasCompulsoryWish(): boolean {
|
||||||
|
let wishes = this.#ending?.wishOptions ?? [];
|
||||||
|
|
||||||
|
for (let w = 0; w < wishes.length; w++) {
|
||||||
|
if (getWishes().get(wishes[w]).isCompulsory) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
#addCandidate(ix: number, at: Point) {
|
#addCandidate(ix: number, at: Point) {
|
||||||
@ -214,6 +278,9 @@ export class EndgameModal {
|
|||||||
if (talentValue > 0) {
|
if (talentValue > 0) {
|
||||||
D.drawText(`(+${talentValue})`, at.offset(xys[i].offset(new Point(56, 0))), fg)
|
D.drawText(`(+${talentValue})`, at.offset(xys[i].offset(new Point(56, 0))), fg)
|
||||||
}
|
}
|
||||||
|
if (talentValue < 0) {
|
||||||
|
D.drawText(`(${talentValue})`, at.offset(xys[i].offset(new Point(56, 0))), fg)
|
||||||
|
}
|
||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -225,7 +292,11 @@ export class EndgameModal {
|
|||||||
enabled,
|
enabled,
|
||||||
|
|
||||||
() => {
|
() => {
|
||||||
|
if (this.#selectedSuccessor == ix) {
|
||||||
|
this.#selectedSuccessor = null
|
||||||
|
} else {
|
||||||
this.#selectedSuccessor = ix;
|
this.#selectedSuccessor = ix;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -235,9 +306,12 @@ export class EndgameModal {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let wishOption = wishOptions[ix];
|
let wishOption = wishOptions[ix];
|
||||||
|
if (wishOption == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
let selected = this.#selectedWish == ix;
|
let selected = this.#selectedWish == ix;
|
||||||
let w = 128;
|
let w = 128;
|
||||||
let h = 72;
|
let h = 88;
|
||||||
let generalRect = new Rect(at, new Size(w, h));
|
let generalRect = new Rect(at, new Size(w, h));
|
||||||
let enabled = true;
|
let enabled = true;
|
||||||
|
|
||||||
@ -262,13 +336,20 @@ export class EndgameModal {
|
|||||||
alignX: AlignX.Center,
|
alignX: AlignX.Center,
|
||||||
alignY: AlignY.Middle,
|
alignY: AlignY.Middle,
|
||||||
});
|
});
|
||||||
|
D.drawText(wishData.profile.note, at.offset(new Point(w / 2, h)), FG_TEXT, {
|
||||||
|
alignX: AlignX.Center
|
||||||
|
});
|
||||||
},
|
},
|
||||||
generalRect,
|
generalRect,
|
||||||
enabled,
|
enabled,
|
||||||
|
|
||||||
() => {
|
() => {
|
||||||
|
if (this.#selectedWish == ix) {
|
||||||
|
this.#selectedWish = null;
|
||||||
|
} else {
|
||||||
this.#selectedWish = ix;
|
this.#selectedWish = ix;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,9 +25,12 @@ export class Hud {
|
|||||||
D.drawText(`${s}`, new Point(0, y), FG_BOLD)
|
D.drawText(`${s}`, new Point(0, y), FG_BOLD)
|
||||||
D.drawText(`${prog.getStat(s)}`, new Point(32, y), FG_TEXT)
|
D.drawText(`${prog.getStat(s)}`, new Point(32, y), FG_TEXT)
|
||||||
let talent = prog.getTalent(s);
|
let talent = prog.getTalent(s);
|
||||||
if (talent) {
|
if (talent > 0) {
|
||||||
D.drawText(`(+${talent})`, new Point(56, y), FG_TEXT)
|
D.drawText(`(+${talent})`, new Point(56, y), FG_TEXT)
|
||||||
}
|
}
|
||||||
|
if (talent < 0) {
|
||||||
|
D.drawText(`(${talent})`, new Point(56, y), FG_TEXT)
|
||||||
|
}
|
||||||
y += 16;
|
y += 16;
|
||||||
}
|
}
|
||||||
D.drawText("EXP", new Point(0, 144), FG_BOLD);
|
D.drawText("EXP", new Point(0, 144), FG_BOLD);
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import {hostGame} from "./engine/internal/host.ts";
|
import {hostGame} from "./engine/internal/host.ts";
|
||||||
import {game} from "./game.ts";
|
import {game} from "./game.ts";
|
||||||
import {getStateManager} from "./statemanager.ts";
|
import {getStateManager} from "./statemanager.ts";
|
||||||
import {batFreak} from "./wishes.ts";
|
|
||||||
|
|
||||||
getStateManager().startGame({
|
getStateManager().startGame({
|
||||||
name: "Pyrex",
|
name: "Pyrex",
|
||||||
@ -9,5 +8,8 @@ getStateManager().startGame({
|
|||||||
note: null,
|
note: null,
|
||||||
stats: {AGI: 10, INT: 10, CHA: 10, PSI: 10},
|
stats: {AGI: 10, INT: 10, CHA: 10, PSI: 10},
|
||||||
talents: {AGI: 0, INT: 0, CHA: 0, PSI: 0},
|
talents: {AGI: 0, INT: 0, CHA: 0, PSI: 0},
|
||||||
}, batFreak);
|
skills: [],
|
||||||
|
isCompulsory: false,
|
||||||
|
inPenance: false,
|
||||||
|
}, null);
|
||||||
hostGame(game);
|
hostGame(game);
|
@ -5,6 +5,7 @@ export class PlayerProgress {
|
|||||||
#name: string
|
#name: string
|
||||||
#stats: Record<Stat, number>
|
#stats: Record<Stat, number>
|
||||||
#talents: Record<Stat, number>
|
#talents: Record<Stat, number>
|
||||||
|
#isInPenance: boolean;
|
||||||
#wish: Wish | null;
|
#wish: Wish | null;
|
||||||
#exp: number;
|
#exp: number;
|
||||||
#blood: number
|
#blood: number
|
||||||
@ -16,6 +17,7 @@ export class PlayerProgress {
|
|||||||
this.#name = asSuccessor.name;
|
this.#name = asSuccessor.name;
|
||||||
this.#stats = {...asSuccessor.stats};
|
this.#stats = {...asSuccessor.stats};
|
||||||
this.#talents = {...asSuccessor.talents};
|
this.#talents = {...asSuccessor.talents};
|
||||||
|
this.#isInPenance = asSuccessor.inPenance;
|
||||||
this.#wish = withWish;
|
this.#wish = withWish;
|
||||||
this.#exp = 0;
|
this.#exp = 0;
|
||||||
this.#blood = 0;
|
this.#blood = 0;
|
||||||
@ -28,7 +30,7 @@ export class PlayerProgress {
|
|||||||
|
|
||||||
applyEndOfTurn() {
|
applyEndOfTurn() {
|
||||||
for (let stat of ALL_STATS.values()) {
|
for (let stat of ALL_STATS.values()) {
|
||||||
this.#stats[stat] += this.#talents[stat];
|
this.add(stat, this.#talents[stat]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -36,11 +38,15 @@ export class PlayerProgress {
|
|||||||
return this.#name;
|
return this.#name;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get isInPenance(): boolean {
|
||||||
|
return this.#isInPenance;
|
||||||
|
}
|
||||||
|
|
||||||
refill() {
|
refill() {
|
||||||
this.#blood = 2000;
|
this.#blood = 2000;
|
||||||
|
|
||||||
let learnableSkills = []; // TODO: Also include costing info
|
let learnableSkills = []; // TODO: Also include costing info
|
||||||
for (let skill of getSkills().getAllAvailableSkills().values()) {
|
for (let skill of getSkills().getAvailableSkills(this.#isInPenance).values()) {
|
||||||
if (this.#canBeAvailable(skill)) {
|
if (this.#canBeAvailable(skill)) {
|
||||||
learnableSkills.push(skill);
|
learnableSkills.push(skill);
|
||||||
}
|
}
|
||||||
@ -99,10 +105,8 @@ export class PlayerProgress {
|
|||||||
if (amount != Math.floor(amount)) {
|
if (amount != Math.floor(amount)) {
|
||||||
throw `stat increment must be integer: ${amount}`
|
throw `stat increment must be integer: ${amount}`
|
||||||
}
|
}
|
||||||
if (amount <= 0) {
|
|
||||||
throw `stat increment must be >0: ${amount}`
|
|
||||||
}
|
|
||||||
this.#stats[stat] += amount;
|
this.#stats[stat] += amount;
|
||||||
|
this.#stats[stat] = Math.min(Math.max(this.#stats[stat], -99), 999);
|
||||||
}
|
}
|
||||||
|
|
||||||
addExperience(amt: number) {
|
addExperience(amt: number) {
|
||||||
@ -169,7 +173,9 @@ export class PlayerProgress {
|
|||||||
}
|
}
|
||||||
return learnedSkills;
|
return learnedSkills;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
getStats() { return {...this.#stats} }
|
||||||
|
getTalents() { return {...this.#talents} } }
|
||||||
|
|
||||||
let active: PlayerProgress | null = null;
|
let active: PlayerProgress | null = null;
|
||||||
|
|
||||||
|
@ -3,7 +3,7 @@ import {getPlayerProgress} from "./playerprogress.ts";
|
|||||||
import {getSkills} from "./skills.ts";
|
import {getSkills} from "./skills.ts";
|
||||||
import {Ending, SCORING_CATEGORIES, ScoringCategory} from "./datatypes.ts";
|
import {Ending, SCORING_CATEGORIES, ScoringCategory} from "./datatypes.ts";
|
||||||
import {sceneBat, sceneCharm, sceneLore, sceneParty, sceneStare, sceneStealth} from "./endings.ts";
|
import {sceneBat, sceneCharm, sceneLore, sceneParty, sceneStare, sceneStealth} from "./endings.ts";
|
||||||
import {generateWishes} from "./wishes.ts";
|
import {generateWishes, getWishes, isWishCompleted} from "./wishes.ts";
|
||||||
import {generateSuccessors} from "./successors.ts";
|
import {generateSuccessors} from "./successors.ts";
|
||||||
|
|
||||||
class Scorer {
|
class Scorer {
|
||||||
@ -26,7 +26,7 @@ class Scorer {
|
|||||||
vampiricSkills += 1;
|
vampiricSkills += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
mortalServants = Math.floor(mortalServants);
|
mortalServants = Math.max(Math.floor(mortalServants), 0);
|
||||||
|
|
||||||
// NOTE: This approach isn't efficient but it's easy to understand
|
// NOTE: This approach isn't efficient but it's easy to understand
|
||||||
// and it allows me to arbitrate ties however I want
|
// and it allows me to arbitrate ties however I want
|
||||||
@ -49,40 +49,68 @@ class Scorer {
|
|||||||
let scene: VNScene;
|
let scene: VNScene;
|
||||||
let rank: string;
|
let rank: string;
|
||||||
let domicile: string;
|
let domicile: string;
|
||||||
|
let reignSentence: string;
|
||||||
|
let penance: boolean = false;
|
||||||
|
let successorVerb: string = "Appoint a Successor";
|
||||||
|
|
||||||
|
// Let the player
|
||||||
|
let wish = getPlayerProgress().getWish();
|
||||||
|
if (wish != null) {
|
||||||
|
let data = getWishes().get(wish);
|
||||||
|
if (isWishCompleted(wish)) {
|
||||||
|
scene = data.onVictory
|
||||||
|
rank = data.profile.name;
|
||||||
|
domicile = data.profile.domicile;
|
||||||
|
reignSentence = data.profile.reignSentence;
|
||||||
|
} else {
|
||||||
|
scene = data.onFailure;
|
||||||
|
rank = data.profile.failureName;
|
||||||
|
domicile = data.profile.failureDomicile;
|
||||||
|
reignSentence = data.profile.failureReignSentence;
|
||||||
|
penance = true;
|
||||||
|
successorVerb = data.profile.failureSuccessorVerb;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
// TODO: Award different ranks depending on second-to-top skill
|
// TODO: Award different ranks depending on second-to-top skill
|
||||||
// TODO: Award different domiciles based on overall score
|
// TODO: Award different domiciles based on overall score
|
||||||
// TODO: Force the rank to match the wish if one existed
|
// TODO: Force the rank to match the wish if one existed
|
||||||
if (isMax("stare", 3)) {
|
else if (isMax("stare", 3)) {
|
||||||
scene = sceneStare;
|
scene = sceneStare;
|
||||||
rank = "Hypno-Chiropteran";
|
rank = "Hypno-Chiropteran";
|
||||||
domicile = "Village of Brainwashed Mortals";
|
domicile = "Village of Brainwashed Mortals";
|
||||||
|
reignSentence = "You rule with a fair but unflinching gaze.";
|
||||||
}
|
}
|
||||||
else if (isMax("lore", 3)) {
|
else if (isMax("lore", 3)) {
|
||||||
scene = sceneLore;
|
scene = sceneLore;
|
||||||
rank = "Loremaster";
|
rank = "Loremaster";
|
||||||
domicile = "Vineyard";
|
domicile = "Vineyard";
|
||||||
|
reignSentence = "You're well on the path to ultimate knowledge.";
|
||||||
}
|
}
|
||||||
else if (isMax("charm", 2)) {
|
else if (isMax("charm", 2)) {
|
||||||
scene = sceneCharm;
|
scene = sceneCharm;
|
||||||
rank = "Seducer";
|
rank = "Seducer";
|
||||||
domicile = "Guest House";
|
domicile = "Guest House";
|
||||||
|
reignSentence = "You get to sink your fangs into anyone you want.";
|
||||||
}
|
}
|
||||||
else if (isMax("party", 1)) {
|
else if (isMax("party", 1)) {
|
||||||
scene = sceneParty;
|
scene = sceneParty;
|
||||||
rank = "Party Animal";
|
rank = "Party Animal";
|
||||||
domicile = "Nightclub";
|
domicile = "Nightclub";
|
||||||
|
reignSentence = "Everyone thinks you're too cool to disobey.";
|
||||||
}
|
}
|
||||||
else if (isMax("stealth", 0)) {
|
else if (isMax("stealth", 0)) {
|
||||||
scene = sceneStealth;
|
scene = sceneStealth;
|
||||||
rank = "Invisible";
|
rank = "Invisible";
|
||||||
domicile = "Townhouse";
|
domicile = "Townhouse";
|
||||||
|
reignSentence = "People don't see you but they do most of what you want.";
|
||||||
}
|
}
|
||||||
// if (isMax("bat")) {
|
// if (isMax("bat")) {
|
||||||
else {
|
else {
|
||||||
scene = sceneBat;
|
scene = sceneBat;
|
||||||
rank = "Bat";
|
rank = "Bat";
|
||||||
domicile = "Cave";
|
domicile = "Cave";
|
||||||
|
reignSentence = "Your skreeking verdicts are irresistible to your subjects.";
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Analytics tracker
|
// TODO: Analytics tracker
|
||||||
@ -91,12 +119,14 @@ class Scorer {
|
|||||||
vampiricSkills,
|
vampiricSkills,
|
||||||
mortalServants,
|
mortalServants,
|
||||||
}
|
}
|
||||||
let successorOptions = generateSuccessors(0); // TODO: generate nImprovements from mortalServants and the player's bsae improvements
|
let successorOptions = generateSuccessors(0, penance); // TODO: generate nImprovements from mortalServants and the player's bsae improvements
|
||||||
let wishOptions = generateWishes();
|
let wishOptions = generateWishes(penance);
|
||||||
|
|
||||||
|
let progenerateVerb = penance ? "Repent" : "Progenerate";
|
||||||
|
|
||||||
return {
|
return {
|
||||||
scene,
|
scene,
|
||||||
personal: {rank, domicile},
|
personal: {rank, domicile, reignSentence, successorVerb, progenerateVerb},
|
||||||
analytics,
|
analytics,
|
||||||
successorOptions,
|
successorOptions,
|
||||||
wishOptions,
|
wishOptions,
|
||||||
|
@ -19,9 +19,11 @@ class SkillsTable {
|
|||||||
return this.#skills[skill.id]
|
return this.#skills[skill.id]
|
||||||
}
|
}
|
||||||
|
|
||||||
getAllAvailableSkills(): Skill[] {
|
getAvailableSkills(includeDegrading: boolean): Skill[] {
|
||||||
let skills = [];
|
let skills = [];
|
||||||
for (let i = 0; i < this.#skills.length; i++) {
|
for (let i = 0; i < this.#skills.length; i++) {
|
||||||
|
let isDegrading = this.#skills[i].isDegrading ?? false;
|
||||||
|
if (isDegrading && !includeDegrading) { continue; }
|
||||||
skills.push({id: i});
|
skills.push({id: i});
|
||||||
}
|
}
|
||||||
return skills;
|
return skills;
|
||||||
@ -35,6 +37,10 @@ class SkillsTable {
|
|||||||
governingStatValue += getPlayerProgress().getStat(stat) / data.governing.stats.length;
|
governingStatValue += getPlayerProgress().getStat(stat) / data.governing.stats.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (data.governing.flipped) {
|
||||||
|
governingStatValue = - governingStatValue + 10;
|
||||||
|
}
|
||||||
|
|
||||||
let mult = getCostMultiplier(getPlayerProgress().getWish(), skill);
|
let mult = getCostMultiplier(getPlayerProgress().getWish(), skill);
|
||||||
let [underTarget, target] = [data.governing.underTarget, data.governing.target];
|
let [underTarget, target] = [data.governing.underTarget, data.governing.target];
|
||||||
underTarget = mult * underTarget;
|
underTarget = mult * underTarget;
|
||||||
@ -62,14 +68,14 @@ function geomInterpolate(
|
|||||||
return lowOut * Math.pow(highOut / lowOut, proportion)
|
return lowOut * Math.pow(highOut / lowOut, proportion)
|
||||||
}
|
}
|
||||||
|
|
||||||
type Difficulty = 0 | 1 | 2 | 3
|
type Difficulty = 0 | 1 | 1.25 | 2 | 3
|
||||||
type GoverningTemplate = {
|
type GoverningTemplate = {
|
||||||
stats: Stat[],
|
stats: Stat[],
|
||||||
note: string
|
note: string
|
||||||
scoring: SkillScoring,
|
scoring: SkillScoring,
|
||||||
}
|
}
|
||||||
|
|
||||||
type Track = "bat" | "stealth" | "charm" | "stare" | "party" | "lore"
|
type Track = "bat" | "stealth" | "charm" | "stare" | "party" | "lore" | "penance"
|
||||||
let templates: Record<Track, GoverningTemplate> = {
|
let templates: Record<Track, GoverningTemplate> = {
|
||||||
bat: {
|
bat: {
|
||||||
stats: ["AGI", "AGI", "PSI"],
|
stats: ["AGI", "AGI", "PSI"],
|
||||||
@ -101,9 +107,14 @@ let templates: Record<Track, GoverningTemplate> = {
|
|||||||
note: "Cheaper with INT and CHA.",
|
note: "Cheaper with INT and CHA.",
|
||||||
scoring: {lore: 1},
|
scoring: {lore: 1},
|
||||||
},
|
},
|
||||||
|
penance: {
|
||||||
|
stats: ["AGI", "INT", "CHA", "PSI"],
|
||||||
|
note: "Lower your stats for this.",
|
||||||
|
scoring: {},
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
function governing(track: Track, difficulty: Difficulty, flipped?: boolean): SkillGoverning {
|
||||||
let template = templates[track];
|
let template = templates[track];
|
||||||
let underTarget: number
|
let underTarget: number
|
||||||
let target: number
|
let target: number
|
||||||
@ -112,9 +123,15 @@ function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
|||||||
switch(difficulty) {
|
switch(difficulty) {
|
||||||
case 0: underTarget = 5; target = 15; cost = 50; mortalServantValue = 1; break;
|
case 0: underTarget = 5; target = 15; cost = 50; mortalServantValue = 1; break;
|
||||||
case 1: underTarget = 15; target = 40; cost = 100; mortalServantValue = 2; break;
|
case 1: underTarget = 15; target = 40; cost = 100; mortalServantValue = 2; break;
|
||||||
|
case 1.25: underTarget = 17; target = 42; cost = 100; mortalServantValue = 2; break;
|
||||||
case 2: underTarget = 30; target = 70; cost = 125; mortalServantValue = 3; break;
|
case 2: underTarget = 30; target = 70; cost = 125; mortalServantValue = 3; break;
|
||||||
case 3: underTarget = 50; target = 100; cost = 150; mortalServantValue = 10; break;
|
case 3: underTarget = 50; target = 100; cost = 150; mortalServantValue = 10; break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (flipped) {
|
||||||
|
mortalServantValue = -mortalServantValue;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
stats: template.stats,
|
stats: template.stats,
|
||||||
underTarget: underTarget,
|
underTarget: underTarget,
|
||||||
@ -122,13 +139,15 @@ function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
|||||||
cost: cost,
|
cost: cost,
|
||||||
note: template.note,
|
note: template.note,
|
||||||
scoring: template.scoring,
|
scoring: template.scoring,
|
||||||
mortalServantValue: mortalServantValue
|
mortalServantValue: mortalServantValue,
|
||||||
|
flipped: flipped ?? false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let table = new SkillsTable();
|
let table = new SkillsTable();
|
||||||
|
|
||||||
export let bat0 = table.add({
|
export let bat0 = table.add({
|
||||||
|
isDegrading: false,
|
||||||
governing: governing("bat", 0),
|
governing: governing("bat", 0),
|
||||||
profile: {
|
profile: {
|
||||||
name: "Screech",
|
name: "Screech",
|
||||||
@ -323,6 +342,37 @@ export let lore3 = table.add({
|
|||||||
prereqs: [lore2]
|
prereqs: [lore2]
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export let sorry0 = table.add({
|
||||||
|
isDegrading: true,
|
||||||
|
governing: governing("penance", 0, true),
|
||||||
|
profile: {
|
||||||
|
name: "I'm Sorry",
|
||||||
|
description: "You really hurt your Master, you know? Shame on you."
|
||||||
|
},
|
||||||
|
prereqs: [],
|
||||||
|
})
|
||||||
|
|
||||||
|
export let sorry1 = table.add({
|
||||||
|
isDegrading: true,
|
||||||
|
governing: governing("penance", 1, true),
|
||||||
|
profile: {
|
||||||
|
name: "I'm So Sorry",
|
||||||
|
description: "You should have known better! You should have done what you were told."
|
||||||
|
},
|
||||||
|
prereqs: [],
|
||||||
|
})
|
||||||
|
|
||||||
|
export let sorry2 = table.add({
|
||||||
|
isDegrading: true,
|
||||||
|
// difficulty 2 is genuinely brutal
|
||||||
|
governing: governing("penance", 1.25, true),
|
||||||
|
profile: {
|
||||||
|
name: "Forgive Me",
|
||||||
|
description: "Nothing you say will ever be enough to make up for your indiscretion.",
|
||||||
|
},
|
||||||
|
prereqs: [],
|
||||||
|
})
|
||||||
|
|
||||||
export function getSkills(): SkillsTable {
|
export function getSkills(): SkillsTable {
|
||||||
return table;
|
return table;
|
||||||
}
|
}
|
@ -1,8 +1,13 @@
|
|||||||
import {ALL_STATS, Stat, SuccessorOption} from "./datatypes.ts";
|
import {ALL_STATS, Skill, Stat, SuccessorOption} from "./datatypes.ts";
|
||||||
import {generateName, generateTitle} from "./namegen.ts";
|
import {generateName, generateTitle} from "./namegen.ts";
|
||||||
import {choose} from "./utils.ts";
|
import {choose} from "./utils.ts";
|
||||||
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
|
|
||||||
|
export function generateSuccessors(nImprovements: number, penance: boolean): SuccessorOption[] {
|
||||||
|
if (penance) {
|
||||||
|
return [generateSuccessorFromPlayer()];
|
||||||
|
}
|
||||||
|
|
||||||
export function generateSuccessors(nImprovements: number): SuccessorOption[] {
|
|
||||||
let options = [];
|
let options = [];
|
||||||
while (options.length < 3) {
|
while (options.length < 3) {
|
||||||
let option = generateSuccessor(nImprovements);
|
let option = generateSuccessor(nImprovements);
|
||||||
@ -23,6 +28,25 @@ function isEligible(existing: SuccessorOption[], added: SuccessorOption) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function generateSuccessorFromPlayer(): SuccessorOption {
|
||||||
|
let progress = getPlayerProgress();
|
||||||
|
let successor = {
|
||||||
|
name: progress.name,
|
||||||
|
title: "Penitent",
|
||||||
|
note: "Failed at Master's bidding",
|
||||||
|
stats: {...progress.getStats()},
|
||||||
|
talents: {...progress.getTalents()},
|
||||||
|
skills: [...progress.getLearnedSkills()],
|
||||||
|
inPenance: true,
|
||||||
|
isCompulsory: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let stat of ALL_STATS.values()) {
|
||||||
|
successor.talents[stat] = -8;
|
||||||
|
}
|
||||||
|
return successor;
|
||||||
|
}
|
||||||
|
|
||||||
export function generateSuccessor(nImprovements: number): SuccessorOption {
|
export function generateSuccessor(nImprovements: number): SuccessorOption {
|
||||||
let name = generateName();
|
let name = generateName();
|
||||||
let title = generateTitle();
|
let title = generateTitle();
|
||||||
@ -50,5 +74,8 @@ export function generateSuccessor(nImprovements: number): SuccessorOption {
|
|||||||
improvement();
|
improvement();
|
||||||
}
|
}
|
||||||
|
|
||||||
return {name, title, note, stats, talents};
|
let skills: Skill[] = [];
|
||||||
|
let inPenance = false;
|
||||||
|
let isCompulsory = false;
|
||||||
|
return {name, title, note, stats, talents, skills, inPenance, isCompulsory};
|
||||||
}
|
}
|
180
src/wishes.ts
180
src/wishes.ts
@ -6,15 +6,17 @@ import {
|
|||||||
charm0,
|
charm0,
|
||||||
charm1,
|
charm1,
|
||||||
charm2,
|
charm2,
|
||||||
charm3,
|
charm3, getSkills,
|
||||||
lore0, lore1, lore2,
|
lore0, lore1, lore2,
|
||||||
party0,
|
party0,
|
||||||
party1, party2, party3, stare0, stare1, stare2, stare3,
|
party1, party2, party3, sorry0, sorry1, sorry2, stare0, stare1, stare2, stare3,
|
||||||
stealth0,
|
stealth0,
|
||||||
stealth1,
|
stealth1,
|
||||||
stealth2,
|
stealth2,
|
||||||
stealth3
|
stealth3
|
||||||
} from "./skills.ts";
|
} from "./skills.ts";
|
||||||
|
import {compile, VNSceneBasisPart} from "./vnscene.ts";
|
||||||
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
|
|
||||||
class WishesTable {
|
class WishesTable {
|
||||||
#wishes: WishData[]
|
#wishes: WishData[]
|
||||||
@ -33,11 +35,13 @@ class WishesTable {
|
|||||||
return this.#wishes[wish.id];
|
return this.#wishes[wish.id];
|
||||||
}
|
}
|
||||||
|
|
||||||
getAllPossibleWishes(): Wish[] {
|
getAllRandomWishes(): Wish[] {
|
||||||
let wishes: Wish[] = [];
|
let wishes: Wish[] = [];
|
||||||
for (let i = 0; i < this.#wishes.length; i++) {
|
for (let i = 0; i < this.#wishes.length; i++) {
|
||||||
|
if (this.#wishes[i].isRandomlyAvailable) {
|
||||||
wishes.push({id: i});
|
wishes.push({id: i});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return wishes;
|
return wishes;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -47,32 +51,174 @@ export function getWishes(): WishesTable {
|
|||||||
return table;
|
return table;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const whisper: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "whisper.mp3"
|
||||||
|
}
|
||||||
|
|
||||||
export const celebritySocialite = table.add({
|
export const celebritySocialite = table.add({
|
||||||
profile: {name: "Celebrity Socialite"},
|
profile: {
|
||||||
|
name: "Celebrity Socialite",
|
||||||
|
note: "+Charm -Lore",
|
||||||
|
domicile: "Party Mansion",
|
||||||
|
reignSentence: "A lot of people know who you are and like you.",
|
||||||
|
failureName: "Z-List Bloodstarver",
|
||||||
|
failureDomicile: "Obscure Soap Ad",
|
||||||
|
failureReignSentence: "Nobody really knows who you are.",
|
||||||
|
failureSuccessorVerb: "Apologize For Your Failure",
|
||||||
|
},
|
||||||
|
isRandomlyAvailable: true,
|
||||||
|
isCompulsory: false,
|
||||||
bannedSkills: () => [lore0],
|
bannedSkills: () => [lore0],
|
||||||
discouragedSkills: () => [stealth0, stealth1, stealth2, stealth3],
|
discouragedSkills: () => [stealth0, stealth1, stealth2, stealth3],
|
||||||
encouragedSkills: () => [party0, party1, party2, party3],
|
encouragedSkills: () => [party0, party1, party2, party3],
|
||||||
requiredSkills: () => [charm0, charm1, charm2],
|
requiredSkills: () => [charm0, charm1, charm2],
|
||||||
|
prologue: compile([
|
||||||
|
whisper,
|
||||||
|
"Master?",
|
||||||
|
whisper,
|
||||||
|
"I see.",
|
||||||
|
"You. I -- should I buy a guitar or something?",
|
||||||
|
whisper,
|
||||||
|
"My looks and my party skills...",
|
||||||
|
]),
|
||||||
|
onFailure: compile([
|
||||||
|
whisper,
|
||||||
|
"You're displeased...",
|
||||||
|
whisper,
|
||||||
|
"I'm not popular enough?",
|
||||||
|
"I see.",
|
||||||
|
]),
|
||||||
|
onVictory: compile([
|
||||||
|
whisper,
|
||||||
|
"I did as you commanded.",
|
||||||
|
"You're pleased?",
|
||||||
|
"... I'm free.",
|
||||||
|
])
|
||||||
});
|
});
|
||||||
|
|
||||||
export const nightswornAlchemist = table.add({
|
export const nightswornAlchemist = table.add({
|
||||||
profile: {name: "Nightsworn Alchemist"},
|
profile: {
|
||||||
|
name: "Nightsworn Alchemist",
|
||||||
|
note: "+Lore -Party",
|
||||||
|
domicile: "Alchemical Lab",
|
||||||
|
reignSentence: "You understand the fundamental connection between wine and blood.",
|
||||||
|
failureName: "Failure of Science",
|
||||||
|
failureDomicile: "Remedial College",
|
||||||
|
failureReignSentence: "You don't understand much of anything.",
|
||||||
|
failureSuccessorVerb: "Apologize For Your Failure",
|
||||||
|
},
|
||||||
|
isRandomlyAvailable: true,
|
||||||
|
isCompulsory: false,
|
||||||
bannedSkills: () => [party0],
|
bannedSkills: () => [party0],
|
||||||
discouragedSkills: () => [charm0, charm1, charm2, charm3],
|
discouragedSkills: () => [charm0, charm1, charm2, charm3],
|
||||||
encouragedSkills: () => [stare0, stare1, stare2, stare3],
|
encouragedSkills: () => [stare0, stare1, stare2, stare3],
|
||||||
requiredSkills: () => [lore0, lore1, lore2]
|
requiredSkills: () => [lore0, lore1, lore2],
|
||||||
|
prologue: compile([
|
||||||
|
whisper,
|
||||||
|
"Master?",
|
||||||
|
whisper,
|
||||||
|
"I see.",
|
||||||
|
"You. I -- should dedicate my life to the vampiric sciences.",
|
||||||
|
whisper,
|
||||||
|
"My looks and my party skills...",
|
||||||
|
]),
|
||||||
|
onFailure: compile([
|
||||||
|
whisper,
|
||||||
|
"You're displeased...",
|
||||||
|
whisper,
|
||||||
|
"I should have learned more lore.",
|
||||||
|
]),
|
||||||
|
onVictory: compile([
|
||||||
|
whisper,
|
||||||
|
"I did as you commanded.",
|
||||||
|
"You're pleased?",
|
||||||
|
"... I'm free.",
|
||||||
|
])
|
||||||
});
|
});
|
||||||
|
|
||||||
export const batFreak = table.add({
|
export const batFreak = table.add({
|
||||||
profile: {name: "Bat Freak"},
|
profile: {
|
||||||
|
name: "Bat Freak",
|
||||||
|
note: "++Bat -All",
|
||||||
|
domicile: "Master's Chiropteriary",
|
||||||
|
reignSentence: "You're an idol among bats.",
|
||||||
|
failureName: "Practically Mortal",
|
||||||
|
failureDomicile: "Right Side Up",
|
||||||
|
failureReignSentence: "Bats can tell you don't skreek correctly.",
|
||||||
|
failureSuccessorVerb: "Apologize -- SKREEK!",
|
||||||
|
},
|
||||||
|
isRandomlyAvailable: true,
|
||||||
|
isCompulsory: false,
|
||||||
bannedSkills: () => [charm0, stare0, party0, lore0],
|
bannedSkills: () => [charm0, stare0, party0, lore0],
|
||||||
discouragedSkills: () => [],
|
discouragedSkills: () => [],
|
||||||
encouragedSkills: () => [stealth0, stealth1, stealth2, stealth3],
|
encouragedSkills: () => [stealth0, stealth1, stealth2, stealth3],
|
||||||
requiredSkills: () => [bat0, bat1, bat2, bat3]
|
requiredSkills: () => [bat0, bat1, bat2, bat3],
|
||||||
|
prologue: compile([
|
||||||
|
whisper,
|
||||||
|
"Master?",
|
||||||
|
whisper,
|
||||||
|
"I see.",
|
||||||
|
"You -- SKKREEK -- want me to become a -- SKKREEK --",
|
||||||
|
]),
|
||||||
|
onFailure: compile([
|
||||||
|
whisper,
|
||||||
|
"You're displeased...",
|
||||||
|
whisper,
|
||||||
|
"I -- SKREEEEK -- should have spent more time becoming a bat...",
|
||||||
|
]),
|
||||||
|
onVictory: compile([
|
||||||
|
whisper,
|
||||||
|
"SKRSKRSKRSK.",
|
||||||
|
"I'm FREEEEEEEEEE --",
|
||||||
|
])
|
||||||
});
|
});
|
||||||
|
|
||||||
export function generateWishes(): Wish[] {
|
export const repent = table.add({
|
||||||
let possibleWishes = table.getAllPossibleWishes();
|
profile: {
|
||||||
|
name: "Not Even Fit To Be Bat Food",
|
||||||
|
note: "--All",
|
||||||
|
domicile: "Master's Home",
|
||||||
|
reignSentence: "You are almost, but not quite loved.",
|
||||||
|
failureName: "Can't Even Repent Correctly",
|
||||||
|
failureDomicile: "Homeless",
|
||||||
|
failureReignSentence: "You are unloved and disrespected.",
|
||||||
|
failureSuccessorVerb: "Apologize Again",
|
||||||
|
},
|
||||||
|
isRandomlyAvailable: false,
|
||||||
|
isCompulsory: true,
|
||||||
|
bannedSkills: () => getSkills().getAvailableSkills(false),
|
||||||
|
discouragedSkills: () => [],
|
||||||
|
encouragedSkills: () => [],
|
||||||
|
requiredSkills: () => [sorry0, sorry1, sorry2],
|
||||||
|
prologue: compile([
|
||||||
|
whisper,
|
||||||
|
"I'm sorry.",
|
||||||
|
"Please...",
|
||||||
|
whisper,
|
||||||
|
"I must repent."
|
||||||
|
]),
|
||||||
|
onFailure: compile([
|
||||||
|
whisper,
|
||||||
|
"I can't --",
|
||||||
|
"I must --",
|
||||||
|
whisper,
|
||||||
|
"Master -- please, no, I --"
|
||||||
|
]),
|
||||||
|
onVictory: compile([
|
||||||
|
whisper,
|
||||||
|
"Yes, I see.",
|
||||||
|
"I'm free...?"
|
||||||
|
])
|
||||||
|
});
|
||||||
|
|
||||||
|
export function generateWishes(penance: boolean): Wish[] {
|
||||||
|
if (penance) {
|
||||||
|
return [repent];
|
||||||
|
}
|
||||||
|
|
||||||
|
let possibleWishes = table.getAllRandomWishes();
|
||||||
shuffle(possibleWishes);
|
shuffle(possibleWishes);
|
||||||
|
|
||||||
let selectedWishes: Wish[] = [];
|
let selectedWishes: Wish[] = [];
|
||||||
@ -104,3 +250,17 @@ export function getCostMultiplier(wish: Wish | null, skill: Skill): number {
|
|||||||
|
|
||||||
return 1.0;
|
return 1.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function isWishCompleted(wish: Wish): boolean {
|
||||||
|
let player = getPlayerProgress();
|
||||||
|
|
||||||
|
let wishData = getWishes().get(wish);
|
||||||
|
|
||||||
|
for (let subj of wishData.requiredSkills()) {
|
||||||
|
if (!player.hasLearned(subj)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user