Visible costing in many places

This commit is contained in:
Pyrex 2025-02-22 11:44:10 -08:00
parent 258795d76c
commit ddca881b36
7 changed files with 66 additions and 8 deletions

View File

@ -1,6 +1,6 @@
import { DrawPile } from "./drawpile.ts"; import { DrawPile } from "./drawpile.ts";
import { AlignX, AlignY, Point, Rect, Size } from "./engine/datatypes.ts"; import { AlignX, AlignY, Point, Rect, Size } from "./engine/datatypes.ts";
import {BG_INSET, FG_BOLD, FG_TEXT, FG_TEXT_DISABLED} from "./colors.ts"; import {BG_INSET, FG_BOLD, FG_TEXT, FG_TEXT_DISABLED, FG_TEXT_ENDORSED} from "./colors.ts";
import { D } from "./engine/public.ts"; import { D } from "./engine/public.ts";
export function addButton( export function addButton(
@ -9,6 +9,9 @@ export function addButton(
rect: Rect, rect: Rect,
enabled: boolean, enabled: boolean,
cbClick: () => void, cbClick: () => void,
options?: {
endorse?: boolean
}
) { ) {
let padding = 2; let padding = 2;
let topLeft = rect.top; let topLeft = rect.top;
@ -26,6 +29,12 @@ export function addButton(
if (!enabled) { if (!enabled) {
fgLabel = FG_TEXT_DISABLED; fgLabel = FG_TEXT_DISABLED;
} }
if (enabled && options?.endorse) {
fg = FG_TEXT_ENDORSED;
fgLabel = FG_TEXT_ENDORSED;
}
if (hover) { if (hover) {
[bg, fg, fgLabel] = [FG_BOLD, BG_INSET, BG_INSET]; [bg, fg, fgLabel] = [FG_BOLD, BG_INSET, BG_INSET];
} }

View File

@ -6,6 +6,7 @@ export const BG_INSET = Color.parseHexCode("#242234");
export const FG_TEXT = Color.parseHexCode("#c0c0c0"); export const FG_TEXT = Color.parseHexCode("#c0c0c0");
export const FG_TEXT_DISABLED = Color.parseHexCode("#808080"); export const FG_TEXT_DISABLED = Color.parseHexCode("#808080");
export const FG_TOO_EXPENSIVE = Color.parseHexCode("#ff8080"); export const FG_TOO_EXPENSIVE = Color.parseHexCode("#ff8080");
export const FG_TEXT_ENDORSED = Color.parseHexCode("#80ff80");
export const FG_BOLD = Color.parseHexCode("#ffffff"); export const FG_BOLD = Color.parseHexCode("#ffffff");
export const BG_CEILING = Color.parseHexCode("#143464"); export const BG_CEILING = Color.parseHexCode("#143464");
export const FG_MOULDING = FG_TEXT; export const FG_MOULDING = FG_TEXT;

View File

@ -4,10 +4,14 @@ import { withCamera } from "./layout.ts";
import { getSkillsModal } from "./skillsmodal.ts"; import { getSkillsModal } from "./skillsmodal.ts";
import { addButton } from "./button.ts"; import { addButton } from "./button.ts";
import { getSleepModal } from "./sleepmodal.ts"; import { getSleepModal } from "./sleepmodal.ts";
import {getPlayerProgress} from "./playerprogress.ts";
import {getSkills} from "./skills.ts";
type Button = { type Button = {
label: string; label: string;
cbClick: () => void; cbClick: () => void;
enabled: boolean;
endorse: boolean;
}; };
export class Hotbar { export class Hotbar {
@ -33,6 +37,8 @@ export class Hotbar {
cbClick: () => { cbClick: () => {
getSkillsModal().setShown(true); getSkillsModal().setShown(true);
}, },
enabled: getPlayerProgress().getAvailableSkills().length > 0,
endorse: getPlayerProgress().anyAffordableSkillsAtMinimum(),
}); });
/* /*
buttons.push({ buttons.push({
@ -44,6 +50,8 @@ export class Hotbar {
cbClick: () => { cbClick: () => {
getSleepModal().setShown(true); getSleepModal().setShown(true);
}, },
enabled: true,
endorse: getPlayerProgress().getBlood() < 100,
}); });
return buttons; return buttons;
} }
@ -65,8 +73,9 @@ export class Hotbar {
this.#drawpile, this.#drawpile,
b.label, b.label,
new Rect(new Point(x, 0), cellSize), new Rect(new Point(x, 0), cellSize),
true, b.enabled,
b.cbClick, b.cbClick,
{endorse: b.endorse}
); );
x += cellSize.w; x += cellSize.w;
} }

View File

@ -1,6 +1,6 @@
import { D } from "./engine/public.ts"; import { D } from "./engine/public.ts";
import { Point, Size } from "./engine/datatypes.ts"; import { Point, Size } from "./engine/datatypes.ts";
import {BG_OUTER, FG_BOLD, FG_TEXT, FG_TOO_EXPENSIVE} from "./colors.ts"; import {BG_OUTER, FG_BOLD, FG_TEXT, FG_TEXT_ENDORSED, FG_TOO_EXPENSIVE} from "./colors.ts";
import { ALL_STATS } from "./datatypes.ts"; import { ALL_STATS } from "./datatypes.ts";
import { getPlayerProgress } from "./playerprogress.ts"; import { getPlayerProgress } from "./playerprogress.ts";
import { getHuntMode } from "./huntmode.ts"; import { getHuntMode } from "./huntmode.ts";
@ -42,6 +42,7 @@ export class Hud {
D.drawText("BLD", new Point(0, 160), FG_BOLD); D.drawText("BLD", new Point(0, 160), FG_BOLD);
let bloodAmount = prog.getBlood(); let bloodAmount = prog.getBlood();
let bloodColor = FG_TEXT; let bloodColor = FG_TEXT;
if (bloodAmount > 2000) { bloodColor = FG_TEXT_ENDORSED; }
if (bloodAmount < 100) { bloodColor = FG_TOO_EXPENSIVE; } if (bloodAmount < 100) { bloodColor = FG_TOO_EXPENSIVE; }
D.drawText(`${prog.getBlood()}cc`, new Point(32, 160), bloodColor); D.drawText(`${prog.getBlood()}cc`, new Point(32, 160), bloodColor);
} }

View File

@ -274,6 +274,18 @@ export class PlayerProgress {
} }
return ItemStage.Untouched; return ItemStage.Untouched;
} }
anyAffordableSkillsAtMinimum() {
let skills = this.getAvailableSkills();
for (let skill of skills.values()) {
if (getSkills().isAtMinimum(skill)) {
if (getPlayerProgress().getExperience() > getSkills().computeCost(skill)) {
return true;
}
}
}
return false;
}
} }
let active: PlayerProgress | null = null; let active: PlayerProgress | null = null;

View File

@ -68,6 +68,12 @@ class SkillsTable {
), ),
); );
} }
isAtMinimum(skill: Skill) {
let minimumCost = this.get(skill).governing.cost
let currentCost = this.computeCost(skill);
return currentCost <= minimumCost;
}
} }
function geomInterpolate( function geomInterpolate(

View File

@ -2,7 +2,7 @@ import { getPartLocation, withCamera } from "./layout.ts";
import { AlignX, Point, Rect, Size } from "./engine/datatypes.ts"; import { AlignX, 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 } from "./engine/public.ts";
import { BG_INSET, FG_BOLD, FG_TEXT } from "./colors.ts"; import {BG_INSET, FG_BOLD, FG_TEXT, FG_TEXT_DISABLED, FG_TEXT_ENDORSED} from "./colors.ts";
import { addButton } from "./button.ts"; import { addButton } from "./button.ts";
import { getSkills } from "./skills.ts"; import { getSkills } from "./skills.ts";
import { getPlayerProgress } from "./playerprogress.ts"; import { getPlayerProgress } from "./playerprogress.ts";
@ -54,7 +54,7 @@ export class SkillsModal {
this.#fixSkillSelection(availableSkills); this.#fixSkillSelection(availableSkills);
let y = 0; let y = 0;
for (let skill of availableSkills) { availableSkills.forEach((skill) => {
let data = getSkills().get(skill); let data = getSkills().get(skill);
let cost = getSkills().computeCost(skill); let cost = getSkills().computeCost(skill);
let y_ = y; let y_ = y;
@ -66,9 +66,23 @@ export class SkillsModal {
0, 0,
(hover) => { (hover) => {
// two column layout // two column layout
let [bg, fg] = [BG_INSET, FG_TEXT]; let [bg, fg] = [BG_INSET, FG_BOLD];
let overpriced = getSkills().computeCost(skill) > getPlayerProgress().getExperience();
let atMinimum = getSkills().isAtMinimum(skill);
if (overpriced) {
fg = FG_TEXT_DISABLED;
} else if (atMinimum) {
fg = FG_TEXT_ENDORSED;
}
if (selected || hover) { if (selected || hover) {
[bg, fg] = [FG_BOLD, BG_INSET]; [bg, fg] = [FG_BOLD, BG_INSET];
if (overpriced) {
// still use the same BG, for contrast
} else if (atMinimum) {
bg = FG_TEXT_ENDORSED;
}
} }
D.fillRect(skillRect.top, skillRect.size, bg); D.fillRect(skillRect.top, skillRect.size, bg);
D.drawText(data.profile.name, new Point(4, y_), fg); D.drawText(data.profile.name, new Point(4, y_), fg);
@ -83,7 +97,7 @@ export class SkillsModal {
}, },
); );
y += 16; y += 16;
} });
// add skill description // add skill description
let selection = this.#skillSelection; let selection = this.#skillSelection;
@ -107,21 +121,27 @@ export class SkillsModal {
); );
let canAfford = getPlayerProgress().getExperience() >= cost; let canAfford = getPlayerProgress().getExperience() >= cost;
let caption = `Learn ${data.profile.name}`; let caption = `Learn ${data.profile.name}`;
let endorse = getSkills().isAtMinimum(selection);
if (!canAfford) { if (!canAfford) {
caption = `Can't Afford`; caption = `Can't Afford`;
endorse = false;
} }
addButton(this.#drawpile, caption, drawButtonRect, canAfford, () => { addButton(this.#drawpile, caption, drawButtonRect, canAfford, () => {
getPlayerProgress().spendExperience(cost); getPlayerProgress().spendExperience(cost);
getPlayerProgress().learnSkill(selection); getPlayerProgress().learnSkill(selection);
}, {
endorse
}); });
} }
// add close button // add close button
let endorseBack = availableSkills.length == 0;
let closeRect = new Rect(new Point(0, 96), new Size(160, 32)); let closeRect = new Rect(new Point(0, 96), new Size(160, 32));
addButton(this.#drawpile, "Back", closeRect, true, () => { addButton(this.#drawpile, "Back", closeRect, true, () => {
this.setShown(false); this.setShown(false);
}); }, {endorse: endorseBack});
this.#drawpile.executeOnClick(); this.#drawpile.executeOnClick();
} }