# 期末專題-當機等候畫面Game
## befort

## after

### 改變
你是否曾經在等待遊戲載入、更新或連線時,感到無聊而焦躁?現在,遊戲化正在改變這一切!遊戲化不再只局限於遊戲本身,而是將遊戲元素融入各個領域,讓原本乏味的等待時間變得更有趣,更具互動性。
#### 新增功能
* 遊戲提示: 在遊戲開始時,顯示一個簡短的提示,說明遊戲目標和操作方式。
* 玩家外觀: 修改了玩家的造型,使用圖片代替了之前的簡單方塊,讓玩家更具辨識度。
* 障礙物外觀: 修改了障礙物的造型,使用圖片代替了之前的簡單方塊,讓障礙物更具視覺衝擊。
* 道具外觀: 修改了加分道具和回血道具的造型,使用圖片代替了之前的簡單方塊,讓道具更直觀易懂。
* 遊戲速度: 遊戲進行中逐漸加速,增加遊戲難度和挑戰性。
* 遊戲結束: 當玩家生命值歸零時,遊戲結束,顯示遊戲結束畫面。
### Code
:::spoiler Code
```javascript=
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>網站當機等候畫面</title>
<style>
body {
font-family: Arial, sans-serif;
margin: 0;
padding: 0;
display: flex;
justify-content: center;
align-items: center;
height: 100vh;
background-image: url('https://s1.pearlcdn.com/TW/Upload/thumbnail/2024/9FIJ2EMXMQ55VBF720240214151849536.400x225.jpg'); /* 替换为你的背景图片路径 */
background-size: cover;
background-position: center;
}
.content {
text-align: center;
background-color: rgba(255, 255, 255, 0.8); /* 添加半透明的背景色 */
padding: 20px;
border-radius: 10px;
}
h1 {
color: #333;
}
p {
color: #666;
}
.loader {
border: 8px solid #f3f3f3;
border-top: 8px solid #3498db;
border-radius: 50%;
width: 60px;
height: 60px;
animation: spin 2s linear infinite;
margin: 20px auto;
}
@keyframes spin {
0% { transform: rotate(0deg); }
100% { transform: rotate(360deg); }
}
#game-container {
margin-top: 20px;
}
#runner-game {
position: relative;
width: 300px;
height: 150px;
background-color: #f0f0f0;
overflow: hidden;
border: 2px solid #333;
}
.runner {
position: absolute;
width: 30px;
height: 30px;
/* 更改玩家的外觀,例如使用圖片 */
background-image: url('https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcRKEgB1iujPXolSvCCGoqyabEI17TzlT7oiaw&s'); /* 替换为你的玩家图片路径 */
background-size: cover;
bottom: 0;
left: 10px;
}
.obstacle {
position: absolute;
width: 30px;
height: 30px;
/* 更改障礙物的外觀,例如使用圖片 */
background-image: url('https://img.lovepik.com/free-png/20211207/lovepik-prohibition-of-obstacles-png-image_401378948_wh1200.png'); /* 替换为你的障礙物图片路径 */
background-size: cover;
bottom: 0;
}
.powerup {
position: absolute;
width: 30px;
height: 30px;
/* 更改回血道具的外觀,例如使用圖片 */
background-image: url(''); /* 替换为你的回血道具图片路径 */
background-size: cover;
bottom: 0;
}
.bonus {
position: absolute;
width: 30px;
height: 30px;
/* 更改加分道具的外觀,例如使用圖片 */
background-image: url('https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcQJl_xjkeqjU6PvCyvVJVdFe4vHkGgCKViJ5A&s');
background-size: cover;
bottom: 0;
}
#game-over {
display: none;
color: red;
font-weight: bold;
}
#start-screen {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background-color: rgba(255, 255, 255, 0.8);
z-index: 1;
}
#start-screen h2 {
font-size: 2em;
margin-bottom: 10px;
}
#start-screen p {
font-size: 1.2em;
margin-bottom: 20px;
}
#start-button {
padding: 10px 20px;
background-color: #3498db;
color: white;
border: none;
border-radius: 5px;
cursor: pointer;
font-size: 1.2em;
}
</style>
</head>
<body>
<div class="content">
<h1>遊戲正在加載中</h1>
<p>請稍等片刻,我們正在處理問題!</p>
<div class="loader"></div>
<div id="start-screen">
<h2>歡迎來到跑酷遊戲!</h2>
<p>使用↑↓鍵控制角色上下移動,躲避障礙物,收集加分道具,盡情奔跑吧!</p>
<p>按下score play game!!</p>
</div>
<div id="game-container">
<div id="runner-game">
<div class="runner" id="runner"></div>
</div>
<p id="score">Score: 0</p>
<p id="lives">HP: 3</p>
<p id="game-over">遊戲結束!按下space重新開始</p>
</div>
</div>
<script>
const runnerGame = document.getElementById('runner-game');
const runner = document.getElementById('runner');
const scoreElement = document.getElementById('score');
const livesElement = document.getElementById('lives');
const gameOverElement = document.getElementById('game-over');
const startScreen = document.getElementById('start-screen');
// const startButton = document.getElementById('start-button'); // 不需要了
let score = 0;
let gameRunning = false;
let speed = 2;
let lives = 3;
let obstacles = [];
let powerups = [];
let bonuses = [];
// 移動功能
function moveRunner(event) {
const runnerTop = parseInt(window.getComputedStyle(runner).getPropertyValue('top'));
if (event.key === 'ArrowUp' && runnerTop > 0) {
runner.style.top = `${runnerTop - 30}px`;
} else if (event.key === 'ArrowDown' && runnerTop < 120) {
runner.style.top = `${runnerTop + 30}px`;
}
}
// 每次按下上下鍵移動
document.addEventListener('keydown', moveRunner);
// 創建障礙物
function createObstacle() {
const obstacle = document.createElement('div');
obstacle.classList.add('obstacle');
obstacle.style.right = '0px';
obstacle.style.top = `${Math.floor(Math.random() * 3) * 30}px`;
runnerGame.appendChild(obstacle);
obstacles.push(obstacle);
}
// 創建回命道具
function createPowerup() {
const powerup = document.createElement('div');
powerup.classList.add('powerup');
powerup.style.right = '0px';
powerup.style.top = `${Math.floor(Math.random() * 3) * 30}px`;
runnerGame.appendChild(powerup);
powerups.push(powerup);
}
// 創建加分道具
function createBonus() {
const bonus = document.createElement('div');
bonus.classList.add('bonus');
bonus.style.right = '0px';
bonus.style.top = `${Math.floor(Math.random() * 3) * 30}px`;
runnerGame.appendChild(bonus);
bonuses.push(bonus);
}
// 移動物體(障礙物和道具)
function moveItems(items) {
items.forEach((item, index) => {
let itemRight = parseInt(window.getComputedStyle(item).getPropertyValue('right'));
if (itemRight > 330) {
item.remove();
items.splice(index, 1);
} else {
item.style.right = `${itemRight + speed}px`;
}
});
}
// 碰撞檢測
function checkCollision() {
const runnerRect = runner.getBoundingClientRect();
// 障礙物碰撞檢測
obstacles.forEach((obstacle, index) => {
const obstacleRect = obstacle.getBoundingClientRect();
if (
runnerRect.right > obstacleRect.left &&
runnerRect.left < obstacleRect.right &&
runnerRect.bottom > obstacleRect.top &&
runnerRect.top < obstacleRect.bottom
) {
lives--;
livesElement.textContent = `HP: ${lives}`;
obstacle.remove();
obstacles.splice(index, 1);
if (lives === 0) {
gameRunning = false;
gameOverElement.style.display = 'block';
}
}
});
// 回命道具碰撞檢測
powerups.forEach((powerup, index) => {
const powerupRect = powerup.getBoundingClientRect();
if (
runnerRect.right > powerupRect.left &&
runnerRect.left < powerupRect.right &&
runnerRect.bottom > powerupRect.top &&
runnerRect.top < powerupRect.bottom
) {
if (lives < 3) { // 只有在生命少于3的时候才出现回血道具
lives++;
livesElement.textContent = `HP: ${lives}`;
}
powerup.remove();
powerups.splice(index, 1);
}
});
// 加分道具碰撞檢測
bonuses.forEach((bonus, index) => {
const bonusRect = bonus.getBoundingClientRect();
if (
runnerRect.right > bonusRect.left &&
runnerRect.left < bonusRect.right &&
runnerRect.bottom > bonusRect.top &&
runnerRect.top < bonusRect.bottom
) {
score += 5;
scoreElement.textContent = `score: ${score}`;
bonus.remove();
bonuses.splice(index, 1);
}
});
}
// 檢查道具是否重疊
function checkItemOverlap(newItem, items) {
const newItemRect = newItem.getBoundingClientRect();
const runnerRect = runner.getBoundingClientRect(); // 检查与玩家角色的重叠
// 检查与其他道具的重叠
for (const item of items) {
const itemRect = item.getBoundingClientRect();
if (
newItemRect.right > itemRect.left &&
newItemRect.left < itemRect.right &&
newItemRect.bottom > itemRect.top &&
newItemRect.top < itemRect.bottom
) {
return true;
}
}
// 检查与玩家角色的重叠
if (
newItemRect.right > runnerRect.left &&
newItemRect.left < runnerRect.right &&
newItemRect.bottom > runnerRect.top &&
newItemRect.top < runnerRect.bottom
) {
return true;
}
return false;
}
// 檢查是否越過障礙物
function checkObstaclePassed(obstacle) {
const obstacleRight = parseInt(window.getComputedStyle(obstacle).getPropertyValue('right'));
if (obstacleRight < 0) {
score++;
scoreElement.textContent = `score: ${score}`;
obstacle.remove();
return true;
}
return false;
}
// 遊戲主循環
function gameLoop() {
if (gameRunning) {
// 根據遊戲時間調整出現機率
let obstacleChance = 0.02 + (score / 100); // 障礙物出現機率稍微提升
let powerupChance = 0.005 + (score / 450); // 回血道具出現機率降低
let bonusChance = 0.002 + (score / 300); // 加分道具出現機率降低
// 控制出現頻率,使道具更分散
if (Math.random() < obstacleChance) {
if (obstacles.length < 3) { // 最多只允許存在兩個障礙物
createObstacle();
// 檢查新創建的障礙物是否与其他道具重疊
if (checkItemOverlap(obstacles[obstacles.length - 1], powerups.concat(bonuses))) {
obstacles[obstacles.length - 1].remove();
obstacles.pop();
}
}
}
if (Math.random() < powerupChance && powerups.length < 1 && lives < 3) { // 最多只允許存在一個回血道具,且生命值小于3时才出现
createPowerup();
// 檢查新創建的回血道具是否与其他道具重疊
if (checkItemOverlap(powerups[powerups.length - 1], obstacles.concat(bonuses))) {
powerups[powerups.length - 1].remove();
powerups.pop();
}
}
if (Math.random() < bonusChance && bonuses.length < 1) { // 最多只允許存在一個加分道具
createBonus();
// 檢查新創建的加分道具是否与其他道具重疊
if (checkItemOverlap(bonuses[bonuses.length - 1], obstacles.concat(powerups))) {
bonuses[bonuses.length - 1].remove();
bonuses.pop();
}
}
// 檢查障礙物是否越過
obstacles.forEach((obstacle, index) => {
if (checkObstaclePassed(obstacle)) {
obstacles.splice(index, 1);
}
});
moveItems(obstacles);
moveItems(powerups);
moveItems(bonuses);
checkCollision();
speed += 0.0015; // 遊戲進行中逐漸加速
requestAnimationFrame(gameLoop);
}
}
// 初始化遊戲
function startGame() {
score = 0;
speed = 2;
lives = 3;
gameRunning = true;
gameOverElement.style.display = 'none';
obstacles.forEach(obstacle => obstacle.remove());
powerups.forEach(powerup => powerup.remove());
bonuses.forEach(bonus => bonus.remove());
obstacles = [];
powerups = [];
bonuses = [];
runner.style.top = '60px'; // 重置角色位置
scoreElement.textContent = `score: ${score}`;
livesElement.textContent = `HP: ${lives}`;
gameLoop();
startScreen.style.display = 'none';
}
// 遊戲結束後重新開始
document.addEventListener('keydown', (event) => {
if (event.key === ' ' && !gameRunning) {
startGame();
}
});
</script>
</body>
</html>
```
:::
### 總結
**1. 遊戲邏輯**
* 玩家控制角色上下移動躲避障礙物
* 遊戲會不斷生成障礙物,並隨機生成道具
* 程式碼會檢查角色是否碰到障礙物或道具
* 遊戲速度會逐漸加快,難度增加
**2. 程式碼結構**
* 角色移動:moveRunner 函數
* 障礙物生成:createObstacle 函數
* 道具生成:createPowerup、createBonus 函數
* 碰撞檢測:checkCollision 函數
* 遊戲循環:gameLoop 函數
**3. 重要概念**
* event.key:取得玩家按下的鍵盤方向
* getComputedStyle:取得元素的樣式屬性值
* requestAnimationFrame:實現遊戲循環,讓畫面不斷更新
玩家需要控制角色上下移動,躲避從右邊不斷出現的障礙物,同時收集加分道具和回血道具來提升得分和存活率。遊戲會逐漸加速,增加難度,直到玩家生命值歸零,遊戲結束。
##### **Git:**[https://sansan20036.github.io/work1/](https://)