為了方便繪圖測試,把sdl 3給加一層包裝,懶人專用… 使用例…
#include <stdlib.h>
#define LAZY_GRAPHIC_IMPLEMENTATION
#include "lazygraphic.h"
#include <math.h>
#include <time.h>
int main(int argc, char* argv[])
{
RGBA bkgColor = { .color = 0xFF181818 };
RGBA cirColor = { .color = 0xFF18D818 };
lazyGr.Init(800, 600, "LAZY !!");
srand(time(NULL));
while (!lazyGr.Done()) {
lazyGr.Clear(bkgColor);
for (int i = 0; i < 50; i++) {
int rx = rand() % 400;
int ry = rand() % 300;
int rr = rand() % 200;
cirColor.color = rand() | 0xFF000000;
lazyGr.Disk(400 - rr/2 + rx/2, 300 - rr/2 + ry/2 , rr, cirColor);
}
lazyGr.Update();
lazyGr.Delay(16);
}
lazyGr.Fini();
return 0;
}
#ifndef __LAZY_GRAPHIC_H__
#define __LAZY_GRAPHIC_H__
#include <SDL3\SDL.h>
#include <SDL3\SDL_main.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
// 全局變量定義
typedef struct {
union {
SDL_Color c;
uint32_t color;
};
} RGBA;
// 預定義顏色常量,使用 uint32_t 格式表示顏色,包含 Alpha 通道 (0xFF)
const RGBA RGB_Black = { .color = 0xFF000000 }; // 黑色
const RGBA RGB_Red = { .color = 0xFFFF0000 }; // 紅色
const RGBA RGB_Green = { .color = 0xFF00FF00 }; // 綠色
const RGBA RGB_Blue = { .color = 0xFF0000FF }; // 藍色
const RGBA RGB_Cyan = { .color = 0xFF00FFFF }; // 青色
const RGBA RGB_Magenta = { .color = 0xFFFF00FF }; // 洋紅色
const RGBA RGB_Yellow = { .color = 0xFFFFFF00 }; // 黃色
const RGBA RGB_White = { .color = 0xFFFFFFFF }; // 白色
const RGBA RGB_Gray = { .color = 0xFF808080 }; // 灰色
const RGBA RGB_Grey = { .color = 0xFFC0C0C0 }; // 淺灰色 (與 Gray 類似,但更淺)
const RGBA RGB_Maroon = { .color = 0xFF000080 }; // 栗色
const RGBA RGB_Darkgreen = { .color = 0xFF008000 }; // 深綠色
const RGBA RGB_Navy = { .color = 0xFF800000 }; // 海軍藍
const RGBA RGB_Teal = { .color = 0xFF808000 }; // 青色 (與 Cyan 類似,但更深)
const RGBA RGB_Purple = { .color = 0xFF800080 }; // 紫色
const RGBA RGB_Olive = { .color = 0xFF008080 }; // 橄欖綠
// LazyGraphic 結構體,封裝了圖形庫的實例所需的數據
typedef struct {
SDL_Window* Window; // SDL 窗口指針
SDL_Renderer* Renderer; // SDL 渲染器指針
SDL_Event event; // SDL 事件結構體,用於處理事件
RGBA SelColor; // 當前選定的繪圖顏色
uint32_t w; // 窗口寬度
uint32_t h; // 窗口高度
const bool* inkeys; // 鍵盤按鍵狀態數組指針
float mouseX; // 鼠標 X 坐標
float mouseY; // 鼠標 Y 坐標
bool LMB; // 鼠標左鍵狀態 (true: 按下, false: 鬆開)
bool RMB; // 鼠標右鍵狀態 (true: 按下, false: 鬆開)
} LazyGraphic;
// LazyGraphic 的實例 (靜態全局變量,在實現文件中定義)
// LazyFnTable 結構體,包含了圖形庫提供的函數指針
typedef struct {
bool (*KeyDown)(int key); // 檢查按鍵是否被按下
bool (*Init)(int w, int h, char* title); // 初始化圖形庫和窗口
int (*ModeNone)(void); // 設置混合模式為 None (不混合)
int (*ModeBlend)(void); // 設置混合模式為 Blend (Alpha 混合)
int (*ModeAdd)(void); // 設置混合模式為 Add (加法混合)
int (*ModeMod)(void); // 設置混合模式為 Mod (調製混合)
void (*Update)(void); // 更新渲染,顯示畫面
void (*ReadKeys)(void); // 讀取鍵盤狀態
void (*GetMouseState)(int* x, int* y); // 獲取鼠標狀態和坐標
unsigned long (*GetTicks)(void); // 獲取程序運行時間 (毫秒)
void (*Delay)(int ms); // 程序延遲 (毫秒)
bool (*Done)(void); // 檢查程序是否應該結束 (例如,按下 ESC 或關閉窗口)
void (*Fini)(void); // 釋放圖形庫資源
void (*Clear)(RGBA col); // 使用指定顏色清空屏幕
void (*Point)(int x, int y, RGBA col); // 繪製一個點
bool (*Line)(int x1, int y1, int x2, int y2, RGBA col); // 繪製一條線段
bool (*Circle)(int xc, int yc, int radius, RGBA col); // 繪製一個圓環
bool (*Disk)(int xc, int yc, int radius, RGBA col); // 繪製一個實心圓
bool (*Rect)(int x, int y, int w, int h, RGBA col); // 繪製一個矩形框
bool (*FillRect)(int x, int y, int w, int h, RGBA col); // 繪製一個實心矩形
} LazyFnTable;
// SB_Image 結構體,用於表示和操作簡單位圖圖像
typedef struct SB_Image {
SDL_Texture* Tex; // SDL 紋理指針,存儲圖像數據
int w, h; // 圖像寬度和高度
void (*Boundary)(struct SB_Image* self, int* w, int* h); // 獲取圖像邊界 (寬度和高度)
void (*Draw)(struct SB_Image* simg, int x, int y, float angle); // 繪製圖像到屏幕
void (*Destroy)(struct SB_Image* self); // 銷毀圖像資源
} SB_Image;
// 創建新的 SB_Image 對象,從文件中加載圖像
SB_Image* NewSBImage(char* filename);
#endif
#ifdef LAZY_GRAPHIC_IMPLEMENTATION
////////////////////////////////////////////////////
// 靜態全局 LazyGraphic 實例,用於封裝圖形庫狀態
static LazyGraphic lg = { 0 };
// _Key_Down 函數,檢查指定按鍵是否被按下
static bool _Key_Down(int key)
{
// 如果鍵盤狀態未初始化,則返回 false
if (!lg.inkeys)
return false;
// 檢查指定按鍵的狀態,如果非 0 則表示按下
return (lg.inkeys[key] != 0);
}
/*
_Init 函數,初始化圖形庫和窗口
*/
static bool _Init(int w, int h, char* title)
{
// 初始化標誌,默認初始化成功
bool bRet = true;
// 設置 LazyGraphic 實例的寬度和高度
lg.w = w;
lg.h = h;
// 通知 SDL 主程序已準備好
SDL_SetMainReady();
// 初始化 SDL 視頻子系統
if (!SDL_Init(SDL_INIT_VIDEO)) {
printf("SDL could not initialize! SDL_Error: %s\n", SDL_GetError());
bRet = false; // 初始化失敗
return bRet;
};
// 創建 SDL 窗口
lg.Window = SDL_CreateWindow(title, w, h, 0);
if (lg.Window == NULL) {
printf("Window could not be created! SDL_Error: %s\n", SDL_GetError());
bRet = false; // 窗口創建失敗
return bRet;
};
// 創建 SDL 渲染器,用於在窗口上繪圖,使用硬件加速
lg.Renderer = SDL_CreateRenderer(lg.Window, NULL);
if (lg.Renderer == NULL) {
printf("Renderer could not be created! SDL_Error: %s\n", SDL_GetError());
bRet = false; // 渲染器創建失敗
return bRet;
};
return bRet; // 初始化成功
}
////////////////////////////////////////////////////////////////////////////////////
// _Mode_None 函數,設置渲染器混合模式為 None (不混合)
static int _Mode_None()
{
return SDL_SetRenderDrawBlendMode(lg.Renderer, SDL_BLENDMODE_NONE);
}
////////////////////////////////////////////////////////////////////////////////////
// _Mode_Blend 函數,設置渲染器混合模式為 Blend (Alpha 混合)
static int _Mode_Blend()
{
return SDL_SetRenderDrawBlendMode(lg.Renderer, SDL_BLENDMODE_BLEND);
}
////////////////////////////////////////////////////////////////////////////////////
// _Mode_Add 函數,設置渲染器混合模式為 Add (加法混合)
static int _Mode_Add()
{
return SDL_SetRenderDrawBlendMode(lg.Renderer, SDL_BLENDMODE_ADD);
}
////////////////////////////////////////////////////////////////////////////////////
// _Mode_Mod 函數,設置渲染器混合模式為 Mod (調製混合)
static int _Mode_Mod()
{
return SDL_SetRenderDrawBlendMode(lg.Renderer, SDL_BLENDMODE_MOD);
}
////////////////////////////////////////////////////////////////////////////////////
// _Update 函數,更新渲染,顯示當前幀畫面
static void _Update() { SDL_RenderPresent(lg.Renderer); }
///////////////////////////////////////////////////////////////////
// sdl input function ...
///////////////////////////////////////////////////////////////////
// _ReadKeys 函數,讀取當前鍵盤狀態
static void _ReadKeys()
{
SDL_PumpEvents(); // 處理事件隊列,更新鍵盤和鼠標狀態
lg.inkeys = SDL_GetKeyboardState(NULL); // 獲取當前鍵盤按鍵狀態數組
}
////////////////////////////////////////////////////////////////////////////////////
// _GetMouseState 函數,獲取鼠標狀態和坐標
static void _GetMouseState(int* x, int* y)
{
uint8_t mouseState = SDL_GetMouseState(&lg.mouseX, &lg.mouseY); // 獲取鼠標坐標和按鍵狀態
*x = (int)lg.mouseX; // 存儲鼠標 X 坐標到指針
*y = (int)lg.mouseY; // 存儲鼠標 Y 坐標到指針
// 檢查鼠標左鍵是否按下 (SDL_BUTTON_LMASK == 1)
if (mouseState & SDL_BUTTON_LMASK)
lg.LMB = true;
else
lg.LMB = false;
// 檢查鼠標右鍵是否按下 (SDL_BUTTON_RMASK == 4)
if (mouseState & SDL_BUTTON_RMASK)
lg.RMB = true;
else
lg.RMB = false;
}
// _GetTicks 函數,返回程序啟動以來經過的毫秒數
static unsigned long _GetTicks() { return SDL_GetTicks(); }
//////////////////////////////////////////////////////////////////////
// _Delay 函數,程序延遲指定毫秒數
static void _Delay(int ms) { SDL_Delay(ms); }
////////////////////////////////////////////////////////////////////////////////////
// _Done 函數,檢查程序是否應該結束 (按下 ESC 鍵或窗口關閉事件)
static bool _Done(void)
{
// 檢查是否有待處理的 SDL 事件
while (SDL_PollEvent(&lg.event)) {
// 如果事件類型為 SDL_EVENT_QUIT (窗口關閉事件)
if (lg.event.type == SDL_EVENT_QUIT)
return true; // 返回 true,表示程序應該結束
}
_ReadKeys(); // 讀取鍵盤狀態
// 檢查 ESC 鍵是否被按下 (SDL_SCANCODE_ESCAPE)
if (lg.inkeys[SDL_SCANCODE_ESCAPE])
return true; // 返回 true,表示程序應該結束
return false; // 返回 false,表示程序繼續運行
}
////////////////////////////////////////////////////////////////////////////////////
// _Fini 函數,釋放圖形庫資源
static void _Fini()
{
SDL_DestroyRenderer(lg.Renderer); // 銷毀渲染器
lg.Renderer = NULL;
SDL_DestroyWindow(lg.Window); // 銷毀窗口
lg.Window = NULL;
SDL_Quit(); // 退出 SDL 子系統
}
////////////////////////////////////////////////////////////////////////////////////
// _Clear 函數,使用指定顏色清空屏幕
static void _Clear(RGBA col)
{
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
SDL_RenderClear(lg.Renderer); // 使用當前繪圖顏色清空渲染目標
}
////////////////////////////////////////////////////////////////////////////////////
// _Point 函數,繪製一個點
static void _Point(int x, int y, RGBA col)
{
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
SDL_RenderPoint(lg.Renderer, x, y); // 繪製一個點
}
////////////////////////////////////////////////////////////////////////////////////
// _Line 函數,繪製一條線段
static bool _Line(int x1, int y1, int x2, int y2, RGBA col)
{
// 邊界檢查:確保線段端點坐標在有效範圍內
if (x1 < 0 || x1 >= lg.w || x2 < 0 || x2 >= lg.w || y1 < 0 || y1 >= lg.h || y2 < 0 || y2 >= lg.h)
return false; // 坐標超出屏幕範圍,繪製失敗
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
SDL_RenderLine(lg.Renderer, x1, y1, x2, y2); // 繪製線段
return true; // 繪製成功
}
////////////////////////////////////////////////////////////////////////////
// _Circle 函數,使用 Bresenham 算法繪製圓環
static bool _Circle(int xc, int yc, int radius, RGBA col)
{
// 邊界檢查:確保圓環在屏幕範圍內
if (xc - radius < 0 || xc + radius >= lg.w || yc - radius < 0 || yc + radius >= lg.h)
return false; // 圓環超出屏幕範圍,繪製失敗
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
int x = 0;
int y = radius;
int p = 3 - (radius << 1); // Bresenham 算法的初始決策參數
int a, b, c, d, e, f, g, h;
while (x <= y) {
// 利用圓的八分對稱性,一次計算八個點
a = xc + x; // 8-way symmetry calculation
b = yc + y;
c = xc - x;
d = yc - y;
e = xc + y;
f = yc + x;
g = xc - y;
h = yc - x;
SDL_RenderPoint(lg.Renderer, a, b); // 繪製點 (x, y)
SDL_RenderPoint(lg.Renderer, c, d); // 繪製點 (-x, -y)
SDL_RenderPoint(lg.Renderer, e, f); // 繪製點 (y, x)
SDL_RenderPoint(lg.Renderer, g, f); // 繪製點 (-y, x)
if (x > 0) { // 避免重複繪製在同一位置的點
SDL_RenderPoint(lg.Renderer, a, d); // 繪製點 (x, -y)
SDL_RenderPoint(lg.Renderer, c, b); // 繪製點 (-x, y)
SDL_RenderPoint(lg.Renderer, e, h); // 繪製點 (y, -x)
SDL_RenderPoint(lg.Renderer, g, h); // 繪製點 (-y, -x)
}
if (p < 0)
p += (x++ << 2) + 6; // 決策參數小於 0,選擇上方像素
else
p += ((x++ - y--) << 2) + 10; // 決策參數大於等於 0,選擇右上方像素
}
return true; // 繪製成功
}
///////////////////////////////////////////////////////////////////////////////
// _Disk 函數,使用填充的 Bresenham 算法繪製實心圓
static bool _Disk(int xc, int yc, int radius, RGBA col)
{
// 邊界檢查:確保圓在屏幕範圍內
if (xc + radius < 0 || xc - radius >= lg.w || yc + radius < 0 || yc - radius >= lg.h)
return false; // 圓超出屏幕範圍,繪製失敗 (優化:提前判斷全部像素是否超出屏幕)
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
int x = 0;
int y = radius;
int p = 3 - (radius << 1); // Bresenham 算法的初始決策參數
int a, b, c, d, e, f, g, h;
int pb = yc + radius + 1, // 前一個 y 坐標值,用於優化水平線繪製,初始值設置為範圍外
pd = yc + radius + 1; // 前一個 -y 坐標值,初始值設置為範圍外
while (x <= y) {
// 計算八分對稱點
a = xc + x;
b = yc + y;
c = xc - x;
d = yc - y;
e = xc + y;
f = yc + x;
g = xc - y;
h = yc - x;
// 繪製水平線,避免重複繪製同一水平線
if (b != pb)
SDL_RenderLine(lg.Renderer, c, b, a, b); // 繪製水平線 (y 坐標)
if (d != pd)
SDL_RenderLine(lg.Renderer, c, d, a, d); // 繪製水平線 (-y 坐標)
if (f != b)
SDL_RenderLine(lg.Renderer, g, f, e, f); // 繪製水平線 (x 坐標)
if (h != d && h != f) // 避免重複繪製以及與前兩條線重疊
SDL_RenderLine(lg.Renderer, g, h, e, h); // 繪製水平線 (-x 坐標)
pb = b; // 更新前一個 y 坐標值
pd = d; // 更新前一個 -y 坐標值
if (p < 0)
p += (x++ << 2) + 6; // 決策參數小於 0,選擇上方像素
else
p += ((x++ - y--) << 2) + 10; // 決策參數大於等於 0,選擇右上方像素
}
return true; // 繪製成功
}
// _Rect 函數,繪製矩形框
static bool _Rect(int x, int y, int w, int h, RGBA col)
{
// 邊界檢查:確保矩形在屏幕範圍內
if (x < 0 || x >= lg.w || x + w <= 0 || x + w > lg.w || y < 0 || y >= lg.h || y + h <= 0 || y + h > lg.h)
return false; // 矩形超出屏幕範圍,繪製失敗
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
SDL_FRect rect;
rect.h = (float)h;
rect.w = (float)w;
rect.x = (float)x;
rect.y = (float)y;
SDL_RenderRect(lg.Renderer, &rect); // 繪製矩形框
return true; // 繪製成功
}
// _FillRect 函數,繪製實心矩形
static bool _FillRect(int x, int y, int w, int h, RGBA col)
{
// 邊界檢查:確保矩形在屏幕範圍內
if (x < 0 || x >= lg.w || x + w <= 0 || x + w > lg.w || y < 0 || y >= lg.h || y + h <= 0 || y + h > lg.h)
return false; // 矩形超出屏幕範圍,繪製失敗
// 檢查是否需要設置新的繪圖顏色,避免重複設置
if (col.color != lg.SelColor.color) {
lg.SelColor.color = col.color; // 更新當前選定顏色
SDL_SetRenderDrawColor(lg.Renderer, col.c.r, col.c.g, col.c.b, col.c.a); // 設置渲染器繪圖顏色
}
SDL_FRect rect;
rect.h = (float)h;
rect.w = (float)w;
rect.x = (float)x;
rect.y = (float)y;
SDL_RenderFillRect(lg.Renderer, &rect); // 繪製實心矩形
return true; // 繪製成功
}
// lazyGr 函數指針表,提供圖形庫 API
LazyFnTable lazyGr = {
.KeyDown = _Key_Down,
.Init = _Init,
.ModeNone = _Mode_None,
.ModeBlend = _Mode_Blend,
.ModeAdd = _Mode_Add,
.ModeMod = _Mode_Mod,
.Update = _Update,
.ReadKeys = _ReadKeys,
.GetMouseState = _GetMouseState,
.GetTicks = _GetTicks,
.Delay = _Delay,
.Done = _Done,
.Fini = _Fini,
.Clear = _Clear,
.Point = _Point,
.Line = _Line,
.Circle = _Circle,
.Disk = _Disk,
.Rect = _Rect,
.FillRect = _FillRect,
};
/////////////////////////////////////////////////////////////////////////////////////
/*
SB Image 實現
*/
// _SB_image_destroy 函數,銷毀 SB_Image 對象,釋放紋理和內存
static void _SB_image_destroy(SB_Image* simg)
{
if (simg != NULL) {
SDL_DestroyTexture(simg->Tex); // 銷毀紋理
free(simg); // 釋放 SB_Image 結構體內存
simg = NULL; // 將指針置空,防止野指針
}
}
// _boundary 函數,獲取 SB_Image 的寬度和高度
static void _boundary(SB_Image* simg, int* w, int* h)
{
*w = simg->w; // 返回圖像寬度
*h = simg->h; // 返回圖像高度
}
// _draw 函數,繪製 SB_Image 到屏幕上,可以指定位置和旋轉角度
static void _draw(SB_Image* simg, int x, int y, float angle)
{
SDL_FRect src = { 0, 0, (float)simg->w, (float)simg->h }; // 源矩形,整個圖像
SDL_FRect dest = { x - (float)simg->w / 2, y - (float)simg->h / 2, (float)simg->w, (float)simg->h }; // 目標矩形,居中繪製
SDL_RenderTextureRotated(lg.Renderer, simg->Tex, &src, &dest, angle, NULL,
SDL_FLIP_NONE); // 旋轉並繪製紋理
}
// NewSBImage 函數,創建新的 SB_Image 對象並從 BMP 文件加載圖像
SB_Image* NewSBImage(char* filename)
{
SB_Image* simg = (SB_Image*)malloc(sizeof(SB_Image)); // 分配 SB_Image 結構體內存
SDL_Surface* surface = SDL_LoadBMP(filename); // 從 BMP 文件加載表面
if (!surface) {
// 加載失敗,釋放已分配的內存並返回 NULL
if (simg != NULL) {
free(simg);
simg = NULL;
}
return NULL;
}
SDL_SetSurfaceColorKey(surface, true, 0); // 設置顏色鍵 (透明色),顏色鍵為黑色 (0)
simg->w = surface->w; // 記錄圖像寬度
simg->h = surface->h; // 記錄圖像高度
simg->Tex = SDL_CreateTextureFromSurface(lg.Renderer, surface); // 從表面創建紋理
SDL_DestroySurface(surface); // 銷毀臨時表面,紋理已創建
SDL_SetTextureBlendMode(simg->Tex, SDL_BLENDMODE_BLEND); // 設置紋理混合模式為 Blend (Alpha 混合)
// 初始化函數指針
simg->Boundary = &_boundary;
simg->Draw = &_draw;
simg->Destroy = &_SB_image_destroy;
return simg; // 返回新創建的 SB_Image 對象
}
#endif // header guard