generated from nhcarrigan/template
1195b657a0
Working through open issues — fixes, balance changes, and features. ## Closed - Closes #161 - Closes #181 - Closes #191 - Closes #199 - Closes #201 - Closes #202 - Closes #203 - Closes #204 - Closes #205 - Closes #206 - Closes #208 - Closes #211 - Closes #212 - Closes #213 - Closes #214 - Closes #216 - Closes #219 - Closes #220 - Closes #221 - Closes #222 - Closes #224 - Closes #225 - Closes #226 - Closes #228 - Closes #229 - Closes #230 - Closes #231 - Closes #232 - Closes #233 - Closes #234 - Closes #235 - Closes #236 ✨ This PR was created with help from Hikari~ 🌸 Reviewed-on: #238 Co-authored-by: Hikari <hikari@nhcarrigan.com> Co-committed-by: Hikari <hikari@nhcarrigan.com>
464 lines
15 KiB
TypeScript
464 lines
15 KiB
TypeScript
/**
|
||
* @file Prestige panel component for ascending and purchasing runestone upgrades.
|
||
* @copyright nhcarrigan
|
||
* @license Naomi's Public License
|
||
* @author Naomi Carrigan
|
||
*/
|
||
/* eslint-disable max-lines-per-function -- Complex component with many render paths */
|
||
/* eslint-disable complexity -- Many conditional render paths */
|
||
/* eslint-disable max-lines -- Large panel with prestige and shop tabs */
|
||
/* eslint-disable max-statements -- Prestige panel manages many local state variables */
|
||
import { useState, type JSX } from "react";
|
||
import { prestige } from "../../api/client.js";
|
||
import { useGame } from "../../context/gameContext.js";
|
||
import {
|
||
PRESTIGE_UPGRADE_CATEGORY_LABELS,
|
||
PRESTIGE_UPGRADES,
|
||
} from "../../data/prestigeUpgrades.js";
|
||
import {
|
||
computeProjectedRunestones,
|
||
} from "../../engine/tick.js";
|
||
import { cdnImage } from "../../utils/cdn.js";
|
||
import { sendNotification } from "../../utils/notification.js";
|
||
import { playSound } from "../../utils/sound.js";
|
||
import type { PrestigeUpgradeCategory } from "@elysium/types";
|
||
|
||
const baseThreshold = 1_000_000;
|
||
|
||
/**
|
||
* Calculates the prestige threshold for a given prestige count.
|
||
* Mirrors the server formula: BASE * (count + 1)^2.5.
|
||
* @param prestigeCount - The current prestige count.
|
||
* @returns The required gold to prestige.
|
||
*/
|
||
const calculateThreshold = (prestigeCount: number): number => {
|
||
return baseThreshold * Math.pow(prestigeCount + 1, 2.5);
|
||
};
|
||
|
||
/**
|
||
* Calculates the production multiplier for a given prestige count.
|
||
* @param prestigeCount - The number of times the player has prestiged.
|
||
* @returns The compounding multiplier applied to all income sources.
|
||
*/
|
||
const calculateProductionMultiplier = (prestigeCount: number): number => {
|
||
return Math.pow(1.3, prestigeCount);
|
||
};
|
||
|
||
const categoryOrder: Array<PrestigeUpgradeCategory> = [
|
||
"income",
|
||
"click",
|
||
"essence",
|
||
"crystals",
|
||
"runestones",
|
||
"utility",
|
||
];
|
||
|
||
/**
|
||
* Renders the prestige panel with ascension and runestone shop tabs.
|
||
* @returns The JSX element.
|
||
*/
|
||
const PrestigePanel = (): JSX.Element => {
|
||
const {
|
||
state,
|
||
reloadSilent,
|
||
formatInteger,
|
||
formatNumber,
|
||
buyPrestigeUpgrade,
|
||
enableNotifications,
|
||
enableSounds,
|
||
toggleAutoAdventurer,
|
||
toggleAutoPrestige,
|
||
toggleAutoPrestigeMaxRunestones,
|
||
triggerPrestigeToast,
|
||
} = useGame();
|
||
const [ isPending, setIsPending ] = useState(false);
|
||
const [ result, setResult ] = useState<{
|
||
runestones: number;
|
||
count: number;
|
||
milestoneRunestones: number;
|
||
} | null>(null);
|
||
const [ prestigeError, setPrestigeError ] = useState<string | null>(null);
|
||
const [ buyingId, setBuyingId ] = useState<string | null>(null);
|
||
const [ activeTab, setActiveTab ] = useState<"prestige" | "shop">("prestige");
|
||
|
||
if (state === null) {
|
||
return (
|
||
<section className="panel">
|
||
<p>{"Loading..."}</p>
|
||
</section>
|
||
);
|
||
}
|
||
|
||
const { autoAdventurer, prestige: prestigeData, player } = state;
|
||
const threshold = calculateThreshold(prestigeData.count);
|
||
const isEligible = player.totalGoldEarned >= threshold;
|
||
const runestonePreview = computeProjectedRunestones(state);
|
||
const nextMultiplier = calculateProductionMultiplier(prestigeData.count + 1);
|
||
const baseRunestones = Math.min(
|
||
Math.floor(Math.cbrt(player.totalGoldEarned / threshold)) * 15,
|
||
200,
|
||
);
|
||
const isAtMaxRunestones = baseRunestones >= 200;
|
||
|
||
async function handlePrestige(): Promise<void> {
|
||
setIsPending(true);
|
||
setPrestigeError(null);
|
||
try {
|
||
const data = await prestige({});
|
||
setResult({
|
||
count: data.newPrestigeCount,
|
||
milestoneRunestones: data.milestoneRunestones,
|
||
runestones: data.runestones,
|
||
});
|
||
triggerPrestigeToast();
|
||
if (enableSounds) {
|
||
playSound("prestige");
|
||
}
|
||
if (enableNotifications) {
|
||
sendNotification(
|
||
"⭐ Prestige!",
|
||
`You've reached prestige level ${data.newPrestigeCount.toString()}!`,
|
||
);
|
||
}
|
||
await reloadSilent();
|
||
} catch (error_: unknown) {
|
||
setPrestigeError(
|
||
error_ instanceof Error
|
||
? error_.message
|
||
: "Prestige failed",
|
||
);
|
||
} finally {
|
||
setIsPending(false);
|
||
}
|
||
}
|
||
|
||
async function handleBuyUpgrade(upgradeId: string): Promise<void> {
|
||
setBuyingId(upgradeId);
|
||
try {
|
||
await buyPrestigeUpgrade(upgradeId);
|
||
} finally {
|
||
setBuyingId(null);
|
||
}
|
||
}
|
||
|
||
const upgradesByCategory = categoryOrder.map((categoryId) => {
|
||
const label = PRESTIGE_UPGRADE_CATEGORY_LABELS[categoryId] ?? categoryId;
|
||
const upgrades = PRESTIGE_UPGRADES.filter((upgrade) => {
|
||
return upgrade.category === categoryId;
|
||
});
|
||
return { categoryId, label, upgrades };
|
||
});
|
||
|
||
function handlePrestigeClick(): void {
|
||
void handlePrestige();
|
||
}
|
||
|
||
function handleAutoAdventurerToggle(): void {
|
||
toggleAutoAdventurer();
|
||
}
|
||
|
||
function handleAutoPrestigeToggle(): void {
|
||
toggleAutoPrestige();
|
||
}
|
||
|
||
function handleAutoPrestigeMaxRunestonesToggle(): void {
|
||
toggleAutoPrestigeMaxRunestones();
|
||
}
|
||
|
||
function handlePrestigeTabClick(): void {
|
||
setActiveTab("prestige");
|
||
}
|
||
|
||
function handleShopTabClick(): void {
|
||
setActiveTab("shop");
|
||
}
|
||
|
||
const progressRatio = player.totalGoldEarned / threshold;
|
||
const progressPct = (progressRatio * 100).toFixed(1);
|
||
|
||
return (
|
||
<section className="panel prestige-panel">
|
||
<h2>{"⭐ Prestige"}</h2>
|
||
|
||
<div className="prestige-tabs">
|
||
<button
|
||
className={`prestige-tab ${activeTab === "prestige"
|
||
? "active"
|
||
: ""}`}
|
||
onClick={handlePrestigeTabClick}
|
||
type="button"
|
||
>
|
||
{"Ascend"}
|
||
</button>
|
||
<button
|
||
className={`prestige-tab ${activeTab === "shop"
|
||
? "active"
|
||
: ""}`}
|
||
onClick={handleShopTabClick}
|
||
type="button"
|
||
>
|
||
{"🔮 Runestone Shop ("}
|
||
{formatInteger(prestigeData.runestones)}
|
||
{" stones)"}
|
||
</button>
|
||
</div>
|
||
|
||
{activeTab === "prestige"
|
||
&& <>
|
||
<p>
|
||
{"Prestige resets your progress but grants "}
|
||
<strong>{"Runestones"}</strong>
|
||
{"— permanent currency used for powerful upgrades."}
|
||
{" Each prestige multiplies your global production by ×1.3"}
|
||
{" (compounding each run)."}
|
||
</p>
|
||
|
||
<div className="prestige-status">
|
||
<p>
|
||
{"Total gold this run: "}
|
||
<strong>{formatNumber(player.totalGoldEarned)}</strong>
|
||
</p>
|
||
<p>
|
||
{"Required to prestige: "}
|
||
<strong>{formatNumber(threshold)}</strong>
|
||
</p>
|
||
<p>
|
||
{"Prestige count: "}
|
||
<strong>{prestigeData.count}</strong>
|
||
</p>
|
||
<p>
|
||
{"Current production multiplier: "}
|
||
<strong>
|
||
{"×"}
|
||
{prestigeData.productionMultiplier.toFixed(2)}
|
||
</strong>
|
||
</p>
|
||
<p>
|
||
{"After next prestige: "}
|
||
<strong>
|
||
{"×"}
|
||
{nextMultiplier.toFixed(2)}
|
||
</strong>
|
||
</p>
|
||
<p>
|
||
{"Runestones: "}
|
||
<strong>{formatInteger(prestigeData.runestones)}</strong>
|
||
</p>
|
||
{isEligible
|
||
? <p className="runestone-preview">
|
||
{"Runestones on prestige: "}
|
||
<strong>
|
||
{"+"}
|
||
{formatInteger(runestonePreview)}
|
||
</strong>
|
||
{isAtMaxRunestones
|
||
? <span className="runestone-max-badge">{" ⚡ MAX"}</span>
|
||
: null
|
||
}
|
||
</p>
|
||
: null}
|
||
{isEligible && !isAtMaxRunestones
|
||
? <p className="runestone-progress-hint">
|
||
{"Earn more gold to increase your runestone yield "
|
||
+ "(capped at ×14³ the prestige threshold)."}
|
||
</p>
|
||
: null}
|
||
{isEligible
|
||
? null
|
||
: <p className="prestige-progress">
|
||
{"Progress: "}
|
||
{formatNumber(player.totalGoldEarned)}
|
||
{" / "}
|
||
{formatNumber(threshold)}
|
||
{" ("}
|
||
{progressPct}
|
||
{"%"}
|
||
{")"}
|
||
</p>
|
||
}
|
||
</div>
|
||
|
||
{isEligible
|
||
? <div className="prestige-form">
|
||
<p>{"You are ready to prestige!"}</p>
|
||
<button
|
||
className="prestige-button"
|
||
disabled={isPending}
|
||
onClick={handlePrestigeClick}
|
||
type="button"
|
||
>
|
||
{isPending
|
||
? "Ascending..."
|
||
: `✨ Ascend (+${formatInteger(runestonePreview)} Runestones)`}
|
||
</button>
|
||
{prestigeError === null
|
||
? null
|
||
: <p className="error">{prestigeError}</p>
|
||
}
|
||
{result === null
|
||
? null
|
||
: <p className="success">
|
||
{"Ascended to Prestige "}
|
||
{result.count}
|
||
{"! Earned "}
|
||
{formatInteger(result.runestones)}
|
||
{" Runestones."}
|
||
{result.milestoneRunestones > 0
|
||
&& <>
|
||
{" 🎉 Milestone bonus: +"}
|
||
{formatInteger(result.milestoneRunestones)}
|
||
{" Runestones!"}
|
||
</>
|
||
}
|
||
</p>
|
||
}
|
||
</div>
|
||
: <p className="prestige-locked">
|
||
{"Earn "}
|
||
{formatNumber(threshold - player.totalGoldEarned)}
|
||
{" more gold to unlock prestige."}
|
||
</p>
|
||
}
|
||
</>
|
||
}
|
||
|
||
{activeTab === "shop"
|
||
&& <div className="runestone-shop">
|
||
<p className="shop-balance">
|
||
{"Balance: "}
|
||
<strong>
|
||
{formatInteger(prestigeData.runestones)}
|
||
{" Runestones"}
|
||
</strong>
|
||
</p>
|
||
|
||
{upgradesByCategory.map(({ categoryId, label, upgrades }) => {
|
||
return (
|
||
<div className="shop-category" key={categoryId}>
|
||
<h3>{label}</h3>
|
||
<div className="shop-upgrades">
|
||
{upgrades.map((upgrade) => {
|
||
const purchased = prestigeData.purchasedUpgradeIds.includes(
|
||
upgrade.id,
|
||
);
|
||
const canAfford
|
||
= prestigeData.runestones >= upgrade.runestonesCost;
|
||
const isLoading = buyingId === upgrade.id;
|
||
|
||
const isAutoAdventurerToggle
|
||
= upgrade.id === "auto_adventurer" && purchased;
|
||
const autoAdventurerEnabled = autoAdventurer ?? false;
|
||
const isAutoPrestigeToggle
|
||
= upgrade.id === "auto_prestige" && purchased;
|
||
const autoPrestigeEnabled
|
||
= prestigeData.autoPrestigeEnabled ?? false;
|
||
const autoPrestigeMaxRunestonesOnly
|
||
= prestigeData.autoPrestigeMaxRunestonesOnly ?? false;
|
||
|
||
function handleBuyClick(): void {
|
||
void handleBuyUpgrade(upgrade.id);
|
||
}
|
||
|
||
return (
|
||
<div
|
||
className={`shop-upgrade-card ${
|
||
purchased
|
||
? "purchased"
|
||
: ""
|
||
} ${!canAfford && !purchased
|
||
? "unaffordable"
|
||
: ""}`}
|
||
key={upgrade.id}
|
||
>
|
||
<img
|
||
alt={upgrade.name}
|
||
className="card-thumbnail"
|
||
src={cdnImage("prestige-upgrades", upgrade.id)}
|
||
/>
|
||
<div className="shop-upgrade-info">
|
||
<h4>{upgrade.name}</h4>
|
||
<p>{upgrade.description}</p>
|
||
<p className="upgrade-cost">
|
||
{purchased
|
||
? "✅ Purchased"
|
||
: `🔮 ${formatInteger(upgrade.runestonesCost)} Runestones`}
|
||
</p>
|
||
</div>
|
||
{isAutoAdventurerToggle
|
||
? <button
|
||
className={`auto-prestige-toggle ${
|
||
autoAdventurerEnabled
|
||
? "enabled"
|
||
: "disabled"
|
||
}`}
|
||
onClick={handleAutoAdventurerToggle}
|
||
type="button"
|
||
>
|
||
{autoAdventurerEnabled
|
||
? "⚡ Auto ON"
|
||
: "⏸ Auto OFF"}
|
||
</button>
|
||
: null}
|
||
{isAutoPrestigeToggle
|
||
? <>
|
||
<button
|
||
className={`auto-prestige-toggle ${
|
||
autoPrestigeEnabled
|
||
? "enabled"
|
||
: "disabled"
|
||
}`}
|
||
onClick={handleAutoPrestigeToggle}
|
||
type="button"
|
||
>
|
||
{autoPrestigeEnabled
|
||
? "⚡ Auto ON"
|
||
: "⏸ Auto OFF"}
|
||
</button>
|
||
{autoPrestigeEnabled
|
||
? <button
|
||
className={`auto-prestige-toggle ${
|
||
autoPrestigeMaxRunestonesOnly
|
||
? "enabled"
|
||
: "disabled"
|
||
}`}
|
||
onClick={handleAutoPrestigeMaxRunestonesToggle}
|
||
title="Only fire at max runestone yield"
|
||
type="button"
|
||
>
|
||
{autoPrestigeMaxRunestonesOnly
|
||
? "⚡ Max Runes Only"
|
||
: "⏸ Max Runes OFF"}
|
||
</button>
|
||
: null}
|
||
</>
|
||
: null}
|
||
{purchased
|
||
? null
|
||
: <button
|
||
className="buy-upgrade-button"
|
||
disabled={
|
||
!canAfford || isLoading || buyingId !== null
|
||
}
|
||
onClick={handleBuyClick}
|
||
type="button"
|
||
>
|
||
{isLoading
|
||
? "Buying..."
|
||
: "Buy"}
|
||
</button>
|
||
}
|
||
</div>
|
||
);
|
||
})}
|
||
</div>
|
||
</div>
|
||
);
|
||
})}
|
||
</div>
|
||
}
|
||
</section>
|
||
);
|
||
};
|
||
|
||
export { PrestigePanel };
|