posts - 126,  comments - 73,  trackbacks - 0

  在这个例子里我们将学习如何使用SDL程序中打开和显示图片。示例程序将绘制一个漂亮的背景,上面显示一个正方形图案,并可以使用键盘的方向键移动它。如果比较熟悉“推箱子”这个游戏,可以看出这个程序实际就是推箱子游戏的基础。

  在程序首部包含以下头文件(stdlib.h供调用atexit()时使用):

#include <stdio.h>
#include <stdlib.h>
#include “SDL.h”

  声明3个SDL_Surface类型的全局变量,同时声明2个整型变量用于记录正方形的坐标:

SDL_Surface *back;
SDL_Surface *image;
SDL_Surface *screen;

int xpos = 0, ypos = 0;

  我们构造一个函数InitImage函数来打开bitmap文件(.bmp)中的图像信息,传递给SDL_Surface显示。InitImage将会被main()函数调用。在InitImage函数中我们使用到了SDL_LoadBMP函数,它把bmp文件的文件名作为参数传入,返回指向存储图像文件信息内存区域的指针。InitImage中我们将背景图片bg.bmp打开并使用back指针进行记录,将正方形图片image.bmp打开并使用image指针记录。

int InitImages()
{
 back = SDL_LoadBMP("bg.bmp");
 image = SDL_LoadBMP("image.bmp");
 return 0;
}

  下面我们介绍将图像绘制到屏幕的两个函数,两个函数都被命名为DrawIMG。第一个DrawIMG函数使用SDL_BlitSurface函数来在屏幕上显示图像。在SDL中SDL_BliSurface的函数原型如下:

int SDL_BlitSurface(SDL_Surface *src, SDL_Rect *srcrect,
SDL_Surface *dst, SDL_Rect *dstrect);

  src是需要进行绘制的surface而dst是进行显示的surface。SDL_Rect是一个包含4个16位整型变量的结构:x, y, w(width)和h(height)。srcrect用来描述源surface中需要绘制部分,而dstrect用来描述在目的surface何处进行绘制。如果设置srcrect为NULL,那么源文件中包含的整个图像都会被显示。dstrect中的x和y变量指定了在何处blit SDL_Surface src。对于dstrect来说,w(width)和h(height)这两个变量是被忽略不计的。第一个DrawIMG函数非常简单:

void DrawIMG(SDL_Surface *img, int x, int y)
{
 SDL_Rect dest;
 dest.x = x;
 dest.y = y;
 SDL_BlitSurface(img, NULL, screen, &dest);
}

  下面我们考虑一个复杂点的情况,如图:


  如果我们需要将阴影区域传递给srcrect进行绘制,我们就需要使用到srcrect结构里的几个变量了:x,y指定了所要绘制的源区域的起点坐标,而w和h分别指定了源区域的宽度和高度。如果图中的阴影坐标起点坐标为(20,25),宽61个象素高70个象素,那么我们可以得到:srcrect中x = 20, y = 25, w = 61, h = 70。

  第二个DrawIMG定义如下:

void DrawIMG(SDL_Surface *img, int x, int y, int w, int h, int x2, int y2)
{
 SDL_Rect dest;
 dest.x = x;
 dest.y = y;
 SDL_Rect dest2;
 dest2.x = x2;
 dest2.y = y2;
 dest2.w = w;
 dest2.h = h;
 SDL_BlitSurface(img, &dest2, screen, &dest);
}

  绘制背景的函数DrawBG()比较简单:

void DrawBG()
{
 Slock(screen);
 DrawIMG(back, 0, 0);
 Sulock(screen);
}

  绘制正方形图像的函数将会比较复杂,首先我们用背景图案填充当前正方形图像所在区域,如果不这样做的话,正方形图像的移动就会造成背景上留下黑色的方形移动轨迹,如图所示:


  这里,我们只填充正方形图像移动后的轨迹区域,这样比填充整个区域速度快。由于正方形区域大小是128x128,由于每次正方形只能朝某个方向移动1个象素(pixel),为了彻底消除移动轨迹,我们背景的新填充区域扩大到132x132,这样就可以完全覆盖由于移动造成的轨迹残留。最后使用SDL_Flip对新的图像绘制区域进行更新。函数如下:

void DrawScene()
{
 Slock(screen);
 DrawIMG(back, xpos-2, ypos-2, 132, 132, xpos-2, ypos-2);
 DrawIMG(image, xpos, ypos);

 SDL_Flip(screen);
 Sulock(screen);
}

  由于要移动正方形图像,我们需要对键盘的方向键的按下事件进行响应,因此在main函数开始时定义:

Uint8* keys;

  keys用来获得每一时间的键盘状态。获得键盘状态的函数为SDL_GetKeyState(),它返回一个指向Uin8类型的数组头部的指针。数组的每个元素都对应记录了某个按键是否被按下的标志。这里的实现,我们不在事件轮询SDL_PollEvent中检查按键,因为事件轮询中是只有触发事件也即SDL_PollEvent(&event) == 1才能进入循环的,因而如果我们一直按下某键不放开是不会触发新的事件发生的,要使得正方形移动我们必须一下又一下的敲击某个方向键,显示这不是我们所要的。我们希望是按下某键不放开的话,正方形一直保持向此方向移动。因此我们将检查按键的程序段放到事件轮询之后处理。这里没有使用else if…结构,因此可以多个方向键同时按下进行移动,程序段如下:

int done=0;

while(done == 0)
{
 SDL_Event event;

 while ( SDL_PollEvent(&event) )
 {
  if ( event.type == SDL_QUIT ) { done = 1; }

  if ( event.type == SDL_KEYDOWN )
  {
   if ( event.key.keysym.sym == SDLK_ESCAPE ) { done = 1; }
  }
 }
 keys = SDL_GetKeyState(NULL);
 if ( keys[SDLK_UP] ) { ypos -= 1; }
 if ( keys[SDLK_DOWN] ) { ypos += 1; }
 if ( keys[SDLK_LEFT] ) { xpos -= 1; }
 if ( keys[SDLK_RIGHT] ) { xpos += 1; }

 DrawScene();
}

  程序运行结果如下(我们已经将正方形图像区域移动到了窗口中央):


  如果想学习本程序的完整代码,可以从以下地址下载:http://cone3d.gamedev.net/cgi-bin/index.pl?page=tutorials/gfxsdl/download.pl?file=lesson2.zip&blah=1

  以上我们简单介绍了VC6下SDL的安装和简单应用,并举了两个小例子,当然SDL的功能远远不止这些,包括音频、定时器和线程编程等等,这里就不一一鳌数了。如果有兴趣,SDL官方网站提供了非常详尽的函数说明、文档和使用指南,你可以在网站上获取非常多的信息。由于SDL是跨平台和开发工具的,方便游戏在各种平台下的移植,同时还支持常用的各种开发语言。因此,有足够的理由相信,在游戏产业蓬勃发展的今天,SDL一定会得到更多开发者的青睐,用它开发出各种有趣的游戏来。
posted on 2010-08-11 13:59 我风 阅读(1182) 评论(0)  编辑 收藏 引用 所属分类: SDL

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理


<2010年8月>
25262728293031
1234567
891011121314
15161718192021
22232425262728
2930311234

常用链接

留言簿(12)

随笔分类

随笔档案

文章档案

相册

收藏夹

C++

MyFavorite

搜索

  •  

积分与排名

  • 积分 - 323969
  • 排名 - 75

最新评论

阅读排行榜

评论排行榜