STC15F104E做的LED小台灯(USB供电+3级触摸调光+掉电记忆)
主要特点:
3级亮度(就是亮1、2、3个LED,非PWM无闪烁)
触摸控制(本来是做电容式的,实际上需要触碰到电极)
掉电状态记忆(比如亮2个LED,拔掉电再插上还是亮2个)
低功耗(手上只有104E用的空闲模式,如果用104W可以在掉电模式定时唤醒)
操作方法:
每次触摸切换灯状态(关 - 1个亮 - 2个亮 - 3个亮 - 关),长触摸直接关。
电路很简单,上图:
LED功率不大直接用IO驱动的,如果是大功率LED可以加MOS管驱动,这里简单起见没加。
触摸按键是经典的RC充放电(姑且认为是吧),R5起到一定保护作用。
触摸按键代码参考了论坛一个帖子,当时没记链接,抱歉:
- // 测量引脚电平上升时间
- static uint8_t Touch_Mes(void)
- {
- uint8_t i;
-
- KEY_TOUCH = 0; // 拉低TOUCH口
-
- // 确保TOUCH口已经拉低,i控制溢出
- for (i = 255; i != 0; i--);
- for (i = 1; KEY_TOUCH && i; i++);
-
- TH0 = 0; // 初始化TH0为0
- TL0 = 0; // 初始化TL0为0
- TR0 = 1; // 启动定时器
- KEY_TOUCH = 1; // 释放TOUCH口,此时TOUCH会慢慢升高电平
- // 等待电平升高到单片机所认为的高电平,i控制溢出
- for (i = 1; !KEY_TOUCH && i; i++);
- TR0 = 0; // 停止计数器
-
- return TL0;
- }
复制代码
掉电记忆使用Flash一个扇区模拟EEPROM,有简易的磨损均衡算法:
- // 保存配置
- static void SaveCfg(void)
- {
- uint16_t i;
- uint8_t dat;
-
- // 每次从扇区前面往后存,直到扇区全满时擦除重来
- for (i = 0; i < IAP_SEC_SIZE; i++) {
- dat = IAP_ReadByte(IAP_ADDR_BASE + i);
- if (dat == 0xff) {
- IAP_ProgramByte(IAP_ADDR_BASE + i, gLightStatus);
- return;
- }
- }
-
- // 扇区全满
- IAP_EraseSector(IAP_ADDR_BASE);
- IAP_ProgramByte(IAP_ADDR_BASE, gLightStatus);
- }
-
- // 读取配置
- static void LoadCfg(void)
- {
- uint16_t i;
- uint8_t dat;
-
- // 从扇区后面往前查找最近的配置
- for (i = IAP_SEC_SIZE; i > 0; i--) {
- dat = IAP_ReadByte(IAP_ADDR_BASE + i - 1);
- if (dat < LED_STATUS_MAX) {
- gLightStatus = dat;
- return;
- }
- }
-
- gLightStatus = LED_STATUS_1;
- }
复制代码
再贴几张实物照片:
直接飞线的看起来有点乱,触摸位置就是尾部粘的那块铝箔。