演示程序之打游戏 -- 慕司板IAP15
2016-04-03 14:20
357 查看
上位机和协议制定我的大学舍友(他的微博:http://weibo.com/lesshst?topnav=1&wvr=5&topsug=1)毕业前百忙之中使用Python花了一个下午完成的,先对他表示谢意。
演示视频如下:
http://v.youku.com/v_show/id_XODExMjY3ODY0.html
下面介绍演示程序的使用步骤:
①把慕司板演示程序下载进“慕司板”中,右边的摇杆拨到上面。
②打开lizhibo文件夹下的config.ini文件,修改串口号和波特率。
③打开浏览器的魂斗罗游戏界面:http://www.4399.com/flash/59630_2.htm
或者超级玛丽: http://www.3366.com/flash/69976.shtml
![](http://img.blog.csdn.net/20160403140909929)
慕司板打游戏其实就是串口虚拟电脑键盘。
④打开http://www.cnblogs.com/longfan/articles/1841136.html
查看上图中WASDJKFH几个按键对应的键值
87 W 上 按键1
65 A 左 按键2
68 D 右 按键3
83 S 下(文中写错了)按键4
74 J 射击 按键6
75 K 跳跃 按键7
70 F 选择 按键5
72H 跳跃 按键10
如果需要双人游戏,再插上一个慕司板,修改config.ini的串口号,波特率,键值(上下左右?>)运行软件即可。
附上main.c代码:
interrup.c代码:
完整工程参见资源汇总帖.
原作于 2014年10月
CSDN发表于2016年4月
weifengdq
演示视频如下:
http://v.youku.com/v_show/id_XODExMjY3ODY0.html
下面介绍演示程序的使用步骤:
①把慕司板演示程序下载进“慕司板”中,右边的摇杆拨到上面。
②打开lizhibo文件夹下的config.ini文件,修改串口号和波特率。
③打开浏览器的魂斗罗游戏界面:http://www.4399.com/flash/59630_2.htm
或者超级玛丽: http://www.3366.com/flash/69976.shtml
慕司板打游戏其实就是串口虚拟电脑键盘。
④打开http://www.cnblogs.com/longfan/articles/1841136.html
查看上图中WASDJKFH几个按键对应的键值
87 W 上 按键1
65 A 左 按键2
68 D 右 按键3
83 S 下(文中写错了)按键4
74 J 射击 按键6
75 K 跳跃 按键7
70 F 选择 按键5
72H 跳跃 按键10
如果需要双人游戏,再插上一个慕司板,修改config.ini的串口号,波特率,键值(上下左右?>)运行软件即可。
附上main.c代码:
/************************************************************************************** 作 者:weifengdq(新浪微博:weifengdq) 注意事项:按ALT+F7 配置->Target->Memory Mode->Large:variables in XDATA,如果需要用到仿真, Debug中设置好仿真和串口号; 初始化有液晶背光调节演示,需要把P19的跳线帽插到上面,使LCD背光引脚与P26相连,使 用PWM1来调节液晶背光亮度; 使用串口1的1通道发送数据打游戏:在lkb.h中增加了KEYMSG结构体 mpu6050.c中Accel_x、Accel_y、Accel_z升为全局变量,增加; Accel_y = (float)GetData(ACCEL_YOUT_H)/286.0f; //读取Y轴加速度 ADC初始化把AD采样时间改小了 修改日期:2014/10/17 *************************************************************************************/ #include "config0.h" #include "delay.h" #include "lkb.h" #include "timer.h" #include "uart.h" #include <stdio.h> #include "lcd.h" #include "adc.h" #include "mpu6050.h" #include "image0.h" #include "pca_pwm.h" #include "hmc5883l.h" #include "nrf24l01.h" #include <math.h> #include "int.h" unsigned int adc0,adc1,adc2,adc3,adc4; float li_voltage=0; KEYMSG left; KEYMSG right; KEYMSG up; KEYMSG down; KEYMSG kb; KEYMSG a_jump; //跳跃 KEYMSG a_shoot;//射击 KEYMSG a_left; KEYMSG a_right; void Encoder_Task(); void MPU6050_Task(); void HMC5883_Task(); void KEY_Task(); void ADC_Task(); void Encoder_Task() { LCD_Show2Num(0,112,Freq,6); printf("%d\n",Cnt_Now); } void MPU6050_Task() { static unsigned int j=0; Angle_Calcu(); //倾角计算 将得到下面两个全局变量,使用了网上摘抄的卡尔曼滤波 //float Angle; //最终倾斜角度 //float Gyro_y; //Y轴角速度 if(Accel_x >=20 && Accel_x<=90) { sendMsg(&a_jump, 7, 1); } else if(Accel_x >=-90 && Accel_x<=-20) { sendMsg(&a_shoot, 6, 1); } else { if(a_jump.msg == 1) { sendMsg(&a_jump, 7, 0); } else if(a_shoot.msg == 1) { sendMsg(&a_shoot, 6, 0); } } if(Accel_y >=20 && Accel_y<=90) { sendMsg(&a_right, 3, 1); } else if(Accel_y >=-90 && Accel_y<=-20) { sendMsg(&a_left, 2, 1); } else { if(a_right.msg == 1) { sendMsg(&a_right, 3, 0); } else if(a_left.msg == 1) { sendMsg(&a_left, 2, 0); } } //printf("%.2f,%.2f\n",Angle,Gyro_y); j++; if(j>=240) { j=0; LCD_Fill(0,140,239,318,Black); POINT_COLOR = Red; BACK_COLOR=Black; LCD_DrawLine(0,229,239,229); POINT_COLOR = Green; } if((int)Angle>-90 && (int)Angle<90 ) //排除初始不稳定杂波 { LCD_DrawPoint(j,229-(int)Angle); //通过NRF24L01发送出去 if(Angle>=0) { Tx_Buf[2] = '+'; Tx_Buf[3] =(unsigned char)Angle; } else { Tx_Buf[2] = '-'; Tx_Buf[3] = (unsigned char)(Angle*(-1)); } Transmit(Tx_Buf); //24L01发送缓冲数组 sta=SPI_Read(READ_REG + STATUS); //读24L01的状态寄存器 if(TX_DS) //sbit TX_DS =sta^5;数据发送完成中断,接收到应答信号时此位置1,写1清除中断 { //Delay_ms(5); SPI_RW_Reg(WRITE_REG + STATUS,sta); //写状态寄存器,清除中断 } if(MAX_RT) //如果是发送超时 sbit MAX_RT =sta^4; { //达到最多次重发中断,MAX_RT中断产生则必须清除后系统才能进行通讯,写1清除中断 //Delay_ms(5); SPI_RW_Reg(WRITE_REG + STATUS,sta); } } } void HMC5883_Task() { //static unsigned int m=0; static float Last_HMC5883_Angle=0; // float Avg_HMC5883_Angle=0; // float Last_Avg_HMC5883_Angle=0; // float temp_HMC5883_Angle=0; Multiple_Read_HMC5883(); //角度的变化似乎并不是线性的 LCD_Show2Num(32,80,(int)HMC5883_Angle,3); POINT_COLOR=BACK_COLOR; //三角函数使用弧度而不是度! LCD_DrawLine(120,100,120+(int)(28.0*cos((Last_HMC5883_Angle-180.0)/57.3)),100+(int)(28.0*sin((Last_HMC5883_Angle-180.0)/57.3))); POINT_COLOR=Yellow; LCD_DrawLine(120,100,120+(int)(28.0*cos((HMC5883_Angle-180.0)/57.3)),100+(int)(28.0*sin((HMC5883_Angle-180.0)/57.3))); Last_HMC5883_Angle=HMC5883_Angle; // //平均值滤波 // m++; // temp_HMC5883_Angle+=HMC5883_Angle; // if(m==5) // { // m=0; // Avg_HMC5883_Angle=temp_HMC5883_Angle/5.0; //与前面四个值相关联 // //printf("\n\n%.2f\n\n",HMC5883_Angle); //HMC5883_Angle: float型,0~360 // LCD_Show2Num(32,80,(int)Avg_HMC5883_Angle,3); // POINT_COLOR=BACK_COLOR; // LCD_DrawLine(120,100,120+28*cos((Last_Avg_HMC5883_Angle-180.0)/57.3),100+28*sin((Last_Avg_HMC5883_Angle-180.0)/57.3)); // POINT_COLOR=Yellow; // LCD_DrawLine(120,100,120+28*cos((Avg_HMC5883_Angle-180.0)/57.3),100+28*sin((Avg_HMC5883_Angle-180.0)/57.3)); // Last_Avg_HMC5883_Angle=Avg_HMC5883_Angle; // } } void ADC_Task() { static unsigned int i=0; static unsigned int k=0; POINT_COLOR= Green; i++; switch(i) { case 1: adc0=(int)ADC_GetResult(0);LCD_Show2Num(48,0,adc0,3); break; case 2: adc1=(int)ADC_GetResult(1);LCD_Show2Num(48,16,adc1,3); break; case 3: adc2=(int)ADC_GetResult(2);LCD_Show2Num(168,0,adc2,3); break; case 4: adc3=(int)ADC_GetResult(3);LCD_Show2Num(168,16,adc3,3); break; case 5: li_voltage=(float)ADC_GetResult(4)*0.019f; break; //adc4*3.3/256*(6.8+3.3)/6.8=adc4*0.019 default: i=0; LCD_Show2Num(96,32,(int)li_voltage,1); //显示整数部分 LCD_Show2Num(112,32,((int)(li_voltage*100))%100,2); //显示小数部分 //printf("%d,%d,%d,%d,%.2f\n",adc0,adc1,adc2,adc3,li_voltage); }//switch 结束 k++; //printf("%d,%d,", adc1, adc0); if(k%2) { if(adc1>=0 && adc1<=80) { sendMsg(&up, 1, 1); } else if(adc1>=170 && adc1<=255) { sendMsg(&down, 4, 1); } else { if(up.msg == 1) { sendMsg(&up, 1, 0); } else if(down.msg == 1) { sendMsg(&down, 4, 0); } } } else { if(adc0>=0 && adc0<=80) { sendMsg(&left, 3, 1); } else if(adc0>=170 && adc0<=255) { sendMsg(&right, 2, 1); } else { if(left.msg == 1) { sendMsg(&left, 3, 0); } else if(right.msg == 1) { sendMsg(&right, 2, 0); } } } } //void KEY_Task() //{ // if (key_up==0) // { // sendMsg(&kb, Key_Num, 1); //sendMsg()函数里面有printf // } // else //if(key_up == 1) // { // // if (kb.msg == 1) ; // sendMsg(&kb, kb.key, 0); // } // //printf("%d\n",Key_Num); //} /*可以使用单步仿真一步步查看效果*/ void main() { unsigned int i; Delay_ms(10); Delay_us(10); PWM2=0; //电机制动,可以下拉10k设置引脚为推挽制动 /*2.2寸、320*240分辨率、SPI接口的LCD初始化,使用STC的硬件SPI,LCD驱动芯片为ILI9340C*/ LCD_LED=0; //LCD背光最亮 SPI_Init(); //IAP15硬件SPI初始化,切换到了SPI的第二通道:SS/P24、MOSI/P23、MISO/P22、SCK/P21 LCD_Init(); //LCD初始化,厂家提供 /*LCD休眠演示*/ LCD_Sleep(); //LCD进入休眠状态 Delay_ms(100); LCD_ExitSleep();//LCD退出休眠状态 /*LCD刷屏演示:红橙黄绿青蓝紫 白黑*/ LCD_Clear(Red); LCD_Clear(Orange); LCD_Clear(Yellow); LCD_Clear(Green); LCD_Clear(Cyan); LCD_Clear(Blue); LCD_Clear(Purple); LCD_Clear(White); LCD_Clear(Black); /*LCD画线、英文、汉字、数字显示测试*/ /*液晶屏左上角坐标为(0,0),右下角坐标为(239,319)*/ POINT_COLOR=Green; //全局变量 BACK_COLOR=Black; //前景色类似于在黑板写字的各种颜色的粉笔,黑板的颜色就是背景色 LCD_ShowString(0,0,"Hello,world!"); //从(0,0)坐标开始显示字符串“Hello,World!” POINT_COLOR=Blue; //可以随时更改画笔颜色,显示不同颜色的数字 LCD_DrawLine(100,0,240,15); //画一条线段,起始坐标(100,0),终点坐标(240,15) LCD_Show2Num(0,16,65535,5); //从(0,16)坐标开始显示0~65535的数字,5为数字位数,即显示5位数字 LCD_ShowNum(100,16,987654321,9); //从(100,16)坐标开始显示unsigned long型数字(0~4294967295),10为数字位数,即10位数字 POINT_COLOR=Yellow; for(i=32;i<320;i+=16) { LCD_ShowString(0,i,"abcdefghijklmnopqrstuvwxyz!"); } LCD_LED=1; //LCD背光关闭 Delay_ms(500); LCD_LED=0; //LCD背光最亮 Delay_ms(500); LCD_Clear(Black); //清屏:黑色 //LCD_Fill(120,120,160,160,Yellow); //在矩形区域填充色块,在函数处按F12可以查看函数原型 LCD_ShowImage(gImage_musi,80,120,78,80);//显示彩色图像: 慕司LOGO Delay_ms(2000); LCD_Clear(Black); POINT_COLOR=Red; LCD_DrawRectangle(0,0,239,60); //画矩形:对角坐标(左上、右下)为(0,0),(239,60) POINT_COLOR=Purple; LCD_ShowString(80,2,"慕司"); //显示16*16汉字 LCD_ShowString(16,22,"为一线微茫憧憬一意孤行,"); LCD_ShowString(8,42,"这众生芸芸谁不曾如此任性!"); POINT_COLOR=Cyan; LCD_ShowGB3232(80,160,"青春"); POINT_COLOR=Green; LCD_DrawCircle(112,176,48); Delay_ms(500); PCA_PWM_Init(); //PWM初始化,切换到PWM的第三通道P25/P26/P27,这里仅使用了PWM1(P26,即LCD_LED引脚) LCD_LED=0; //LCD背光最亮 for(i=255;i<=0;i--) //液晶背光256级亮度调节演示:由亮到暗 { CCAP1H = CCAP1L = i; //通过调节占空比来调节屏幕亮度 Delay_ms(20); } LCD_LED=1; //LCD背光关闭 for(i=0;i<255;i++) //液晶背光256级亮度调节演示:由暗到亮 { CCAP1H = CCAP1L = i; //通过调节占空比来调节屏幕亮度 Delay_ms(20); } Delay_ms(1000); LCD_Clear(Black); LKB_Init(); //LED Key Buzzer:LED、按键、蜂鸣器 初始化(设置蜂鸣器端口为推挽输出) Buzzer_Func(500,1000); //蜂鸣器500Hz@1000ms ADC_Init(); //AD初始化:设置P1的0 1 2 3(摇杆)4(锂电池电压)口为AD口, POINT_COLOR = Green; //液晶屏画笔颜色 BACK_COLOR=Black; //背景色 LCD_ShowString(0,0 ,"adc0: "); LCD_ShowString(0,16,"adc1: "); LCD_ShowString(120,0,"adc2: "); LCD_ShowString(120,16,"adc3: "); LCD_ShowString(0,32,"li_voltage: V"); LCD_ShowString(104,32,"."); NRF24L01_Init(); TX_Mode(); //发送模式 LCD_ShowString(0,48,"NRF24L01_State: Tx_Mode"); MPU6050_Init(); Delay_ms(100); POINT_COLOR = Red; LCD_DrawLine(0,319,239,319); LCD_DrawLine(0,139,239,139); LCD_DrawLine(0,229,239,229); HMC5883_Init(); Delay_ms(100); POINT_COLOR = Green; LCD_DrawCircle(120,100,31); LCD_ShowString(0,64,"direction:"); LCD_ShowString(0,96,"speed:"); Timer_Init(100); //每10ms中断一次,参数为百us:100*100us = 10ms UART_Init(115200); //串口初始化,使用了串口1的第一通道,设置波特率115200,用于仿真时可以切换到第二通道P36/P37 TI=1; SBUF=0; //这一句使用printf有时是必须的 INT_Init(); while(1) { if(LED_flag) //每1s切换一次LED的状态 {LED_flag=0;LED1 = ~LED1;} if(ADC_flag) {ADC_flag=0;ADC_Task();} if(adc2 >=80 && Encoder_flag) //使用adc2作开关 { Encoder_flag=0; CCAP2H = CCAP2L = (adc3 - 10)*2; //使用摇杆的ADC3调节电机转速 Encoder_Task(); } else if(adc2 <=70) { if(MPU6050_flag) //MPU6050的几个函数使用了Delay_us(5) {MPU6050_flag=0;MPU6050_Task();} if(HMC5883_flag) {HMC5883_flag=0;HMC5883_Task();} //interrupt.c中把Key_Scan()改成支持连按 //uart.c中把串口1切换回P30/P31 if(KEY_flag) { // KEY_Task(); if (key_up==0) { sendMsg(&kb, Key_Num, 1); //sendMsg()函数里面有printf } else //if(key_up == 1) { KEY_flag=0; if (kb.msg == 1) ; sendMsg(&kb, kb.key, 0); } //printf("%d\n",Key_Num); } } else {;} } }
interrup.c代码:
#include "config0.h" #include "lkb.h" bit LED_flag=0; bit BEEP_flag=0; bit KEY_flag=0; bit ADC_flag=0; bit MPU6050_flag=0; bit HMC5883_flag=0; bit Encoder_flag=0; bit Rotate_Dir=0; //用于判断编码器旋转方向 int Cnt_Last; //记录上一次的捕获值 int Cnt_Now=500; //记录本次的捕获值 unsigned int Freq=0; //仅作参考,并未使用 void exint0() interrupt 0 using 2 //INT0中断入口 { //先判断转向 if(B_Phase) //如果B相脉冲为正电压,表示正转 { Rotate_Dir=1; Cnt_Now++; if(Cnt_Now>=1000) Cnt_Now=0; Cnt_Last=Cnt_Now; } else { Rotate_Dir=0; Cnt_Now--; if(Cnt_Now<=0) Cnt_Now=1000; } } //外部中断2服务程序 //void exint2() interrupt 10 //INT2中断入口 (下降沿) //{ // //先判断转向 // if(B_Phase) //如果B相脉冲为正电压,表示正转 // { // Rotate_Dir=1; // Cnt_Now++; // if(Cnt_Now>=1000) Cnt_Now=0; // Cnt_Last=Cnt_Now; // } // else // { // Rotate_Dir=0; // Cnt_Now--; // if(Cnt_Now<=0) Cnt_Now=1000; // } // //// INT_CLKO &= 0xEF; //若需要手动清除中断标志,可先关闭中断,此时系统会自动清除内部的中断标志 //// INT_CLKO |= 0x10; //然后再开中断即可 //} /********************************************* * Timer0中断服务程序 *********************************************/ void tm0_isr() interrupt 1 using 1 //10ms中断一次 { static unsigned int i=0; //用于LED static unsigned int j=0; //用于蜂鸣器 static unsigned int k=0; //用于MPU6050 // static unsigned int m=0; //用于HMC5883L static unsigned int n=0; //用于ADC static unsigned int p=0; //用于编码器 //使用Aslong的JGA25-371直流减速电机:334线编码器,减速比为 21.3,12V额定电压,额定转速201rpm //那么额定转速下10ms输出脉冲数:201*21.3*334/60/100=238.3257个脉冲 unsigned char ch,cl; static unsigned int temp=0; static unsigned int temp_1=0; //上次的值 cl=TL1; //先读低位(高位变得没那么快) ch=TH1; temp_1=temp; temp=ch*256+cl; //用左移怎么实现? ch<<8+cl //if(temp>=temp_1) Freq=(temp-temp_1)/5; // *200/1000 kHz //20kHz 每5ms 计100个数 //else Freq=(65536-temp_1 + temp)/5; if(temp>=temp_1) Freq=(temp-temp_1)*100; //1s的脉冲数,即频率 else Freq=(65536-temp_1 + temp)*100; p++; if(p>=1) { p=0; Encoder_flag=1; } i++; if(i>=100) { i=0; LED_flag=1; } Key_Scan(1,1); //支持连按;按键消抖时间,10ms if(Key_Num) { //if(Key_Num==10) BEEP_flag=1; //按键10按下,蜂鸣器鸣响 KEY_flag=1; } if(BEEP_flag) { j++; BEEP=~BEEP; //10ms翻转一次,50Hz if(j>10) //响100ms { j=0; BEEP_flag=0; BEEP=0; //关闭蜂鸣器 } } k++; if(k%2) //10ms采样一次 { //k=0; MPU6050_flag=1; } else HMC5883_flag=1; // m++; // if(m>=1) // { // m=0; // HMC5883_flag=1; // } n++; if(n>=1) //ad: 10ms采样一次 { n=0; ADC_flag=1; } }
完整工程参见资源汇总帖.
原作于 2014年10月
CSDN发表于2016年4月
weifengdq
相关文章推荐
- 多分类问题中混淆矩阵(Confusion Matrix)的Matlab画法
- 【java开发系列】— JDOM创建、改动、删除、读取XML文件
- java项目日志
- Cocos2d-x 3.9教程:1.场景的切换
- C++ Standard-Library Random Numbers
- 三星Mega 6.3(i9200)删除kingroot
- 在线编程--用递归函数和栈操作逆序栈
- 适配器模式
- sourceinsight显示中文乱码解决
- 物联网用到自己的新家里:改造家里的灯光照明系统
- eclipse xdebug调试
- service builder不会自动建表/更新字段
- 【简单DP】 数字三角形
- maven之Transitive dependencies(默认树的先序遍历算法处理依赖冲突)
- Fibonacci数——求第n个斐波拉契数
- 美女时钟+电子时钟
- linux下日期时间自动同步设置(rdate,ntpdate两种方法)
- WindowListener
- 二进制字符串转化为十进制数字,感觉明明是在范围内却报异常了
- 1660:小邪的烦恼(贪心)