Code - Pong Game

a simple game with raylib

為什麼會有這個東東呢? 因為想要知道ai的註解是否好用,不想寫註解的人有福了,找ai就對了。

#include "raylib.h"
#include <stdio.h>
#include <stdlib.h>

// 螢幕寬度常數
const int screenWidth = 800;
// 螢幕高度常數
const int screenHeight = 600;

// 球拍移動方向列舉
typedef enum {
    UP, // 向上
    DOWN, // 向下
} Dir;

// Clamp 函式:限制浮點數值在指定範圍內
float Clamp(float value, float min, float max)
{
    float result = (value < min) ? min : value; // 若 value 小於 min,則結果為 min
    if (result > max) // 若結果大於 max,則結果為 max
        result = max;
    return result;
}

// 球體結構定義
typedef struct Ball {
    float x, y; // 球心 X, Y 座標
    float speedx, speedy; // 球在 X, Y 軸的速度
    float radius; // 球的半徑
    void (*draw)(struct Ball*); // 繪製球的函式指標
    void (*update)(struct Ball*); // 更新球狀態的函式指標
} Ball;

// 球體繪製函式 (畫圓)
static void ball_drawCircle(Ball* b)
{
    DrawCircle((int)b->x, (int)b->y, b->radius, WHITE); // 使用 raylib 繪製白色圓形
}

// 球體更新函式
static void ball_update(Ball* b)
{
    // 更新球的位置
    b->x += b->speedx * GetFrameTime(); // X 軸位置根據速度和幀時間更新
    b->y += b->speedy * GetFrameTime(); // Y 軸位置根據速度和幀時間更新

    /* 邊界碰撞檢測 */
    // 上下邊界反彈
    if (b->y < 0) { // 若球超出上邊界
        b->y = 0; // 將球 Y 座標設為邊界位置
        b->speedy *= -1; // Y 軸速度反向 (反彈)
    }
    if (b->y > screenHeight) { // 若球超出下邊界
        b->y = screenHeight; // 將球 Y 座標設為邊界位置
        b->speedy *= -1; // Y 軸速度反向 (反彈)
    }
    // 左右邊界碰撞在 Update 函式中不處理,輸贏判斷在 Update 函式中處理
}

// 球體建立函式
Ball CreateBall(float x, float y, float speedx, float speedy, float radius)
{
    // 使用複合字面值初始化 Ball 結構體
    return (Ball) {
        .x = x, // 初始化 X 座標
        .y = y, // 初始化 Y 座標
        .speedx = speedx, // 初始化 X 軸速度
        .speedy = speedy, // 初始化 Y 軸速度
        .radius = radius, // 初始化半徑
        .draw = ball_drawCircle, // 設定繪製函式為 ball_drawCircle
        .update = ball_update, // 設定更新函式為 ball_update
    };
}

// 球拍結構定義
typedef struct Paddle {
    float x, y; // 球拍中心 X, Y 座標
    float speed; // 球拍移動速度
    float width, height; // 球拍寬度和高度
    Rectangle (*getRect)(struct Paddle*); // 取得球拍矩形範圍的函式指標
    void (*draw)(struct Paddle*); // 繪製球拍的函式指標
    void (*update)(struct Paddle*, Dir); // 更新球拍狀態的函式指標,Dir 參數表示移動方向
} Paddle;

// 取得球拍矩形範圍函式
Rectangle paddle_GetRect(struct Paddle* p)
{
    // 計算球拍的 Rectangle 結構,用於碰撞檢測和繪製
    return (Rectangle) {
        p->x - p->width / 2, // 左上角 X 座標 (中心 X 減去寬度一半)
        p->y - p->height / 2, // 左上角 Y 座標 (中心 Y 減去高度一半)
        p->width, // 寬度
        p->height // 高度
    };
}

// 球拍繪製函式
void paddle_draw(struct Paddle* p)
{
    DrawRectangleRec(p->getRect(p), WHITE); // 使用 raylib 繪製白色矩形,使用 getRect 取得矩形範圍
}

// 球拍更新函式
void paddle_update(struct Paddle* p, Dir d)
{
    switch (d) {
    case UP: // 向上移動
        p->y -= p->speed * GetFrameTime(); // Y 座標減去速度 (向上移動)
        p->y = Clamp(p->y, p->height / 2, screenHeight - p->height / 2); // 限制球拍 Y 座標在螢幕上下邊界內
        break;
    case DOWN: // 向下移動
        p->y += p->speed * GetFrameTime(); // Y 座標加上速度 (向下移動)
        p->y = Clamp(p->y, p->height / 2, screenHeight - p->height / 2); // 限制球拍 Y 座標在螢幕上下邊界內 (修正:原程式碼 `DOWN` 缺少 Clamp)
        break;
    default:
        break;
    }
}

// 球拍建立函式
Paddle CreatePaddle(float x, float y, float w, float h, float speed)
{
    // 使用複合字面值初始化 Paddle 結構體
    return (Paddle) {
        .x = x, // 初始化 X 座標
        .y = y, // 初始化 Y 座標
        .width = w, // 初始化寬度
        .height = h, // 初始化高度
        .speed = speed, // 初始化速度
        .getRect = paddle_GetRect, // 設定取得矩形範圍函式為 paddle_GetRect
        .draw = paddle_draw, // 設定繪製函式為 paddle_draw
        .update = paddle_update, // 設定更新函式為 paddle_update
    };
}

// 遊戲結構體,包含球和兩個球拍
typedef struct Game {
    Ball ball; // 球
    Paddle Player; // 玩家球拍
    Paddle ai; // AI 球拍
} Game;

Game game = {}; // 宣告一個 Game 型別的變數 game 並初始化 (全域變數)

// 更新遊戲狀態函式
char* Update(Game* game)
{
    game->ball.update(&game->ball); // 更新球的狀態

    /* 玩家控制 */
    // 左球拍控制 (W/S鍵)
    if (IsKeyDown(KEY_W))
        game->Player.update(&game->Player, UP); // 按下 W 鍵,玩家球拍向上移動
    if (IsKeyDown(KEY_S))
        game->Player.update(&game->Player, DOWN); // 按下 S 鍵,玩家球拍向下移動

    /* AI控制 */
    // 只在球向右移動時反應 (speedx > 0)
    if (game->ball.speedx > 0) {
        // 計算球和 AI 球拍的中心 Y 位置
        float ballCenterY = game->ball.y;
        float paddleCenterY = game->ai.y;

        // 根據位置調整 AI 球拍
        if (ballCenterY < paddleCenterY) { // 球在 AI 球拍上方
            game->ai.y -= game->ai.speed * GetFrameTime(); // AI 球拍向上移動
        } else if (ballCenterY > paddleCenterY) { // 球在 AI 球拍下方
            game->ai.y += game->ai.speed * GetFrameTime(); // AI 球拍向下移動
        }
        game->ai.y = Clamp(game->ai.y, game->ai.height / 2, screenHeight - game->ai.height / 2); // 限制 AI 球拍 Y 座標在螢幕上下邊界內 (新增 AI 球拍邊界限制)
    }

    /* 碰撞檢測 */
    // 左球拍 (玩家) 碰撞
    if (CheckCollisionCircleRec((Vector2) { game->ball.x, game->ball.y }, game->ball.radius, game->Player.getRect(&game->Player))) {
        if (game->ball.speedx < 0) { // 只處理球向左移動時與左球拍的碰撞
            game->ball.speedx *= -1.1f; // X 軸速度反向並稍微加速 (1.1 倍)
            // 根據碰撞位置調整 Y 軸速度,模擬旋轉效果
            game->ball.speedy = (game->ball.y - game->Player.y) / (game->Player.height / 2) * game->ball.speedx;
        }
    }

    // 右球拍 (AI) 碰撞
    if (CheckCollisionCircleRec((Vector2) { game->ball.x, game->ball.y }, game->ball.radius, game->ai.getRect(&game->ai))) {
        if (game->ball.speedx > 0) { // 只處理球向右移動時與右球拍的碰撞
            game->ball.speedx *= -1.1f; // X 軸速度反向並稍微加速 (1.1 倍)
            // 根據碰撞位置調整 Y 軸速度,模擬旋轉效果
            game->ball.speedy = (game->ball.y - game->ai.y) / (game->ai.height / 2) * -game->ball.speedx; // 注意 AI 球拍的 Y 軸速度方向相反
        }
    }

    char* winnerText = NULL; // 勝利訊息,預設為 NULL

    /* 勝利條件檢測 */
    if (game->ball.x < 0) // 球超出左邊界,AI 獲勝
        winnerText = "AI Wins!"; // 設定 AI 獲勝訊息
    if (game->ball.x > screenWidth) // 球超出右邊界,玩家獲勝
        winnerText = "Player Wins!"; // 設定玩家獲勝訊息

    /* 遊戲重置 */
    if (winnerText && IsKeyPressed(KEY_SPACE)) { // 若有勝利者且按下空白鍵
        // 重置球的位置和速度
        game->ball = CreateBall(screenWidth / 2, screenHeight / 2, 300, 300, 5); // 將球放回螢幕中心,恢復初始速度
        // 隨機化球的初始 X 軸速度方向 (改進:讓遊戲開始時球的方向更隨機)
        if (GetRandomValue(0, 1)) {
            game->ball.speedx *= -1; // 50% 機率反轉 X 軸速度
        }
        winnerText = NULL; // 清除勝利訊息,準備開始新局
    }

    return winnerText; // 返回勝利訊息 (若有,否則返回 NULL)
}

// 繪製遊戲畫面函式
void Draw(Game* game)
{
    game->ball.draw(&game->ball);    // 繪製球
    game->Player.draw(&game->Player); // 繪製玩家球拍
    game->ai.draw(&game->ai);        // 繪製 AI 球拍
}

// 主函式
int main()
{
    // 初始化視窗
    InitWindow(screenWidth, screenHeight, "Pong-Game -- Ethical Aniruddha"); // 建立視窗,設定寬高和標題
    SetWindowState(FLAG_VSYNC_HINT); // 啟用垂直同步 (建議啟用,減少畫面撕裂)

    // 初始化遊戲物件
    game.ball = CreateBall(
        screenWidth / 2,     // 初始 X 位置 (螢幕中心)
        screenHeight / 2,    // 初始 Y 位置 (螢幕中心)
        300, 300, 5         // 初始 X 軸速度, 初始 Y 軸速度, 半徑
    );
    // 隨機化球的初始 X 軸速度方向 (改進:讓遊戲開始時球的方向更隨機)
    if (GetRandomValue(0, 1)) {
        game.ball.speedx *= -1; // 50% 機率反轉 X 軸速度
    }
    game.Player = CreatePaddle(
        50, screenHeight / 2, // X 位置 (靠近左邊界), Y 位置 (螢幕垂直中心)
        10, 100, 500         // 寬度, 高度, 速度
    );
    game.ai = CreatePaddle(
        screenWidth - 50, screenHeight / 2, // X 位置 (靠近右邊界), Y 位置 (螢幕垂直中心)
        10, 100, 500         // 寬度, 高度, 速度
    );

    const char* winnerText = NULL; // 宣告勝利訊息指標,初始為 NULL

    // 主遊戲迴圈
    while (!WindowShouldClose()) { // 視窗未關閉時持續執行迴圈
        winnerText = Update(&game); // 更新遊戲狀態,並取得勝利訊息 (若有)

        BeginDrawing();           // 開始繪製畫面
        ClearBackground(BLACK);    // 清除背景為黑色

        // 繪製所有遊戲物件
        Draw(&game);

        // 顯示勝利訊息 (若有)
        if (winnerText) {
            int textWidth = MeasureText(winnerText, 60); // 計算勝利訊息文字寬度
            DrawText(winnerText,                               // 繪製文字
                screenWidth / 2 - textWidth / 2,           // X 座標置中
                screenHeight / 2 - 30,                      // Y 座標略為偏上
                60, YELLOW);                                // 字體大小 60,顏色黃色
        }

        DrawFPS(10, 10); // 顯示 FPS (幀率) 在左上角
        EndDrawing();             // 結束繪製,將畫面送出顯示
    }

    CloseWindow(); // 關閉視窗並釋放資源
    return 0;      // 程式結束
}

See also