0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

分享一個(gè)STM32菜單框架

STM32嵌入式開發(fā) ? 來(lái)源:STM32嵌入式開發(fā) ? 2023-05-31 09:42 ? 次閱讀

相信很多攻城獅都用過(guò)液晶屏,想寫好一點(diǎn)的ui好像不太可能或且花費(fèi)很多時(shí)間,直接寫吧,感覺(jué)好像很零碎,coding都怕了。

下面介紹一個(gè)簡(jiǎn)單易用的菜單框架,你會(huì)發(fā)現(xiàn)它能做多層菜單而且結(jié)果清晰。

基本原理: 99d1faae-fee3-11ed-90ce-dac502259ad0.png ????

如上圖液晶顯示一屏我們定義為一個(gè)page,page中的項(xiàng)目定義為item;這樣page就是item的容器了。當(dāng)我們選中其中的一個(gè)item進(jìn)去后是不是又是一個(gè)page呢,如下圖。 99d898dc-fee3-11ed-90ce-dac502259ad0.png ????

這樣的話每一個(gè)item的下面都對(duì)應(yīng)一個(gè)page,這樣是不是就構(gòu)成一個(gè)多層的菜單了。

99e4b180-fee3-11ed-90ce-dac502259ad0.png ? ????

他們是什么關(guān)系呢?

一個(gè)page中有item,那么用結(jié)構(gòu)體就可以實(shí)現(xiàn)啦;item下面又有page,那么在item中加一個(gè)page的指針指向item對(duì)應(yīng)的page頁(yè)。

前面都是從上到下的,那么怎么返回呢?

觀察發(fā)現(xiàn)返回就是子page返回父page,這樣在page結(jié)構(gòu)體中假如一項(xiàng)父page的指針不就ok了。

具體實(shí)現(xiàn)請(qǐng)看源文件。


/******************************************************************************************************/
//主菜單
//定義Item項(xiàng)             //顯示方式&序號(hào)  項(xiàng)目的名字    項(xiàng)目指向的頁(yè)(Page)
const struct Item main_item[]={ 0x00, "信息",   &SMS_Page,
        0x01, "設(shè)置",   &Setting_Page,
        0x02, "版本",   &Version_Page,
        0x03, "時(shí)間",   &Time_Page,
        0x04, "狀態(tài)",   0,
        0x05, "報(bào)警",   0,
        0x06, "飛信",   0,
        0x07, "問(wèn)答",   0
};
//定義一個(gè)Page        父頁(yè) 該頁(yè)的回調(diào)函數(shù) 該頁(yè)的項(xiàng)          項(xiàng)的個(gè)數(shù)    
const struct PAGE mainPage={0,mainPageCallBack,main_item,sizeof(main_item)/sizeof(struct Item)};
/*********************************************************************************************************/




const struct PAGE Version_Page={&mainPage,Version_CallBack,0,0};
/***************************************************************************************************************/


//定義Item項(xiàng)              //顯示方式&序號(hào)    項(xiàng)目的名字      項(xiàng)目指向的頁(yè)(Page)
const struct Item Setting_item[]={ 0x10, " 00.設(shè)0",   0,
         0x11, " 01.設(shè)1",   0,
         0x12, " 02.設(shè)2",   0,
         0x13, " 03.設(shè)3",   0,
         0x14, " 04.設(shè)4",   0,
         0x15, " 05.設(shè)5",   0,
         0x16, " 06.設(shè)6 你好",  0,
         0x17, " 07.設(shè)7",   0,
         0x18, " 08.設(shè)8",   0,
         0x19, " 09.設(shè)9",   0,
         0x1A, " 10.設(shè)10",   0
         };
const struct PAGE Setting_Page={&mainPage,Setting_CallBack,Setting_item,sizeof(Setting_item)/sizeof(struct Item)};
/***************************************************************************************************************/


const struct PAGE Time_Page={&mainPage,Time_CallBack,0,0};


/***************************************************************************************************************/
//定義Item項(xiàng)              //顯示方式&序號(hào)    項(xiàng)目的名字      項(xiàng)目指向的頁(yè)(Page)
const struct Item SMS_item[]={ 
         0x10, " 00.",   &SMS_Text_Page,
         0x11, " 01.",   &SMS_Text_Page,
         0x12, " 02.",   &SMS_Text_Page,
         0x13, " 03.",   &SMS_Text_Page,
         0x14, " 04.",   &SMS_Text_Page,
         0x15, " 05.",   &SMS_Text_Page,
         0x16, " 06.",   &SMS_Text_Page,
         0x17, " 07.",   &SMS_Text_Page,
         0x18, " 08.",   &SMS_Text_Page,
         0x19, " 09.",   &SMS_Text_Page,
         0x1A, " 10.",   &SMS_Text_Page
         };


const struct PAGE SMS_Page={&mainPage,SMS_CallBack,SMS_item,sizeof(Setting_item)/sizeof(struct Item)};




Menu.h:

#ifndef _Menu_H_BAB
#define _Menu_H_BAB


#include "stm32f10x.h"
#include "LCD.h"
#include "Key.h"


#define KEY_Special  255 ///<這個(gè)保留用于特別事件


//菜單調(diào)試,在調(diào)試時(shí)最好定義,可以幫助發(fā)現(xiàn)問(wèn)題;當(dāng)發(fā)布時(shí)把其置為0可以加快速度
#define MENU_DEBUG 1


void Menu_Show(void);


struct PAGE
{
 const struct PAGE *pParent;
 void (*Function)(u8 key);
 const struct Item *pItem;
 const u8 ItemNum;
};
struct Item
{
 /**
 高4位作為特殊用途(bit4=1表示列表顯示否則兩列顯示),低4位用于標(biāo)記Item的序號(hào)  

 如果為列表模式時(shí)*pText的格式為:" xx.string",最前面保留一個(gè)空格用于個(gè)光標(biāo)(>)使用,xx.為兩位序號(hào)不要"."一定要有,string是要顯示的文字,最多能顯示6個(gè)漢字  

 如果是兩列顯示則pText,即為要顯示的文本(最多2個(gè)漢字)
 */
 const u8 TypeAndIndex; 
 const u8 *pText;
 const struct PAGE *pChildrenPage;
};


extern const struct PAGE *pPage;


void SetMainPage(const struct PAGE *pMainPage);
void ShowMenu(const struct PAGE *pPage);
void ShowPage(const struct PAGE *pPage);
void ShowParentPage(void);
void ShowItemPage(void);
void SelPageItem(u8 ItemIndex);
u8 Menu_GetSelItem(void);


void GetShowLst(u8 *pOutMin,u8 *pOutMax);


void KeySelItem(u8 key);


#endif 

Menu.c:

#include "Menu.h"


//保存選中的菜單項(xiàng)變量
static u8 SelItem=0;


/**
用于當(dāng)前LCD列表中顯示著哪幾項(xiàng)
高4位:最大序號(hào)
低4為:最小序號(hào)
*/
static u8 ListShow=0x00;


const struct PAGE *pPage;


void SelItemOfList(u8 index);


void SetMainPage(const struct PAGE *pMainPage)
{
 pPage=pMainPage;
}
/**
獲得當(dāng)前選中的菜單項(xiàng)
@return 返回菜單序號(hào)
*/
u8 Menu_GetSelItem(void)
{
 return SelItem;
}


/**
獲取當(dāng)前顯示列表的范圍
@param pOutMin 當(dāng)前顯示的最小序號(hào)
@param pOutMax 當(dāng)前顯示的最大序號(hào)
*/
void GetShowLst(u8 *pOutMin,u8 *pOutMax)
{
 *pOutMin=ListShow&0x0f; 
 *pOutMax=ListShow>>4;
}
void ShowList(u8 min,u8 max)
{
 u8 i=0,index=0;
 #if MENU_DEBUG
  if(max-min>3)
  {
   Lcd_Clr_Scr();
   LCD_Write_Str(0,0,"err:ShowList>3");
   while (1);
  }
  
  if ((pPage->pItem[0].TypeAndIndex & 0x10)==0)///<如果是使用列表方式
  {
   
    Lcd_Clr_Scr();
    LCD_Write_Str(0,0,"不是列表類型不能不能列出");
    while (1); 
  }
 #endif
 
 Lcd_Clr_Scr();
 for (index=min;index<=max;index++)
 {


  LCD_Write_Str(i++,0,pPage->pItem[index].pText);
 }
 ListShow=(max<<4)|min; ///<記錄當(dāng)前顯示的Item
 
}
/**
頁(yè)顯示


1.當(dāng)這個(gè)頁(yè)有項(xiàng)目(Item)時(shí):顯示Item并同時(shí)選中Item 0   

2.沒(méi)有時(shí):會(huì)調(diào)用該P(yáng)age的回調(diào)函數(shù)并傳入KEY_Special 參數(shù) 

@param pPage 指向一個(gè)page
*/
void ShowPage( const struct PAGE *pPage)
{
 s8 i;
 ///清屏
 Lcd_Clr_Scr();
   
 if(pPage->pItem==0) 
 {
  pPage->Function(KEY_Special);
  return; ///<如果沒(méi)有Item項(xiàng)則不顯示Item,直接返回
 }
  
 if (pPage->pItem[0].TypeAndIndex & 0x10)///<如果是使用列表方式
 {
  ShowList(0,3);
  SelItemOfList(0);
  pPage->Function(KEY_Special);
 }
 else
 { 
  ///取出page中的Item并顯示
  for (i=0;iItemNum;i++)
  {
   if (i<4)
   {
    LCD_Write_Str(i,1,pPage->pItem[i].pText);
   }
   else
   {
    LCD_Write_Str(i-4,5,pPage->pItem[i].pText);
   }
   
  }
  SelPageItem(0);///<選中Item 0
  pPage->Function(KEY_Special);
 }
 
};


/**
顯示父頁(yè)(ParentPage)
*/
void ShowParentPage(void)
{
 pPage=pPage->pParent;
 ShowPage(pPage);
}


/**
顯示項(xiàng)目(Item)下對(duì)應(yīng)的頁(yè)(Page)
*/
void ShowItemPage(void)
{
 //如果該項(xiàng)下沒(méi)有頁(yè),這警告或返回
 if (pPage->pItem[Menu_GetSelItem()].pChildrenPage ==0)
 {
  #if MENU_DEBUG
   Lcd_Clr_Scr();
   LCD_Write_Str(0,0,"該項(xiàng)下無(wú)顯示請(qǐng)修正");
   while (1);
  #else
   return;
  #endif 
 }
 pPage=pPage->pItem[Menu_GetSelItem()].pChildrenPage; //獲得菜單項(xiàng)(Item)對(duì)應(yīng)的page


 ShowPage(pPage);
}


/**
選擇page中的Item項(xiàng)
@param ItemIndex page中Item的索引號(hào) 0~7
*/
void SelPageItem(u8 ItemIndex)
{
 ///檢查是否有錯(cuò)誤調(diào)用
#if MENU_DEBUG


 if (ItemIndex>=8)
 {
  LCD_Write_Str(0,0,"設(shè)置菜單項(xiàng)溢出");
  return;
 }
#endif


///清除上次選中的
   if (SelItem<4)
   {
  LCD_Write_Str(SelItem,0,"  ");
  LCD_Write_Str(SelItem,3,"  ");
 
   }
   else
   {
  LCD_Write_Str(SelItem-4,4,"  ");
  LCD_Write_Str(SelItem-4,7,"  ");
   }
///選中這次要選中的  
   if (ItemIndex<4)
   {
  LCD_Write_Str(ItemIndex,0,"【");
  LCD_Write_Str(ItemIndex,3,"】");
  SelItem=ItemIndex;
   }
   else
   {
  LCD_Write_Str(ItemIndex-4,4,"【");
  LCD_Write_Str(ItemIndex-4,7,"】");
  SelItem=ItemIndex;
   } 
};
void SelItemOfList(u8 index)
{
 u8 max;
 u8 min;
 
 max=ListShow>>4;
 min=ListShow&0x0f;
 
 if (index>max) ///<超出最大當(dāng)前顯示的序號(hào)
 {
  
  LCD_Write_Str(Menu_GetSelItem()-min,0," ");
  
  min+=1;
  max+=1;
  ShowList(min,max);
  ListShow=(max<<4)|min;
  
  LCD_Write_Str(index-min,0,">");
  
 }
 else if(index>=min)///<在最小和最大序號(hào)之間
 {
  LCD_Write_Str(Menu_GetSelItem()-min,0," ");
  LCD_Write_Str(index-min,0,">");
 }
 else     ///<低于最小當(dāng)前顯示最小序號(hào)
 {
  LCD_Write_Str(Menu_GetSelItem()-min,0," ");
  
  min-=1;
  max-=1;
  ShowList(min,max);
  ListShow=(max<<4)|min;
  
  LCD_Write_Str(index-min,0,">");
 }
 SelItem=index;
}
void KeySelItem(u8 key)
{
 s8 index;
 if (pPage->pItem[0].TypeAndIndex & 0x10)///<如果是使用列表方式
 {
  switch(key)
  {
   case KEY_UP:
    index=Menu_GetSelItem()-1;
    if(index<0) break;
    
    SelItemOfList(index);
    break;
   case KEY_Down:
    index=Menu_GetSelItem()+1;
    if(index>(pPage->ItemNum-1)) break;;
    SelItemOfList(index);
    break;
  }
  return;
 }
 switch(key)
 {
  case KEY_UP:
   index=Menu_GetSelItem()-1;
   if(index<0) index=pPage->ItemNum-1;
   SelPageItem(index);
   break;
  case KEY_Down:
   index=Menu_GetSelItem()+1;
   if(index>(pPage->ItemNum-1)) index=0;
   SelPageItem(index);
   break;
  case KEY_Left:
  case KEY_Right: 
   index=Menu_GetSelItem();
   if (index<4)
   {
    if((index+4)>(pPage->ItemNum-1)) return; //右沒(méi)有Item項(xiàng),無(wú)法選中右邊項(xiàng);所以返回
    index+=4;        //右邊有Item時(shí)把index定位到右邊的Item
   } 
   else     index-=4;      //因?yàn)橛疫呌蠭tem項(xiàng)時(shí),左邊一定有Item項(xiàng);因?yàn)槭前错樞虬才诺?   SelPageItem(index);
   break;
 }
}

篇幅有限,MenuAPP代碼未貼出。 

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 液晶
    +關(guān)注

    關(guān)注

    6

    文章

    606

    瀏覽量

    69690
  • STM32
    +關(guān)注

    關(guān)注

    2270

    文章

    10921

    瀏覽量

    356941
  • 液晶屏
    +關(guān)注

    關(guān)注

    18

    文章

    720

    瀏覽量

    42939
  • 指針
    +關(guān)注

    關(guān)注

    1

    文章

    480

    瀏覽量

    70592

原文標(biāo)題:一個(gè)STM32菜單框架,文末附工程文件

文章出處:【微信號(hào):c-stm32,微信公眾號(hào):STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    12864液晶使用教程分享 基于MCU菜單框架設(shè)計(jì)方案

    個(gè)12864里面有菜單功能。 以前可能覺(jué)得菜單高大上,其實(shí)并不是想象中的復(fù)雜,本文為大家分享個(gè)用單色屏做的
    發(fā)表于 02-19 17:29 ?1w次閱讀
    12864液晶使用教程分享 基于MCU<b class='flag-5'>菜單</b><b class='flag-5'>框架</b>設(shè)計(jì)方案

    STM32簡(jiǎn)易多級(jí)菜單(數(shù)組查表法)顯示方法

    本篇介紹了種簡(jiǎn)易的多級(jí)菜單的顯示方法,本質(zhì)是通過(guò)數(shù)組查表,實(shí)現(xiàn)各級(jí)菜單的各個(gè)頁(yè)面(狀態(tài))的切換(跳轉(zhuǎn)),并在STM32上編程實(shí)現(xiàn),通過(guò)OLED屏幕,以及借助U8g2圖形庫(kù),測(cè)試了多級(jí)
    的頭像 發(fā)表于 06-07 09:11 ?8801次閱讀
    <b class='flag-5'>STM32</b>簡(jiǎn)易多級(jí)<b class='flag-5'>菜單</b>(數(shù)組查表法)顯示方法

    輕量級(jí)多級(jí)菜單控制框架

    有很多這種菜單框架的代碼,但是大多耦合性太強(qiáng),無(wú)法獨(dú)立出來(lái)適配不同的菜單設(shè)計(jì)。 本文介紹個(gè)降低了耦合性,完全獨(dú)立的
    發(fā)表于 10-12 09:36

    如何設(shè)計(jì)個(gè)產(chǎn)品級(jí)MCU菜單框架

    聲明:本處所說(shuō)的菜單是用在128*64這種小屏幕的菜單,例如下面這種,不是彩屏上的GUI。作為個(gè)底層驅(qū)動(dòng)工程師,驅(qū)動(dòng)寫完了,是要寫硬件測(cè)試程序的。這個(gè)測(cè)試程序,
    發(fā)表于 11-04 06:43

    如何搭建基于STM32驅(qū)動(dòng)OLED屏顯示三級(jí)菜單界面框架?

    什么是主界面?如何控制界面之間的切換?如何搭建基于STM32驅(qū)動(dòng)OLED屏顯示三級(jí)菜單界面框架
    發(fā)表于 12-17 06:45

    STM32系統(tǒng)時(shí)鐘框架

    STM32系統(tǒng)時(shí)鐘框架圖,能夠幫你詳細(xì)了解STM32單片機(jī)時(shí)鐘。
    發(fā)表于 08-18 18:24 ?19次下載

    單片機(jī)實(shí)例:個(gè)用單色屏做的菜單框架資料下載

    電子發(fā)燒友網(wǎng)為你提供單片機(jī)實(shí)例:個(gè)用單色屏做的菜單框架資料下載的電子資料下載,更有其他相關(guān)的電路圖、源代碼、課件教程、中文資料、英文資料、參考設(shè)計(jì)、用戶指南、解決方案等資料,希望可以
    發(fā)表于 03-27 08:42 ?16次下載
    單片機(jī)實(shí)例:<b class='flag-5'>一</b><b class='flag-5'>個(gè)</b>用單色屏做的<b class='flag-5'>菜單</b><b class='flag-5'>框架</b>資料下載

    個(gè)產(chǎn)品級(jí)MCU菜單框架設(shè)計(jì)~

    聲明:本處所說(shuō)的菜單是用在128*64這種小屏幕的菜單,例如下面這種,不是彩屏上的GUI。作為個(gè)底層驅(qū)動(dòng)工程師,驅(qū)動(dòng)寫完了,是要寫硬件測(cè)試程序的。這個(gè)測(cè)試程序,
    發(fā)表于 10-29 11:20 ?15次下載
    <b class='flag-5'>一</b><b class='flag-5'>個(gè)</b>產(chǎn)品級(jí)MCU<b class='flag-5'>菜單</b><b class='flag-5'>框架</b>設(shè)計(jì)~

    基于STM32F407的簡(jiǎn)易菜單設(shè)計(jì)+LCD+按鍵

    基于STM32F407的簡(jiǎn)易多級(jí)菜單設(shè)計(jì)+LCD+按鍵實(shí)現(xiàn)原理主要使用 雙向鏈表 結(jié)構(gòu)實(shí)現(xiàn)的菜單://定義菜單中功能項(xiàng)的類型#define TYPE_SUBMENU 101 //具有
    發(fā)表于 12-04 10:06 ?56次下載
    基于<b class='flag-5'>STM32</b>F407的簡(jiǎn)易<b class='flag-5'>菜單</b>設(shè)計(jì)+LCD+按鍵

    一個(gè)STM32CubeIDE項(xiàng)目

    使用STM32CubeIDE的第一個(gè)項(xiàng)目開始第一個(gè)項(xiàng)目添加代碼今天開始做一個(gè)STM32CubeIDE的第
    發(fā)表于 12-29 19:29 ?11次下載
    第<b class='flag-5'>一個(gè)</b><b class='flag-5'>STM32</b>CubeIDE項(xiàng)目

    個(gè)帶LCD的簡(jiǎn)單Arduino菜單

    電子發(fā)燒友網(wǎng)站提供《個(gè)帶LCD的簡(jiǎn)單Arduino菜單.zip》資料免費(fèi)下載
    發(fā)表于 11-15 14:38 ?1次下載
    <b class='flag-5'>一</b><b class='flag-5'>個(gè)</b>帶LCD的簡(jiǎn)單Arduino<b class='flag-5'>菜單</b>

    個(gè)產(chǎn)品級(jí)MCU菜單框架設(shè)計(jì)

    本處所說(shuō)的菜單是用在128*64這種小屏幕的菜單,例如下面這種,不是彩屏上的GUI。
    的頭像 發(fā)表于 09-07 09:34 ?871次閱讀
    <b class='flag-5'>一</b><b class='flag-5'>個(gè)</b>產(chǎn)品級(jí)MCU<b class='flag-5'>菜單</b><b class='flag-5'>框架</b>設(shè)計(jì)

    基于LCD驅(qū)動(dòng)架構(gòu)的MCU菜單框架設(shè)計(jì)

    當(dāng)前代碼: 1實(shí)現(xiàn)了雙列菜單,用數(shù)字鍵選擇進(jìn)入下層。每頁(yè)最多顯示8個(gè)菜單(4*4鍵盤用1-8鍵) 2 實(shí)現(xiàn)了單列菜單,通過(guò)上下翻查看
    發(fā)表于 10-11 14:51 ?1456次閱讀
    基于LCD驅(qū)動(dòng)架構(gòu)的MCU<b class='flag-5'>菜單</b><b class='flag-5'>框架</b>設(shè)計(jì)

    STM32 OLED多菜單操作

    stm32 ?oled多菜單操作
    發(fā)表于 10-09 11:01 ?2次下載

    STM32CubeMX的菜單介紹

    相信初學(xué)者打開STM32CubeMX定是臉懵逼,里面都是全英文的,不知道從何入手。這里先給大家簡(jiǎn)單講解下。在新建工程這欄里,我們最常
    的頭像 發(fā)表于 12-25 21:03 ?155次閱讀
    <b class='flag-5'>STM32</b>CubeMX的<b class='flag-5'>菜單</b>介紹