From 047248adb635cef28cb4efdc33813ce56a14d8a6 Mon Sep 17 00:00:00 2001 From: Nyeogmi Date: Sun, 2 Feb 2025 22:26:56 -0800 Subject: [PATCH] Make ladders work --- src/art/pickups/ladder.png | Bin 0 -> 179 bytes src/art/tilesets/drips.png | Bin 240 -> 216 bytes src/game.ts | 8 +- src/hud.ts | 3 +- src/huntmode.ts | 165 ++++++++++++------------------------- src/mapgen.ts | 97 ++++++++++++++++++++++ src/playerprogress.ts | 6 +- src/skills.ts | 2 +- src/sleepmodal.ts | 5 +- src/sprites.ts | 8 +- 10 files changed, 172 insertions(+), 122 deletions(-) create mode 100644 src/art/pickups/ladder.png create mode 100644 src/mapgen.ts diff --git a/src/art/pickups/ladder.png b/src/art/pickups/ladder.png new file mode 100644 index 0000000000000000000000000000000000000000..04007105fc22fe9fe1a72b4e9cb97415b67909e4 GIT binary patch literal 179 zcmeAS@N?(olHy`uVBq!ia0vp^3P3Et!3HGD8EPYe6k~CayA#8@b22Z1oLo;A#}JRs z`Zz$d}F an1NwmfMQhW^d0oVbM7k>~41^@s6rssJn00001b5ch_0Itp)=>Px#l1W5CR9J=Wm$40i zAP_~rj6&#b*ulyvtSp_4iXCj|iK}RZl}5XVk=Kv_$-(6g{_mXiyFl``c4T}*2A3pf zy`1$}d^`bFP}+bR8;!>Q2r((d^lJWY@29EdQilA4YvA`fz(VA0rO$wJRsh)wq!7C_ zA|)Aj)$z3chVlR|^MtBF%!!4VG!8v3hfymk@qXX{j5t~H9LbO800000NkvXXu0mjf D(T`h& delta 225 zcmV<703QF?0q_Bk7k?NC1^@s6O0@e#00001b5ch_0Itp)=>Px#s!2paR9J=Wn5_+f zFc3yR7^SJlbO#huppeXlVh2n;VHHp$Ku9ozcM11Q<#)fm_KFzNDv)`Z3R1r$#U&|* zq!<#Wj|ad6dmB)v<2cT5!N(veZUOuqFfS8r0t?*&Cquqw7+SN2F@LW6$WvC{$eI=B zv7gw{`KGQYGx$sJ>ik`0SgDp*vEme70W=-}+5j380U#GL_!#WF7TpS=6+q7a9dsov b@q57y5t&?3H7orN00000NkvXXu0mjf8YNjY diff --git a/src/game.ts b/src/game.ts index 4d38f76..686c93d 100644 --- a/src/game.ts +++ b/src/game.ts @@ -1,7 +1,7 @@ import {BG_OUTER} from "./colors.ts"; import {D, I} from "./engine/public.ts"; import {IGame, Point, Size} from "./engine/datatypes.ts"; -import {HuntMode} from "./huntmode.ts"; +import {getHuntMode} from "./huntmode.ts"; import {getPageLocation, Page, withCamera} from "./layout.ts"; import {getHud} from "./hud.ts"; import {getHotbar, Hotbar} from "./hotbar.ts"; @@ -32,7 +32,6 @@ class MenuCamera { export class Game implements IGame { camera: MenuCamera; page: Page; - huntMode: HuntMode; #bottomThing: SkillsModal | SleepModal | Hotbar | null; constructor() { @@ -42,7 +41,6 @@ export class Game implements IGame { }); this.page = "Gameplay"; - this.huntMode = HuntMode.generate({depth: 1}); this.#bottomThing = null; } @@ -86,7 +84,7 @@ export class Game implements IGame { this.#chooseBottomThing(); withCamera("Gameplay", () => { - this.huntMode.update(); + getHuntMode().update(); }); withCamera("HUD", () => { getHud().update() }) this.#bottomThing?.update(); @@ -94,7 +92,7 @@ export class Game implements IGame { drawGameplay() { withCamera("Gameplay", () => { - this.huntMode.draw(); + getHuntMode().draw(); }); withCamera("HUD", () => { getHud().draw() }) this.#bottomThing?.draw() diff --git a/src/hud.ts b/src/hud.ts index fa156c3..086aa61 100644 --- a/src/hud.ts +++ b/src/hud.ts @@ -3,6 +3,7 @@ import {Point, Size} from "./engine/datatypes.ts"; import {FG_BOLD, FG_TEXT} from "./colors.ts"; import {ALL_STATS} from "./datatypes.ts"; import {getPlayerProgress} from "./playerprogress.ts"; +import {getHuntMode} from "./huntmode.ts"; export class Hud { get size(): Size { @@ -14,7 +15,7 @@ export class Hud { draw() { // 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("Level 1", new Point(0, 16), FG_TEXT) + D.drawText(`Level ${getHuntMode().getDepth()}`, new Point(0, 16), FG_TEXT) let y = 48; let prog = getPlayerProgress(); diff --git a/src/huntmode.ts b/src/huntmode.ts index e60b58e..c36e636 100644 --- a/src/huntmode.ts +++ b/src/huntmode.ts @@ -1,11 +1,11 @@ import {Grid, Point, Rect, Size} from "./engine/datatypes.ts"; -import {ConceptualCell, maps} from "./maps.ts"; import {ALL_STATS, Resource, Stat} from "./datatypes.ts"; import {DrawPile} from "./drawpile.ts"; import {D} from "./engine/public.ts"; -import {sprDrips, sprRaccoonWalking, sprResourcePickup, sprStatPickup} from "./sprites.ts"; +import {sprDrips, sprLadder, sprRaccoonWalking, sprResourcePickup, sprStatPickup} from "./sprites.ts"; import {BG_INSET, FG_TEXT} from "./colors.ts"; import {getPlayerProgress} from "./playerprogress.ts"; +import {generate} from "./mapgen.ts"; export type MapCellContent = {type: "statPickup", stat: Stat} | @@ -21,122 +21,53 @@ export type MapCell = { nextMoveAccessible: boolean, } -export class HuntMode { - depth: number - cells: Grid +export type LoadedMap = { + cells: Grid, player: Point +} +export class HuntMode { + map: LoadedMap drawpile: DrawPile frame: number + depth: number - constructor({depth, cells, player}: {depth: number, cells: Grid, player: Point }) { - this.depth = depth; - this.cells = cells; - this.player = player; + constructor() { + this.map = null!; // initialized in replaceMap this.drawpile = new DrawPile(); this.frame = 0; + this.depth = 1; + this.replaceMap(); } - // == map generator == - static generate({depth}: {depth: number}) { - let mapNames: Array = Object.keys(maps); - let mapName = mapNames[Math.floor(Math.random() * mapNames.length)]; - let map = maps[mapName]; + replaceMap(deeper?: boolean) { + this.map = generate(); + this.#updateVisibilityAndPossibleMoves(); - let baseCells = map.map((ccell, _xy) => { - return this.#generateCell(ccell); - }) - - let cells = new Grid( - new Size(baseCells.size.w + 2, baseCells.size.h + 2), (xy) => { - let offset = xy.offset(new Point(-1, -1)); - if (offset.x == -1 || offset.y == -1 || offset.x == baseCells.size.w || offset.y == baseCells.size.h) { - return this.#generateBoundaryCell(); - } - return baseCells.get(offset) - } - ) - - let validSpawns = []; - for (let x = 0; x < cells.size.w; x++) { - for (let y = 0; y < cells.size.h; y++) { - let position = new Point(x, y); - if (cells.get(position).isValidSpawn) { - validSpawns.push(position); - } - } - } - let player = choose(validSpawns); - cells.get(player).content = {type: "empty"}; - - if (Math.random() < 0.75) { - while (true) { - let x = Math.floor(Math.random() * cells.size.w); - let y = Math.floor(Math.random() * cells.size.h); - let xy = new Point(x, y); - - let item = cells.get(new Point(x, y)); - if (player.equals(xy)) { - continue; - } - if (item.content.type == "block") { - continue; - } - item.content = {type: "stairs"} - break; - } - } - - let hm = new HuntMode({depth, cells, player}) - hm.#updateVisibilityAndPossibleMoves(); - return hm; - } - - static #generateCell(conceptual: ConceptualCell): MapCell { - switch (conceptual) { - case "X": - return { content: {type: "block"}, revealed: false, isValidSpawn: false, nextMoveAccessible: false}; - case " ": - return { content: HuntMode.#generateContent(), revealed: false, isValidSpawn: false, nextMoveAccessible: false }; - case ".": - return { content: HuntMode.#generateContent(), revealed: false, isValidSpawn: true, nextMoveAccessible: false }; + if (deeper) { + this.depth += 1; } } - static #generateBoundaryCell() { - return this.#generateCell("X"); - } - - static #generateContent(): MapCellContent { - // stat pickup - let gsp = (): MapCellContent => { - return {type: "statPickup", stat: choose(ALL_STATS)} - }; - let exp = (): MapCellContent => { - return {type: "resourcePickup", resource: "EXP"} - } - // TODO: Other objects? - return choose([ - gsp, gsp, gsp, gsp, - exp, - ])(); + getDepth() { + return this.depth; } // == update logic == #updateVisibilityAndPossibleMoves() { - for (let x = 0; x < this.cells.size.w; x++) { - for (let y = 0; y < this.cells.size.h; y++) { + for (let x = 0; x < this.map.cells.size.w; x++) { + for (let y = 0; y < this.map.cells.size.h; y++) { let position = new Point(x, y); - let data = this.cells.get(position); + let data = this.map.cells.get(position); data.nextMoveAccessible = false; if ( - Math.abs(x - this.player.x) <= 1 && - Math.abs(y - this.player.y) <= 1 + Math.abs(x - this.map.player.x) <= 1 && + Math.abs(y - this.map.player.y) <= 1 ) { data.revealed = true; - if (!this.player.equals(position)) { + if (!this.map.player.equals(position)) { data.nextMoveAccessible = true; } } @@ -146,7 +77,12 @@ export class HuntMode { } #collectResources() { - let present = this.cells.get(this.player); + let present = this.map.cells.get(this.map.player); + + if (present.content.type == "stairs") { + getPlayerProgress().addBlood(1000); + this.replaceMap(true); + } if (present.content.type == "statPickup") { let stat = present.content.stat; @@ -170,10 +106,13 @@ export class HuntMode { } #computeCostToMoveTo(mapPosition: Point): number | null { - let present = this.cells.get(mapPosition); + let present = this.map.cells.get(mapPosition); if (present.content.type == "statPickup" || present.content.type == "resourcePickup") { return 100; } + if (present.content.type == "stairs") { + return 0; + } if (present.content.type == "empty") { return 10; } @@ -181,7 +120,7 @@ export class HuntMode { } movePlayerTo(newPosition: Point) { - this.player = newPosition; + this.map.player = newPosition; this.#updateVisibilityAndPossibleMoves(); this.#collectResources(); } @@ -192,18 +131,18 @@ export class HuntMode { this.drawpile.clear(); let globalOffset = - new Point(this.player.x * MAP_CELL_ONSCREEN_SIZE.w, this.player.y * MAP_CELL_ONSCREEN_SIZE.h).offset( + new Point(this.map.player.x * MAP_CELL_ONSCREEN_SIZE.w, this.map.player.y * MAP_CELL_ONSCREEN_SIZE.h).offset( new Point(-192, -192) ) - let map = this.cells; + let map = this.map.cells; for (let y = 0; y < map.size.h; y += 1) { for (let x = 0; x < map.size.w; x += 1) { let cellOffset = new Point(x * MAP_CELL_ONSCREEN_SIZE.w, y * MAP_CELL_ONSCREEN_SIZE.h).offset(globalOffset.negate()); - let cellData = this.cells.get(new Point(x, y)) + let cellData = this.map.cells.get(new Point(x, y)) let belowIsBlock = true; if (y < map.size.h - 1) { - let below = this.cells.get(new Point(x, y + 1)); + let below = this.map.cells.get(new Point(x, y + 1)); belowIsBlock = !below.revealed || below.content.type == "block"; } @@ -241,7 +180,7 @@ export class HuntMode { if (cellData.content.type == "block") { if (!belowIsBlock) { this.drawpile.add(inAir, () => { - D.drawSprite(sprDrips, cellOffset.offset(new Point(0, -cellSize.h)), 1, {xScale: 3, yScale: 3}) + D.drawSprite(sprDrips, cellOffset.offset(new Point(0, -cellSize.h / 2)), 1, {xScale: 3, yScale: 3}) }) } return; @@ -252,6 +191,11 @@ export class HuntMode { this.drawpile.addClickable(onFloor, (hover: boolean) => { D.fillRect(cellTopLeft, cellSize, hover ? FG_TEXT : BG_INSET) + + if (cellData.content.type == "stairs") { + // draw ladder if applicable + D.drawSprite(sprLadder, cellTopLeft, 0, {xScale: 3, yScale: 3}); + } }, new Rect(cellTopLeft, cellSize), cellData.nextMoveAccessible && cost != null && cost <= getPlayerProgress().getBlood(), @@ -263,10 +207,11 @@ export class HuntMode { } ); + if (belowIsBlock) { // draw the underhang this.drawpile.add(onFloor, () => { - D.drawSprite(sprDrips, cellOffset.offset(new Point(0, cellSize.h/2)), 0, {xScale: 3, yScale: 3}) + D.drawSprite(sprDrips, cellOffset.offset(new Point(0, cellSize.h / 2)), 0, {xScale: 3, yScale: 3}) }) } @@ -304,10 +249,10 @@ export class HuntMode { #drawPlayer(globalOffset: Point) { let cellOffset = new Point( - this.player.x * MAP_CELL_ONSCREEN_SIZE.w, - this.player.y * MAP_CELL_ONSCREEN_SIZE.h + this.map.player.x * MAP_CELL_ONSCREEN_SIZE.w, + this.map.player.y * MAP_CELL_ONSCREEN_SIZE.h ).offset(globalOffset.negate()) - this.drawpile.add(this.player.y, () => { + this.drawpile.add(this.map.player.y, () => { D.drawSprite( sprRaccoonWalking, cellOffset.offset(new Point(0, 22)), @@ -322,9 +267,7 @@ export class HuntMode { const MAP_CELL_ONSCREEN_SIZE: Size = new Size(96, 48) -function choose(array: Array): T { - if (array.length == 0) { - throw `array cannot have length 0 for choose` - } - return array[Math.floor(Math.random() * array.length)] +let active = new HuntMode(); +export function getHuntMode() { + return active; } diff --git a/src/mapgen.ts b/src/mapgen.ts new file mode 100644 index 0000000..91b1c0b --- /dev/null +++ b/src/mapgen.ts @@ -0,0 +1,97 @@ +import {ConceptualCell, maps} from "./maps.ts"; +import {Grid, Point, Size} from "./engine/datatypes.ts"; +import {ALL_STATS} from "./datatypes.ts"; +import {LoadedMap, MapCell, MapCellContent} from "./huntmode.ts"; + +export function generate(): LoadedMap { + let mapNames: Array = Object.keys(maps); + let mapName = mapNames[Math.floor(Math.random() * mapNames.length)]; + let map = maps[mapName]; + + let baseCells = map.map((ccell, _xy) => { + return generateCell(ccell); + }) + + let cells = new Grid( + new Size(baseCells.size.w + 2, baseCells.size.h + 2), (xy) => { + let offset = xy.offset(new Point(-1, -1)); + if (offset.x == -1 || offset.y == -1 || offset.x == baseCells.size.w || offset.y == baseCells.size.h) { + return generateBoundaryCell(); + } + return baseCells.get(offset) + } + ) + + let validSpawns = []; + for (let x = 0; x < cells.size.w; x++) { + for (let y = 0; y < cells.size.h; y++) { + let position = new Point(x, y); + if (cells.get(position).isValidSpawn) { + validSpawns.push(position); + } + } + } + let player = choose(validSpawns); + cells.get(player).content = {type: "empty"}; + + let nStairs = choose([1, 1, 1, 0]); + for (let i = 0; i < nStairs; i++) { + while (true) { + let x = Math.floor(Math.random() * cells.size.w); + let y = Math.floor(Math.random() * cells.size.h); + let xy = new Point(x, y); + + let item = cells.get(new Point(x, y)); + if (player.equals(xy)) { + continue; + } + if (item.content.type == "block" || item.content.type == "stairs") { + continue; + } + item.content = {type: "stairs"} + break; + } + } + + return { + cells, + player, + } +} + +function generateCell(conceptual: ConceptualCell): MapCell { + switch (conceptual) { + case "X": + return { content: {type: "block"}, revealed: false, isValidSpawn: false, nextMoveAccessible: false}; + case " ": + return { content: generateContent(), revealed: false, isValidSpawn: false, nextMoveAccessible: false }; + case ".": + return { content: generateContent(), revealed: false, isValidSpawn: true, nextMoveAccessible: false }; + } +} + +function generateBoundaryCell() { + return generateCell("X"); +} + +function generateContent(): MapCellContent { + // stat pickup + let gsp = (): MapCellContent => { + return {type: "statPickup", stat: choose(ALL_STATS)} + }; + let exp = (): MapCellContent => { + return {type: "resourcePickup", resource: "EXP"} + } + // TODO: Other objects? + return choose([ + gsp, gsp, gsp, gsp, + exp, + ])(); +} + +function choose(array: Array): T { + if (array.length == 0) { + throw `array cannot have length 0 for choose` + } + return array[Math.floor(Math.random() * array.length)] +} diff --git a/src/playerprogress.ts b/src/playerprogress.ts index 66ebad7..a17d9f3 100644 --- a/src/playerprogress.ts +++ b/src/playerprogress.ts @@ -107,6 +107,11 @@ export class PlayerProgress { return Math.floor(Math.max(this.#blood, 0)); } + addBlood(amt: number) { + this.#blood += amt; + this.#blood = Math.min(this.#blood, 5000) + } + spendBlood(amt: number) { this.#blood -= amt; } @@ -127,7 +132,6 @@ export class PlayerProgress { }); return skillsAvailable.slice(0, 6) } - } let active: PlayerProgress = new PlayerProgress(); diff --git a/src/skills.ts b/src/skills.ts index 3622eb3..ec8427e 100644 --- a/src/skills.ts +++ b/src/skills.ts @@ -79,7 +79,7 @@ function governing(track: Track, difficulty: Difficulty): SkillGoverning { let cost: number switch(difficulty) { case 0: underTarget = 5; target = 15; cost = 50; break; - case 1: underTarget = 50; target = 100; cost = 100; break; + case 1: underTarget = 15; target = 40; cost = 100; break; case 2: underTarget = 100; target = 150; cost = 250; break; case 3: underTarget = 175; target = 250; cost = 500; break; } diff --git a/src/sleepmodal.ts b/src/sleepmodal.ts index 8f37172..589168e 100644 --- a/src/sleepmodal.ts +++ b/src/sleepmodal.ts @@ -6,6 +6,7 @@ import {D} from "./engine/public.ts"; import {BG_INSET} from "./colors.ts"; import {getSkillsModal} from "./skillsmodal.ts"; import {getPlayerProgress} from "./playerprogress.ts"; +import {getHuntMode} from "./huntmode.ts"; export class SleepModal { #drawpile: DrawPile; @@ -60,10 +61,10 @@ export class SleepModal { let remainingWidth = size.w - 160; let nextRect = new Rect(new Point(160, 96), new Size(remainingWidth, 32)); - addButton(this.#drawpile, "Sleep All Day", nextRect, true, () => { + addButton(this.#drawpile, "Sleep (Next Day)", nextRect, true, () => { getPlayerProgress().refill(); + getHuntMode().replaceMap(); getSleepModal().setShown(false); - // TODO: Advance huntmode }); this.#drawpile.executeOnClick(); diff --git a/src/sprites.ts b/src/sprites.ts index 8a0273c..a784f96 100644 --- a/src/sprites.ts +++ b/src/sprites.ts @@ -9,6 +9,7 @@ import imgRaccoon from "./art/characters/raccoon.png"; import imgRaccoonWalking from "./art/characters/raccoon_walking.png"; import imgResourcePickup from "./art/pickups/resources.png"; import imgStatPickup from "./art/pickups/stats.png"; +import imgLadder from "./art/pickups/ladder.png"; import imgDrips from "./art/tilesets/drips.png"; import {Point, Size} from "./engine/datatypes.ts"; @@ -40,6 +41,11 @@ export let sprStatPickup = new Sprite( ); export let sprDrips = new Sprite( - imgDrips, new Size(32, 24), new Point(16, 0), + imgDrips, new Size(32, 16), new Point(16, 0), new Size(2, 1), 2 ); + +export let sprLadder = new Sprite( + imgLadder, new Size(32, 24), new Point(0, 0), + new Size(1, 1), 1 +);