![背面图.png 背面图.png](https://static.assets-stash.eet-china.com/forum/201906/04/121905ws6z6fm8nc28s68z.png)
整个贪吃蛇小项目分为两部分:第一部分是对uFun开发板的编程,另一部分是在电脑上做个简单的贪吃蛇游戏界面。先说一下uFun开发板的编程。
出于娱乐精神,看到开发板背面还有些其它资源,就忍不住为“手柄”添加了一些花里胡哨的东西,最后整个“手柄”要实现以下的功能:
- 通过串口和电脑收发数据,控制贪吃蛇行进方向。
- 每有按键按下时,RGB LED灯就会亮起,松开灯灭。
- 当贪吃蛇吃到食物时,蜂鸣器会响一下。
关于触摸按键位置在原理图中的对应关系以及在本程序中的设定如下图所示。
![触摸按键.png 触摸按键.png](https://static.assets-stash.eet-china.com/forum/201906/04/123637tq3oj3eumnv5m5nn.png)
还是通过STM32CubeMX生成初始化代码,查看板子原理图,需要用到引脚及功能如下列举:
- 触摸按键:向右-PC5; 向上-PC4; 向左-PB3; 向下-PB4。配置为浮空输入模式,查询到高电平则有按键按下。(也可以开启外部中断,边沿检测更为准确)
- RGB灯:对应TIM2的1、2、3通道(TIM2-CH1、TIM2-CH2、TIM2-CH3),低电平灯亮。均配置为PWM输出模式。
- 蜂鸣器:对应TIM1的通道一(TIM1_CH1),高电平导通。配置为PWM输出模式。
- 串口通信:micro-USB接口使用的是USART1,配置为波特率115200,8位数据位,1位停止位,无奇偶校验位。打开接收中断,主要是为了当吃到食物时电脑会给板子发送信息,从而控制蜂鸣器发声。
- 先定义一个标志变量flag,按键没有按下,flag=0;按键按下时,flag=1。该变量主要是为了防止在主循环内按下一次,误判断为按下多次。
- 当判断有按键按下时,要通过串口向电脑发送约定好的数据格式。
- 由于电脑发来的数据在串口接收中断里处理了,并且开启蜂鸣器也在中断里做了,因此主函数要做的只是判断蜂鸣器响没响,响了的话就过段时间把蜂鸣器关了,必竟一直想也太吵了。
int main(void){ /* USER CODE BEGIN 1 */ uint8_t upFlag = 0; // 用来标识四个按键是否被按下 uint8_t downFlag = 0; uint8_t leftFlag = 0; uint8_t rightFlag = 0; uint16_t beepCount = 0; // 用来计数蜂鸣器鸣响在主循环的次数 /* USER CODE END 1 */ /* MCU Configuration--------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* Configure the system clock */ SystemClock_Config(); /* Initialize all configured peripherals */ MX_GPIO_Init(); MX_TIM1_Init(); MX_USART1_UART_Init(); MX_TIM2_Init(); /* USER CODE BEGIN 2 */ // 初始化串口接收相关变量,开启串口接收中断 rcvIndex = 0; // 清空接收帧的内容 frameStart = 0; // 没有接收到一帧数据 HAL_UART_Receive_IT(&huart1, &ch, 1); // 开启接收中断 // 打开RGB三个引脚的PWM输出 HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_2); HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_3); /* USER CODE END 2 */ /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { // up if(HAL_GPIO_ReadPin(KEY_UP_GPIO_Port, KEY_UP_Pin) == 1 && upFlag == 0){ RGB_On(); SendString("<W>", 3); upFlag = 1; } if(HAL_GPIO_ReadPin(KEY_UP_GPIO_Port, KEY_UP_Pin) == 0 && upFlag == 1){ RGB_Off(); upFlag = 0; } // down if(HAL_GPIO_ReadPin(KEY_DOWN_GPIO_Port, KEY_DOWN_Pin) == 1 && downFlag == 0){ RGB_On(); SendString("<S>", 3); downFlag = 1; } if(HAL_GPIO_ReadPin(KEY_DOWN_GPIO_Port, KEY_DOWN_Pin) == 0 && downFlag == 1){ RGB_Off(); downFlag = 0; } // left if(HAL_GPIO_ReadPin(KEY_LEFT_GPIO_Port, KEY_LEFT_Pin) == 1 && leftFlag == 0){ RGB_On(); SendString("<A>", 3); leftFlag = 1; } if(HAL_GPIO_ReadPin(KEY_LEFT_GPIO_Port, KEY_LEFT_Pin) == 0 && leftFlag == 1){ RGB_Off(); leftFlag = 0; } // right if(HAL_GPIO_ReadPin(KEY_RIGHT_GPIO_Port, KEY_RIGHT_Pin) == 1 && rightFlag == 0){ RGB_On(); SendString("<D>", 3); rightFlag = 1; } if(HAL_GPIO_ReadPin(KEY_RIGHT_GPIO_Port, KEY_RIGHT_Pin) == 0 && rightFlag == 1){ RGB_Off(); rightFlag = 0; } // 控制蜂鸣器,判断蜂鸣器是否在响 if(isBeeping() == 1){ beepCount++; if(beepCount >= 0x1FFF){ Beep_Stop(); // 关闭蜂鸣器 beepCount = 0; } } /* USER CODE END WHILE */ } }
复制代码在串口通信中,我习惯自定义信息格式,每一个信息帧(信息包)都由如下格式组成:
<chars>
其中“<”和“>”是包头包尾,遇到包头才会真正接受信息,遇到包尾说明一帧接收结束,开始解析里面的信息。这次一共定义了5中信息:
- <A>:表示向左
- <S>:表示向下
- <D>:表示向右
- <W>:表示向上
- <F>:表示吃到了食物(food)
char rcvData[5]; // 存储串口接收到的自定义的一帧数据uint8_t rcvIndex; // 用来记录rcvData数组接收到数据的最高位下标 char ch; // 串口接收Buffer uint8_t frameStart; // 标识遇到自定义的数据包头,一帧数据开始
复制代码void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart){ if(ch == '<'){ rcvIndex = 0; frameStart = 1; } else if(frameStart == 1){ if(ch == '>'){ switch (rcvData[0]){ case 'F': Beep_Start(); break; default: break; } frameStart = 0; } else{ rcvData[rcvIndex] = ch; rcvIndex++; } } while(HAL_UART_Receive_IT(&huart1, &ch, 1) != HAL_OK); }
复制代码![手柄.gif 手柄.gif](https://static.assets-stash.eet-china.com/forum/201906/04/134414a0n0shvhusgouez8.gif)
STM32CubeMX+TrueSTUDIO的工程文件:
![](static/image/filetype/zip.gif)
继续阅读本篇相关更多标签
全部回复 3