Wire up endgame screen to real data
This commit is contained in:
parent
fc3c9ce02a
commit
5ecafa0d4a
@ -1,3 +1,4 @@
|
|||||||
|
import {VNScene} from "./vnscene.ts";
|
||||||
|
|
||||||
export type Stat = "AGI" | "INT" | "CHA" | "PSI";
|
export type Stat = "AGI" | "INT" | "CHA" | "PSI";
|
||||||
export const ALL_STATS: Array<Stat> = ["AGI", "INT", "CHA", "PSI"];
|
export const ALL_STATS: Array<Stat> = ["AGI", "INT", "CHA", "PSI"];
|
||||||
@ -31,3 +32,39 @@ export type SkillScoring = {[P in ScoringCategory]?: number};
|
|||||||
export type Skill = {
|
export type Skill = {
|
||||||
id: number
|
id: number
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export type Wish = "celebritySocialite" | "nightswornAlchemist" | "batFreak";
|
||||||
|
|
||||||
|
// endings
|
||||||
|
|
||||||
|
export type Ending = {
|
||||||
|
scene: VNScene
|
||||||
|
personal: EndingPersonal,
|
||||||
|
analytics: EndingAnalytics,
|
||||||
|
successorOptions: SuccessorOption[],
|
||||||
|
wishOptions: Wish[],
|
||||||
|
|
||||||
|
// forcedSuccessors: number[] | null,
|
||||||
|
// forcedWishes: number[] | null
|
||||||
|
}
|
||||||
|
|
||||||
|
export type EndingPersonal = {
|
||||||
|
rank: string,
|
||||||
|
domicile: string,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type EndingAnalytics = {
|
||||||
|
itemsPurloined: number,
|
||||||
|
vampiricSkills: number,
|
||||||
|
mortalServants: number,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SuccessorOption = {
|
||||||
|
name: string,
|
||||||
|
title: string,
|
||||||
|
note: string | null, // ex "already a vampire"
|
||||||
|
stats: Record<Stat, number>,
|
||||||
|
talents: Record<Stat, number>
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -4,33 +4,34 @@ import {BG_INSET, FG_BOLD, FG_TEXT} from "./colors.ts";
|
|||||||
import {AlignX, AlignY, Point, Rect, Size} from "./engine/datatypes.ts";
|
import {AlignX, AlignY, Point, Rect, Size} from "./engine/datatypes.ts";
|
||||||
import {DrawPile} from "./drawpile.ts";
|
import {DrawPile} from "./drawpile.ts";
|
||||||
import {addButton} from "./button.ts";
|
import {addButton} from "./button.ts";
|
||||||
import {ALL_STATS} from "./datatypes.ts";
|
import {ALL_STATS, Ending, Wish} from "./datatypes.ts";
|
||||||
|
import {getScorer} from "./scorer.ts";
|
||||||
|
|
||||||
const WIDTH = 384;
|
const WIDTH = 384;
|
||||||
const HEIGHT = 384;
|
const HEIGHT = 384;
|
||||||
|
|
||||||
export class EndgameModal {
|
export class EndgameModal {
|
||||||
#isShown: boolean;
|
|
||||||
#drawpile: DrawPile;
|
#drawpile: DrawPile;
|
||||||
#page: number;
|
#page: number;
|
||||||
|
|
||||||
constructor() {
|
#ending: Ending | null;
|
||||||
this.#isShown = false;
|
|
||||||
this.#drawpile = new DrawPile();
|
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.#drawpile = new DrawPile();
|
||||||
this.#page = 0;
|
this.#page = 0;
|
||||||
|
|
||||||
|
this.show(getScorer().pickEnding());
|
||||||
|
|
||||||
// debug
|
// debug
|
||||||
this.show();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get isShown(): boolean {
|
get isShown(): boolean {
|
||||||
return this.#isShown;
|
return this.#ending != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
show() {
|
show(ending: Ending) {
|
||||||
this.#isShown = true;
|
|
||||||
this.#page = 0;
|
this.#page = 0;
|
||||||
|
this.#ending = ending;
|
||||||
}
|
}
|
||||||
|
|
||||||
update() {
|
update() {
|
||||||
@ -44,17 +45,30 @@ export class EndgameModal {
|
|||||||
#update() {
|
#update() {
|
||||||
this.#drawpile.clear();
|
this.#drawpile.clear();
|
||||||
if (this.#page == 0) {
|
if (this.#page == 0) {
|
||||||
|
let analytics = this.#ending?.analytics;
|
||||||
|
let rank = this.#ending?.personal?.rank ?? "No Rank";
|
||||||
|
let domicile = this.#ending?.personal?.domicile ?? "No Domicile";
|
||||||
|
let itemsPurloined = analytics?.itemsPurloined ?? 0;
|
||||||
|
let vampiricSkills = analytics?.vampiricSkills ?? 0;
|
||||||
|
let mortalServants = analytics?.mortalServants ?? 0;
|
||||||
|
|
||||||
this.#drawpile.add(0, () => {
|
this.#drawpile.add(0, () => {
|
||||||
D.drawText("It is time to announce the sentence of fate.", new Point(0, 0), FG_TEXT)
|
D.drawText("It is time to announce the sentence of fate.", new Point(0, 0), FG_TEXT)
|
||||||
D.drawText("You are no longer a fledgling. Your new rank:", new Point(0, 32), FG_TEXT)
|
D.drawText("You are no longer a fledgling. Your new rank:", new Point(0, 32), FG_TEXT)
|
||||||
D.drawText("Progenitor", new Point(WIDTH / 2, 64), FG_BOLD, {alignX: AlignX.Center})
|
D.drawText(rank, new Point(WIDTH / 2, 64), FG_BOLD, {alignX: AlignX.Center})
|
||||||
D.drawText("You have achieved a DOMICILE STATUS of:", new Point(0, 96), FG_TEXT)
|
D.drawText("You have achieved a DOMICILE STATUS of:", new Point(0, 96), FG_TEXT)
|
||||||
D.drawText("Guest House", new Point(WIDTH / 2, 128), FG_BOLD, {alignX: AlignX.Center})
|
D.drawText(domicile, new Point(WIDTH / 2, 128), FG_BOLD, {alignX: AlignX.Center})
|
||||||
D.drawText("where you live with many friends.", new Point(0, 160), FG_TEXT)
|
D.drawText("where you live with many friends.", new Point(0, 160), FG_TEXT) // TODO: Vary this text
|
||||||
D.drawText("You have achieved:", new Point(0, 192), FG_TEXT)
|
D.drawText("You have achieved:", new Point(0, 192), FG_TEXT)
|
||||||
D.drawText("48 items purloined\n96 vampiric skills\n50 mortal servants", new Point(WIDTH / 2, 224), FG_TEXT, {alignX: AlignX.Center})
|
D.drawText(
|
||||||
D.drawText("48 \n96 \n50 ", new Point(WIDTH / 2, 224), FG_BOLD, {alignX: AlignX.Center})
|
`${itemsPurloined} items purloined\n${vampiricSkills} vampiric skills\n${mortalServants} mortal servants`,
|
||||||
D.drawText("That feels like a lot!", new Point(0, 288), FG_TEXT)
|
new Point(WIDTH / 2, 224), FG_TEXT, {alignX: AlignX.Center}
|
||||||
|
)
|
||||||
|
D.drawText(
|
||||||
|
`${itemsPurloined} \n${vampiricSkills} \n${mortalServants} `,
|
||||||
|
new Point(WIDTH / 2, 224), FG_BOLD, {alignX: AlignX.Center}
|
||||||
|
);
|
||||||
|
D.drawText("That feels like a lot!", new Point(0, 288), FG_TEXT) // TODO: Vary this text
|
||||||
D.drawText("Your reign continues unimpeded from the shadows. It is now time to", new Point(0, 320), FG_TEXT, {forceWidth: WIDTH})
|
D.drawText("Your reign continues unimpeded from the shadows. It is now time to", new Point(0, 320), FG_TEXT, {forceWidth: WIDTH})
|
||||||
})
|
})
|
||||||
addButton(
|
addButton(
|
||||||
@ -114,7 +128,15 @@ export class EndgameModal {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#addCandidate(ix: number, at: Point) {
|
#addCandidate(ix: number, at: Point) {
|
||||||
let names = ["Marty", "Karla", "Steve"];
|
let candidates = this.#ending?.successorOptions;
|
||||||
|
if (candidates == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let candidate = candidates[ix];
|
||||||
|
if (candidate == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let w = WIDTH;
|
let w = WIDTH;
|
||||||
let h = 64;
|
let h = 64;
|
||||||
|
|
||||||
@ -136,7 +158,8 @@ export class EndgameModal {
|
|||||||
at.offset(new Point(0, 4)), new Size(w, h - 8), fg,
|
at.offset(new Point(0, 4)), new Size(w, h - 8), fg,
|
||||||
)
|
)
|
||||||
|
|
||||||
D.drawText(names[ix], at.offset(new Point(4, 8)), fgBold)
|
D.drawText(candidate.name + ", " + candidate.title, at.offset(new Point(4, 8)), fg);
|
||||||
|
D.drawText(candidate.name, at.offset(new Point(4, 8)), fgBold);
|
||||||
|
|
||||||
let xys = [
|
let xys = [
|
||||||
new Point(4, 24), new Point(4, 40),
|
new Point(4, 24), new Point(4, 40),
|
||||||
@ -144,15 +167,20 @@ export class EndgameModal {
|
|||||||
];
|
];
|
||||||
let i = 0;
|
let i = 0;
|
||||||
for (let s of ALL_STATS.values()) {
|
for (let s of ALL_STATS.values()) {
|
||||||
|
let statValue = candidate.stats[s];
|
||||||
|
let talentValue = candidate.talents[s];
|
||||||
|
|
||||||
D.drawText(s, at.offset(xys[i]), fg)
|
D.drawText(s, at.offset(xys[i]), fg)
|
||||||
D.drawText("10", at.offset(xys[i].offset(new Point(32, 0))), fgBold)
|
D.drawText(`${statValue}`, at.offset(xys[i].offset(new Point(32, 0))), fgBold)
|
||||||
D.drawText("(+4)", 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
D.drawText("Former barista", at.offset(new Point(224, 24)), fg)
|
if (candidate.note != null) {
|
||||||
if (ix == 2) {
|
D.drawText(candidate.note, at.offset(new Point(224, 24)), fg, {forceWidth: w - 224})
|
||||||
D.drawText("Already a vampire", at.offset(new Point(224, 40)), fg)
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
generalRect,
|
generalRect,
|
||||||
@ -164,17 +192,19 @@ export class EndgameModal {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#addWish(ix: number, at: Point) {
|
#addWish(ix: number, at: Point) {
|
||||||
let wishes = [
|
let wishOptions = this.#ending?.wishOptions;
|
||||||
"Celebrity Socialite",
|
if (wishOptions == null) {
|
||||||
"Nightsworn Alchemist",
|
return;
|
||||||
"Bat Freak"
|
}
|
||||||
]
|
let wishOption = wishOptions[ix];
|
||||||
let selected = ix == 1;
|
let selected = ix == 1;
|
||||||
let w = 128;
|
let w = 128;
|
||||||
let h = 72;
|
let h = 72;
|
||||||
let generalRect = new Rect(at, new Size(w, h));
|
let generalRect = new Rect(at, new Size(w, h));
|
||||||
let enabled = true;
|
let enabled = true;
|
||||||
|
|
||||||
|
let wishLabel = wishLabels[wishOption];
|
||||||
|
|
||||||
this.#drawpile.addClickable(
|
this.#drawpile.addClickable(
|
||||||
0,
|
0,
|
||||||
(hover) => {
|
(hover) => {
|
||||||
@ -189,7 +219,7 @@ export class EndgameModal {
|
|||||||
at.offset(new Point(2, 4)), new Size(w - 4, h - 8), fg,
|
at.offset(new Point(2, 4)), new Size(w - 4, h - 8), fg,
|
||||||
)
|
)
|
||||||
|
|
||||||
D.drawText(wishes[ix], at.offset(new Point(w / 2,h / 2 )), fgBold, {
|
D.drawText(wishLabel, at.offset(new Point(w / 2,h / 2 )), fgBold, {
|
||||||
forceWidth: w - 4,
|
forceWidth: w - 4,
|
||||||
alignX: AlignX.Center,
|
alignX: AlignX.Center,
|
||||||
alignY: AlignY.Middle,
|
alignY: AlignY.Middle,
|
||||||
@ -213,6 +243,12 @@ export class EndgameModal {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const wishLabels: Record<Wish, string> = {
|
||||||
|
celebritySocialite: "Celebrity Socialite",
|
||||||
|
nightswornAlchemist: "Nightsworn Alchemist",
|
||||||
|
batFreak: "Bat Freak"
|
||||||
|
}
|
||||||
|
|
||||||
let active = new EndgameModal();
|
let active = new EndgameModal();
|
||||||
export function getEndgameModal() {
|
export function getEndgameModal() {
|
||||||
return active;
|
return active;
|
||||||
|
@ -2,6 +2,7 @@ import {ConceptualCell, maps} from "./maps.ts";
|
|||||||
import {Grid, Point, Size} from "./engine/datatypes.ts";
|
import {Grid, Point, Size} from "./engine/datatypes.ts";
|
||||||
import {ALL_STATS} from "./datatypes.ts";
|
import {ALL_STATS} from "./datatypes.ts";
|
||||||
import {LoadedMap, MapCell, MapCellContent} from "./huntmode.ts";
|
import {LoadedMap, MapCell, MapCellContent} from "./huntmode.ts";
|
||||||
|
import {choose} from "./utils.ts";
|
||||||
|
|
||||||
export function generate(): LoadedMap {
|
export function generate(): LoadedMap {
|
||||||
let mapNames: Array<string> = Object.keys(maps);
|
let mapNames: Array<string> = Object.keys(maps);
|
||||||
@ -89,9 +90,3 @@ function generateContent(): MapCellContent {
|
|||||||
])();
|
])();
|
||||||
}
|
}
|
||||||
|
|
||||||
function choose<T>(array: Array<T>): T {
|
|
||||||
if (array.length == 0) {
|
|
||||||
throw `array cannot have length 0 for choose`
|
|
||||||
}
|
|
||||||
return array[Math.floor(Math.random() * array.length)]
|
|
||||||
}
|
|
||||||
|
50
src/namegen.ts
Normal file
50
src/namegen.ts
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
import {choose} from "./utils.ts";
|
||||||
|
|
||||||
|
const names = [
|
||||||
|
// vampires
|
||||||
|
"Vlad", "Drek",
|
||||||
|
// generic American names I like
|
||||||
|
"Kyle",
|
||||||
|
// friends I can defame
|
||||||
|
"Bhijn", "Myr", "Narry",
|
||||||
|
// aggressively furry names
|
||||||
|
"Tech",
|
||||||
|
// deities
|
||||||
|
"Quetzal", "Zotz",
|
||||||
|
// Nameberry's unique names
|
||||||
|
"Teleri", "Artis", "Lautaro", "Corbett", "Kestrel",
|
||||||
|
"Averil", "Sparrow", "Quillan", "Pipit", "Capella",
|
||||||
|
"Altair", "Lowell", "Leonie", "Vega", "Kea",
|
||||||
|
"Shai", "Teddy", "Howard", "Khalid", "Ozias",
|
||||||
|
"Zuko", "Ezio", "Zeno", "Thisby", "Calloway",
|
||||||
|
"Fenna", "Lupin", "Finlo", "Tycho", "Talmadge",
|
||||||
|
// others
|
||||||
|
"Jeff", "Jon", "Garrett", "Russell", "Tyson",
|
||||||
|
"Gervase", "Sonja", "Sue", "Richard", "Jankie",
|
||||||
|
// highly trustworthy individuals
|
||||||
|
"Nef", "Matt", "Sam"
|
||||||
|
]
|
||||||
|
export function generateName() {
|
||||||
|
return choose(names);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TOOD: Associate these with a stat difference
|
||||||
|
// mix of Liberal Crime Squad jobs, SS13 jobs, and jobs that amuse me
|
||||||
|
const titles = [
|
||||||
|
"Artist",
|
||||||
|
"Barista",
|
||||||
|
"Bartender",
|
||||||
|
"Blogger",
|
||||||
|
"Dragon Handler",
|
||||||
|
"Game Developer",
|
||||||
|
"Hypnotist",
|
||||||
|
"Journalist",
|
||||||
|
"Poker Player",
|
||||||
|
"Priest",
|
||||||
|
"Magician",
|
||||||
|
"Writer"
|
||||||
|
];
|
||||||
|
|
||||||
|
export function generateTitle() {
|
||||||
|
return choose(titles);
|
||||||
|
}
|
@ -1,13 +1,14 @@
|
|||||||
import {VNScene} from "./vnscene.ts";
|
import {VNScene} from "./vnscene.ts";
|
||||||
import {getPlayerProgress} from "./playerprogress.ts";
|
import {getPlayerProgress} from "./playerprogress.ts";
|
||||||
import {getSkills} from "./skills.ts";
|
import {getSkills} from "./skills.ts";
|
||||||
import {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 {generateSuccessors} from "./successors.ts";
|
||||||
|
|
||||||
class Scorer {
|
class Scorer {
|
||||||
constructor() { }
|
constructor() { }
|
||||||
|
|
||||||
|
|
||||||
pickEnding(): Ending {
|
pickEnding(): Ending {
|
||||||
let learnedSkills = getPlayerProgress().getLearnedSkills();
|
let learnedSkills = getPlayerProgress().getLearnedSkills();
|
||||||
let scores: Record<string, number> = {};
|
let scores: Record<string, number> = {};
|
||||||
@ -21,46 +22,78 @@ class Scorer {
|
|||||||
|
|
||||||
// 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
|
||||||
|
let runningScores: Record<string, number> = {...scores};
|
||||||
const isMax = (cat: ScoringCategory, min: number) => {
|
const isMax = (cat: ScoringCategory, min: number) => {
|
||||||
let score = scores[cat] ?? 0;
|
let score = runningScores[cat] ?? 0;
|
||||||
scores[cat] = 0; // each category, once checked, can't disqualify any other category
|
runningScores[cat] = 0; // each category, once checked, can't disqualify any other category
|
||||||
|
|
||||||
if (score < min) {
|
if (score < min) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
for (let cat of SCORING_CATEGORIES.values()) {
|
for (let cat of SCORING_CATEGORIES.values()) {
|
||||||
if (scores[cat] > score) {
|
if (runningScores[cat] > score) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let scene: VNScene;
|
||||||
|
let rank: string;
|
||||||
|
let domicile: string;
|
||||||
|
|
||||||
|
// TODO: Award different ranks depending on second-to-top skill
|
||||||
|
// TODO: Award different domiciles based on overall score
|
||||||
|
// TODO: Force the rank to match the wish if one existed
|
||||||
if (isMax("stare", 3)) {
|
if (isMax("stare", 3)) {
|
||||||
return {scene: sceneStare}
|
scene = sceneStare;
|
||||||
|
rank = "Hypno-Chiropteran";
|
||||||
|
domicile = "Village of Brainwashed Mortals";
|
||||||
}
|
}
|
||||||
if (isMax("lore", 3)) {
|
else if (isMax("lore", 3)) {
|
||||||
return {scene: sceneLore};
|
scene = sceneLore;
|
||||||
|
rank = "Loremaster";
|
||||||
|
domicile = "Vineyard";
|
||||||
}
|
}
|
||||||
if (isMax("charm", 2)) {
|
else if (isMax("charm", 2)) {
|
||||||
return {scene: sceneCharm}
|
scene = sceneCharm;
|
||||||
|
rank = "Seducer";
|
||||||
|
domicile = "Guest House";
|
||||||
}
|
}
|
||||||
if (isMax("party", 1)) {
|
else if (isMax("party", 1)) {
|
||||||
return {scene: sceneParty};
|
scene = sceneParty;
|
||||||
|
rank = "Party Animal";
|
||||||
|
domicile = "Nightclub";
|
||||||
}
|
}
|
||||||
if (isMax("stealth", 0)) {
|
else if (isMax("stealth", 0)) {
|
||||||
return {scene: sceneStealth}
|
scene = sceneStealth;
|
||||||
|
rank = "Invisible";
|
||||||
|
domicile = "Townhouse";
|
||||||
}
|
}
|
||||||
// if (isMax("bat")) {
|
// if (isMax("bat")) {
|
||||||
{
|
else {
|
||||||
return {scene: sceneBat};
|
scene = sceneBat;
|
||||||
}
|
rank = "Bat";
|
||||||
|
domicile = "Cave";
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
// TODO: Analytics tracker
|
||||||
|
let analytics = {
|
||||||
|
itemsPurloined: 0,
|
||||||
|
vampiricSkills: 0,
|
||||||
|
mortalServants: 0,
|
||||||
|
}
|
||||||
|
let successorOptions = generateSuccessors(0); // TODO: generate nImprovements from score
|
||||||
|
let wishOptions = generateWishes();
|
||||||
|
|
||||||
type Ending = {
|
return {
|
||||||
scene: VNScene
|
scene,
|
||||||
|
personal: {rank, domicile},
|
||||||
|
analytics,
|
||||||
|
successorOptions,
|
||||||
|
wishOptions,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let active = new Scorer();
|
let active = new Scorer();
|
||||||
|
@ -29,7 +29,7 @@ export class StateManager {
|
|||||||
// TODO: Play a specific scene
|
// TODO: Play a specific scene
|
||||||
let ending = getScorer().pickEnding();
|
let ending = getScorer().pickEnding();
|
||||||
getVNModal().play(ending.scene);
|
getVNModal().play(ending.scene);
|
||||||
getEndgameModal().show();
|
getEndgameModal().show(ending);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
56
src/successors.ts
Normal file
56
src/successors.ts
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
import {ALL_STATS, Stat, SuccessorOption} from "./datatypes.ts";
|
||||||
|
import {generateName, generateTitle} from "./namegen.ts";
|
||||||
|
import {choose} from "./utils.ts";
|
||||||
|
|
||||||
|
// TODO: Take a "number of improvements", use that to improve
|
||||||
|
// each successor N times
|
||||||
|
export function generateSuccessors(nImprovements: number): SuccessorOption[] {
|
||||||
|
let options = [];
|
||||||
|
while (options.length < 3) {
|
||||||
|
let option = generateSuccessor(nImprovements);
|
||||||
|
if (!isEligible(options, option)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
options.push(option);
|
||||||
|
}
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isEligible(existing: SuccessorOption[], added: SuccessorOption) {
|
||||||
|
for (let e of existing.values()) {
|
||||||
|
if (e.name == added.name || e.title == added.title) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function generateSuccessor(nImprovements: number): SuccessorOption {
|
||||||
|
let name = generateName();
|
||||||
|
let title = generateTitle();
|
||||||
|
let note = null;
|
||||||
|
let stats: Record<Stat, number> = {
|
||||||
|
"AGI": 10 + choose([1, 2]),
|
||||||
|
"INT": 10 + choose([1, 2]),
|
||||||
|
"CHA": 10 + choose([1, 2]),
|
||||||
|
"PSI": 10 + choose([1, 2]),
|
||||||
|
}
|
||||||
|
let talents: Record<Stat, number> = {
|
||||||
|
"AGI": 0,
|
||||||
|
"INT": 0,
|
||||||
|
"CHA": 0,
|
||||||
|
"PSI": 0,
|
||||||
|
}
|
||||||
|
|
||||||
|
let improvements = [
|
||||||
|
() => { stats[choose(ALL_STATS)] += choose([3, 4, 5, 6]); }, // avg 4.5
|
||||||
|
() => { talents[choose(ALL_STATS)] += 1; },
|
||||||
|
];
|
||||||
|
let nTotalImprovements = nImprovements + 5;
|
||||||
|
for (let i = 0; i < nTotalImprovements; i++) {
|
||||||
|
let improvement = improvements[Math.floor(Math.random() * improvements.length)];
|
||||||
|
improvement();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {name, title, note, stats, talents};
|
||||||
|
}
|
6
src/utils.ts
Normal file
6
src/utils.ts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
export function choose<T>(array: Array<T>): T {
|
||||||
|
if (array.length == 0) {
|
||||||
|
throw `array cannot have length 0 for choose`
|
||||||
|
}
|
||||||
|
return array[Math.floor(Math.random() * array.length)]
|
||||||
|
}
|
5
src/wishes.ts
Normal file
5
src/wishes.ts
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
import {Wish} from "./datatypes.ts";
|
||||||
|
|
||||||
|
export function generateWishes(): Wish[] {
|
||||||
|
return ["celebritySocialite", "nightswornAlchemist", "batFreak"];
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user