/****************************************************************************/
/* 電子日歷,有時(shí)間顯示、鬧鈴、日期、秒表及鍵盤(pán)設置功能 */
/* 功能鍵A: 設置位數字+1 鬧鐘模式下為鬧鐘開(kāi)關(guān) 秒表模式下為記時(shí)開(kāi)關(guān) */
/* 功能鍵B: 設置位數字-1 鬧鐘模式下為鬧鐘開(kāi)關(guān) */
/* 功能鍵C:設置模式及設置位選擇 秒表模式下為清零鍵 */
/* 功能鍵D:在四種工作模式下切換 設置鬧鐘開(kāi)關(guān) */
/* 曹宇 03電子 0201029 */
/* 2006.6.3 更新 */
/****************************************************************************/
#include
#include
/***************這里設置程序初始化時(shí)顯示的時(shí)間****************/
#define SET_HOUR 12 /*設置初始化小時(shí)*/
#define SET_MINUTE 00 /*設置初始化分鐘*/
#define SET_SECOND 00 /*設置初始化秒數*/
/*************************系統地址****************************/
#define BASE_PORT 0x8000 /*選通基地址*/
#define KEY_LINE BASE_PORT+1 /*鍵盤(pán)行線(xiàn)地址*/
#define KEY_COLUMN BASE_PORT+2 /*鍵盤(pán)列線(xiàn)地址*/
#define LED_SEG BASE_PORT+4 /*數碼管段選地址*/
#define LED_BIT BASE_PORT+2 /*數碼管位選地址*/
#define LED_ON(x) XBYTE[LED_BIT]=(0x01<#define LED_OFF XBYTE[LED_SEG]=0x00 /*LED顯示空*/
/**************在設置模式下對秒分時(shí)的宏定義*****************/
#define SECOND 0 /*對應數碼管右邊兩位*/
#define MINUTE 1 /*對應數碼管中間兩位*/
#define HOUR 2 /*對應數碼管左邊兩位*/
/********************定義四種工作模式***********************/
#define CLOCK clockstr /*時(shí)鐘模式*/
#define ALART alartstr /*鬧鐘模式*/
#define DATE datestr /*日期模式*/
#define TIMER timerstr /*秒表模式*/
/****************以下是所有子函數的聲明*********************/
void sys_init(void); /*系統的初始化程序*/
void display(void); /*動(dòng)態(tài)刷新一次數碼管子程序*/
void clockplus(void); /*時(shí)間加1S的子程序*/
void update_clockstr(void); /*更新時(shí)間顯示編碼*/
void update_alartstr(void); /*更新鬧鐘時(shí)間的顯示編碼*/
void update_datestr(void); /*更新日期顯示編碼*/
void update_timerstr(void); /*更新秒表時(shí)間的顯示編碼*/
void deley(int); /*延時(shí)子程序*/
void update_dispbuf(unsigned char *); /*更新顯示緩沖區*/
unsigned char getkeycode(void); /*獲取鍵值子程序*/
void keyprocess(unsigned char); /*鍵值處理子程序*/
unsigned char getmonthdays(unsigned int,unsigned char);/*計算某月的天數子程序*/
/*功能鍵功能子函數*/
void Akey(void); /*當前設置位+1 開(kāi)關(guān)鬧鐘 開(kāi)關(guān)秒表*/
void Bkey(void); /*當前設置位-1 開(kāi)關(guān)鬧鐘 */
void Ckey(void); /*設置位選擇 秒表清零*/
void Dkey(void); /*切換四種工作模式*/
/**********************全局變量聲明部分*********************/
unsigned char led[10]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};/*從0~9的LED編碼*/
unsigned char ledchar[3]={0x5c,0x54,0x71};/*o n f*/
//unsigned char key[24]={ /* 鍵值代碼數組 對應鍵位:*/
// 0x70,0x71,0x72,0x73,0x74,0x75, /* 7 8 9 A TRACE RESET*/
// 0xb0,0xb1,0xb2,0xb3,0xb4,0xb5, /* 4 5 6 B STEP MON */
// 0xd0,0xd1,0xd2,0xd3,0xd4,0xd5, /* 1 2 3 C HERE LAST */
// 0xe0,0xe1,0xe2,0xe3,0xe4,0xe5}; /* 0 F E D EXEC NEXT */
struct{ /*時(shí)間結構體變量*/
unsigned char s;
unsigned char m;
unsigned char h;
}clock={SET_SECOND,SET_MINUTE,SET_HOUR};
struct{ /*鬧鈴時(shí)間結構體變量*/
unsigned char m;
unsigned char h;
}alart={SET_MINUTE,SET_HOUR};
struct{ /*日期結構體變量*/
unsigned int year;
unsigned char month;
unsigned char day;
}date={6,1,1};
struct{ /*秒表時(shí)間結構體變量*/
unsigned char ms;
unsigned char s;
unsigned char m;
}timer={0,0,0};
unsigned char dispbuf[6]; /*顯示緩沖區數組*/
unsigned char clockstr[6]; /*時(shí)間顯示的數碼管編碼數組*/
unsigned char alartstr[6]; /*鬧鐘顯示的數碼管編碼數組*/
unsigned char datestr[6]; /*日期顯示的數碼管編碼數組*/
unsigned char timerstr[6]; /*秒表顯示的數碼管編碼數組*/
unsigned int itime=0,idot; /*定時(shí)器0中斷計數*/
unsigned char itime1=0; /*定時(shí)器1中斷計數*/
sbit P3_1=P3^1; /*外接蜂鳴器的管腳*/
bdata bit IsSet=0; /*設置模式標志位 0:正常走時(shí) 1:設置模式*/
bdata bit Alart_EN=0; /*鬧鈴功能允許位 0:禁止鬧鈴 1:允許鬧鈴*/
bdata bit IsBeep=0; /*響鈴標志位 0:未響鈴 1:正在響鈴*/
unsigned char SetSelect=0; /*在設置模式IsSet=1時(shí),正在被設置的位,對應上面的宏*/
unsigned char *CurrentMode; /*標志當前正設置的功能,如CurrentMode=CLOCK或CurrentMode=ALART等*/
void timerplus(void);
/**************************函數部分*************************/
void main(void)
{
sys_init();
while(1)
{
XBYTE[KEY_COLUMN,0x00]; /*給鍵盤(pán)列線(xiàn)賦全零掃描碼,判斷是否有鍵按下 */
while((XBYTE[KEY_LINE]&0x0f)==0x0f) /*檢測是否有鍵按下,無(wú)則一直進(jìn)行LED的刷新顯示*/
{
if(Alart_EN&&(clock.h==alart.h)&&(clock.m==alart.m)) {IsBeep=1;}
else
{ IsBeep=0;
P3_1=0;}
display();
}
keyprocess(getkeycode()); /*有鍵按下時(shí)得到鍵值,并送入鍵值處理程序*/
display(); /*可要可不要*/
}
}
void sys_init(void)
{
TMOD=0x22; /*定時(shí)器0和1都設置為工作方式2,基準定時(shí)250×2=500us=0.5ms*/
TH0=6; /*定時(shí)器0中斷服務(wù)用來(lái)產(chǎn)生1秒時(shí)鐘定時(shí)及鬧鐘蜂鳴器蜂鳴脈沖*/
TL0=6; /*定時(shí)器1中斷服務(wù)留給秒表使用,產(chǎn)生1/100秒定時(shí)*/
TH1=6;
TL1=6;
ET0=1;
ET1=1;
EA=1;
TR0=1;
update_clockstr(); /*初始化時(shí)鐘顯示編碼數組*/
update_alartstr(); /*初始化鬧鐘顯示編碼數組*/
update_datestr(); /*初始化日期顯示編碼數組*/
update_timerstr(); /*初始化秒表顯示編碼數組*/
update_dispbuf(clockstr);/*初始化顯示緩沖數組*/
CurrentMode=CLOCK; /*默認的顯示摸式為時(shí)鐘*/
P3_1=0; /*蜂鳴器接線(xiàn)引腳復位*/
}
void timer0(void) interrupt 1 using 1 /*定時(shí)器0中斷服務(wù)器,用來(lái)產(chǎn)生1秒定時(shí)*/
{
itime++;
if(itime==1000)
{
if(IsSet) /*在設置模式下,對正在設置的位閃爍顯示*/
{
dispbuf[SetSelect*2]=0; /*對正在設置的位所對應的顯示緩沖區元素賦0,使LED滅*/
dispbuf[SetSelect*2+1]=0;
}
if(IsBeep) P3_1=!P3_1; /*鬧鐘模式時(shí),產(chǎn)生峰鳴器響脈沖*/
if(CurrentMode==CLOCK)
{
dispbuf[2]=dispbuf[2]&0x7f;
dispbuf[4]=dispbuf[4]&0x7f;
}
}
if(itime==2000) /*兩千次計數為1S 2000×0.5ms=1s*/
{
itime=0; /*定時(shí)1s時(shí)間到,軟計數清零*/
clockplus(); /*時(shí)間結構體變量秒數加1 */
update_clockstr(); /* 更新時(shí)間顯示編碼數組 */
if(CurrentMode!=TIMER) update_dispbuf(CurrentMode); /* 用時(shí)間編碼數組更新顯示緩沖區 */
}
}
void timer1(void) interrupt 3 using 2 /*定時(shí)器1中斷服務(wù)器,用來(lái)產(chǎn)生1/100秒定時(shí)*/
{
idot++;
if(++itime1==20) /*20*0.5ms=10ms*/
{
itime1=0;
timerplus();
update_timerstr();
if(CurrentMode==TIMER)
{
update_dispbuf(timerstr);
dispbuf[2]=dispbuf[2]&0x7f; /*關(guān)閉小數點(diǎn)的顯示*/
dispbuf[4]=dispbuf[4]&0x7f;
if(idot<1000) /*閃爍顯示小數點(diǎn)*/
{
dispbuf[2]=dispbuf[2]|0x80;
dispbuf[4]=dispbuf[4]|0x80;
}else{
dispbuf[2]=dispbuf[2]&0x7f;
dispbuf[4]=dispbuf[4]&0x7f;
}
}
}
if(idot==2000) idot=0;
}
/*功能模塊子函數*/
void clockplus(void) /*時(shí)間加1s判斷分,時(shí)子函數*/
{
if(++clock.s==60) /*秒位判斷*/
{
clock.s=0;
if(++clock.m==60) /*分位判斷*/
{
clock.m=0;
if(++clock.h==24) /*時(shí)位判斷*/
{
clock.h=0;
if(++date.day==(getmonthdays(date.year,date.month)+1))
{
date.day=1;
if(++date.month==13) date.month=1;
}
}
}
}
}
void timerplus() /*秒表1/100秒位加1,判斷秒、分子程序*/
{
if(++timer.ms==100)
{
timer.ms=0;
if(++timer.s==60)
{
timer.s=0;
if(++timer.m==60)
{
timer.m=0;
}
}
}
}
void update_clockstr(void) /*更新時(shí)鐘顯示代碼數組clockstr*/
{
clockstr[0]=led[clock.s%10]; /*給元素0賦相應數碼管顯示編碼,編碼序號是秒數的個(gè)位*/
clockstr[1]=led[(int)(clock.s/10)]; /*給元素1賦相應數碼管顯示編碼,編碼序號是秒數的十位*/
clockstr[2]=led[clock.m%10]; /*以下類(lèi)推*/
clockstr[3]=led[(int)(clock.m/10)];
clockstr[4]=led[clock.h%10];
clockstr[5]=led[(int)(clock.h/10)];
}
void update_alartstr(void) /*更新鬧鐘顯示代碼數組alartstr*/
{ /*右邊兩位顯示on:鬧鐘開(kāi)啟 of:鬧鐘關(guān)閉*/
if(Alart_EN) alartstr[0]=ledchar[1];/*顯示字母n*/
else alartstr[0]=ledchar[2]; /*顯示字母f*/
alartstr[1]=ledchar[0]; /*顯示字母o*/
alartstr[2]=led[alart.m%10];
alartstr[3]=led[(int)(alart.m/10)];
alartstr[4]=led[alart.h%10];
alartstr[5]=led[(int)(alart.h/10)];
}
void update_datestr(void) /*更新日期顯示代碼數組datestr*/
{
datestr[0]=led[date.day%10];
datestr[1]=led[(int)(date.day/10)];
datestr[2]=led[date.month%10];
datestr[3]=led[(int)(date.month/10)];
datestr[4]=led[date.year%10];
datestr[5]=led[(int)(date.year/10)];
}
void update_timerstr(void) /*更新秒表顯示代碼數組timerstr*/
{
timerstr[0]=led[timer.ms%10];
timerstr[1]=led[(int)(timer.ms/10)];
timerstr[2]=led[timer.s%10];
timerstr[3]=led[(int)(timer.s/10)];
timerstr[4]=led[timer.m%10];
timerstr[5]=led[(int)(timer.m/10)];
}
void display(void) /*刷新顯示六位LED一次*/
{
unsigned char i;
for(i=0;i<6;i++)
{
LED_ON(i); /*選通相應位*/
XBYTE[LED_SEG]=dispbuf[i]; /*寫(xiě)顯示段碼*/
deley(50); /*延時(shí)顯示*/
LED_OFF; /*寫(xiě)LED全滅段碼*/
}
}
void update_dispbuf(unsigned char *str) /*更新顯示緩沖區子函數,參數為要用來(lái)更新緩沖區的源字符數組的首地址*/
{
dispbuf[0]=str[0]; /*將要更新的源字符數組內容COPY至dispbuf數組,用作顯示緩沖區*/
dispbuf[1]=str[1];
dispbuf[2]=str[2]|0x80; /*默認把時(shí)位和分位后面的小數點(diǎn)顯示出來(lái),根據需要再取舍*/
dispbuf[3]=str[3];
dispbuf[4]=str[4]|0x80;
dispbuf[5]=str[5];
}
void deley(int i) /*延時(shí)子函數*/
{
while(i--);
}
unsigned char getkeycode(void) /*鍵盤(pán)掃描子程序,返回獲得的鍵碼*/
{
unsigned char keycode; /*鍵碼變量,一開(kāi)始存行碼*/
unsigned char scancode=0x20; /*列掃描碼*/
unsigned char icolumn=0; /*鍵的列號*/
display(); /*用刷新數碼管顯示的時(shí)間去抖*/
XBYTE[KEY_COLUMN]=0x00;
keycode=XBYTE[KEY_LINE]&0x0f; /*從行端口讀入四位行碼*/
while((scancode&0x3f)!=0) /*取scancode的低六位,只要沒(méi)變?yōu)槿?,則執行循環(huán)*/
{
XBYTE[KEY_COLUMN]=(~scancode)&0x3f; /*給列賦掃描碼,第一次為011111*/
if((XBYTE[KEY_LINE]&0x0f)==keycode) break; /*檢測按鍵所在的列跳出循環(huán)*/
scancode=scancode>>1; /*列掃描碼右移一位*/
icolumn++; /*列號加1*/
}
keycode=keycode<<4; /*把行碼移到高四位*/
keycode=keycode|icolumn; /*由行碼和列碼組成鍵碼*/
//等待按鍵放開(kāi)
XBYTE[KEY_COLUMN]=0x00;
while((XBYTE[KEY_LINE]&0x0f)!=0x0f) display();
return keycode;
}
void keyprocess(unsigned char keycode) /*鍵值處理函數*/
{
switch (keycode)
{
case 0x73: Akey();
break;
case 0xB3: Bkey();
break;
case 0xD3: Ckey();
break;
case 0xE3: Dkey();
break;
default: break;
}
update_dispbuf(CurrentMode);
}
unsigned char getmonthdays(unsigned int year,unsigned char month)/*得到某月的天數*/
{
unsigned char days;
switch (month)
{
case 4:
case 6:
case 9:
case 11:days=30;
break;
case 2: if(year%4==0) days=29;
else days=28;
break;
default:days=31;
break;
}
return days;
}
/*功能鍵子函數部分*/
void Akey(void) /*對當前設置位進(jìn)行加一操作,如果設置秒位,則給秒位清零*/
{
if(CurrentMode==TIMER) /*秒表模式下啟閉走時(shí)*/
{ TR1=!TR1;
return;
}
if(!IsSet) return; /*如果不是設置模式退出*/
switch (SetSelect)
{
case SECOND:if(CurrentMode==CLOCK)
{
clock.s=0; /*如果當前被設置位是秒位,則清零秒位*/
update_clockstr();
}
if(CurrentMode==ALART)
{
Alart_EN=!Alart_EN;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.day==(getmonthdays(date.year,date.month)+1)) date.day=1;
update_datestr();
}
if(CurrentMode==TIMER)
{
TR1=!TR1;
}
break;
case MINUTE:if(CurrentMode==CLOCK)
{
if(++clock.m==60) clock.m=0; /*如果當前被設置分位,則分位加1*/
update_clockstr();
}
if(CurrentMode==ALART)
{
if(++alart.m==60) alart.m=0;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.month==13) date.month=1;
update_datestr();
}
break;
case HOUR: if(CurrentMode==CLOCK)
{
if(++clock.h==24) clock.h=0; /*如果當前被設置時(shí)位,則時(shí)位加1*/
update_clockstr();
}
if(CurrentMode==ALART)
{
if(++alart.h==24) alart.h=0;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(++date.year==100) date.year=0;
update_datestr();
}
break;
default: break;
}
update_dispbuf(CurrentMode);
}
void Bkey(void) /*對當前設置位進(jìn)行減一操作,如果設置秒分,則給秒位清零,類(lèi)比Akey()函數*/
{
if(!IsSet) return;
switch (SetSelect)
{
case SECOND:if(CurrentMode==CLOCK)
{
clock.s=0;
update_clockstr();
}
if(CurrentMode==ALART)
{
Alart_EN=!Alart_EN;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(--date.day==0x00) date.day=getmonthdays(date.year,date.month);
update_datestr();
}
break;
case MINUTE:if(CurrentMode==CLOCK)
{
if(--clock.m==0xff) clock.m=59;
update_clockstr();
}
if(CurrentMode==ALART)
{
if(--alart.m==0xff) alart.m=59;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(--date.month==0x00) date.month=12;
update_datestr();
}
break;
case HOUR: if(CurrentMode==CLOCK)
{
if(--clock.h==0xff) clock.h=23;
update_clockstr();
}
if(CurrentMode==ALART)
{
if(--alart.h==0xff) alart.h=23;
update_alartstr();
}
if(CurrentMode==DATE)
{
if(--date.year==0xffff) date.year=99;
update_datestr();
}
break;
default: break;
}
update_dispbuf(CurrentMode);
}
void Ckey(void) /*正常走時(shí)模式和設置模式的切換*/
{
if(CurrentMode==TIMER)
{
TR1=0; /*初始化定時(shí)器1設置,停止秒表記時(shí)*/
TH1=6;
TL1=6;
timer.ms=0; /*初始化秒表數組*/
timer.s=0;
timer.m=0;
update_timerstr();
}else
{
if(IsSet==0) /*在非秒表模式下,第一次按下C鍵進(jìn)入設置模式,設置時(shí)位*/
{
IsSet=1; /*置位標志位,進(jìn)入設置模式 */
SetSelect=HOUR;
return;
} /*第二次按C鍵設置分位,第三次按鍵設置秒位,第四次按鍵完成退出設置*/
if(SetSelect==0) /*按到第四次,即設置完秒位后,將標志位IsSet置0,完成設置*/
{
IsSet=0; /*復位標志位,進(jìn)入正常走時(shí)模式*/
return;
}
if(SetSelect>0) SetSelect--; /*設置位的標志變量SetSelect=0:時(shí)位 1:分位 2:秒位*/
}
}
void Dkey(void) /*工作狀態(tài)切換:時(shí)鐘、鬧鐘、日期、秒表*/
{
if(CurrentMode==CLOCK) /*切換至鬧鐘,同時(shí)開(kāi)關(guān)鬧鐘*/
{ CurrentMode=ALART;
Alart_EN=!Alart_EN;
update_alartstr();
return;
}
if(CurrentMode==ALART) /*切換至日期*/
{ CurrentMode=DATE;
return;
}
if(CurrentMode==DATE) /*切換至秒表,同時(shí)關(guān)閉設置模式*/
{
CurrentMode=TIMER;
IsSet=0;
return;
}
if(CurrentMode==TIMER) /*切換至時(shí)鐘*/
{
CurrentMode=CLOCK;
return;
}
}