Non-grid-based movement

This commit is contained in:
Pyrex 2025-02-22 15:50:03 -08:00
parent a528ffd9e0
commit b45f81e6c6
4 changed files with 303 additions and 84 deletions

View File

@ -84,6 +84,13 @@ export class Point {
return new Point(this.x * other.w, this.y * other.h); return new Point(this.x * other.w, this.y * other.h);
} }
unscale(other: Point | Size) {
if (other instanceof Point) {
return new Point(this.x / other.x, this.y / other.y);
}
return new Point(this.x / other.w, this.y / other.h);
}
subtract(top: Point): Size { subtract(top: Point): Size {
return new Size(this.x - top.x, this.y - top.y); return new Size(this.x - top.x, this.y - top.y);
} }
@ -91,6 +98,13 @@ export class Point {
manhattan(other: Point) { manhattan(other: Point) {
return Math.abs(this.x - other.x) + Math.abs(this.y - other.y); return Math.abs(this.x - other.x) + Math.abs(this.y - other.y);
} }
snap(x: number, y: number) {
return new Point(
lerp(x, Math.floor(this.x), Math.ceil(this.x)),
lerp(y, Math.floor(this.y), Math.ceil(this.y))
);
}
} }
export class Size { export class Size {
@ -118,12 +132,16 @@ export class Size {
export class Rect { export class Rect {
readonly top: Point; readonly top: Point;
readonly size: Size; readonly size: Size;
constructor(top: Point, size: Size) { constructor(top: Point, size: Size) {
this.top = top; this.top = top;
this.size = size; this.size = size;
} }
toString(): string {
return `Rect(${this.top},${this.size})`;
}
offset(offset: Point) { offset(offset: Point) {
return new Rect(this.top.offset(offset), this.size); return new Rect(this.top.offset(offset), this.size);
} }
@ -137,6 +155,28 @@ export class Rect {
); );
} }
overlappedCells(size: Size) {
let x0 = this.top.x;
let y0 = this.top.y;
let x1 = x0 + this.size.w;
let y1 = y0 + this.size.w;
let cx0 = Math.floor(x0 / size.w);
let cy0 = Math.floor(y0 / size.h);
let cx1 = Math.ceil(x1 / size.w);
let cy1 = Math.ceil(y1 / size.h);
let cells = [];
for (let cy = cy0; cy < cy1; cy++) {
for (let cx = cx0; cx < cx1; cx++) {
let px0 = cx * size.w;
let py0 = cy * size.h;
cells.push(new Rect(new Point(px0, py0), size));
}
}
return cells;
}
overlaps(other: Rect) { overlaps(other: Rect) {
let ax0 = this.top.x; let ax0 = this.top.x;
let ay0 = this.top.y; let ay0 = this.top.y;
@ -254,3 +294,13 @@ export enum AlignY {
Middle = 1, Middle = 1,
Bottom = 2, Bottom = 2,
} }
export function lerp(amt: number, lo: number, hi: number) {
if (amt <= 0) {
return lo;
}
if (amt >= 1) {
return hi;
}
return lo + (hi - lo) * amt;
};

View File

@ -8,7 +8,7 @@ export const CENTER = new Point(192, 192);
export const MOULDING_SZ = new Size(1, 1); export const MOULDING_SZ = new Size(1, 1);
export class GridArt { export class GridArt {
#at: Point; #atPixel: Point;
#floorCenter: Point; #floorCenter: Point;
#ceilingCenter: Point; #ceilingCenter: Point;
@ -17,27 +17,29 @@ export class GridArt {
#floorBr: Point; #floorBr: Point;
#ceilingBr: Point; #ceilingBr: Point;
constructor(at: Point) { constructor(atPixel: Point) {
this.#at = at; this.#atPixel = atPixel;
this.#floorCenter = at.scale(FLOOR_CELL_SIZE).offset(CENTER); let at = atPixel.unscale(FLOOR_CELL_SIZE);
this.#floorCenter = atPixel.offset(CENTER);
this.#ceilingCenter = at.scale(CEILING_CELL_SIZE).offset(CENTER); this.#ceilingCenter = at.scale(CEILING_CELL_SIZE).offset(CENTER);
this.#floorTl = at this.#floorTl = atPixel.offset(new Point(-FLOOR_CELL_SIZE.w / 2, -FLOOR_CELL_SIZE.h / 2))
.offset(new Point(-0.5, -0.5))
.scale(FLOOR_CELL_SIZE)
.offset(CENTER); .offset(CENTER);
this.#ceilingTl = at this.#ceilingTl = at
.offset(new Point(-0.5, -0.5)) .offset(new Point(-0.5, -0.5))
.scale(CEILING_CELL_SIZE) .scale(CEILING_CELL_SIZE)
.offset(CENTER); .offset(CENTER)
this.#floorBr = at .snap(0, 0)
.offset(new Point(0.5, 0.5)) this.#floorBr = atPixel.offset(new Point(FLOOR_CELL_SIZE.w / 2, FLOOR_CELL_SIZE.h / 2))
.scale(FLOOR_CELL_SIZE)
.offset(CENTER); .offset(CENTER);
this.#ceilingBr = at this.#ceilingBr = at
.offset(new Point(0.5, 0.5)) .offset(new Point(0.5, 0.5))
.scale(CEILING_CELL_SIZE) .scale(CEILING_CELL_SIZE)
.offset(CENTER); .offset(CENTER)
.snap(0, 0)
// console.log(`floorTl: ${this.#floorTl}`)
// console.log(`floorBr: ${this.#floorBr}`)
} }
get floorRect(): Rect { get floorRect(): Rect {
@ -83,84 +85,134 @@ export class GridArt {
} }
drawWallTop(color: Color) { drawWallTop(color: Color) {
if (this.#at.y > 0) { if (this.#atPixel.y > FLOOR_CELL_SIZE.h / 2) {
return; return;
} }
this.#drawWallTop(color); this.#drawWallTop(color);
} }
drawWallLeft(color: Color) { drawWallLeft(color: Color) {
if (this.#at.x > 0) { if (this.#atPixel.x > FLOOR_CELL_SIZE.w / 2) {
return; return;
} }
this.#drawWallLeft(color); this.#drawWallLeft(color);
} }
drawWallBottom(color: Color) { drawWallBottom(color: Color) {
if (this.#at.y < 0) { if (this.#atPixel.y < -FLOOR_CELL_SIZE.h / 2) {
return; return;
} }
new GridArt(this.#at.offset(new Point(0, 1))).#drawWallTop(color); new GridArt(this.#atPixel.offset(new Point(0, FLOOR_CELL_SIZE.h))).#drawWallTop(color);
} }
drawWallRight(color: Color) { drawWallRight(color: Color) {
if (this.#at.x < 0) { if (this.#atPixel.x < -FLOOR_CELL_SIZE.w / 2) {
return; return;
} }
new GridArt(this.#at.offset(new Point(1, 0))).#drawWallLeft(color); new GridArt(this.#atPixel.offset(new Point(FLOOR_CELL_SIZE.w, 0))).#drawWallLeft(color);
} }
drawMouldingTop(color: Color) { drawMouldingTop(color: Color) {
let lhs = this.#ceilingTl.offset(new Point(0, -MOULDING_SZ.h)); let x0 = this.#ceilingTl.x;
D.fillRect(lhs, new Size(CEILING_CELL_SIZE.w, MOULDING_SZ.h), color); let y0 = this.#ceilingTl.y - MOULDING_SZ.h;
let x1 = this.#ceilingBr.x;
let y1 = this.#ceilingTl.y;
D.fillRect(
new Point(x0, y0),
new Size(x1 - x0, y1 - y0),
color
);
} }
drawMouldingTopLeft(color: Color) { drawMouldingTopLeft(color: Color) {
let x0 = this.#ceilingTl.x - MOULDING_SZ.w;
let y0 = this.#ceilingTl.y - MOULDING_SZ.h;
let x1 = this.#ceilingTl.x;
let y1 = this.#ceilingTl.y;
D.fillRect( D.fillRect(
this.#ceilingTl.offset(new Point(-MOULDING_SZ.w, -MOULDING_SZ.h)), new Point(x0, y0),
MOULDING_SZ, new Size(x1 - x0, y1 - y0),
color, color
); );
} }
drawMouldingLeft(color: Color) { drawMouldingLeft(color: Color) {
let lhs = this.#ceilingTl.offset(new Point(-MOULDING_SZ.w, 0)); let x0 = this.#ceilingTl.x - MOULDING_SZ.w;
D.fillRect(lhs, new Size(MOULDING_SZ.w, CEILING_CELL_SIZE.h), color); let y0 = this.#ceilingTl.y;
let x1 = this.#ceilingTl.x;
let y1 = this.#ceilingBr.y;
D.fillRect(
new Point(x0, y0),
new Size(x1 - x0, y1 - y0),
color
);
} }
drawMouldingTopRight(color: Color) { drawMouldingTopRight(color: Color) {
let x0 = this.#ceilingBr.x;
let y0 = this.#ceilingTl.y - MOULDING_SZ.h;
let x1 = this.#ceilingBr.x + MOULDING_SZ.w;
let y1 = this.#ceilingTl.y;
D.fillRect( D.fillRect(
this.#ceilingTl.offset(new Point(CEILING_CELL_SIZE.w, -MOULDING_SZ.h)), new Point(x0, y0),
MOULDING_SZ, new Size(x1 - x0, y1 - y0),
color, color
); );
} }
drawMouldingBottom(color: Color) { drawMouldingBottom(color: Color) {
let lhs = this.#ceilingTl.offset(new Point(0, CEILING_CELL_SIZE.h)); let x0 = this.#ceilingTl.x;
D.fillRect(lhs, new Size(CEILING_CELL_SIZE.w, MOULDING_SZ.h), color); let y0 = this.#ceilingBr.y;
let x1 = this.#ceilingBr.x;
let y1 = this.#ceilingBr.y + MOULDING_SZ.h;
D.fillRect(
new Point(x0, y0),
new Size(x1 - x0, y1 - y0),
color
);
} }
drawMouldingBottomLeft(color: Color) { drawMouldingBottomLeft(color: Color) {
let x0 = this.#ceilingTl.x - MOULDING_SZ.w;
let y0 = this.#ceilingBr.y;
let x1 = this.#ceilingTl.x;
let y1 = this.#ceilingBr.y + MOULDING_SZ.h;
D.fillRect( D.fillRect(
this.#ceilingTl.offset(new Point(-MOULDING_SZ.w, CEILING_CELL_SIZE.h)), new Point(x0, y0),
MOULDING_SZ, new Size(x1 - x0, y1 - y0),
color, color,
); );
} }
drawMouldingRight(color: Color) { drawMouldingRight(color: Color) {
let lhs = this.#ceilingTl.offset(new Point(CEILING_CELL_SIZE.w, 0)); let x0 = this.#ceilingBr.x;
D.fillRect(lhs, new Size(MOULDING_SZ.w, CEILING_CELL_SIZE.h), color); let y0 = this.#ceilingTl.y;
let x1 = this.#ceilingBr.x + MOULDING_SZ.w;
let y1 = this.#ceilingBr.y;
D.fillRect(
new Point(x0, y0),
new Size(x1 - x0, y1 - y0),
color
);
} }
drawMouldingBottomRight(color: Color) { drawMouldingBottomRight(color: Color) {
let x0 = this.#ceilingBr.x;
let y0 = this.#ceilingBr.y;
let x1 = this.#ceilingBr.x + MOULDING_SZ.w;
let y1 = this.#ceilingBr.y + MOULDING_SZ.h;
D.fillRect( D.fillRect(
this.#ceilingTl.offset( new Point(x0, y0),
new Point(CEILING_CELL_SIZE.w, CEILING_CELL_SIZE.h), new Size(x1 - x0, y1 - y0),
), color
MOULDING_SZ,
color,
); );
} }

View File

@ -1,7 +1,7 @@
import { Point, Size } from "./engine/datatypes.ts"; import {Point, Rect, Size} from "./engine/datatypes.ts";
import { DrawPile } from "./drawpile.ts"; import {DrawPile} from "./drawpile.ts";
import { D } from "./engine/public.ts"; import {D, I} from "./engine/public.ts";
import { sprThrallLore } from "./sprites.ts"; import {sprThrallLore} from "./sprites.ts";
import { import {
BG_INSET, BG_INSET,
BG_WALL_OR_UNREVEALED, BG_WALL_OR_UNREVEALED,
@ -11,16 +11,16 @@ import {
FG_TEXT_ENDORSED, FG_TEXT_ENDORSED,
FG_TOO_EXPENSIVE, FG_TOO_EXPENSIVE,
} from "./colors.ts"; } from "./colors.ts";
import { getPlayerProgress } from "./playerprogress.ts"; import {getPlayerProgress} from "./playerprogress.ts";
import { Architecture, LoadedNewMap } from "./newmap.ts"; import {Architecture, LoadedNewMap} from "./newmap.ts";
import { FLOOR_CELL_SIZE, GridArt } from "./gridart.ts"; import {FLOOR_CELL_SIZE, GridArt} from "./gridart.ts";
import { shadowcast } from "./shadowcast.ts"; import {shadowcast} from "./shadowcast.ts";
import { getCheckModal } from "./checkmodal.ts"; import {withCamera} from "./layout.ts";
import { withCamera } from "./layout.ts";
export class HuntMode { export class HuntMode {
map: LoadedNewMap; map: LoadedNewMap;
player: Point; floatingPlayer: Point;
velocity: Point;
faceLeft: boolean; faceLeft: boolean;
drawpile: DrawPile; drawpile: DrawPile;
@ -29,7 +29,8 @@ export class HuntMode {
constructor(depth: number, map: LoadedNewMap) { constructor(depth: number, map: LoadedNewMap) {
this.map = map; this.map = map;
this.player = map.entrance; this.floatingPlayer = map.entrance.offset(new Point(0.5, 0.5));
this.velocity = new Point(0, 0);
this.faceLeft = false; this.faceLeft = false;
this.drawpile = new DrawPile(); this.drawpile = new DrawPile();
@ -39,13 +40,27 @@ export class HuntMode {
// getCheckModal().show(standardVaultTemplates[5].checks[1], null) // getCheckModal().show(standardVaultTemplates[5].checks[1], null)
} }
get gridifiedPlayer(): Point {
return new Point(
Math.floor(this.floatingPlayer.x),
Math.floor(this.floatingPlayer.y),
);
}
get pixelPlayer(): Point {
return new Point(
Math.floor(this.floatingPlayer.x * FLOOR_CELL_SIZE.w - FLOOR_CELL_SIZE.w / 2),
Math.floor(this.floatingPlayer.y * FLOOR_CELL_SIZE.h - FLOOR_CELL_SIZE.h / 2),
)
}
getDepth() { getDepth() {
return this.depth; return this.depth;
} }
// == update logic == // == update logic ==
#collectResources() { #collectResources() {
let cell = this.map.get(this.player); let cell = this.map.get(this.gridifiedPlayer);
let pickup = cell.pickup; let pickup = cell.pickup;
if (pickup != null) { if (pickup != null) {
@ -61,11 +76,11 @@ export class HuntMode {
} }
let dist = Math.max( let dist = Math.max(
Math.abs(mapPosition.x - this.player.x), Math.abs(mapPosition.x - this.gridifiedPlayer.x),
Math.abs(mapPosition.y - this.player.y), Math.abs(mapPosition.y - this.gridifiedPlayer.y),
); );
if (dist != 1) { if (dist > 1) {
return null; return null;
} }
@ -76,6 +91,7 @@ export class HuntMode {
return pickup.computeCostToClick(); return pickup.computeCostToClick();
} }
/*
movePlayerTo(newPosition: Point) { movePlayerTo(newPosition: Point) {
let oldX = this.player.x; let oldX = this.player.x;
let newX = newPosition.x; let newX = newPosition.x;
@ -88,9 +104,10 @@ export class HuntMode {
} }
this.#collectResources(); this.#collectResources();
} }
*/
getZoneLabel(): string | null { getZoneLabel(): string | null {
return this.map.get(this.player).zoneLabel; return this.map.get(this.gridifiedPlayer).zoneLabel;
} }
// draw // draw
@ -109,18 +126,20 @@ export class HuntMode {
this.frame += 1; this.frame += 1;
this.drawpile.clear(); this.drawpile.clear();
let globalOffset = new Point( let globalOffset = this.pixelPlayer.offset(
this.player.x * FLOOR_CELL_SIZE.w, new Point(-192, -192)
this.player.y * FLOOR_CELL_SIZE.h, );
).offset(new Point(-192, -192));
this.#updatePlayer();
this.#updateFov(); this.#updateFov();
this.#updatePickups(); this.#updatePickups();
for (let y = 0; y < this.map.size.h; y += 1) { for (let y = 0; y < this.map.size.h; y += 1) {
for (let x = 0; x < this.map.size.w; x += 1) { for (let x = 0; x < this.map.size.w; x += 1) {
let offsetInCells = new Point(x - this.player.x, y - this.player.y); let offsetInPixels = new Point(x, y).scale(FLOOR_CELL_SIZE).offset(
this.#drawMapCell(offsetInCells, new Point(x, y)); this.pixelPlayer.negate()
);
this.#drawMapCell(offsetInPixels, new Point(x, y));
} }
} }
this.#drawPlayer(globalOffset); this.#drawPlayer(globalOffset);
@ -130,6 +149,60 @@ export class HuntMode {
this.drawpile.executeOnClick(); this.drawpile.executeOnClick();
} }
#updatePlayer() {
let dx = this.velocity.x;
let dy = this.velocity.y;
let amt = 0.005;
if (I.isKeyDown("w")) { dy -= amt; }
if (I.isKeyDown("s")) { dy += amt; }
if (I.isKeyDown("a")) { dx -= amt; }
if (I.isKeyDown("d")) { dx += amt; }
dx *= 0.9;
dy *= 0.9;
this.velocity = new Point(dx, dy);
let nSteps = 40;
let szX = 0.5;
let szY = 0.5;
for (let i = 0; i < nSteps; i++) {
let oldXy = this.floatingPlayer;
let newXy = oldXy.offset(new Point(this.velocity.x / nSteps, 0));
let bbox = new Rect(
newXy.offset(new Point(-szX / 2, -szY / 2)),
new Size(szX, szY)
)
for (let cell of bbox.overlappedCells(new Size(1, 1)).values()) {
if (this.#blocksMovement(cell.top)) {
this.velocity = new Point(0, this.velocity.y);
newXy = oldXy;
}
}
oldXy = newXy;
newXy = oldXy.offset(new Point(0, this.velocity.y / nSteps));
bbox = new Rect(
newXy.offset(new Point(-szX / 2, -szY / 2)),
new Size(szX, szY)
)
for (let cell of bbox.overlappedCells(new Size(1, 1)).values()) {
if (this.#blocksMovement(cell.top)) {
this.velocity = new Point(this.velocity.x, 0);
newXy = oldXy;
}
}
this.floatingPlayer = newXy;
}
}
#updateFov() { #updateFov() {
for (let y = 0; y < this.map.size.h; y += 1) { for (let y = 0; y < this.map.size.h; y += 1) {
for (let x = 0; x < this.map.size.w; x += 1) { for (let x = 0; x < this.map.size.w; x += 1) {
@ -137,15 +210,15 @@ export class HuntMode {
} }
} }
this.map.get(new Point(this.player.x, this.player.y)).revealed = true; this.map.get(this.gridifiedPlayer).revealed = true;
shadowcast( shadowcast(
[this.player.x, this.player.y], [this.gridifiedPlayer.x, this.gridifiedPlayer.y],
([x, y]: [number, number]): boolean => { ([x, y]: [number, number]): boolean => {
let cell = this.map.get(new Point(x, y)); let cell = this.map.get(new Point(x, y));
let pickup = cell.pickup; let pickup = cell.pickup;
return ( return (
cell.architecture == Architecture.Wall || cell.architecture == Architecture.Wall ||
(pickup != null && pickup.isObstructive()) (pickup != null && pickup.obstructsVision())
); );
}, },
([x, y]: [number, number]) => { ([x, y]: [number, number]) => {
@ -167,8 +240,8 @@ export class HuntMode {
} }
#inVisibilityRange(x: number, y: number): boolean { #inVisibilityRange(x: number, y: number): boolean {
let dx = x - this.player.x; let dx = x - this.gridifiedPlayer.x;
let dy = y - this.player.y; let dy = y - this.gridifiedPlayer.y;
return dx * dx + dy * dy < 13; return dx * dx + dy * dy < 13;
} }
@ -176,14 +249,14 @@ export class HuntMode {
this.drawpile.draw(); this.drawpile.draw();
} }
#drawMapCell(offsetInCells: Point, mapPosition: Point) { #drawMapCell(offsetInPixels: Point, mapPosition: Point) {
const OFFSET_UNDER_FLOOR = -512 + mapPosition.y; const OFFSET_UNDER_FLOOR = -512 + mapPosition.y;
const OFFSET_FLOOR = -256 + mapPosition.y; const OFFSET_FLOOR = -256 + mapPosition.y;
const OFFSET_AIR = 0 + mapPosition.y; const OFFSET_AIR = 0 + mapPosition.y;
const OFFSET_TOP = 256 + mapPosition.y; const OFFSET_TOP = 256 + mapPosition.y;
const OFFSET_TOP_OF_TOP = 512 + mapPosition.y; const OFFSET_TOP_OF_TOP = 512 + mapPosition.y;
const gridArt = new GridArt(offsetInCells); const gridArt = new GridArt(offsetInPixels);
let cellData = this.map.get(mapPosition); let cellData = this.map.get(mapPosition);
@ -240,8 +313,10 @@ export class HuntMode {
} }
getPlayerProgress().spendBlood(cost); getPlayerProgress().spendBlood(cost);
this.movePlayerTo(mapPosition);
getCheckModal().show(null, null); // TODO: This isn't a thing anymore
// this.movePlayerTo(mapPosition);
// getCheckModal().show(null, null);
}, },
onSqueeze: () => { onSqueeze: () => {
// the cost _gates_ squeezes // the cost _gates_ squeezes
@ -320,10 +395,7 @@ export class HuntMode {
} }
#drawPlayer(globalOffset: Point) { #drawPlayer(globalOffset: Point) {
let cellOffset = new Point( let cellOffset = this.pixelPlayer.offset(globalOffset.negate());
this.player.x * FLOOR_CELL_SIZE.w,
this.player.y * FLOOR_CELL_SIZE.h,
).offset(globalOffset.negate());
this.drawpile.add(0, () => { this.drawpile.add(0, () => {
D.drawSprite(sprThrallLore, cellOffset, 1, { D.drawSprite(sprThrallLore, cellOffset, 1, {
xScale: this.faceLeft ? -2 : 2, xScale: this.faceLeft ? -2 : 2,
@ -379,6 +451,20 @@ export class HuntMode {
); );
}); });
} }
#blocksMovement(xy: Point) {
let cell = this.map.get(xy);
if (cell.architecture == Architecture.Wall) {
return true;
}
if (cell.pickup?.blocksMovement()) {
return true;
}
// TODO: Other cases
return false;
}
} }
let active: HuntMode | null = null; let active: HuntMode | null = null;

View File

@ -44,7 +44,11 @@ export class LockPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return true; return true;
} }
@ -90,7 +94,11 @@ export class BreakableBlockPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return true; return true;
} }
@ -175,7 +183,11 @@ export class LadderPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return false; return false;
} }
@ -217,7 +229,11 @@ export class ThrallPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return false; return false;
} }
@ -263,7 +279,11 @@ export class ThrallPosterPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return false; return false;
} }
@ -310,7 +330,11 @@ export class ThrallRecruitedPickup {
return !this.bitten; return !this.bitten;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return false; return false;
} }
@ -425,7 +449,11 @@ export class ThrallCollectionPlatePickup {
return !this.rewarded; return !this.rewarded;
} }
isObstructive() { blocksMovement() {
return false;
}
obstructsVision() {
return false; return false;
} }
@ -530,7 +558,10 @@ export class ThrallItemPickup {
return true; return true;
} }
isObstructive() { blocksMovement() {
return true;
}
obstructsVision() {
return false; return false;
} }