Viết game flappy Javascript thuần túy

10 min read

Mục tiêu: Làm game flappy bird với Javascript thuần túy

Chuẩn bị:

Các hình ảnh dùng trong game:

Lấy base64 string từ trang https://www.base64-image.de/

Bắt đầu code

Viết 1 trang index.html thuần túy, trong đó cần chứa:

  • Canvas
  • Include game.js: là source code game chúng ta sẽ code bên trong
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Flappy Bird</title>
    <style>
        body {
            margin: 0;
            display: flex;
            justify-content: center;
            align-items: center;
            height: 100vh;
            background-color: #70c5ce;
        }

        #game-container {
            position: relative;
            width: 800px;
            height: 600px;
            overflow: hidden;
            border: 2px solid #000;
        }

        canvas {
            display: block;
        }
    </style>
</head>
<body>
    <div id="game-container">
        <canvas id="gameCanvas"></canvas>
    </div>
    <script src="game.js"></script>
</body>
</html>

Game.js file

Javascript sử dụng Canvas để vẽ các hình ảnh lên đó.

Ví dụ: Để vẽ 1 hình chữ nhật, ta sẽ cần code như sau

const canvas = document.getElementById('gameCanvas');
const context = canvas.getContext('2d');

canvas.width = 800;
canvas.height = 600;

// Draw a rectangle
context.fillStyle = 'blue'; // Set the fill color
context.fillRect(100, 100, 200, 150); // Draw the rectangle (x, y, width, height)

Để vẽ nhân vật Chim, chúng ta sẽ load base64 string vào Image và hiển thị lên canvas

const canvas = document.getElementById('gameCanvas');
const context = canvas.getContext('2d');

canvas.width = 800;
canvas.height = 600;

// Base64 image of the Bird
const base64Image = '';

// Create a new image
const image = new Image();
image.onload = function() {
    context.drawImage(image, 0, 0, canvas.width, canvas.height);
};
image.src = base64Image;

Để Chim có thể di chuyển, chúng ta cần Thay đổi tọa độ hiển thị của Chim theo Thời gian.
Đây là khái niệm “game loop”

Hãy viết 1 game loop đơn giản

function updateAndDraw() {    
}

function gameLoop() {
    updateAndDraw();
    requestAnimationFrame(gameLoop); // Call gameLoop again on the next frame
}

gameLoop();

Chúng ta cần Thay đổi tọa độ vẽ Chim trong hàm updateAndDraw

let x = 0;
let speed = 2;
function updateAndDraw() {
    x = x + speed;
    context.drawImage(image, x, 0, canvas.width, canvas.height);
}

Sử dụng document.addEventListener(‘keydown’, (event) => {}) để lắng nghe sự kiện phím “Space” được gõ

Gom tất cả các logic bên trên vào 1 file, gồm:

  • Load canvas
  • Hiển thị ảnh từ base64 string
  • Game loop & di chuyển nhân vật
  • Lắng nghe sự kiện từ bàn phím

Ta sẽ có 1 file html gồm source code như sau <code full để mn tham khảo>

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Flappy Bird</title>
    <style>
        body {
            margin: 0;
            display: flex;
            justify-content: center;
            align-items: center;
            height: 100vh;
            background-color: #70c5ce;
        }

        #game-container {
            position: relative;
            width: 800px;
            height: 600px;
            overflow: hidden;
            border: 2px solid #000;
        }

        canvas {
            display: block;
        }
    </style>
</head>
<body>
    <div id="game-container">
        <canvas id="gameCanvas"></canvas>
    </div>
    <script>
        const canvas = document.getElementById('gameCanvas');
        const context = canvas.getContext('2d');

        canvas.width = 800;
        canvas.height = 600;

        // Base64 images
        const birdImage = ''; // Replace with actual base64 string
        const pipeImage = ''; // Replace with actual base64 string
        const backgroundImage = ''; // Replace with actual base64 string

        const bird = {
            x: 50,
            y: 150,
            width: 34, // Set the width of bird image
            height: 24, // Set the height of bird image
            gravity: 0.6,
            lift: -15,
            velocity: 0
        };

        const pipes = [];
        const pipeWidth = 52; // Set the width of pipe image
        const pipeGap = 150;
        const pipeFrequency = 90; // Frames
        let frameCount = 0;
        let score = 0;
        let gameOver = false;

        // Create image objects
        const birdImg = new Image();
        birdImg.src = birdImage;

        const pipeImg = new Image();
        pipeImg.src = pipeImage;

        const bgImg = new Image();
        bgImg.src = backgroundImage;

        function updateAndDraw() {
            // Update bird's position
            bird.velocity += bird.gravity;
            bird.y += bird.velocity;

            // Check for collision with the top and bottom of the canvas
            if (bird.y + bird.height > canvas.height || bird.y < 0) {
                gameOver = true;
            }

            // Update pipes
            if (frameCount % pipeFrequency === 0) {
                const top = Math.floor(Math.random() * (canvas.height - pipeGap));
                pipes.push({ x: canvas.width, top: top });
            }

            pipes.forEach(pipe => {
                pipe.x -= 2;
            });

            // Remove off-screen pipes and update score
            pipes.forEach(pipe => {
                if (pipe.x + pipeWidth < 0) {
                    pipes.shift();
                    score++;
                }
            });

            // Check for collisions with pipes
            pipes.forEach(pipe => {
                if (
                    bird.x < pipe.x + pipeWidth &&
                    bird.x + bird.width > pipe.x &&
                    (bird.y < pipe.top || bird.y + bird.height > pipe.top + pipeGap)
                ) {
                    gameOver = true;
                }
            });

            // Draw everything
            context.clearRect(0, 0, canvas.width, canvas.height);
            context.drawImage(bgImg, 0, 0, canvas.width, canvas.height);
            pipes.forEach(pipe => {
                context.drawImage(pipeImg, pipe.x, pipe.top - pipeImg.height); // Draw top pipe
                context.drawImage(pipeImg, pipe.x, pipe.top + pipeGap); // Draw bottom pipe
            });
            context.drawImage(birdImg, bird.x, bird.y, bird.width, bird.height);
            context.fillStyle = 'black';
            context.font = '20px Arial';
            context.fillText(`Score: ${score}`, 10, 20);

            if (gameOver) {
                context.fillStyle = 'black';
                context.font = '40px Arial';
                context.fillText('Game Over', canvas.width / 2 - 100, canvas.height / 2);
                context.fillText(`Score: ${score}`, canvas.width / 2 - 100, canvas.height / 2 + 50);
                context.fillText('Press Space to Restart', canvas.width / 2 - 150, canvas.height / 2 + 100);
            } else {
                frameCount++;
                requestAnimationFrame(updateAndDraw);
            }
        }

        function resetGame() {
            bird.y = 150;
            bird.velocity = 0;
            pipes.length = 0;
            score = 0;
            frameCount = 0;
            gameOver = false;
        }

        document.addEventListener('keydown', (event) => {
            if (event.code === 'Space') {
                if (gameOver) {
                    resetGame();
                    updateAndDraw();
                } else {
                    bird.velocity = bird.lift;
                }
            }
        });

        // Start the game loop
        bgImg.onload = () => {
            updateAndDraw();
        };
    </script>
</body>
</html>
Avatar photo

Leave a Reply

Your email address will not be published. Required fields are marked *