747 lines
15 KiB
TypeScript
747 lines
15 KiB
TypeScript
import { getThralls, ItemStage, LifeStage, Thrall } from "./thralls.ts";
|
|
import { CellView, CheckData } from "./newmap.ts";
|
|
import { getPlayerProgress } from "./playerprogress.ts";
|
|
import { getHuntMode, HuntMode, initHuntMode } from "./huntmode.ts";
|
|
import { generateMap } from "./mapgen.ts";
|
|
import { ALL_STATS, Stat } from "./datatypes.ts";
|
|
import { D } from "./engine/public.ts";
|
|
import {
|
|
sprCollectibles,
|
|
sprCollectiblesSilhouettes,
|
|
sprLadder,
|
|
sprLock,
|
|
} from "./sprites.ts";
|
|
import { GridArt } from "./gridart.ts";
|
|
import { getCheckModal } from "./checkmodal.ts";
|
|
import { Point, Size } from "./engine/datatypes.ts";
|
|
import { choose } from "./utils.ts";
|
|
import { FG_BOLD, FG_TEXT, SWATCH_EXP, SWATCH_STAT } from "./colors.ts";
|
|
import { Block3D } from "./world3d.ts";
|
|
import { DrawPile } from "./drawpile.ts";
|
|
import { Floater } from "./floater.ts";
|
|
|
|
export type Pickup =
|
|
| LockPickup
|
|
| BreakableBlockPickup
|
|
| LadderPickup
|
|
| ThrallPickup
|
|
| ThrallPosterPickup
|
|
| ThrallRecruitedPickup;
|
|
|
|
export class LockPickup {
|
|
check: CheckData;
|
|
|
|
constructor(check: CheckData) {
|
|
this.check = check;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return true;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
for (let z = 0; z < 5; z += 0.25) {
|
|
D.drawSprite(sprLock, gridArt.project(z), 0, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(cell: CellView): boolean {
|
|
getCheckModal().show(this.check, () => (cell.pickup = null));
|
|
return true;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export type BreakableBlockPickupCallbacks =
|
|
| StatPickupCallbacks
|
|
| ExperiencePickupCallbacks;
|
|
|
|
const RECOVERY_PER_TICK: number = 0.1;
|
|
export class BreakableBlockPickup {
|
|
callbacks: BreakableBlockPickupCallbacks;
|
|
breakProgress: number;
|
|
|
|
constructor(callbacks: BreakableBlockPickupCallbacks) {
|
|
this.callbacks = callbacks;
|
|
this.breakProgress = 0.0;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return this.callbacks.cost;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return true;
|
|
}
|
|
|
|
get #adjustedProgress(): number {
|
|
return Math.pow(this.breakProgress, 2.15);
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(384, () => {
|
|
let progress = this.#adjustedProgress;
|
|
let extraMult = 1.0;
|
|
let angleRange = 0;
|
|
if (progress != 0) {
|
|
extraMult = 1.2;
|
|
angleRange = 10;
|
|
}
|
|
|
|
this.callbacks.draw(gridArt, {
|
|
xScale: 2 * (1.0 - progress * 0.7) * extraMult,
|
|
yScale: 2 * (1.0 - progress * 0.7) * extraMult,
|
|
angle: (2 * progress - 1) * angleRange,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return this.callbacks.getBlock(this.breakProgress);
|
|
}
|
|
|
|
update(cellData: CellView) {
|
|
if (this.breakProgress >= 1.0) {
|
|
getPlayerProgress().spendBlood(this.callbacks.cost);
|
|
cellData.pickup = null;
|
|
|
|
let n = choose([1, 1, 1, 1, 1, 2, 3]);
|
|
for (let i = 0; i < n; i++) {
|
|
let floater = new Floater(
|
|
cellData.xy.offset(new Point(0.5, 0.5)),
|
|
50,
|
|
this.callbacks,
|
|
);
|
|
let speed = 0.015;
|
|
let direction = Math.random() * Math.PI * 2;
|
|
floater.velocity = new Point(
|
|
Math.cos(direction) * speed,
|
|
Math.sin(direction) * speed * 0.1,
|
|
);
|
|
floater.velZ = 0.8;
|
|
getHuntMode().spawnFloater(floater);
|
|
}
|
|
}
|
|
|
|
this.breakProgress = Math.max(0.0, this.breakProgress - RECOVERY_PER_TICK);
|
|
}
|
|
|
|
onClick(): boolean {
|
|
return true;
|
|
}
|
|
|
|
onSqueeze(_cellData: CellView) {
|
|
this.breakProgress = Math.min(
|
|
this.breakProgress + 0.02 + RECOVERY_PER_TICK,
|
|
1.0,
|
|
);
|
|
}
|
|
}
|
|
|
|
export class StatPickupCallbacks {
|
|
#stat: Stat;
|
|
|
|
constructor(stat: Stat) {
|
|
this.#stat = stat;
|
|
}
|
|
|
|
get cost(): number {
|
|
return 100;
|
|
}
|
|
|
|
obtain() {
|
|
getPlayerProgress().add(this.#stat, 1);
|
|
getPlayerProgress().purloinItem();
|
|
}
|
|
|
|
getBlock(progress: number) {
|
|
return new Block3D(
|
|
progress > 0.6 ? FG_BOLD : SWATCH_STAT[this.#stat][1],
|
|
progress > 0.6 ? FG_TEXT : SWATCH_STAT[this.#stat][0],
|
|
progress > 0.6 ? FG_BOLD : SWATCH_STAT[this.#stat][1],
|
|
);
|
|
}
|
|
|
|
draw(
|
|
gridArt: GridArt,
|
|
options: { xScale?: number; yScale?: number; angle?: number },
|
|
) {
|
|
let statIndex = ALL_STATS.indexOf(this.#stat);
|
|
if (statIndex == -1) {
|
|
return;
|
|
}
|
|
|
|
let at = gridArt.project(100);
|
|
D.drawSprite(sprCollectiblesSilhouettes, at, statIndex, options);
|
|
}
|
|
|
|
drawParticle(at: Point, isShadow: boolean, rotation: number) {
|
|
let statIndex = ALL_STATS.indexOf(this.#stat);
|
|
if (statIndex == -1) {
|
|
return;
|
|
}
|
|
D.drawSprite(
|
|
isShadow ? sprCollectiblesSilhouettes : sprCollectibles,
|
|
at,
|
|
statIndex,
|
|
{ xScale: 2 * Math.cos(rotation), yScale: 2 },
|
|
);
|
|
}
|
|
}
|
|
|
|
export class ExperiencePickupCallbacks {
|
|
constructor() {}
|
|
|
|
get cost(): number {
|
|
return 100;
|
|
}
|
|
|
|
obtain() {
|
|
getPlayerProgress().addExperience(250);
|
|
getPlayerProgress().purloinItem();
|
|
}
|
|
|
|
getBlock(progress: number) {
|
|
return new Block3D(
|
|
progress > 0.6 ? FG_BOLD : SWATCH_EXP[1],
|
|
progress > 0.6 ? FG_TEXT : SWATCH_EXP[0],
|
|
progress > 0.6 ? FG_BOLD : SWATCH_EXP[1],
|
|
);
|
|
}
|
|
|
|
draw(
|
|
gridArt: GridArt,
|
|
options: { xScale?: number; yScale?: number; angle?: number },
|
|
) {
|
|
let at = gridArt.project(100);
|
|
D.drawSprite(sprCollectiblesSilhouettes, at, 4, options);
|
|
}
|
|
|
|
drawParticle(at: Point, isShadow: boolean, rotation: number) {
|
|
D.drawSprite(
|
|
isShadow ? sprCollectiblesSilhouettes : sprCollectibles,
|
|
at,
|
|
4,
|
|
{ xScale: 2 * Math.cos(rotation), yScale: 2 },
|
|
);
|
|
}
|
|
}
|
|
|
|
export class LadderPickup {
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(-128, () => {
|
|
D.drawSprite(sprLadder, gridArt.project(0.0), 0, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(): boolean {
|
|
getPlayerProgress().addBlood(1000);
|
|
initHuntMode(new HuntMode(getHuntMode().depth + 1, generateMap()));
|
|
return false;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export class ThrallPickup {
|
|
thrall: Thrall;
|
|
|
|
constructor(thrall: Thrall) {
|
|
this.thrall = thrall;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
let data = getThralls().get(this.thrall);
|
|
D.drawSprite(data.sprite, gridArt.project(0.0), 0, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(cell: CellView): boolean {
|
|
let data = getThralls().get(this.thrall);
|
|
getCheckModal().show(data.initialCheck, () => {
|
|
getPlayerProgress().unlockThrall(this.thrall);
|
|
cell.pickup = null;
|
|
});
|
|
return true;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export class ThrallPosterPickup {
|
|
thrall: Thrall;
|
|
|
|
constructor(thrall: Thrall) {
|
|
this.thrall = thrall;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
let data = getThralls().get(this.thrall);
|
|
D.drawSprite(data.sprite, gridArt.project(0.0), 2, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(cell: CellView): boolean {
|
|
let data = getThralls().get(this.thrall);
|
|
getCheckModal().show(data.posterCheck, () => (cell.pickup = null));
|
|
return true;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export class ThrallRecruitedPickup {
|
|
thrall: Thrall;
|
|
spokenTo: boolean;
|
|
bitten: boolean;
|
|
|
|
constructor(thrall: Thrall) {
|
|
this.thrall = thrall;
|
|
this.spokenTo = false;
|
|
this.bitten = false;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return !this.spokenTo;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return !this.bitten;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
let data = getThralls().get(this.thrall);
|
|
let lifeStage = getPlayerProgress().getThrallLifeStage(this.thrall);
|
|
let ix = 0;
|
|
let rot = 0;
|
|
|
|
if (lifeStage == LifeStage.Vampirized) {
|
|
ix = 1;
|
|
}
|
|
if (lifeStage == LifeStage.Dead) {
|
|
ix = 1;
|
|
rot = 270;
|
|
}
|
|
D.drawSprite(data.sprite, gridArt.project(0.0), ix, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
angle: rot,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(_cell: CellView): boolean {
|
|
this.spokenTo = true;
|
|
if (this.bitten) {
|
|
return true;
|
|
}
|
|
|
|
let data = getThralls().get(this.thrall);
|
|
let lifeStage = getPlayerProgress().getThrallLifeStage(this.thrall);
|
|
let text = data.lifeStageText[lifeStage];
|
|
getCheckModal().show(
|
|
{
|
|
label: `${text.prebite}`,
|
|
options: [
|
|
{
|
|
isChoice: true,
|
|
countsAsSuccess: true,
|
|
unlockable: "Bite!",
|
|
success: text.postbite,
|
|
},
|
|
{
|
|
isChoice: true,
|
|
countsAsSuccess: false,
|
|
unlockable: "Refrain",
|
|
success: "Maybe next time.",
|
|
},
|
|
],
|
|
},
|
|
() => {
|
|
this.bitten = true;
|
|
getPlayerProgress().addBlood(
|
|
lifeStage == LifeStage.Fresh
|
|
? 1000
|
|
: lifeStage == LifeStage.Average
|
|
? 500
|
|
: lifeStage == LifeStage.Poor
|
|
? 300
|
|
: lifeStage == LifeStage.Vampirized
|
|
? 1500 // lethal bite
|
|
: // lifeStage == LifeStage.Dead ?
|
|
100,
|
|
);
|
|
getPlayerProgress().damageThrall(this.thrall, choose([0.9]));
|
|
},
|
|
);
|
|
return true;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export class ThrallCollectionPlatePickup {
|
|
thrall: Thrall;
|
|
rewarded: boolean;
|
|
|
|
constructor(thrall: Thrall) {
|
|
this.thrall = thrall;
|
|
this.rewarded = false;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
let lifeStage = getPlayerProgress().getThrallLifeStage(this.thrall);
|
|
let itemStage = getPlayerProgress().getThrallItemStage(this.thrall);
|
|
if (itemStage == ItemStage.Obtained) {
|
|
// the player should deliver it! make sure they see a badge informing them of that
|
|
return true;
|
|
}
|
|
|
|
if (
|
|
itemStage == ItemStage.Delivered &&
|
|
lifeStage != LifeStage.Dead &&
|
|
!this.rewarded
|
|
) {
|
|
// the player should collect it! make sure they see a badge informing them of that
|
|
return true;
|
|
}
|
|
|
|
// OK, the only interaction is to get a hint
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return !this.rewarded;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return false;
|
|
}
|
|
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
let itemStage = getPlayerProgress().getThrallItemStage(this.thrall);
|
|
let data = getThralls().get(this.thrall);
|
|
|
|
if (itemStage != ItemStage.Delivered) {
|
|
D.drawRect(
|
|
gridArt.project(0).offset(new Point(-18, -18)),
|
|
new Size(36, 36),
|
|
FG_TEXT,
|
|
);
|
|
} else {
|
|
D.drawSprite(data.sprite, gridArt.project(2), 3, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(cell: CellView): boolean {
|
|
let lifeStage = getPlayerProgress().getThrallLifeStage(this.thrall);
|
|
let itemStage = getPlayerProgress().getThrallItemStage(this.thrall);
|
|
let data = getThralls().get(this.thrall);
|
|
|
|
// if (itemStage == ItemStage.Untouched) { itemStage = ItemStage.Obtained; }
|
|
|
|
if (itemStage == ItemStage.Untouched) {
|
|
if (lifeStage == LifeStage.Dead) {
|
|
getCheckModal().show(
|
|
{
|
|
label: "There's no point in delivering this now.",
|
|
options: [],
|
|
},
|
|
null,
|
|
);
|
|
} else {
|
|
getCheckModal().show(
|
|
{
|
|
label: data.itemHint,
|
|
options: [],
|
|
},
|
|
null,
|
|
);
|
|
}
|
|
} else if (itemStage == ItemStage.Obtained) {
|
|
getPlayerProgress().deliverThrallItem(this.thrall);
|
|
if (lifeStage != LifeStage.Dead) {
|
|
getCheckModal().show(
|
|
{
|
|
label: data.deliveryMessage,
|
|
options: [],
|
|
},
|
|
null,
|
|
);
|
|
}
|
|
} else {
|
|
if (lifeStage == LifeStage.Dead) {
|
|
// nothing happens
|
|
} else if (this.rewarded) {
|
|
// nothing happens
|
|
} else {
|
|
this.rewarded = true;
|
|
getCheckModal().show(
|
|
{
|
|
label: data.rewardMessage,
|
|
options: [],
|
|
},
|
|
null,
|
|
);
|
|
data.rewardCallback((what) => this.spawn(cell.xy, what));
|
|
}
|
|
}
|
|
|
|
// getCheckModal().show(this.check, () => (cell.pickup = null));
|
|
return true;
|
|
}
|
|
|
|
spawn(xy: Point, what: Stat | "EXP") {
|
|
let callbacks = null;
|
|
if (what == "EXP") {
|
|
callbacks = new ExperiencePickupCallbacks();
|
|
} else {
|
|
callbacks = new StatPickupCallbacks(what);
|
|
}
|
|
if (callbacks == null) { return; }
|
|
let floater = new Floater(
|
|
xy.offset(new Point(0.5, 0.5)),
|
|
50,
|
|
callbacks,
|
|
);
|
|
let speed = 0.015;
|
|
let direction = Math.random() * Math.PI * 2;
|
|
floater.velocity = new Point(
|
|
Math.cos(direction) * speed,
|
|
Math.sin(direction) * speed * 0.1,
|
|
);
|
|
floater.velZ = 0.8;
|
|
getHuntMode().spawnFloater(floater);
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|
|
|
|
export class ThrallItemPickup {
|
|
thrall: Thrall;
|
|
|
|
constructor(thrall: Thrall) {
|
|
this.thrall = thrall;
|
|
}
|
|
|
|
computeCostToClick() {
|
|
return 0;
|
|
}
|
|
|
|
advertisesBadge() {
|
|
return false;
|
|
}
|
|
|
|
advertisesClickable() {
|
|
return true;
|
|
}
|
|
|
|
blocksMovement() {
|
|
return true;
|
|
}
|
|
obstructsVision() {
|
|
return false;
|
|
}
|
|
|
|
draw(drawpile: DrawPile, gridArt: GridArt) {
|
|
drawpile.add(0, () => {
|
|
let data = getThralls().get(this.thrall);
|
|
|
|
D.drawSprite(data.sprite, gridArt.project(2), 3, {
|
|
xScale: 2.0,
|
|
yScale: 2.0,
|
|
});
|
|
});
|
|
}
|
|
|
|
getBlock(): Block3D | null {
|
|
return null;
|
|
}
|
|
|
|
update() {}
|
|
|
|
onClick(cell: CellView): boolean {
|
|
let data = getThralls().get(this.thrall);
|
|
|
|
cell.pickup = null;
|
|
getCheckModal().show(
|
|
{
|
|
label: data.itemPickupMessage,
|
|
options: [],
|
|
},
|
|
null,
|
|
);
|
|
getPlayerProgress().obtainThrallItem(this.thrall);
|
|
return true;
|
|
}
|
|
|
|
onSqueeze() {}
|
|
}
|