找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 25653|回复: 39
打印 上一主题 下一主题
收起左侧

原创51单片机俄罗斯方块程序 12864屏显示 代码有详细注释

  [复制链接]
跳转到指定楼层
楼主
给51黑电子论坛的朋友们分享一个很优秀的俄罗斯方块程序,lcd12864屏幕 是用普通的51单片机做的,P3.4口,短按就可以开始运行了。效果如下图所示:

完整的源码下载:
俄罗斯方块全部源码.zip (147.88 KB, 下载次数: 599)



Tetris.c源程序文件内容:
  1. /*******************************************************
  2. * 文件名称:Tetris.c
  3. * 单 片 机:STC89C52RC
  4. * 简    述:使用LCD12864显示的俄罗斯方块程序
  5. * 功    能:计分,下一个方块预览,欢迎结束界面,长按连续左右移,暂停(按键Left+Turn)
  6. * 作    者:刘琦
  7. * 完成日期:4月21日
  8. * IO口设定:按键
  9.                         sbit key_sr_left=P3^7;
  10.                         sbit key_sr_turn=P3^6;
  11.                         sbit key_sr_right=P3^5;
  12.                         sbit key_sr_down=P3^4;
  13.                         LCD12864
  14.                         sbit RS_Port=P1^0;
  15.                         sbit RW_Port=P1^1;
  16.                         sbit E_Port=P2^5;
  17.                         sbit PSB_Port=P1^2;
  18.                         sbit RST_Port=P1^4;
  19.                         数码管锁存器
  20.                         sbit dula=P2^6;
  21.                         sbit wela=P2^7;

  22. *******************************************************/

  23. /*****************************头文件区*****************************************************/
  24. #include<reg52.h>
  25. //#include"STC15F2K60S2.h"
  26. #include"task_key.h"
  27. #include"task_activation.h"
  28. #include"12864b.h"
  29. #include"TaskBlock.h"
  30. #include"Block.h"
  31. /*****************************宏定义区*****************************************************/

  32. //需要判断是否刷新的任务总数
  33. //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  34. #define NUM_TASK_FLAG 7
  35. //需要定时刷新的任务总数
  36. //1TaskGoingDown,2KeyService
  37. #define NUM_TASK_REFRESH 3
  38. //刷新频率
  39. #define TIME_PER_SEC 200                                                        //每次进入中断的频率,200Hz
  40. #define TIME_CLOCK 11059200                                                        //晶振频率
  41. #define TIME_KEY_SERVICE_50HZ  TIME_PER_SEC/50          //按键服务频率,0.02s
  42. #define TIME_TASKGOINGDOWN_2HZ  TIME_PER_SEC/2          //下落频率,0.5s

  43. /******************************子函数声明区***********************************************/

  44. void initial_myself(void);   
  45. void initial_peripheral(void);
  46. void delay100ms(void);        

  47. /******************************全局变量定义区***********************************************/
  48. unsigned char uc_state=1;//状态标示:1欢迎界面,2等待开始界面,3下落界面,4死亡界面,5暂停界面
  49. enum Block{S1,S2,Z1,Z2,L1,L2,L3,L4,J1,J2,J3,J4,I1,I2,O1,T_1,T_2,T_3,T_4}enumBlock;//方块编号
  50. unsigned char ucBlockNow=0;//当前方块
  51. unsigned char ucBlockNext=0;//下一个方块

  52. unsigned char ucFlagScreenChange=1;//界面变更标志,界面更改时置1,激活task_activation

  53. unsigned char ucCoordinate[2];//方块的旋转点坐标,横H,竖S
  54. unsigned char xdata ucxMap[10][20];//整张屏幕的标记地图,用来记录未消除方块位置,有方块则为1
  55. unsigned char ucxBlockPosition[4][2];//方块的4个小方块坐标,[0][0],[0][1]为旋转点坐标H,S

  56. unsigned int uc_delay_task_cnt[NUM_TASK_REFRESH];//任务刷新延迟
  57. unsigned char uc_flag_taskrefresh[NUM_TASK_FLAG];//任务刷新标志

  58. unsigned long int ucScore=0;//得分

  59. /******************************主函数开始***********************************************/
  60. void main(void)
  61. {
  62.         unsigned char i=0;
  63.         initial_myself();
  64.         delay100ms();   
  65.     initial_peripheral();


  66.         while(1)
  67.         {
  68.                 if(ucFlagScreenChange==1)//界面变更标志为1时,进行任务激活,防止一直刷新界面
  69.                 {
  70.                         task_activation(uc_state);
  71.                         ucFlagScreenChange=0;
  72.                 }
  73.                 if(uc_delay_task_cnt[2]==0)//20HZ频率刷新key_service
  74.                 {
  75.                         task_key_service();
  76.                         ET0=0;//在中断中也有可能变化的变量在更改前时先关闭中断
  77.                         uc_delay_task_cnt[2]=TIME_KEY_SERVICE_50HZ;//延迟重置
  78.                         ET0=1;
  79.                 }
  80.                 if(uc_flag_taskrefresh[5]&&uc_delay_task_cnt[1]==0)//需要时,2HZ频率刷新TaskGoingDown
  81.                 {
  82.                         if(!TaskGoingDown())//下落失败
  83.                         {
  84.                                 //将当前方块位置写入Map
  85.                                 for(i=0;i<4;i++)
  86.                                 {
  87.                                         ucxMap[ ucxBlockPosition[i][0] ][ ucxBlockPosition[i][1] ]=1;
  88.                                 }
  89.                                 //调用消除方块函数
  90.                                 TaskClear();
  91.                         }
  92.                         ET0=0;//在中断中也有可能变化的变量在更改前时先关闭中断
  93.                         uc_delay_task_cnt[1]=TIME_TASKGOINGDOWN_2HZ;//延迟重置
  94.                         ET0=1;
  95.                 }

  96.         }
  97. }

  98. //中断函数
  99. void timer0(void) interrupt 1
  100. {
  101.         unsigned char i;
  102.         TR0=0;
  103.         TH0=255-TIME_CLOCK/TIME_PER_SEC/12/256;
  104.         TL0=255-TIME_CLOCK/TIME_PER_SEC/12%256;
  105.         //task_delay[]减到0时,相应的函数准备就绪
  106.         for(i=0;i<NUM_TASK_REFRESH;i++)
  107.         {
  108.                 if(uc_delay_task_cnt[i]!=0)//延迟不为0时才减
  109.                         {uc_delay_task_cnt[i]--;};
  110.         }
  111.         task_key_scan();//按键扫描函数,放在中断中,保证准时扫描
  112.         TR0=1;
  113. }

  114. //初始化区
  115. void initial_myself(void)//第一区 初始化单片机
  116. {
  117.         unsigned char i;
  118.         for(i=0;i<NUM_TASK_REFRESH;i++)uc_delay_task_cnt[i]=0;//初始化让所有任务就绪
  119.         TMOD=0X21; //定时器0为16位不自动重装,用来定时;定时器1为8位自动重装,用来产生随机数
  120.         TH0=255-TIME_CLOCK/TIME_PER_SEC/12/256;
  121.         TL0=255-TIME_CLOCK/TIME_PER_SEC/12%256;
  122.         TH1=0;
  123.         TL1=0;
  124. }

  125. void initial_peripheral(void) //第二区 初始化外围
  126. {
  127.         Lcd_Init();//初始化12864
  128.         EA=1;     //开总中断
  129.         ET0=1;    //允许定时器0中断,定时器1无需开启中断
  130.     TR0=1;    //启动定时器0
  131.     TR1=1;         //启动定时器1
  132. }

  133. void delay100ms(void)                //@11.0592MHz
  134. {
  135.         unsigned char i, j, k;

  136.         ;
  137.         ;
  138.         i = 5;
  139.         j = 52;
  140.         k = 195;
  141.         do
  142.         {
  143.                 do
  144.                 {
  145.                         while (--k);
  146.                 } while (--j);
  147.         } while (--i);
  148. }
复制代码
  1. /*******************************************************
  2. * 文件名称:task_activation.c
  3. * 单 片 机:STC89C52RC
  4. * 简    述:任务激活子程序,uc_state:1欢迎界面,2等待界面,3下落界面,4死亡界面
  5. * 作    者:刘琦
  6. * 完成日期:2015年5月1日
  7. *******************************************************/
  8. #include"reg52.h"
  9. #include"12864b.h"
  10. #include"tetris.h"
  11. #include"TaskBlock.h"
  12. #include"Score.h"
  13. //#include"STC15F2K60S2.h"

  14. #define  xchar unsigned char code

  15. //欢迎词
  16. xchar CorpInf[]=
  17. {
  18.         "来玩俄罗斯方块!"
  19. };
  20. //死亡词
  21. xchar CorpInf1[]=
  22. {
  23.         "不好意思你死了!"
  24. };

  25. void task_activation(unsigned char state)        
  26. {
  27.         unsigned char i=0,j=0;
  28.         switch(state)
  29.         {
  30.                 case 1: //欢迎界面
  31.                                 DisGBStr(CorpInf);                                //写入欢迎词
  32.                         //当前界面需要刷新的任务
  33.                                 //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  34.                                 uc_flag_taskrefresh[1]=0;               
  35.                             uc_flag_taskrefresh[2]=0;
  36.                             uc_flag_taskrefresh[3]=0;
  37.                             uc_flag_taskrefresh[4]=1;
  38.                             uc_flag_taskrefresh[5]=0;
  39.                             uc_flag_taskrefresh[6]=0;
  40.                             break;
  41.                 case 2:        //等待界面
  42.                                 //清空所有显示
  43.                                 Cram_Off();                //清空文本显示
  44.                                 Clr_GDRAM();        //清空绘图显示
  45.                                 //画出边框
  46.                                 DrawLineY(7,1,62,1);
  47.                                 DrawLineX(7,127,1,1);
  48.                                 DrawLineX(7,127,62,1);

  49.                                 //将地图数据清空
  50.                                 for(i=0;i<10;i++)
  51.                                 {
  52.                                         for(j=0;j<20;j++)
  53.                                         {
  54.                                                 ucxMap[i][j]=0;
  55.                                         }
  56.                                 }

  57.                                 ucScore=0;//计分清0

  58.                         //当前界面需要刷新的任务
  59.                                 //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  60.                         uc_flag_taskrefresh[1]=0;
  61.                             uc_flag_taskrefresh[2]=0;
  62.                             uc_flag_taskrefresh[3]=0;
  63.                             uc_flag_taskrefresh[4]=1;
  64.                             uc_flag_taskrefresh[5]=0;
  65.                             uc_flag_taskrefresh[6]=0;
  66.                             break;
  67.                 case 3:        //下落界面
  68.                                 TaskScore ();                //写分,刷新最开始的0000000
  69.                                 TaskComingNew();        //出新方块
  70.                         //当前界面需要刷新的任务
  71.                                 //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  72.                                 uc_flag_taskrefresh[1]=1;
  73.                             uc_flag_taskrefresh[2]=1;
  74.                             uc_flag_taskrefresh[3]=1;
  75.                             uc_flag_taskrefresh[4]=1;
  76.                             uc_flag_taskrefresh[5]=1;
  77.                             uc_flag_taskrefresh[6]=1;
  78.                             break;
  79.                 case 4:        //死亡界面
  80.                                 Cram_Off();                //清空文本显示
  81.                                 Clr_GDRAM();        //清空绘图显示
  82.                                 DisGBStr(CorpInf1);
  83.                         //当前界面需要刷新的任务
  84.                                 //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  85.                         uc_flag_taskrefresh[1]=0;
  86.                             uc_flag_taskrefresh[2]=0;
  87.                             uc_flag_taskrefresh[3]=0;
  88.                             uc_flag_taskrefresh[4]=1;
  89.                             uc_flag_taskrefresh[5]=0;
  90.                             uc_flag_taskrefresh[6]=0;
  91.                             break;
  92.                 case 5:        //暂停界面
  93.                         //当前界面需要刷新的任务
  94.                                 //1KeyLeft,2KeyTurn,3KeyRight,4KeyDown,5TaskGoingDown,6KeyPause
  95.                         uc_flag_taskrefresh[1]=0;
  96.                             uc_flag_taskrefresh[2]=0;
  97.                             uc_flag_taskrefresh[3]=0;
  98.                             uc_flag_taskrefresh[4]=1;
  99.                             uc_flag_taskrefresh[5]=0;
  100.                             uc_flag_taskrefresh[6]=0;
  101.                             break;
  102.                 default:
  103.                             break;
  104.         }
  105. }
复制代码
  1. #include"reg52.h"
  2. //#include"STC15F2K60S2.h"
  3. #include"tetris.h"
  4. #include"Block.h"
  5. #include"TaskBlock.h"

  6. #define CONST_KEY_TIME 10    //按键去抖动延时的时间

  7. //#define const_time_0_25s  111   //0.25秒钟的时间需要的定时中断次数
  8. //#define const_time_0_5s     222   //0.5秒钟的时间需要的定时中断次数

  9. #define const_time_0_25s  60   //0.125秒钟的时间需要的定时中断次数
  10. #define const_time_0_5s     60   //0.125秒钟的时间需要的定时中断次数


  11. sbit key_sr_left=P3^7;
  12. sbit key_sr_turn=P3^6;
  13. sbit key_sr_right=P3^5;
  14. sbit key_sr_down=P3^4;

  15. unsigned char ucKeySec=0;   //被触发的按键编号

  16. unsigned int  uiKeyTimeCnt_left=0; //按键去抖动延时计数器
  17. unsigned int  uiKeyCtntyCnt_left=0; //按键连续触发的间隔延时计数器
  18. unsigned char ucKeyLock_left=0; //按键触发后自锁的变量标志

  19. unsigned int  uiKeyTimeCnt_turn=0; //按键去抖动延时计数器
  20. //unsigned int  uiKeyCtntyCnt_turn=0; //按键连续触发的间隔延时计数器
  21. unsigned char ucKeyLock_turn=0; //按键触发后自锁的变量标志

  22. unsigned int  uiKeyTimeCnt_right=0; //按键去抖动延时计数器
  23. unsigned int  uiKeyCtntyCnt_right=0; //按键连续触发的间隔延时计数器
  24. unsigned char ucKeyLock_right=0; //按键触发后自锁的变量标志

  25. unsigned int  uiKeyTimeCnt_down=0; //按键去抖动延时计数器
  26. unsigned char ucKeyLock_down=0; //按键触发后自锁的变量标志

  27. unsigned int  uiKeyTimeCnt_pause=0; //按键去抖动延时计数器
  28. unsigned char ucKeyLock_pause=0; //按键触发后自锁的变量标志

  29. //放在中断中,其中的每个按键的标志位为1时才刷新
  30. void task_key_scan(void)
  31. {
  32.         if(uc_flag_taskrefresh[1])
  33.         {
  34.                 if(key_sr_left==1)//left
  35.                   {
  36.                      ucKeyLock_left=0;
  37.                      uiKeyTimeCnt_left=0;
  38.                          uiKeyCtntyCnt_left=0;   
  39.                  }
  40.             else if(ucKeyLock_left==0)
  41.                  {
  42.                      uiKeyTimeCnt_left++;
  43.                      if(uiKeyTimeCnt_left>CONST_KEY_TIME)
  44.                      {
  45.                         uiKeyTimeCnt_left=0;
  46.                         ucKeyLock_left=1;  
  47.                         ucKeySec=1;    //触发1号键,left
  48.                      }
  49.             }
  50.                 else if(uiKeyTimeCnt_left<const_time_0_5s)
  51.                 {
  52.                          uiKeyTimeCnt_left++;
  53.                 }
  54.                 else
  55.                 {
  56.                          uiKeyCtntyCnt_left++;
  57.                          if(uiKeyCtntyCnt_left>const_time_0_25s)
  58.                          {
  59.                                  uiKeyCtntyCnt_left=0;
  60.                                 ucKeySec=1;    //触发1号键,left
  61.                          }
  62.                 }
  63.         }

  64.         if(uc_flag_taskrefresh[2])
  65.         {
  66.                 if(key_sr_turn==1)//turn
  67.                   {
  68.                      ucKeyLock_turn=0;
  69.                      uiKeyTimeCnt_turn=0;
  70. //                         uiKeyCtntyCnt_turn=0;   
  71.                  }
  72.             else if(ucKeyLock_turn==0)
  73.                  {
  74.                      uiKeyTimeCnt_turn++;
  75.                      if(uiKeyTimeCnt_turn>CONST_KEY_TIME)
  76.                      {
  77.                         uiKeyTimeCnt_turn=0;
  78.                         ucKeyLock_turn=1;  
  79.                         ucKeySec=2;    //触发2号键,turn
  80.                      }
  81.             }
  82.         }

  83.         if(uc_flag_taskrefresh[3])
  84.         {
  85.                 if(key_sr_right==1)//right
  86.                   {
  87.                      ucKeyLock_right=0;
  88.                      uiKeyTimeCnt_right=0;
  89.                          uiKeyCtntyCnt_right=0;   
  90.                  }
  91.             else if(ucKeyLock_right==0)
  92.                  {
  93.                      uiKeyTimeCnt_right++;
  94.                      if(uiKeyTimeCnt_right>CONST_KEY_TIME)
  95.                      {
  96.                         uiKeyTimeCnt_right=0;
  97.                         ucKeyLock_right=1;  
  98.                         ucKeySec=3;    //触发3号键,right
  99.                      }
  100.             }
  101.                 else if(uiKeyTimeCnt_right<const_time_0_5s)
  102.                 {
  103.                          uiKeyTimeCnt_right++;
  104.                 }
  105.                 else
  106.                 {
  107.                          uiKeyCtntyCnt_right++;
  108.                          if(uiKeyCtntyCnt_right>const_time_0_25s)
  109.                          {
  110.                                  uiKeyCtntyCnt_right=0;
  111.                                 ucKeySec=3;    //触发3号键,right
  112.                          }
  113.                 }
  114.         }

  115.         if(uc_flag_taskrefresh[4])
  116.         {
  117.                 if(key_sr_down==1)//down
  118.                   {
  119.                      ucKeyLock_down=0; //按键自锁标志清零
  120.                      uiKeyTimeCnt_down=0;//按键去抖动延时计数器清零   
  121.                  }
  122.             else if(ucKeyLock_down==0)//有按键按下,且是第一次被按下,并且当前界面支持这个按键
  123.                  {
  124.                      uiKeyTimeCnt_down++; //累加定时中断次数
  125.                           if(uiKeyTimeCnt_down>CONST_KEY_TIME)
  126.                      {
  127.                         uiKeyTimeCnt_down=0;
  128.                         ucKeyLock_down=1;  //自锁按键置位,避免一直触发
  129.                         ucKeySec=4;    //触发4号键,down
  130.                      }
  131.             }
  132.         }

  133.         if(uc_flag_taskrefresh[6])
  134.         {                        
  135.           if(key_sr_left==1||key_sr_turn==1)//pause,按键左和按键旋转未同时按下
  136.           {
  137.                  ucKeyLock_pause=0; //按键自锁标志清零
  138.                  uiKeyTimeCnt_pause=0;//按键去抖动延时计数器清零      
  139.           }
  140.                 else if(ucKeyLock_pause==0)//有按键按下,且是第一次被按下
  141.                 {
  142.                     uiKeyTimeCnt_pause++; //累加定时中断次数
  143.                     if(uiKeyTimeCnt_pause>CONST_KEY_TIME)
  144.                     {
  145.                             uiKeyTimeCnt_pause=0;
  146.                         ucKeyLock_pause=1;  //自锁按键置位,避免一直触发
  147.                     ucKeySec=6;    //触发6号键,暂停
  148.                     }
  149.                 }
  150.         }
  151. }

  152. //按键服务函数
  153. void task_key_service(void)
  154. {
  155.         unsigned char ucCheck=0;
  156.         unsigned char ucOldEnum=0;
  157.         unsigned char ucNewEnum=0;
  158.         unsigned char i=0;
  159.         switch(ucKeySec) //按键服务状态切换
  160.           {
  161.             case 1:// 1号键,left
  162.                                 ucCoordinate[0]++;                                //旋转点坐标H+1
  163.                                 WriteBlockCoordinate();                        //写入其他3个方块坐标
  164.                                 ucCheck=TaskCheck();                        //检测碰撞
  165.                                 if(ucCheck)                                                //未碰撞,则左移
  166.                                 {
  167.                                         ucCoordinate[0]--;
  168.                                         ChangeBlock(0);
  169.                                         ucCoordinate[0]++;
  170.                                         ChangeBlock(1);
  171.                                 }
  172.                                 else                                                    //碰撞,不变
  173.                                 {
  174.                                         ucCoordinate[0]--;
  175.                                         WriteBlockCoordinate();
  176.                                 }
  177.             break;
  178.                 case 2:// 2号键,turn
  179.                                 ucOldEnum=enumBlock;
  180.                                 switch(enumBlock)                                //旋转后改变方块编号
  181.                                 {
  182.                                         case S1:
  183.                                                         enumBlock=S2;
  184.                                         break;
  185.                                         case S2:
  186.                                                         enumBlock=S1;
  187.                                         break;
  188.                                         case Z1:
  189.                                                         enumBlock=Z2;
  190.                                         break;
  191.                                         case Z2:
  192.                                                         enumBlock=Z1;
  193.                                         break;
  194.                                         case L1:
  195.                                                         enumBlock=L2;
  196.                                         break;
  197.                                         case L2:
  198.                                                         enumBlock=L3;
  199.                                         break;
  200.                                         case L3:
  201.                                                         enumBlock=L4;
  202.                                         break;
  203.                                         case L4:
  204.                                                         enumBlock=L1;
  205.                                         break;
  206.                                         case J1:
  207.                                                         enumBlock=J2;
  208.                                         break;
  209.                                         case J2:
  210.                                                         enumBlock=J3;
  211.                                         break;
  212.                                         case J3:
  213.                                                         enumBlock=J4;
  214.                                         break;
  215.                                         case J4:
  216.                                                         enumBlock=J1;
  217.                                         break;
  218.                                         case I1:
  219.                                                         enumBlock=I2;
  220.                                         break;
  221.                                         case I2:
  222.                                                         enumBlock=I1;
  223.                                         break;
  224.                                         case O1:
  225.                                                         enumBlock=O1;
  226.                                         break;
  227.                                         case T_1:
  228.                                                         enumBlock=T_2;
  229.                                         break;
  230.                                         case T_2:
  231.                                                         enumBlock=T_3;
  232.                                         break;
  233.                                         case T_3:
  234.                                                         enumBlock=T_4;
  235.                                         break;
  236.                                         case T_4:
  237.                                                         enumBlock=T_1;
  238.                                         break;
  239.                                         default:
  240.                                         break;
  241.                                 }
  242.                                 WriteBlockCoordinate();
  243.                                 ucCheck=TaskCheck();
  244.                                 if(ucCheck)
  245.                                 {
  246.                                         ucNewEnum=enumBlock;
  247.                                         enumBlock=ucOldEnum;
  248.                                         ChangeBlock(0);
  249.                                         enumBlock=ucNewEnum;
  250.                                         ChangeBlock(1);
  251.                                 }
  252.                                 else
  253.                                 {
  254.                                         enumBlock=ucOldEnum;
  255.                                         WriteBlockCoordinate();
  256.                                 }

  257.             break;
  258.             case 3:// 3号键,right
  259.                                 ucCoordinate[0]--;
  260.                                 WriteBlockCoordinate();
  261.                                 ucCheck=TaskCheck();
  262.                                 if(ucCheck)
  263.                                 {
  264.                                         ucCoordinate[0]++;
  265.                                         ChangeBlock(0);
  266.                                         ucCoordinate[0]--;
  267.                                         ChangeBlock(1);
  268.                                 }
  269.                                 else
  270.                                 {
  271.                                         ucCoordinate[0]++;
  272.                                         WriteBlockCoordinate();
  273.                                 }

  274.             break;
  275.                 case 4:// 4号键,down
  276.                                 //根据所处界面不同,按键down的功能不同
  277.                                 switch(uc_state)
  278.                                 {
  279.                                         case 1:
  280.                                                         ucBlockNow=TL1%7;                //准备进入等待界面,就先产生当前随机方块
  281.                                                         uc_state=2;                                 //进入等待界面
  282.                                                         ucFlagScreenChange=1;        //激活界面变更
  283.                                         break;
  284.                                         case 2:
  285.                                                         uc_state=3;                                //进入下落界面
  286.                                                         ucFlagScreenChange=1;
  287.                                         break;
  288.                                         case 3:        //在下落界面下,down为直接下落到底
  289.                                                         ET0=0;//关闭计时器0,出新后开启,防止直接下落
  290.                                                         ChangeBlock(0);                 //先清除当前的图形
  291.                                                         do                                                //循环模拟下落并检测,直到失败
  292.                                                         {
  293.                                                                 ucCoordinate[1]++;
  294.                                                                 WriteBlockCoordinate();
  295.                                                                 ucCheck=TaskCheck();
  296.                                                         }
  297.                                                         while(ucCheck);
  298.                                                         ucCoordinate[1]--;                //恢复到最后一次成功的坐标
  299.                                                         WriteBlockCoordinate();
  300.                                                         ChangeBlock(1);                        //画方块

  301.                                                         //立即写入,并消除,防止到底后还可以移动
  302.                                                         //写入Map
  303.                                                         for(i=0;i<4;i++)
  304.                                                         {
  305.                                                                 ucxMap[ ucxBlockPosition[i][0] ][ ucxBlockPosition[i][1] ]=1;
  306.                                                         }
  307.                                                         //调用消除
  308.                                                         TaskClear();
  309.                                                         //刷新下落延迟,防止出新后,直接下落
  310.                                                         ET0=1;
  311.                                         break;
  312.                                         case 4:        //死亡界面
  313.                                                         ucBlockNow=TL1%7;                //准备进入等待界面,就先产生当前随机方块
  314.                                                         uc_state=2;                                 //进入等待界面
  315.                                                         ucFlagScreenChange=1;
  316.                                         break;
  317.                                         case 5:        //暂停界面
  318.                                                         uc_state=3;                                //进入下落界面
  319.                                                         //开启下落界面时的相关任务
  320.                                                         uc_flag_taskrefresh[1]=1;
  321.                                                     uc_flag_taskrefresh[2]=1;
  322.                                                     uc_flag_taskrefresh[3]=1;
  323.                                                     uc_flag_taskrefresh[4]=1;
  324.                                                     uc_flag_taskrefresh[5]=1;
  325.                                                     uc_flag_taskrefresh[6]=1;
  326.                                         break;
  327.                                 }
  328.             break;
  329.                 case 6:// 6号键,pause
  330.                                 uc_state=5;
  331.                                 ucFlagScreenChange=1;
  332.                 break;
  333.                 default:
  334.                 break;                 
  335.           }
  336.         ucKeySec=0;
  337. }
复制代码
  1. /*******************************************************
  2. * 文件名称:TaskBlock.c
  3. * 单 片 机:STC89C52RC
  4. * 简    述:方块子程序,包含,下落,出新,消除,检测
  5. * 作    者:刘琦
  6. * 完成日期:2015年4月21日
  7. *******************************************************/
  8. #include<reg52.h>
  9. #include"Tetris.h"
  10. #include"Block.h"
  11. #include"tetris.h"
  12. #include"Score.h"
  13. /*******************************************************
  14. * 函 数 名:TaskCheck(void)
  15. * 功能描述:检测
  16. * 函数说明:检测左右移,下落,出新,旋转
  17. * 输入参数:无
  18. * 返 回 值:成功失败标示,ucReturn
  19. *******************************************************/
  20. unsigned char TaskCheck(void)
  21. {
  22.         unsigned char ucReturn=0;
  23.         unsigned char i=0;
  24.         //加入检测部分,包括边界检测,碰撞检测
  25.         //边界检测
  26.         for(i=0;i<4;i++)
  27.         {
  28.                 if(ucxBlockPosition[i][0]>=0&&ucxBlockPosition[i][0]<=9                        //检测左右
  29.                    &&ucxBlockPosition[i][1]>=0&&ucxBlockPosition[i][1]<=19)                //检测上下
  30.                 {
  31.                         ucReturn=1;
  32.                 }
  33.                 else
  34.                 {
  35.                         ucReturn=0;
  36.                         return ucReturn;//如果边界检测失败,就不进行下面检测,直接返回失败
  37.                 }
  38.         }
  39.         //碰撞检测
  40.         for(i=0;i<4;i++)
  41.         {
  42.                 if(ucxMap[ ucxBlockPosition[i][0] ][ ucxBlockPosition[i][1] ]!=0)//与地图标记位置进行比较
  43.                 {
  44.                         ucReturn=0;
  45.                         return ucReturn;
  46.                 }
  47.         }
  48.         return ucReturn;
  49. }
  50. /*******************************************************
  51. * 函 数 名:unsigned char TaskGoingDown(void)
  52. * 功能描述:方块下落
  53. * 函数说明:
  54. * 输入参数:无
  55. * 返 回 值:ucResult,1无碰撞,0碰撞
  56. *******************************************************/
  57. unsigned char TaskGoingDown(void)
  58. {
  59.         unsigned char ucCheck=0;
  60.         unsigned char ucResult=0;
  61.         ucCoordinate[1]++;                                //旋转点坐标,竖S+1
  62.         WriteBlockCoordinate();                        //将其他三个小方块坐标写入数组
  63.         ucCheck=TaskCheck();                        //检测碰撞
  64.         if(ucCheck)                                                //无碰撞
  65.         {
  66.                 ucCoordinate[1]--;                        //坐标复位
  67.                 ChangeBlock(0);                                //在老位置消除方块,只刷新有改变的位置,加快显示速度
  68.                 ucCoordinate[1]++;                        //坐标+1
  69.                 ChangeBlock(1);                                //在新位置画方块
  70.                 ucResult=1;
  71.         }
  72.         else                                                        //有碰撞,则不变
  73.         {
  74.                 ucCoordinate[1]--;                        
  75.                 WriteBlockCoordinate();
  76.                 ucResult=0;
  77.         }
  78.         return ucResult;
  79. }
  80. /*******************************************************
  81. * 函 数 名:unsigned char TaskComingNew(void)
  82. * 功能描述:方块出新
  83. * 函数说明:
  84. * 输入参数:无
  85. * 返 回 值:ucResult,1成功,0失败
  86. *******************************************************/
  87. unsigned char TaskComingNew(void)
  88. {
  89.         unsigned char ucCheck=0;
  90.         unsigned char ucResult=0;
  91.         //写入当前方块的相关信息
  92.         switch(ucBlockNow)                                         
  93.         {
  94.                 case 0:
  95.                                 enumBlock=S1;
  96.                                 ucCoordinate[0]=4;
  97.                                 ucCoordinate[1]=0;
  98.                 break;
  99.                 case 1:
  100.                                 enumBlock=Z1;
  101.                                 ucCoordinate[0]=4;
  102.                                 ucCoordinate[1]=0;
  103.                 break;
  104.                 case 2:
  105.                                 enumBlock=L1;
  106.                                 ucCoordinate[0]=4;
  107.                                 ucCoordinate[1]=1;
  108.                 break;
  109.                 case 3:
  110.                                 enumBlock=J1;
  111.                                 ucCoordinate[0]=4;
  112.                                 ucCoordinate[1]=1;
  113.                 break;
  114.                 case 4:
  115.                                 enumBlock=I1;
  116.                                 ucCoordinate[0]=5;
  117.                                 ucCoordinate[1]=0;
  118.                 break;
  119.                 case 5:
  120.                                 enumBlock=O1;
  121.                                 ucCoordinate[0]=5;
  122.                                 ucCoordinate[1]=0;
  123.                 break;
  124.                 case 6:
  125.                                 enumBlock=T_1;
  126.                                 ucCoordinate[0]=4;
  127.                                 ucCoordinate[1]=0;
  128.                 break;
  129.                 default:
  130.                 break;
  131.         }
  132.         //检测碰撞
  133.         WriteBlockCoordinate();        
  134.         ucCheck=TaskCheck();
  135.         if(ucCheck)                                                //成功
  136.         {
  137.                 ChangeSmallBlock(0);                //将预览的方块清除
  138.                 ucBlockNext=TL1%7;                        //产生随机数,作为预览方块
  139.                 ChangeBlock(1);                         //画新的方块
  140.                 ChangeSmallBlock(1);                //画新的预览方块
  141.                 ucBlockNow=ucBlockNext;
  142.                 ucResult=1;
  143.         }
  144.         else
  145.         {
  146.                 ucResult=0;
  147.         }

  148.         return ucResult;
  149. }
  150. /*******************************************************
  151. * 函 数 名:void TaskClear(void)
  152. * 功能描述:消除满行
  153. * 函数说明:消除满行,并出新
  154. * 输入参数:无
  155. * 返 回 值:无
  156. *******************************************************/
  157. void TaskClear(void)
  158. {
  159.         char i=0,j=0,n=0;
  160.         unsigned char m=0;
  161.         unsigned char k=0;
  162.         unsigned char flag_man=1,flag_kong=1,flag_clear=0;//满行、空行、待清除标志
  163.         unsigned char man[4],count=0;                                          //满行位置数组,满行数量
  164.         unsigned char kong=20;                                                          //空行位置
  165.         unsigned char oldkong=20;                                                  //用来储存旧的空行位置

  166.         unsigned char ucResult=0;

  167.         //标记出来满行及第一个空行
  168.         for(i=19;i>=0;i--)
  169.         {
  170.                 for(j=9;j>=0;j--)
  171.                 {
  172.                         if(ucxMap[j][i]==0)
  173.                         {
  174.                                 flag_man=0;
  175.                         }
  176.                         else
  177.                         {
  178.                                 flag_kong=0;
  179.                         }
  180.                 }
  181.                 if(flag_man==1)
  182.                 {
  183.                         flag_clear=1;//激活消除标记
  184.                         man[k]=i;//标记满行
  185.                         count++;
  186.                         k++;
  187.                 }
  188.                 else if(flag_kong==1)
  189.                 {
  190.                          kong=i;
  191.                         break;//检测到第一个空行,就退出循环
  192.                 }
  193.                 flag_man=1;//将满行标志复位
  194.                 flag_kong=1;
  195.         }

  196.         //计分
  197.         switch(count)
  198.         {
  199.                 case 0:
  200.                                 ucScore=ucScore+100;
  201.                                 break;
  202.                 case 1:
  203.                                 ucScore=ucScore+200;
  204.                                 break;
  205.                 case 2:
  206.                                 ucScore=ucScore+400;
  207.                                 break;
  208.                 case 3:
  209.                                 ucScore=ucScore+700;
  210.                                 break;
  211.                 case 4:
  212.                                 ucScore=ucScore+1100;
  213.                                 break;
  214.                 default:
  215.                                 break;
  216.         }

  217.         oldkong=kong;
  218.         //消除满行
  219.         if(flag_clear==1)
  220.         {
  221.                 while(count!=0)//至少有一个满行,才进行消除
  222.                 {
  223.                         //逐行下移
  224.                         for(m=0;m<man[count-1]-kong-1;m++)
  225.                         {
  226.                                 for(n=9;n>=0;n--)
  227.                                 {
  228.                                         ucxMap[n][man[count-1]-m]=ucxMap[n][man[count-1]-m-1];        
  229.                                 }
  230.                         }
  231.                         //最上行清空
  232.                         for(n=0;n<10;n++)
  233.                         {
  234.                                 ucxMap[n][kong+1]=0;        
  235.                         }

  236.                         kong++;
  237.                         count--;
  238.                 }
  239.                 ChangeMap(oldkong-1,man[0]); //写入当前Map,只写有改变的行,加快显示速度
  240.         }

  241.         //将分数写入屏幕
  242.         TaskScore();
  243.         //出新的方块
  244.         ucResult=TaskComingNew();
  245.         //如果失败,就进入失败界面
  246.         if(ucResult==0)
  247.         {
  248.                 uc_state=4;
  249.                 ucFlagScreenChange=1;
  250.         }
  251. }
复制代码

评分

参与人数 7黑币 +46 收起 理由
adminhh + 12 很给力!
分经济 + 5 太强了,膜拜,哈哈
周超然 + 5 绝世好帖!
stkh + 5 很给力!
踏步如云 + 5 很给力!
苏眉ZH + 2 赞一个!
zhaok2013 + 12 绝世好帖!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏35 分享淘帖 顶6 踩
回复

使用道具 举报

沙发
ID:130231 发表于 2017-3-17 19:41 | 只看该作者
太厉害了
回复

使用道具 举报

板凳
ID:162136 发表于 2017-3-18 09:45 | 只看该作者
这个要支持。正打算用汇编写这个
回复

使用道具 举报

地板
ID:143525 发表于 2017-5-4 10:18 | 只看该作者
谢谢分享!!!!1
回复

使用道具 举报

5#
ID:199694 发表于 2017-5-15 15:51 | 只看该作者
这个io口怎么改?能不能详细说下
回复

使用道具 举报

6#
ID:99130 发表于 2017-6-28 19:01 | 只看该作者
编译过不了,
回复

使用道具 举报

7#
ID:99130 发表于 2017-6-30 16:28 | 只看该作者
花屏有点严重啊 为什么图里的很清楚??怎么解决
回复

使用道具 举报

8#
ID:209954 发表于 2017-7-2 22:04 来自触屏版 | 只看该作者
越快乐越堕落 发表于 2017-6-30 16:28
花屏有点严重啊 为什么图里的很清楚??怎么解决

图里用的晶振是11兆的,如果你用的晶振是12兆的就要修改延时,否则可能会花屏。
回复

使用道具 举报

9#
ID:223554 发表于 2017-7-31 14:06 | 只看该作者
积分不够
回复

使用道具 举报

10#
ID:216536 发表于 2017-8-1 08:34 | 只看该作者
学习学习
回复

使用道具 举报

11#
ID:231984 发表于 2017-11-23 19:36 | 只看该作者
锁存器用的什么型号的呢?
回复

使用道具 举报

12#
ID:266764 发表于 2017-12-26 19:35 | 只看该作者
我用清翔的开发板,其他引脚都改好了,但是dula=P2^6和wela=P2^7是什么东西的引脚
回复

使用道具 举报

13#
ID:166341 发表于 2017-12-28 16:17 | 只看该作者
孤狼131 发表于 2017-12-26 19:35
我用清翔的开发板,其他引脚都改好了,但是dula=P2^6和wela=P2^7是什么东西的引脚

控制数码管的段选与位选锁存器的
回复

使用道具 举报

14#
ID:266764 发表于 2017-12-28 23:17 | 只看该作者
Jerry0925 发表于 2017-12-28 16:17
控制数码管的段选与位选锁存器的

那你知道这个锁存器是哪个型号的吗
回复

使用道具 举报

15#
ID:224487 发表于 2018-1-5 00:33 | 只看该作者
按照接线连好单片机,为什么进入界面后无法运行,只有一个边框。本人刚接触单片机,还有很多不懂
回复

使用道具 举报

16#
ID:196311 发表于 2018-1-7 21:34 | 只看该作者
楼主很强大,下载学习下算法,谢谢分享
回复

使用道具 举报

17#
ID:273035 发表于 2018-1-8 09:28 来自触屏版 | 只看该作者
学习学习
回复

使用道具 举报

18#
ID:228939 发表于 2018-1-15 21:13 | 只看该作者
坐等超级玛丽
回复

使用道具 举报

19#
ID:288471 发表于 2018-3-6 22:57 | 只看该作者
高手啊  不错!!
回复

使用道具 举报

20#
ID:306312 发表于 2018-4-12 08:20 | 只看该作者
感谢楼主的无私奉献,给了我很大的启发
回复

使用道具 举报

21#
ID:319242 发表于 2018-5-2 22:39 | 只看该作者
感谢分享代码,经测试能够运行,但是移植到串行12864模块上很难,单片机RAM太小,无法缓存所有屏幕点阵,而并口LCD则可以直接读取屏幕内容。
回复

使用道具 举报

22#
ID:59433 发表于 2018-5-2 23:29 | 只看该作者
给个元器件电路图咯  谢谢
回复

使用道具 举报

23#
ID:320508 发表于 2018-11-29 14:22 | 只看该作者
好帖收藏一波
回复

使用道具 举报

24#
ID:20468 发表于 2018-11-29 19:41 | 只看该作者
这是好资料
回复

使用道具 举报

25#
ID:445953 发表于 2018-12-13 20:31 | 只看该作者
厉害厉害,支持一波
回复

使用道具 举报

26#
ID:3802 发表于 2019-1-25 14:31 | 只看该作者
太厉害了
回复

使用道具 举报

27#
ID:479216 发表于 2019-2-22 11:00 来自触屏版 | 只看该作者
楼主有没有更详细的资料啊,要买啥,焊哪里
回复

使用道具 举报

28#
ID:479299 发表于 2019-2-23 01:33 | 只看该作者
mark!mark!mark!mark!
回复

使用道具 举报

29#
ID:414624 发表于 2019-4-10 16:24 | 只看该作者
DB0-DB7是P0口吗,那要上啦电阻输出吗
回复

使用道具 举报

30#
ID:533791 发表于 2019-5-12 00:27 来自触屏版 | 只看该作者
孤狼131 发表于 2017-12-26 19:35
我用清翔的开发板,其他引脚都改好了,但是dula=P2^6和wela=P2^7是什么东西的引脚

我用的也是清翔的开发板,想请问你这个子程序到底怎么弄啊。我萌新,这个复制过去编译都过不了
回复

使用道具 举报

31#
ID:538436 发表于 2019-5-14 22:42 | 只看该作者
萌新求一下资料 么有币。。
回复

使用道具 举报

32#
ID:634589 发表于 2019-11-2 23:07 | 只看该作者
哇!!!这个才是硬核!!!!
回复

使用道具 举报

33#
ID:625113 发表于 2019-11-26 09:29 来自触屏版 | 只看该作者
大佬大佬,来学习学习
回复

使用道具 举报

34#
ID:663957 发表于 2019-12-28 10:40 | 只看该作者
linlexiao 发表于 2019-5-12 00:27
我用的也是清翔的开发板,想请问你这个子程序到底怎么弄啊。我萌新,这个复制过去编译都过不了

所以到底怎么搞啊
回复

使用道具 举报

35#
ID:59884 发表于 2020-1-6 19:54 | 只看该作者
谢谢楼主分享!
回复

使用道具 举报

36#
ID:669226 发表于 2020-1-24 19:30 | 只看该作者
感谢分享!!!
回复

使用道具 举报

37#
ID:185815 发表于 2020-2-4 21:25 | 只看该作者
感谢分享了,困家里没事做,D一个
回复

使用道具 举报

38#
ID:614068 发表于 2020-2-5 18:23 | 只看该作者
谢谢楼主分享,这个算是大型程序了,值得我学习啊。
回复

使用道具 举报

39#
ID:624009 发表于 2020-2-10 21:44 | 只看该作者
谢谢分享,楼主太厉害了
回复

使用道具 举报

40#
ID:808771 发表于 2020-8-24 09:31 | 只看该作者
感谢楼主的分享,稍后自己也试试看
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表