modified: index.html

modified:   js/config.js
	modified:   js/entities.js
	modified:   js/game.js
	modified:   js/helpers.js
	modified:   js/ui.js
This commit is contained in:
xbl
2026-02-17 23:23:42 +01:00
parent 8d5b673044
commit 0bbfa7261e
6 changed files with 159 additions and 16 deletions

View File

@ -102,6 +102,10 @@ a1.7 1.7 0 0 0-1.5 1z"></path>
<div id="ktype-level"></div> <div id="ktype-level"></div>
<button id="ktype-upgrade-btn" class="upgrade-btn" disabled>Upgrade</button> <button id="ktype-upgrade-btn" class="upgrade-btn" disabled>Upgrade</button>
</div> </div>
<div class="upgrade-row">
<div id="gtype-level"></div>
<button id="gtype-upgrade-btn" class="upgrade-btn" disabled>Upgrade</button>
</div>
</div> </div>
<!-- Leaderboard Toggle --> <!-- Leaderboard Toggle -->

View File

@ -37,8 +37,8 @@ var CONFIG = {
PLANET_BASE_COST: 1e24, PLANET_BASE_COST: 1e24,
GIANT_BASE_COST: 1e27, GIANT_BASE_COST: 1e27,
MTYPE_BASE_COST: 1e29, MTYPE_BASE_COST: 1e29,
KTYPE_BASE_COST: 1e31, KTYPE_BASE_COST: 1e30,
GTYPE_BASE_COST: 1e33, GTYPE_BASE_COST: 1e31,
// Upgrade scaling // Upgrade scaling
UPGRADE_COST_MULTIPLIER_ASTER: 1.25, UPGRADE_COST_MULTIPLIER_ASTER: 1.25,
@ -75,6 +75,17 @@ var CONFIG = {
gtype: [1.59e30, 2.07e30] // K-Type: 0.8-1.04 M☉ gtype: [1.59e30, 2.07e30] // K-Type: 0.8-1.04 M☉
}, },
// Visual Sizes
VISUAL_SIZE_SMALL: 1,
VISUAL_SIZE_MEDIUM: 2,
VISUAL_SIZE_LARGE: 3,
VISUAL_SIZE_COMET: 3,
VISUAL_SIZE_PLANT: 5,
VISUAL_SIZE_GIANT: 8,
VISUAL_SIZE_MTYPE: 12,
VISUAL_SIZE_KTYPE: 14,
VISUAL_SIZE_GTYPE: 16,
// Planet color schemes // Planet color schemes
PLANET_COLORS: [ PLANET_COLORS: [
{ light: '#6b7c87', mid: '#3d4d5a', dark: '#1f2a33' }, { light: '#6b7c87', mid: '#3d4d5a', dark: '#1f2a33' },

View File

@ -66,11 +66,11 @@ function Asteroid(type, blackHole, canvas) {
} }
Asteroid.prototype.initializeTypeSpecificProperties = function() { Asteroid.prototype.initializeTypeSpecificProperties = function() {
// Visual size (unchanged) // Visual size
if (this.type === 'comet') { if (this.type === 'comet') {
this.size = 3 + Math.random() * 1; this.size = CONFIG.VISUAL_SIZE_COMET + Math.random() * 1;
} else if (this.type === 'mtype') { } else if (this.type === 'mtype') {
this.size = 15 + Math.random() * 3; this.size = CONFIG.VISUAL_SIZE_MTYPE + Math.random() * 2;
this.orbitSpeed *= 0.2; this.orbitSpeed *= 0.2;
this.decayRate *= 0.3; this.decayRate *= 0.3;
this.planetColors = CONFIG.MTYPE_COLORS[Math.floor(Math.random() * CONFIG.MTYPE_COLORS.length)]; this.planetColors = CONFIG.MTYPE_COLORS[Math.floor(Math.random() * CONFIG.MTYPE_COLORS.length)];
@ -100,7 +100,7 @@ Asteroid.prototype.initializeTypeSpecificProperties = function() {
}); });
} }
} else if (this.type === 'ktype') { } else if (this.type === 'ktype') {
this.size = 16 + Math.random() * 4; this.size = CONFIG.VISUAL_SIZE_KTYPE + Math.random() * 2;
this.orbitSpeed *= 0.15; this.orbitSpeed *= 0.15;
this.decayRate *= 0.22; this.decayRate *= 0.22;
this.planetColors = CONFIG.KTYPE_COLORS[Math.floor(Math.random() * CONFIG.KTYPE_COLORS.length)]; this.planetColors = CONFIG.KTYPE_COLORS[Math.floor(Math.random() * CONFIG.KTYPE_COLORS.length)];
@ -138,6 +138,64 @@ Asteroid.prototype.initializeTypeSpecificProperties = function() {
}); });
} }
// Sparkles — simulating spikes toward the viewer
this.sparkles = [];
var sparkleCount = 4 + Math.floor(Math.random() * 4); // 48
for (var k = 0; k < sparkleCount; k++) {
var angle = Math.random() * Math.PI * 2;
var dist = Math.random() * this.size * 0.75;
this.sparkles.push({
x: Math.cos(angle) * dist,
y: Math.sin(angle) * dist,
dist: dist, // ← add
angle: angle, // ← add
wanderSpeed: (Math.random() - 0.5) * 0.008, // ← add, slow angular drift
size: 0.4 + Math.random() * 0.8,
phase: Math.random() * Math.PI * 2,
speed: 0.05 + Math.random() * 1.8,
armLen: 0.4 + Math.random() * 8,
armWidth: 0.2 + Math.random() * 0.4
});
}
} else if (this.type === 'gtype') {
this.size = CONFIG.VISUAL_SIZE_GTYPE + Math.random() * 2;
this.orbitSpeed *= 0.13;
this.decayRate *= 0.20;
this.planetColors = CONFIG.GTYPE_COLORS[Math.floor(Math.random() * CONFIG.GTYPE_COLORS.length)];
// Surface granulation cells
this.granules = [];
var granCount = 8 + Math.floor(Math.random() * 10);
for (var i = 0; i < granCount; i++) {
var angle = Math.random() * Math.PI * 2;
var dist = Math.random() * this.size * 0.8;
this.granules.push({
x: Math.cos(angle) * dist,
y: Math.sin(angle) * dist,
radius: 1.5 + Math.random() * this.size * 0.2,
phase: Math.random() * Math.PI * 2
});
}
// Spikes — each one fully independent
this.spikes = [];
var spikeCount = 22 + Math.floor(Math.random() * 11); // 2233
for (var s = 0; s < spikeCount; s++) {
this.spikes.push({
baseAngle: (s / spikeCount) * Math.PI * 2 + (Math.random() - 0.5) * 0.4,
rotSpeed: (Math.random() - 0.5) * 0.015, // some drift CW, some CCW
lenBase: 0.35 + Math.random() * 0.45, // 0.350.80 × size
lenAmp: 0.05 + Math.random() * 0.20, // oscillation range
lenSpeed: 0.3 + Math.random() * 1.0, // oscillation speed
lenPhase: Math.random() * Math.PI * 2,
widthBase: 0.1 + Math.random() * 0.045, // tip width
alphaBase: 0.10 + Math.random() * 0.14,
alphaAmp: 0.03 + Math.random() * 0.07,
alphaPhase: Math.random() * Math.PI * 2,
alphaSpeed: 0.4 + Math.random() * 0.9
});
}
// Sparkles — simulating spikes toward the viewer // Sparkles — simulating spikes toward the viewer
this.sparkles = []; this.sparkles = [];
var sparkleCount = 4 + Math.floor(Math.random() * 4); // 48 var sparkleCount = 4 + Math.floor(Math.random() * 4); // 48
@ -158,7 +216,7 @@ Asteroid.prototype.initializeTypeSpecificProperties = function() {
}); });
} }
} else if (this.type === 'giant') { } else if (this.type === 'giant') {
this.size = 10 + Math.random() * 5; this.size = CONFIG.VISUAL_SIZE_GIANT + Math.random() * 5;
this.orbitSpeed *= 0.3; this.orbitSpeed *= 0.3;
this.decayRate *= 0.4; this.decayRate *= 0.4;
this.planetColors = CONFIG.GIANT_COLORS[Math.floor(Math.random() * CONFIG.GIANT_COLORS.length)]; this.planetColors = CONFIG.GIANT_COLORS[Math.floor(Math.random() * CONFIG.GIANT_COLORS.length)];
@ -174,16 +232,16 @@ Asteroid.prototype.initializeTypeSpecificProperties = function() {
}); });
} }
} else if (this.type === 'planet') { } else if (this.type === 'planet') {
this.size = 6 + Math.random() * 4; this.size = CONFIG.VISUAL_SIZE_PLANT + Math.random() * 4;
this.orbitSpeed *= 0.5; this.orbitSpeed *= 0.5;
this.decayRate *= 0.6; this.decayRate *= 0.6;
this.planetColors = CONFIG.PLANET_COLORS[Math.floor(Math.random() * CONFIG.PLANET_COLORS.length)]; this.planetColors = CONFIG.PLANET_COLORS[Math.floor(Math.random() * CONFIG.PLANET_COLORS.length)];
} else if (this.type === 'large') { } else if (this.type === 'large') {
this.size = 3 + Math.random() * 1; this.size = CONFIG.VISUAL_SIZE_LARGE + Math.random() * 1;
} else if (this.type === 'medium') { } else if (this.type === 'medium') {
this.size = 2 + Math.random() * 1; this.size = CONFIG.VISUAL_SIZE_MEDIUM + Math.random() * 1;
} else { } else {
this.size = 1 + Math.random() * 1; this.size = CONFIG.VISUAL_SIZE_SMALL + Math.random() * 1;
} }
// Physical mass (kg) // Physical mass (kg)
@ -206,6 +264,8 @@ Asteroid.prototype.draw = function(ctx) {
this.drawMType(ctx, performance.now() * 0.002); this.drawMType(ctx, performance.now() * 0.002);
} else if (this.type === 'ktype') { } else if (this.type === 'ktype') {
this.drawKType(ctx, performance.now() * 0.002); this.drawKType(ctx, performance.now() * 0.002);
} else if (this.type === 'gtype') {
this.drawKType(ctx, performance.now() * 0.002);
} else if (this.type === 'giant') { } else if (this.type === 'giant') {
// Initialize per-giant tilt if not already done // Initialize per-giant tilt if not already done
if (this._ringTiltBase === undefined) { if (this._ringTiltBase === undefined) {

View File

@ -20,12 +20,15 @@ var Game = (function() {
mtypeUpgradeCost: CONFIG.MTYPE_BASE_COST, mtypeUpgradeCost: CONFIG.MTYPE_BASE_COST,
ktypeUpgradeLevel: 0, ktypeUpgradeLevel: 0,
ktypeUpgradeCost: CONFIG.KTYPE_BASE_COST, ktypeUpgradeCost: CONFIG.KTYPE_BASE_COST,
gtypeUpgradeLevel: 0,
gtypeUpgradeCost: CONFIG.GTYPE_BASE_COST,
currentAsteroidSpawnInterval: CONFIG.BASE_ASTEROID_SPAWN_INTERVAL, currentAsteroidSpawnInterval: CONFIG.BASE_ASTEROID_SPAWN_INTERVAL,
currentCometSpawnInterval: CONFIG.BASE_COMET_SPAWN_INTERVAL, currentCometSpawnInterval: CONFIG.BASE_COMET_SPAWN_INTERVAL,
currentPlanetSpawnInterval: CONFIG.BASE_PLANET_SPAWN_INTERVAL, currentPlanetSpawnInterval: CONFIG.BASE_PLANET_SPAWN_INTERVAL,
currentGiantSpawnInterval: CONFIG.BASE_GIANT_SPAWN_INTERVAL, currentGiantSpawnInterval: CONFIG.BASE_GIANT_SPAWN_INTERVAL,
currentMtypeSpawnInterval: CONFIG.BASE_MTYPE_SPAWN_INTERVAL, currentMtypeSpawnInterval: CONFIG.BASE_MTYPE_SPAWN_INTERVAL,
currentKtypeSpawnInterval: CONFIG.BASE_KTYPE_SPAWN_INTERVAL, currentKtypeSpawnInterval: CONFIG.BASE_KTYPE_SPAWN_INTERVAL,
currentGtypeSpawnInterval: CONFIG.BASE_GTYPE_SPAWN_INTERVAL,
asteroidSpawnCount: 0, asteroidSpawnCount: 0,
lastAsteroidSpawn: Date.now(), lastAsteroidSpawn: Date.now(),
lastCometSpawn: Date.now(), lastCometSpawn: Date.now(),
@ -33,6 +36,7 @@ var Game = (function() {
lastGiantSpawn: Date.now(), lastGiantSpawn: Date.now(),
lastMtypeSpawn: Date.now(), lastMtypeSpawn: Date.now(),
lastKtypeSpawn: Date.now(), lastKtypeSpawn: Date.now(),
lastGtypeSpawn: Date.now(),
sM: 0, sT: Date.now(), sM: 0, sT: Date.now(),
mM: 0, mT: Date.now(), mM: 0, mT: Date.now(),
lM: 0, lT: Date.now(), lM: 0, lT: Date.now(),
@ -45,7 +49,8 @@ var Game = (function() {
planetUnlocked: false, planetUnlocked: false,
giantUnlocked: false, giantUnlocked: false,
mtypeUnlocked: false, mtypeUnlocked: false,
ktypeUnlocked: false ktypeUnlocked: false,
gtypeUnlocked: false
}; };
var blackHole; var blackHole;
@ -93,7 +98,8 @@ var Game = (function() {
planet: handlePlanetUpgrade, planet: handlePlanetUpgrade,
giant: handleGiantUpgrade, giant: handleGiantUpgrade,
mtype: handleMtypeUpgrade, mtype: handleMtypeUpgrade,
ktype: handleKtypeUpgrade ktype: handleKtypeUpgrade,
gtype: handleGtypeUpgrade
}); });
Server.init() Server.init()
@ -164,6 +170,7 @@ var Game = (function() {
state.giantUpgradeLevel = savedState.giantUpgradeLevel || 0; state.giantUpgradeLevel = savedState.giantUpgradeLevel || 0;
state.mtypeUpgradeLevel = savedState.mtypeUpgradeLevel || 0; state.mtypeUpgradeLevel = savedState.mtypeUpgradeLevel || 0;
state.ktypeUpgradeLevel = savedState.ktypeUpgradeLevel || 0; state.ktypeUpgradeLevel = savedState.ktypeUpgradeLevel || 0;
state.gtypeUpgradeLevel = savedState.gtypeUpgradeLevel || 0;
// Derive unlock states from levels // Derive unlock states from levels
state.cometUnlocked = state.asteroidUpgradeLevel >= 20; state.cometUnlocked = state.asteroidUpgradeLevel >= 20;
@ -171,6 +178,7 @@ var Game = (function() {
state.giantUnlocked = state.planetUpgradeLevel >= 10; state.giantUnlocked = state.planetUpgradeLevel >= 10;
state.mtypeUnlocked = state.giantUpgradeLevel >= 5; state.mtypeUnlocked = state.giantUpgradeLevel >= 5;
state.ktypeUnlocked = state.mtypeUpgradeLevel >= 5; state.ktypeUnlocked = state.mtypeUpgradeLevel >= 5;
state.gtypeUnlocked = state.ktypeUpgradeLevel >= 5;
// Derive costs and intervals from levels + CONFIG // Derive costs and intervals from levels + CONFIG
recalculateUpgradeCosts(); recalculateUpgradeCosts();
@ -183,6 +191,7 @@ var Game = (function() {
state.lastGiantSpawn = savedState.lastGiantSpawn || Date.now(); state.lastGiantSpawn = savedState.lastGiantSpawn || Date.now();
state.lastMtypeSpawn = savedState.lastMtypeSpawn || Date.now(); state.lastMtypeSpawn = savedState.lastMtypeSpawn || Date.now();
state.lastKtypeSpawn = savedState.lastKtypeSpawn || Date.now(); state.lastKtypeSpawn = savedState.lastKtypeSpawn || Date.now();
state.lastGtypeSpawn = savedState.lastGtypeSpawn || Date.now();
// Consumption rates // Consumption rates
state.sM = savedState.sM || 0; state.sM = savedState.sM || 0;
@ -260,6 +269,8 @@ var Game = (function() {
(1 + state.mtypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_MTYPE); (1 + state.mtypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_MTYPE);
state.currentKtypeSpawnInterval = CONFIG.BASE_KTYPE_SPAWN_INTERVAL / state.currentKtypeSpawnInterval = CONFIG.BASE_KTYPE_SPAWN_INTERVAL /
(1 + state.ktypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_KTYPE); (1 + state.ktypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_KTYPE);
state.currentGtypeSpawnInterval = CONFIG.BASE_GTYPE_SPAWN_INTERVAL /
(1 + state.gtypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_GTYPE);
} }
function recalculateUpgradeCosts() { function recalculateUpgradeCosts() {
@ -269,6 +280,7 @@ var Game = (function() {
state.giantUpgradeCost = Math.floor(CONFIG.GIANT_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_GIANT, state.giantUpgradeLevel)); state.giantUpgradeCost = Math.floor(CONFIG.GIANT_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_GIANT, state.giantUpgradeLevel));
state.mtypeUpgradeCost = Math.floor(CONFIG.MTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_MTYPE, state.mtypeUpgradeLevel)); state.mtypeUpgradeCost = Math.floor(CONFIG.MTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_MTYPE, state.mtypeUpgradeLevel));
state.ktypeUpgradeCost = Math.floor(CONFIG.KTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_KTYPE, state.ktypeUpgradeLevel)); state.ktypeUpgradeCost = Math.floor(CONFIG.KTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_KTYPE, state.ktypeUpgradeLevel));
state.gtypeUpgradeCost = Math.floor(CONFIG.GTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_GTYPE, state.gtypeUpgradeLevel));
} }
function handleAsteroidUpgrade() { function handleAsteroidUpgrade() {
@ -374,6 +386,27 @@ var Game = (function() {
state.ktypeUpgradeCost = Math.floor( state.ktypeUpgradeCost = Math.floor(
CONFIG.KTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_KTYPE, state.ktypeUpgradeLevel) CONFIG.KTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_KTYPE, state.ktypeUpgradeLevel)
); );
// Unlock K-Type at m-type level 5
if (state.ktypeUpgradeLevel >= 5 && !state.gtypeUnlocked) {
state.gtypeUnlocked = true;
state.lastGtypeSpawn = Date.now();
showUnlockNotification('gtype');
asteroids.push(new Asteroid('gtype', blackHole, canvas));
}
updateSpawnIntervals();
UI.update(state, CONFIG);
}
}
function handleGtypeUpgrade() {
if (state.totalMassConsumed >= state.gtypeUpgradeCost) {
state.totalMassConsumed -= state.gtypeUpgradeCost;
state.gtypeUpgradeLevel++;
state.gtypeUpgradeCost = Math.floor(
CONFIG.GTYPE_BASE_COST * Math.pow(CONFIG.UPGRADE_COST_MULTIPLIER_GTYPE, state.gtypeUpgradeLevel)
);
updateSpawnIntervals(); updateSpawnIntervals();
UI.update(state, CONFIG); UI.update(state, CONFIG);
} }
@ -432,6 +465,12 @@ var Game = (function() {
asteroids.push(new Asteroid('ktype', blackHole, canvas)); asteroids.push(new Asteroid('ktype', blackHole, canvas));
} }
} }
if (state.gtypeUnlocked) {
if (currentTime - state.lastGtypeSpawn > state.currentGtypeSpawnInterval) {
state.lastGtypeSpawn = currentTime;
asteroids.push(new Asteroid('gtype', blackHole, canvas));
}
}
} }

View File

@ -222,7 +222,8 @@ function showUnlockNotification(type) {
planet: 'Planets Unlocked', planet: 'Planets Unlocked',
giant: 'Gas & Ice Giants Unlocked', giant: 'Gas & Ice Giants Unlocked',
mtype: 'M-Type Stars Unlocked', mtype: 'M-Type Stars Unlocked',
ktype: 'K-Type Stars Unlocked' ktype: 'K-Type Stars Unlocked',
gtype: 'G-Type Stars Unlocked'
}; };
showNotification({ showNotification({

View File

@ -15,12 +15,14 @@ var UI = {
giantLevel: document.getElementById('giant-level'), giantLevel: document.getElementById('giant-level'),
mtypeLevel: document.getElementById('mtype-level'), mtypeLevel: document.getElementById('mtype-level'),
ktypeLevel: document.getElementById('ktype-level'), ktypeLevel: document.getElementById('ktype-level'),
gtypeLevel: document.getElementById('gtype-level'),
asteroidUpgradeBtn: document.getElementById('asteroid-upgrade-btn'), asteroidUpgradeBtn: document.getElementById('asteroid-upgrade-btn'),
cometUpgradeBtn: document.getElementById('comet-upgrade-btn'), cometUpgradeBtn: document.getElementById('comet-upgrade-btn'),
planetUpgradeBtn: document.getElementById('planet-upgrade-btn'), planetUpgradeBtn: document.getElementById('planet-upgrade-btn'),
giantUpgradeBtn: document.getElementById('giant-upgrade-btn'), giantUpgradeBtn: document.getElementById('giant-upgrade-btn'),
mtypeUpgradeBtn: document.getElementById('mtype-upgrade-btn'), mtypeUpgradeBtn: document.getElementById('mtype-upgrade-btn'),
ktypeUpgradeBtn: document.getElementById('ktype-upgrade-btn'), ktypeUpgradeBtn: document.getElementById('ktype-upgrade-btn'),
gtypeUpgradeBtn: document.getElementById('gtype-upgrade-btn'),
gearIcon: document.getElementById('gear-icon'), gearIcon: document.getElementById('gear-icon'),
settingsMenu: document.getElementById('settings-menu'), settingsMenu: document.getElementById('settings-menu'),
resetBtn: document.getElementById('reset-btn'), resetBtn: document.getElementById('reset-btn'),
@ -205,6 +207,7 @@ var UI = {
this.elements.giantUpgradeBtn.addEventListener('click', handlers.giant); this.elements.giantUpgradeBtn.addEventListener('click', handlers.giant);
this.elements.mtypeUpgradeBtn.addEventListener('click', handlers.mtype); this.elements.mtypeUpgradeBtn.addEventListener('click', handlers.mtype);
this.elements.ktypeUpgradeBtn.addEventListener('click', handlers.ktype); this.elements.ktypeUpgradeBtn.addEventListener('click', handlers.ktype);
this.elements.gtypeUpgradeBtn.addEventListener('click', handlers.gtype);
}, },
update: function(gameState, config) { update: function(gameState, config) {
@ -305,13 +308,22 @@ var UI = {
this.elements.ktypeLevel.parentElement.style.display = 'none'; this.elements.ktypeLevel.parentElement.style.display = 'none';
} }
// Only show gtype upgrade if unlocked
if (gameState.gtypeUnlocked) {
this.updateGtypeUpgrade(gameState);
this.elements.gtypeLevel.parentElement.style.display = '';
} else {
this.elements.gtypeLevel.parentElement.style.display = 'none';
}
var canAffordAny = var canAffordAny =
gameState.totalMassConsumed >= gameState.asteroidUpgradeCost || gameState.totalMassConsumed >= gameState.asteroidUpgradeCost ||
(gameState.cometUnlocked && gameState.totalMassConsumed >= gameState.cometUpgradeCost) || (gameState.cometUnlocked && gameState.totalMassConsumed >= gameState.cometUpgradeCost) ||
(gameState.planetUnlocked && gameState.totalMassConsumed >= gameState.planetUpgradeCost) || (gameState.planetUnlocked && gameState.totalMassConsumed >= gameState.planetUpgradeCost) ||
(gameState.giantUnlocked && gameState.totalMassConsumed >= gameState.giantUpgradeCost) || (gameState.giantUnlocked && gameState.totalMassConsumed >= gameState.giantUpgradeCost) ||
(gameState.mtypeUnlocked && gameState.totalMassConsumed >= gameState.mtypeUpgradeCost) || (gameState.mtypeUnlocked && gameState.totalMassConsumed >= gameState.mtypeUpgradeCost) ||
(gameState.ktypeUnlocked && gameState.totalMassConsumed >= gameState.ktypeUpgradeCost); (gameState.ktypeUnlocked && gameState.totalMassConsumed >= gameState.ktypeUpgradeCost) ||
(gameState.gtypeUnlocked && gameState.totalMassConsumed >= gameState.gtypeUpgradeCost);
var holeIcon = document.getElementById('hole-icon'); var holeIcon = document.getElementById('hole-icon');
if (holeIcon) { if (holeIcon) {
@ -333,7 +345,8 @@ var UI = {
gameState.planetUpgradeLevel + gameState.planetUpgradeLevel +
gameState.giantUpgradeLevel + gameState.giantUpgradeLevel +
gameState.mtypeUpgradeLevel + gameState.mtypeUpgradeLevel +
gameState.ktypeUpgradeLevel; gameState.ktypeUpgradeLevel +
gameState.gtypeUpgradeLevel;
el.innerHTML = el.innerHTML =
'<span class="tooltip-trigger">' + '<span class="tooltip-trigger">' +
@ -462,6 +475,7 @@ var UI = {
var rate = (CONFIG.BASE_KTYPE_SPAWN_INTERVAL / gameState.currentKtypeSpawnInterval).toFixed(2); var rate = (CONFIG.BASE_KTYPE_SPAWN_INTERVAL / gameState.currentKtypeSpawnInterval).toFixed(2);
var bonusPercent = (gameState.ktypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_KTYPE * 100).toFixed(0); var bonusPercent = (gameState.ktypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_KTYPE * 100).toFixed(0);
var tooltipText = 'K-types, also known as orange dwarfs, are medium-sized stars that are cooler than the Sun, with masses ranging from about 0.45 to 0.8 M☉. They are known for their stability and long lifespans (20 to 70 billion years), making them potential candidates for supporting inhabited planets.<br><br>Rate: ' + rate + '/2days<br>Bonus: ' + bonusPercent + '%'; var tooltipText = 'K-types, also known as orange dwarfs, are medium-sized stars that are cooler than the Sun, with masses ranging from about 0.45 to 0.8 M☉. They are known for their stability and long lifespans (20 to 70 billion years), making them potential candidates for supporting inhabited planets.<br><br>Rate: ' + rate + '/2days<br>Bonus: ' + bonusPercent + '%';
if (!gameState.gtypeUnlocked) tooltipText += '<br>Unlocks G-Type at level 5';
this.elements.ktypeLevel.innerHTML = '<span class="tooltip-trigger">K-Type: Level ' + this.elements.ktypeLevel.innerHTML = '<span class="tooltip-trigger">K-Type: Level ' +
gameState.ktypeUpgradeLevel + gameState.ktypeUpgradeLevel +
@ -472,6 +486,20 @@ var UI = {
gameState.totalMassConsumed < gameState.ktypeUpgradeCost; gameState.totalMassConsumed < gameState.ktypeUpgradeCost;
}, },
updateGtypeUpgrade: function(gameState) {
var rate = (CONFIG.BASE_GTYPE_SPAWN_INTERVAL / gameState.currentGtypeSpawnInterval).toFixed(2);
var bonusPercent = (gameState.gtypeUpgradeLevel * CONFIG.UPGRADE_BONUS_PER_LEVEL_GTYPE * 100).toFixed(0);
var tooltipText = 'G-Types, also known as yellow dwarfs, are medium-sized stars that are about the size of our Sun (~1 M☉). The term yellow dwarf is a misnomer, as most G-Types (including the Sun) are in fact white. These stars are stable and long-lived, making them strong candidates for hosting habitable planets.<br><br>Rate: ' + rate + '/3days<br>Bonus: ' + bonusPercent + '%';
this.elements.gtypeLevel.innerHTML = '<span class="tooltip-trigger">G-Type: Level ' +
gameState.gtypeUpgradeLevel +
'<span class="tooltip">' + tooltipText + '</span></span>';
this.elements.gtypeUpgradeBtn.textContent =
'Upgrade (Cost: ' + this.formatMass(gameState.gtypeUpgradeCost) + ')';
this.elements.gtypeUpgradeBtn.disabled =
gameState.totalMassConsumed < gameState.gtypeUpgradeCost;
},
formatMass: function(massKg, options = {}) { formatMass: function(massKg, options = {}) {
if (massKg == null) return '0'; // fallback if (massKg == null) return '0'; // fallback
const SOLAR_SWITCH_KG = CONFIG.SOLAR_MASS_KG * 0.01; // ~1% solar mass const SOLAR_SWITCH_KG = CONFIG.SOLAR_MASS_KG * 0.01; // ~1% solar mass