Ending selector, VN sequences
This commit is contained in:
parent
047248adb6
commit
3631144f3c
@ -6,7 +6,12 @@ export type Resource = "EXP";
|
|||||||
export const ALL_RESOURCES: Array<Resource> = ["EXP"]
|
export const ALL_RESOURCES: Array<Resource> = ["EXP"]
|
||||||
|
|
||||||
export type SkillGoverning = {
|
export type SkillGoverning = {
|
||||||
stats: Stat[], underTarget: number, target: number, cost: number, note: string
|
stats: Stat[],
|
||||||
|
underTarget: number,
|
||||||
|
target: number,
|
||||||
|
cost: number,
|
||||||
|
note: string,
|
||||||
|
scoring: SkillScoring,
|
||||||
};
|
};
|
||||||
export type SkillProfile = {
|
export type SkillProfile = {
|
||||||
name: string,
|
name: string,
|
||||||
@ -19,6 +24,10 @@ export type SkillData = {
|
|||||||
prereqs: Skill[]
|
prereqs: Skill[]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ScoringCategory = "bat" | "stealth" | "charm" | "stare" | "party" | "lore";
|
||||||
|
export const SCORING_CATEGORIES: ScoringCategory[] = ["bat", "stealth", "charm", "stare", "party", "lore"];
|
||||||
|
export type SkillScoring = {[P in ScoringCategory]?: number};
|
||||||
|
|
||||||
export type Skill = {
|
export type Skill = {
|
||||||
id: number
|
id: number
|
||||||
}
|
}
|
||||||
|
129
src/endings.ts
Normal file
129
src/endings.ts
Normal file
@ -0,0 +1,129 @@
|
|||||||
|
import {compile, VNScene, VNSceneBasisPart} from "./vnscene.ts";
|
||||||
|
|
||||||
|
const squeak: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "squeak.mp3"
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sceneBat: VNScene = compile([
|
||||||
|
squeak,
|
||||||
|
"I didn't.",
|
||||||
|
"I tried to, but --",
|
||||||
|
squeak,
|
||||||
|
"Well, you know what they say.",
|
||||||
|
"But I didn't disappear, right?",
|
||||||
|
"I'm still here.",
|
||||||
|
squeak,
|
||||||
|
"That's actually _why_ I stopped.",
|
||||||
|
"Talking is --",
|
||||||
|
squeak,
|
||||||
|
"Well. You know.",
|
||||||
|
squeak,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const doorbell: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "doorbell.mp3"
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sceneStealth: VNScene = compile([
|
||||||
|
doorbell,
|
||||||
|
"Yeah, you can let yourself in.",
|
||||||
|
doorbell,
|
||||||
|
"I'll have it moved.",
|
||||||
|
"Just -- don't call Susan, OK?",
|
||||||
|
doorbell,
|
||||||
|
"Believe me, I'm good for the money.",
|
||||||
|
"I'm doing... a lot better than it looks like.",
|
||||||
|
doorbell,
|
||||||
|
"The fangs? They're not real.",
|
||||||
|
"I'm just like you.",
|
||||||
|
doorbell,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const phoneBeep: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "phonebeep.mp3"
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sceneCharm: VNScene = compile([
|
||||||
|
phoneBeep,
|
||||||
|
"How do I sound?",
|
||||||
|
phoneBeep,
|
||||||
|
"*chuckle*",
|
||||||
|
"Sorry. I didn't plan ahead of time.",
|
||||||
|
"We're not on the air?",
|
||||||
|
phoneBeep,
|
||||||
|
"Well, I want a song.",
|
||||||
|
"Can you put me through?",
|
||||||
|
phoneBeep,
|
||||||
|
"I really want it.",
|
||||||
|
"It's for my boyfriend. First boyfriend, sorry.",
|
||||||
|
phoneBeep,
|
||||||
|
"*chuckle*",
|
||||||
|
"Yeah. I guess I do.",
|
||||||
|
"Is that bad?",
|
||||||
|
phoneBeep,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const sleepyBreath: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "sleepyBreath.mp3"
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sceneStare: VNScene = compile([
|
||||||
|
sleepyBreath,
|
||||||
|
"Don't wake up.",
|
||||||
|
"You're home.",
|
||||||
|
"A lot of things have been happening pretty fast.",
|
||||||
|
sleepyBreath,
|
||||||
|
"You have a lot of friends. You stick together.",
|
||||||
|
"You didn't have that when you were younger.",
|
||||||
|
sleepyBreath,
|
||||||
|
"It's not bad when things change.",
|
||||||
|
"And all that other stuff. It was a long time ago.",
|
||||||
|
"I couldn't really stop myself...",
|
||||||
|
sleepyBreath,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const party: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "party.mp3"
|
||||||
|
};
|
||||||
|
|
||||||
|
export const sceneParty: VNScene = compile([
|
||||||
|
party, // party noises
|
||||||
|
"IT'S LOUD?",
|
||||||
|
party, // party noises
|
||||||
|
"I KNOW.",
|
||||||
|
party, // party noises
|
||||||
|
"LISTEN, I --",
|
||||||
|
party, // party noises
|
||||||
|
"IF I DON'T SEE YOU AGAIN, I --",
|
||||||
|
party, // party noises
|
||||||
|
]);
|
||||||
|
|
||||||
|
const ghost: VNSceneBasisPart = {
|
||||||
|
type: "message",
|
||||||
|
text: "...",
|
||||||
|
sfx: "ghost.mp3"
|
||||||
|
};
|
||||||
|
|
||||||
|
export const sceneLore: VNScene = compile([
|
||||||
|
ghost,
|
||||||
|
"Oh yeah, I canceled those.",
|
||||||
|
ghost,
|
||||||
|
"I'll tell you the answers later.",
|
||||||
|
"I guess if I said it now that would be --",
|
||||||
|
"It'd be like cheating, right?",
|
||||||
|
ghost,
|
||||||
|
"Don't say that! I visit you, don't I?",
|
||||||
|
ghost,
|
||||||
|
"Yeah. They remember.",
|
||||||
|
ghost,
|
||||||
|
]);
|
@ -111,6 +111,16 @@ class Input {
|
|||||||
isKeyReleased(key: string) : boolean {
|
isKeyReleased(key: string) : boolean {
|
||||||
return !this.#keyDown[key] && this.#previousKeyDown[key];
|
return !this.#keyDown[key] && this.#previousKeyDown[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isAnythingPressed(): boolean {
|
||||||
|
for (let k of Object.keys(this.#keyDown)) {
|
||||||
|
if (this.#keyDown[k] && !this.#previousKeyDown[k]) { return true }
|
||||||
|
}
|
||||||
|
for (let k of Object.keys(this.#mouseDown)) {
|
||||||
|
if (this.#mouseDown[k] && !this.#previousMouseDown[k]) { return true }
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let active = new Input();
|
let active = new Input();
|
||||||
|
@ -7,6 +7,7 @@ import {getHud} from "./hud.ts";
|
|||||||
import {getHotbar, Hotbar} from "./hotbar.ts";
|
import {getHotbar, Hotbar} from "./hotbar.ts";
|
||||||
import {getSkillsModal, SkillsModal} from "./skillsmodal.ts";
|
import {getSkillsModal, SkillsModal} from "./skillsmodal.ts";
|
||||||
import {getSleepModal, SleepModal} from "./sleepmodal.ts";
|
import {getSleepModal, SleepModal} from "./sleepmodal.ts";
|
||||||
|
import {getEndgameModal} from "./vnmodal.ts";
|
||||||
|
|
||||||
class MenuCamera {
|
class MenuCamera {
|
||||||
// measured in whole screens
|
// measured in whole screens
|
||||||
@ -88,6 +89,7 @@ export class Game implements IGame {
|
|||||||
});
|
});
|
||||||
withCamera("HUD", () => { getHud().update() })
|
withCamera("HUD", () => { getHud().update() })
|
||||||
this.#bottomThing?.update();
|
this.#bottomThing?.update();
|
||||||
|
getEndgameModal().update();
|
||||||
}
|
}
|
||||||
|
|
||||||
drawGameplay() {
|
drawGameplay() {
|
||||||
@ -96,6 +98,7 @@ export class Game implements IGame {
|
|||||||
});
|
});
|
||||||
withCamera("HUD", () => { getHud().draw() })
|
withCamera("HUD", () => { getHud().draw() })
|
||||||
this.#bottomThing?.draw()
|
this.#bottomThing?.draw()
|
||||||
|
getEndgameModal().draw();
|
||||||
}
|
}
|
||||||
|
|
||||||
#chooseBottomThing() {
|
#chooseBottomThing() {
|
||||||
|
14
src/hud.ts
14
src/hud.ts
@ -4,10 +4,11 @@ import {FG_BOLD, FG_TEXT} from "./colors.ts";
|
|||||||
import {ALL_STATS} from "./datatypes.ts";
|
import {ALL_STATS} from "./datatypes.ts";
|
||||||
import {getPlayerProgress} from "./playerprogress.ts";
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
import {getHuntMode} from "./huntmode.ts";
|
import {getHuntMode} from "./huntmode.ts";
|
||||||
|
import {getStateManager} from "./statemanager.ts";
|
||||||
|
|
||||||
export class Hud {
|
export class Hud {
|
||||||
get size(): Size {
|
get size(): Size {
|
||||||
return new Size(96, 160)
|
return new Size(96, 176)
|
||||||
}
|
}
|
||||||
|
|
||||||
update() { }
|
update() { }
|
||||||
@ -16,18 +17,19 @@ export class Hud {
|
|||||||
// D.fillRect(new Point(-4, -4), this.size.add(new Size(8, 8)), BG_INSET)
|
// D.fillRect(new Point(-4, -4), this.size.add(new Size(8, 8)), BG_INSET)
|
||||||
D.drawText("Pyrex", new Point(0, 0), FG_BOLD)
|
D.drawText("Pyrex", new Point(0, 0), FG_BOLD)
|
||||||
D.drawText(`Level ${getHuntMode().getDepth()}`, new Point(0, 16), FG_TEXT)
|
D.drawText(`Level ${getHuntMode().getDepth()}`, new Point(0, 16), FG_TEXT)
|
||||||
|
D.drawText(`Turn ${getStateManager().getTurn()}/${getStateManager().getMaxTurns()}`, new Point(0, 32), FG_TEXT)
|
||||||
|
|
||||||
let y = 48;
|
let y = 64;
|
||||||
let prog = getPlayerProgress();
|
let prog = getPlayerProgress();
|
||||||
for (let s of ALL_STATS.values()) {
|
for (let s of ALL_STATS.values()) {
|
||||||
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)
|
||||||
y += 16;
|
y += 16;
|
||||||
}
|
}
|
||||||
D.drawText("EXP", new Point(0, 128), FG_BOLD);
|
D.drawText("EXP", new Point(0, 144), FG_BOLD);
|
||||||
D.drawText(`${prog.getExperience()}`, new Point(32, 128), FG_TEXT);
|
D.drawText(`${prog.getExperience()}`, new Point(32, 144), FG_TEXT);
|
||||||
D.drawText("BLD", new Point(0, 144), FG_BOLD);
|
D.drawText("BLD", new Point(0, 160), FG_BOLD);
|
||||||
D.drawText(`${prog.getBlood()}cc`, new Point(32, 144), FG_TEXT);
|
D.drawText(`${prog.getBlood()}cc`, new Point(32, 160), FG_TEXT);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,10 +52,12 @@ export function withCamera(part: UIPart, cb: () => void) {
|
|||||||
|
|
||||||
// specific
|
// specific
|
||||||
export type Page = "Gameplay" | "Thralls";
|
export type Page = "Gameplay" | "Thralls";
|
||||||
export type UIPart = "BottomModal" | "Hotbar" | "HUD" | "Gameplay" | "Thralls";
|
export type UIPart = "BottomModal" | "FullscreenPopover" | "Hotbar" | "HUD" | "Gameplay" | "Thralls";
|
||||||
|
|
||||||
export function getPartPage(part: UIPart): Page {
|
export function getPartPage(part: UIPart): Page | null {
|
||||||
switch (part) {
|
switch (part) {
|
||||||
|
case "FullscreenPopover":
|
||||||
|
return null
|
||||||
case "BottomModal":
|
case "BottomModal":
|
||||||
case "Hotbar":
|
case "Hotbar":
|
||||||
case "HUD":
|
case "HUD":
|
||||||
@ -83,9 +85,15 @@ export function getPageLocation(page: Page): Point {
|
|||||||
export function getPartLocation(part: UIPart): Rect {
|
export function getPartLocation(part: UIPart): Rect {
|
||||||
// TODO: in pixels, not screens
|
// TODO: in pixels, not screens
|
||||||
let {w: screenW, h: screenH} = D.size;
|
let {w: screenW, h: screenH} = D.size;
|
||||||
let pageOffset = getPageLocation(getPartPage(part));
|
let page = getPartPage(part);
|
||||||
|
let pageOffset = page ? getPageLocation(page) : null;
|
||||||
let layoutRect = internalGetPartLayoutRect(part);
|
let layoutRect = internalGetPartLayoutRect(part);
|
||||||
|
|
||||||
|
if (pageOffset == null) {
|
||||||
|
// follow camera
|
||||||
|
return layoutRect.offset(D.camera);
|
||||||
|
}
|
||||||
|
|
||||||
return layoutRect.offset(new Point(
|
return layoutRect.offset(new Point(
|
||||||
pageOffset.x * screenW,
|
pageOffset.x * screenW,
|
||||||
pageOffset.y * screenH
|
pageOffset.y * screenH
|
||||||
@ -100,6 +108,8 @@ export function internalGetPartLayoutRect(part: UIPart) {
|
|||||||
alignX: AlignX.Center,
|
alignX: AlignX.Center,
|
||||||
alignY: AlignY.Bottom,
|
alignY: AlignY.Bottom,
|
||||||
});
|
});
|
||||||
|
case "FullscreenPopover":
|
||||||
|
return getLayoutRect(new Size(384, 384));
|
||||||
case "Gameplay":
|
case "Gameplay":
|
||||||
case "Thralls":
|
case "Thralls":
|
||||||
return getLayoutRect(new Size(384, 384));
|
return getLayoutRect(new Size(384, 384));
|
||||||
|
@ -132,6 +132,14 @@ export class PlayerProgress {
|
|||||||
});
|
});
|
||||||
return skillsAvailable.slice(0, 6)
|
return skillsAvailable.slice(0, 6)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getLearnedSkills() {
|
||||||
|
let learnedSkills = []
|
||||||
|
for (let s of this.#skillsLearned.values()) {
|
||||||
|
learnedSkills.push({id: s})
|
||||||
|
}
|
||||||
|
return learnedSkills;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let active: PlayerProgress = new PlayerProgress();
|
let active: PlayerProgress = new PlayerProgress();
|
||||||
|
69
src/scorer.ts
Normal file
69
src/scorer.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import {VNScene} from "./vnscene.ts";
|
||||||
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
|
import {getSkills} from "./skills.ts";
|
||||||
|
import {SCORING_CATEGORIES, ScoringCategory} from "./datatypes.ts";
|
||||||
|
import {sceneBat, sceneCharm, sceneLore, sceneParty, sceneStare, sceneStealth} from "./endings.ts";
|
||||||
|
|
||||||
|
class Scorer {
|
||||||
|
constructor() { }
|
||||||
|
|
||||||
|
|
||||||
|
pickEnding(): Ending {
|
||||||
|
let learnedSkills = getPlayerProgress().getLearnedSkills();
|
||||||
|
let scores: Record<string, number> = {};
|
||||||
|
|
||||||
|
for (const skill of learnedSkills.values()) {
|
||||||
|
let data = getSkills().get(skill);
|
||||||
|
for (let [category, number] of Object.entries(data.governing.scoring)) {
|
||||||
|
scores[category] = (scores[category] ?? 0) + number;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: This approach isn't efficient but it's easy to understand
|
||||||
|
// and it allows me to arbitrate ties however I want
|
||||||
|
const isMax = (cat: ScoringCategory, min: number) => {
|
||||||
|
let score = scores[cat] ?? 0;
|
||||||
|
scores[cat] = 0; // each category, once checked, can't disqualify any other category
|
||||||
|
|
||||||
|
if (score < min) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (let cat of SCORING_CATEGORIES.values()) {
|
||||||
|
if (scores[cat] > score) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isMax("stare", 3)) {
|
||||||
|
return {scene: sceneStare}
|
||||||
|
}
|
||||||
|
if (isMax("lore", 3)) {
|
||||||
|
return {scene: sceneLore};
|
||||||
|
}
|
||||||
|
if (isMax("charm", 2)) {
|
||||||
|
return {scene: sceneCharm}
|
||||||
|
}
|
||||||
|
if (isMax("party", 1)) {
|
||||||
|
return {scene: sceneParty};
|
||||||
|
}
|
||||||
|
if (isMax("stealth", 0)) {
|
||||||
|
return {scene: sceneStealth}
|
||||||
|
}
|
||||||
|
// if (isMax("bat")) {
|
||||||
|
{
|
||||||
|
return {scene: sceneBat};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
type Ending = {
|
||||||
|
scene: VNScene
|
||||||
|
}
|
||||||
|
|
||||||
|
let active = new Scorer();
|
||||||
|
export function getScorer(): Scorer {
|
||||||
|
return active;
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
import {Skill, SkillData, SkillGoverning, Stat} from "./datatypes.ts";
|
import {Skill, SkillData, SkillGoverning, SkillScoring, Stat} from "./datatypes.ts";
|
||||||
import {getPlayerProgress} from "./playerprogress.ts";
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
|
|
||||||
class SkillsTable {
|
class SkillsTable {
|
||||||
@ -60,16 +60,41 @@ type Difficulty = 0 | 1 | 2 | 3
|
|||||||
type GoverningTemplate = {
|
type GoverningTemplate = {
|
||||||
stats: Stat[],
|
stats: Stat[],
|
||||||
note: string
|
note: string
|
||||||
|
scoring: SkillScoring,
|
||||||
}
|
}
|
||||||
|
|
||||||
type Track = "bat" | "stealth" | "charm" | "stare" | "party" | "lore"
|
type Track = "bat" | "stealth" | "charm" | "stare" | "party" | "lore"
|
||||||
let templates: Record<Track, GoverningTemplate> = {
|
let templates: Record<Track, GoverningTemplate> = {
|
||||||
bat: { stats: ["AGI", "AGI", "PSI"], note: "Cheaper with AGI and PSI." },
|
bat: {
|
||||||
stealth: { stats: ["AGI", "AGI", "INT"], note: "Cheaper with AGI and INT." },
|
stats: ["AGI", "AGI", "PSI"],
|
||||||
charm: { stats: ["CHA", "PSI", "PSI"], note: "Cheaper with CHA and PSI." },
|
note: "Cheaper with AGI and PSI.",
|
||||||
stare: { stats: ["PSI", "PSI"], note: "Cheaper with PSI." },
|
scoring: {bat: 1},
|
||||||
party: { stats: ["CHA", "CHA", "PSI"], note: "Cheaper with CHA and PSI." },
|
},
|
||||||
lore: { stats: ["INT", "INT", "CHA"], note: "Cheaper with INT and CHA." },
|
stealth: {
|
||||||
|
stats: ["AGI", "AGI", "INT"],
|
||||||
|
note: "Cheaper with AGI and INT.",
|
||||||
|
scoring: {stealth: 1},
|
||||||
|
},
|
||||||
|
charm: {
|
||||||
|
stats: ["CHA", "PSI", "PSI"],
|
||||||
|
note: "Cheaper with CHA and PSI.",
|
||||||
|
scoring: {charm: 1},
|
||||||
|
},
|
||||||
|
stare: {
|
||||||
|
stats: ["PSI", "PSI"],
|
||||||
|
note: "Cheaper with PSI.",
|
||||||
|
scoring: {stare: 1},
|
||||||
|
},
|
||||||
|
party: {
|
||||||
|
stats: ["CHA", "CHA", "PSI"],
|
||||||
|
note: "Cheaper with CHA and PSI.",
|
||||||
|
scoring: {party: 1},
|
||||||
|
},
|
||||||
|
lore: {
|
||||||
|
stats: ["INT", "INT", "CHA"],
|
||||||
|
note: "Cheaper with INT and CHA.",
|
||||||
|
scoring: {lore: 1},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
||||||
@ -89,6 +114,7 @@ function governing(track: Track, difficulty: Difficulty): SkillGoverning {
|
|||||||
target: target,
|
target: target,
|
||||||
cost: cost,
|
cost: cost,
|
||||||
note: template.note,
|
note: template.note,
|
||||||
|
scoring: template.scoring,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,8 +5,7 @@ import {addButton} from "./button.ts";
|
|||||||
import {D} from "./engine/public.ts";
|
import {D} from "./engine/public.ts";
|
||||||
import {BG_INSET} from "./colors.ts";
|
import {BG_INSET} from "./colors.ts";
|
||||||
import {getSkillsModal} from "./skillsmodal.ts";
|
import {getSkillsModal} from "./skillsmodal.ts";
|
||||||
import {getPlayerProgress} from "./playerprogress.ts";
|
import {getStateManager} from "./statemanager.ts";
|
||||||
import {getHuntMode} from "./huntmode.ts";
|
|
||||||
|
|
||||||
export class SleepModal {
|
export class SleepModal {
|
||||||
#drawpile: DrawPile;
|
#drawpile: DrawPile;
|
||||||
@ -62,9 +61,7 @@ export class SleepModal {
|
|||||||
let remainingWidth = size.w - 160;
|
let remainingWidth = size.w - 160;
|
||||||
let nextRect = new Rect(new Point(160, 96), new Size(remainingWidth, 32));
|
let nextRect = new Rect(new Point(160, 96), new Size(remainingWidth, 32));
|
||||||
addButton(this.#drawpile, "Sleep (Next Day)", nextRect, true, () => {
|
addButton(this.#drawpile, "Sleep (Next Day)", nextRect, true, () => {
|
||||||
getPlayerProgress().refill();
|
getStateManager().advance();
|
||||||
getHuntMode().replaceMap();
|
|
||||||
getSleepModal().setShown(false);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
this.#drawpile.executeOnClick();
|
this.#drawpile.executeOnClick();
|
||||||
|
42
src/statemanager.ts
Normal file
42
src/statemanager.ts
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
|
import {getHuntMode} from "./huntmode.ts";
|
||||||
|
import {getSleepModal} from "./sleepmodal.ts";
|
||||||
|
import {getEndgameModal} from "./vnmodal.ts";
|
||||||
|
import {getScorer} from "./scorer.ts";
|
||||||
|
|
||||||
|
const N_TURNS: number = 9;
|
||||||
|
|
||||||
|
export class StateManager {
|
||||||
|
#turn: number;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.#turn = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
getTurn(): number {
|
||||||
|
return this.#turn
|
||||||
|
}
|
||||||
|
|
||||||
|
advance() {
|
||||||
|
this.#turn += 1;
|
||||||
|
|
||||||
|
if (this.#turn <= N_TURNS) {
|
||||||
|
getPlayerProgress().refill();
|
||||||
|
getHuntMode().replaceMap();
|
||||||
|
getSleepModal().setShown(false);
|
||||||
|
} else {
|
||||||
|
// TODO: Play a specific scene
|
||||||
|
let ending = getScorer().pickEnding();
|
||||||
|
getEndgameModal().play(ending.scene);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getMaxTurns() {
|
||||||
|
return N_TURNS
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let active: StateManager = new StateManager();
|
||||||
|
export function getStateManager(): StateManager {
|
||||||
|
return active
|
||||||
|
}
|
118
src/vnmodal.ts
Normal file
118
src/vnmodal.ts
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
import {D, I} from "./engine/public.ts";
|
||||||
|
import {AlignX, AlignY, Color, Point} from "./engine/datatypes.ts";
|
||||||
|
import {BG_OUTER, FG_BOLD} from "./colors.ts";
|
||||||
|
import {withCamera} from "./layout.ts";
|
||||||
|
import {VNScene, VNSceneMessage, VNScenePart} from "./vnscene.ts";
|
||||||
|
|
||||||
|
const WIDTH = 384;
|
||||||
|
const HEIGHT = 384;
|
||||||
|
|
||||||
|
class VNModal {
|
||||||
|
#scene: VNScene | null;
|
||||||
|
#nextIndex = 0;
|
||||||
|
#cathexis: SceneCathexis | null;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.#scene = null;
|
||||||
|
this.#nextIndex = 0;
|
||||||
|
this.#cathexis = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
get isShown(): boolean {
|
||||||
|
return this.#scene != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
play(scene: VNScene) {
|
||||||
|
this.#scene = scene
|
||||||
|
this.#nextIndex = 0;
|
||||||
|
this.#cathexis = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fixCathexis() {
|
||||||
|
if (this.#cathexis?.isDone()) {
|
||||||
|
this.#cathexis = null;
|
||||||
|
}
|
||||||
|
if (this.#scene == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.#cathexis == null) {
|
||||||
|
let ix = this.#nextIndex
|
||||||
|
if (ix < this.#scene?.length) {
|
||||||
|
this.#cathexis = createCathexis(this.#scene[ix])
|
||||||
|
this.#nextIndex += 1;
|
||||||
|
} else {
|
||||||
|
this.#scene = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
update() {
|
||||||
|
this.#fixCathexis()
|
||||||
|
if (!this.isShown) { return }
|
||||||
|
|
||||||
|
withCamera("FullscreenPopover", () => this.#update())
|
||||||
|
}
|
||||||
|
|
||||||
|
draw() {
|
||||||
|
if (!this.isShown) { return }
|
||||||
|
|
||||||
|
D.fillRect(new Point(0, 0), D.size, new Color(BG_OUTER.r, BG_OUTER.g, BG_OUTER.b, 255));
|
||||||
|
withCamera("FullscreenPopover", () => this.#draw())
|
||||||
|
}
|
||||||
|
|
||||||
|
#update() {
|
||||||
|
this.#cathexis?.update();
|
||||||
|
}
|
||||||
|
|
||||||
|
#draw() {
|
||||||
|
this.#cathexis?.draw();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SceneCathexis {
|
||||||
|
isDone(): boolean;
|
||||||
|
update(): void;
|
||||||
|
draw(): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
function createCathexis(part: VNScenePart): SceneCathexis {
|
||||||
|
switch (part.type) {
|
||||||
|
case "message":
|
||||||
|
return new SceneMessageCathexis(part)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class SceneMessageCathexis {
|
||||||
|
#message: VNSceneMessage;
|
||||||
|
#done: boolean;
|
||||||
|
|
||||||
|
constructor (message: VNSceneMessage) {
|
||||||
|
this.#message = message;
|
||||||
|
this.#done = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
isDone() {
|
||||||
|
return this.#done;
|
||||||
|
}
|
||||||
|
|
||||||
|
update() {
|
||||||
|
// TODO: SFX
|
||||||
|
if (I.isAnythingPressed()) {
|
||||||
|
this.#done = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
draw() {
|
||||||
|
D.drawText(this.#message.text, new Point(WIDTH/2, HEIGHT/2), FG_BOLD, {
|
||||||
|
alignX: AlignX.Center,
|
||||||
|
alignY: AlignY.Middle,
|
||||||
|
forceWidth: WIDTH
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let active: VNModal = new VNModal();
|
||||||
|
export function getEndgameModal() {
|
||||||
|
return active;
|
||||||
|
}
|
25
src/vnscene.ts
Normal file
25
src/vnscene.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
export type VNSceneMessage = {
|
||||||
|
type: "message",
|
||||||
|
text: string,
|
||||||
|
sfx?: string,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type VNSceneBasisPart = string | VNSceneMessage;
|
||||||
|
export type VNSceneBasis = VNSceneBasisPart[];
|
||||||
|
export type VNScenePart = VNSceneMessage;
|
||||||
|
export type VNScene = VNScenePart[];
|
||||||
|
|
||||||
|
export function compile(basis: VNSceneBasis): VNScene {
|
||||||
|
let out: VNScene = [];
|
||||||
|
for (let item of basis.values()) {
|
||||||
|
if (typeof item == 'string') {
|
||||||
|
out.push({
|
||||||
|
type: "message",
|
||||||
|
text: item,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
out.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user