0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

【开源小项目】基于STM32的OLED舵机菜单显示

嵌入式悦翔园 来源:CSDN-混分巨兽龙某某 2023-01-16 12:06 次阅读

一、前言

本文的OLED多级菜单UI为一个综合性的STM32小项目,使用多传感器与OLED显示屏实现智能终端的效果。项目中的多级菜单UI使用了较为常见的结构体索引法去实现功能与功能之间的来回切换,搭配DHT11,RTC,LED,KEY等器件实现高度智能化一体化操作。

后期自己打板设计结构,可以衍生为智能手表等小玩意。目前,项目属于裸机状态(CPU占用率100%),后期可能会加上RTOS系统。(本项目源码在本文末尾进行开源!)

二、硬件实物图

41ca6564-9550-11ed-bfe3-dac502259ad0.png

温度计:

41e1c31c-9550-11ed-bfe3-dac502259ad0.png41f51ee4-9550-11ed-bfe3-dac502259ad0.png

游戏机:

42731f6a-9550-11ed-bfe3-dac502259ad0.png429218a2-9550-11ed-bfe3-dac502259ad0.png

三、硬件引脚图

OLED模块:

VCC-->3.3V

GND-->GND

SCL-->PB10

SDA-->PB11

DHT11模块:

DATA-->PB9

VCC-->3.3V

GND-->GND

KEY模块(这部分笔者直接使用了正点原子精英板上的):

KEY0-->PE4

KEY1-->PE3

KEY_UP-->PA0

四、多级菜单

随着工业化和自动化的发展,如今基本上所有项目都离不开显示终端。而多级菜单更是终端显示项目中必不可少的组成因素,其实TFT-LCD屏幕上可以借鉴移植很多优秀的开源多级菜单(GUI,比如:LVGL),而0.96寸的OLED屏幕上通常需要自己去适配和编程多级菜单。

网上的普遍采用的多级菜单的方案是基于索引或者结构树,其中,索引法居多。索引法的优点:可阅读性好,拓展性也不错,查找的性能差不多是最优,就是有点占用内存空间。

4.1 索引法多级菜单实现

网上关于索引法实现多级菜单功能有很多基础教程,笔者就按照本项目中的具体实现代码过程给大家讲解一下索引法实现多级菜单。特别说明:本项目直接使用了正点原子的精英板作为核心板,所以读者朋友复现代码还是很简单的。

首先,基于索引法实现多级菜单的首要条件是先确定项目中将使用到几个功能按键(比如:向前,向后,确定,退出等等)本项目中,笔者使用到了3个按键:下一个(next),确定(enter),退出(back)。所以,接下首先定义一个结构体,结构体中一共有5个变量(3+2),分别为:当前索引序号(current),向下一个(next),确定(enter),退出(back),当前执行函数(void)。其中,标红的为需要设计的按键(笔者这里有3个),标绿的则为固定的索引号与该索引下需要执行的函数。

typedefstruct
{
u8current;//当前状态索引号
u8next;//向下一个
u8enter;//确定
6u8back;//退出
void(*current_operation)(void);//当前状态应该执行的操作
}Menu_table;

接下来就是定义一个数组去决定整个项目菜单的逻辑顺序(利用索引号)

Menu_tabletable[30]=
{
{0,0,1,0,(*home)},//一级界面(主页面)索引,向下一个,确定,退出

{1,2,5,0,(*Temperature)},//二级界面温湿度
{2,3,6,0,(*Palygame)},//二级界面游戏
{3,4,7,0,(*Setting)},//二级界面设置
{4,1,8,0,(*Info)},//二级界面信息

{5,5,5,1,(*TestTemperature)},//三级界面:DHT11测量温湿度
{6,6,6,2,(*ControlGame)},//三级界面:谷歌小恐龙Dinogame
{7,7,9,3,(*Set)},//三级界面:设置普通外设状态LED
{8,8,8,4,(*Information)},//三级界面:作者和相关项目信息

{9,9,7,3,(*LED)},//LED控制
};

这里解释一下这个数组中各元素的意义,由于我们在前面先定义了Menu_table结构体,结构体成员变量分别与数组中元素对应。比如:{0,0,1,0,(*home)},代表了索引号为0,按向下键(next)转入索引号为0,按确定键(enter)转入索引号为1,按退出键(back)转入索引号为0,索引号为0时执行home函数。

在举一个例子帮助大家理解一下,比如,我们当前程序处在索引号为2(游戏界面),就会执行Playgame函数。此时,如果按下next按键,程序当前索引号就会变为3,并且执行索引号为3时候的Setting函数。如果按下enter按键,程序当前索引号就会变为6,并且执行索引号为6时候的ControlGame函数。如果按下back按键,程序当前索引号就会变为0,并且执行索引号为0时候的home函数。

再接下就是按键处理函数:

uint8_tfunc_index=0;//主程序此时所在程序的索引值

voidMenu_key_set(void)
{
if((KEY_Scan(1)==1)&&(func_index!=6))//屏蔽掉索引6下的情况,适配游戏
{
func_index=table[func_index].next;//按键next按下后的索引号
OLED_Clear();
}

if((KEY_Scan(1)==2)&&(func_index!=6))
{
func_index=table[func_index].enter;//按键enter按下后的索引号
OLED_Clear();
}

if(KEY_Scan(1)==3)
{
func_index=table[func_index].back;//按键back按下后的索引号
OLED_Clear();
}

current_operation_index=table[func_index].current_operation;//执行当前索引号所对应的功能函数
(*current_operation_index)();//执行当前操作函数
}


//按键函数
u8KEY_Scan(u8mode)
{
staticu8key_up=1;
if(mode)key_up=1;
if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
{
HAL_Delay(100);//消抖
key_up=0;
if(KEY0==0)return1;
elseif(KEY1==0)return2;
elseif(WK_UP==1)return3;
}elseif(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1;
return0;
}

说明2点:

(1)由于是目前本项目是裸机状态下运行的,所以CPU占用率默认是100%的,所以这里使用按键支持连按时,对于菜单的切换更好些。

(2)可能部分索引号下的执行函数,需要使用到已经定义的3个按键(比如,本项目中的DInogame中)。所以,可以在需要差别化的索引号下去屏蔽原先的按键功能。如下:

if((KEY_Scan(1)==1)&&(func_index!=6))//屏蔽掉索引6下的情况,适配游戏
{
func_index=table[func_index].next;//按键next按下后的索引号
OLED_Clear();
}

if((KEY_Scan(1)==2)&&(func_index!=6))//屏蔽掉索引6下的情况,适配游戏
{
func_index=table[func_index].enter;//按键enter按下后的索引号
OLED_Clear();
}

(3)笔者这里是使用全屏刷新去切换功能界面,同时,没有启用高级算法去加速显示,所以可能在切换界面的时候效果一般。读者朋友可以试试根据自己的UI情况使用局部刷新,这样可能项目会更加丝滑一点。

本项目中的菜单索引图:

435e37ac-9550-11ed-bfe3-dac502259ad0.png

4.2 内部功能实现(简化智能手表)

OLED就是正常的驱动与显示,有能力的读者朋友可以使用高级算法去加速OLED屏幕的刷新率,可以使自己的多级菜单切换起来更丝滑。

唯一需要注意的点就是需要去制作菜单里面的UI图标(注意图片大小是否合适):

436ae556-9550-11ed-bfe3-dac502259ad0.png

如果是黑白图片的话,可以直接使用PCtoLCD2002完美版进行取模:

43bb502c-9550-11ed-bfe3-dac502259ad0.png

4.3 KEY按键

KEY按键注意消抖(建议裸机情况下支持连续按动),同时注意自己实际硬件情况去进行编程(电阻是否存在上拉或者下拉)。

43c9c5d0-9550-11ed-bfe3-dac502259ad0.png

4.4 DinoGame实现

43d6f1ec-9550-11ed-bfe3-dac502259ad0.png谷歌公司最近比较流行的小游戏,笔者之前有文章进行了STM32的成功复刻。博客地址:基于STM32的小游戏——谷歌小恐龙(Chrome Dino Game)_混分巨兽龙某某的博客-CSDN博客_谷歌恐龙

4.5 LED控制和DHT11模块

LED和DHT11模块其实都属于外设控制,这里读者朋友可以根据自己的实际情况去取舍。需要注意的是尽可能适配一下自己多级菜单(外设控制也需要注意一下按键安排,可以参考笔者项目的设计)。

五、CubeMX配置

1、RCC配置外部高速晶振(精度更高)——HSE;

43ef31bc-9550-11ed-bfe3-dac502259ad0.png

2、SYS配置:Debug设置成Serial Wire(否则可能导致芯片自锁);

441f8a24-9550-11ed-bfe3-dac502259ad0.png

3、I2C2配置:这里不直接使用CubeMX的I2C2,使用GPIO模拟(PB10:CLK;PB11:SDA)

442eb2d8-9550-11ed-bfe3-dac502259ad0.png

4、RTC配置:年月日,时分秒;

44404a0c-9550-11ed-bfe3-dac502259ad0.png446fb1f2-9550-11ed-bfe3-dac502259ad0.png

5、TIM2配置:由上面可知DHT11的使用需要us级的延迟函数,HAL库自带只有ms的,所以需要自己设计一个定时器

448305b8-9550-11ed-bfe3-dac502259ad0.png

6、KEY按键配置:PE3,PE4和PA0设置为端口输入(开发板原理图)

44b15170-9550-11ed-bfe3-dac502259ad0.png

7、时钟树配置:

44e1bf72-9550-11ed-bfe3-dac502259ad0.png

8、文件配置

44ee37f2-9550-11ed-bfe3-dac502259ad0.png

六、代码

6.1 OLED驱动代码

此部分OLED的基本驱动函数,笔者使用的是I2C驱动的0.96寸OLED屏幕。所以,首先需要使用GPIO模拟I2C通讯。随后,使用I2C通讯去驱动OLED。(此部分代码包含了屏幕驱动与基础显示,如果对OLED显示不太理解的朋友可以去看看上文提到的笔者的另一篇文章)

oled.h:

#ifndef__OLED_H
#define__OLED_H

#include"main.h"

#defineu8uint8_t
#defineu32uint32_t

#defineOLED_CMD0//写命令
#defineOLED_DATA1//写数据

#defineOLED0561_ADD0x78//OLEDI2C地址
#defineCOM0x00//OLED
#defineDAT0x40//OLED

#defineOLED_MODE0
#defineSIZE8
#defineXLevelL0x00
#defineXLevelH0x10
#defineMax_Column128
#defineMax_Row64
#defineBrightness0xFF
#defineX_WIDTH128
#defineY_WIDTH64


//-----------------OLEDIICGPIO进行模拟----------------

#defineOLED_SCLK_Clr()HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_RESET)//GPIO_ResetBits(GPIOB,GPIO_Pin_10)//SCL
#defineOLED_SCLK_Set()HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_SET)//GPIO_SetBits(GPIOB,GPIO_Pin_10)

#defineOLED_SDIN_Clr()HAL_GPIO_WritePin(GPIOB,GPIO_PIN_11,GPIO_PIN_RESET)//GPIO_ResetBits(GPIOB,GPIO_Pin_11)//SDA
#defineOLED_SDIN_Set()HAL_GPIO_WritePin(GPIOB,GPIO_PIN_11,GPIO_PIN_SET)//GPIO_SetBits(GPIOB,GPIO_Pin_11)


//I2CGPIO模拟
voidIIC_Start();
voidIIC_Stop();
voidIIC_WaitAck();
voidIIC_WriteByte(unsignedcharIIC_Byte);
voidIIC_WriteCommand(unsignedcharIIC_Command);
voidIIC_WriteData(unsignedcharIIC_Data);
voidOLED_WR_Byte(unsigneddat,unsignedcmd);


//功能函数
voidOLED_Init(void);
voidOLED_WR_Byte(unsigneddat,unsignedcmd);

voidOLED_FillPicture(unsignedcharfill_Data);
voidOLED_SetPos(unsignedcharx,unsignedchary);
voidOLED_DisplayOn(void);
voidOLED_DisplayOff(void);
voidOLED_Clear(void);
voidOLED_On(void);
voidOLED_ShowChar(u8x,u8y,u8chr,u8Char_Size);
u32oled_pow(u8m,u8n);
voidOLED_ShowNum(u8x,u8y,u32num,u8len,u8size2);
voidOLED_ShowString(u8x,u8y,u8*chr,u8Char_Size);

#endif

oled.c:

#include"oled.h"
#include"asc.h"//字库(可以自己制作)
#include"main.h"



/********************GPIO模拟I2C*******************/
//注意:这里没有直接使用HAL库中的模拟I2C
/**********************************************
//IICStart
**********************************************/
voidIIC_Start()
{

OLED_SCLK_Set();
OLED_SDIN_Set();
OLED_SDIN_Clr();
OLED_SCLK_Clr();
}

/**********************************************
//IICStop
**********************************************/
voidIIC_Stop()
{
OLED_SCLK_Set();
OLED_SDIN_Clr();
OLED_SDIN_Set();

}

voidIIC_WaitAck()
{
OLED_SCLK_Set();
OLED_SCLK_Clr();
}
/**********************************************
//IICWritebyte
**********************************************/

voidIIC_WriteByte(unsignedcharIIC_Byte)
{
unsignedchari;
unsignedcharm,da;
da=IIC_Byte;
OLED_SCLK_Clr();
for(i=0;i<8;i++)
 {
   m=da;
  // OLED_SCLK_Clr();
  m=m&0x80;
  if(m==0x80)
  {OLED_SDIN_Set();}
  else OLED_SDIN_Clr();
   da=da<<1;
  OLED_SCLK_Set();
  OLED_SCLK_Clr();
 }
 
 
}
/**********************************************
// IIC Write Command
**********************************************/
void IIC_WriteCommand(unsigned char IIC_Command)
{
   IIC_Start();
   IIC_WriteByte(0x78);            //Slave address,SA0=0
 IIC_WaitAck();
   IIC_WriteByte(0x00);   //write command
 IIC_WaitAck();
   IIC_WriteByte(IIC_Command);
 IIC_WaitAck();
   IIC_Stop();
}
/**********************************************
// IIC Write Data
**********************************************/
void IIC_WriteData(unsigned char IIC_Data)
{
   IIC_Start();
   IIC_WriteByte(0x78);   //D/C#=0; R/W#=0
 IIC_WaitAck();
   IIC_WriteByte(0x40);   //write data
 IIC_WaitAck();
   IIC_WriteByte(IIC_Data);
 IIC_WaitAck();
   IIC_Stop();
}
 
void OLED_WR_Byte(unsigned dat,unsigned cmd)
{
 if(cmd)
 {
  IIC_WriteData(dat);
 }
 else
 {
  IIC_WriteCommand(dat);
 }
}
 
void OLED_Init(void)
{
 HAL_Delay(100);  //这个延迟很重要
 
 OLED_WR_Byte(0xAE,OLED_CMD);//--display off
 OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
 OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
 OLED_WR_Byte(0x40,OLED_CMD);//--set start line address
 OLED_WR_Byte(0xB0,OLED_CMD);//--set page address
 OLED_WR_Byte(0x81,OLED_CMD); // contract control
 OLED_WR_Byte(0xFF,OLED_CMD);//--128
 OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap
 OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse
 OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
 OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty
 OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction
 OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset
 OLED_WR_Byte(0x00,OLED_CMD);//
 
 OLED_WR_Byte(0xD5,OLED_CMD);//set osc division
 OLED_WR_Byte(0x80,OLED_CMD);//
 
 OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off
 OLED_WR_Byte(0x05,OLED_CMD);//
 
 OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period
 OLED_WR_Byte(0xF1,OLED_CMD);//
 
 OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion
 OLED_WR_Byte(0x12,OLED_CMD);//
 
 OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh
 OLED_WR_Byte(0x30,OLED_CMD);//
 
 OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable
 OLED_WR_Byte(0x14,OLED_CMD);//
 
 OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel
 HAL_Delay(100); 
 OLED_FillPicture(0x0);
 
}
 
 
/********************************************
// OLED_FillPicture
********************************************/
void OLED_FillPicture(unsigned char fill_Data)
{
 unsigned char m,n;
 for(m=0;m<8;m++)
 {
  OLED_WR_Byte(0xb0+m,0);  //page0-page1
  OLED_WR_Byte(0x00,0);  //low column start address
  OLED_WR_Byte(0x10,0);  //high column start address
  for(n=0;n<128;n++)
   {
    OLED_WR_Byte(fill_Data,1);
   }
 }
}
 
//坐标设置
void OLED_SetPos(unsigned char x, unsigned char y)
{  OLED_WR_Byte(0xb0+y,OLED_CMD);
 OLED_WR_Byte(((x&0xf0)>>4)|0x10,OLED_CMD);
OLED_WR_Byte((x&0x0f),OLED_CMD);
}
//开启OLED显示
voidOLED_DisplayOn(void)
{
OLED_WR_Byte(0X8D,OLED_CMD);//SETDCDC命令
OLED_WR_Byte(0X14,OLED_CMD);//DCDCON
OLED_WR_Byte(0XAF,OLED_CMD);//DISPLAYON
}
//关闭OLED显示
voidOLED_DisplayOff(void)
{
OLED_WR_Byte(0X8D,OLED_CMD);//SETDCDC命令
OLED_WR_Byte(0X10,OLED_CMD);//DCDCOFF
OLED_WR_Byte(0XAE,OLED_CMD);//DISPLAYOFF
}
//清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!
voidOLED_Clear(void)
{
u8i,n;
for(i=0;i<8;i++)
 {
  OLED_WR_Byte (0xb0+i,OLED_CMD);    //设置页地址(0~7)
  OLED_WR_Byte (0x00,OLED_CMD);      //设置显示位置—列低地址
  OLED_WR_Byte (0x10,OLED_CMD);      //设置显示位置—列高地址
  for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA);
 } //更新显示
}
void OLED_On(void)
{
 u8 i,n;
 for(i=0;i<8;i++)
 {
  OLED_WR_Byte (0xb0+i,OLED_CMD);    //设置页地址(0~7)
  OLED_WR_Byte (0x00,OLED_CMD);      //设置显示位置—列低地址
  OLED_WR_Byte (0x10,OLED_CMD);      //设置显示位置—列高地址
  for(n=0;n<128;n++)OLED_WR_Byte(1,OLED_DATA);
 } //更新显示
}
//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//mode:0,反白显示;1,正常显示
//size:选择字体 16/12
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 Char_Size)
{
 unsigned char c=0,i=0;
  c=chr-' ';//得到偏移后的值
  if(x>Max_Column-1){x=0;y=y+2;}
if(Char_Size==16)
{
OLED_SetPos(x,y);
for(i=0;i<8;i++)
   OLED_WR_Byte(F8X16[c*16+i],OLED_DATA);
   OLED_SetPos(x,y+1);
   for(i=0;i<8;i++)
   OLED_WR_Byte(F8X16[c*16+i+8],OLED_DATA);
   }
   else {
    OLED_SetPos(x,y);
    for(i=0;i<6;i++)
    OLED_WR_Byte(F6x8[c][i],OLED_DATA);
 
   }
}
 
//m^n函数
u32 oled_pow(u8 m,u8 n)
{
 u32 result=1;
 while(n--)result*=m;
 return result;
}
 
//显示2个数字
//x,y :起点坐标
//len :数字的位数
//size:字体大小
//mode:模式 0,填充模式;1,叠加模式
//num:数值(0~4294967295);
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size2)
{
 u8 t,temp;
 u8 enshow=0;
 for(t=0;t120){x=0;y+=2;}
j++;
}
}

6.2 谷歌小恐龙游戏图形绘制代码

该部分为整个项目代码的核心部分之一,任何一个游戏都是需要去绘制和构建游戏的图形以及模型的。好的游戏往往都具有很好的游戏模型和精美UI,很多3A大作都具备这样的特性。

dinogame.h:

#ifndef__DINOGAME_H
#define__DINOGAME_H

voidOLED_DrawBMP(unsignedcharx0,unsignedchary0,unsignedcharx1,unsignedchary1,unsignedcharBMP[]);
voidOLED_DrawBMPFast(constunsignedcharBMP[]);
voidoled_drawbmp_block_clear(intbx,intby,intclear_size);
voidOLED_DrawGround();
voidOLED_DrawCloud();
voidOLED_DrawDino();
voidOLED_DrawCactus();
intOLED_DrawCactusRandom(unsignedcharver,unsignedcharreset);
intOLED_DrawDinoJump(charreset);
voidOLED_DrawRestart();
voidOLED_DrawCover();

#endif

dinogame.c代码:

#include"oled.h"
#include"oledfont.h"
#include"stdlib.h"

/***********功能描述:显示显示BMP图片128×64起始点坐标(x,y),x的范围0~127,y为页的范围0~7*****************/
voidOLED_DrawBMP(unsignedcharx0,unsignedchary0,unsignedcharx1,unsignedchary1,unsignedcharBMP[])
{
unsignedintj=0;
unsignedcharx,y;

if(y1%8==0)y=y1/8;
elsey=y1/8+1;
for(y=y0;y128)break;
IIC_WriteByte(0x0);
IIC_WaitAck();
}
IIC_Stop();
}

voidOLED_DrawGround()
{
staticunsignedintpos=0;
unsignedcharspeed=5;
unsignedintground_length=sizeof(GROUND);
unsignedcharx;

OLED_SetPos(0,7);
IIC_Start();
IIC_WriteByte(0x78);
IIC_WaitAck();
IIC_WriteByte(0x40);
IIC_WaitAck();
for(x=0;x< 128; x++)
 {
  IIC_WriteByte(GROUND[(x+pos)%ground_length]);
  IIC_WaitAck();
 }
 IIC_Stop();
 
 pos = pos + speed;
 //if(pos>ground_length)pos=0;
}


//绘制云朵
voidOLED_DrawCloud()
{
staticintpos=128;
staticcharheight=0;
charspeed=3;
unsignedinti=0;
intx;
intstart_x=0;
intlength=sizeof(CLOUD);
unsignedcharbyte;

//if(pos+length<= -speed) pos = 128;
 
 if (pos + length <= -speed)
 {
  pos = 128;
  height = rand()%3;
 }
 if(pos < 0)
 {
  start_x = -pos;
  OLED_SetPos(0, 1+height);
 }
 else
 {
  OLED_SetPos(pos, 1+height);
 }
 
 IIC_Start();
 IIC_WriteByte(0x78);
 IIC_WaitAck();
 IIC_WriteByte(0x40);
 IIC_WaitAck();
 for (x = start_x; x < length + speed; x++)
 {
  if (pos + x >127)break;
if(x< length) byte = CLOUD[x];
  else byte = 0x0;
 
  IIC_WriteByte(byte);
  IIC_WaitAck();
 }
 IIC_Stop();
 
 pos = pos - speed;
}
 
// 绘制小恐龙
void OLED_DrawDino()
{
 static unsigned char dino_dir = 0;
 unsigned int j=0;
 unsigned char x, y;
 unsigned char byte;
 
 dino_dir++;
 dino_dir = dino_dir%2;
 for(y=0; y<2; y++)
 {
  OLED_SetPos(16, 6+y);
  IIC_Start();
  IIC_WriteByte(0x78);
  IIC_WaitAck();
  IIC_WriteByte(0x40);
  IIC_WaitAck();
  for (x = 0; x < 16; x++)
  {
   j = y*16 + x;
   byte = DINO[dino_dir][j];
 
   IIC_WriteByte(byte);
   IIC_WaitAck();
  }
  IIC_Stop();
 }
}
 
// 绘制仙人掌障碍物
void OLED_DrawCactus()
{
 char speed = 5;
 static int pos = 128;
 int start_x = 0;
 int length = sizeof(CACTUS_2)/2;
 
 unsigned int j=0;
 unsigned char x, y;
 unsigned char byte;
 
 if (pos + length <= 0)
 {
  oled_drawbmp_block_clear(0, 6, speed);
  pos = 128;
 }
 
 for(y=0; y<2; y++)
 {
  if(pos < 0)
  {
   start_x = -pos;
   OLED_SetPos(0, 6+y);
  }
  else
  {
   OLED_SetPos(pos, 6+y);
  }
 
  IIC_Start();
  IIC_WriteByte(0x78);
  IIC_WaitAck();
  IIC_WriteByte(0x40);
  IIC_WaitAck();
 
  for (x = start_x; x < length; x++)
  {
   if (pos + x >127)break;
j=y*length+x;
byte=CACTUS_2[j];
IIC_WriteByte(byte);
IIC_WaitAck();
}
IIC_Stop();
}
oled_drawbmp_block_clear(pos+length,6,speed);//清除残影
pos=pos-speed;
}


//绘制随机出现的仙人掌障碍物
intOLED_DrawCactusRandom(unsignedcharver,unsignedcharreset)
{
charspeed=5;
staticintpos=128;
intstart_x=0;
intlength=0;

unsignedinti=0,j=0;
unsignedcharx,y;
unsignedcharbyte;
if(reset==1)
{
pos=128;
oled_drawbmp_block_clear(0,6,speed);
return128;
}
if(ver==0)length=8;//sizeof(CACTUS_1)/2;
elseif(ver==1)length=16;//sizeof(CACTUS_2)/2;
elseif(ver==2||ver==3)length=24;

for(y=0;y<2; y++)
 {
  if(pos < 0)
  {
   start_x = -pos;
   OLED_SetPos(0, 6+y);
  }
  else
  {
   OLED_SetPos(pos, 6+y);
  }
 
  IIC_Start();
  IIC_WriteByte(0x78);
  IIC_WaitAck();
  IIC_WriteByte(0x40);
  IIC_WaitAck();
 
  for (x = start_x; x < length; x++)
  {
   if (pos + x >127)break;

j=y*length+x;
if(ver==0)byte=CACTUS_1[j];
elseif(ver==1)byte=CACTUS_2[j];
elseif(ver==2)byte=CACTUS_3[j];
elsebyte=CACTUS_4[j];

IIC_WriteByte(byte);
IIC_WaitAck();
}
IIC_Stop();
}

oled_drawbmp_block_clear(pos+length,6,speed);

pos=pos-speed;
returnpos+speed;
}




//绘制跳跃小恐龙
intOLED_DrawDinoJump(charreset)
{
charspeed_arr[]={1,1,3,3,4,4,5,6,7};
staticcharspeed_idx=sizeof(speed_arr)-1;
staticintheight=0;
staticchardir=0;
//charspeed=4;

unsignedintj=0;
unsignedcharx,y;
charoffset=0;
unsignedcharbyte;
if(reset==1)
{
height=0;
dir=0;
speed_idx=sizeof(speed_arr)-1;
return0;
}
if(dir==0)
{
height+=speed_arr[speed_idx];
speed_idx--;
if(speed_idx<0) speed_idx = 0;
 }
 if (dir==1)
 {
  height -= speed_arr[speed_idx];
  speed_idx ++;
  if (speed_idx>sizeof(speed_arr)-1)speed_idx=sizeof(speed_arr)-1;
}
if(height>=31)
{
dir=1;
height=31;
}
if(height<= 0)
 {
  dir = 0;
  height = 0;
 }
 if(height <= 7) offset = 0;
 else if(height <= 15) offset = 1;
 else if(height <= 23) offset = 2;
 else if(height <= 31) offset = 3;
 else offset = 4;
 
 for(y=0; y<3; y++) // 4
 {
  OLED_SetPos(16, 5- offset + y);
 
  IIC_Start();
  IIC_WriteByte(0x78);
  IIC_WaitAck();
  IIC_WriteByte(0x40);
  IIC_WaitAck();
  for (x = 0; x < 16; x++) // 32
  {
   j = y*16 + x; // 32
   byte = DINO_JUMP[height%8][j];
 
   IIC_WriteByte(byte);
   IIC_WaitAck();
  }
  IIC_Stop();
 }
 if (dir == 0) oled_drawbmp_block_clear(16, 8- offset, 16);
 if (dir == 1) oled_drawbmp_block_clear(16, 4- offset, 16);
 return height;
}
 
// 绘制重启
void OLED_DrawRestart()
{
 unsigned int j=0;
 unsigned char x, y;
 unsigned char byte;
 //OLED_SetPos(0, 0);
 for (y = 2; y < 5; y++)
 {
  OLED_SetPos(52, y);
  IIC_Start();
  IIC_WriteByte(0x78);
  IIC_WaitAck();
  IIC_WriteByte(0x40);
  IIC_WaitAck();
  for (x = 0; x < 24; x++)
  {
   byte = RESTART[j++];
   IIC_WriteByte(byte);
   IIC_WaitAck();
  }
  IIC_Stop();
 }
 OLED_ShowString(10, 3, "GAME", 16);
 OLED_ShowString(86, 3, "OVER", 16);
}
// 绘制封面
void OLED_DrawCover()
{
 OLED_DrawBMPFast(COVER);
}

6.3 谷歌小恐龙的运行控制代码

control.h:

#ifndef__CONTROL_H
#define__CONTROL_H

intget_key();
voidGame_control();

#endif****

control.c:

#include"control.h"
#include"oled.h"
#include"dinogame.h"
#include"stdlib.h"

unsignedcharkey_num=0;
unsignedcharcactus_category=0;
unsignedcharcactus_length=8;
unsignedintscore=0;
unsignedinthighest_score=0;
intheight=0;
intcactus_pos=128;
unsignedcharcur_speed=30;
charfailed=0;
charreset=0;


intget_key()
{
if(HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_4)==0)
{
HAL_Delay(10);//延迟
if(HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_4)==0)
{
return2;
}
}

if(HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_3)==0)
{
HAL_Delay(10);//延迟
if(HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_3)==0)
{
return1;
}
}

if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==1)
{
HAL_Delay(10);//延迟
if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==1)
{
return3;
}
}

return0;
}

voidGame_control()
{

while(1)
{

if(get_key()==3)//wk_up按键按下强制退出一次循环
{
break;
}

if(failed==1)
{
OLED_DrawRestart();

key_num=get_key();
if(key_num==2)
{
if(score>highest_score)highest_score=score;
score=0;
failed=0;
height=0;
reset=1;
OLED_DrawDinoJump(reset);
OLED_DrawCactusRandom(cactus_category,reset);
OLED_Clear();
}
continue;
}


score++;
if(height<= 0) key_num = get_key();
 
  OLED_DrawGround();
  OLED_DrawCloud();
 
  if (height>0||key_num==1)height=OLED_DrawDinoJump(reset);
elseOLED_DrawDino();

cactus_pos=OLED_DrawCactusRandom(cactus_category,reset);
if(cactus_category==0)cactus_length=8;
elseif(cactus_category==1)cactus_length=16;
elsecactus_length=24;

if(cactus_pos+cactus_length< 0)
  {
    cactus_category = rand()%4;
   OLED_DrawCactusRandom(cactus_category, 1);
  }
 
  if ((height < 16) && ( (cactus_pos>=16&&cactus_pos<=32) || (cactus_pos + cactus_length>=16&&cactus_pos+cactus_length<=32)))
  {
   failed = 1;
  }
 
  
  OLED_ShowString(35, 0, "HI:", 12);
  OLED_ShowNum(58, 0, highest_score, 5, 12);
  OLED_ShowNum(98, 0, score, 5, 12);
 
 
  reset = 0;
 
  cur_speed = score/20;
  if (cur_speed >29)cur_speed=29;
HAL_Delay(30-cur_speed);
//HAL_Delay(500);
key_num=0;

}

}

6.4 多级菜单核心代码:

menu.h:

#ifndef__MENU_H
#define__MENU_H

#include"main.h"
#defineu8unsignedchar

//按键定义
#defineKEY0HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_4)//低电平有效KEY0
#defineKEY1HAL_GPIO_ReadPin(GPIOE,GPIO_PIN_3)//低电平有效
#defineWK_UPHAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)//高电平有效


typedefstruct
{
u8current;//当前状态索引号
u8next;//向下一个
u8enter;//确定
u8back;//退出
void(*current_operation)(void);//当前状态应该执行的操作
}Menu_table;

//界面UI
voidhome();
voidTemperature();
voidPalygame();
voidSetting();
voidInfo();


voidMenu_key_set(void);
u8KEY_Scan(u8mode);

voidTestTemperature();
voidConrtolGame();
voidSet();
voidInformation();

voidLED();
voidRTC_display();

#endif

menu.c:

#include"menu.h"
#include"oled.h"
#include"gpio.h"
#include"dinogame.h"
#include"control.h"
#include"DHT11.h"
#include"rtc.h"

RTC_DateTypeDefGetData;//获取日期结构体

RTC_TimeTypeDefGetTime;//获取时间结构体


//UI界面
//主页
/****************************************************/
//UI库

/****************************************************/

void(*current_operation_index)();

Menu_tabletable[30]=
{
{0,0,1,0,(*home)},//一级界面(主页面)索引,向下一个,确定,退出

{1,2,5,0,(*Temperature)},//二级界面温湿度
{2,3,6,0,(*Palygame)},//二级界面游戏
{3,4,7,0,(*Setting)},//二级界面设置
{4,1,8,0,(*Info)},//二级界面信息

{5,5,5,1,(*TestTemperature)},//三级界面:DHT11测量温湿度
{6,6,6,2,(*ConrtolGame)},//三级界面:谷歌小恐龙Dinogame
{7,7,9,3,(*Set)},//三级界面:设置普通外设状态LED
{8,8,8,4,(*Information)},//三级界面:作者和相关项目信息

{9,9,7,3,(*LED)},//LED控制
};

uint8_tfunc_index=0;//主程序此时所在程序的索引值

voidMenu_key_set(void)
{
if((KEY_Scan(1)==1)&&(func_index!=6))
{
func_index=table[func_index].next;//按键next按下后的索引号
OLED_Clear();
}

if((KEY_Scan(1)==2)&&(func_index!=6))
{
func_index=table[func_index].enter;//按键enter按下后的索引号
OLED_Clear();
}

if(KEY_Scan(1)==3)
{
func_index=table[func_index].back;//按键back按下后的索引号
OLED_Clear();
}

current_operation_index=table[func_index].current_operation;//执行当前索引号所对应的功能函数
(*current_operation_index)();//执行当前操作函数
}


voidhome()
{
RTC_display();
OLED_DrawBMP(0,0,20,3,signal_BMP);
OLED_DrawBMP(20,0,36,2,gImage_bulethouch);
OLED_DrawBMP(112,0,128,2,gImage_engery);
OLED_DrawBMP(4,6,20,8,gImage_yes);
OLED_DrawBMP(12,4,28,6,gImage_left);
OLED_DrawBMP(40,2,88,8,gImage_home);
OLED_DrawBMP(99,4,115,6,gImage_right);
OLED_DrawBMP(107,6,123,8,gImage_back);
}

voidTemperature()
{
RTC_display();
OLED_DrawBMP(0,0,20,3,signal_BMP);
OLED_DrawBMP(20,0,36,2,gImage_bulethouch);
OLED_DrawBMP(112,0,128,2,gImage_engery);
OLED_DrawBMP(4,6,20,8,gImage_yes);
OLED_DrawBMP(12,4,28,6,gImage_left);
OLED_DrawBMP(40,2,88,8,gImage_temp);
OLED_DrawBMP(99,4,115,6,gImage_right);
OLED_DrawBMP(107,6,123,8,gImage_back);
}

voidPalygame()
{
RTC_display();
OLED_DrawBMP(0,0,20,3,signal_BMP);
OLED_DrawBMP(20,0,36,2,gImage_bulethouch);
OLED_DrawBMP(112,0,128,2,gImage_engery);
OLED_DrawBMP(4,6,20,8,gImage_yes);
OLED_DrawBMP(12,4,28,6,gImage_left);
OLED_DrawBMP(40,2,88,8,gImage_playgame);
OLED_DrawBMP(99,4,115,6,gImage_right);
OLED_DrawBMP(107,6,123,8,gImage_back);
}

voidSetting()
{
RTC_display();
OLED_DrawBMP(0,0,20,3,signal_BMP);
OLED_DrawBMP(20,0,36,2,gImage_bulethouch);
OLED_DrawBMP(112,0,128,2,gImage_engery);
OLED_DrawBMP(4,6,20,8,gImage_yes);
OLED_DrawBMP(12,4,28,6,gImage_left);
OLED_DrawBMP(40,2,88,8,gImage_setting);
OLED_DrawBMP(99,4,115,6,gImage_right);
OLED_DrawBMP(107,6,123,8,gImage_back);
}

voidInfo()
{
RTC_display();
OLED_DrawBMP(0,0,20,3,signal_BMP);
OLED_DrawBMP(20,0,36,2,gImage_bulethouch);
OLED_DrawBMP(112,0,128,2,gImage_engery);
OLED_DrawBMP(4,6,20,8,gImage_yes);
OLED_DrawBMP(12,4,28,6,gImage_left);
OLED_DrawBMP(40,2,88,8,gImage_info);
OLED_DrawBMP(99,4,115,6,gImage_right);
OLED_DrawBMP(107,6,123,8,gImage_back);
}


//按键函数,不支持连按
u8KEY_Scan(u8mode)
{
staticu8key_up=1;
if(mode)key_up=1;
if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))
{
HAL_Delay(100);//消抖
key_up=0;
if(KEY0==0)return1;
elseif(KEY1==0)return2;
elseif(WK_UP==1)return3;
}elseif(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1;
return0;
}

voidTestTemperature()
{
DHT11();
}

voidConrtolGame()
{
Game_control();
}

voidSet()
{
OLED_ShowString(0,0,"Peripherals:Lights",16);
OLED_ShowString(0,2,"Status:Closed",16);
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_5,GPIO_PIN_SET);
}

voidInformation()
{
OLED_ShowString(0,0,"Author:Sneak",16);
OLED_ShowString(0,2,"Date:2022/8/23",16);
OLED_ShowString(0,4,"Lab:Multi-levelmenu",16);
}

voidLED()
{
OLED_ShowString(0,0,"Peripherals:Lights",16);
OLED_ShowString(0,2,"Status:Open",16);
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_5,GPIO_PIN_RESET);
}



voidRTC_display()//RTC????
{
/*GettheRTCcurrentTime*/
HAL_RTC_GetTime(&hrtc,&GetTime,RTC_FORMAT_BIN);
/*GettheRTCcurrentDate*/
HAL_RTC_GetDate(&hrtc,&GetData,RTC_FORMAT_BIN);

/*DisplaydateFormat:yy/mm/dd*/

/*DisplaytimeFormat:hhss*/
OLED_ShowNum(40,0,GetTime.Hours,2,16);//hour
OLED_ShowString(57,0,":",16);
OLED_ShowNum(66,0,GetTime.Minutes,2,16);//min
OLED_ShowString(83,0,":",16);
OLED_ShowNum(93,0,GetTime.Seconds,2,16);//seconds
}

七、总结与代码开源

总结:本项目目前还处于最初代版本,十分简易,后期笔者将抽时间去精进优化该多级菜单项目。其中,UI界面中的电池与信号目前都还处于贴图状态,后期笔者会加上库仑计测量电池电量等。文章中指出了需要注意的地方与可以改进的点,感兴趣的朋友可以彼此交流交流。

代码下载地址:https://download.csdn.net/download/black_sneak/86469358

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • OLED
    +关注

    关注

    118

    文章

    5986

    浏览量

    221472
  • STM32
    +关注

    关注

    2240

    文章

    10674

    浏览量

    348806
  • 菜单
    +关注

    关注

    0

    文章

    33

    浏览量

    13356
  • 开源
    +关注

    关注

    3

    文章

    2989

    浏览量

    41720
  • RTOS
    +关注

    关注

    20

    文章

    776

    浏览量

    118796

原文标题:【开源小项目】基于STM32的OLED舵机菜单显示

文章出处:【微信号:嵌入式悦翔园,微信公众号:嵌入式悦翔园】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    STM32简易多级菜单(数组查表法)显示方法

    本篇介绍了一种简易的多级菜单显示方法,本质是通过数组查表,实现各级菜单的各个页面(状态)的切换(跳转),并在STM32上编程实现,通过OLED
    的头像 发表于 06-07 09:11 7250次阅读
    <b class='flag-5'>STM32</b>简易多级<b class='flag-5'>菜单</b>(数组查表法)<b class='flag-5'>显示</b>方法

    ST7920_12864液晶图文菜单显示.C

    ST7920_12864液晶图文菜单显示.C
    发表于 08-17 21:40

    pic16f1937与lcd12864实现多级菜单控制(按键分选择,确定,返回主菜单是四首诗的题目,子菜单显示四首诗的内容)

    pic16f1937与lcd12864实现多级菜单控制(按键分选择,确定,返回主菜单是四首诗的题目,子菜单显示四首诗的内容)
    发表于 01-19 11:48

    急求一个msp430f149三级菜单显示程序,(LCD12864显示

    急求一个msp430f149三级菜单显示程序,(LCD12864显示
    发表于 07-07 21:24

    单片机小项目开源分享

    转眼间已经工作一年,目前从事linux/android驱动工作,将大学期间单片机的小项目开源出来,供大家交流学习,比较简单。源码在git clone http://www.github.com
    发表于 09-17 11:33

    STM32CubeMX菜单乱码

    win10 + SetupSTM32CubeMX-4.24.0 安装成功,但在主界面,菜单显示乱码,如下,何故?
    发表于 12-26 08:51

    多级菜单显示

    前言只要有显示屏的地方,就要用到多级菜单显示。在很多初学者眼里,多级菜单显示是很难的,今天我做完了这个
    发表于 08-11 07:19

    如何搭建基于STM32驱动OLED显示三级菜单界面框架?

    什么是主界面?如何控制界面之间的切换?如何搭建基于STM32驱动OLED显示三级菜单界面框架?
    发表于 12-17 06:45

    【野火鲁班猫2开发板体验】Debian + 物联网 + 综合小项目

    野火鲁班猫2单板电脑=物联网+综合小项目野火开发板,鲁班猫2单板电脑,这次分享一个物联网+综合小项目使用到外设有火焰传感器,蜂鸣器,语音模块,led灯,0.96存iic接口的oled屏幕,超声波
    发表于 03-08 20:12

    【鲁班猫创意氛围赛】鲁班猫2单板电脑=物联网+综合小项目

    野火鲁班猫2单板电脑=物联网+综合小项目 项目概述 这次分享一个物联网+综合小项目 使用的开发板 野火家的开发板,鲁班猫2单板电脑开发板 项目当中使用到的外设模块 火焰传感器,蜂鸣器,
    发表于 05-18 20:06

    多级操作菜单显示系统设计

    本文旨在提供一个轻量级的单片机多级菜单实现方法,以较少的系统资源消耗和简单方便的方法完成菜单设计。考虑到菜单程序需要具备3个基本要素:一是每个菜单窗口要
    发表于 03-26 15:07 6977次阅读
    多级操作<b class='flag-5'>菜单显示</b>系统设计

    STM32二级菜单通过按键切换自定义任务OLED显示的程序和工程文件

    本文档的主要内容详细介绍的是STM32二级菜单通过按键切换自定义任务OLED显示的程序和工程文件。
    发表于 08-31 08:00 47次下载
    <b class='flag-5'>STM32</b>二级<b class='flag-5'>菜单</b>通过按键切换自定义任务<b class='flag-5'>OLED</b><b class='flag-5'>显示</b>的程序和工程文件

    PADS在WIN10系统中菜单显示不全怎么解决?

    到这里PADS菜单显示的问题解决了,但是因为我们修改了系统默认字体,会影响到其它软件,比如笔者使用的webstorm就会显示乱码。
    的头像 发表于 03-14 11:57 8253次阅读

    STM32学习——入门小项目

    STM32学习——入门小项目
    发表于 12-07 17:21 70次下载
    <b class='flag-5'>STM32</b>学习——入门<b class='flag-5'>小项目</b>

    基于嵌入式STM32的智能手表设计实现

    本文的OLED多级菜单UI为一个综合性的STM32小项目,使用多传感器与OLED显示屏实现智能终
    发表于 09-25 09:51 1366次阅读
    基于嵌入式<b class='flag-5'>STM32</b>的智能手表设计实现