自己动手写俄罗斯方块(一)

一.思路
1.  设定小方块的大小和游戏区的座标
每个俄罗斯方块都是有4个小方块构成的,所以我们要先设定好每个小方块的固定大小,还要确定游戏区的起始座标(左上角座标);
2.  建立游戏区
根据自己设定的座标建立一个2010列的游戏区。每行每列交叉处构成的方格即为一
个方块的大小。设定好方格大小后,我们就可以获得游戏区的终止座标(右下角座标)。如起始座标为(50,50),方格大小为20,那终止座标就是(450250);(50+20*20,50+10*20);
3.  初始化俄罗斯方块的7种图形
我们都知道俄罗斯方块有7种图形,分别为:SZLJIOT这几个字母的形状。每种图形经过4个方向的旋转又可以得到4种图形(有的可能是1种或2种,因为有的旋转过之后跟以前的图形还是一样的)。我们将这些图形做一下初始化,等需要时可以直接调用。我用一个POINT类型的三维数组来实现。如:可以定义一个这样的三维数组:
const POINT Terics[7][4][4] =

{

  {

      0,0,1,0,0,1,-1,1,

      0,0,0,1,1,1,1,2,

      0,0,1,0,0,1,-1,1,

      0,0,0,1,1,1,1,2

  },

  {

      0,0,1,0,1,1,2,1,

      0,0,0,1,-1,1,-1,2,

      0,0,1,0,1,1,2,1,

      0,0,0,1,-1,1,-1,2

  },

  {

      0,0,0,1,0,2,1,2,

      0,0,0,1,-1,1,-2,1,

      0,0,1,0,1,1,1,2,

      0,0,0,1,1,0,2,0

  },

  {

      0,0,0,1,0,2,-1,2,

      0,0,1,0,2,0,2,1,

      0,0,1,0,0,1,0,2,

      0,0,0,1,1,1,2,1

  },

  {

      0,0,0,1,0,2,0,3,

      0,0,1,0,2,0,3,0,

      0,0,0,1,0,2,0,3,

      0,0,1,0,2,0,3,0

  },

  {

      0,0,1,0,0,1,1,1,

      0,0,1,0,0,1,1,1,

      0,0,1,0,0,1,1,1,

      0,0,1,0,0,1,1,1

  },

  {

      0,0,1,0,2,0,1,1,

      0,0,0,1,0,2,1,1,

      0,0,0,1,-1,1,1,1,

      0,0,0,1,0,2,-1,1

  }

};

第一维的7表示7种图形,第二维的4表示4个方向,第三维的4表示4point型的数。通过这个三维数组和当前座标我们就可以得出每种图形4个方块的左上角座标,也就可以在当前位置画出各种各样的图形了。可能有的人不理解这个三维数组中的数字代表什么意思,其实我使用的是相对偏移量,在每种图形的每个方向的4个小方块中选择一个作为基准方块,它的左上角的座标是(0,0),其他三个小方块的左上角座标就可以用相对于这个基准方块的左上角座标的偏移量来表示。具体实现方法在下面会讨论。注意:这个三维数组的初始化不是唯一的,根据自己的喜欢,先初始化哪种图形都可以。
4.  游戏按键
我们只需用到上下左右四个键就可以了,上键用于改变方向,下键用于加速方块的下移,左右键用于方块的左右移动。注意我们每次移动的是一个小方块的距离。
当用户点击游戏开始时,方块会自上而下落下,如果用户不按任何键,则方块正常下落。这个可以由SetTimer来完成,我们在WM_CREATE消息中调用SetTimer(hwnd,1,1000,NULL);如何用户觉得下落时间太慢,也可以自己设定间隔时间。然后就可以在WM_TIMER中通过改变图形的纵座标来实现方块下移。
若用户按:
上键:有规律的变换方向,即根据用户初始化时的方向依次变换。
下键:加速下移,通过增加纵座标来实现。
左键:向左移动一个位置,通过减少横座标来实现。
右键:向右移动一个位置,通过增加横座标来实现。
这些游戏按键我们可以在WM_KEYDOWN消息中实现,当窗口函数获得WM_KEYDOWN消息时,它的第三个参数wParam保存的将是按键的详细内容。我们可以通过一个switch……case来查找我们想要的消息。
这四个按键对应的消息分别是:VK_UP(上键),VK_DOWN(下键),VK_LEFT(左键),VK_RIGHT(右键)。
5.  判断范围
每次用户按下键后都要判断是否超出了范围,若超出了范围,则对用户的按键不响应。
判断方法:两个条件:1)根据用户当前的座标计算出该形状的四个方块的左上角座标,然后判断这些座标是否超出了游戏区范围。2)判断该方块要移动到的区域是否已经有方块存在了。如果上面两个条件中有一个成立,那么用户的按键将不予响应。
6.  保存游戏区状态
我们如何知道当前位置是否有方块存在呢,这就需要我们保存游戏区当前的状态。一方面是为了第5步的判断用,另一方面是为了窗口刷新时加载已存在的方块。
我们可以定义一个结构体,里面有两个变量,一个表示状态,另一个表示形状。另外还需要定义一个结构体变量,这个变量我们定义成一个整型的二维数组,这个二维数组的横纵座标当然应该是游戏区的行数和列数啦。如:
struct CurrentTericsStates
{
    Int states;
    Int shape;
}CurrentTericsStatesInfo[10][20];
何时用来保存当前的状态呢?
当然是当方块落到游戏区的底部或是遇到其他方块不能再往下移动时就需要保存当前方块的信息了。
7.  图形显示问题
当图形自动下移或者我们移动图形时,会发现原来的图形还存在,该如何解决这个问题呢?我想了两种方法:
1)              我们需要用两个POINT型变量来保存当前图形和上一个图形的形状、方向、坐
标。然后在每次移动时用背景色的画笔重画上一个图形,这样上一个图形就被覆盖了,我们就看不到了。最后再把当前的图形画上就可以了。
2)              利用擦除背景法,即我们每次在移动一个图形后,将整个游戏区用背景色的画
刷再刷一遍,这样所有图形都没有了。我们只需将游戏区,原有图形和当前图形都画上就可以了。
8.  如何消去一行
当游戏区中的某一行都填满了方块后,我们该如何消去这一行呢?这时我们就需要用到前面定义的结构体变量了。利用查找法,查看数组中是否有一行都为1,如果都为1的话,就说明该行已满,那么就消去该行。如何消去呢?其实,我们就是用上一行来填充这一行,也就是用上一行的状态来代替该行的状态,这样该行就被消去了。另外我们还需要两个全局变量countscore,每当消去一行,count就加1,并计算出相应的分数,例如每消去一行就加10分等等。最后我使用了一个弹出对话框来输出所得的分数。
9.         如何判断游戏结束呢?
这个问题看起来有点难,其实我们只需要判断游戏区最上边一行的状态就可以了,如果最上边一行中有状态为1的方块,就说明有方块已经到达了游戏区顶部,那么游戏也就结束了。
到这里,一个俄罗斯方块的大体思路已经出来了,具体的代码和一些细节会在以后的文章中提到。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章