This commit is contained in:
M1zantrop 2026-06-17 00:31:14 +07:00
parent 3e6af9939f
commit cd37894f3f
3 changed files with 1100 additions and 0 deletions

195
cs.css Normal file
View File

@ -0,0 +1,195 @@
* {
user-select: none;
-webkit-tap-highlight-color: transparent;
}
body {
margin: 0;
min-height: 100vh;
background: radial-gradient(circle at 20% 30%, #03030f, #000000);
display: flex;
justify-content: center;
align-items: center;
font-family: 'Orbitron', 'Segoe UI', 'Courier New', monospace;
overflow: hidden;
}
.game-container {
background: #0a0c1a;
border-radius: 28px;
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.6), inset 0 1px 1px rgba(255, 255, 255, 0.05);
padding: 12px;
border: 1px solid #2a3f6e;
}
canvas {
display: block;
margin: 0 auto;
border-radius: 20px;
box-shadow: 0 0 0 2px #1e2a3a, 0 10px 25px -5px black;
cursor: crosshair;
}
.info-panel {
margin-top: 16px;
display: flex;
flex-wrap: wrap;
gap: 14px;
justify-content: space-between;
background: rgba(8, 12, 25, 0.85);
backdrop-filter: blur(12px);
border-radius: 20px;
padding: 12px 20px;
border: 1px solid #2e4b7c;
color: #b7e4ff;
text-shadow: 0 0 3px #0a5f8a;
}
.stats {
display: flex;
gap: 28px;
flex-wrap: wrap;
font-weight: bold;
}
.stat {
background: #010314;
padding: 6px 14px;
border-radius: 40px;
border-left: 3px solid #3e8ed0;
font-size: 0.9rem;
}
.stat span {
color: #ffd966;
font-weight: bold;
font-size: 1.1rem;
}
button {
background: #162b40;
border: none;
color: #c0e4ff;
font-family: monospace;
font-weight: bold;
padding: 6px 18px;
border-radius: 60px;
font-size: 0.8rem;
cursor: pointer;
transition: 0.2s;
box-shadow: 0 2px 6px rgba(0, 0, 0, 0.4);
letter-spacing: 1px;
}
button:hover {
background: #2a577a;
transform: scale(0.96);
color: white;
box-shadow: 0 0 8px #3f95cf;
}
.action-group {
display: flex;
gap: 12px;
align-items: center;
}
.status-msg {
background: #000000aa;
border-radius: 20px;
padding: 5px 15px;
font-size: 0.75rem;
font-family: monospace;
max-width: 260px;
border-right: 2px solid #2e8bc0;
}
.system-info {
background: #03071cee;
border-radius: 16px;
padding: 8px 14px;
margin-top: 10px;
font-size: 0.7rem;
max-width: 380px;
border-left: 3px solid #ffaa55;
}
.system-info h4 {
margin: 0 0 5px 0;
font-size: 0.8rem;
color: #ffcc88;
}
.planet-list {
display: flex;
flex-wrap: wrap;
gap: 8px;
margin-top: 6px;
}
.planet-badge {
background: #1f2a44;
border-radius: 20px;
padding: 4px 12px;
font-size: 0.7rem;
cursor: pointer;
transition: 0.1s;
display: flex;
align-items: center;
gap: 6px;
}
.planet-badge:hover {
background: #3e5a8f;
transform: scale(1.02);
}
/* Панель настроек */
.settings-panel {
background: #05161fe6;
border-radius: 20px;
padding: 10px 16px;
margin-top: 10px;
display: flex;
flex-wrap: wrap;
gap: 20px;
align-items: center;
font-size: 0.7rem;
border: 1px solid #2e6b8f;
}
.setting-item {
display: flex;
align-items: center;
gap: 12px;
background: #02101a;
padding: 4px 12px;
border-radius: 40px;
}
.setting-item label {
font-weight: bold;
color: #aae0ff;
}
input[type="range"] {
width: 140px;
background: #1f4a6e;
height: 3px;
border-radius: 5px;
}
.setting-value {
background: black;
padding: 2px 8px;
border-radius: 20px;
min-width: 40px;
text-align: center;
font-family: monospace;
}
.small-btn {
background: #2f5f7a;
padding: 2px 10px;
font-size: 0.7rem;
}

69
index.html Normal file
View File

@ -0,0 +1,69 @@
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=no">
<title>Галактическая Империя — настраиваемая скорость и ми111р</title>
</head>
<body>
<div>
<div class="game-container">
<canvas id="gameCanvas" width="1100" height="650" style="width:1100px;height:650px;"></canvas>
<div class="info-panel">
<div class="stats">
<div class="stat">🛸 КОРАБЛЬ: <span id="shipStatus">НА ОРБИТЕ</span></div>
<div class="stat">СКОРОСТЬ: <span id="speedVal">0</span> AU/s</div>
<div class="stat">💎 МЕТАЛЛ: <span id="metalCount">0</span> / <span id="metalCap"></span></div>
<div class="stat">🌫 ГАЗ: <span id="gasCount">0</span> / <span id="gasCap"></span></div>
<div class="stat">🔮 КРИСТАЛЛ: <span id="crystalCount">0</span> / <span id="crystalCap"></span>
</div>
</div>
<div class="action-group">
<button id="enterSystemBtn" style="background:#1f5940;">🚀 ВОЙТИ В СИСТЕМУ</button>
<button id="exitToGalaxyBtn" style="background:#2f4f6f;">🌌 ВЫЙТИ В ГАЛАКТИКУ</button>
<button id="mineBtn" style="background:#3a2a1f;">⛏ ДОБЫЧА</button>
</div>
<div class="status-msg" id="gameMsg">
▶ Кликните на систему, затем на планету в панели
</div>
</div>
<div id="systemInfoPanel" class="system-info" style="display: none;">
<h4 id="infoSysName">Система</h4>
<div id="planetListContainer" class="planet-list"></div>
<div style="margin-top: 6px; font-size:0.65rem;">✨ клик по планете — мгновенный телепорт на орбиту</div>
</div>
<div class="settings-panel">
<div class="setting-item">
<label>⚡Макс.скорость</label>
<input type="range" id="speedSlider" min="200" max="1200" step="10" value="520">
<span id="speedValue" class="setting-value">520</span>
</div>
<div class="setting-item">
<label>📦Вместимость</label>
<input type="range" id="capacitySlider" min="100" max="2000" step="50" value="800">
<span id="capacityValue" class="setting-value">800</span>
</div>
<div class="setting-item">
<label>🪐Кол-во систем</label>
<input type="range" id="systemsCountSlider" min="5" max="18" step="1" value="10">
<span id="systemsCountValue" class="setting-value">10</span>
<button id="regenerateWorldBtn" class="small-btn">🌍 Обновить мир</button>
</div>
<div class="setting-item">
<label>🌍Планет в системе</label>
<input type="range" id="planetCountSlider" min="2" max="8" step="1" value="4">
<span id="planetCountValue" class="setting-value">4</span>
<button id="updatePlanetsBtn" class="small-btn">🔄 Обновить планеты</button>
</div>
</div>
</div>
</div>
<script src="n.ts"></script>
</body>
</html>

836
n.ts Normal file
View File

@ -0,0 +1,836 @@
<script>
(function () {
const canvas = document.getElementById('gameCanvas');
const ctx = canvas.getContext('2d');
let HEX_SIZE = 58;
function hexToPixel(q, r) {
const x = HEX_SIZE * (Math.sqrt(3) * q + Math.sqrt(3) / 2 * r);
const y = HEX_SIZE * (3 / 2 * r);
return { x, y };
}
// Глобальные переменные для генерации мира
let systems = [];
let axialCoords = []; // будет заполняться динамически
let selectedSystem = null;
let camera = { offsetX: 0, offsetY: 0, zoom: 1.0 };
// Параметры игры
let maxShipSpeed = 520;
let cargoCapacity = 800;
let gameState = {
view: "galaxy",
currentSystemId: null,
targetSystemId: null,
targetPlanet: null,
pendingAutoEnter: false,
ship: {
posX: 0, posY: 0,
velX: 0, velY: 0,
maxSpeed: maxShipSpeed,
accel: 680,
drag: 1.15,
radius: 9
},
inventory: { metal: 0, gas: 0, crystal: 0 },
messages: []
};
// Функция генерации планет
function generatePlanetsForSystem(sysName, targetCount = null) {
const templates = [
{ name: "Вулкания", type: "металлическая", resource: "metal", baseYield: 12, icon: "💎" },
{ name: "Акварис", type: "океаническая", resource: "gas", baseYield: 8, icon: "🌫" },
{ name: "Кристаллис", type: "ледяная", resource: "crystal", baseYield: 10, icon: "🔮" },
{ name: "Терра Нова", type: "землеподобная", resource: "metal", baseYield: 15, icon: "💎" },
{ name: "Газовая бухта", type: "газовый гигант", resource: "gas", baseYield: 20, icon: "🌫" },
{ name: "Кремниевый пояс", type: "астероидное поле", resource: "crystal", baseYield: 9, icon: "🔮" },
{ name: "Инферно", type: "раскалённая", resource: "metal", baseYield: 11, icon: "💎" },
{ name: "Эфир", type: "туманность", resource: "crystal", baseYield: 14, icon: "🔮" }
];
let planetsCount = targetCount !== null ? targetCount : (3 + Math.floor(Math.random() * 4));
let shuffled = [...templates];
for (let i = shuffled.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[shuffled[i], shuffled[j]] = [shuffled[j], shuffled[i]];
}
let planets = [];
for (let i = 0; i < planetsCount; i++) {
let tpl = shuffled[i % shuffled.length];
let orbitRadius = 70 + i * 40 + Math.random() * 20;
let angle = (i * 97) % 360;
planets.push({
id: `${sysName}_p${i}`,
name: tpl.name + " " + (i + 1),
type: tpl.type,
resource: tpl.resource,
yield: tpl.baseYield + Math.floor(Math.random() * 7),
orbitRadius: orbitRadius,
angle: angle,
icon: tpl.icon,
posX: 0, posY: 0
});
}
return planets;
}
// Генерация гексагональной сетки с заданным числом систем
function generateHexGrid(systemsCount) {
// Генерируем координаты по спирали или просто кольцами. Используем простой метод: создаём список осевых координат вокруг центра.
let coords = [{ q: 0, r: 0, name: "SOL PRIME" }];
let names = ["ORIONIS", "SIRIUS", "ANTARES", "RIGEL", "VEGA", "ALTAIR", "THUBAN", "POLARIS", "BETELGEUSE", "CENTAURI", "DRACO", "LYRA", "AQUILA", "CYGNUS", "ANDROMEDA"];
let ring = 1;
while (coords.length < systemsCount) {
for (let i = -ring; i <= ring; i++) {
for (let j = -ring; j <= ring; j++) {
let k = -i - j;
if (Math.max(Math.abs(i), Math.abs(j), Math.abs(k)) === ring) {
if (coords.length < systemsCount && !coords.some(c => c.q === i && c.r === j)) {
let idx = coords.length % names.length;
coords.push({ q: i, r: j, name: names[idx] + (ring) });
}
}
}
}
ring++;
}
return coords;
}
// Построение систем (полная перегенерация)
function rebuildWorld(systemsCount, planetsPerSystem = null) {
axialCoords = generateHexGrid(systemsCount);
let newSystems = [];
for (let idx = 0; idx < axialCoords.length; idx++) {
let a = axialCoords[idx];
let pixel = hexToPixel(a.q, a.r);
let systemId = `sys_${idx}`;
let planets = generatePlanetsForSystem(a.name, planetsPerSystem);
newSystems.push({
id: systemId,
name: a.name,
q: a.q,
r: a.r,
x: pixel.x + canvas.width / 2,
y: pixel.y + canvas.height / 2,
planets: planets,
visited: false
});
}
// Вычисляем соседей
for (let sys of newSystems) {
sys.neighbors = [];
for (let other of newSystems) {
if (sys.id === other.id) continue;
let dq = Math.abs(sys.q - other.q);
let dr = Math.abs(sys.r - other.r);
let ds = Math.abs((sys.q + sys.r) - (other.q + other.r));
let dist = Math.max(dq, dr, ds);
if (dist === 1) sys.neighbors.push(other.id);
}
}
systems = newSystems;
// Если текущая система не существует, выбираем первую
if (!systems.find(s => s.id === gameState.currentSystemId)) {
gameState.currentSystemId = systems[0].id;
gameState.ship.posX = systems[0].x;
gameState.ship.posY = systems[0].y;
} else {
let curSys = systems.find(s => s.id === gameState.currentSystemId);
if (curSys) {
gameState.ship.posX = curSys.x;
gameState.ship.posY = curSys.y;
}
}
gameState.targetSystemId = null;
gameState.targetPlanet = null;
if (gameState.view === "system") gameState.view = "galaxy";
selectedSystem = null;
updateSystemInfoPanel();
addMessage(`🌌 Вселенная обновлена: ${systems.length} систем, настройки планет применены.`);
updateUIStats();
}
// Обновить планеты в выбранной системе (или текущей)
function updatePlanetsInSelectedSystem(planetCount) {
let targetSys = selectedSystem;
if (!targetSys && gameState.view === "system") targetSys = getCurrentSystem();
if (!targetSys) {
addMessage("Сначала выберите систему (кликните на карте)", true);
return;
}
let newPlanets = generatePlanetsForSystem(targetSys.name, planetCount);
targetSys.planets = newPlanets;
if (gameState.view === "system" && gameState.currentSystemId === targetSys.id) {
// Если мы внутри этой системы — телепортируем корабль в центр, чтобы избежать конфликтов
gameState.ship.posX = 0;
gameState.ship.posY = 0;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
gameState.targetPlanet = null;
}
updateSystemInfoPanel(); // обновить отображение в панели
addMessage(`🪐 Планеты системы ${targetSys.name} обновлены (${newPlanets.length} шт.)`);
}
// Ограничение по вместимости
function applyCapacityLimit() {
if (gameState.inventory.metal > cargoCapacity) gameState.inventory.metal = cargoCapacity;
if (gameState.inventory.gas > cargoCapacity) gameState.inventory.gas = cargoCapacity;
if (gameState.inventory.crystal > cargoCapacity) gameState.inventory.crystal = cargoCapacity;
document.getElementById('metalCap').innerText = cargoCapacity;
document.getElementById('gasCap').innerText = cargoCapacity;
document.getElementById('crystalCap').innerText = cargoCapacity;
}
function addMessage(msg, isError = false) {
gameState.messages.unshift(msg);
if (gameState.messages.length > 3) gameState.messages.pop();
let msgDiv = document.getElementById('gameMsg');
msgDiv.innerHTML = `🛸 ${msg}`;
if (isError) msgDiv.style.borderRightColor = "#ff8866";
else msgDiv.style.borderRightColor = "#2e8bc0";
setTimeout(() => {
if (document.getElementById('gameMsg').innerHTML === `🛸 ${msg}`)
msgDiv.style.borderRightColor = "#2e8bc0";
}, 2000);
}
function updateUIStats() {
document.getElementById('metalCount').innerText = gameState.inventory.metal;
document.getElementById('gasCount').innerText = gameState.inventory.gas;
document.getElementById('crystalCount').innerText = gameState.inventory.crystal;
let spd = Math.hypot(gameState.ship.velX, gameState.ship.velY);
document.getElementById('speedVal').innerText = spd.toFixed(1);
let statusText = "В ГАЛАКТИКЕ";
if (gameState.view === "system") statusText = `В СИСТЕМЕ ${getCurrentSystem().name}`;
if (gameState.targetSystemId) statusText = "ПЕРЕЛЁТ ...";
if (gameState.targetPlanet) statusText = "К ПЛАНЕТЕ";
document.getElementById('shipStatus').innerHTML = statusText;
applyCapacityLimit();
}
function getCurrentSystem() {
return systems.find(s => s.id === gameState.currentSystemId);
}
function updatePlanetPositions(system) {
if (!system) return;
let now = Date.now() / 1000;
for (let p of system.planets) {
let ang = (p.angle + now * 0.45) % 360;
let rad = p.orbitRadius;
p.posX = Math.cos(ang * Math.PI / 180) * rad;
p.posY = Math.sin(ang * Math.PI / 180) * rad;
}
}
const DOCK_DIST = 22;
const DOCK_ZONE = 40;
function handleSmoothDocking(deltaSec) {
if (gameState.view !== "system") return;
let sys = getCurrentSystem();
if (!sys) return;
for (let planet of sys.planets) {
let dx = planet.posX - gameState.ship.posX;
let dy = planet.posY - gameState.ship.posY;
let dist = Math.hypot(dx, dy);
if (dist < DOCK_ZONE) {
let angle = Math.atan2(dy, dx);
let targetX = planet.posX - Math.cos(angle) * DOCK_DIST;
let targetY = planet.posY - Math.sin(angle) * DOCK_DIST;
let toTargetX = targetX - gameState.ship.posX;
let toTargetY = targetY - gameState.ship.posY;
let force = 4.5 * deltaSec;
if (force > 0.8) force = 0.8;
gameState.ship.velX += toTargetX * force;
gameState.ship.velY += toTargetY * force;
let distToTarget = Math.hypot(toTargetX, toTargetY);
if (distToTarget < 3) {
gameState.ship.velX *= 0.95;
gameState.ship.velY *= 0.95;
}
}
}
}
function updatePhysics(deltaSec) {
if (deltaSec > 0.033) deltaSec = 0.033;
const ship = gameState.ship;
let targetX = null, targetY = null;
let isMoving = false;
if (gameState.view === "system") {
let sys = getCurrentSystem();
if (sys) updatePlanetPositions(sys);
handleSmoothDocking(deltaSec);
}
if (gameState.view === "galaxy" && gameState.targetSystemId) {
let targetSys = systems.find(s => s.id === gameState.targetSystemId);
if (targetSys) {
targetX = targetSys.x;
targetY = targetSys.y;
isMoving = true;
let dx = targetX - ship.posX;
let dy = targetY - ship.posY;
let dist = Math.hypot(dx, dy);
if (dist < 15) {
ship.posX = targetSys.x;
ship.posY = targetSys.y;
ship.velX = 0;
ship.velY = 0;
gameState.currentSystemId = targetSys.id;
gameState.targetSystemId = null;
addMessage(`✧ Прибыли в систему ${targetSys.name}`);
targetSys.visited = true;
if (gameState.pendingAutoEnter) {
gameState.pendingAutoEnter = false;
gameState.view = "system";
gameState.targetPlanet = null;
gameState.ship.posX = 0;
gameState.ship.posY = 0;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
addMessage(`✨ Погружение в систему ${targetSys.name}.`);
}
updateUIStats();
isMoving = false;
}
} else {
gameState.targetSystemId = null;
}
}
else if (gameState.view === "system" && gameState.targetPlanet) {
let sys = getCurrentSystem();
if (sys && gameState.targetPlanet) {
let freshPlanet = sys.planets.find(p => p.id === gameState.targetPlanet.id);
if (freshPlanet) {
gameState.targetPlanet = freshPlanet;
targetX = freshPlanet.posX;
targetY = freshPlanet.posY;
isMoving = true;
let dx = targetX - ship.posX;
let dy = targetY - ship.posY;
let dist = Math.hypot(dx, dy);
if (dist < DOCK_DIST + 5) {
let angle = Math.atan2(dy, dx);
let finalX = freshPlanet.posX - Math.cos(angle) * DOCK_DIST;
let finalY = freshPlanet.posY - Math.sin(angle) * DOCK_DIST;
ship.posX = finalX;
ship.posY = finalY;
ship.velX = 0;
ship.velY = 0;
addMessage(`🔻 Корабль на орбите ${freshPlanet.name}`);
gameState.targetPlanet = null;
isMoving = false;
}
} else {
gameState.targetPlanet = null;
}
} else {
gameState.targetPlanet = null;
}
}
if (isMoving && targetX !== null && targetY !== null) {
let dx = targetX - ship.posX;
let dy = targetY - ship.posY;
let dist = Math.hypot(dx, dy);
if (dist > 0.5) {
let dirX = dx / dist;
let dirY = dy / dist;
let thrust = gameState.ship.accel * deltaSec;
ship.velX += dirX * thrust;
ship.velY += dirY * thrust;
}
let spd = Math.hypot(ship.velX, ship.velY);
if (spd > gameState.ship.maxSpeed) {
ship.velX = (ship.velX / spd) * gameState.ship.maxSpeed;
ship.velY = (ship.velY / spd) * gameState.ship.maxSpeed;
}
}
let dragForce = 1 - (gameState.ship.drag * deltaSec);
if (dragForce < 0) dragForce = 0;
ship.velX *= dragForce;
ship.velY *= dragForce;
ship.posX += ship.velX * deltaSec;
ship.posY += ship.velY * deltaSec;
if (gameState.view === "galaxy") {
ship.posX = Math.min(Math.max(ship.posX, -400), canvas.width + 400);
ship.posY = Math.min(Math.max(ship.posY, -300), canvas.height + 300);
}
if (gameState.view === "system") {
let limit = 950;
ship.posX = Math.min(Math.max(ship.posX, -limit), limit);
ship.posY = Math.min(Math.max(ship.posY, -limit), limit);
}
updateUIStats();
}
function teleportToPlanet(system, planet) {
if (gameState.currentSystemId !== system.id && gameState.targetSystemId !== system.id) {
gameState.ship.posX = system.x;
gameState.ship.posY = system.y;
gameState.currentSystemId = system.id;
gameState.targetSystemId = null;
addMessage(`📡 Гиперпрыжок в систему ${system.name}`);
}
if (gameState.view !== "system") {
gameState.view = "system";
gameState.targetPlanet = null;
gameState.pendingAutoEnter = false;
addMessage(`✨ Вход в систему ${system.name}`);
}
updatePlanetPositions(system);
let angleToPlanet = Math.atan2(planet.posY, planet.posX);
let orbitX = planet.posX - Math.cos(angleToPlanet) * DOCK_DIST;
let orbitY = planet.posY - Math.sin(angleToPlanet) * DOCK_DIST;
gameState.ship.posX = orbitX;
gameState.ship.posY = orbitY;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
gameState.targetPlanet = null;
addMessage(`🪐 Телепорт на орбиту ${planet.name} (${planet.type})`);
updateUIStats();
}
function updateSystemInfoPanel() {
const panel = document.getElementById('systemInfoPanel');
if (gameState.view !== "galaxy" || !selectedSystem) {
panel.style.display = 'none';
return;
}
panel.style.display = 'block';
document.getElementById('infoSysName').innerHTML = `🌟 ${selectedSystem.name}`;
const container = document.getElementById('planetListContainer');
container.innerHTML = '';
for (let planet of selectedSystem.planets) {
const badge = document.createElement('div');
badge.className = 'planet-badge';
let resourceIcon = "";
if (planet.resource === 'metal') resourceIcon = "💎";
else if (planet.resource === 'gas') resourceIcon = "🌫";
else resourceIcon = "🔮";
badge.innerHTML = `${resourceIcon} ${planet.name} (${planet.type}) +${planet.yield}`;
badge.addEventListener('click', (e) => {
e.stopPropagation();
teleportToPlanet(selectedSystem, planet);
});
container.appendChild(badge);
}
}
function getMouseCanvasCoords(e) {
const rect = canvas.getBoundingClientRect();
const scaleX = canvas.width / rect.width;
const scaleY = canvas.height / rect.height;
let mouseX = (e.clientX - rect.left) * scaleX;
let mouseY = (e.clientY - rect.top) * scaleY;
if (gameState.view === "galaxy") {
let worldX = (mouseX - camera.offsetX) / camera.zoom;
let worldY = (mouseY - camera.offsetY) / camera.zoom;
return { worldX, worldY, mouseX, mouseY };
} else {
return { worldX: mouseX, worldY: mouseY, mouseX, mouseY };
}
}
function findSystemAt(worldX, worldY) {
for (let sys of systems) {
let dx = sys.x - worldX;
let dy = sys.y - worldY;
let dist = Math.hypot(dx, dy);
if (dist < HEX_SIZE * 0.85) return sys;
}
return null;
}
function handleGalaxyClick(e) {
let { worldX, worldY } = getMouseCanvasCoords(e);
let system = findSystemAt(worldX, worldY);
if (system) {
selectedSystem = system;
updateSystemInfoPanel();
addMessage(`🔍 Выбрана система ${system.name}. В панели планеты — кликните для телепорта.`);
} else {
selectedSystem = null;
updateSystemInfoPanel();
}
}
function handleSystemClick(e) {
const rect = canvas.getBoundingClientRect();
const scaleX = canvas.width / rect.width;
const scaleY = canvas.height / rect.height;
let mouseCanvasX = (e.clientX - rect.left) * scaleX;
let mouseCanvasY = (e.clientY - rect.top) * scaleY;
let sys = getCurrentSystem();
if (!sys) return;
updatePlanetPositions(sys);
for (let planet of sys.planets) {
let screenX = canvas.width / 2 + planet.posX;
let screenY = canvas.height / 2 + planet.posY;
let dx = mouseCanvasX - screenX;
let dy = mouseCanvasY - screenY;
if (Math.hypot(dx, dy) < 28) {
if (gameState.targetPlanet === planet) {
addMessage(`Уже направляемся к ${planet.name}`);
return;
}
gameState.targetPlanet = planet;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
addMessage(`🪐 Курс на ${planet.name}`);
return;
}
}
}
function tryMine() {
if (gameState.view !== "system") {
addMessage("❌ Добыча доступна только внутри системы!", true);
return;
}
let sys = getCurrentSystem();
if (!sys) return;
let ship = gameState.ship;
let closePlanet = null;
for (let p of sys.planets) {
let dx = ship.posX - p.posX;
let dy = ship.posY - p.posY;
let dist = Math.hypot(dx, dy);
if (dist < 40) {
closePlanet = p;
break;
}
}
if (!closePlanet) {
addMessage("🌍 Нет планеты в зоне добычи. Используйте телепорт из панели или подлетите.", true);
return;
}
let yieldAmount = closePlanet.yield + Math.floor(Math.random() * 8);
let resource = closePlanet.resource;
let newVal = gameState.inventory[resource] + yieldAmount;
if (newVal > cargoCapacity) newVal = cargoCapacity;
gameState.inventory[resource] = newVal;
addMessage(`⛏ Добыто ${yieldAmount} ${resource === 'metal' ? 'металла' : (resource === 'gas' ? 'газа' : 'кристаллов')} с ${closePlanet.name}`);
updateUIStats();
}
function enterSystemWithAutoFlight() {
if (gameState.view !== "galaxy") {
addMessage("Вы уже внутри системы", true);
return;
}
let targetSys = selectedSystem;
if (!targetSys) {
addMessage("Кликните на систему, чтобы выбрать.", true);
return;
}
if (gameState.targetSystemId === targetSys.id) {
addMessage(`Перелёт к ${targetSys.name} уже идёт.`);
gameState.pendingAutoEnter = true;
return;
}
let distToSys = Math.hypot(gameState.ship.posX - targetSys.x, gameState.ship.posY - targetSys.y);
if (gameState.currentSystemId === targetSys.id || distToSys < 20) {
gameState.view = "system";
gameState.targetPlanet = null;
gameState.targetSystemId = null;
gameState.ship.posX = 0;
gameState.ship.posY = 0;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
addMessage(`✨ Погружение в ${targetSys.name}`);
updateUIStats();
return;
}
gameState.targetSystemId = targetSys.id;
gameState.pendingAutoEnter = true;
addMessage(`🚀 Курс на ${targetSys.name}, после прибытия вход авто.`);
}
function exitToGalaxy() {
if (gameState.view !== "system") {
addMessage("Уже в галактике", true);
return;
}
let currentSys = getCurrentSystem();
if (currentSys) {
gameState.view = "galaxy";
gameState.targetPlanet = null;
gameState.targetSystemId = null;
gameState.pendingAutoEnter = false;
gameState.ship.posX = currentSys.x;
gameState.ship.posY = currentSys.y;
gameState.ship.velX = 0;
gameState.ship.velY = 0;
addMessage(`🌌 Выход к системе ${currentSys.name}`);
updateUIStats();
selectedSystem = currentSys;
updateSystemInfoPanel();
}
}
// Настройки UI
const speedSlider = document.getElementById('speedSlider');
const speedValue = document.getElementById('speedValue');
const capacitySlider = document.getElementById('capacitySlider');
const capacityValue = document.getElementById('capacityValue');
const systemsCountSlider = document.getElementById('systemsCountSlider');
const systemsCountValue = document.getElementById('systemsCountValue');
const planetCountSlider = document.getElementById('planetCountSlider');
const planetCountValue = document.getElementById('planetCountValue');
const regenerateWorldBtn = document.getElementById('regenerateWorldBtn');
const updatePlanetsBtn = document.getElementById('updatePlanetsBtn');
speedSlider.addEventListener('input', () => {
maxShipSpeed = parseInt(speedSlider.value);
speedValue.innerText = maxShipSpeed;
gameState.ship.maxSpeed = maxShipSpeed;
addMessage(`Максимальная скорость изменена на ${maxShipSpeed}`);
});
capacitySlider.addEventListener('input', () => {
cargoCapacity = parseInt(capacitySlider.value);
capacityValue.innerText = cargoCapacity;
applyCapacityLimit();
addMessage(`Вместимость хранилища: ${cargoCapacity}`);
});
systemsCountSlider.addEventListener('input', () => {
systemsCountValue.innerText = systemsCountSlider.value;
});
planetCountSlider.addEventListener('input', () => {
planetCountValue.innerText = planetCountSlider.value;
});
regenerateWorldBtn.addEventListener('click', () => {
let newCount = parseInt(systemsCountSlider.value);
let planetsCount = parseInt(planetCountSlider.value);
rebuildWorld(newCount, planetsCount);
});
updatePlanetsBtn.addEventListener('click', () => {
let planetsCount = parseInt(planetCountSlider.value);
updatePlanetsInSelectedSystem(planetsCount);
});
// Инициализация мира
function initWorld() {
let sysCount = parseInt(systemsCountSlider.value);
let planetsPer = parseInt(planetCountSlider.value);
axialCoords = generateHexGrid(sysCount);
rebuildWorld(sysCount, planetsPer);
}
// Отрисовка
function drawGalaxy() {
ctx.save();
ctx.translate(camera.offsetX, camera.offsetY);
ctx.scale(camera.zoom, camera.zoom);
ctx.fillStyle = "#010008";
ctx.fillRect(-camera.offsetX / camera.zoom, -camera.offsetY / camera.zoom, canvas.width / camera.zoom, canvas.height / camera.zoom);
for (let i = 0; i < 800; i++) {
ctx.fillStyle = `rgba(255,240,200,${Math.random() * 0.6 + 0.2})`;
ctx.beginPath();
ctx.arc((i * 131) % canvas.width - camera.offsetX / camera.zoom, (i * 253) % canvas.height - camera.offsetY / camera.zoom, 1 + Math.random() * 2, 0, Math.PI * 2);
ctx.fill();
}
for (let sys of systems) {
for (let nId of sys.neighbors) {
let neigh = systems.find(s => s.id === nId);
if (neigh && sys.id < neigh.id) {
ctx.beginPath();
ctx.moveTo(sys.x, sys.y);
ctx.lineTo(neigh.x, neigh.y);
ctx.strokeStyle = "#4f7fbf";
ctx.lineWidth = 2.5;
ctx.stroke();
}
}
}
for (let sys of systems) {
let angles = [];
for (let i = 0; i < 6; i++) {
let ang = i * Math.PI * 2 / 6 - Math.PI / 6;
let x = sys.x + Math.cos(ang) * HEX_SIZE;
let y = sys.y + Math.sin(ang) * HEX_SIZE;
angles.push({ x, y });
}
ctx.beginPath();
ctx.moveTo(angles[0].x, angles[0].y);
for (let i = 1; i < 6; i++) ctx.lineTo(angles[i].x, angles[i].y);
ctx.closePath();
ctx.fillStyle = "#071a2be6";
ctx.fill();
ctx.strokeStyle = "#3e7bb3";
ctx.lineWidth = 1.8;
ctx.stroke();
let gradient = ctx.createRadialGradient(sys.x - 4, sys.y - 4, 5, sys.x, sys.y, 20);
gradient.addColorStop(0, '#ffdd88');
gradient.addColorStop(1, '#aa7733');
ctx.fillStyle = gradient;
ctx.beginPath();
ctx.arc(sys.x, sys.y, 13, 0, 2 * Math.PI);
ctx.fill();
ctx.fillStyle = "white";
ctx.font = `bold ${14}px "Orbitron"`;
ctx.shadowBlur = 4;
ctx.fillText(sys.name, sys.x - 28, sys.y - 15);
ctx.fillStyle = "#bbddff";
ctx.font = `10px monospace`;
ctx.fillText(`🪐${sys.planets.length}`, sys.x + 10, sys.y - 8);
if (selectedSystem && selectedSystem.id === sys.id) {
ctx.beginPath();
ctx.arc(sys.x, sys.y, HEX_SIZE * 0.9, 0, 2 * Math.PI);
ctx.strokeStyle = "#ffcc44";
ctx.lineWidth = 3;
ctx.setLineDash([6, 8]);
ctx.stroke();
ctx.setLineDash([]);
}
}
ctx.shadowBlur = 6;
ctx.beginPath();
ctx.arc(gameState.ship.posX, gameState.ship.posY, 9, 0, 2 * Math.PI);
ctx.fillStyle = "#6fcbff";
ctx.fill();
ctx.fillStyle = "white";
ctx.beginPath();
ctx.moveTo(gameState.ship.posX + 12, gameState.ship.posY);
ctx.lineTo(gameState.ship.posX + 18, gameState.ship.posY - 4);
ctx.lineTo(gameState.ship.posX + 18, gameState.ship.posY + 4);
ctx.fill();
ctx.fillStyle = "#c8ffff";
ctx.font = "bold 12px monospace";
ctx.fillText("⚡", gameState.ship.posX - 4, gameState.ship.posY - 4);
ctx.restore();
}
function drawSystemView() {
ctx.clearRect(0, 0, canvas.width, canvas.height);
let sys = getCurrentSystem();
if (!sys) return;
updatePlanetPositions(sys);
ctx.fillStyle = "#030318";
ctx.fillRect(0, 0, canvas.width, canvas.height);
for (let i = 0; i < 300; i++) {
ctx.fillStyle = `rgba(255,210,150,0.5)`;
ctx.beginPath();
ctx.arc((i * 97) % canvas.width, (i * 353) % canvas.height, 1.2, 0, Math.PI * 2);
ctx.fill();
}
let grad = ctx.createRadialGradient(canvas.width / 2 - 20, canvas.height / 2 - 20, 10, canvas.width / 2, canvas.height / 2, 70);
grad.addColorStop(0, '#ffaa55');
grad.addColorStop(1, '#cc4400');
ctx.fillStyle = grad;
ctx.beginPath();
ctx.arc(canvas.width / 2, canvas.height / 2, 45, 0, 2 * Math.PI);
ctx.fill();
for (let p of sys.planets) {
let x = canvas.width / 2 + p.posX;
let y = canvas.height / 2 + p.posY;
ctx.beginPath();
ctx.arc(x, y, 14, 0, 2 * Math.PI);
ctx.fillStyle = p.resource === 'metal' ? "#a07a4a" : (p.resource === 'gas' ? "#7fbbbb" : "#c9a0dc");
ctx.fill();
ctx.fillStyle = "#eef4ff";
ctx.font = "10px monospace";
ctx.fillText(p.name, x - 18, y - 12);
ctx.fillStyle = "#ddddaa";
ctx.fillText(`${p.type.substring(0, 3)}`, x - 10, y + 18);
if (gameState.targetPlanet && gameState.targetPlanet.id === p.id) {
ctx.beginPath();
ctx.arc(x, y, 22, 0, 2 * Math.PI);
ctx.strokeStyle = "#ffaa44";
ctx.lineWidth = 2.5;
ctx.setLineDash([4, 6]);
ctx.stroke();
ctx.setLineDash([]);
}
}
let shipX = canvas.width / 2 + gameState.ship.posX;
let shipY = canvas.height / 2 + gameState.ship.posY;
ctx.beginPath();
ctx.arc(shipX, shipY, 10, 0, 2 * Math.PI);
ctx.fillStyle = "#4ad4ff";
ctx.fill();
ctx.fillStyle = "white";
ctx.beginPath();
ctx.moveTo(shipX + 14, shipY);
ctx.lineTo(shipX + 22, shipY - 5);
ctx.lineTo(shipX + 22, shipY + 5);
ctx.fill();
if (gameState.targetPlanet) {
ctx.beginPath();
ctx.moveTo(shipX, shipY);
let tarX = canvas.width / 2 + gameState.targetPlanet.posX;
let tarY = canvas.height / 2 + gameState.targetPlanet.posY;
ctx.lineTo(tarX, tarY);
ctx.strokeStyle = "#ffaa66";
ctx.setLineDash([8, 6]);
ctx.stroke();
ctx.setLineDash([]);
}
ctx.font = "11px monospace";
ctx.fillStyle = "#bbddff";
ctx.fillText("Клик по планете — полёт. Панель в галактике — телепорт", 20, 40);
}
let lastTime = performance.now();
function animate() {
let now = performance.now();
let delta = Math.min(0.033, (now - lastTime) / 1000);
if (delta > 0.001) updatePhysics(delta);
lastTime = now;
if (gameState.view === "galaxy") {
let targetOffsetX = canvas.width / 2 - gameState.ship.posX * camera.zoom;
let targetOffsetY = canvas.height / 2 - gameState.ship.posY * camera.zoom;
camera.offsetX = targetOffsetX;
camera.offsetY = targetOffsetY;
drawGalaxy();
} else {
drawSystemView();
}
requestAnimationFrame(animate);
}
function init() {
initWorld();
document.getElementById('enterSystemBtn').addEventListener('click', enterSystemWithAutoFlight);
document.getElementById('exitToGalaxyBtn').addEventListener('click', exitToGalaxy);
document.getElementById('mineBtn').addEventListener('click', tryMine);
canvas.addEventListener('click', (e) => {
if (gameState.view === "galaxy") handleGalaxyClick(e);
else if (gameState.view === "system") handleSystemClick(e);
});
canvas.addEventListener('wheel', (e) => {
if (gameState.view !== "galaxy") return;
e.preventDefault();
let delta = e.deltaY > 0 ? 0.9 : 1.1;
let newZoom = camera.zoom * delta;
newZoom = Math.min(Math.max(newZoom, 0.5), 2.2);
camera.zoom = newZoom;
});
canvas.addEventListener('dblclick', (e) => {
if (gameState.view !== "galaxy") return;
let { worldX, worldY } = getMouseCanvasCoords(e);
let sys = findSystemAt(worldX, worldY);
if (sys) {
selectedSystem = sys;
updateSystemInfoPanel();
enterSystemWithAutoFlight();
}
});
updateUIStats();
addMessage("Добро пожаловать! Настройте скорость и мир под себя.");
animate();
}
init();
})();
</script>