不到两百行的俄罗斯方块游戏,教学用:
/*===================================================================== // FOOLTRS.C: Fool tetris game v1.0 by skywind // 最简俄罗斯,游戏设计入门傻瓜版,林伟 1999年8月11日 // // 游戏并无任何扩展,仅以最简单的方式向人说明俄罗斯方块的基本功能实现 // 程序运行于MS-DOS,虽然现在写游戏的平台早已变化了许多, // 但是我尽量容入了一些不变理论,希望才入门的游戏设计者能得到一些帮助 // 用方向键/空格游戏,代码不超过两百行,初学C者皆能读懂,请用TC20编译 // //=====================================================================*/ #include <stdio.h> #include <stdlib.h> #include <time.h> #include <conio.h> #include <string.h> #include <dos.h> /*--------------------------------------------------------------------- // 基本定义:地图和砖块 //---------------------------------------------------------------------*/ int map[26][10]; /* 游戏地图定义: 所有砖块将被描述在地图中 ............*/ typedef struct { int d[4][4]; } Block; /* 砖块结构定义 ...............*/ int BlockList[7][4][4] = { /* 七种传统的方块定义 .................*/ { { 0, 0, 0, 0 }, { 0, 1, 0, 0 }, { 0, 1, 0, 0 }, { 0, 1, 1, 0 } }, { { 0, 0, 0, 0 }, { 0, 0, 2, 0 }, { 0, 0, 2, 0 }, { 0, 2, 2, 0 } }, { { 0, 3, 0, 0 }, { 0, 3, 0, 0 }, { 0, 3, 0, 0 }, { 0, 3, 0, 0 } }, { { 0, 4, 0, 0 }, { 0, 4, 4, 0 }, { 0, 0, 4, 0 }, { 0, 0, 0, 0 } }, { { 0, 0, 5, 0 }, { 0, 5, 5, 0 }, { 0, 5, 0, 0 }, { 0, 0, 0, 0 } }, { { 0, 6, 6, 6 }, { 0, 0, 6, 0 }, { 0, 0, 0, 0 }, { 0, 0, 0, 0 } }, { { 0, 0, 0, 0 }, { 0, 7, 7, 0 }, { 0, 7, 7, 0 }, { 0, 0, 0, 0 } }}; unsigned char cmap[]={0,0x72,0x30,0x47,0x57,0x27,0x60,0x74}; /* 颜色表 */ long GameMode = 0, GameSpeed = -1, GameScore = 0; /* 全局定义:各状态 .*/ /*--------------------------------------------------------------------- // 视频相关数据定义 //---------------------------------------------------------------------*/ char far* TextBuf = (char far*)0xb8000000l; /* 文本状态的显存地址 .....*/ char TextBck[20][10]; /* 文本状态的二级缓存 .....*/ #define SCREEN(x,y) (TextBuf[((y) * 160) + ((x) * 2)]) /* 显存中字符 */ #define SCOLOR(x,y) (TextBuf[((y) * 160) + ((x) * 2) + 1]) /* 屏幕颜色 */ /*--------------------------------------------------------------------- // 基本砖块操作函数 //---------------------------------------------------------------------*/ void Rotate(const Block* src, Block* dest) /* 旋转砖块src到dest ......*/ { int i, j; for (j = 0; j < 4; j++) for (i = 0; i < 4; i++) dest->d[i][j] = src->d[j][3 - i]; } int MapCheck(int x, int y) /* 检测地图中某坐标是否有砖块 ..............*/ { if (x < 0 || x >=10 || y < 0 || y >= 26) return -1; return map[y][x]; } int BlockCheck(int x, int y, const Block* src) /* 检测砖块位置合法性 ..*/ { int i, j; for (j = 0; j < 4; j++) for (i = 0; i < 4; i++) if (MapCheck(x + j, y - i) && src->d[i][j]) return 0; return 1; } void CreateBlock(Block* src) /* 随机创建一个砖块,然后保存到src .......*/ { int i,j,k = rand() % 7; Block t; for (i = 0; i < 4; i++) for (j = 0; j < 4; j++) src->d[i][j] = BlockList[k][i][j]; for (i = rand() % 4; i > 0; i--, *src = t) Rotate(src, &t); } long TimePass = 0, TimeExit = -1, GameTime = -1, w; void GameMain(void); /*--------------------------------------------------------------------- // 程序入口函数 //---------------------------------------------------------------------*/ void main(void) { clrscr(); /* 初始化:清屏 ...........*/ randomize(); /* 初始化:随机数种子 .....*/ for (w = 0; w < 26; w++) memset(map[w], 0, sizeof(int) * 10); for (w = 0; w < 22; w++) /* 初始化:绘制边框 .......*/ SCREEN(w,0)=254, SCOLOR(w,0)=8, SCREEN(w,21)=254, SCOLOR(w,21)=8, SCREEN(0,w)=254, SCOLOR(0,w)=8, SCREEN(21,w)=254, SCOLOR(21,w)=8; GameSpeed = 2; /* 初始化:设置速度 .......*/ GameMain(); /* 进入游戏主程序 .........*/ } int readkey(void) /* 无等待的接收一个键盘消息 ...............*/ { if (!kbhit()) return 0; /* 如果当前没有按键消息则直接返回 .........*/ if ((w = getch()) == 0) return getch(); /* 返回非控制键 .........*/ return (int)w; } int SpeedX[10] = { 17, 15, 13, 11, 9, 7, 5, 4, 3, 2 }; /* 速度级别 ....*/ int x = 3, y = 20, v = 0, level; /* 全局定义:砖块的坐标,速度 .....*/ Block A, B, C; /* 全局定义:A, B, C三个活动砖块 ..*/ void GameDriver(void); void GameView(void); /*--------------------------------------------------------------------- // 游戏逻辑核心(重要): 每秒循环18.2次,处理砖块移动/下落/消除等逻辑 //---------------------------------------------------------------------*/ void GameMain(void) { long key, i, j, m, n, old_time = 0; /* 定义各变量 .........*/ CreateBlock(&A); CreateBlock(&C); /* 初始化砖块 .........*/ for (GameMode = 0, v = SpeedX[GameSpeed]; GameMode == 0; TimePass++) { while (clock() == old_time); /* 速度控制,clock()为 18.2Hz ..*/ old_time = clock(); nosound(); key = readkey(); /* 读入一个键盘消息 ............*/ if (key == 75) if (BlockCheck(x - 1, y, &A)) x--; /* 左移 ...*/ if (key == 77) if (BlockCheck(x + 1, y, &A)) x++; /* 右移 ...*/ if (key == 80) v = 0; /* 下落: SET v = 0 ....*/ if (key == ' ') { /* 直接下落 ...........*/ for (i = y; i >= 0; i--) if (!BlockCheck(x, i - 1, &A)) break; y = i; v = 0; } if (key == 27) GameMode = 1; /* ESC: 退出游戏 ......*/ if (key == 72) { /* UP: 转动砖块 .......*/ Rotate(&A, &B); if (BlockCheck(x, y, &B)) A = B; } if (--v <= 0) { /* 到达了下落的时间 .......*/ v = SpeedX[GameSpeed]; /* 重置砖块速度 ...........*/ if (BlockCheck(x, y - 1, &A) == 0) {/* 如果到达map底部 ....*/ for (i=0, m=y; i<4; i++,m--) for (j=0,n=x; j<4; j++,n++) if (MapCheck(n,m)==0 && A.d[i][j]) /* 将A复制到map */ map[m][n] = A.d[i][j]; A = C; /* 将C复制到A .........*/ CreateBlock(&C); /* 创建新的C砖块 ......*/ if (y >= 20) GameMode = -1; /* 判断死亡 ...........*/ else y = 20, x = 3; GameDriver(); /* 消除完成的行 .......*/ } else y--; /* 否则:下降 .........*/ } GameView(); /* 显示: 更新屏幕 .....*/ } while (kbhit()) getch(); /* 按任意键结束 .......*/ while (!kbhit()); nosound(); /* 关闭扬声器 .........*/ } void GameDriver(void) /* 处理消除行 ..............................*/ { int inc[5]={10,100,300,600,1000}, count = 0, x, y, k, i, j; for (y = 0; y < 25; y++) for (; ; count++) { /* 扫描map测试消行 ...*/ for (x = 0, k = 0; x < 10; x++) k += (map[y][x])? 1 : 0; if (k < 10) break; /* 计算一行是否可以消去 .......*/ for (j = y; j < 25; j++) memcpy(map[j], map[j+1], 10 * sizeof(int)); } if ((GameScore / 10000) != ((GameScore + inc[count]) / 10000)) GameSpeed = (GameSpeed + 1) % 10; /* 计算速度升级 ...........*/ GameScore += inc[count]; /* 计算新的分数 ...........*/ sound((count == 0)? 75 : 900); } void GameView(void) /* 绘制:将map里面的砖块绘制到屏幕 .........*/ { int i, j, m, n; for (i=0; i<10; i++) for (j=0; j<20; j++) /* 更新TextBck二级缓存 ..*/ TextBck[j][i] = map[j][i]; for (i=0, n=x; i<4; i++, n++) for (j = 0, m=y; j < 4; j++, m--) if (m>=0&&m<20 && n>=0&&n<10 && A.d[j][i]) TextBck[m][n] = A.d[j][i]; for (i = 0; i < 10; i++) for (j = 0; j < 20; j++) { /* 显示地图 ...*/ m = TextBck[19 - j][i], n = i * 2; SCREEN(n+1,j+1) = (m)? '[':'.', SCOLOR(n+1,j+1) = (m)? cmap[m] : 8; SCREEN(n+2,j+1) = (m)? ']':'.', SCOLOR(n+2,j+1) = (m)? cmap[m] : 8; } for (j = 0; j < 4; j++) for (i = 0; i < 4; i++) { /* 显示下个砖块*/ m = C.d[j][i], n = 160 * (j + 8) + 4 * i + 58; TextBuf[n+0] = (m)? '[':'.', TextBuf[n+1] = (m)? cmap[m] : 8; TextBuf[n+2] = (m)? ']':'.', TextBuf[n+3] = (m)? cmap[m] : 8; } if (TimePass & 3) return; textcolor(0xa-8); gotoxy(30,1); printf("POSITION: [%2d,%2d]", x, y); /* 显示分数等 .*/ gotoxy(30,2); printf("SPEED: %d", GameSpeed); gotoxy(30,6); printf("SCORE: "); cprintf("%ld", GameScore); gotoxy(30,5); printf("TIME: "); /* 显示时间 ...*/ m = (int)(TimePass / 18.2) / 60, n = (int)(TimePass / 18.2) % 60; cprintf("%d%d:%d%d", (m/10), (m%10), (n/10), (n%10)); gotoxy(1, 23); }