# 期末專題-當機等候畫面Game ## befort ![螢幕擷取畫面 2024-06-05 023052](https://hackmd.io/_uploads/SkJtpAnVC.png) ## after ![螢幕擷取畫面 2024-06-05 022858](https://hackmd.io/_uploads/ryy9aC24A.png) ### 改變 你是否曾經在等待遊戲載入、更新或連線時,感到無聊而焦躁?現在,遊戲化正在改變這一切!遊戲化不再只局限於遊戲本身,而是將遊戲元素融入各個領域,讓原本乏味的等待時間變得更有趣,更具互動性。 #### 新增功能 * 遊戲提示: 在遊戲開始時,顯示一個簡短的提示,說明遊戲目標和操作方式。 * 玩家外觀: 修改了玩家的造型,使用圖片代替了之前的簡單方塊,讓玩家更具辨識度。 * 障礙物外觀: 修改了障礙物的造型,使用圖片代替了之前的簡單方塊,讓障礙物更具視覺衝擊。 * 道具外觀: 修改了加分道具和回血道具的造型,使用圖片代替了之前的簡單方塊,讓道具更直觀易懂。 * 遊戲速度: 遊戲進行中逐漸加速,增加遊戲難度和挑戰性。 * 遊戲結束: 當玩家生命值歸零時,遊戲結束,顯示遊戲結束畫面。 ### 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://)