实践丨手把手教你用STM32设计WiFi语音播报日程表

x33g5p2x  于2022-07-19 转载在 其他  
字(14.7k)|赞(0)|评价(0)|浏览(333)

**摘要:**随着电子产品的发展,数字日程表这项应用在人们工作和生活中起到越来越重要的作用。本文带领大家基于STM32自己动手制作一个WiFi语音播报日程表。

本文分享自华为云社区《基于STM32设计的WiFi语音播报日程表》,作者: DS小龙哥 。

1. 前言

近年来,随着电子产品的发展,数字日程表这项应用在人们工作和生活中起到越来越重要的作用。时间对人们来说总是那么宝贵,工作的忙碌性和繁杂性容易使人忘记当前的时间,忘记了要做的事情,当事情不是很重要的时候,这种遗忘无伤大雅。但是,遇上重要事务,一时的耽误可能酿成大祸。

因此从人们的日常生活到公司办公,从台式电脑到便携式智能手机,都要求标配上日程表的作用。人们要求随时随地都能快速准确的提醒当前事务,并且要求日程表能够更直观、更可靠、更便宜。这种要求催生了新型日程表的产生。除此之外,由于对社会责任的更多承担,人们要求所设计的产品能够产生尽量少的垃圾、能够消耗尽量少的能量。因此人们对日程表的又有了体积小、功耗低的要求。

2. 功能需求

2.1 硬件部分

整个项目在正点原子stm32f103mini开发板环境下进行。开发板主控是stm32f103rct6。

项目模块由一些部分组成:

(1)2.8寸tft触摸屏负责显示和进行交互;

(2)pcf8563t用作时钟计时,并把日期和时间显示在屏幕上。

(3)时间采用24小时制显示,上位机要支持设备端RTC日期及时间信息更新.

(4)DHT11温湿度传感器检测环境信息,并把信息显示在液晶屏幕上;

(5)使用esp8266作WiFi模块与手机app进行通信;

(6)单片机将接收到的内容存储在w25Q64内,同时可以在屏幕上指定位置将内容显示出来。内容包括具体日程的文字内容以及日程开始、结束的时间;

(7)使用蜂鸣器以及通过syn6288语音芯片合成的语音信息,通过喇叭播报实现提醒功能。在日程开始、结束前五分钟提前(这个提前提醒的时间要可以修改)通过蜂鸣器以及喇叭发出语音提示。

(8)显示屏(横屏显示)上应包含:
基础的日期、时间、温湿度信息显示,屏幕主体部分通过列表的方式显示从手机app端接收到的日程内容;屏幕上设置一个触屏按钮,按下该按钮是可以跳当前进行中的,或者还未开始的下一项即将开始的日程。

日程显示部分:因为2.8寸显示屏的空间有限,所以同屏范围内只显示一到两个日程的具体内容。需要显示出日程的文字内容,开始和结束的时间。当日程时间即将开始以及即将结束时触发语音提示;日程结束之后要从当前显示位置上清除,同时删除w25Q64上存储的信息。同时显示下一个待开始的日程。

显示的文本部分要求能够支持显示16和24大小的,包括中文字符在内的所有字符(无法兼容就做成24大小即可)。从手机端发送的中文文本信息在在屏幕上显示的同时要存储在w25q64内。并且单片机终端上要支持存储最少十五条日程内容。单片机要能识别具体日程的时间信息,根据时间排序,同时判断日程是否过时,过时的日程直接删除,删除和跳过日程不需要触发提示。

(9)语音提示内容:
1:您有待开始的日程,请注意时间。(如果可以实现将手机端输入的日程内容(主要是汉字)读出来,那么此句改为:下一项日程:XXX即将开始,请注意时间。(XXX内即为日程内容))

2:当前日程即将结束

3:连接成功(成功连上app时播报)

4:连接失败

5:日程已设置(单片机接收到手机上发送的日程内容。)

(10)每句提示播报前蜂鸣器响一声,响完后停顿一秒再播报。

2.2 软件部分

软件部分主要就是手机上的控制app,手机的app包括可以选择添加日程的按钮,可以输入信息的文本框,可以将文本框内的内容发送到单片机的按钮。同时要可以在app上查看单片机终端的已录入的日程内容,以及加入一个可以删除已录入内容的按钮。文本框分为三部分,一部分输入文本内容(两到八个汉字字符左右的长度即可。)输入开始时间的文本框,输入结束时间的文本框。

2.3 功能总结

(1)STM32采用正点原子mini板。正点原子的2.8寸tft触摸屏

(2)pcf8563t用作时钟计时,并把日期和时间显示在屏幕上

(3)DHT11温湿度传感器检测环境信息,并把信息显示在液晶屏幕上

(4)使用esp8266作WiFi模块与手机app进行通信;

(5)w25Q64 烧录字库,存放字库,存放日程提醒信息
实现思路: 将W25Q64安装FATFS文件系统,方便数据存放读取,读写日程信息,字库信息。

(6)syn6288语音芯片合成的语音信息,通过喇叭播报实现提醒功能

(7)开发手机APP输入提醒日程,单片机接收到手机上发送的日程内容。
单片机将接收到的内容存储在w25Q64内,同时可以在屏幕上指定位置将内容显示出来。内容包括具体日程的文字内容以及日程开始、结束的时间;日程信息采用文件形式存储,修改、读写都针对文件进行操作。

3. 软件运行效果

软件打开之后先输入设备端的IP地址和端口,连接成功之后就可以进行功能操作。

软件上有日程表查看页面(也就是主页面)、日志页面、新增日程提醒页面。

软件分为windows桌面版本和Android手机版本,下面演示的截图以windows桌面版本为例。

软件采用QT设计,Qt Creator是跨平台的 Qt IDE, Qt Creator 是 Qt 被 Nokia 收购后推出的一款新的轻量级集成开发环境(IDE)。此 IDE 能够跨平台运行,支持的系统包括 Linux(32 位及 64 位)、Mac OS X 以及 Windows。

Qt Creator官网下载地址:Download Qt | Embedded System | Real Time Embedded Systems | Qt

QT所有版本下载地址:Index of /archive/qt

QT环境搭建,入门开发专栏: https://blog.csdn.net/xiaolong1126626497/category_11400392.html

1)日程表查看页面(也就是主页面),查看日程提醒事件,点击更新日程按钮,可以从设备端获取最新的数据过来。

(2)日志页面用来查看软件与设备间交互的过程,可以调试了解发送的数据是否正常。

(3)新增日程提醒页面

在这个页面上可以填入提醒的事件内容,输入提醒的起始时间、结束时间,提前提醒的时,输入完毕后,点击新增提醒事件按钮,就可以将数据发送给设备端,并且在主页面添加数据显示。

(4)Android手机运行效果

4. 通信协议

设备端与软件上位机之间数据交互的的协议:

(1)
软件上位机对STM32发送:
#update          让STM32发送当前存储的所有日程数据过来

STM32向上位机返回的数据格式:
$update,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)
$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5

(2)
给STM32发送一条日程数据过去
格式:$add,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)
$add,2022/02/22 13:15,2022/02/23 12:17,吃饭,5

(3)
给STM32发送校准时间  *20220222131338

(4)
删除STM32上存储的日程数据
$del,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)
$del,2022/02/22 13:15,2022/02/23 12:17,吃饭,5

5. 测试流程总结

设备端采用ESP8266与上位机进行通信,ESP8266上电初始化为AP+TCP服务器模式,设置固定端口号。

采用电脑或者手机运行APP测试之前,先搜索ESP8266创建的WIFI热点连接上,然后打开软件,在软件里输入ESP8266服务器的IP地址和端口号点击连接,连接成功之后就可以与设备端进行交互。

如果没有设备端,也可以采用网络调试助手与上位机之前交互,测试功能。

6. 硬件部分

6.1 硬件实物

板子的串口正常提示:

(1)上电提示

(2) 更新事件提示

(3)SD卡上生成的文件

6.2 外设硬件连线

(1) ESP8266 WIFI
PB10--->ESP8266-RX
PB11--->ESP8266-TX
3.3v--->VCC
GND---->GND
(2) SYN6628
PA2(TX)---SYN6628-RX
PA3(RX)---SYN6628-TX
3.3v---->VCC
GND----->GND
(3) DHT11 温湿度传感器
PA5 ---->DHT11-OUT
3.3v---->VCC
GND----->GND

剩下的用的硬件是开发板本身自带--正点原子STM32F1战舰V3开发板,硬件连接详情看原理图接口。

6.3 字库创建

6.4 SD卡上存放的字库文件

7. 设备端核心代码及实现思路

如果需要整个工程直接使用,可以去这里获取: https://download.csdn.net/download/xiaolong1126626497/85892788

7.1 字库读取

目前设备端LCD屏字库存放在SD卡上,通过fatfs文件系统读取字模进行显示,这样做的优点: 更换字库方便,直接把SD卡拔出来放在电脑上拷贝字库即可。

核心代码如下:

void NT35310_DisplayGBKData(u32 x,u32 y,u32 size,u8 *p,u16 c1,u16 c2)
{
      FIL fp;
	  UINT br;
	  u8 L,H;
	  u32 Addr;
	  u16 font_size=size/8*size; //字体占用的点阵码字节大小
	  u8 *buff=NULL;
		H=*p;
		L=*(p+1);
		if(L<0x7f)L=L-0x40;
		else L=L-0x41;
		H=H-0x81;
		Addr=(190*H+L)*font_size; //中文在字库里的偏移量
		buff=malloc(font_size);   //使用的堆空间
		if(buff==NULL)return;

		switch(size)
		{
			case 16:
				if(f_open(&fp,"0:SYSTEM/FONT/GBK16-H.DZK",FA_READ)!=FR_OK)
                {
                      printf("f_open error.\r\n");
                }
				f_lseek(&fp,Addr);
				f_read(&fp,buff,font_size,&br);
				f_close(&fp);
 
				break;
			case 24:
                f_open(&fp,"0:SYSTEM/FONT/GBK24-H.DZK",FA_READ);
				f_lseek(&fp,Addr);
				f_read(&fp,buff,font_size,&br);
				f_close(&fp);
				break;
			case 32:
 
				break;
		}
		//显示中文
		NT35310_DisplayData(x,y,size,size,buff,c1,c2);
 
		//释放空间
		free(buff);
}
void NT35310_DisplayData(u32 x,u32 y,u32 w,u32 h,u8 *p,u16 c1,u16 c2)
{
   u16 i,j,x0=x;
   u8 data;
    u16 colortemp=POINT_COLOR;    
   for(i=0;i<w/8*h;i++) //取出的模型总字节数
   {
       data=p[i]; //取出数组里一个字节的数据
 
       for(j=0;j<8;j++)
       {
          if(data&0x80)Draw_Point(x0,y,c1); //字体颜色    
          else 
          {
              Draw_Point(x0,y,c2); //背景颜色
          }
          data<<=1; //继续判断下一位
          x0++; //继续画下一个点
       }
       if(x0-x==w) //判断是否需要换行
       {
          x0=x;//横坐标归位
          y++; //纵坐标自增
       }
   }
   POINT_COLOR=colortemp;	  
}

7.2 解析ESP8266数据

主函数里通过轮询方式检测,ESP8266是否收到上位机的命令,收到之后进行解析处理

核心代码如下:

//ESP8266 WIFI 返回的数据
if(USART3_RX_FLAG)
{
	USART3_RX_BUFFER[USART3_RX_CNT]='\0';
	printf("%s",USART3_RX_BUFFER);

	//解析WIFI返回的数据
 
	//如果是校准RTC时间 +IPD,0,15:*20220304220552
	if(strstr((char*)USART3_RX_BUFFER,":*"))
	{
		 printf("校准时间.\r\n");
		 rtc_time_update((char*)USART3_RX_BUFFER);
	}
	//如果是请求更新提醒
	else if(strstr((char*)USART3_RX_BUFFER,"#update"))
	{
		printf("请求更新事件.\r\n");
		update_enev();
	}
	//如果是新增提醒
	//+IPD,0,49:$add,2022/03/04 21:56,2022/03/04 21:56,
	else if(strstr((char*)USART3_RX_BUFFER,"$add"))
	{
		 printf("新增提醒事件.\r\n");
		 add_enev((char*)USART3_RX_BUFFER);
	}
	//如果是删除某个提醒
	//+IPD,0,49:$del,2022/03/04 21:56,2022/03/04 21:56,水水水水,0
	else if(strstr((char*)USART3_RX_BUFFER,"$del"))
	{
		 printf("删除某个提醒.\r\n");
		del_enev((char*)USART3_RX_BUFFER);
	}

	USART3_RX_CNT=0;
	USART3_RX_FLAG=0;
}

7.3 向SD卡存放事件信息

事件提醒都是存放在SD卡上,以文件的形式存放,上面封装的几个函数里,主要是就是读写文件。

核心代码:

/*
函数功能:从buf里面得到第cnt个逗号所在的位置
返 回 值:0~254,代表逗号所在位置的偏移.
255,代表不存在第cnt个逗号
*/
u8 GetCommaOffset(char *buf,u8 cnt)
{
	char *p=buf;
	while(cnt)
	{
		if(*buf==',')cnt--;
		buf++;
	}
	return buf-p; //计算偏移量
}
/*
写文件
*/
void FATFS_Write(const TCHAR *FileName,const char *WriteBuff)
{
    FIL fp;
    FRESULT res;
    UINT cnt; //存放写入成功的数量
 
    /*1. 创建文件*/  
    res=f_open(&fp,FileName, FA_WRITE | FA_CREATE_ALWAYS);
    if(res!=0)
    {
        printf("%s文件创建失败!\n",FileName);
        return;
    }
 
    /*2. 写入数据*/
    res=f_write(&fp,WriteBuff,strlen(WriteBuff),&cnt);
    if(res!=0)
    {
        printf("%s文件写入失败!\n",FileName);
        return;
    }
 
    /*3. 关闭文件*/
    f_close(&fp);
 
    printf("%s文件创建成功,成功写入:%d\n",FileName,cnt);
}

//提取指定逗号位置的数据
void GetCommaOffsetBuff(char *buf_in,char *buf_out,u8 cnt)
{
	while (cnt)
	{
		if (*buf_in == ',')cnt--;
		if (*buf_in == '\0')break;
		buf_in++;
	}
 
	while (*buf_in != ',' && *buf_in != '\0')
	{
		*buf_out=*buf_in;
		buf_in++;
		buf_out++;
	}
	*buf_out = '\0';
}

//字符串替换
//sub1替换前 sub2替换后字符
void StringSubstitution(char *p,char sub1,char sub2)
{
	while (*p!='\0')
	{
		if (*p == sub1)
		{
			*p = sub2;
		}
		p++;
	}
}
//新增提醒
 //+IPD,0,49:$add,2022/03/04 21:56,2022/03/04 21:56,
void add_enev(char *p)
{
   char buf_out[20];
	u8 offset = 0;
	char *p2;
	offset = GetCommaOffset(p, 3);
	p2 = p + offset;
	printf("%d,%s\r\n", offset, p2);

	//提取提醒的起始时间 例如:2022/03/04 21:56
	GetCommaOffsetBuff(p,buf_out,3);
	printf("buf_out1=%s\r\n",buf_out);

	StringSubstitution(buf_out,'/','-');
	StringSubstitution(buf_out, ' ','-');
	StringSubstitution(buf_out, ':','-');
	printf("buf_out2=%s\r\n", buf_out);

	strcat(buf_out,".ev");
	printf("buf_out3=%s\r\n", buf_out);

	FATFS_Write(buf_out, p2);
 
    //更新事件列表
    update_event_list();
}
//删除某个提醒
 //+IPD,0,49:$del,2022/03/04 21:56,2022/03/04 21:56,水水水水,0
void del_enev(char *p)
{
    char buf_out[20];
    char dir_file_path[50];
	//提取提醒的起始时间 例如:2022/03/04 21:56
	GetCommaOffsetBuff(p, buf_out, 3);
	printf("buf_out1=%s\n", buf_out);

	StringSubstitution(buf_out, '/', '-');
	StringSubstitution(buf_out, ' ', '-');
	StringSubstitution(buf_out, ':', '-');
	printf("buf_out2=%s\n", buf_out);

	strcat(buf_out, ".ev");
	printf("buf_out3=%s\n", buf_out);
 
    //拼接目录名称
    sprintf(dir_file_path,"0:/%s",buf_out);
 
    //删除文件
    if(f_unlink(dir_file_path)==FR_OK)
    {
        printf("%s\r\n删除成功.",dir_file_path);
    }
    else
    {
        printf("%s\r\n删除失败.",dir_file_path);
    }    
 
    //更新事件列表
    update_event_list();
}
//向APP终端更新提醒
//+IPD,0,7:#update
/*
STM32向上位机返回的数据格式:
$update,起始时间,结束时间,事件内容,提前提醒时间(0~59分钟)
$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5
*/
u8 update_enev(void)
{
//    u8 buff[]="$update,2022/02/22 13:15,2022/02/23 12:17,吃饭,5";
//    ESP8266_ServerSendData(0,buff,strlen((char*)buff));
 
    char *path="0:/"; //目录位置
    DIR dir;
    FIL file; //文件指针
    FRESULT res; 
    FILINFO fno; //存放读取的文件信息
    char *abs_path=NULL;  
    char cmd[]="$update,"; //请求的命令头
    char out_buff[100];
    UINT cnt;
 
    strcpy(out_buff,cmd);
 
    /*1. 打开目录*/    
    res=f_opendir(&dir,path);
    if(res!=FR_OK)return res;
 
    /*2. 循环读取目录*/
     while(1)
     {
        res=f_readdir(&dir,&fno);
        if(fno.fname[0] == 0 || res!=0)break;
        //printf("文件名称: %s,文件大小: %ld 字节\r\n",fno.fname,fno.fsize);

        /*过滤目录*/
        if(strstr(fno.fname,".ev"))
        {
            //申请存放文件名称的长度
            abs_path=malloc(strlen(path)+strlen(fno.fname)+1);
            if(abs_path==NULL)break;
 
            strcpy(abs_path,path);
            strcat(abs_path,"/");
            strcat(abs_path,fno.fname);
            //读取文件数据
            f_open(&file,abs_path,FA_READ);
            f_read(&file,out_buff+strlen(cmd),100,&cnt);
            free(abs_path);      
            printf("abs_path=%s,读取:%d\r\n",abs_path,cnt);
            out_buff[cnt+strlen(cmd)]='\0';
 
            //发送给上位机
            ESP8266_ServerSendData(0,(u8*)out_buff,strlen((char*)out_buff));
            printf("发送:%s\r\n",out_buff);
            delay_ms(100);
        }
    }
 
    /*3. 关闭目录*/
    f_closedir(&dir);
    return 0;
}
//提取数据,存放到全局事件结构体里
//参数: i 索引值  buf_out 源数据内容
void ExtractData(int i,char *buf_out)
{
	//std_to_sec(u16 syear, u8 smon, u8 sday, u8 hour, u8 min)

	char buf_num[50];
 
    //格式化处理
	StringSubstitution(buf_out, '/', ',');
	StringSubstitution(buf_out, ' ', ',');
	StringSubstitution(buf_out, ':', ',');
	printf("buf_out=%s\r\n",buf_out);

	//2022-03-05-10-11-2022-03-05-10-11-打酱油-5

	//1. 提取起始时间
	GetCommaOffsetBuff(buf_out,buf_num,0);
	event_buff[i].s_year = atoi(buf_num);
	printf("起始-年: %d,%s\r\n", event_buff[i].s_year, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 1);
	event_buff[i].s_month = atoi(buf_num);
	printf("起始-月: %d,%s\r\n", event_buff[i].s_month, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 2);
	event_buff[i].s_date = atoi(buf_num);
	printf("起始-日: %d,%s\r\n", event_buff[i].s_date, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 3);
	event_buff[i].s_hour = atoi(buf_num);
	printf("起始-时: %d,%s\r\n", event_buff[i].s_hour, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 4);
	event_buff[i].s_min = atoi(buf_num);
	printf("起始-分: %d,%s\r\n", event_buff[i].s_min, buf_num);

	//2. 提取结束时间
	GetCommaOffsetBuff(buf_out, buf_num, 5);
	event_buff[i].e_year = atoi(buf_num);
	printf("结束-年: %d,%s\r\n", event_buff[i].e_year, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 6);
	event_buff[i].e_month = atoi(buf_num);
	printf("结束-月: %d,%s\r\n", event_buff[i].e_month, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 7);
	event_buff[i].e_date = atoi(buf_num);
	printf("结束-日: %d,%s\r\n", event_buff[i].e_date, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 8);
	event_buff[i].e_hour = atoi(buf_num);
	printf("结束-时: %d,%s\r\n", event_buff[i].e_hour, buf_num);

	GetCommaOffsetBuff(buf_out, buf_num, 9);
	event_buff[i].e_min = atoi(buf_num);
	printf("结束-分: %d,%s\r\n", event_buff[i].e_min, buf_num);

	//3. 提取事件内容
	GetCommaOffsetBuff(buf_out, buf_num,10);
	strcpy((char*)event_buff[i].event_buff, buf_num);
	printf("事件内容:%s\r\n", event_buff[i].event_buff);

	//4. 提前提醒时间
	GetCommaOffsetBuff(buf_out, buf_num, 11);
	event_buff[i].time_min_advance = atoi(buf_num);
	printf("提前提醒时间:%d分钟\r\n", event_buff[i].time_min_advance);

	//5. 设置标志位
	event_buff[i].flag = 1;
}
//最多15个事件提醒
struct Event event_buff[15];

//更新事件列表
u8 update_event_list(void)
{
    char *path="0:/"; //目录位置
    DIR dir;
    FIL file; //文件指针
    FRESULT res; 
    FILINFO fno; //存放读取的文件信息
    char *abs_path=NULL;  
    char out_buff[100];
    UINT cnt;
    int index=0; //数组索引
 
    //先将数组全部清0
    memset(event_buff,0,sizeof(event_buff));
 
    /*1. 打开目录*/    
    res=f_opendir(&dir,path);
    if(res!=FR_OK)return res;
 
    /*2. 循环读取目录*/
     while(1)
     {
        res=f_readdir(&dir,&fno);
        if(fno.fname[0] == 0 || res!=0)break;
        //printf("文件名称: %s,文件大小: %ld 字节\r\n",fno.fname,fno.fsize);

        /*过滤目录*/
        if(strstr(fno.fname,".ev"))
        {
            //申请存放文件名称的长度
            abs_path=malloc(strlen(path)+strlen(fno.fname)+1);
            if(abs_path==NULL)break;
 
            strcpy(abs_path,path);
            strcat(abs_path,"/");
            strcat(abs_path,fno.fname);
            //读取文件数据
            f_open(&file,abs_path,FA_READ);
            f_read(&file,out_buff,100,&cnt);
            free(abs_path); 
            out_buff[cnt]='\0'; //添加结束符            
            printf("abs_path=%s,读取:%d,%s\r\n",abs_path,cnt,out_buff);
 
            //解析数据存放到数组里
            //2022/03/05 10:11,2022/03/05 10:11,打酱油,5
            ExtractData(index++,out_buff);
            if(index>=15)break; //最大存放15个事件
        }
    }
 
    /*3. 关闭目录*/
    f_closedir(&dir);
    return 0;
}

7.4 时间更新

RTC时间更新,通过上位机发送时间进行更新。

核心代码:

void rtc_time_update(char *buff)
{
    char *time;
    /*判断是否收到客户端发来的数据  */
    char *p=strstr((char*)buff,"+IPD");
    if(p!=NULL) //正常数据格式: +IPD,0,7:LED1_ON    +IPD,0表示第0个客户端   7:LED1_ON表示数据长度与数据
    {
            /*解析上位机发来的数据*/
            p=strstr((char*)buff,":");
            if(p!=NULL)
            {
                    p+=1; //向后偏移1个字节
                    if(*p=='*')  //设置RTC时间
                    {
                            p+=1; //向后偏移,指向正确的时间
                            time=p;
                            calendar.w_year=(time[0]-48)*1000+(time[1]-48)*100+(time[2]-48)*10+(time[3]-48)*1;
                            calendar.w_month=(time[4]-48)*10+(time[5]-48)*1;
                            calendar.w_date=(time[6]-48)*10+(time[7]-48)*1;
                            calendar.hour=(time[8]-48)*10+(time[9]-48)*1;
                            calendar.min=(time[10]-48)*10+(time[11]-48)*1;
                            calendar.sec=(time[12]-48)*10+(time[13]-48)*1;
 
                        RTC_Set(calendar.w_year,
                                    calendar.w_month,
                        calendar.w_date,
                        calendar.hour,
                        calendar.min,
                        calendar.sec);
 
                        printf("时间设置成功:%s\r\n",buff);
                    }
            }
    }
}

7.5 LCD屏几个主要页面

主循环里通过轮询按键,检测是否需要切换显示页面。

目前设计有3个主页面+N个事件显示页面。

(1)页面1: 模拟电子时钟页面

(2)页面2:日历显示页面

(3)页面3-N : 待办事件显示页面

7.6 RTC时钟

RTC开启了秒中断,在秒中断里绘制模拟时钟页面,更新当前的系统时间。

7.7 事件时间判断页面

主函数里使用定时器1,以10秒的频率,判断待办事件时间是否到达,是否需要语音播报。

/*
函数功能: 定时器1的更新中断服务函数
*/
#include "app.h"
#include "rtc.h"
#include "syn6628.h"
u32 time_ev_cnt=0;
void TIM1_UP_IRQHandler(void)
{
    int i=0;
    u32 t1=0;
    u32 t2=0;
    u32 t3=0;
    char dir_file_path[50];
    if(TIM1->SR&1<<0)
    {
      TIM1->SR&=~(1<<0);
    }
 
    time_ev_cnt++;

    //10秒时间
    if(time_ev_cnt>=2)
    {
        time_ev_cnt=0;
            printf("10秒时间到达.\r\n");
 
            for(i=0;i<15;i++)
            {
                if(event_buff[i].flag==0)break;
 
                //当前时间
                t1=std_to_sec(calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min);
                //提醒开始时间
                t2=std_to_sec(event_buff[i].s_year,event_buff[i].s_month,event_buff[i].s_date,event_buff[i].s_hour,event_buff[i].s_min);
                //提醒结束时间
                t3=std_to_sec(event_buff[i].e_year,event_buff[i].e_month,event_buff[i].e_date,event_buff[i].s_hour,event_buff[i].e_min);
 
                //加上提前提醒的分钟
                t2+=event_buff[i].time_min_advance*60;
 
                //判断有没有时间满足需求,满足就语音提示
                if(t1>=t2 && t1<=t3)
                {
                    //时间达到,语音播报
                    SYN6288_Speech("您有待开始的日程,请注意");
 
                    //如果需要播放具体的内容,直接播放结构体里的成员即可
                }
 
                //如果时间超过,就可以删除事件
                if(t1>=t3)
                {
                    //删除SD卡上的这个文件
                    //拼接路径
                    sprintf(dir_file_path,"0:/%d-%d-%d-%d-%d.ev",event_buff[i].s_year,event_buff[i].s_month,event_buff[i].s_date,event_buff[i].s_hour,event_buff[i].s_min);
                    if(f_unlink(dir_file_path)==FR_OK)
                    {
                        printf("%s\r\n删除成功.",dir_file_path);
                         //更新事件列表
                        update_event_list();
                    }
                    else
                    {
                        printf("%s\r\n删除失败.",dir_file_path);
                    }  
                }
            }
    }
}

点击关注,第一时间了解华为云新鲜技术~

相关文章