找回密码
 立即注册

QQ登录

只需一步,快速开始

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

STM32打地鼠小游戏程序设计与源码说明

  [复制链接]
跳转到指定楼层
楼主
ID:361500 发表于 2018-12-13 09:12 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
打地鼠程序说明文档:
项目是一款打地鼠的游戏,程序用到了延时函数、LED、电阻屏、按键灯功能,这些功能的初始化例程里面都有,我们可以直接调用来完成初始化。
    完成了初始化就开始程序的主干了,既然我们用到了电阻屏,肯定就需要对它进行校准,校准函数在触摸屏例程也有,我们可以通过KEY0按键直接调用校准函数,用来防止触摸屏不准。
    然后我们先进行的是界面显示,先通过图片取模软件取出一个背景图,注意取模的时候由于例程显示那里标注了,在液晶上画图(仅支持:从左到右,从上到下 or 从上到下,从左到右 的扫描方式!),图片数据(必须包含图片信息头, 4096色/16位真彩色/18位真彩色/24位真彩色/32位真彩色的图像数据头),针对STM32,不能选择image2lcd的"高位在前(MSB First)"选项,否则imginfo的数据将不正确!我们按照标注选择了16位真彩色的取模方式,通过PS将图片改为屏幕合适尺寸,就可以在屏幕上显示出背景了。将前20行用来显示分数,等级,生命属性,他们通过字符串和数字变量的显示函数来实现, 后面300行做成3*3的地鼠洞,地鼠洞通过例程里面的画实心圆来表示,用一个for来遍历每个圆心坐标,让代码简介,在用个随机数函数产生0-8的数字来代表老鼠出现的位置,然后通过同样的方法取出老鼠的图片,根据洞圆心的坐标,找到老鼠显示的坐标显示出老鼠,这里背景是指显示一次的,老鼠变换一次刷新一次屏幕,刷新就只用刷新你目前的状态,洞的显示和老鼠的位置,让洞把老鼠覆盖就可以不用刷新背景了。再游戏结束之后再重新刷新背景显示游戏结束和继续就好了。
    显示完成了接下来就是触摸,由于它的例程里面自带触摸函数,可以直接调用来返回当前的x,y坐标,这样触摸就很容易了
    key_9=(tp_dev.x[0]+1)/80+3*(tp_dev.y[0]+1-20)/100;
    用以上这段代码就可以识别出触摸的是之前分的3*3的那个位置,返回的1-9,然后再和老鼠对应的坐标进行判断就可以知道打对与否,对就加分,分到就升级,升级了就提高地鼠刷新次数,错就减分,减生命,分为0时就不减了,生命为0就游戏结束。还有不管对错,只要点击了屏幕地鼠的持续时间就会减短,马上刷新下一只地鼠。就不用多余的等待。

单片机源程序如下:
  1. #include "led.h"
  2. #include "delay.h"
  3. #include "sys.h"
  4. #include "usart.h"
  5. #include "lcd.h"
  6. #include "key.h"  
  7. #include "24cxx.h"
  8. #include "myiic.h"
  9. #include "touch.h"
  10. #include "stdlib.h"

  11. #ifndef __IMAGE2LCD_H
  12. #define __IMAGE2LCD_H
  13. #include "sys.h"

  14. __packed typedef struct _HEADCOLOR
  15. {
  16.    unsigned char scan;
  17.    unsigned char gray;
  18.    unsigned short w;
  19.    unsigned short h;
  20.    unsigned char is565;
  21.    unsigned char rgb;
  22. }HEADCOLOR;  
  23. void image_display(u16 x,u16 y,u8 * imgx);//在指定位置显示图片
  24. void image_show(u16 xsta,u16 ysta,u16 xend,u16 yend,u8 scan,u8 *p);//在指定区域开始显示图片
  25. u16 image_getcolor(u8 mode,u8 *str);//获取颜色
  26. extern const u8 gImage_1[];
  27. extern const u8 gImage_2[];
  28. #endif



  29. //ALIENTEK Mini STM32开发板范例代码21
  30. //触摸屏实验  
  31. //技术支持:www.openedv.com
  32. //广州市星翼电子科技有限公司

  33. void Load_Drow_Dialog(void)
  34. {
  35.         LCD_Clear(WHITE);//清屏   
  36.          //POINT_COLOR=BLUE;//设置字体为蓝色
  37.         //LCD_ShowString(lcddev.width-24,0,200,16,16,"RST");//显示清屏区域
  38.           POINT_COLOR=RED;//设置画笔蓝色
  39. }
  40. ////////////////////////////////////////////////////////////////////////////////
  41. //电容触摸屏专有部分
  42. //画水平线
  43. //x0,y0:坐标
  44. //len:线长度
  45. //color:颜色
  46. void gui_draw_hline(u16 x0,u16 y0,u16 len,u16 color)
  47. {
  48.         if(len==0)return;
  49.         LCD_Fill(x0,y0,x0+len-1,y0,color);        
  50. }
  51. //画实心圆
  52. //x0,y0:坐标
  53. //r:半径
  54. //color:颜色
  55. void gui_fill_circle(u16 x0,u16 y0,u16 r,u16 color)
  56. {                                                                                          
  57.         u32 i;
  58.         u32 imax = ((u32)r*707)/1000+1;
  59.         u32 sqmax = (u32)r*(u32)r+(u32)r/2;
  60.         u32 x=r;
  61.         gui_draw_hline(x0-r,y0,2*r,color);
  62.         for (i=1;i<=imax;i++)
  63.         {
  64.                 if ((i*i+x*x)>sqmax)// draw lines from outside  
  65.                 {
  66.                          if (x>imax)
  67.                         {
  68.                                 gui_draw_hline (x0-i+1,y0+x,2*(i-1),color);
  69.                                 gui_draw_hline (x0-i+1,y0-x,2*(i-1),color);
  70.                         }
  71.                         x--;
  72.                 }
  73.                 // draw lines from inside (center)  
  74.                 gui_draw_hline(x0-x,y0+i,2*x,color);
  75.                 gui_draw_hline(x0-x,y0-i,2*x,color);
  76.         }
  77. }  
  78. //两个数之差的绝对值
  79. //x1,x2:需取差值的两个数
  80. //返回值:|x1-x2|
  81. u16 my_abs(u16 x1,u16 x2)
  82. {                        
  83.         if(x1>x2)return x1-x2;
  84.         else return x2-x1;
  85. }  
  86. //画一条粗线
  87. //(x1,y1),(x2,y2):线条的起始坐标
  88. //size:线条的粗细程度
  89. //color:线条的颜色
  90. void lcd_draw_bline(u16 x1, u16 y1, u16 x2, u16 y2,u8 size,u16 color)
  91. {
  92.         u16 t;
  93.         int xerr=0,yerr=0,delta_x,delta_y,distance;
  94.         int incx,incy,uRow,uCol;
  95.         if(x1<size|| x2<size||y1<size|| y2<size)return;
  96.         delta_x=x2-x1; //计算坐标增量
  97.         delta_y=y2-y1;
  98.         uRow=x1;
  99.         uCol=y1;
  100.         if(delta_x>0)incx=1; //设置单步方向
  101.         else if(delta_x==0)incx=0;//垂直线
  102.         else {incx=-1;delta_x=-delta_x;}
  103.         if(delta_y>0)incy=1;
  104.         else if(delta_y==0)incy=0;//水平线
  105.         else{incy=-1;delta_y=-delta_y;}
  106.         if( delta_x>delta_y)distance=delta_x; //选取基本增量坐标轴
  107.         else distance=delta_y;
  108.         for(t=0;t<=distance+1;t++ )//画线输出
  109.         {  
  110.                 gui_fill_circle(uRow,uCol,size,color);//画点
  111.                 xerr+=delta_x ;
  112.                 yerr+=delta_y ;
  113.                 if(xerr>distance)
  114.                 {
  115.                         xerr-=distance;
  116.                         uRow+=incx;
  117.                 }
  118.                 if(yerr>distance)
  119.                 {
  120.                         yerr-=distance;
  121.                         uCol+=incy;
  122.                 }
  123.         }  
  124. }   
  125. const u16 POINT_COLOR_TBL[CT_MAX_TOUCH]={RED,GREEN,BLUE,BROWN,GRED};
  126. ////////////////////////////////////////////////////////////////////////////////
  127. //5个触控点的颜色                                                                                                
  128. //电阻触摸屏测试函数
  129.         u8 flag_1=5;
  130. void rtp_test(void)
  131. {
  132.         u8 key;
  133.         u8 i=0;         
  134.         u8 z=0;        
  135.         u16 o=rand()%9;
  136.         u8 key_9=0;
  137.         u16 score=0;
  138.         u8 flag=1;
  139.         u8 flag_2=1;
  140.         u8 level=1;
  141. //        for(z=0;z<9;z++)
  142. //        {
  143. //                        gui_fill_circle((z%3)*80+40,(z/3)*100+70,40,BLUE);        
  144. //        }
  145. //        LCD_ShowString(0,0,200,16,16,"SCORE");
  146.         
  147.         while(1)
  148.         {
  149.                  key=KEY_Scan(0);
  150.                 tp_dev.scan(0);
  151.                 key_9=(tp_dev.x[0]+1)/80+3*(tp_dev.y[0]+1-20)/100;
  152.                 if(flag_1==5)
  153.                 {
  154.                         
  155.                         LCD_Clear(WHITE);
  156.                         image_display(0,0,(u8*)gImage_1);
  157.                         LCD_ShowString(0,0,200,16,16,"SCORE");
  158.                         LCD_ShowString(79,0,200,16,16,"LEVEL");
  159.                         LCD_ShowString(159,0,200,16,16,"LIFE");
  160.                         flag_1=0;
  161.                 }
  162.                 if(flag_1<=3)
  163.                 {
  164.                         if(i==0)
  165.                         {
  166.                                 LCD_ShowNum(189,0,4-flag_1,4,16);
  167.                                 LCD_ShowNum(29,0,score,4,16);
  168.                                 LCD_ShowNum(109,0,level,4,16);
  169.                                 
  170.                                 for(z=0;z<9;z++)
  171.                                 {
  172.                                 gui_fill_circle((z%3)*80+40,(z/3)*100+70,25,BRRED);        
  173.                                 }
  174.                                 delay_ms(10);
  175.                                 //LCD_ShowNum(0,0,o,4,16);
  176.                                 //LCD_ShowNum((o%3)*80+36,(o/3)*100+62,o,1,16);
  177.                                 image_display((o%3)*80-17+40,(o/3)*100-17+70,(u8*)gImage_2);
  178.                         }
  179.                         //gui_fill_circle(119,159,40,BLUE);
  180.                 if(tp_dev.sta&TP_PRES_DOWN)                        //触摸屏被按下
  181.                 {        
  182.                          if(tp_dev.x[0]<lcddev.width&&tp_dev.y[0]<lcddev.height)
  183.                         {        
  184. //                                if(tp_dev.x[0]>(lcddev.width-24)&&tp_dev.y[0]<16)Load_Drow_Dialog();//清除
  185. //                                else
  186. //                                {
  187. //                                        TP_Draw_Big_Point(tp_dev.x[0],tp_dev.y[0],POINT_COLOR_TBL[i%5]);
  188. //                                       
  189. //                                }                                        //画图         
  190.                                 //        key_9=(tp_dev.x[0]+1)/80+3*(tp_dev.y[0]+1-20)/100;
  191.                                 
  192.                                 if((o+1)==key_9&&flag)
  193.                                 {
  194.                                         score++;
  195.                                         flag=0;
  196.                                 }
  197.                                 else if(flag)
  198.                                 {
  199.                                         if(score>0)score--;
  200.                                         flag=0;
  201.                                         flag_1++;
  202.                                 }
  203.                                 
  204.                                         i=(int)(100/(level*0.75)-15);
  205.                                 
  206.                         }
  207.                 }else delay_ms(10);        //没有按键按下的时候            
  208.                 if(key==KEY0_PRES)        //KEY0按下,则执行校准程序
  209.                 {
  210.                         LCD_Clear(WHITE);//清屏
  211.                     TP_Adjust();  //屏幕校准
  212.                         TP_Save_Adjdata();         
  213.                         Load_Drow_Dialog();
  214.                 }
  215.                
  216.                 if(score>19){level++;score=0;}
  217.                 i++;
  218.                 if(i>(100/(level*0.75)))
  219.                 {
  220.                         o=rand()%9;
  221.                         LED0=!LED0;
  222.                         flag=1;
  223.                         i=0;
  224.                 }
  225.         }
  226.                 if(flag_1>3)
  227.                 {
  228.                          if(flag_2){tp_dev.x[0]=0;tp_dev.y[0]=0;flag_2=0;
  229.                                 LCD_Clear(WHITE);//清屏
  230.                                  image_display(0,0,(u8*)gImage_1);
  231.                                 if(tp_dev.touchtype!=0XFF)LCD_ShowString(90,130,200,16,16,"GAME OVER");
  232.                         if(tp_dev.touchtype!=0XFF)LCD_ShowString(110,160,200,16,16,"continue");
  233.                          }
  234.                         //LCD_Clear(WHITE);
  235.                                 if(key_9==5)
  236.                                 {
  237.                                                 flag_1=5;
  238.                                                 level=1;
  239.                                                 score=0;
  240.                                                 flag=1;
  241.                                                 flag_2=1;
  242.                                 }
  243.                 }
  244.         }
  245. }

  246. //电容触摸屏测试函数
  247. void ctp_test(void)
  248. {
  249.         u8 t=0;
  250.         u8 i=0;                     
  251.          u16 lastpos[5][2];                //最后一次的数据
  252.         while(1)
  253.         {
  254.                 tp_dev.scan(0);
  255.                 for(t=0;t<CT_MAX_TOUCH;t++)//最多5点触摸
  256.                 {
  257.                         if((tp_dev.sta)&(1<<t))//判断是否有点触摸?
  258.                         {
  259.                                 if(tp_dev.x[t]<lcddev.width&&tp_dev.y[t]<lcddev.height)//在LCD范围内
  260.                                 {
  261.                                         if(lastpos[t][0]==0XFFFF)
  262.                                         {
  263.                                                 lastpos[t][0] = tp_dev.x[t];
  264.                                                 lastpos[t][1] = tp_dev.y[t];
  265.                                         }
  266.                                         lcd_draw_bline(lastpos[t][0],lastpos[t][1],tp_dev.x[t],tp_dev.y[t],2,POINT_COLOR_TBL[t]);//画线
  267.                                         lastpos[t][0]=tp_dev.x[t];
  268.                                         lastpos[t][1]=tp_dev.y[t];
  269.                                         if(tp_dev.x[t]>(lcddev.width-24)&&tp_dev.y[t]<16)
  270.                                         {
  271.                                                 Load_Drow_Dialog();//清除
  272.                                         }
  273.                                 }
  274.                         }else lastpos[t][0]=0XFFFF;
  275.                 }
  276.                
  277.                 delay_ms(5);i++;
  278.                 if(i%20==0)LED0=!LED0;
  279.         }        
  280. }
  281. int main(void)
  282. {
  283.          delay_init();                     //延时函数初始化         
  284.         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
  285.         uart_init(9600);                 //串口初始化为9600
  286.         LED_Init();                                  //初始化与LED连接的硬件接口
  287.         LCD_Init();                                   //初始化LCD         
  288.         KEY_Init();                                //按键初始化                        
  289.         tp_dev.init();                        //触摸屏初始化
  290.         POINT_COLOR=RED;//设置字体为红色
  291.          
  292.         Load_Drow_Dialog();                 
  293.                 if(tp_dev.touchtype&0X80)ctp_test();        //电容屏测试
  294.                 else rtp_test();          //电阻屏测试
  295.                  
  296. }
复制代码

所有资料51hei提供下载(源码+文档):
打地鼠V1.0.7z (414.54 KB, 下载次数: 135)
说明文档.docx (15.47 KB, 下载次数: 99)


评分

参与人数 1黑币 +100 收起 理由
admin + 100 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

沙发
ID:452469 发表于 2018-12-23 10:07 | 只看该作者
请问在这个程序中,是怎样把打地鼠图片的加入到程序中的,两者之间是如何建立起关联的啊
回复

使用道具 举报

板凳
ID:361500 发表于 2018-12-26 15:15 | 只看该作者
嵌入式系统 发表于 2018-12-23 10:07
请问在这个程序中,是怎样把打地鼠图片的加入到程序中的,两者之间是如何建立起关联的啊

地鼠和背景图片都是用的取模软件,在工程里面有这个函数
回复

使用道具 举报

地板
ID:457316 发表于 2018-12-28 15:00 | 只看该作者
STM32战舰V3可以用吗
回复

使用道具 举报

5#
ID:202445 发表于 2019-9-29 10:03 | 只看该作者
试一下看行不行
回复

使用道具 举报

6#
ID:436894 发表于 2019-12-11 10:13 来自手机 | 只看该作者
为什么打不开
回复

使用道具 举报

7#
ID:736884 发表于 2020-4-26 13:46 | 只看该作者
楼主这个key_9=(tp_dev.x[0]+1)/80+3*(tp_dev.y[0]+1-20)/100;是如何计算的?
回复

使用道具 举报

8#
ID:736884 发表于 2020-4-26 13:48 | 只看该作者
楼主这个触摸点 key_9=(tp_dev.x[0]+1)/80+3*(tp_dev.y[0]+1-20)/100;是如何计算的?
回复

使用道具 举报

9#
ID:999593 发表于 2022-1-4 10:21 | 只看该作者
显示无法打开pdsc文件是什么原因
回复

使用道具 举报

10#
ID:262 发表于 2022-1-4 18:27 | 只看该作者
759699071 发表于 2022-1-4 10:21
显示无法打开pdsc文件是什么原因

用Keil5来编译  你看我编译的

51hei.png (59.64 KB, 下载次数: 17)

51hei.png
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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