fledgling/src/huntmode.ts

310 lines
7.9 KiB
TypeScript

import { Point } from "./engine/datatypes.ts";
import { DrawPile } from "./drawpile.ts";
import { D } from "./engine/public.ts";
import { sprThrallLore } from "./sprites.ts";
import {
BG_INSET,
BG_WALL_OR_UNREVEALED,
FG_BOLD,
FG_MOULDING,
FG_TEXT, FG_TOO_EXPENSIVE,
} from "./colors.ts";
import { getPlayerProgress } from "./playerprogress.ts";
import { Architecture, LoadedNewMap } from "./newmap.ts";
import { FLOOR_CELL_SIZE, GridArt } from "./gridart.ts";
import { shadowcast } from "./shadowcast.ts";
import { getCheckModal } from "./checkmodal.ts";
import {withCamera} from "./layout.ts";
export class HuntMode {
map: LoadedNewMap;
player: Point;
faceLeft: boolean;
drawpile: DrawPile;
frame: number;
depth: number;
constructor(depth: number, map: LoadedNewMap) {
this.map = map;
this.player = map.entrance;
this.faceLeft = false;
this.drawpile = new DrawPile();
this.frame = 0;
this.depth = depth;
// getCheckModal().show(standardVaultTemplates[5].checks[1], null)
}
getDepth() {
return this.depth;
}
// == update logic ==
#collectResources() {
let cell = this.map.get(this.player);
let pickup = cell.pickup;
if (pickup != null) {
cell.pickup = null;
}
}
#computeCostToClick(mapPosition: Point): number | null {
let present = this.map.get(mapPosition);
if (present.architecture != Architecture.Floor) {
return null;
}
let dist = Math.max(
Math.abs(mapPosition.x - this.player.x),
Math.abs(mapPosition.y - this.player.y),
);
if (dist != 1) {
return null;
}
let pickup = present.pickup;
if (pickup == null) {
return 10;
}
return pickup.computeCostToClick();
}
movePlayerTo(newPosition: Point) {
let oldX = this.player.x;
let newX = newPosition.x;
this.player = newPosition;
if (newX < oldX) {
this.faceLeft = true;
}
if (oldX < newX) {
this.faceLeft = false;
}
this.#collectResources();
}
// draw
update() {
withCamera("Gameplay", () => { this.#update() });
}
draw() {
withCamera("Gameplay", () => { this.#draw() });
}
#update() {
this.frame += 1;
this.drawpile.clear();
let globalOffset = new Point(
this.player.x * FLOOR_CELL_SIZE.w,
this.player.y * FLOOR_CELL_SIZE.h,
).offset(new Point(-192, -192));
this.#updateFov();
for (let y = 0; y < this.map.size.h; y += 1) {
for (let x = 0; x < this.map.size.w; x += 1) {
let offsetInCells = new Point(x - this.player.x, y - this.player.y);
this.#drawMapCell(offsetInCells, new Point(x, y));
}
}
this.#drawPlayer(globalOffset);
this.drawpile.executeOnClick();
}
#updateFov() {
for (let y = 0; y < this.map.size.h; y += 1) {
for (let x = 0; x < this.map.size.w; x += 1) {
this.map.get(new Point(x, y)).revealed = false;
}
}
this.map.get(new Point(this.player.x, this.player.y)).revealed = true;
shadowcast(
[this.player.x, this.player.y],
([x, y]: [number, number]): boolean => {
let cell = this.map.get(new Point(x, y));
let pickup = cell.pickup;
return (
cell.architecture == Architecture.Wall ||
(pickup != null && pickup.isObstructive())
);
},
([x, y]: [number, number]) => {
let dx = x - this.player.x;
let dy = y - this.player.y;
if (dx * dx + dy * dy >= 13) {
return;
}
this.map.get(new Point(x, y)).revealed = true;
},
);
}
#draw() {
this.drawpile.draw();
}
#drawMapCell(offsetInCells: Point, mapPosition: Point) {
const OFFSET_UNDER_FLOOR = -512 + mapPosition.y;
const OFFSET_FLOOR = -256 + mapPosition.y;
const OFFSET_AIR = 0 + mapPosition.y;
const OFFSET_TOP = 256 + mapPosition.y;
const OFFSET_TOP_OF_TOP = 512 + mapPosition.y;
const gridArt = new GridArt(offsetInCells);
let cellData = this.map.get(mapPosition);
this.drawpile.add(OFFSET_UNDER_FLOOR, () => {
gridArt.drawCeiling(BG_WALL_OR_UNREVEALED);
});
if (cellData.architecture == Architecture.Wall || !cellData.revealed) {
this.drawpile.add(OFFSET_TOP, () => {
gridArt.drawCeiling(BG_WALL_OR_UNREVEALED);
});
return;
}
if (!cellData.revealed) {
return;
}
let pickup = cellData.pickup;
// draw inset zone
let cost = this.#computeCostToClick(mapPosition);
let tooExpensive = cost != null && (cost > getPlayerProgress().getBlood());
this.drawpile.addClickable(
OFFSET_FLOOR,
(hover: boolean) => {
let highlighted = hover;
if (cost == null) {
highlighted = false;
}
if (!(pickup?.advertisesClickable() ?? true)) {
highlighted = false;
}
let color = BG_INSET;
if (highlighted) {
color = FG_TEXT;
if (tooExpensive) {
color = FG_TOO_EXPENSIVE;
}
}
gridArt.drawFloor(color);
pickup?.drawFloor(gridArt);
},
gridArt.floorRect,
true,
() => {
if (cost == null || tooExpensive) { return; }
if (pickup?.onClick(cellData)) {
return;
}
if (cost != null) {
getPlayerProgress().spendBlood(cost);
this.movePlayerTo(mapPosition);
getCheckModal().show(null, null);
}
},
);
if (pickup != null) {
this.drawpile.add(OFFSET_AIR, () => {
pickup.drawInAir(gridArt);
});
}
const isRevealedBlock = (dx: number, dy: number) => {
let other = this.map.get(mapPosition.offset(new Point(dx, dy)));
return other.revealed && other.architecture == Architecture.Wall;
};
if (isRevealedBlock(0, -1) && isRevealedBlock(-1, 0)) {
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingTopLeft(FG_MOULDING);
});
}
if (isRevealedBlock(0, -1)) {
this.drawpile.add(OFFSET_AIR, () => {
gridArt.drawWallTop(FG_TEXT);
});
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingTop(FG_MOULDING);
});
}
if (isRevealedBlock(0, -1) && isRevealedBlock(1, 0)) {
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingTopRight(FG_MOULDING);
});
}
if (isRevealedBlock(-1, 0)) {
this.drawpile.add(OFFSET_AIR, () => {
gridArt.drawWallLeft(FG_TEXT);
});
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingLeft(FG_MOULDING);
});
}
if (isRevealedBlock(0, 1) && isRevealedBlock(-1, 0)) {
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingBottomLeft(FG_MOULDING);
});
}
if (isRevealedBlock(0, 1)) {
this.drawpile.add(OFFSET_AIR, () => {
gridArt.drawWallBottom(FG_BOLD);
});
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingBottom(FG_MOULDING);
});
}
if (isRevealedBlock(0, 1) && isRevealedBlock(1, 0)) {
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingBottomRight(FG_MOULDING);
});
}
if (isRevealedBlock(1, 0)) {
this.drawpile.add(OFFSET_AIR, () => {
gridArt.drawWallRight(FG_BOLD);
});
this.drawpile.add(OFFSET_TOP_OF_TOP, () => {
gridArt.drawMouldingRight(FG_MOULDING);
});
}
}
#drawPlayer(globalOffset: Point) {
let cellOffset = new Point(
this.player.x * FLOOR_CELL_SIZE.w,
this.player.y * FLOOR_CELL_SIZE.h,
).offset(globalOffset.negate());
this.drawpile.add(this.player.y, () => {
D.drawSprite(sprThrallLore, cellOffset, 1, {
xScale: this.faceLeft ? -2 : 2,
yScale: 2,
});
});
}
}
let active: HuntMode | null = null;
export function initHuntMode(huntMode: HuntMode) {
active = huntMode;
}
export function getHuntMode() {
if (active == null) {
throw new Error(`trying to get hunt mode before it has been initialized`);
}
return active;
}