From 46a249352ddaed3d560868a434b62773b37f14b6 Mon Sep 17 00:00:00 2001 From: Nyeogmi Date: Sat, 1 Feb 2025 22:04:40 -0800 Subject: [PATCH] OK, fine, draw the map --- src/art/pickups/stats.png | Bin 0 -> 643 bytes src/colors.ts | 3 +- src/drawpile.ts | 21 +++++ src/engine/datatypes.ts | 53 ++++++++++--- src/engine/internal/sprite.ts | 1 - src/game.ts | 145 ++++++++++++++++++++++++++++------ src/maps.ts | 47 ++++------- src/sprites.ts | 28 ++++++- 8 files changed, 227 insertions(+), 71 deletions(-) create mode 100644 src/art/pickups/stats.png create mode 100644 src/drawpile.ts diff --git a/src/art/pickups/stats.png b/src/art/pickups/stats.png new file mode 100644 index 0000000000000000000000000000000000000000..dd33da510e4a0b7d1d9aa98222787d3f24eaf7b9 GIT binary patch literal 643 zcmV-}0(||6P)Px%JxN4CRCt{2n!8E^K@^71N=T6+q!1xS1UpGB2oh}a1a>JbwaFXUrSlEam?A|A zJD(s;5Fub0WE&fCEkcT5ks=7O2&1x@nVrj--EniiCS;jg=Ktr+nb`mk1VIo4K@bG7 zT?APVz=Y$=#})5+IB1~%o}04{w=W%TpE};ov(~_t?b60%M3d@9xuemmlOHS+J`R=< zFvMs0Y69t?c!I~_86XaZAyma@*qQ)SdSAnvUVRc%WRqS*ck^0o9c+s+(kjouo#J-G%Yw zxk?7JU^tJo#K*u!eAPA*4UiSONncuRYNB%?fu&#dYpQ>7x4CSUiSaDj!N5Jp8|+qn4pB&+uuler)j>7)L@L zMh4(T0avTv)C9Az53^^yNpmHvDykt}x2)F@fT3f8S=e8HB8m?nf3qht2j<&AoozrT zT3AievRzt5lWJ;vzsv+ocmO={vHlZqw|qx8Bo_b50F2!e{z`B$$)dh=@Kt<*AP9mW d2!bHCk#8oW8cv?h9x4C;002ovPDHLkV1g|J8G--+ literal 0 HcmV?d00001 diff --git a/src/colors.ts b/src/colors.ts index 27452e8..168f60c 100644 --- a/src/colors.ts +++ b/src/colors.ts @@ -1,5 +1,6 @@ import {Color} from "./engine/datatypes.ts"; -export const BG_OUTER = Color.parseHexCode("#200500"); +export const BG_OUTER = Color.parseHexCode("#143464"); +export const BG_INSET = Color.parseHexCode("#242234"); export const FG_TEXT = Color.parseHexCode("#c0c0c0") export const FG_BOLD = Color.parseHexCode("#ffffff") diff --git a/src/drawpile.ts b/src/drawpile.ts new file mode 100644 index 0000000..08ceccf --- /dev/null +++ b/src/drawpile.ts @@ -0,0 +1,21 @@ +export class DrawPile { + readonly #draws: {depth: number, op: () => void}[] + + constructor() { + this.#draws = [] + } + + add(depth: number, op: () => void) { + this.#draws.push({depth, op}); + } + + execute() { + let draws = [...this.#draws]; + draws.sort( + (d0, d1) => d0.depth - d1.depth + ); + for (let d of draws.values()) { + d.op(); + } + } +} \ No newline at end of file diff --git a/src/engine/datatypes.ts b/src/engine/datatypes.ts index d138db5..b562c9a 100644 --- a/src/engine/datatypes.ts +++ b/src/engine/datatypes.ts @@ -66,6 +66,10 @@ export class Point { negate() { return new Point(-this.x, -this.y); } + + equals(other: Point): boolean { + return this.x == other.x && this.y == other.y; + } } export class Size { @@ -79,17 +83,19 @@ export class Size { } export class Grid { - data: T[][]; + readonly size: Size; + #data: T[][]; - constructor({size, cbDefault}: {size: Size, cbDefault: (xy: Point) => T}) { - this.data = []; + constructor(size: Size, cbDefault: (xy: Point) => T) { + this.size = size; + this.#data = []; for (let y = 0; y < size.h; y++) { let row = []; for (let x = 0; x < size.w; x++) { row.push(cbDefault(new Point(x, y))) } - this.data.push(row); + this.#data.push(row); } } @@ -105,12 +111,12 @@ export class Grid { w = w1; } - return new Grid({ - size: new Size(w, h), - cbDefault: (xy) => { + return new Grid( + new Size(w, h), + (xy) => { return ary[xy.y].charAt(xy.x); } - }) + ) } static createGridFromJaggedArray(ary: Array>): Grid { @@ -125,12 +131,35 @@ export class Grid { w = w1; } - return new Grid({ - size: new Size(w, h), - cbDefault: (xy) => { + return new Grid( + new Size(w, h), + (xy) => { return ary[xy.y][xy.x]; } - }) + ) + } + + map(cbCell: (content: T, position: Point) => T2) { + return new Grid(this.size, (xy) => cbCell(this.get(xy), xy)); + } + + #checkPosition(position: Point) { + if ( + (position.x < 0 || position.x >= this.size.w || Math.floor(position.x) != position.x) || + (position.y < 0 || position.y >= this.size.h || Math.floor(position.y) != position.y) + ) { + throw `invalid position for ${this.size}: ${position}` + } + } + + get(position: Point): T { + this.#checkPosition(position); + return this.#data[position.y][position.x]; + } + + set(position: Point, value: T) { + this.#checkPosition(position); + this.#data[position.y][position.x] = value; } } diff --git a/src/engine/internal/sprite.ts b/src/engine/internal/sprite.ts index 1f72672..d492150 100644 --- a/src/engine/internal/sprite.ts +++ b/src/engine/internal/sprite.ts @@ -41,7 +41,6 @@ export class Sprite { let srcCy = Math.floor(ix / this.cellsPerSheet.w); let srcPx = srcCx * this.pixelsPerSubimage.w; let srcPy = srcCy * this.pixelsPerSubimage.h; - console.log(`src px and py ${srcPx} ${srcPy}`) ctx.drawImage(me, srcPx, srcPy, this.pixelsPerSubimage.w, this.pixelsPerSubimage.h, 0, 0, this.pixelsPerSubimage.w, this.pixelsPerSubimage.h); } } diff --git a/src/game.ts b/src/game.ts index ed0e850..cf99dfc 100644 --- a/src/game.ts +++ b/src/game.ts @@ -1,13 +1,15 @@ import {desiredHeight, desiredWidth, getScreen} from "./engine/internal/screen.ts"; -import {BG_OUTER, FG_TEXT} from "./colors.ts"; +import {BG_INSET, BG_OUTER, FG_TEXT} from "./colors.ts"; import {checkGrid, ConceptualCell, maps, mapSzX, mapSzY} from "./maps.ts"; import {D, I} from "./engine/public.ts"; -import {IGame, Point, Size} from "./engine/datatypes.ts"; +import {Grid, IGame, Point, Size} from "./engine/datatypes.ts"; +import {sprRaccoonWalking, sprStatPickup} from "./sprites.ts"; +import {DrawPile} from "./drawpile.ts"; class MenuCamera { // measured in whole screens position: Point; - target: Point + target: Point; constructor({position, target}: {position: Point, target: Point}) { this.position = position; @@ -43,6 +45,7 @@ export class Game implements IGame { camera: MenuCamera; state: GameState; huntMode: HuntMode; + frame: number; constructor() { this.camera = new MenuCamera({ @@ -52,9 +55,11 @@ export class Game implements IGame { this.state = "Gameplay"; this.huntMode = HuntMode.generate({depth: 1}); + this.frame = 0; } update() { + this.frame += 1; if (I.isKeyPressed("w")) { this.state = "Gameplay" } @@ -125,8 +130,89 @@ export class Game implements IGame { drawGameplay() { let region = this.getPaneRegionForGameState("Gameplay") // TODO: Draw + let oldCamera = D.camera; + D.camera = D.camera.offset(region.small.position.negate()); - D.drawText("hello", region.small.position, FG_TEXT); + let drawpile = new DrawPile(); + + let globalOffset = + new Point(this.huntMode.player.x * 32, this.huntMode.player.y * 32).offset( + new Point(-192, -128) + ) + + + for (let y = 0; y < mapSzY; y += 1) { + for (let x = 0; x < mapSzX; x += 1) { + let cellOffset = new Point(x * 32, y * 32).offset(globalOffset.negate()); + let cellData = this.huntMode.cells.get(new Point(x, y)) + + this.#drawMapCell(drawpile, cellOffset, new Point(x, y), cellData); + } + } + this.#drawPlayer(drawpile, globalOffset); + + drawpile.execute(); + + D.drawText("hello", new Point(0, 0), FG_TEXT); + + D.camera = oldCamera; + } + + #drawMapCell( + drawpile: DrawPile, + cellOffset: Point, + mapPosition: Point, + cellData: MapCell, + ) { + const OFFSET_FLOOR = -256; + const OFFSET_AIR = 0; + const depth = cellOffset.y; + const onFloor = OFFSET_FLOOR + depth; + const inAir = OFFSET_AIR + depth; + + if (cellData.content.type == "block") { + return; + } + /* + if (!cellData.revealed) { + return; + } + */ + + // draw inset zone + drawpile.add(onFloor, () => + D.fillRect(cellOffset.offset(new Size(-16, -26)), new Size(32, 32), BG_INSET) + ); + + /* + if (!cellData.revealed) { + // TODO: draw some kind of question mark + D.drawText("?", cellOffset.offset(new Point(12, 8)), FG_TEXT); + return + } + */ + if (cellData.content.type == "statPickup") { + let content = cellData.content; + let extraXOffset = 0; // Math.cos(this.frame / 80 + mapPosition.x + mapPosition.y) * 1; + let extraYOffset = 0; // Math.sin(this.frame / 50 + mapPosition.x * 2+ mapPosition.y * 0.75) * 6 - 3; + drawpile.add(inAir, () => { + D.drawSprite( + sprStatPickup, + cellOffset.offset(new Point(extraXOffset, extraYOffset)), + ALL_STATS.indexOf(content.stat) + ) + }); + } + } + + #drawPlayer(drawpile: DrawPile, globalOffset: Point) { + let cellOffset = new Point(this.huntMode.player.x * 32, this.huntMode.player.y * 32).offset(globalOffset.negate()) + drawpile.add(this.huntMode.player.y, () => { + D.drawSprite( + sprRaccoonWalking, + cellOffset + ) + }); } } @@ -135,19 +221,21 @@ const ALL_STATS: Array = ["AGI", "INT", "CHA", "PSI"]; type MapCellContent = {type: "statPickup", stat: Stat} | {type: "stairs"} | + {type: "empty"} | {type: "block"} type MapCell = { content: MapCellContent, + isValidSpawn: boolean, revealed: boolean } class HuntMode { depth: number - cells: Array> - player: Point | null + cells: Grid + player: Point - constructor({depth, cells, player}: {depth: number, cells: Array>, player: Point | null}) { + constructor({depth, cells, player}: {depth: number, cells: Grid, player: Point }) { this.depth = depth; this.cells = cells; this.player = player; @@ -160,44 +248,55 @@ class HuntMode { let mapName = mapNames[Math.floor(Math.random() * mapNames.length)]; let map = maps[mapName]; - let rows = []; - for (let y = 0; y < mapSzY; y++) { - let row = []; - for (let x = 0; x < mapSzX; x++) { - let src = map[y][x]; - row.push(HuntMode.#generateCell(src)) + let cells = map.map((ccell, _xy) => { + return this.#generateCell(ccell); + }) + + 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); + } } - rows.push(row); } + let player = choose(validSpawns); + cells.get(player).content = {type: "empty"}; if (Math.random() < 0.75) { while (true) { let x = Math.floor(Math.random() * mapSzX); let y = Math.floor(Math.random() * mapSzY); + let xy = new Point(x, y); - let item = rows[y][x]; - if (item.content.type != "block") { - item.content = {type: "stairs"} - break; + let item = cells.get(new Point(x, y)); + if (player.equals(xy)) { + continue; } + if (item.content.type == "block") { + continue; + } + item.content = {type: "stairs"} + break; } } - return new HuntMode({depth, cells: rows, player: null}) + return new HuntMode({depth, cells, player}) } static #generateCell(conceptual: ConceptualCell): MapCell { switch (conceptual) { case "X": - return { content: {type: "block"}, revealed: true}; + return { content: {type: "block"}, revealed: true, isValidSpawn: false}; case " ": - return { content: HuntMode.#generateContent(true), revealed: true }; + return { content: HuntMode.#generateContent(), revealed: false, isValidSpawn: false }; case ".": - return { content: HuntMode.#generateContent(false), revealed: true }; + return { content: HuntMode.#generateContent(), revealed: false, isValidSpawn: true }; } } - static #generateContent(_revealed: boolean): MapCellContent { + static #generateContent(): MapCellContent { // stat pickup let gsp = (): MapCellContent => { return {type: "statPickup", stat: choose(ALL_STATS)} diff --git a/src/maps.ts b/src/maps.ts index d6b8417..be200cf 100644 --- a/src/maps.ts +++ b/src/maps.ts @@ -1,43 +1,30 @@ +import {Grid} from "./engine/datatypes.ts"; + export const mapSzX = 12; export const mapSzY= 9; -export function checkGrid(cells: Array>): Array> { - if (cells.length != mapSzY) { - throw `map must be ${mapSzX}x${mapSzY}` +export function checkGrid(grid: Grid): Grid { + if (grid.size.w != mapSzX || grid.size.h != mapSzY) { + throw `map must be ${mapSzX}x${mapSzY}, not ${grid.size}` } - - for (let row of cells.values()) { - if (row.length != mapSzX) { - throw `map must be ${mapSzX}x${mapSzY}`; - } - } - return cells; + return grid; } export type ConceptualCell = "X" | "." | " "; -function loadMap(map: Array): Array> { - let newRows: Array> = []; - for (let oldRow of map.values()) { - let newRow: Array = [] - for (let i = 0; i < oldRow.length; i++) { - let char = oldRow.charAt(i); - let cell: ConceptualCell - switch(char) { - case "X": cell = "X"; break; - case ".": cell = "."; break; - case " ": cell = " "; break; - default: - throw `map element not valid: ${char}` - } - newRow.push(cell); - } - newRows.push(newRow); - } - return checkGrid(newRows); +function loadMap(map: Array): Grid { + let src = Grid.createGridFromStringArray(map); + return src.map((char: string): ConceptualCell => { + switch(char) { + case "X": return "X"; + case ".": return "."; + case " ": return " "; + default: + throw `map element not valid: ${char}` + }}); } -export let maps: Record>> = { +export let maps: Record> = { map0: loadMap([ "XX XX", "X. .X", diff --git a/src/sprites.ts b/src/sprites.ts index 52b115f..df6f4ce 100644 --- a/src/sprites.ts +++ b/src/sprites.ts @@ -1,14 +1,34 @@ import {Sprite} from "./engine/internal/sprite.ts"; +/* import imgBat from "./art/characters/bat.png"; import imgKobold from "./art/characters/kobold.png"; -import imgRaccoon from "./art/characters/raccoon.png"; -import imgRaccoonWalking from "./art/characters/raccoon_walking.png"; import imgRobot from "./art/characters/robot.png"; import imgSnake from "./art/characters/snake.png"; + */ +import imgRaccoon from "./art/characters/raccoon.png"; +import imgRaccoonWalking from "./art/characters/raccoon_walking.png"; +import imgStatPickup from "./art/pickups/stats.png"; +import {Point, Size} from "./engine/datatypes.ts"; +/* export let sprBat = new Sprite(imgBat, 64, 64, 32, 32, 1, 1, 1); export let sprKobold = new Sprite(imgKobold, 64, 64, 32, 32, 1, 1, 1); -export let sprRaccoon = new Sprite(imgRaccoon, 64, 64, 32, 32, 1, 1, 1); -export let sprRaccoonWalking = new Sprite(imgRaccoonWalking, 64, 64, 32, 32, 8, 1, 8); export let sprRobot = new Sprite(imgRobot, 64, 64, 32, 32, 1, 1, 1); export let sprSnake = new Sprite(imgSnake, 64, 64, 32, 32, 1, 1, 1); + */ + +export let sprRaccoon = new Sprite( + imgRaccoon, + new Size(64, 64), new Point(32, 32), new Size(1, 1), + 1 +); +export let sprRaccoonWalking = new Sprite( + imgRaccoonWalking, + new Size(64, 64), new Point(32, 32), new Size(8, 1), + 8 +); + +export let sprStatPickup = new Sprite( + imgStatPickup, new Size(32, 32), new Point(16, 26), + new Size(4, 1), 4 +);