Pular para conteúdo

Sensores

Nesta página você conecta dois sensores e envia seus dados ao portal. Primeiro SHT31 (clima do gabinete), depois termistor (temperatura do aquecedor). Este é o passo "obtemos dados" antes de adicionar a lógica de controle.

O princípio de trabalho com o núcleo é simples: seu código em loop() escreve leituras frescas nos campos s_link.telemetry.*, e a fachada publica-os automaticamente na nuvem a cada telemetryPeriodMs da Config. Nenhuma necessidade de chamar a publicação manualmente.

Campos de telemetria

Para nosso gabinete, usamos três campos (índice [0] — primeira e única câmera):

Campo O que armazena Flag em Config
s_link.telemetry.airTempC[0] temperatura do ar, °C hasAirTemp
s_link.telemetry.airHumidityPct[0] umidade do ar, % hasAirHumidity
s_link.telemetry.heaterTempC[0] temperatura do aquecedor, °C hasHeaterTemp

Todos os três sinalizadores já foram incluídos no Config do passo anterior.

Regra: código de sensor não deve bloquear loop()

A fachada idryer-core serve Wi-Fi e MQTT no mesmo loop(). Portanto, ao ler sensores, você não pode chamar delay() — uma pausa quebra a sessão de rede. O sensor é consultado por um temporizador, e o valor pronto é simplesmente lido. Os drivers prontos do ecossistema já são estruturados assim.

Passo 1. SHT31: clima do gabinete

Você não precisa escrever o driver SHT31 do zero — a classe pronta Sht31ClimateSensor está no exemplo iDryer-Storage. Ele usa a biblioteca robtillaart/SHT31 e lê o sensor sem bloqueio.

  1. Adicione a biblioteca SHT31 em lib_deps do seu platformio.ini:

    lib_deps =
        file://path/to/idryer-core
        bblanchon/ArduinoJson @ ^6.21.0
        knolleary/PubSubClient
        robtillaart/SHT31 @ ^0.5.0
    
  2. Copie para sua pasta src/ quatro arquivos de iDryer-Storage/src/storage/sensors/: Sht31ClimateSensor.h, Sht31ClimateSensor.cpp, IClimateSensor.h e sensor_reading.h.

  3. Conecte o sensor por I2C (veja Esquema de conexão) e leia-o em src/main.cpp:

#include <Wire.h>
#include <iDryer.h>
#include "Sht31ClimateSensor.h"

static Sht31ClimateSensor s_climate(&Wire);
static bool               s_climateOk = false;

void setup() {
    Serial.begin(115200);
    Wire.begin(8, 9);                 // SDA, SCL — pinos da sua placa
    s_climateOk = s_climate.begin();  // encontra automaticamente o endereço 0x44 ou 0x45
    s_link.begin();
}

void loop() {
    s_link.loop();

    if (s_climateOk) {
        s_climate.tick(millis());
        SensorReading r = s_climate.get();
        if (r.ok) {
            s_link.telemetry.airTempC[0]       = r.temperature;
            s_link.telemetry.airHumidityPct[0] = r.humidity;
        }
    }
}

A estrutura SensorReading (campos ok, temperature, humidity) é declarada em sensor_reading.h. Após flashear, o portal mostrará temperatura e umidade do gabinete — este é o primeiro feedback do dispositivo.

Passo 2. Termistor: temperatura do aquecedor

Não tenho uma classe pronta de termistor para ESP32, então lemos e escrevemos diretamente em src/main.cpp. O termistor está conectado ao pino ADC através de um conversor de tensão (veja Esquema de conexão): o controlador mede a tensão no ponto central, por ela calcula a resistência do termistor, depois a temperatura.

#include <math.h>

static const int   THERM_PIN  = 2;         // pino ADC
static const float SERIES_R   = 4700.0f;   // resistor divisor, Ω
static const float NOMINAL_R  = 100000.0f; // resistência do termistor a 25 °C, Ω
static const float NOMINAL_T  = 25.0f;     // °C
static const float BETA       = 3950.0f;   // coeficiente B do termistor

// Retorna a temperatura do aquecedor em °C.
static float readHeaterTempC() {
    int   raw = analogRead(THERM_PIN);          // 0..4095 em ESP32
    float v   = (float)raw / 4095.0f;           // fração da escala completa
    float r   = SERIES_R * (1.0f - v) / v;      // resistência do termistor, Ω
    // Equação de Steinhart–Hart (parâmetro B):
    float tK  = 1.0f / (1.0f / (NOMINAL_T + 273.15f) + logf(r / NOMINAL_R) / BETA);
    return tK - 273.15f;
}

Em loop() escreva o resultado na telemetria perto da leitura SHT31:

s_link.telemetry.heaterTempC[0] = readHeaterTempC();

Esta é leitura simplificada — ajuste os parâmetros para seu termistor

Constantes NOMINAL_R e BETA dependem do termistor específico — pegue-as de sua folha de dados (termistor doméstico comum — Genérico 3950, 100 kΩ). A fórmula divisor corresponde ao esquema de Esquema de conexão: termistor para 3.3V, resistor para GND. Com outro roteamento a fórmula muda. ADC em ESP32 é não-linear, então para medições precisas as leituras são calibradas — nos controladores iDryer em série uma tabela de termistor é usada (biblioteca Thermistor).

Verificação de termistor com multímetro — Verificação de termistor.

Completo src/main.cpp após este capítulo

Abaixo — o arquivo inteiro. Novas linhas em relação ao capítulo anterior são marcadas // ← capítulo 5; o resto não mudou.

??? nota «O que foi — src/main.cpp após capítulo 4»

```cpp
#include <iDryer.h>

static const iDryer::Config CFG = {
    .deviceType        = iDryer::DeviceType::Dryer,
    .unitsCount        = 1,
    .hasHeater         = true,
    .hasFan            = true,
    .hasAirTemp        = true,
    .hasAirHumidity    = true,
    .hasHeaterTemp     = true,
    .telemetryPeriodMs = 5000,
    .statusPeriodMs    = 10000,
    .hardwareVersion   = "1.0",
    .firmwareVersion   = "0.1.0",
    .model             = "DIY Storage Cabinet",
};
static iDryer::Link s_link(CFG);

void setup() {
    Serial.begin(115200);
    s_link.begin();
}

void loop() {
    s_link.loop();
}
```
#include <iDryer.h>
#include <Wire.h>                  // ← capítulo 5
#include <math.h>                  // ← capítulo 5
#include "Sht31ClimateSensor.h"    // ← capítulo 5

static const iDryer::Config CFG = {
    .deviceType        = iDryer::DeviceType::Dryer,
    .unitsCount        = 1,
    .hasHeater         = true,
    .hasFan            = true,
    .hasAirTemp        = true,
    .hasAirHumidity    = true,
    .hasHeaterTemp     = true,
    .telemetryPeriodMs = 5000,
    .statusPeriodMs    = 10000,
    .hardwareVersion   = "1.0",
    .firmwareVersion   = "0.1.0",
    .model             = "DIY Storage Cabinet",
};
static iDryer::Link s_link(CFG);

// ← capítulo 5: sensor de clima SHT31
static Sht31ClimateSensor s_climate(&Wire);
static bool               s_climateOk = false;

// ← capítulo 5: termistor do aquecedor
static const int   THERM_PIN  = 2;
static const float SERIES_R   = 4700.0f;
static const float NOMINAL_R  = 100000.0f;
static const float NOMINAL_T  = 25.0f;
static const float BETA       = 3950.0f;

static float readHeaterTempC() {
    int   raw = analogRead(THERM_PIN);
    float v   = (float)raw / 4095.0f;
    float r   = SERIES_R * (1.0f - v) / v;
    float tK  = 1.0f / (1.0f / (NOMINAL_T + 273.15f) + logf(r / NOMINAL_R) / BETA);
    return tK - 273.15f;
}

void setup() {
    Serial.begin(115200);
    Wire.begin(8, 9);                 // ← capítulo 5  (SDA, SCL — pinos da sua placa)
    s_climateOk = s_climate.begin();  // ← capítulo 5
    s_link.begin();
}

void loop() {
    s_link.loop();

    if (s_climateOk) {                                       // ← capítulo 5
        s_climate.tick(millis());
        SensorReading r = s_climate.get();
        if (r.ok) {
            s_link.telemetry.airTempC[0]       = r.temperature;
            s_link.telemetry.airHumidityPct[0] = r.humidity;
        }
    }
    s_link.telemetry.heaterTempC[0] = readHeaterTempC();     // ← capítulo 5
}

Verificação de resultado

Após este passo, o portal deve exibir três valores:

  • temperatura do ar no gabinete;
  • umidade no gabinete;
  • temperatura do aquecedor.

Se as leituras "flutuam" ou são claramente incorretas:

  • verifique terra comum e roteamento (interferência de fios de potência) — Erros de fiação;
  • verifique o valor do resistor divisor e tipo do termistor;
  • certifique-se de que SHT31 responde em I2C (endereço correto e linhas).

Diagnóstico "sensor mostra bobagem" — Verificação de termistor e Erros típicos.

O que vem a seguir

Dados de sensores existem. Agora descrevemos as configurações do dispositivo (temperatura-alvo, histerese) em Menu em YAML, para que possam ser alteradas do portal e armazenadas na memória.