移动的精灵,示例 SDL2 的图片分割、键盘消息
博客专区 > 四彩 的博客 > 博客详情
移动的精灵,示例 SDL2 的图片分割、键盘消息
四彩 发表于2年前
移动的精灵,示例 SDL2 的图片分割、键盘消息
  • 发表于 2年前
  • 阅读 67
  • 收藏 2
  • 点赞 1
  • 评论 0

这是一个移动的精灵,用到了SDL2 的图片分割、键盘事件。

// spirit.c
// SDL2 移动的精灵

//#define _DEBUG_

#include <SDL2/SDL.h>
#include <SDL2/SDL_image.h>

// 图片文件
// 文件名
char *ImageFileName[] =
{
    "res/Town.jpg",     // 背景图文件
    "res/Fighter.png",  // Player文件
    "res/chick.png"     // NPC文件
};
// 图形属性:x = 横向分几幅,y = 纵向分几幅,w = 宽度,h = 高度
SDL_Rect ImageInfo[] =
{
    {1, 2, 640, 640},
    {4, 4, 128, 192},
    {4, 4, 128, 192}
};
// 图片的背景色
int BackColor = 0xFFFFFF;

// 字符串常量
char szWindowTitle[] = "SDL2 移动的精灵";

SDL_Texture *GetImageTexture(SDL_Renderer *pRenderer, char *FileName, _Bool bTransparent, int color);
void UpdateWindow(SDL_Window *pWindow, SDL_Renderer *pRenderer, SDL_Texture **pImageTexture,
                  int bx, int by, int px, int py, int nx, int ny, SDL_Rect *pd, SDL_Rect *nd);

#undef main
int main(int argc, char **argv)
{
    int nWindowWidth  = 640;        // 屏幕尺寸
    int nWindowHeight = 320;
    SDL_Window   *pWindow;          // 主窗口
    SDL_Renderer *pRenderer;        // 主窗口渲染器
    SDL_Texture  *pImageTexture[3]; // 棋盘背景、黑白棋子图纹理
    SDL_Event    event;     // 事件
    _Bool        bRun = 1;  // 持续等待事件控制循环标识

    // 背景、人物、NPC 的图幅坐标
    int bx = 0, by = 0, px = 0, py = 0, nx = 0, ny = 0;
    // 人物、NPC 的目标位置及大小
    SDL_Rect pr = {300, 300, 32, 48}, nr = {380, 230, 16, 24};
    // 移动速度
    int speed = 10;

    // 初始化:SDL2、SDL_Image(jpg)
    if(SDL_Init(SDL_INIT_VIDEO) == -1 || IMG_Init(IMG_INIT_JPG | IMG_INIT_PNG) == -1)
    {
#ifdef _DEBUG_
        fprintf(stderr, "1 %s", SDL_GetError());
#endif
        return 1;
    }
    // 创建主窗口及其渲染器
    if(SDL_CreateWindowAndRenderer(640, 480, SDL_WINDOW_SHOWN, &pWindow, &pRenderer) == -1)
    {
#ifdef _DEBUG_
        fprintf(stderr, "2 %s", SDL_GetError());
#endif
        goto label_error;
    }
    SDL_SetWindowTitle(pWindow, szWindowTitle);
    // 加载图片文件
    if(NULL == (pImageTexture[0] = GetImageTexture(pRenderer, ImageFileName[0], 0, 0))
            || NULL == (pImageTexture[1] = GetImageTexture(pRenderer, ImageFileName[1], 1, BackColor))
            || NULL == (pImageTexture[2] = GetImageTexture(pRenderer, ImageFileName[2], 1, BackColor)))
    {
#ifdef _DEBUG_
        fprintf(stderr, "3 %s", IMG_GetError());
#endif
        goto label_error;
    }

    // 等待事件
    while(bRun && SDL_WaitEvent(&event))
    {
        switch(event.type)
        {
        case SDL_KEYDOWN :      // 键盘按键消息
            switch(event.key.keysym.sym)
            {
            case SDLK_RIGHT :
                py = 2;
                px++;
                pr.x = SDL_min(pr.x+speed, nWindowWidth-pr.w);
                break;

            case SDLK_LEFT :
                py = 1;
                px++;
                pr.x = SDL_max(pr.x-speed, 0);
                break;

            case SDLK_DOWN :
                py = 0;
                px++;
                pr.y = SDL_min(pr.y+speed, nWindowHeight-pr.h);
                break;

            case SDLK_UP :
                py = 3;
                px++;
                pr.y = SDL_max(pr.y-speed, 0);
                break;

            default :
                break;
            }
            px %= 4;
            UpdateWindow(pWindow, pRenderer, pImageTexture, bx, by, px, py, nx, ny, &pr, &nr);
            break;

        case SDL_WINDOWEVENT :      //  有窗口消息,重新计算窗口尺寸
            SDL_GetWindowSize(pWindow, &nWindowWidth, &nWindowHeight);
            UpdateWindow(pWindow, pRenderer, pImageTexture, bx, by, px, py, nx, ny, &pr, &nr);
            break;

        case SDL_QUIT :
            bRun = 0;
            break;

        default :
            break;
        }
    }

label_error:
// 清理
    if(pImageTexture[0] != NULL) SDL_DestroyTexture(pImageTexture[0]);
    if(pImageTexture[1] != NULL) SDL_DestroyTexture(pImageTexture[1]);
    if(pImageTexture[2] != NULL) SDL_DestroyTexture(pImageTexture[2]);
    IMG_Quit();
    SDL_Quit();
    return 0;
}

// 重绘窗口
void UpdateWindow(SDL_Window *pWindow, SDL_Renderer *pRenderer, SDL_Texture **pImageTexture,
                  int bx, int by, int px, int py, int nx, int ny, SDL_Rect *pr, SDL_Rect *nr)
{
    SDL_Rect srt;

    SDL_RenderClear(pRenderer);

    srt.x = bx*ImageInfo[0].w/ImageInfo[0].x;
    srt.y = by*ImageInfo[0].h/ImageInfo[0].y;
    srt.w = ImageInfo[0].w/ImageInfo[0].x;
    srt.h = ImageInfo[0].h/ImageInfo[0].y;
    SDL_RenderCopy(pRenderer, pImageTexture[0], &srt, NULL);

    srt.x = px*ImageInfo[1].w/ImageInfo[1].x;
    srt.y = py*ImageInfo[1].h/ImageInfo[1].y;
    srt.w = ImageInfo[1].w/ImageInfo[1].x;
    srt.h = ImageInfo[1].h/ImageInfo[1].y;
    SDL_RenderCopy(pRenderer, pImageTexture[1], &srt, pr);

    srt.x = nx*ImageInfo[2].w/ImageInfo[2].x;
    srt.y = ny*ImageInfo[2].h/ImageInfo[2].y;
    srt.w = ImageInfo[2].w/ImageInfo[2].x;
    srt.h = ImageInfo[2].h/ImageInfo[2].y;
    SDL_RenderCopy(pRenderer, pImageTexture[2], &srt, nr);

    SDL_RenderPresent(pRenderer);
}


// 取得图片文件纹理
// 参数:pRenderer = 渲染器;FileName = 图片文件名;bTransparent = 是否透明处理;color = 背景色
// 返回值:纹理指针
SDL_Texture *GetImageTexture(SDL_Renderer *pRenderer, char *FileName, _Bool bTransparent, int color)
{
    SDL_Texture *pTexture;
    SDL_Surface *pSurface;

    if((pSurface = IMG_Load(FileName)) == NULL)
        return NULL;
    if(bTransparent)
        SDL_SetColorKey(pSurface, 1, SDL_MapRGB(pSurface->format, color>>16, (color>>8)&0xFF, color&0xFF));
    pTexture = SDL_CreateTextureFromSurface(pRenderer, pSurface);
    SDL_FreeSurface(pSurface);

    return pTexture;
}

图片是从 C4droid  百度吧找来的。

人物移动位置没有进行合理性检查,这需要对地图进行分格定性。

共有 人打赏支持
粉丝 26
博文 49
码字总数 77226
×
四彩
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: