用C++编写的小游戏源代码
五子棋的代码:
#include《iostream》
#include《stdio.h》
#include《stdlib.h》
#include 《time.h》
using namespace std;
const int N=15; //15*15的棋盘
const char ChessBoardflag = ’ ’; //棋盘标志
const char flag1=’o’; //玩家1或电脑的棋子标志
const char flag2=’X’; //玩家2的棋子标志
typedef struct Coordinate //坐标类
{
int x; //代表行
int y; //代表列
}Coordinate;
class GoBang //五子棋类
{
public:
GoBang() //初始化
{
InitChessBoard();
}
void Play() //下棋
{
Coordinate Pos1; // 玩家1或电脑
Coordinate Pos2; //玩家2
int n = 0;
while (1)
{
int mode = ChoiceMode();
while (1)
{
if (mode == 1) //电脑vs玩家
{
ComputerChess(Pos1,flag1); // 电脑下棋
if (GetVictory(Pos1, 0, flag1) == 1) //0表示电脑,真表示获胜
break;
PlayChess(Pos2, 2, flag2); //玩家2下棋
if (GetVictory(Pos2, 2, flag2)) //2表示玩家2
break;
}
else //玩家1vs玩家2
{
PlayChess(Pos1, 1, flag1); // 玩家1下棋
if (GetVictory(Pos1, 1, flag1)) //1表示玩家1
break;
PlayChess(Pos2, 2, flag2); //玩家2下棋
if (GetVictory(Pos2, 2, flag2)) //2表示玩家2
break;
}
}
cout 《《 “***再来一局***“ 《《 endl;
cout 《《 “y or n :“;
char c = ’y’;
cin 》》 c;
if (c == ’n’)
break;
}
}
protected:
int ChoiceMode() //选择模式
{
int i = 0;
system(“cls“); //系统调用,清屏
InitChessBoard(); //重新初始化棋盘
cout 《《 “***0、退出 1、电脑vs玩家 2、玩家vs玩家***“ 《《 endl;
while (1)
{
cout 《《 “请选择:“;
cin 》》 i;
if (i == 0) //选择0退出
exit(1);
if (i == 1 || i == 2)
return i;
cout 《《 “输入不合法“ 《《 endl;
}
}
void InitChessBoard() //初始化棋盘
{
for (int i = 0; i 《 N + 1; ++i)
{
for (int j = 0; j 《 N + 1; ++j)
{
_ChessBoard[i][j] = ChessBoardflag;
}
}
}
void PrintChessBoard() //打印棋盘,这个函数可以自己调整
{
system(“cls“); //系统调用,清空屏幕
for (int i = 0; i 《 N+1; ++i)
{
for (int j = 0; j 《 N+1; ++j)
{
if (i == 0) //打印列数字
{
if (j!=0)
printf(“%d “, j);
else
printf(“ “);
}
else if (j == 0) //打印行数字
printf(“%2d “, i);
else
{
if (i 《 N+1)
{
printf(“%c |“,_ChessBoard[i][j]);
}
}
}
cout 《《 endl;
cout 《《 “ “;
for (int m = 0; m 《 N; m++)
{
printf(“–|“);
}
cout 《《 endl;
}
}
void PlayChess(Coordinate& pos, int player, int flag) //玩家下棋
{
PrintChessBoard(); //打印棋盘
while (1)
{
printf(“玩家%d输入坐标:“, player);
cin 》》 pos.x 》》 pos.y;
if (JudgeValue(pos) == 1) //坐标合法
break;
cout 《《 “坐标不合法,重新输入“ 《《 endl;
}
_ChessBoard[pos.x][pos.y] = flag;
}
void ComputerChess(Coordinate& pos, char flag) //电脑下棋
{
PrintChessBoard(); //打印棋盘
int x = 0;
int y = 0;
while (1)
{
x = (rand() % N) + 1; //产生1~N的随机数
srand((unsigned int) time(NULL));
y = (rand() % N) + 1; //产生1~N的随机数
srand((unsigned int) time(NULL));
if (_ChessBoard[x][y] == ChessBoardflag) //如果这个位置是空的,也就是没有棋子
break;
}
pos.x = x;
pos.y = y;
_ChessBoard[pos.x][pos.y] = flag;
}
int JudgeValue(const Coordinate& pos) //判断输入坐标是不是合法
{
if (pos.x 》 0 && pos.x 《= N&&pos.y 》 0 && pos.y 《= N)
{
if (_ChessBoard[pos.x][pos.y] == ChessBoardflag)
{
return 1; //合法
}
}
return 0; //非法
}
int JudgeVictory(Coordinate pos, char flag) //判断有没有人胜负(底层判断)
{
int begin = 0;
int end = 0;
int begin1 = 0;
int end1 = 0;
//判断行是否满足条件
(pos.y – 4) 》 0 ? begin = (pos.y – 4) : begin = 1;
(pos.y + 4) 》N ? end = N : end = (pos.y + 4);
for (int i = pos.x, j = begin; j + 4 《= end; j++)
{
if (_ChessBoard[i][j] == flag&&_ChessBoard[i][j + 1] == flag&&
_ChessBoard[i][j + 2] == flag&&_ChessBoard[i][j + 3] == flag&&
_ChessBoard[i][j + 4] == flag)
return 1;
}
//判断列是否满足条件
(pos.x – 4) 》 0 ? begin = (pos.x – 4) : begin = 1;
(pos.x + 4) 》 N ? end = N : end = (pos.x + 4);
for (int j = pos.y, i = begin; i + 4 《= end; i++)
{
if (_ChessBoard[i][j] == flag&&_ChessBoard[i + 1][j] == flag&&
_ChessBoard[i + 2][j] == flag&&_ChessBoard[i + 3][j] == flag&&
_ChessBoard[i + 4][j] == flag)
return 1;
}
int len = 0;
//判断主对角线是否满足条件
pos.x 》 pos.y ? len = pos.y – 1 : len = pos.x – 1;
if (len 》 4)
len = 4;
begin = pos.x – len; //横坐标的起始位置
begin1 = pos.y – len; //纵坐标的起始位置
pos.x 》 pos.y ? len = (N – pos.x) : len = (N – pos.y);
if (len》4)
len = 4;
end = pos.x + len; //横坐标的结束位置
end1 = pos.y + len; //纵坐标的结束位置
for (int i = begin, j = begin1; (i + 4 《= end) && (j + 4 《= end1); ++i, ++j)
{
if (_ChessBoard[i][j] == flag&&_ChessBoard[i + 1][j + 1] == flag&&
_ChessBoard[i + 2][j + 2] == flag&&_ChessBoard[i + 3][j + 3] == flag&&
_ChessBoard[i + 4][j + 4] == flag)
return 1;
}
//判断副对角线是否满足条件
(pos.x – 1) 》(N – pos.y) ? len = (N – pos.y) : len = pos.x – 1;
if (len 》 4)
len = 4;
begin = pos.x – len; //横坐标的起始位置
begin1 = pos.y + len; //纵坐标的起始位置
(N – pos.x) 》 (pos.y – 1) ? len = (pos.y – 1) : len = (N – pos.x);
if (len》4)
len = 4;
end = pos.x + len; //横坐标的结束位置
end1 = pos.y – len; //纵坐标的结束位置
for (int i = begin, j = begin1; (i + 4 《= end) && (j – 4 》= end1); ++i, –j)
{
if (_ChessBoard[i][j] == flag&&_ChessBoard[i + 1][j – 1] == flag&&
_ChessBoard[i + 2][j – 2] == flag&&_ChessBoard[i + 3][j – 3] == flag&&
_ChessBoard[i + 4][j – 4] == flag)
return 1;
}
for (int i = 1; i 《 N + 1; ++i) //棋盘有没有下满
{
for (int j =1; j 《 N + 1; ++j)
{
if (_ChessBoard[i][j] == ChessBoardflag)
return 0; //0表示棋盘没满
}
}
return -1; //和棋
}
bool GetVictory(Coordinate& pos, int player, int flag) //对JudgeVictory的一层封装,得到具体那个玩家获胜
{
int n = JudgeVictory(pos, flag); //判断有没有人获胜
if (n != 0) //有人获胜,0表示没有人获胜
{
PrintChessBoard();
if (n == 1) //有玩家赢棋
{
if (player == 0) //0表示电脑获胜,1表示玩家1,2表示玩家2
printf(“***电脑获胜***\n“);
else
printf(“***恭喜玩家%d获胜***\n“, player);
}
else
printf(“***双方和棋***\n“);
return true; //已经有人获胜
}
return false; //没有人获胜
}
private:
char _ChessBoard[N+1][N+1];
};
扩展资料:
设计思路
1、进行问题分析与设计,计划实现的功能为,开局选择人机或双人对战,确定之后比赛开始。
2、比赛结束后初始化棋盘,询问是否继续比赛或退出,后续可加入复盘、悔棋等功能。
3、整个过程中,涉及到了棋子和棋盘两种对象,同时要加上人机对弈时的AI对象,即涉及到三个对象。
c++编程小游戏代码
以下是贪吃蛇源代码:
#include《iostream.h》
#include《windows.h》
#include《time.h》
#include《stdlib.h》
#include《conio.h》
#define N 21
void gotoxy(int x,int y)//位置函数{
COORD pos;
pos.X=2*x;
pos.Y=y;
SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE),pos);
}
void color(int a)//颜色函数{
SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE),a);
}
void init(int apple)//初始化函数(初始化围墙、显示信息、苹果)
{
int i,j;//初始化围墙
int wall[N+2][N+2]={{0}};
for(i=1;i《=N;i++)
{
for(j=1;j《=N;j++)
wall[i][j]=1;
}
color(11);
for(i=0;i《N+2;i++)
{
for(j=0;j《N+2;j++)
{
if(wall[i][j])
cout《《“■“;
else cout《《“□“ ;
}
cout《《endl;
}
gotoxy(N+3,1);//显示信息
color(20);
cout《《“按 W S A D 移动方向“《《endl;
gotoxy(N+3,2);
color(20);
cout《《“按任意键暂停“《《endl;
gotoxy(N+3,3);
color(20);
cout《《“得分:“《《endl;
apple=rand()%N+1;//苹果
apple=rand()%N+1;
gotoxy(apple,apple);
color(12);
cout《《“●“《《endl;
}
int main()
{
int i,j;
int** snake=NULL;
int apple;
int score=0;
int tail;
int len=3;
char ch=’p’;
srand((unsigned)time(NULL));
init(apple);
snake=(int**)realloc(snake,sizeof(int*)*len);
for(i=0;i《len;i++)
snake[i]=(int*)malloc(sizeof(int)*2);
for(i=0;i《len;i++)
{
snake[i]=N/2;
snake[i]=N/2+i;
gotoxy(snake[i],snake[i]);
color(14);
cout《《““《《endl;
}
while(1)//进入消息循环
{
tail=snake[len-1];
tail=snake[len-1];
gotoxy(tail,tail);
color(11);
cout《《“■“《《endl;
for(i=len-1;i》0;i–)
{
snake[i]=snake[i-1];
snake[i]=snake[i-1];
gotoxy(snake[i],snake[i]);
color(14);
cout《《““《《endl;
}
if(kbhit())
{
gotoxy(0,N+2);
ch=getche();
}
switch(ch)
{
case ’w’:snake–;break;
case ’s’:snake++;break;
case ’a’:snake–;break;
case ’d’:snake++;break;
default: break;
}
gotoxy(snake,snake);
color(14);
cout《《““《《endl;
Sleep(abs(200-0.5*score));
if(snake==apple&&snake==apple)//吃掉苹果后蛇分数加1,蛇长加1
{
score++;
len++;
snake=(int**)realloc(snake,sizeof(int*)*len);
snake[len-1]=(int*)malloc(sizeof(int)*2);
apple=rand()%N+1;
apple=rand()%N+1;
gotoxy(apple,apple);
color(12);
cout《《“●“《《endl;
gotoxy(N+5,3);
color(20);
cout《《score《《endl;
}
if(snake==0||snake==N||snake==0||snake==N)//撞到围墙后失败
{
gotoxy(N/2,N/2);
color(30);
cout《《“失败!!!“《《endl;
for(i=0;i《len;i++)
free(snake[i]);
Sleep(INFINITE);
exit(0);
}
}
return 0;
}
怎么编写游戏代码
首先请先学一门语言,c类,java,python其中三选一,当你学会基础之后,就可以采用各种网上有的库来编写一款游戏了。
还有一种办法就是下载一款图形编程软件,类似processing等,这些都需要用到上述其中一种语言,好处就是不用跟各种五花八门的库打交道,坏处是写出来的游戏只能用这些软件运行。
扩展资料:
代码设计的原则包括唯一确定性、标准化和通用性、可扩充性与稳定性、便于识别与记忆、力求短小与格式统一以及容易修改等。 源代码是代码的分支,某种意义上来说,源代码相当于代码。
现代程序语言中,源代码可以书籍或磁带形式出现,但最为常用格式是文本文件,这种典型格式的目的是为了编译出计算机程序。计算机源代码最终目的是将人类可读文本翻译成为计算机可执行的二进制指令,这种过程叫编译,它由通过编译器完成。
参考资料来源:百度百科-代码
游戏一般用什么编程语言开发
一般的大型游戏开发不是单一用某一种软件语言的问题。一个大型游戏的开发需要非常大的团队用各种各样的语言和工具来完成。
总结一下主要有C/C++,汇编语言,着色器语言,脚本语言,高效的开发语言C#或Java。
首先一般的游戏开发架构(Windows平台)从底到顶一般是Direct X™——游戏引擎——游戏。
大型游戏开发的大部分工作其实都是在编写游戏脚本,脚本是大型游戏得以如此高速开发和发布的主要原因。脚本化的开发让游戏开发摆脱了硬编码的种种弊端,让游戏内容可以轻易的修改和调试。比如比较流行的语言。,然后编写脚本将其组织成一个游戏,不需要什么底层的编程语言。
Windows平台比较流行的方法是使用最新版本的Visual Studio,显卡厂商如NVIDIA也会为VS开发一些插件来简化显卡编程和调试。扩展资料:
汇编语言
为了克服机器语言难读、难编、难记和易出错的缺点,人们就用与代码指令实际含义相近的英文缩写词、字母和数字等符号来取代指令代码(如用ADD表示运算符号“+”的机器代码),于是就产生了汇编语言。所以说,汇编语言是一种用助记符表示的仍然面向机器的计算机语言。汇编语言亦称符号语言。
汇编语言由于是采用了助记符号来编写程序,比用机器语言的二进制代码编程要方便些,在一定程度上简化了编程过程。汇编语言的特点是用符号代替了机器指令代码。而且助记符与指令代码一一对应,基本保留了机器语言的灵活性。使用汇编语言能面向机器并较好地发挥机器的特性,得到质量较高的程序。
汇编语言中由于使用了助记符号,用汇编语言编制的程序送入计算机,计算机不能象用机器语言编写的程序一样直接识别和执行,必须通过预先放入计算机的 “汇编程序“的加工和翻译,才能变成能够被计算机识别和处理的二进制代码程序。
用汇编语言等非机器语言书写好的符号程序称源程序,运行时汇编程序要将源程序翻译成目标程序。目标程序是机器语言程序,它一经被安置在内存的预定位置上,就能被计算机的CPU处理和执行。
汇编语言像机器指令一样,是硬件操作的控制信息,因而仍然是面向机器的语言,使用起来还是比较繁琐费时,通用性也差。汇编语言是低级语言。但是,汇编语言用来编制系统软件和过程控制软件,其目标程序占用内存空间少,运行速度快,有着高级语言不可替代的用途。
高级语言
不论是机器语言还是汇编语言都是面向硬件的具体操作的,语言对机器的过分依赖,要求使用者必须对硬件结构及其工作原理都十分熟悉,这对非计算机专业人员是难以做到的,对于计算机的推广应用是不利的。计算机事业的发展,促使人们去寻求一些与人类自然语言相接近且能为计算机所接受的语意确定、规则明确、自然直观和通用易学的计算机语言。
这种与自然语言相近并为计算机所接受和执行的计算机语言称高级语言。高级语言是面向用户的语言。无论何种机型的计算机,只要配备上相应的高级语言的编译或解释程序,则用该高级语言编写的程序就可以通用。
如今被广泛使用的高级语言有BASIC、PASCAL、C、COBOL、FORTRAN、LOGO以及VC、VB等。这些语言都是属于系统软件。
计算机并不能直接地接受和执行用高级语言编写的源程序,源程序在输入计算机时,通过“翻译程序”翻译成机器语言形式的目标程序,计算机才能识别和执行。这种“翻译”通常有两种方式,即编译方式和解释方式。
编译方式是:事先编好一个称为编译程序的机器语言程序,作为系统软件存放在计算机内,当用户由高级语言编写的源程序输入计算机后,编译程序便把源程序整个地翻译成用机器语言表示的与之等价的目标程序,然后计算机再执行该目标程序,以完成源程序要处理的运算并取得结果。解释方式是:源程序进入计算机时,解释程序边扫描边解释作逐句输入逐句翻译,计算机一句句执行,并不产生目标程序。
PASCAL、 FORTRAN、COBOL等高级语言执行编译方式;BASIC语言则以执行解释方式为主;而PASCAL、C语言是能书写编译程序的高级程序设计语言。每一种高级(程序设计)语言,都有自己人为规定的专用符号、英文单词、语法规则和语句结构(书写格式)。高级语言与自然语言(英语)更接近,而与硬件功能相分离(彻底脱离了具体的指令系统),便于广大用户掌握和使用。高级语言的通用性强,兼容性好,便于移植
求一个用C语言编写的小游戏代码
#include 《graphics.h》
#include 《conio.h》
#include 《time.h》
/////////////////////////////////////////////
// 定义常量、枚举量、结构体、全局变量
/////////////////////////////////////////////
#define WIDTH 10 // 游戏区宽度
#define HEIGHT 22 // 游戏区高度
#define SIZE 20 // 每个游戏区单位的实际像素
// 定义操作类型
enum CMD
{
CMD_ROTATE, // 方块旋转
CMD_LEFT, CMD_RIGHT, CMD_DOWN, // 方块左、右、下移动
CMD_SINK, // 方块沉底
CMD_QUIT // 退出游戏
};
// 定义绘制方块的方法
enum DRAW
{
SHOW, // 显示方块
HIDE, // 隐藏方块
FIX // 固定方块
};
// 定义七种俄罗斯方块
struct BLOCK
{
WORD dir; // 方块的四个旋转状态
COLORREF color; // 方块的颜色
} g_Blocks = { {0x0F00, 0x4444, 0x0F00, 0x4444, RED}, // I
{0x0660, 0x0660, 0x0660, 0x0660, BLUE}, // 口
{0x4460, 0x02E0, 0x0622, 0x0740, MAGENTA}, // L
{0x2260, 0x0E20, 0x0644, 0x0470, YELLOW}, // 反L
{0x0C60, 0x2640, 0x0C60, 0x2640, CYAN}, // Z
{0x0360, 0x4620, 0x0360, 0x4620, GREEN}, // 反Z
{0x4E00, 0x4C40, 0x0E40, 0x4640, BROWN}}; // T
// 定义当前方块、下一个方块的信息
struct BLOCKINFO
{
byte id; // 方块 ID
char x, y; // 方块在游戏区中的坐标
byte dir:2; // 方向
} g_CurBlock, g_NextBlock;
// 定义游戏区
BYTE g_World[WIDTH][HEIGHT] = {0};
/////////////////////////////////////////////
// 函数声明
/////////////////////////////////////////////
void Init(); // 初始化游戏
void Quit(); // 退出游戏
void NewGame(); // 开始新游戏
void GameOver(); // 结束游戏
CMD GetCmd(); // 获取控制命令
void DispatchCmd(CMD _cmd); // 分发控制命令
void NewBlock(); // 生成新的方块
bool CheckBlock(BLOCKINFO _block); // 检测指定方块是否可以放下
void DrawBlock(BLOCKINFO _block, DRAW _draw = SHOW); // 画方块
void OnRotate(); // 旋转方块
void OnLeft(); // 左移方块
void OnRight(); // 右移方块
void OnDown(); // 下移方块
void OnSink(); // 沉底方块
/////////////////////////////////////////////
// 函数定义
/////////////////////////////////////////////
// 主函数
void main()
{
Init();
CMD c;
while(true)
{
c = GetCmd();
DispatchCmd(c);
// 按退出时,显示对话框咨询用户是否退出
if (c == CMD_QUIT)
{
HWND wnd = GetHWnd();
if (MessageBox(wnd, _T(“您要退出游戏吗?“), _T(“提醒“), MB_OKCANCEL | MB_ICONQUESTION) == IDOK)
Quit();
}
}
}
// 初始化游戏
void Init()
{
initgraph(640, 480);
srand((unsigned)time(NULL));
// 显示操作说明
setfont(14, 0, _T(“宋体“));
outtextxy(20, 330, _T(“操作说明“));
outtextxy(20, 350, _T(“上:旋转“));
outtextxy(20, 370, _T(“左:左移“));
outtextxy(20, 390, _T(“右:右移“));
outtextxy(20, 410, _T(“下:下移“));
outtextxy(20, 430, _T(“空格:沉底“));
outtextxy(20, 450, _T(“ESC:退出“));
// 设置坐标原点
setorigin(220, 20);
// 绘制游戏区边界
rectangle(-1, -1, WIDTH * SIZE, HEIGHT * SIZE);
rectangle((WIDTH + 1) * SIZE – 1, -1, (WIDTH + 5) * SIZE, 4 * SIZE);
// 开始新游戏
NewGame();
}
// 退出游戏
void Quit()
{
closegraph();
exit(0);
}
// 开始新游戏
void NewGame()
{
// 清空游戏区
setfillstyle(BLACK);
bar(0, 0, WIDTH * SIZE – 1, HEIGHT * SIZE – 1);
ZeroMemory(g_World, WIDTH * HEIGHT);
// 生成下一个方块
g_NextBlock.id = rand() % 7;
g_NextBlock.dir = rand() % 4;
g_NextBlock.x = WIDTH + 1;
g_NextBlock.y = HEIGHT – 1;
// 获取新方块
NewBlock();
}
// 结束游戏
void GameOver()
{
HWND wnd = GetHWnd();
if (MessageBox(wnd, _T(“游戏结束。\n您想重新来一局吗?“), _T(“游戏结束“), MB_YESNO | MB_ICONQUESTION) == IDYES)
NewGame();
else
Quit();
}
// 获取控制命令
DWORD m_oldtime;
CMD GetCmd()
{
// 获取控制值
while(true)
{
// 如果超时,自动下落一格
DWORD newtime = GetTickCount();
if (newtime – m_oldtime 》= 500)
{
m_oldtime = newtime;
return CMD_DOWN;
}
// 如果有按键,返回按键对应的功能
if (kbhit())
{
switch(getch())
{
case ’w’:
case ’W’: return CMD_ROTATE;
case ’a’:
case ’A’: return CMD_LEFT;
case ’d’:
case ’D’: return CMD_RIGHT;
case ’s’:
case ’S’: return CMD_DOWN;
case 27: return CMD_QUIT;
case ’ ’: return CMD_SINK;
case 0:
case 0xE0:
switch(getch())
{
case 72: return CMD_ROTATE;
case 75: return CMD_LEFT;
case 77: return CMD_RIGHT;
case 80: return CMD_DOWN;
}
}
}
// 延时 (降低 CPU 占用率)
Sleep(20);
}
}
// 分发控制命令
void DispatchCmd(CMD _cmd)
{
switch(_cmd)
{
case CMD_ROTATE: OnRotate(); break;
case CMD_LEFT: OnLeft(); break;
case CMD_RIGHT: OnRight(); break;
case CMD_DOWN: OnDown(); break;
case CMD_SINK: OnSink(); break;
case CMD_QUIT: break;
}
}
// 生成新的方块
void NewBlock()
{
g_CurBlock.id = g_NextBlock.id, g_NextBlock.id = rand() % 7;
g_CurBlock.dir = g_NextBlock.dir, g_NextBlock.dir = rand() % 4;
g_CurBlock.x = (WIDTH – 4) / 2;
g_CurBlock.y = HEIGHT + 2;
// 下移新方块直到有局部显示
WORD c = g_Blocks[g_CurBlock.id].dir[g_CurBlock.dir];
while((c & 0xF) == 0)
{
g_CurBlock.y–;
c 》》= 4;
}
// 绘制新方块
DrawBlock(g_CurBlock);
// 绘制下一个方块
setfillstyle(BLACK);
bar((WIDTH + 1) * SIZE, 0, (WIDTH + 5) * SIZE – 1, 4 * SIZE – 1);
DrawBlock(g_NextBlock);
// 设置计时器,用于判断自动下落
m_oldtime = GetTickCount();
}
// 画方块
void DrawBlock(BLOCKINFO _block, DRAW _draw)
{
WORD b = g_Blocks[_block.id].dir[_block.dir];
int x, y;
int color = BLACK;
switch(_draw)
{
case SHOW: color = g_Blocks[_block.id].color; break;
case HIDE: color = BLACK; break;
case FIX: color = g_Blocks[_block.id].color / 3; break;
}
setfillstyle(color);
for(int i=0; i《16; i++)
{
if (b & 0x8000)
{
x = _block.x + i % 4;
y = _block.y – i / 4;
if (y 《 HEIGHT)
{
if (_draw != HIDE)
bar3d(x * SIZE + 2, (HEIGHT – y – 1) * SIZE + 2, (x + 1) * SIZE – 4, (HEIGHT – y) * SIZE – 4, 3, true);
else
bar(x * SIZE, (HEIGHT – y – 1) * SIZE, (x + 1) * SIZE – 1, (HEIGHT – y) * SIZE – 1);
}
}
b 《《= 1;
}
}
// 检测指定方块是否可以放下
bool CheckBlock(BLOCKINFO _block)
{
WORD b = g_Blocks[_block.id].dir[_block.dir];
int x, y;
for(int i=0; i《16; i++)
{
if (b & 0x8000)
{
x = _block.x + i % 4;
y = _block.y – i / 4;
if ((x 《 0) || (x 》= WIDTH) || (y 《 0))
return false;
if ((y 《 HEIGHT) && (g_World[x][y]))
return false;
}
b 《《= 1;
}
return true;
}
// 旋转方块
void OnRotate()
{
// 获取可以旋转的 x 偏移量
int dx;
BLOCKINFO tmp = g_CurBlock;
tmp.dir++; if (CheckBlock(tmp)) { dx = 0; goto rotate; }
tmp.x = g_CurBlock.x – 1; if (CheckBlock(tmp)) { dx = -1; goto rotate; }
tmp.x = g_CurBlock.x + 1; if (CheckBlock(tmp)) { dx = 1; goto rotate; }
tmp.x = g_CurBlock.x – 2; if (CheckBlock(tmp)) { dx = -2; goto rotate; }
tmp.x = g_CurBlock.x + 2; if (CheckBlock(tmp)) { dx = 2; goto rotate; }
return;
rotate:
// 旋转
DrawBlock(g_CurBlock, HIDE);
g_CurBlock.dir++;
g_CurBlock.x += dx;
DrawBlock(g_CurBlock);
}
// 左移方块
void OnLeft()
{
BLOCKINFO tmp = g_CurBlock;
tmp.x–;
if (CheckBlock(tmp))
{
DrawBlock(g_CurBlock, HIDE);
g_CurBlock.x–;
DrawBlock(g_CurBlock);
}
}
// 右移方块
void OnRight()
{
BLOCKINFO tmp = g_CurBlock;
tmp.x++;
if (CheckBlock(tmp))
{
DrawBlock(g_CurBlock, HIDE);
g_CurBlock.x++;
DrawBlock(g_CurBlock);
}
}
// 下移方块
void OnDown()
{
BLOCKINFO tmp = g_CurBlock;
tmp.y–;
if (CheckBlock(tmp))
{
DrawBlock(g_CurBlock, HIDE);
g_CurBlock.y–;
DrawBlock(g_CurBlock);
}
else
OnSink(); // 不可下移时,执行“沉底方块”操作
}
// 沉底方块
void OnSink()
{
int i, x, y;
// 连续下移方块
DrawBlock(g_CurBlock, HIDE);
BLOCKINFO tmp = g_CurBlock;
tmp.y–;
while (CheckBlock(tmp))
{
g_CurBlock.y–;
tmp.y–;
}
DrawBlock(g_CurBlock, FIX);
// 固定方块在游戏区
WORD b = g_Blocks[g_CurBlock.id].dir[g_CurBlock.dir];
for(i = 0; i 《 16; i++)
{
if (b & 0x8000)
{
if (g_CurBlock.y – i / 4 》= HEIGHT)
{ // 如果方块的固定位置超出高度,结束游戏
GameOver();
return;
}
else
g_World[g_CurBlock.x + i % 4][g_CurBlock.y – i / 4] = 1;
}
b 《《= 1;
}
// 检查是否需要消掉行,并标记
int row = {0};
bool bRow = false;
for(y = g_CurBlock.y; y 》= max(g_CurBlock.y – 3, 0); y–)
{
i = 0;
for(x = 0; x 《 WIDTH; x++)
if (g_World[x][y] == 1)
i++;
if (i == WIDTH)
{
bRow = true;
row[g_CurBlock.y – y] = 1;
setfillstyle(WHITE, DIAGCROSS2_FILL);
bar(0, (HEIGHT – y – 1) * SIZE + SIZE / 2 – 2, WIDTH * SIZE – 1, (HEIGHT – y – 1) * SIZE + SIZE / 2 + 2);
}
}
if (bRow)
{
// 延时 200 毫秒
Sleep(200);
// 擦掉刚才标记的行
IMAGE img;
for(i = 0; i 《 4; i++)
{
if (row[i])
{
for(y = g_CurBlock.y – i + 1; y 《 HEIGHT; y++)
for(x = 0; x 《 WIDTH; x++)
{
g_World[x][y – 1] = g_World[x][y];
g_World[x][y] = 0;
}
getimage(&img, 0, 0, WIDTH * SIZE, (HEIGHT – (g_CurBlock.y – i + 1)) * SIZE);
putimage(0, SIZE, &img);
}
}
}
// 产生新方块
NewBlock();
}