mirror of
https://github.com/esphome/esphome.git
synced 2024-11-29 10:14:13 +01:00
Total daily energy methods (#2163)
Co-authored-by: Chris Nussbaum <chris.nussbaum@protolabs.com>
This commit is contained in:
parent
8eb18995cb
commit
ebaa84617f
3 changed files with 40 additions and 1 deletions
|
@ -7,6 +7,7 @@ from esphome.const import (
|
||||||
DEVICE_CLASS_ENERGY,
|
DEVICE_CLASS_ENERGY,
|
||||||
LAST_RESET_TYPE_AUTO,
|
LAST_RESET_TYPE_AUTO,
|
||||||
STATE_CLASS_MEASUREMENT,
|
STATE_CLASS_MEASUREMENT,
|
||||||
|
CONF_METHOD,
|
||||||
)
|
)
|
||||||
|
|
||||||
DEPENDENCIES = ["time"]
|
DEPENDENCIES = ["time"]
|
||||||
|
@ -14,6 +15,12 @@ DEPENDENCIES = ["time"]
|
||||||
CONF_POWER_ID = "power_id"
|
CONF_POWER_ID = "power_id"
|
||||||
CONF_MIN_SAVE_INTERVAL = "min_save_interval"
|
CONF_MIN_SAVE_INTERVAL = "min_save_interval"
|
||||||
total_daily_energy_ns = cg.esphome_ns.namespace("total_daily_energy")
|
total_daily_energy_ns = cg.esphome_ns.namespace("total_daily_energy")
|
||||||
|
TotalDailyEnergyMethod = total_daily_energy_ns.enum("TotalDailyEnergyMethod")
|
||||||
|
TOTAL_DAILY_ENERGY_METHODS = {
|
||||||
|
"trapezoid": TotalDailyEnergyMethod.TOTAL_DAILY_ENERGY_METHOD_TRAPEZOID,
|
||||||
|
"left": TotalDailyEnergyMethod.TOTAL_DAILY_ENERGY_METHOD_LEFT,
|
||||||
|
"right": TotalDailyEnergyMethod.TOTAL_DAILY_ENERGY_METHOD_RIGHT,
|
||||||
|
}
|
||||||
TotalDailyEnergy = total_daily_energy_ns.class_(
|
TotalDailyEnergy = total_daily_energy_ns.class_(
|
||||||
"TotalDailyEnergy", sensor.Sensor, cg.Component
|
"TotalDailyEnergy", sensor.Sensor, cg.Component
|
||||||
)
|
)
|
||||||
|
@ -33,6 +40,9 @@ CONFIG_SCHEMA = (
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_MIN_SAVE_INTERVAL, default="0s"
|
CONF_MIN_SAVE_INTERVAL, default="0s"
|
||||||
): cv.positive_time_period_milliseconds,
|
): cv.positive_time_period_milliseconds,
|
||||||
|
cv.Optional(CONF_METHOD, default="right"): cv.enum(
|
||||||
|
TOTAL_DAILY_ENERGY_METHODS, lower=True
|
||||||
|
),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.extend(cv.COMPONENT_SCHEMA)
|
.extend(cv.COMPONENT_SCHEMA)
|
||||||
|
@ -50,3 +60,4 @@ async def to_code(config):
|
||||||
time_ = await cg.get_variable(config[CONF_TIME_ID])
|
time_ = await cg.get_variable(config[CONF_TIME_ID])
|
||||||
cg.add(var.set_time(time_))
|
cg.add(var.set_time(time_))
|
||||||
cg.add(var.set_min_save_interval(config[CONF_MIN_SAVE_INTERVAL]))
|
cg.add(var.set_min_save_interval(config[CONF_MIN_SAVE_INTERVAL]))
|
||||||
|
cg.add(var.set_method(config[CONF_METHOD]))
|
||||||
|
|
|
@ -20,7 +20,9 @@ void TotalDailyEnergy::setup() {
|
||||||
|
|
||||||
this->parent_->add_on_state_callback([this](float state) { this->process_new_state_(state); });
|
this->parent_->add_on_state_callback([this](float state) { this->process_new_state_(state); });
|
||||||
}
|
}
|
||||||
|
|
||||||
void TotalDailyEnergy::dump_config() { LOG_SENSOR("", "Total Daily Energy", this); }
|
void TotalDailyEnergy::dump_config() { LOG_SENSOR("", "Total Daily Energy", this); }
|
||||||
|
|
||||||
void TotalDailyEnergy::loop() {
|
void TotalDailyEnergy::loop() {
|
||||||
auto t = this->time_->now();
|
auto t = this->time_->now();
|
||||||
if (!t.is_valid())
|
if (!t.is_valid())
|
||||||
|
@ -37,6 +39,7 @@ void TotalDailyEnergy::loop() {
|
||||||
this->publish_state_and_save(0);
|
this->publish_state_and_save(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void TotalDailyEnergy::publish_state_and_save(float state) {
|
void TotalDailyEnergy::publish_state_and_save(float state) {
|
||||||
this->total_energy_ = state;
|
this->total_energy_ = state;
|
||||||
this->publish_state(state);
|
this->publish_state(state);
|
||||||
|
@ -47,13 +50,29 @@ void TotalDailyEnergy::publish_state_and_save(float state) {
|
||||||
this->last_save_ = now;
|
this->last_save_ = now;
|
||||||
this->pref_.save(&state);
|
this->pref_.save(&state);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TotalDailyEnergy::process_new_state_(float state) {
|
void TotalDailyEnergy::process_new_state_(float state) {
|
||||||
if (isnan(state))
|
if (isnan(state))
|
||||||
return;
|
return;
|
||||||
const uint32_t now = millis();
|
const uint32_t now = millis();
|
||||||
|
const float old_state = this->last_power_state_;
|
||||||
|
const float new_state = state;
|
||||||
float delta_hours = (now - this->last_update_) / 1000.0f / 60.0f / 60.0f;
|
float delta_hours = (now - this->last_update_) / 1000.0f / 60.0f / 60.0f;
|
||||||
|
float delta_energy = 0.0f;
|
||||||
|
switch (this->method_) {
|
||||||
|
case TOTAL_DAILY_ENERGY_METHOD_TRAPEZOID:
|
||||||
|
delta_energy = delta_hours * (old_state + new_state) / 2.0;
|
||||||
|
break;
|
||||||
|
case TOTAL_DAILY_ENERGY_METHOD_LEFT:
|
||||||
|
delta_energy = delta_hours * old_state;
|
||||||
|
break;
|
||||||
|
case TOTAL_DAILY_ENERGY_METHOD_RIGHT:
|
||||||
|
delta_energy = delta_hours * new_state;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
this->last_power_state_ = new_state;
|
||||||
this->last_update_ = now;
|
this->last_update_ = now;
|
||||||
this->publish_state_and_save(this->total_energy_ + state * delta_hours);
|
this->publish_state_and_save(this->total_energy_ + delta_energy);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace total_daily_energy
|
} // namespace total_daily_energy
|
||||||
|
|
|
@ -8,11 +8,18 @@
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace total_daily_energy {
|
namespace total_daily_energy {
|
||||||
|
|
||||||
|
enum TotalDailyEnergyMethod {
|
||||||
|
TOTAL_DAILY_ENERGY_METHOD_TRAPEZOID = 0,
|
||||||
|
TOTAL_DAILY_ENERGY_METHOD_LEFT,
|
||||||
|
TOTAL_DAILY_ENERGY_METHOD_RIGHT,
|
||||||
|
};
|
||||||
|
|
||||||
class TotalDailyEnergy : public sensor::Sensor, public Component {
|
class TotalDailyEnergy : public sensor::Sensor, public Component {
|
||||||
public:
|
public:
|
||||||
void set_min_save_interval(uint32_t min_interval) { this->min_save_interval_ = min_interval; }
|
void set_min_save_interval(uint32_t min_interval) { this->min_save_interval_ = min_interval; }
|
||||||
void set_time(time::RealTimeClock *time) { time_ = time; }
|
void set_time(time::RealTimeClock *time) { time_ = time; }
|
||||||
void set_parent(Sensor *parent) { parent_ = parent; }
|
void set_parent(Sensor *parent) { parent_ = parent; }
|
||||||
|
void set_method(TotalDailyEnergyMethod method) { method_ = method; }
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
float get_setup_priority() const override { return setup_priority::DATA; }
|
||||||
|
@ -29,11 +36,13 @@ class TotalDailyEnergy : public sensor::Sensor, public Component {
|
||||||
ESPPreferenceObject pref_;
|
ESPPreferenceObject pref_;
|
||||||
time::RealTimeClock *time_;
|
time::RealTimeClock *time_;
|
||||||
Sensor *parent_;
|
Sensor *parent_;
|
||||||
|
TotalDailyEnergyMethod method_;
|
||||||
uint16_t last_day_of_year_{};
|
uint16_t last_day_of_year_{};
|
||||||
uint32_t last_update_{0};
|
uint32_t last_update_{0};
|
||||||
uint32_t last_save_{0};
|
uint32_t last_save_{0};
|
||||||
uint32_t min_save_interval_{0};
|
uint32_t min_save_interval_{0};
|
||||||
float total_energy_{0.0f};
|
float total_energy_{0.0f};
|
||||||
|
float last_power_state_{0.0f};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace total_daily_energy
|
} // namespace total_daily_energy
|
||||||
|
|
Loading…
Reference in a new issue