2021爱智先行者—智能灯光开关
时间:2022-08-15 01:30:01
2021爱智先行者—智能灯光开关
我怎么度,水文还有机会拿奖金,这我怎么会错过呢???立即开始光速整活!
【本文正在参与"2021爱智先行者-征文大赛"活动】https://mp.weixin.qq.com/s/I2s99dZpbP0QpMiKFneYJA
具体活动帖子可参考以下链接:
https://bbs.csdn.net/topics/603225537
如果你感兴趣,你也可以一起参加。有一等奖 1000 块呢!
新手礼包给朋友!
首先,我发现很多朋友第一次使用爱情智慧。在这里,我想和大家分享一些我自己的经验和我自己的小工具(我绝对不想拯救我悲伤的阅读量!(▼皿▼#))。
设备与爱智交互的协议是我目前使用最多的:
统一了 WiFi 和 ZigBee 上层使用的跨厂商发现并控制 DDC协议介绍_桌面博客的灵感-CSDN博客
我写的爱智设备开发的小调试工具(很挫折,还有BUG,但勉强能用):
第一次尝试! Spirit 1 实现 DDC协议嗅探器_桌面博客的灵感-CSDN博客
我自己写开发写的一个 SDDC_SDK ,设备开发会很方便(虽然可以用,但还是有的BUG,欢迎帮我修理BUG,大悲(?_?) ):
同事逼死官方系列!sddc 协议的SDK框架 sddc_sdk_lib 解析_桌面博客的灵感-CSDN博客
同事逼死官方系列! DDC 嗅探器到 sddc_sdk_lib 的数据解析_桌面博客的灵感-CSDN博客
我还做了很多奇怪的场景和 DEMO,我的财宝吗?想要的话就给你,找出来!我的一切都放在我身上。桌面秘密宝库的灵感 !
以上代码都是开源的(欢迎帮我修理)BUG),如果不会用 git 朋友可以看看这篇文章:
简单无脑,上手即用 - 教你手把手使用 智能红外温度传感器代码及依赖性 gitee 库!_桌面博客的灵感-CSDN博客
怎么样?我准备好超级充分了吗?如果你觉得有用,请表扬我!求你了,我让室友磕头!
这是严肃的前言
几个月前我在这里 从零开始的DIY智能家居 - 基于 ESP32 智能光照传感器_桌面博客的灵感-CSDN博客 埋了一个坑,现在我回来填坑了!这次完成了远程控制灯的场景(秘密告诉你,我偷偷挖了更大的坑)。现在自动开关灯真的可以实现了!
硬件选择
第一,是翼辉的 边缘计算机 Spirit 1 边缘计算机,在这里 Spirit 1 它相当于一个小服务器,大部分操作都在这里完成。
我选择了公牛的远程控制灯设备 遥控开关 但我发现这个开关似乎是改造的 lora 是的,我没有完成 433MHz 模块,于是只能用继电器硬上了。
这继电器 挺好用的,5V就能驱动,3.3V可以触发,体积不大,板加继电器总共0.13A的电流。
接线方面 开发板只需连接一个继电器(VCC-5V, GND-GND,T1-P14)。
代码解析
获取代码
为了方便解释逻辑,我会打乱代码的顺序,可能会被切割。想直接拿代码跑的朋友可以直接去。 桌面秘密宝库的灵感 或直接获取代码 clone:
https://gitee.com/inspiration-desktop/DEV-lib-arduino.git
下载或者 clone这三个文件夹用于代码后:
cjson:我移植的 cjson 库,是标准的 cjson 库,放到 arduino 在安装目录下 libraries 百度在文件夹里 cjson 使用函数就行了。
libsddc:我从官方移植SDDC库和自己写的 SDK,也是放入 libraries 就在文件夹里。里面是。 SDDC 我们不需要管理协议的处理函数。
demo 文件夹是我们的各种传感器 demo 代码了:
红圈的 SW_sddc_sdk_demo我们的代码在文件夹里,点进去就能看到。 SW_sddc_sdk_demo.ino 双击文件自动启动 arduino-IDE 打开代码 -> 端口 选择对应的 COM 然后点击上传将代码烧录到板中:
设备控制命令:
通过 Spirit 1 应用程序或调试工具 嗅探器 发送给传感器设备的命令:
触发开关:
{
"method": "set", "SW_ctrl": "button" }
协议的初始化过程:
基于官方 demo 主要是设备初始化、管脚配置和协议初始化。
/* * 初始化传感器 */ void sensor_init() {
pinMode(sign_pin, OUTPUT); } void setup() {
byte mac[6]; Serial.begin(115200); Serial.setDebugOutput(true); Serial.println();
// 初始化传感器
sensor_init();
// 清除一下按键状态机的状态
button.reset();
// 创建按键扫描线程,长按 IO0 按键,松开后ESP32 将会进入 SmartConfig 模式
sddc_printf("长按按键进入 Smartconfig...\n");
button.attachLongPressStop(esp_io0_key_task);
xTaskCreate(esp_tick_task, "button_tick", ESP_TASK_STACK_SIZE, NULL, ESP_TASK_PRIO, NULL);
// 启动 WiFi 并且连接网络
WiFi.begin(ssid, password);
while (WiFi.status() != WL_CONNECTED)
{
delay(500);
Serial.print(".");
}
// 获取并打印 IP 地址
Serial.println("");
Serial.println("WiFi connected");
Serial.print("'ip :");
Serial.print(WiFi.localIP());
Serial.println("' to connect");
// sddc协议初始化
sddc_lib_main(&sys_cfg);
// 获取并打印网卡 mac 地址
WiFi.macAddress(mac);
sddc_printf("MAC addr: %02x:%02x:%02x:%02x:%02x:%02x\n",
mac[5], mac[4], mac[3], mac[2], mac[1], mac[0]);
// 使用网卡 mac 地址设置设备唯一标识 UID
sddc_set_uid(G_sddc, mac);
}
void loop() {
// 运行 SDDC 协议循环
while (1)
{
sddc_printf("SDDC running...\n");
sddc_run(G_sddc);
sddc_printf("SDDC quit!\n");
}
// 销毁 SDDC 协议
sddc_destroy(G_sddc);
}
配置设备信息
这部分代码可以配置 WiFi 名字和 WiFi 密码,要使用的引脚,并且配置设备在 Spirit 1 上显示的信息:
#define SDDC_CFG_PORT 680U // SDDC 协议使用的端口号
#define PIN_INPUT 0 // 选择 IO0 进行控制
#define ESP_TASK_STACK_SIZE 4096
#define ESP_TASK_PRIO 25
static const int sign_pin = 14; // 开关控制引脚
static const char* ssid = "EOS-Tenda"; // WiFi 名
static const char* password = "1234567890"; // WiFi 密码
OneButton button(PIN_INPUT, true);
/* * 当前设备的信息定义 */
DEV_INFO dev_info = {
.name = "灯光遥控开关",
.type = "light.rc",
.excl = SDDC_FALSE,
.desc = "ESP-32S",
.model = "IDDGYK01B",
.vendor = "inspiration-desktop",
};
/* * 系统注册对象汇聚 */
SDDC_CONFIG_INFO sys_cfg = {
.token = "1234567890", // 设备密码
.devinfo = &dev_info,
.io_dev_reg = io_dev,
.io_dev_reg_num = ARRAY_SIZE(io_dev),
.num_dev_reg = num_dev,
.num_dev_reg_num = ARRAY_SIZE(num_dev),
.state_get_reg = dev_state_get_reg,
.state_get_reg_num = ARRAY_SIZE(dev_state_get_reg),
.dis_dev_reg = dis_dev,
.dis_dev_num = ARRAY_SIZE(dis_dev),
};
回调函数注册
这是收到命令后回调函数注册的位置,在这里注册的函数才能被 SDK 正确的调用,执行正确的动作。
具体 SDK 的解析可以参考 同人逼死官方系列!基于sddc 协议的SDK框架 sddc_sdk_lib 解析 和 同人逼死官方系列!从 DDC 嗅探器到 sddc_sdk_lib 的数据解析
/* * 数字量设备对象函数与处理方法注册 */
NUM_DEV_REGINFO num_dev[] = {
// {"set_num_demo", demo}, // 字符串为输入命令,demo为命令处理函数
};
/* * 显示设备对象函数与处理方法注册 */
DIS_DEV_REGINFO dis_dev[] = {
// {"set_dis_demo", demo}, // 字符串为输入命令,demo为命令处理函数
};
/* * IO设备对象设置函数与处理方法注册 */
IO_DEV_REGINFO io_dev[] = {
// {"set_io_demo", demo}, // 字符串为输入命令,demo为命令处理函数
{
"SW_ctrl", SW_ctrl},
};
/* * 系统对象状态获取注册 */
DEV_STATE_GET dev_state_get_reg[] = {
// {"demo", DEV_NUM_TYPE, num_get_demo}, // demo为输入命令,字符串为命令处理函数
// {"demo", DEV_IO_TYPE, io_get_demo},
// {"demo", DEV_DISPLAY_TYPE, dis_get_demo},
};
数据获取与上报流程
这里是我们自己编写的处理流程 ,可以根据你的需求自己更改,收到 set 或者 get 后根据前面的注册的函数,进入对应的处理函数。
static void button_task(void *arg)
{
Serial.println("触发");
delay(100);
digitalWrite(sign_pin, HIGH);
delay(100);
digitalWrite(sign_pin, LOW);
vTaskDelete(NULL);
}
/* * 开关控制函数 */
sddc_bool_t SW_ctrl(const char* value)
{
printf("进入开关控制函数\n");
// 不能在主循环里面延时,所有单独开一个线程模拟按键触发
xTaskCreate(button_task, "button_task", ESP_TASK_STACK_SIZE, NULL, ESP_TASK_PRIO, NULL);
return SDDC_TRUE;
}
总结
就这?就这?我在开头说了我在憋大招,当然不会这样这点东西,这只是一个开胃菜。请大家敬请期待吧!
本文仅个人学习使用,如有错误,欢迎指正, ( ੭ ˙ᗜ˙ )੭谢谢老板!