Enhance ending sounds

This commit is contained in:
Pyrex 2025-02-23 17:50:02 -08:00
parent 81f498c804
commit f2f20b820e
19 changed files with 94 additions and 29 deletions

BIN
src/art/sounds/vn_bat.mp3 Normal file

Binary file not shown.

Binary file not shown.

BIN
src/art/sounds/vn_dance.mp3 Normal file

Binary file not shown.

Binary file not shown.

BIN
src/art/sounds/vn_ghost.mp3 Normal file

Binary file not shown.

BIN
src/art/sounds/vn_page.mp3 Normal file

Binary file not shown.

BIN
src/art/sounds/vn_phone.mp3 Normal file

Binary file not shown.

View File

@ -1,9 +1,18 @@
import { compile, VNScene, VNSceneBasisPart } from "./vnscene.ts"; import { compile, VNScene, VNSceneBasisPart } from "./vnscene.ts";
import {
sndVnBat,
sndVnBreath,
sndVnDance,
sndVnDoorbell,
sndVnGhost,
sndVnPage,
sndVnPhone,
} from "./sounds.ts";
const squeak: VNSceneBasisPart = { const squeak: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "squeak.mp3", sfx: sndVnBat,
}; };
export const sceneBat: VNScene = compile([ export const sceneBat: VNScene = compile([
@ -25,7 +34,7 @@ export const sceneBat: VNScene = compile([
const doorbell: VNSceneBasisPart = { const doorbell: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "doorbell.mp3", sfx: sndVnDoorbell,
}; };
export const sceneStealth: VNScene = compile([ export const sceneStealth: VNScene = compile([
@ -33,7 +42,7 @@ export const sceneStealth: VNScene = compile([
"Yeah, you can let yourself in.", "Yeah, you can let yourself in.",
doorbell, doorbell,
"I'll have it moved.", "I'll have it moved.",
"Just -- don't call Susan, OK?", "Just -- don't call Liz, OK?",
doorbell, doorbell,
"Believe me, I'm good for the money.", "Believe me, I'm good for the money.",
"I'm doing... a lot better than it looks like.", "I'm doing... a lot better than it looks like.",
@ -46,7 +55,7 @@ export const sceneStealth: VNScene = compile([
const phoneBeep: VNSceneBasisPart = { const phoneBeep: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "phonebeep.mp3", sfx: sndVnPhone,
}; };
export const sceneCharm: VNScene = compile([ export const sceneCharm: VNScene = compile([
@ -61,7 +70,7 @@ export const sceneCharm: VNScene = compile([
"Can you put me through?", "Can you put me through?",
phoneBeep, phoneBeep,
"I really want it.", "I really want it.",
"It's for my boyfriend. First boyfriend, sorry.", "It's for my boyfriend. My old boyfriend, sorry.",
phoneBeep, phoneBeep,
"*chuckle*", "*chuckle*",
"Yeah. I guess I do.", "Yeah. I guess I do.",
@ -72,7 +81,7 @@ export const sceneCharm: VNScene = compile([
const sleepyBreath: VNSceneBasisPart = { const sleepyBreath: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "sleepyBreath.mp3", sfx: sndVnBreath,
}; };
export const sceneStare: VNScene = compile([ export const sceneStare: VNScene = compile([
@ -93,7 +102,7 @@ export const sceneStare: VNScene = compile([
const party: VNSceneBasisPart = { const party: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "party.mp3", sfx: sndVnDance,
}; };
export const sceneParty: VNScene = compile([ export const sceneParty: VNScene = compile([
@ -111,7 +120,7 @@ export const sceneParty: VNScene = compile([
const ghost: VNSceneBasisPart = { const ghost: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "ghost.mp3", sfx: sndVnGhost,
}; };
export const sceneLore: VNScene = compile([ export const sceneLore: VNScene = compile([
@ -127,3 +136,37 @@ export const sceneLore: VNScene = compile([
"Yeah. They remember.", "Yeah. They remember.",
ghost, ghost,
]); ]);
const page: VNSceneBasisPart = {
type: "message",
text: "...",
sfx: sndVnPage,
};
export const sceneTrueEnding: VNScene = compile([
page,
"(This is a taxonomy. It's what nerds write instead of poetry.)",
page,
"INSECTIVORE. INSECTIPHAGE. INSECT-EATER. INSECT-EATING INSECT BAT.",
"CONSUMER OF BUGS.",
"We eat -- flies? They eat beetles.",
"We eat -- various bugs. Yes. And we hang in caves --",
"We deposit the shells in a heap. An absolutely massive heap --",
page,
"FRUCTIVORE. FRUITIPHAGE. FRUIT-EATING FRUIT BAT.",
"We eat -- grapes, melons, that kind of thing.",
"We unearth the heap.",
"We lay the shells in trenches in the furrows of a vineyard.",
page,
"There are two clades and in addition to that is a secret clade.",
page,
"(The pages are stuck together.)",
page,
"HEMOPHAGE. HEMOVORE. HEMATOPHAGE. BLOOD EATER.",
"It is not yet time to announce the sentence of fate.",
"We -- take the wine that grows from the branches.",
"That's a simplification.",
"This is the night deeper than any night that cannot be spoken of.",
page,
"OK -- now roll it.",
]);

View File

@ -67,7 +67,7 @@ export class Hud {
D.drawText("BLD", new Point(0, 160), FG_BOLD); D.drawText("BLD", new Point(0, 160), FG_BOLD);
let bloodAmount = prog.getBlood(); let bloodAmount = prog.getBlood();
let bloodColor = FG_TEXT; let bloodColor = FG_TEXT;
if (bloodAmount > 2000) { if (bloodAmount >= 2000) {
bloodColor = FG_TEXT_ENDORSED; bloodColor = FG_TEXT_ENDORSED;
} }
if (bloodAmount < 100) { if (bloodAmount < 100) {

View File

@ -207,7 +207,9 @@ export class HuntMode {
this.velocity = dxy; this.velocity = dxy;
// let friction do it // let friction do it
if (this.map.imposesBloodCosts) { if (this.map.imposesBloodCosts) {
getPlayerProgress().spendBlood(displacement.distance(new Point(0, 0)) * 10 / 3); getPlayerProgress().spendBlood(
(displacement.distance(new Point(0, 0)) * 10) / 3,
);
} }
} }

View File

@ -1,16 +1,9 @@
import { Architecture, LoadedNewMap } from "./newmap.ts"; import { Architecture, LoadedNewMap } from "./newmap.ts";
import { Grid, Point } from "./engine/datatypes.ts"; import { Grid, Point } from "./engine/datatypes.ts";
import { import { getThralls } from "./thralls.ts";
getThralls,
thrallCharm,
thrallLore,
thrallParty,
thrallStealth,
} from "./thralls.ts";
import { import {
LadderPickup, LadderPickup,
ThrallCollectionPlatePickup, ThrallCollectionPlatePickup,
ThrallPickup,
ThrallPosterPickup, ThrallPosterPickup,
ThrallRecruitedPickup, ThrallRecruitedPickup,
} from "./pickups.ts"; } from "./pickups.ts";

View File

@ -320,7 +320,7 @@ export class LadderPickup {
onClick(): boolean { onClick(): boolean {
if (getHuntMode().map.imposesBloodCosts) { if (getHuntMode().map.imposesBloodCosts) {
getPlayerProgress().addBlood(100); // this used to award 1k; 100 now is equivalent to what 300 blood used to be getPlayerProgress().addBlood(100); // this used to award 1k; 100 now is equivalent to what 300 blood used to be
} }
initHuntMode(new HuntMode(getHuntMode().depth + 1, generateMap())); initHuntMode(new HuntMode(getHuntMode().depth + 1, generateMap()));
return false; return false;

View File

@ -9,6 +9,7 @@ import {
sceneParty, sceneParty,
sceneStare, sceneStare,
sceneStealth, sceneStealth,
sceneTrueEnding,
} from "./endings.ts"; } from "./endings.ts";
import { generateWishes, getWishes, isWishCompleted } from "./wishes.ts"; import { generateWishes, getWishes, isWishCompleted } from "./wishes.ts";
import { generateSuccessors } from "./successors.ts"; import { generateSuccessors } from "./successors.ts";
@ -81,7 +82,12 @@ class Scorer {
// 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
else if (isMax("stare", 3)) { else if (vampiricSkills >= 24) {
scene = sceneTrueEnding;
rank = "Master Vampire";
domicile = "Third Clade";
reignSentence = "You know the truth, or at least your character does.";
} 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";
@ -122,7 +128,10 @@ class Scorer {
vampiricSkills, vampiricSkills,
mortalServants, mortalServants,
}; };
let successorOptions = generateSuccessors(getPlayerProgress().nImprovements + 2, penance); // TODO: generate nImprovements from mortalServants and the player's bsae improvements let successorOptions = generateSuccessors(
getPlayerProgress().nImprovements + 2,
penance,
); // TODO: generate nImprovements from mortalServants and the player's bsae improvements
let wishOptions = generateWishes(penance); let wishOptions = generateWishes(penance);
let progenerateVerb = penance ? "Repent" : "Progenerate"; let progenerateVerb = penance ? "Repent" : "Progenerate";

View File

@ -39,17 +39,18 @@ class SkillsTable {
computeCost(skill: Skill) { computeCost(skill: Skill) {
const _STAT_TO_TRIPS: Record<Stat, number> = { const _STAT_TO_TRIPS: Record<Stat, number> = {
"AGI": 1/7.2, // 8.4 is what I measured, but this seems very overpriced in practice AGI: 1 / 7.2, // 8.4 is what I measured, but this seems very overpriced in practice
"INT": 1/5.4, INT: 1 / 5.4,
"CHA": 1/4.8, CHA: 1 / 4.8,
"PSI": 1/7.0, PSI: 1 / 7.0,
}; };
let data = this.get(skill); let data = this.get(skill);
let governingStatValue = 0; let governingStatValue = 0;
for (let stat of data.governing.stats.values()) { for (let stat of data.governing.stats.values()) {
governingStatValue += governingStatValue +=
getPlayerProgress().getStat(stat) * _STAT_TO_TRIPS[stat] / data.governing.stats.length; (getPlayerProgress().getStat(stat) * _STAT_TO_TRIPS[stat]) /
data.governing.stats.length;
} }
if (data.governing.flipped) { if (data.governing.flipped) {

View File

@ -10,6 +10,13 @@ import audRewardMedium from "./art/sounds/reward_medium.mp3";
import audRewardSmall from "./art/sounds/reward_small.mp3"; import audRewardSmall from "./art/sounds/reward_small.mp3";
import audSilence from "./art/sounds/silence.mp3"; import audSilence from "./art/sounds/silence.mp3";
import audSleep from "./art/sounds/sleep.mp3"; import audSleep from "./art/sounds/sleep.mp3";
import audVnBat from "./art/sounds/vn_bat.mp3";
import audVnBreath from "./art/sounds/vn_breath.mp3";
import audVnDance from "./art/sounds/vn_dance.mp3";
import audVnDoorbell from "./art/sounds/vn_doorbell.mp3";
import audVnGhost from "./art/sounds/vn_ghost.mp3";
import audVnPage from "./art/sounds/vn_page.mp3";
import audVnPhone from "./art/sounds/vn_phone.mp3";
import { Sound } from "./sound.ts"; import { Sound } from "./sound.ts";
export let sndBite = new Sound(audBite); export let sndBite = new Sound(audBite);
@ -24,6 +31,13 @@ export let sndRewardMedium = new Sound(audRewardMedium);
export let sndRewardSmall = new Sound(audRewardSmall); export let sndRewardSmall = new Sound(audRewardSmall);
export let sndSilence = new Sound(audSilence); export let sndSilence = new Sound(audSilence);
export let sndSleep = new Sound(audSleep); export let sndSleep = new Sound(audSleep);
export let sndVnBat = new Sound(audVnBat);
export let sndVnBreath = new Sound(audVnBreath);
export let sndVnDance = new Sound(audVnDance);
export let sndVnDoorbell = new Sound(audVnDoorbell);
export let sndVnGhost = new Sound(audVnGhost);
export let sndVnPage = new Sound(audVnPage);
export let sndVnPhone = new Sound(audVnPhone);
export function sndRewardFor(amount: number) { export function sndRewardFor(amount: number) {
if (amount <= 1) { if (amount <= 1) {

View File

@ -10,7 +10,6 @@ import { openingScene } from "./openingscene.ts";
import { generateName } from "./namegen.ts"; import { generateName } from "./namegen.ts";
import { photogenicThralls } from "./thralls.ts"; import { photogenicThralls } from "./thralls.ts";
import { choose } from "./utils.ts"; import { choose } from "./utils.ts";
import {generateSuccessor} from "./successors.ts";
const N_TURNS: number = 9; const N_TURNS: number = 9;

View File

@ -85,6 +85,7 @@ interface SceneCathexis {
function createCathexis(part: VNScenePart): SceneCathexis { function createCathexis(part: VNScenePart): SceneCathexis {
switch (part.type) { switch (part.type) {
case "message": case "message":
part?.sfx?.play({ volume: 0.5 });
return new SceneMessageCathexis(part); return new SceneMessageCathexis(part);
case "callback": case "callback":
part?.callback(); part?.callback();

View File

@ -1,7 +1,9 @@
import { Sound } from "./sound.ts";
export type VNSceneMessage = { export type VNSceneMessage = {
type: "message"; type: "message";
text: string; text: string;
sfx?: string; sfx?: Sound;
}; };
export type VNSceneCallback = { export type VNSceneCallback = {

View File

@ -31,6 +31,7 @@ import {
} from "./skills.ts"; } from "./skills.ts";
import { compile, VNSceneBasisPart } from "./vnscene.ts"; import { compile, VNSceneBasisPart } from "./vnscene.ts";
import { getPlayerProgress } from "./playerprogress.ts"; import { getPlayerProgress } from "./playerprogress.ts";
import { sndVnBreath } from "./sounds.ts";
class WishesTable { class WishesTable {
#wishes: WishData[]; #wishes: WishData[];
@ -68,7 +69,7 @@ export function getWishes(): WishesTable {
const whisper: VNSceneBasisPart = { const whisper: VNSceneBasisPart = {
type: "message", type: "message",
text: "...", text: "...",
sfx: "whisper.mp3", sfx: sndVnBreath,
}; };
export const celebritySocialite = table.add({ export const celebritySocialite = table.add({