ESP32+时钟、闹钟、温度+microPython程序 (2020-10-09)

ESP32+时钟、闹钟、温度+microPython程序 (2020-10-09),第1张

【硬件】:ESP32芯片、DS3231时钟芯片、五方向按键(带set和rst按键)、有源蜂鸣器、 LCD1602液晶屏(带PCF8574芯片,注意不是背光可调多种颜色的那种RGB1602)

【软件】:microPython

1、开始自动检测WIFI,检测到之后,自动从网络获取时间,并写入esp8266/esp32、同时写入ds3231时钟芯片中

2、如果不能联网,则自动从ds3231芯片获取时间

3、按中键,可以开启关闭屏幕背光

4、按set键,可以进入菜单,可以设置日期、时间、闹钟以及闹钟是否开启(默认每天循环)

5、液晶屏幕上,时间与温度中间,如果有个.显示,表示闹钟是开启状态。 如果没有,则表示闹钟关闭

DS3231micro.py : 从 https://www.jianshu.com/p/93af7d173f98 可以复制

esp8266_i2c_ lcd.py 与 lcd_ api.py : 从 https://www.jianshu.com/p/7798d0c06c69 可以复制

本帖最后由 qhdtc5 于 2015-3-25 08:00 编辑

说明:

使用了LCD5110作为显示屏,库为LCD5110_Basic

借鉴了其他程序的一些思路,用链表作为菜单结构,并使用Action概念来确定菜单项动作

虽然动作类型设想有好几种,但我只写了设置逻辑值的菜单动作,其他的可自行完善(我也会继续写完)

代码未优化和未考虑内存占用等问题

请大家指教

先上图片和视频

[media]http://player.youku.com/player.php/sid/XOTE5MjkyOTg4/v.swf[/media]

代码:

[mw_shl_code=bash,true]#include <Wire.h>

#include <DS3231.h>

#include <LCD5110_Basic.h>

#define LIGHT_PIN 3

#define LED_PIN 13

// 菜单最大显示行数

#define MENU_MAX_ROW 5

/**

* 以下定义菜单项类型

*/

// 具有子菜单的菜单项

#define MENU_SUBMENU 0

// 参数项(用于执行参数设置)

#define MENU_PARAM 1

// 无动作的菜单项

#define MENU_ACTION_NONE 101

// 执行逻辑参数设置的菜单项(开/关、真/假等设置)

#define MENU_ACTION_LOGIC 102

// 执行数值调整设置的菜单项(时间、音量等设置)

#define MENU_ACTION_NUMBER 103

// 执行字符串设置的菜单项(欢迎语、LED显示文字等设置)

#define MENU_ACTION_STRING 104

/**

* 以下定义按键引脚

* 设置上拉电阻,低电平有效

*/

#define KEY_UP 7

#define KEY_DOWN 6

#define KEY_ENTER 5

#define KEY_EXIT 4

// 定义按键消抖延时时间

#define KEY_TREMBLE_TIME 20

// DS3231时钟变量

DS3231 clock

// LCD5110液晶屏变量

LCD5110 myGLCD(8, 9, 10, 11, 12)

// LCD5110使用的字体

extern unsigned char SmallFont[]

/**

* 菜单结构

* 一个数组既代表一级菜单,数组中的一个元素就是一个菜单项,就是一个单独的结构体,

* 数组中的第一个元素中的num值很重要,表示本级菜单具有多少个菜单项。

*

* @var int num 本级菜单数量,必须在第一项中设置正确的菜单项数量

* @var char* label 菜单文本

* @var int type 此项类型,参考宏定义

* @var void (*action)(const char *) 指向动作的函数指针,此项要执行的具体动作函数

* @var mymenu* next 下一级菜单,只需在菜单的第一项设置,其他项置空(NULL)即可

* @var mymenu* prev 上一级菜单,同上

*/

struct mymenu {

int num

char *label//display label

int type//0:item, 1:submenu, 2:param

void (*action)(const char *)

mymenu * next

mymenu * prev

}

/**

* 逻辑参数设置菜单

* 逻辑菜单表示菜单项中的action函数要执行逻辑设置动作,即设置某个参数的逻辑值

* 逻辑菜单只需两个菜单项代表true和false

* 需要遵守的规则为:菜单中的文本需要设置为“ON”和“OFF”,

* 在执行动作函数的时候,可以将正确的参数传递过去,

* 动作函数的规则参照函数说明

*/

mymenu logic_menu[2] = {

{2, "ON", MENU_PARAM, NULL, NULL, NULL},

{2, "OFF", MENU_PARAM, NULL, NULL, NULL}

}

/**

* 下面定义了三级菜单说明了菜单应该如何个定义

*/

// 第二级菜单

mymenu light_menu[2] = {

//第一项什么也不做,所以设置了类型为MENU_ACTION_NONE

{2, "light 1", MENU_ACTION_NONE, NULL, NULL, NULL},

//第二项指向了下级菜单,所以设置了类型为MENU_SUBMENU

{2, "light submenu", MENU_SUBMENU, NULL, NULL, NULL}

}

// 第三级菜单

mymenu test_level3_menu[7] = {

{7, "level3menu-1", MENU_ACTION_NONE, NULL, NULL, NULL},

{7, "level3menu-2", MENU_ACTION_NONE, NULL, NULL, NULL},

{7, "level3menu-3", MENU_ACTION_NONE, NULL, NULL, NULL},

{7, "level3menu-4", MENU_ACTION_NONE, NULL, NULL, NULL},

//定义了一个逻辑动作,这里是控制PIN13脚的LED

{7, "LED TEST", MENU_ACTION_LOGIC, NULL, NULL, NULL},

{7, "level3menu-6", MENU_ACTION_NONE, NULL, NULL, NULL},

{7, "level3menu-7", MENU_ACTION_NONE, NULL, NULL, NULL}

}

// 第一级菜单

mymenu main_menu[4] = {

{4, "item 1", MENU_ACTION_NONE, NULL, NULL, NULL},

{4, "item 2", MENU_ACTION_NONE, NULL, NULL, NULL},

//指向下一级菜单

{4, "item 2.1", MENU_SUBMENU, NULL, NULL, NULL},

{4, "item 3", MENU_ACTION_NONE, NULL, NULL, NULL}

}

// 定义菜单 *** 作需要的全局变量

// 分别为当前菜单、上一级菜单、当前菜单项索引和开始显示的菜单项索引

mymenu *cur_item, *prev_item

int item_index, start_index

bool stat

/**

* DS3231需要的全局变量

*/

bool Century=false

bool h12

bool PM

byte ADay, AHour, AMinute, ASecond, ABits

bool ADy, A12h, Apm

int second,minute,hour,date,month,year,temperature

int oldsecond//此变量用于判断是否更新显示内容

// 定义LCD背光显示计时变量,无按键 *** 作超时就关闭背光

unsigned long starttime

// 定义菜单的 *** 作按键(上、下、进入和返回)状态变量

bool old_up_stat, old_down_stat, old_enter_stat, old_exit_stat

void setup() {

//设置LCD背光引脚为输出

pinMode(LIGHT_PIN, OUTPUT)

//设置按键为上拉电阻输入(低电平有效)

for (int i=4i<=7i++){

pinMode(i, INPUT_PULLUP)

}

//初始化时打开LCD背光

backlight("ON")

stat = false

//初始化LCD5110

myGLCD.InitLCD()

myGLCD.setContrast(127)

myGLCD.setFont(SmallFont)

myGLCD.print("Initialize...", 0, 0)

/**

* 菜单的进一步设置

* 在这里将每一个菜单的关联设置好

* 对照每一个初始设置仔细填写它们之间的关系

*/

//第一级(main_menu)的第三项指向了下一级菜单(light_menu)

main_menu[2].next = light_menu

//第二级(light_menu)的上一级(main_menu)

light_menu[0].prev = main_menu

//第二级(light_menu)的第二项指向了下一级菜单(test_level3_menu)

light_menu[1].next = test_level3_menu

//第三级(test_level3_menu)的上一级(light_menu)

test_level3_menu[0].prev = light_menu

//第三级(test_level3_menu)的第五项定义了个逻辑动作

test_level3_menu[4].action = ledtest

/**

* 初始化当前菜单为第一级(main_menu)

*/

cur_item = main_menu

/**

* 上一级菜单为空

*/

prev_item = NULL

/**

* 当前选择了第一项

*/

item_index = 0

/**

* 从第一项开始显示菜单

*/

start_index = 0

/**

* 设置DS3231的日期和时间,只需在需要调整的时候执行一次

*/

//clock.setClockMode(true)

//I2C总线库初始化

Wire.begin()

//一些状态变量的初始化

oldsecond = -1

old_up_stat = HIGH

old_down_stat = HIGH

old_enter_stat = HIGH

old_exit_stat = HIGH

//为了视频中能看清楚,延时定义了3秒

delay(3000)

//所有初始化完成,清屏并关闭LCD背光

myGLCD.clrRow(0)

backlight("OFF")

starttime = millis()

}

void loop() {

//获取运行时间

unsigned long endtime = millis()

//获取当前菜单的项目数量

int menu_num = cur_item[0].num

//定义一个临时变量

int idx

//绘制菜单

renderMenu(&cur_item[0], menu_num>MENU_MAX_ROW ? MENU_MAX_ROW : menu_num)

//读取按键

int k_down = digitalRead(KEY_DOWN)

int k_up = digitalRead(KEY_UP)

int k_enter = digitalRead(KEY_ENTER)

int k_exit = digitalRead(KEY_EXIT)

//计算无按键时间,决定是否关闭LCD背光

if (endtime-starttime>5000) {

//myGLCD.enableSleep()

backlight("OFF")

}

//有键按下,打开LCD背光,计时清零

if (k_up==LOW || k_down==LOW || k_enter==LOW || k_exit==LOW) {

//myGLCD.disableSleep()

backlight("ON")

starttime = endtime

}

//检测按键,进入相应动作

if (k_up == LOW &&k_up!=old_up_stat){

//消抖

delay(KEY_TREMBLE_TIME)

if (k_up == LOW){

item_index --

}

} else if (k_down == LOW &&k_down!=old_down_stat){

//消抖

delay(KEY_TREMBLE_TIME)

if (k_down == LOW){

item_index ++

}

} else if (k_enter == LOW &&k_enter!=old_enter_stat){

//消抖

delay(KEY_TREMBLE_TIME)

if (k_enter == LOW){

//计算此时的菜单项索引值

idx = start_index+item_index

if (cur_item[idx].next != NULL &&cur_item[idx].type == MENU_SUBMENU){

//条件成立说明此菜单项指向了下一级菜单

//此级菜单变成了上一级菜单

prev_item = cur_item

//将指向的下一级菜单设置为当前菜单

cur_item = cur_item[idx].next

//重置菜单项索引和绘制索引

item_index = 0

start_index = 0

//清屏

myGLCD.clrScr()

} else if (cur_item[idx].action != NULL &&cur_item[idx].type != MENU_PARAM){

//条件成立说明此项菜单是动作

//此级菜单变成上一级菜单

prev_item = cur_item

//根据动作类型调用相应的下一级菜单

switch(cur_item[idx].type){

case MENU_ACTION_LOGIC:

//将动作函数传递给逻辑菜单,使逻辑菜单能够正确执行动作

logic_menu[0].action = cur_item[idx].action

//设置当前菜单为逻辑菜单

cur_item = logic_menu

//重置菜单项索引和绘制索引

item_index = 0

start_index = 0

break

case MENU_ACTION_NUMBER:

break

case MENU_ACTION_STRING:

break

default:

break

}

//清屏

myGLCD.clrScr()

} else if (cur_item[idx].type == MENU_PARAM){

//条件成立说明正在执行动作

//调用相应的动作函数,并传递参数(这里只举例逻辑设置)

cur_item[0].action((const char *)cur_item[idx].label)

}

}

} else if (k_exit == LOW &&k_exit!=old_exit_stat){

//消抖

delay(KEY_TREMBLE_TIME)

if (k_exit == LOW){

//返回上一级菜单的 *** 作

if (prev_item != NULL){

//设置上一级菜单为当前菜单

cur_item = prev_item

//设置当前菜单的上一级菜单

prev_item = cur_item[0].prev

//重置菜单项索引和绘制索引

item_index = 0

start_index = 0

//清屏

myGLCD.clrScr()

}

}

}

/**

* 菜单项上下选择是否越界

*/

if (item_index<0){

item_index = 0

start_index --

if (start_index<0) start_index = 0

}

if (item_index>=MENU_MAX_ROW || item_index>=menu_num){

if (item_index>=menu_num) item_index = menu_num-1

if (item_index>=MENU_MAX_ROW) item_index = MENU_MAX_ROW-1

if (start_index+MENU_MAX_ROW<menu_num) start_index ++

}

//保存按键状态

old_up_stat = digitalRead(KEY_UP)

old_down_stat = digitalRead(KEY_DOWN)

old_enter_stat = digitalRead(KEY_ENTER)

old_exit_stat = digitalRead(KEY_EXIT)

//renderClock(0, 0)

}

/**

* 关闭LCD背光

* 此函数符合菜单动作定义规则,即无返回值、一个字符串参数

*

* @var const char* stat 值为“ON"打开背光,为“OFF”关闭背光

*/

void backlight(const char *stat)

{

//这里定义了一个逻辑动作

//根据参数决定LCD背光的开关

digitalWrite(LIGHT_PIN, strcmp(stat,"ON") ? LOW : HIGH)

}

/**

* 逻辑动作测试函数

*

* @var const char* stat 根据此参数来执行动作,这里测试PIN13的LED

*/

void ledtest(const char *stat)

{

//根据参数决定LED开关

digitalWrite(LED_PIN, strcmp(stat,"ON") ? LOW : HIGH)

}

/**

* 绘制单个菜单项

*

* @var struct mymenu item 要绘制的项目

* @var int row 绘制坐标(LCD5110绘制文字时只需一个行坐标即可,可根据实际显示设备调整)

* @var bool rev 是否反相绘制,默认为不反相绘制(用于当前选中项绘制,可根据实际显示设备调整)

*/

void renderItem(struct mymenu item, int row, bool rev = false)

{

char label[15]

//规范显示项目文字(左对齐,补空格)

sprintf(label, "%-14s", item.label)

if (rev){

//打开5110的反相绘制功能

myGLCD.invertText(true)

}

//绘制菜单项

myGLCD.print(label, 0, row)

if (rev){

//关闭5110的反相绘制功能

myGLCD.invertText(false)

}

}

/**

* 绘制某一级菜单

*

* @var struct mymenu* items 需要绘制的菜单

* @var int menu_num 菜单项目数量

*/

void renderMenu(struct mymenu *items, int menu_num)

{

//绘制数量不能超过每一屏的最大绘制数量

int num = menu_num>MENU_MAX_ROW ? MENU_MAX_ROW : menu_num

for (int i=0i<numi++){

//绘制每个菜单项

renderItem(items[i+start_index], i*8, i==item_index ? true:false)

}

}

void renderClock(int row)

{

char s[12]

second = clock.getSecond()

minute = clock.getMinute()

hour = clock.getHour(h12, PM)

date = clock.getDate()

month = clock.getMonth(Century)

year = clock.getYear()

temperature = clock.getTemperature()

if (oldsecond!=second){

sprintf(s, "20%02d-%02d-%02d", year, month, date)

myGLCD.print(s, 0, row)

sprintf(s, "%02d:%02d:%02d", PM==0 ? hour:hour+12, minute, second)

myGLCD.print(s, 0, row+8)

sprintf(s, "%3d", sizeof(main_menu)/sizeof(mymenu))

myGLCD.print(s, 0, row+16)

oldsecond = second

}

}

void clockSetting(int year, int month, int day, int hour, int minute, int second)

clock.setSecond(second)//配置秒

clock.setMinute(minute

只要这个1602的.h和.c文件都正确有效,那么将这个1602的.h和.c文件放到你的这个工程文件夹里,而后只要主程序中有#include"lcd1602.h"这样的包含头文件声明,就能正确使用了。如下图是用keil uv2建立的一个DS3231的时钟工程,图片中相关文件都能正常打开的:


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/7820653.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-10
下一篇 2023-04-10

发表评论

登录后才能评论

评论列表(0条)

保存