×
嵌入式 > 嵌入式开发 > 详情

基于STM32的多功能数字钟

发布时间:2020-09-02 发布时间:
|
/**************************************************************************************
程序功能:基于STM32的多功能数字钟
**************************************************************************************
1、时钟的基准用STM32F103RCT6自带的RTC实现。
2、这三个按键的功能分别为:"设置"、"+"、"-"。(WAKEUP KEY0 KEY1)
三个按键作为这三个功能:
键盘上的WAKEUP用作"设置",KEY0用作"+",KEY1用作"-"。
设定时间、日期、星期时,按下设定键,
相应的参数会闪动,即一亮一灭,提示正的修改的数据,每按一下设置键,
设定的项目就会变化,顺序为:
秒--分--时--星期--日--月--年。
顺序设定完后,再按一下设定键,就退出设定状态,回到正显示状态。
3、用DS18B20显示温度。精确到小数点后一位。如:33.5℃。
4、增加用红外遥控器的CH CH+ CH- 三个按键进行时间设置,
功能等同于键盘上的(WAKEUP KEY0 KEY1)"设置"、"+"、"-"。
5、通过串行口设置时间日期。
如串行口发送t12:30:00 设置时间;
如串行口发送d2016-08-12 设置日期;
6、通过触摸屏进行时间日期设置
在触摸屏上有 SET + -三个标志,当用触笔去按时,就会产生同开发板上的
按键一样的效果。
触摸按下去和弹起来时,按键的颜色会发生变化。
**************************************************************************************
硬件:LED0----GPIOA.8
KEY0----GPIOC.5
KEY1----GPIOA.15
KEY_WAKEUP----GPIOA.0
TIMER2定时器定时功能。
TIMER5定时器的定时和捕获功能
TFT液晶屏2。8寸屏带触摸功能。
TOUCH触摸屏。
DS18B20---PC4;
HS138红外遥控---PA1;
串行口USART1;
RTC实时时钟
开发板:正点原子MINI开发板
**************************************************************************************
编程目的:让大赛学生尽快学会STM32的综合编程。
**************************************************************************************
编程软件:KEIL5
**************************************************************************************
常用的算法:
按键采用状态机实现按键扫描。在定时器中对按键进行处理。
红外部分采用定时器捕获功能实现。
时间用STM32自带的RTC实现;
**************************************************************************************
时间:2016.8.12
作者:王庐山
工作室:湖北工业职业技术学院电子创新中心
***************************************************************************************/
#include "stm32f10x_conf.h"
#include "delay.h"
#include "sys.h"
#include "LED.h"
#include "key.h"
#include "timer2.h"
#include "24cxx.h"
#include "lcd.h"
#include "touch.h"
#include "usart.h"
#include "rtc.h"
#include "ds18b20.h"
#include "remote.h"
//时间结构体
typedef struct
{
vu8 s_hour;
vu8 s_min;
vu8 s_sec;
//公历日月年周
vu16 s_year;
vu8 s_month;
vu8 s_date;
vu8 s_week;
}Timer;
Timer S_timer;
unsigned char flag_20ms=0x0,flag_500ms=0,flag_250ms=0,flag_1s=0;//定时标志
unsigned char flag_key0=0,flag_key1=0,flag_key2=0,flag_tiao=0,key_remot=0;
void Disp_Tiao_Shan(void);
void Load_Drow_Dialog(void);
int main(void)
{
unsigned char temp[12],sec_back=0xff,t=0,len=0;
short temp5;
delay_init();
NVIC_Configuration();
uart_init(9600);
Led_Init();
key_init();
Timer2_config(71,2000); //定时两毫秒
DS18B20_Init();
Remote_Init();
AT24CXX_Init();
RTC_Init();
LCD_Init();
LCD_Clear(BLUE);
TP_Init();
POINT_COLOR=RED;
BACK_COLOR=BLUE;
LCD_ShowString(10,50,220,24,24,"--MULT FUNC CLOCK--");
LCD_ShowString(60,260,180,24,24,"SET + - ");
// POINT_COLOR=YELLOW ;
LCD_DrawRectangle(50,250,100,290);
LCD_DrawRectangle(110,250,160,290);
LCD_DrawRectangle(170,250,220,290);
while(1)
{
if(tp_dev.sta&TP_PRES_DOWN)//有触摸按下状态。
{//按下时,方框的颜色要发生变化,这样看效果才明显示,
//手松开后,又要恢复原画的颜色。
if((tp_dev.x[0]>50)&&(tp_dev.x[0]<100)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{LCD_Fill(51,251,99,289,YELLOW);LCD_ShowString(60,260,36,24,24,"SET");}
else if((tp_dev.x[0]>110)&&(tp_dev.x[0]<160)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{LCD_Fill(111,251,159,289,YELLOW);LCD_ShowString(120,260,36,24,24," + ");}
else if((tp_dev.x[0]>170)&&(tp_dev.x[0]<220)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{LCD_Fill(171,251,219,289,YELLOW);LCD_ShowString(180,260,36,24,24," - ");}
}
if((USART_RX_STA&0x8000))//收到了一帧串行口数据,进行解析。
{
switch(USART_RX_BUF[0])
{
case t:{//time是调时间
S_timer.s_hour=(USART_RX_BUF[1]-0)*10+(USART_RX_BUF[2]-0);
S_timer.s_min=(USART_RX_BUF[3]-0)*10+(USART_RX_BUF[4]-0);
S_timer.s_sec=(USART_RX_BUF[5]-0)*10+(USART_RX_BUF[6]-0);
USART_RX_STA=0;
printf("set time:%s",USART_RX_BUF);
calendar.sec=S_timer.s_sec;
calendar.hour=S_timer.s_hour;
calendar.min=S_timer.s_min;
calendar.w_date=S_timer.s_date;
calendar.w_month=S_timer.s_month;
calendar.w_year=S_timer.s_year;
calendar.week=S_timer.s_week;
RTC_Set(calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min,calendar.sec);
break;
}
case d:{//date是调日期
S_timer.s_year =(USART_RX_BUF[1]-0)*1000+(USART_RX_BUF[2]-0)*100+(USART_RX_BUF[3]-0)*10+(USART_RX_BUF[4]-0);
S_timer.s_month=(USART_RX_BUF[5]-0)*10+(USART_RX_BUF[6]-0);
S_timer.s_date=(USART_RX_BUF[7]-0)*10+(USART_RX_BUF[8]-0);
USART_RX_STA=0;
printf("set date:%s",USART_RX_BUF);
calendar.sec=S_timer.s_sec;
calendar.hour=S_timer.s_hour;
calendar.min=S_timer.s_min;
calendar.w_date=S_timer.s_date;
calendar.w_month=S_timer.s_month;
calendar.w_year=S_timer.s_year;
calendar.week=S_timer.s_week;
RTC_Set(calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min,calendar.sec);
break;
}
default :break;
}
}
if(flag_tiao!=0)//用于调节时闪烁显示
{
switch(flag_tiao)
{
case 1: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*13,140,24,24,24," ");
break;
}
else goto kk2;
}
case 2: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*10,140,24,24,24," ");
break;
}
else goto kk2;
}
case 3: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*7,140,24,24,24," ");
break;
}
else goto kk2;
}
case 4: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*15,100,24,24,24," ");
break;
}
else goto kk2;
}
case 5: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*12,100,24,24,24," ");
break;
}
else goto kk2;
}
case 6: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*10,100,24,24,24," ");
break;
}
else goto kk2;
}
case 7: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*5,100,48,24,24," ");
break;
}
else goto kk2;
}
kk2:
{
sprintf(temp,"DATE:%d-%d-%d %d",calendar.w_year,calendar.w_month,calendar.w_date,calendar.week);
LCD_ShowString(30,100,210,24,24,temp);
sprintf(temp,"TIME: %d:%d:%d ",calendar.hour,calendar.min,calendar.sec);
LCD_ShowString(30,140,210,24,24,temp);
break;
}
default :break ;
}
}
else//正常显示
{
if(sec_back!=calendar.sec)
{
sprintf(temp,"DATE:%d-%d-%d %d",calendar.w_year,calendar.w_month,calendar.w_date,calendar.week);
LCD_ShowString(30,100,210,24,24,temp);
sprintf(temp,"TIME: %d:%d:%d ",calendar.hour,calendar.min,calendar.sec);
LCD_ShowString(30,140,210,24,24,temp);
}
}
keyfunction();//按键扫描和功能解析(包括了开发板上的按键、红外遥控按键、触摸按键
if(flag_1s ==1)//每秒读取一次温度
{
flag_1s =0;
LCD_ShowString(30+5*12,180,210,24,24," ");
temp5=DS18B20_Get_Temp();
sprintf(temp,"TEMP:%d.%d",temp5/10,temp5%10);
LCD_ShowString(30,180,210,24,24,temp);
}
}
}
/*
函数功能:显示画图框
*/
void Load_Drow_Dialog(void)
{
LCD_Clear(RED);//清屏
POINT_COLOR=BLUE;//设置字体为蓝色
LCD_ShowString(lcddev.width-30,0,200,24,24,"RST");//显示清屏区域
}
void Disp_Tiao_Shan(void)
{
unsigned char temp[12];
switch(flag_tiao)
{
case 1: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*13,140,24,24,24," ");
break;
}
else goto kk2;
}
case 2: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*10,140,24,24,24," ");
break;
}
else goto kk2;
}
case 3: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*7,140,24,24,24," ");
break;
}
else goto kk2;
}
case 4: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*15,100,24,24,24," ");
break;
}
else goto kk2;
}
case 5: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*12,100,24,24,24," ");
break;
}
else goto kk2;
}
case 6: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*10,100,24,24,24," ");
break;
}
else goto kk2;
}
case 7: {
if(flag_250ms==1)
{
LCD_ShowString(30+12*5,100,48,24,24," ");
break;
}
else goto kk2;
}
kk2:
{
sprintf(temp,"DATE:%d-%d-%d %d",calendar.w_year,calendar.w_month,calendar.w_date,calendar.week);
LCD_ShowString(30,100,210,24,24,temp);
sprintf(temp,"TIME: %d:%d:%d ",calendar.hour,calendar.min,calendar.sec);
LCD_ShowString(30,140,210,24,24,temp);
break;
}
default :break ;
}
}
#include "sys.h"
#include "delay.h"
#include "key.h"
#include "rtc.h"
#include "remote.h"
#include "touch.h"
#include "lcd.h"
typedef struct
{
vu8 s_hour;
vu8 s_min;
vu8 s_sec;
//公历日月年周
vu16 s_year;
vu8 s_month;
vu8 s_date;
vu8 s_week;
}Timer;
extern unsigned char flag_20ms;
extern unsigned char flag_key0,flag_key1,flag_key2,flag_tiao;
extern Timer S_timer;
void key_init(void)
{
GPIO_InitTypeDef KEY_InitTypeDef;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOD|RCC_APB2Periph_GPIOC,ENABLE);
KEY_InitTypeDef.GPIO_Pin=GPIO_Pin_0;
KEY_InitTypeDef.GPIO_Mode=GPIO_Mode_IPD;
KEY_InitTypeDef.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&KEY_InitTypeDef);
KEY_InitTypeDef.GPIO_Pin=GPIO_Pin_5;
KEY_InitTypeDef.GPIO_Mode=GPIO_Mode_IPU;
GPIO_Init(GPIOC,&KEY_InitTypeDef);
KEY_InitTypeDef.GPIO_Pin=GPIO_Pin_15;
KEY_InitTypeDef.GPIO_Mode=GPIO_Mode_IPU;
GPIO_Init(GPIOA,&KEY_InitTypeDef);
KEY0=1;
KEY1=1;
WK_UP=0;
}
/*
函数功能:按键扫描函数
输入参数:无
输出参数:返回键码。无按键时返回0;
外部变量:
算法:采用软件延时实现按键扫描。
*/
unsigned char KeyScan(void)
{
unsigned char keynum=0;
if((KEY0!=1)||(KEY1!=1)||(WK_UP!=0))
{
delay_ms(10);
if((KEY0!=1)||(KEY1!=1)||(WK_UP!=0))
{
if(KEY0!=1)
{
while(KEY0!=1);
keynum=1;return keynum;}
else if(KEY1!=1)
{while(KEY1!=1);
keynum=2;return keynum;}
else if(WK_UP!=0)
{while(WK_UP!=0);
keynum=3;return keynum;}
else
return 0;
}
else
return keynum;
}
else
return keynum;
}
/*
函数功能:按键扫描函数
输入参数:无
输出参数:返回键码。无按键时返回0;
外部变量:flag_20ms
算法:采用状态机实现按键扫描。
*/
unsigned char KeyScan2(void)
{
static unsigned char keynum=0,step1=0;
if(flag_20ms==0x01)
{
flag_20ms=0;
switch(step1)
{
case 0:{//状态1:检测是否有按键按下
if((KEY0!=1)||(KEY1!=1)||(WK_UP!=0))
step1=1;
else
step1=0;
return 0;
//break;
}
case 1:{//状态2:检测是否真的有按键按下
if((KEY0!=1)||(KEY1!=1)||(WK_UP!=0))
{
if(KEY0!=1)
{step1=2; keynum=1;}
else if(KEY1!=1)
{step1=2; keynum=2;}
else if(WK_UP!=0)
{step1=2; keynum=3;}
else
{step1=0; }
}
return 0;
//break;
}
case 2:{//状态3:检测按键是否弹起
if((KEY0!=1)||(KEY1!=1)||(WK_UP!=0))
{step1=2; return 0;}
else
{ step1=0;return keynum;}
//break;
}
default :return 0;
}
}
return 0;
}
/*
函数功能:按键功能解析函数
输入参数:无
输入函数:KeyScan2();
输出参数:无
外部变量:flag_key0,flag_key1
*/
void keyfunction(void)
{
unsigned char temp;
temp=(KeyScan2())|(Remote_Scan());
if(TP_Scan(0))//当触摸按键按下并松开后才返回1,否则返回0;
{//按下时,方框的颜色要发生变化,这样看效果才明显示,
//手松开后,又要恢复原画的颜色。
if((tp_dev.x[0]>50)&&(tp_dev.x[0]<100)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{ temp =3;
{LCD_Fill(51,251,99,289,BLUE);LCD_ShowString(60,260,180,24,24,"SET");}
}
else if((tp_dev.x[0]>110)&&(tp_dev.x[0]<160)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{ temp=1;
{LCD_Fill(111,251,159,289,BLUE);LCD_ShowString(120,260,180,24,24," + ");}
}
else if((tp_dev.x[0]>170)&&(tp_dev.x[0]<220)&&(tp_dev.y[0]>250)&&(tp_dev.y[0]<290))
{temp=2;
{LCD_Fill(171,251,219,289,BLUE);LCD_ShowString(180,260,180,24,24," - ");}
}
}
if(temp>0)
{ TIM_Cmd(TIM2,ENABLE);
switch(temp)
{
case 1:{
flag_key0=1;
if(flag_tiao>0)
{
S_timer.s_sec =calendar.sec;
S_timer.s_hour=calendar.hour;
S_timer.s_min =calendar.min;
S_timer.s_date=calendar.w_date;
S_timer.s_month =calendar.w_month;
S_timer.s_year=calendar.w_year;
S_timer.s_week =calendar.week;
switch(flag_tiao)
{
case 1: {
if((S_timer.s_sec)<60);
else
S_timer.s_sec=0;
goto kk1;
}
case 2: {
if(S_timer.s_min++<60);
else
S_timer.s_min=0; goto kk1;
}
case 3: {
if(S_timer.s_hour ++<60);
else
S_timer.s_hour=0; goto kk1;
}
case 4: {
if(S_timer.s_week++<7);
else
S_timer.s_week =1; goto kk1;
}
case 5: {
if(S_timer.s_date++<30);
else
S_timer.s_date=1; goto kk1;
}
case 6: {
if(S_timer.s_month++<12);
else
S_timer.s_month =1; goto kk1;
}
case 7: {
if(S_timer.s_year++<2099);
else
S_timer.s_year =1970; goto kk1;
}
default :break ;
}
}
break;
}
case 2:{
flag_key1=1;
if(flag_tiao>0)
{
S_timer.s_sec =calendar.sec;
S_timer.s_hour=calendar.hour;
S_timer.s_min =calendar.min;
S_timer.s_date=calendar.w_date;
S_timer.s_month =calendar.w_month;
S_timer.s_year=calendar.w_year;
S_timer.s_week =calendar.week;
switch(flag_tiao)
{
case 1: {
if((S_timer.s_sec)-->0);
else
S_timer.s_sec=60; goto kk1;
//break ;
}
case 2: {
if(S_timer.s_min-->0);
else
S_timer.s_min=60; goto kk1;
//break ;
}
case 3: {
if(S_timer.s_hour-->0);
else
S_timer.s_hour=24; goto kk1;
//break ;
}
case 4: {
if(S_timer.s_week-->0);
else
S_timer.s_week =7; goto kk1;
//break ;
}
case 5: {
if(S_timer.s_date-->0);
else
S_timer.s_date=31; goto kk1;
//break ;
}
case 6: {
if(S_timer.s_month-->0);
else
S_timer.s_month =12; goto kk1;
//break ;
}
case 7: {
if(S_timer.s_year-->1970);
else
S_timer.s_year =1970; goto kk1;
//break ;
}
kk1:
calendar.sec=S_timer.s_sec;
calendar.hour=S_timer.s_hour;
calendar.min=S_timer.s_min;
calendar.w_date=S_timer.s_date;
calendar.w_month=S_timer.s_month;
calendar.w_year=S_timer.s_year;
calendar.week=S_timer.s_week;
RTC_Set(calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min,calendar.sec);
break;
default :break ;
}
}
break;
}
case 3:{
flag_key2=1;
if(flag_tiao++<7);
else
flag_tiao=0;
break;
}
default :break;
}
TIM_Cmd(TIM2,ENABLE);
}
}



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
内核日志及printk结构浅析