按键长按短按

长按是在短按的基础上,以一个很短的时间(10ms)再次判断按键是否片于按下状态,当达到一定的时间(1s)后,按键还处于按下状态,说明是长按状态,结合上一节的短按,下面实现一个从0-99的累加,短按时一次加1,长按后,进入快速累加的过程,同时使用了定时器来定时刷新数码管,在Display的处理上就和之前的不同,原来在Display里通过循环刷新每个数码管的显示,现在放到了定时器定时触发的事件里,设置定时时间为2ms,这2ms只负责刷新共阴极的8位数码的一位,所以需要在定时事件里处理每次刷新不同位置的数码管。

#include <reg52.h>



sbit KeyAdd = P0^0;

sbit LED = P0^1;

sbit LATCH1=P2^2;//段锁存

sbit LATCH2=P2^3;//位锁存



unsigned char code DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~9

unsigned char code WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//位码

unsigned char TempData[8]; //存储显示值

unsigned char vT = 0;



void Delay(unsigned int t);

void Display(unsigned char Index);

void InitTime0(void);



void main(void)

{

    unsigned char numadd = 0;  // 累加的结果

    unsigned int vExtend = 0;  // 用于判断是否满足长按标准

    LED = 0;

    TempData[0] = DuanMa[0];  

    TempData[1] = DuanMa[0];

    InitTime0();

    while (1)

    {

        if (!KeyAdd)       // 发现按下

        {

            Delay(1500);   // 延时去抖

            if (!KeyAdd)   // 如果还处于按下

            {

                while (!KeyAdd)     //等待弹起

                {

                    Delay(100);     // 延时,这里要短一点,vExtend长些,这样有利于精确的判断是否放开

                    vExtend++;     // 长按计数

                    if (vExtend == 1000)   // 按下时间大约1s认为长按

                    {

                        vExtend = 0;      //清空计数初值

                        while(!KeyAdd)      // 如果按键还处于按下状态,则快速增加

                        {

                            if (numadd == 99)

                                numadd = 0;

                            else

                                numadd++;

                            TempData[0] = DuanMa[numadd / 10];  

                            TempData[1] = DuanMa[numadd % 10];

                            Delay(10000); //延时让数字显示一段时间,否则数字不动直到放开按钮后突然跳变

                        }    

                    }

                }

                vExtend = 0;   //设置初值,为下一次长按做准备

                if (numadd == 99)

                    numadd = 0;

                else

                    numadd++;

                TempData[0] = DuanMa[numadd / 10];  

                TempData[1] = DuanMa[numadd % 10];

            }

        }

    }

}



void Delay(unsigned int t)

{

    while (--t);

}



void Display(unsigned char Index)

{      

    P1 = 0;   //清空数据,防止有交替重影

    LATCH1 = 1;     //段锁存

    LATCH1 = 0;



    P1 = WeiMa[Index]; //取位码 

    LATCH2=1;     //位锁存

    LATCH2=0;



    P1 = TempData[Index]; //取显示数据,段码

    LATCH1=1;     //段锁存

    LATCH1=0;

}



void InitTime0(void)

{

    TMOD = 0x01;

    TH0 = 0xF8;    // 计数初值F830H = 2000 (65536-63536=2000 * 1us = 2ms)

    TL0 = 0x30;

    ET0 = 1;       // EA即IE^1 等于1时申请中断定时器0中断打开

    EA = 1;           // 总中断打开

    TR0 = 1;       // 启动定时器0

}



void MyInterrept(void) interrupt 1

{

    LED = !LED;

    TH0 = 0xF8;          //重新赋值 2ms

    TL0 = 0x30;

     Display(vT);

    vT++;

    if (vT == 8)

      vT = 0;

}

 

你可能感兴趣的:(按键长按短按)