diff --git a/src/button.ts b/src/button.ts new file mode 100644 index 0000000..44e9c84 --- /dev/null +++ b/src/button.ts @@ -0,0 +1,41 @@ +import {DrawPile} from "./drawpile.ts"; +import {AlignX, AlignY, Point, Rect, Size} from "./engine/datatypes.ts"; +import {BG_INSET, FG_BOLD, FG_TEXT} from "./colors.ts"; +import {D} from "./engine/public.ts"; + +export function addButton( + drawpile: DrawPile, + label: string, + rect: Rect, + enabled: boolean, + cbClick: () => void, +) { + let padding = 2; + let topLeft = rect.top; + let topLeftPadded = topLeft.offset(new Point(padding, padding)); + let sizePadded = new Size(rect.size.w - padding * 2, rect.size.h - padding * 2); + let center = topLeft.offset(new Point(rect.size.w / 2, rect.size.h / 2)); + + drawpile.addClickable( + 0, + (hover) => { + let [bg, fg, fgLabel] = [BG_INSET, FG_TEXT, FG_BOLD]; + if (hover) { + [bg, fg, fgLabel] = [FG_BOLD, BG_INSET, BG_INSET]; + } + D.fillRect( + topLeftPadded.offset(new Point(-1, -1)), + sizePadded.add(new Size(2, 2)), + bg + ); + D.drawRect(topLeftPadded, sizePadded, fg); + D.drawText(label, center, fgLabel, { + alignX: AlignX.Center, + alignY: AlignY.Middle, + }) + }, + new Rect(topLeftPadded, sizePadded), + enabled, + cbClick + ); +} \ No newline at end of file diff --git a/src/datatypes.ts b/src/datatypes.ts index 7a0e5c5..f2a7cab 100644 --- a/src/datatypes.ts +++ b/src/datatypes.ts @@ -1,3 +1,21 @@ export type Stat = "AGI" | "INT" | "CHA" | "PSI"; export const ALL_STATS: Array = ["AGI", "INT", "CHA", "PSI"]; + +export type SkillGoverning = { + stats: Stat[], target: number, cost: number, note: string +}; +export type SkillProfile = { + name: string, + description: string, +} + +export type SkillData = { + governing: SkillGoverning, + profile: SkillProfile, + prereqs: Skill[] +} + +export type Skill = { + id: number +} diff --git a/src/engine/internal/font.ts b/src/engine/internal/font.ts index 190207b..bf1b0f5 100644 --- a/src/engine/internal/font.ts +++ b/src/engine/internal/font.ts @@ -98,13 +98,15 @@ class Font { return this.#glyphwise(text, forceWidth, () => {}); } - #glyphwise(text: string, forceWidth: number | undefined, callback: (x: number, y: number, char: string) => void): {w: number, h: number} { + #glyphwise(text: string, forceWidth: number | undefined, callback: (x: number, y: number, char: string) => void): Size { let cx = 0; let cy = 0; let cw = 0; let ch = 0; let wcx = forceWidth == undefined ? undefined : Math.floor(forceWidth / this.#px); + text = betterWordWrap(text, wcx); + for (let i = 0; i < text.length; i++) { let char = text[i] if (char == '\n') { @@ -126,8 +128,19 @@ class Font { } } - return { w: cw * this.#px, h: ch * this.#py }; + return new Size(cw * this.#px, ch * this.#py); } } + +// https://stackoverflow.com/users/1993501/edi9999 +function betterWordWrap(s: string, wcx?: number) { + if (wcx === undefined) { + return s; + } + return s.replace( + new RegExp(`(?![^\\n]{1,${wcx}}$)([^\\n]{1,${wcx}})\\s`, 'g'), '$1\n' + ); +} + export let mainFont = new Font(fontSheet, new Size(32, 8), new Size(8, 16)); \ No newline at end of file diff --git a/src/game.ts b/src/game.ts index 06bab3b..a6eebf5 100644 --- a/src/game.ts +++ b/src/game.ts @@ -4,7 +4,8 @@ import {IGame, Point, Size} from "./engine/datatypes.ts"; import {HuntMode} from "./huntmode.ts"; import {getPageLocation, Page, withCamera} from "./layout.ts"; import {getHud} from "./hud.ts"; -import {getHotbar} from "./hotbar.ts"; +import {getHotbar, Hotbar} from "./hotbar.ts"; +import {getSkillsModal, SkillsModal} from "./skillsmodal.ts"; class MenuCamera { // measured in whole screens @@ -31,6 +32,7 @@ export class Game implements IGame { camera: MenuCamera; page: Page; huntMode: HuntMode; + #bottomThing: SkillsModal | Hotbar | null; constructor() { this.camera = new MenuCamera({ @@ -40,6 +42,7 @@ export class Game implements IGame { this.page = "Gameplay"; this.huntMode = HuntMode.generate({depth: 1}); + this.#bottomThing = null; } update() { @@ -79,11 +82,13 @@ export class Game implements IGame { } updateGameplay() { + this.#chooseBottomThing(); + withCamera("Gameplay", () => { this.huntMode.update(); }); withCamera("HUD", () => { getHud().update() }) - withCamera("Hotbar", () => { getHotbar().update() }) + this.#bottomThing?.update(); } drawGameplay() { @@ -91,8 +96,26 @@ export class Game implements IGame { this.huntMode.draw(); }); withCamera("HUD", () => { getHud().draw() }) - withCamera("Hotbar", () => { getHotbar().draw() }) + this.#bottomThing?.draw() } + + #chooseBottomThing() { + // This is explicitly chosen because we would prefer that + // the same bottomThing be used in both draw and update, + // meaning that events after this in updateGameplay should not affect + // its value + + let skillsModal = getSkillsModal(); + if (skillsModal.isShown) { + this.#bottomThing = skillsModal; + return; + } + + // use the hotbar only as a matter of last resort + this.#bottomThing = getHotbar(); + } + + // withCamera("Hotbar", () => { getHotbar().draw() }) } diff --git a/src/hotbar.ts b/src/hotbar.ts index 5bcf148..385e4f4 100644 --- a/src/hotbar.ts +++ b/src/hotbar.ts @@ -1,7 +1,8 @@ -import {AlignX, AlignY, Point, Rect, Size} from "./engine/datatypes.ts"; -import {D} from "./engine/public.ts"; -import {BG_INSET, FG_BOLD, FG_TEXT} from "./colors.ts"; +import {Point, Rect, Size} from "./engine/datatypes.ts"; import {DrawPile} from "./drawpile.ts"; +import {withCamera} from "./layout.ts"; +import {getSkillsModal} from "./skillsmodal.ts"; +import {addButton} from "./button.ts"; type Button = { label: string, @@ -28,7 +29,9 @@ export class Hotbar { let buttons: Button[] = []; buttons.push({ label: "Skills", - cbClick: () => alert("beep"), + cbClick: () => { + getSkillsModal().setShown(true) + } }) /* buttons.push({ @@ -39,6 +42,10 @@ export class Hotbar { } update() { + withCamera("Hotbar", () => this.#update()) + } + + #update() { this.#drawpile.clear(); let buttons = this.#computeButtons(); @@ -46,35 +53,8 @@ export class Hotbar { let cellSize = this.#cellSize; let x = 0; - let padding = 2; for (let b of buttons.values()) { - let topLeft = new Point(x, 0); - let topLeftPadded = topLeft.offset(new Point(padding, padding)); - let sizePadded = new Size(cellSize.w - padding * 2, cellSize.h - padding * 2); - let center = topLeft.offset(new Point(cellSize.w / 2, cellSize.h / 2)); - - this.#drawpile.addClickable( - 0, - (hover) => { - let [bg, fg, fgLabel] = [BG_INSET, FG_TEXT, FG_BOLD]; - if (hover) { - [bg, fg, fgLabel] = [FG_TEXT, BG_INSET, BG_INSET]; - } - D.fillRect( - topLeftPadded.offset(new Point(-1, -1)), - sizePadded.add(new Size(2, 2)), - bg - ); - D.drawRect(topLeftPadded, sizePadded, fg); - D.drawText(b.label, center, fgLabel, { - alignX: AlignX.Center, - alignY: AlignY.Middle, - }) - }, - new Rect(topLeftPadded, sizePadded), - true, - b.cbClick, - ); + addButton(this.#drawpile, b.label, new Rect(new Point(x, 0), cellSize), true, b.cbClick); x += cellSize.w; } this.#drawpile.executeOnClick(); @@ -83,7 +63,7 @@ export class Hotbar { draw() { // D.fillRect(new Point(-4, -4), this.size.add(new Size(8, 8)), BG_INSET); - this.#drawpile.draw(); + withCamera("Hotbar", () => this.#drawpile.draw()); } } diff --git a/src/layout.ts b/src/layout.ts index 06c0591..af7bc1d 100644 --- a/src/layout.ts +++ b/src/layout.ts @@ -23,9 +23,6 @@ export function getLayoutRect( let {w: innerW, h: innerH} = size; let remainingSpaceX = marginalScreenW - innerW; let remainingSpaceY = marginalScreenH - innerH; - if (options?.alignY == AlignY.Bottom) { - console.log(`Remaining space y: ${remainingSpaceY}`) - } let alignXCoef = options?.alignX == AlignX.Left ? 0.0 : @@ -55,10 +52,11 @@ export function withCamera(part: UIPart, cb: () => void) { // specific export type Page = "Gameplay" | "Thralls"; -export type UIPart = "Hotbar" | "HUD" | "Gameplay" | "Thralls"; +export type UIPart = "BottomModal" | "Hotbar" | "HUD" | "Gameplay" | "Thralls"; export function getPartPage(part: UIPart): Page { switch (part) { + case "BottomModal": case "Hotbar": case "HUD": case "Gameplay": @@ -97,6 +95,11 @@ export function getPartLocation(part: UIPart): Rect { export function internalGetPartLayoutRect(part: UIPart) { switch (part) { + case "BottomModal": + return getLayoutRect(new Size(384, 128), { + alignX: AlignX.Center, + alignY: AlignY.Bottom, + }); case "Gameplay": case "Thralls": return getLayoutRect(new Size(384, 384)); @@ -108,7 +111,7 @@ export function internalGetPartLayoutRect(part: UIPart) { case "HUD": return getLayoutRect(getHud().size, { alignX: AlignX.Left, - alignY: AlignY.Middle + alignY: AlignY.Top }) } throw `not sure what layout rect to use ${part}` diff --git a/src/playerprogress.ts b/src/playerprogress.ts index 750ccf2..bf32824 100644 --- a/src/playerprogress.ts +++ b/src/playerprogress.ts @@ -1,8 +1,11 @@ -import {Stat} from "./datatypes.ts"; +import {Skill, Stat} from "./datatypes.ts"; +import {getSkills} from "./skills.ts"; export class PlayerProgress { #stats: Record #blood: number + #skillsLearned: number[] // use the raw ID representation for indexOf + #untrimmedSkillsAvailable: Skill[] constructor() { this.#stats = { @@ -10,14 +13,63 @@ export class PlayerProgress { INT: 10, CHA: 10, PSI: 10, - } + }; this.#blood = 0; + this.#skillsLearned = []; + this.#untrimmedSkillsAvailable = [] this.refill(); } refill() { this.#blood = 2000; + + let learnableSkills = []; // TODO: Also include costing info + for (let skill of getSkills().getAllAvailableSkills().values()) { + if (this.#canBeAvailable(skill)) { + learnableSkills.push(skill); + } + } + + this.#untrimmedSkillsAvailable = learnableSkills + } + + hasLearned(skill: Skill) { + return this.#skillsLearned.indexOf(skill.id) !== -1; + } + + learnSkill(skill: Skill) { + if (this.#skillsLearned.indexOf(skill.id) != -1) { + return + } + this.#skillsLearned.push(skill.id); + + // remove entries for that skill + let skills2 = []; + for (let entry of this.#untrimmedSkillsAvailable.values()) { + if (entry.id == skill.id) { continue; } + skills2.push(entry); + } + this.#untrimmedSkillsAvailable = skills2; + } + + #canBeAvailable(skill: Skill) { + // make sure we haven't learned this skill already + if (this.hasLearned(skill)) { + return false; + } + + let data = getSkills().get(skill); + + // make sure the prereqs are met + for (let prereq of data.prereqs.values()) { + if (!this.hasLearned(prereq)) { + return false + } + } + + // ok, we're good!! + return true; } add(stat: Stat, amount: number) { @@ -41,6 +93,11 @@ export class PlayerProgress { spendBlood(amt: number) { this.#blood -= amt; } + + getAvailableSkills(): Skill[] { + // TODO: Sort by cost, then by name, then trim down to first 8 + return this.#untrimmedSkillsAvailable + } } let active: PlayerProgress = new PlayerProgress(); diff --git a/src/skills.ts b/src/skills.ts new file mode 100644 index 0000000..95cc5af --- /dev/null +++ b/src/skills.ts @@ -0,0 +1,262 @@ +import {Skill, SkillData, SkillGoverning, Stat} from "./datatypes.ts"; + +class SkillsTable { + #skills: SkillData[] + + constructor() { + this.#skills = []; + } + + add(data: SkillData): Skill { + let id = this.#skills.length; + this.#skills.push(data); + return {id}; + } + + get(skill: Skill): SkillData { + return this.#skills[skill.id] + } + + getAllAvailableSkills(): Skill[] { + let skills = []; + for (let i = 0; i < this.#skills.length; i++) { + skills.push({id: i}); + } + return skills; + } +} + +type Difficulty = 0 | 1 | 2 | 3 +type GoverningTemplate = { + stats: Stat[], + note: string +} + +type Track = "bat" | "stealth" | "charm" | "stare" | "party" | "lore" +let templates: Record = { + bat: { stats: ["AGI", "AGI", "PSI"], note: "Cheaper with AGI and PSI." }, + stealth: { stats: ["AGI", "AGI", "INT"], note: "Cheaper with AGI and INT." }, + charm: { stats: ["CHA", "PSI", "PSI"], note: "Cheaper with CHA and PSI." }, + stare: { stats: ["PSI", "PSI"], note: "Cheaper with PSI." }, + party: { stats: ["CHA", "CHA", "PSI"], note: "Cheaper with CHA and PSI." }, + lore: { stats: ["INT", "INT", "CHA"], note: "Cheaper with INT and CHA." }, +} + +function governing(track: Track, difficulty: Difficulty): SkillGoverning { + let template = templates[track]; + let target: number + let cost: number + switch(difficulty) { + case 0: target = 30; cost = 50; break; + case 1: target = 100; cost = 100; break; + case 2: target = 150; cost = 250; break; + case 3: target = 250; cost = 500; break; + } + return { + stats: template.stats, + target: target, + cost: cost, + note: template.note, + } +} + +let table = new SkillsTable(); + +export let bat0 = table.add({ + governing: governing("bat", 0), + profile: { + name: "Screech", + description: "Energy fills your body. You can't help but let go. It just feels so good to let that sound rip through you." + }, + prereqs: [] +}); +export let bat1 = table.add({ + governing: governing("bat", 1), + profile: { + name: "Flap", + description: "Sailing on your cloak is pleasurable, but it's better still to shake your limbs and FIGHT the wind." + }, + prereqs: [bat0] +}); +export let bat2 = table.add({ + governing: governing("bat", 2), + profile: { + name: "Transform", + description: "Bare your fangs and let them out further than normal. Your nose wrinkles. You have a SNOUT??" + }, + prereqs: [bat1] +}); +export let bat3 = table.add({ + governing: governing("bat", 3), + profile: { + name: "Eat Bugs", + description: "This is the forbidden pleasure. It supersedes even blood. Go on -- have a bite -- CRUNCH!" + }, + prereqs: [bat2] +}); + +export let stealth0 = table.add({ + governing: governing("stealth", 0), + profile: { + name: "Be Quiet", + description: "There's a thing in the brain that _wants_ to be caught. Mortals have it, vampires don't." + }, + prereqs: [] +}); +export let stealth1 = table.add({ + governing: governing("stealth", 1), + profile: { + name: "Disguise", + description: "First impressions: what you want them to see, they'll see. Just meet their gaze and start talking.", + }, + prereqs: [stealth0] +}); +export let stealth2 = table.add({ + governing: governing("stealth", 2), + profile: { + name: "Sneak", + description: "Your unnatural pallor _should_ make you bright against the shadow. But it likes you, so you fade." + }, + prereqs: [stealth1] +}); +export let stealth3 = table.add({ + governing: governing("stealth", 3), + profile: { + name: "Turn Invisible", + description: "No one sees any more of you than you'd like. You're as ghostly as your own reflection.", + }, + prereqs: [stealth2] +}); + +export let charm0 = table.add({ + governing: governing("charm", 0), + profile: { + name: "Flatter", + description: "No matter how weird you're being, people like praise. Praise in your voice has an intoxicating quality.", + }, + prereqs: [] +}); +export let charm1 = table.add({ + governing: governing("charm", 1), + profile: { + name: "Befriend", + description: "Cute: they think they've met the real you. They're even thinking about you when you're not around." + }, + prereqs: [charm0] +}); +export let charm2 = table.add({ + governing: governing("charm", 2), + profile: { + name: "Seduce", + description: "Transfix them long and deep enough for them to realize how much they want you. \"No\" isn't \"no\" anymore.", + }, + prereqs: [charm1] +}); +export let charm3 = table.add({ + governing: governing("charm", 3), + profile: { + name: "Infatuate", + description: "They were into mortals once. Now they can't get off without the fangs. The eyes. The pale dead flesh." + }, + prereqs: [charm2] +}); +export let stare0 = table.add({ + governing: governing("stare", 0), + profile: { + name: "Dazzle", + description: "Your little light show can reduce anyone to a puddle of their own fluids. Stare and they give in instantly.", + }, + prereqs: [] +}); +export let stare1 = table.add({ + governing: governing("stare", 1), + profile: { + name: "Hypnotize", + description: "Say \"sleep\" and the mortal falls asleep. That is not a person: just a machine that acts when you require it." + }, + prereqs: [stare0] +}); +export let stare2 = table.add({ + governing: governing("stare", 2), + profile: { + name: "Enthrall", + description: "Everyone's mind has room for one master. Reach into the meek fractured exterior and mean for it to be you." + }, + prereqs: [stare1] +}); +export let stare3 = table.add({ + governing: governing("stare", 3), + profile: { + name: "Seal Memory", + description: "There was no existence before you and will be none after. Your mortals cannot imagine another existence." + }, + prereqs: [stare2] +}); +export let party0 = table.add({ + governing: governing("party", 0), + profile: { + name: "Chug", + description: "This undead body can hold SO MUCH whiskey. (BRAAAAP.) \"You, mortal -- fetch me another drink!\"" + }, + prereqs: [] +}); +export let party1 = table.add({ + governing: governing("party", 1), + profile: { + name: "Rave", + description: "You could jam glowsticks in your hair, but your eyes are a lot brighter. And they pulse with the music." + }, + prereqs: [party0] +}); +export let party2 = table.add({ + governing: governing("party", 2), + profile: { + name: "Peer Pressure", + description: "Partying: it gets you out of your head. Makes you do things you wouldn't normally do. Controls you." + }, + prereqs: [party1] +}); +export let party3 = table.add({ + governing: governing("party", 3), + profile: { + name: "Sleep It Off", + description: "Feels good. Never want it to end. But sober up. These feelings aren't for you. They're for your prey." + }, + prereqs: [party2] +}); +export let lore0 = table.add({ + governing: governing("lore", 0), + profile: { + name: "Respect Elders", + description: "You're told not to bother learning much. The test is not to believe that. Bad vampires _disappear_." + }, + prereqs: [] +}); +export let lore1 = table.add({ + governing: governing("lore", 1), + profile: { + name: "Brick by Brick", + description: "Vampire history is a mix of fact and advice. Certain tips -- \"live in a castle\" -- seem very concrete." + }, + prereqs: [lore0] +}); +export let lore2 = table.add({ + governing: governing("lore", 2), + profile: { + name: "Make Wine", + description: "Fruit bats grow the grapes. Insectivores fertilize the soil. What do vampire bats do? Is this a metaphor?" + }, + prereqs: [lore1] +}); +export let lore3 = table.add({ + governing: governing("lore", 3), + profile: { + name: "Third Clade", + description: "Mortals love the day. They hate the night. There's a night deeper than any night that cannot be discussed." + }, + prereqs: [lore2] +}); + +export function getSkills(): SkillsTable { + return table; +} \ No newline at end of file diff --git a/src/skillsmodal.ts b/src/skillsmodal.ts new file mode 100644 index 0000000..44a1293 --- /dev/null +++ b/src/skillsmodal.ts @@ -0,0 +1,146 @@ +import {getPartLocation, withCamera} from "./layout.ts"; +import {AlignX, Point, Rect, Size} from "./engine/datatypes.ts"; +import {DrawPile} from "./drawpile.ts"; +import {D} from "./engine/public.ts"; +import {BG_INSET, FG_BOLD, FG_TEXT} from "./colors.ts"; +import {addButton} from "./button.ts"; +import { + getSkills, +} from "./skills.ts"; +import {getPlayerProgress} from "./playerprogress.ts"; +import {Skill, SkillData} from "./datatypes.ts"; + +export class SkillsModal { + #drawpile: DrawPile; + #shown: boolean; + #skillSelection: Skill | null; + + constructor() { + this.#drawpile = new DrawPile(); + this.#shown = false; + this.#skillSelection = null; + } + + get #size(): Size { + // Instead of calculating this here, compute it from outside + // as it has to be the same for every bottom modal + return getPartLocation("BottomModal").size + } + + get isShown(): boolean { + return this.#shown; + } + + setShown(shown: boolean) { + this.#shown = shown + } + + update() { + withCamera("BottomModal", () => this.#update()) + } + + draw() { + withCamera("BottomModal", () => this.#draw()) + } + + #update() { + this.#drawpile.clear(); + let size = this.#size + this.#drawpile.add(0, () => { + D.fillRect(new Point(-4, -4), size.add(new Size(8, 8)), BG_INSET) + }) + + // draw skills + let availableSkills = getPlayerProgress().getAvailableSkills(); + + this.#fixSkillSelection(availableSkills); + + let y = 0; + for (let skill of availableSkills) { + let data = getSkills().get(skill); + let y_ = y; + let selected = this.#skillSelection?.id == skill.id; + let skillRect = new Rect(new Point(0, y_), new Size(160 + 4, 16)); + let enabled = true; + + this.#drawpile.addClickable( + 0, + (hover) => { + // two column layout + let [bg, fg] = [BG_INSET, FG_TEXT]; + if (selected || hover) { + [bg, fg] = [FG_BOLD, BG_INSET]; + } + D.fillRect(skillRect.top, skillRect.size, bg); + D.drawText(data.profile.name, new Point(4, y_), fg); + D.drawText("100", new Point(160 - 4, y_), fg, {alignX: AlignX.Right}); + }, + skillRect, + enabled, + () => { + this.#skillSelection = skill; + } + ) + y += 16; + } + + // add skill description + let selection = this.#skillSelection; + if (selection != null) { + let data = getSkills().get(selection); + let size = this.#size; + let remainingWidth = size.w - 160; + + this.#drawpile.add(0, () => { + D.fillRect(new Point(160, 0), new Size(remainingWidth, 96), FG_BOLD) + D.drawText(createFullDescription(data), new Point(164, 0), BG_INSET, {forceWidth: remainingWidth - 8}); + }); + + // add learn button + let drawButtonRect = new Rect(new Point(160, 96), new Size(remainingWidth, 32)) + let caption = `Learn ${data.profile.name}` + addButton(this.#drawpile, caption, drawButtonRect, true, () => { + getPlayerProgress().learnSkill(selection); + }) + } + + + // add close button + let closeRect = new Rect(new Point(0, 96), new Size(160, 32)) + addButton(this.#drawpile, "Back", closeRect, true, () => { + this.setShown(false); + }) + this.#drawpile.executeOnClick(); + } + + #fixSkillSelection(skills: Skill[]) { + // if we have selected a skill that is really available, + // that's fine + for (let s of skills.values()) { + if (s.id == this.#skillSelection?.id) { + return; + } + } + + // select the first skill if one exists + if (skills.length == 0) { + this.#skillSelection = null; + } else { + this.#skillSelection = skills[0]; + } + } + + #draw() { + this.#drawpile.draw(); + } +} + +let active = new SkillsModal(); + +export function getSkillsModal(): SkillsModal { + return active; +} + +function createFullDescription(data: SkillData) { + return data.profile.description + "\n\n" + data.governing.note +} \ No newline at end of file