Files
paper-racing-gpi/solution-player/index.html
2025-10-20 21:07:28 +05:00

359 lines
11 KiB
HTML
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<!DOCTYPE html>
<html lang="ru">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Визуализатор решений - Гонки на бумаге</title>
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
min-height: 100vh;
padding: 20px;
display: flex;
justify-content: center;
align-items: center;
}
.container {
background: white;
border-radius: 20px;
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.3);
padding: 30px;
width: 100%;
}
h1 {
text-align: center;
color: #333;
margin-bottom: 10px;
font-size: 2em;
}
.subtitle {
text-align: center;
color: #666;
margin-bottom: 30px;
font-size: 0.9em;
}
.controls {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 20px;
margin-bottom: 30px;
}
.control-group {
background: #f8f9fa;
padding: 20px;
border-radius: 12px;
border: 2px solid #e9ecef;
}
.control-group h3 {
margin-bottom: 15px;
color: #495057;
font-size: 1.1em;
border-bottom: 2px solid #dee2e6;
padding-bottom: 8px;
}
.buttons {
display: flex;
gap: 10px;
flex-wrap: wrap;
}
button {
flex: 1;
padding: 12px 24px;
border: none;
border-radius: 8px;
font-size: 16px;
font-weight: 600;
cursor: pointer;
transition: all 0.3s;
min-width: 120px;
}
button:hover {
transform: translateY(-2px);
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.2);
}
button:active {
transform: translateY(0);
}
.btn-primary {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
color: white;
}
.btn-success {
background: linear-gradient(135deg, #43e97b 0%, #38f9d7 100%);
color: #333;
}
.canvas-wrapper {
margin-top: 30px;
display: flex;
justify-content: center;
background: #f8f9fa;
padding: 20px;
border-radius: 12px;
overflow: auto;
}
canvas {
border: 3px solid #dee2e6;
border-radius: 8px;
box-shadow: 0 5px 20px rgba(0, 0, 0, 0.1);
}
.info {
margin-top: 20px;
padding: 15px;
background: #e7f3ff;
border-left: 4px solid #2196f3;
border-radius: 8px;
color: #0d47a1;
}
.info strong {
display: block;
margin-bottom: 5px;
}
#mapInput, #solutionInput {
display: none;
}
.legend {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(150px, 1fr));
gap: 10px;
margin-top: 15px;
}
.legend-item {
display: flex;
align-items: center;
gap: 10px;
padding: 8px;
background: white;
border-radius: 6px;
}
.legend-color {
width: 30px;
height: 30px;
border-radius: 4px;
border: 2px solid #dee2e6;
}
.road { background: #f8f9fa; }
.stone { background: #6c757d; }
.snow { background: #e3f2fd; }
.ice { background: #b3e5fc; }
.checkpoint { background: #fff3cd; }
.start { background: #d4edda; }
.playback-controls {
display: flex;
gap: 10px;
align-items: center;
flex-wrap: wrap;
margin-top: 15px;
}
.playback-btn {
padding: 10px 20px;
border: none;
border-radius: 8px;
font-size: 16px;
font-weight: 600;
cursor: pointer;
transition: all 0.3s;
background: #667eea;
color: white;
min-width: auto;
}
.playback-btn:hover {
transform: translateY(-2px);
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.2);
}
.playback-btn:disabled {
opacity: 0.5;
cursor: not-allowed;
transform: none;
}
.speed-control {
display: flex;
align-items: center;
gap: 10px;
}
.speed-control input[type="range"] {
width: 150px;
}
.step-info {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 15px;
margin-top: 15px;
padding: 15px;
background: white;
border-radius: 8px;
}
.info-item {
display: flex;
flex-direction: column;
}
.info-label {
font-size: 0.85em;
color: #6c757d;
margin-bottom: 5px;
}
.info-value {
font-size: 1.2em;
font-weight: 600;
color: #333;
}
.hidden {
display: none;
}
.visualization-panel {
margin-top: 20px;
padding: 20px;
background: #f8f9fa;
border-radius: 12px;
border: 2px solid #dee2e6;
}
.visualization-panel h3 {
margin-bottom: 15px;
color: #495057;
}
</style>
</head>
<body>
<div class="container">
<h1>🎬 Визуализатор решений</h1>
<p class="subtitle">Гонки на бумаге / Paper Racing</p>
<div class="controls">
<div class="control-group">
<h3>📂 Загрузка файлов</h3>
<div class="buttons">
<button class="btn-primary" id="loadMapBtn" onclick="document.getElementById('mapInput').click()">📂 Загрузить карту</button>
<button class="btn-success" id="loadSolutionBtn" onclick="document.getElementById('solutionInput').click()">🎬 Загрузить решение</button>
</div>
<input type="file" id="mapInput" accept=".json" onchange="loadMap(event)">
<input type="file" id="solutionInput" accept=".json" onchange="loadSolution(event)">
</div>
</div>
<div class="canvas-wrapper">
<canvas id="mapCanvas"></canvas>
</div>
<div id="playbackControls" class="visualization-panel hidden">
<h3>🎮 Управление воспроизведением</h3>
<div class="playback-controls">
<button class="playback-btn" onclick="playVisualization()" id="playBtn">▶ Play</button>
<button class="playback-btn" onclick="pauseVisualization()" id="pauseBtn" disabled>⏸ Pause</button>
<button class="playback-btn" onclick="resetVisualization()">⏮ Reset</button>
<button class="playback-btn" onclick="stepBackward()">⏪ Back</button>
<button class="playback-btn" onclick="stepForward()">⏩ Forward</button>
<div class="speed-control">
<label for="speedSlider">Скорость:</label>
<input type="range" id="speedSlider" min="1" max="10" value="5" onchange="updateSpeed()">
<span id="speedValue">5x</span>
</div>
</div>
</div>
<div id="stepInfo" class="step-info hidden">
<div class="info-item">
<span class="info-label">Шаг</span>
<span class="info-value" id="stepNumber">0 / 0</span>
</div>
<div class="info-item">
<span class="info-label">Позиция (x, y)</span>
<span class="info-value" id="positionValue">(0, 0)</span>
</div>
<div class="info-item">
<span class="info-label">Скорость (vx, vy)</span>
<span class="info-value" id="velocityValue">(0, 0)</span>
</div>
<div class="info-item">
<span class="info-label">Ускорение (ax, ay)</span>
<span class="info-value" id="accelerationValue">(0, 0)</span>
</div>
</div>
<div class="info">
<strong>💡 Инструкция:</strong>
• Сначала загрузите карту с точкой старта (тип 5)<br>
• Затем загрузите файл решения с векторами ускорений<br>
• Используйте кнопки управления для просмотра анимации<br>
• 🔵 Синяя линия - пройденная траектория<br>
• 🔴 Красный круг - текущая позиция<br>
• ➡️ Красная стрелка - направление и скорость движения
</div>
<div class="control-group">
<h3>📖 Легенда цветов</h3>
<div class="legend">
<div class="legend-item">
<div class="legend-color road"></div>
<span>Дорога (0)</span>
</div>
<div class="legend-item">
<div class="legend-color stone"></div>
<span>Камень (1)</span>
</div>
<div class="legend-item">
<div class="legend-color snow"></div>
<span>Снег (2)</span>
</div>
<div class="legend-item">
<div class="legend-color ice"></div>
<span>Лёд (3)</span>
</div>
<div class="legend-item">
<div class="legend-color checkpoint"></div>
<span>Чекпоинт (4)</span>
</div>
<div class="legend-item">
<div class="legend-color start"></div>
<span>Старт (5)</span>
</div>
</div>
</div>
</div>
<script src="player.js"></script>
</body>
</html>