前言
这里这么多设施,忽然发现我做的如同都是传感器之类的居多如同没啥输入端设施,每天采集一堆数据,然而没留神看手机就不晓得,我那甲醛传感器丢柜子外面大半月都遗记它的存在了,然而每次掏手机就去打游戏刷视频去了,懒得专门去看(名正言顺!)
我得再搞个法子让我连手机都不必看,就能晓得须要的信息,于是我搞了一个语音播报模块。
硬件抉择
板子仍然用便宜好用的的安信可的 ESP32S (我是不是真得去找安信可要资助啊?至多把我买板子的钱报销一下啊!)
语音播报模块我买的是:亚博智能语音合成播报模块 DF家的用太多啦,换一家省得你们说我是托 o(* ̄3 ̄)o
这个模块应用 I2C 管制和通信(连贯开发板:SDA->P21,SCL->P22)
服务器用的翼辉的 翼辉的 Spirit 1 边缘计算机(其实并不是真正的服务器,只是在这里的作用相似于服务器)。
代码解析
获取代码
为了不便解说逻辑,我会打乱代码的程序可能还会进行裁剪,要是想间接拿代码跑的敌人能够间接去 灵感桌面的机密宝库 获取代码,或者间接 clone:
https://gitee.com/inspiration-desktop/DEV-lib-arduino.git
下载或者 clone代码后这次用到的是这个四个文件夹:

设施管制命令:
通过 Spirit 1 的应用程序或者嗅探器 向传感器设施发送的命令:
语音播报命令:
{ "method": "set", "speech": "string" // string为须要播报的文本。编码方式必须为ANSI或者GB2312}
语速设置命令:
{ "method": "set", //设置语速1~10 "set_speed": 1-10}
语调设置命令:
{ "method": "set", "set_intonation": 1-10 //设置语调1~10}
音量设置命令:
{ "method": "set", "set_volume": 1-10 //设置音量1~10}
设施和协定初始化流程:
基于官网 demo 写的不须要做什么批改,次要是设施初始化,管脚配置,和协定初始化局部。
/* * 初始化语音播报模块 */static void XFS_Init(){ uint8_t n = 5; Serial.println("初始化语音播报模块"); xfs.Begin(0x30);//设施i2c地址,地址为0x50 delay(n); Serial.println("设置发音人"); xfs.SetReader(XFS5152CE::Reader_XiaoYan); //设置发音人 delay(n); Serial.println("文本的编码格局"); xfs.SetEncodingFormat(XFS5152CE::GB2312); //文本的编码格局 delay(n); Serial.println("语种判断"); xfs.SetLanguage(xfs.Language_Auto); //语种判断 delay(n); Serial.println("合成格调设置"); xfs.SetStyle(XFS5152CE::Style_Continue); //合成格调设置 delay(n); Serial.println("设置单词的发音形式"); xfs.SetArticulation(XFS5152CE::Articulation_Letter); //设置单词的发音形式 delay(n); Serial.println("设置语速"); xfs.SetSpeed(5); //设置语速1~10 delay(n); Serial.println("设置语调"); xfs.SetIntonation(5); //设置语调1~10 delay(n); Serial.println("设置音量"); xfs.SetVolume(5); //设置音量1~10 delay(n);}/* * 初始化传感器 */void sensor_init(){ // 初始化语音播报模块 XFS_Init();}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/*超时设置,示例为30S*/static uint32_t LastSpeakTime = 0;#define SpeakTimeOut 10000static const char* ssid = "EOS-Tenda"; // WiFi 名static const char* password = "1234567890"; // WiFi 明码static double illu_val;OneButton button(PIN_INPUT, true);XFS5152CE xfs;/* * 以后设施的信息定义 */DEV_INFO dev_info = { .name = "语音合成模块", .type = "device.speech", .excl = SDDC_FALSE, .desc = "ESP-32S + 亚博语音合成播报模块", .model = "IDSPEECHU01B", .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为命令处理函数 {"设置语速",SetSpeed}, {"设置语调",SetIntonation}, {"设置音量",SetVolume},};/* * 显示设施对象函数与解决办法注册 */DIS_DEV_REGINFO dis_dev[] = {// {"set_dis_demo", demo}, // 字符串为输出命令,demo为命令处理函数};/* * IO设施对象设置函数与解决办法注册 */IO_DEV_REGINFO io_dev[] = {// {"set_io_demo", demo}, // 字符串为输出命令,demo为命令处理函数 {"语音播报",speech},};/* * 零碎对象状态获取注册 */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 后依据后面的注册的函数,进入对应的处理函数。
语言播报模块会将语音播报命令中的字符串转换成中文,合成语音播报进去,还能够通过 set 命令来设置模块的性能,这里只举了用了三个罕用性能,然而实际上还有更多的性能能够设置:
/* * 语音播报函数 */sddc_bool_t speech(const char* value){ printf("开始播报\n"); xfs.StartSynthesis(value); // 抉择播报的语句 while(xfs.GetChipStatus() != xfs.ChipStatus_Idle) //查看判断是否实现 { delay(30); } return SDDC_TRUE;}/* * 设置语速函数 */sddc_bool_t SetSpeed(const uint64_t value){ printf("设置语速\n"); xfs.SetSpeed(value); return SDDC_TRUE;}/* * 设置语调函数 */sddc_bool_t SetIntonation(const uint64_t value){ printf("设置语调\n"); xfs.SetIntonation(value); return SDDC_TRUE;}/* * 设置音量函数 */sddc_bool_t SetVolume(const uint64_t value){ xfs.SetVolume(value); return SDDC_TRUE;}
总结
好了之后只有写个利用,定一个工夫或者阈值获取传感器数据让语言播报模块通知我就好了,完满!
本文仅集体学习应用,如有谬误,欢送斧正, ( )谢谢老板!