mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 05:03:52 +01:00 
			
		
		
		
	Add support for fan preset modes (#5694)
Co-authored-by: J. Nick Koston <nick@koston.org>
This commit is contained in:
		| @@ -365,6 +365,7 @@ message ListEntitiesFanResponse { | |||||||
|   bool disabled_by_default = 9; |   bool disabled_by_default = 9; | ||||||
|   string icon = 10; |   string icon = 10; | ||||||
|   EntityCategory entity_category = 11; |   EntityCategory entity_category = 11; | ||||||
|  |   repeated string supported_preset_modes = 12; | ||||||
| } | } | ||||||
| enum FanSpeed { | enum FanSpeed { | ||||||
|   FAN_SPEED_LOW = 0; |   FAN_SPEED_LOW = 0; | ||||||
| @@ -387,6 +388,7 @@ message FanStateResponse { | |||||||
|   FanSpeed speed = 4 [deprecated = true]; |   FanSpeed speed = 4 [deprecated = true]; | ||||||
|   FanDirection direction = 5; |   FanDirection direction = 5; | ||||||
|   int32 speed_level = 6; |   int32 speed_level = 6; | ||||||
|  |   string preset_mode = 7; | ||||||
| } | } | ||||||
| message FanCommandRequest { | message FanCommandRequest { | ||||||
|   option (id) = 31; |   option (id) = 31; | ||||||
| @@ -405,6 +407,8 @@ message FanCommandRequest { | |||||||
|   FanDirection direction = 9; |   FanDirection direction = 9; | ||||||
|   bool has_speed_level = 10; |   bool has_speed_level = 10; | ||||||
|   int32 speed_level = 11; |   int32 speed_level = 11; | ||||||
|  |   bool has_preset_mode = 12; | ||||||
|  |   string preset_mode = 13; | ||||||
| } | } | ||||||
|  |  | ||||||
| // ==================== LIGHT ==================== | // ==================== LIGHT ==================== | ||||||
|   | |||||||
| @@ -293,6 +293,8 @@ bool APIConnection::send_fan_state(fan::Fan *fan) { | |||||||
|   } |   } | ||||||
|   if (traits.supports_direction()) |   if (traits.supports_direction()) | ||||||
|     resp.direction = static_cast<enums::FanDirection>(fan->direction); |     resp.direction = static_cast<enums::FanDirection>(fan->direction); | ||||||
|  |   if (traits.supports_preset_modes()) | ||||||
|  |     resp.preset_mode = fan->preset_mode; | ||||||
|   return this->send_fan_state_response(resp); |   return this->send_fan_state_response(resp); | ||||||
| } | } | ||||||
| bool APIConnection::send_fan_info(fan::Fan *fan) { | bool APIConnection::send_fan_info(fan::Fan *fan) { | ||||||
| @@ -307,6 +309,8 @@ bool APIConnection::send_fan_info(fan::Fan *fan) { | |||||||
|   msg.supports_speed = traits.supports_speed(); |   msg.supports_speed = traits.supports_speed(); | ||||||
|   msg.supports_direction = traits.supports_direction(); |   msg.supports_direction = traits.supports_direction(); | ||||||
|   msg.supported_speed_count = traits.supported_speed_count(); |   msg.supported_speed_count = traits.supported_speed_count(); | ||||||
|  |   for (auto const &preset : traits.supported_preset_modes()) | ||||||
|  |     msg.supported_preset_modes.push_back(preset); | ||||||
|   msg.disabled_by_default = fan->is_disabled_by_default(); |   msg.disabled_by_default = fan->is_disabled_by_default(); | ||||||
|   msg.icon = fan->get_icon(); |   msg.icon = fan->get_icon(); | ||||||
|   msg.entity_category = static_cast<enums::EntityCategory>(fan->get_entity_category()); |   msg.entity_category = static_cast<enums::EntityCategory>(fan->get_entity_category()); | ||||||
| @@ -328,6 +332,8 @@ void APIConnection::fan_command(const FanCommandRequest &msg) { | |||||||
|   } |   } | ||||||
|   if (msg.has_direction) |   if (msg.has_direction) | ||||||
|     call.set_direction(static_cast<fan::FanDirection>(msg.direction)); |     call.set_direction(static_cast<fan::FanDirection>(msg.direction)); | ||||||
|  |   if (msg.has_preset_mode) | ||||||
|  |     call.set_preset_mode(msg.preset_mode); | ||||||
|   call.perform(); |   call.perform(); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -1375,6 +1375,10 @@ bool ListEntitiesFanResponse::decode_length(uint32_t field_id, ProtoLengthDelimi | |||||||
|       this->icon = value.as_string(); |       this->icon = value.as_string(); | ||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
|  |     case 12: { | ||||||
|  |       this->supported_preset_modes.push_back(value.as_string()); | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|     default: |     default: | ||||||
|       return false; |       return false; | ||||||
|   } |   } | ||||||
| @@ -1401,6 +1405,9 @@ void ListEntitiesFanResponse::encode(ProtoWriteBuffer buffer) const { | |||||||
|   buffer.encode_bool(9, this->disabled_by_default); |   buffer.encode_bool(9, this->disabled_by_default); | ||||||
|   buffer.encode_string(10, this->icon); |   buffer.encode_string(10, this->icon); | ||||||
|   buffer.encode_enum<enums::EntityCategory>(11, this->entity_category); |   buffer.encode_enum<enums::EntityCategory>(11, this->entity_category); | ||||||
|  |   for (auto &it : this->supported_preset_modes) { | ||||||
|  |     buffer.encode_string(12, it, true); | ||||||
|  |   } | ||||||
| } | } | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
| void ListEntitiesFanResponse::dump_to(std::string &out) const { | void ListEntitiesFanResponse::dump_to(std::string &out) const { | ||||||
| @@ -1451,6 +1458,12 @@ void ListEntitiesFanResponse::dump_to(std::string &out) const { | |||||||
|   out.append("  entity_category: "); |   out.append("  entity_category: "); | ||||||
|   out.append(proto_enum_to_string<enums::EntityCategory>(this->entity_category)); |   out.append(proto_enum_to_string<enums::EntityCategory>(this->entity_category)); | ||||||
|   out.append("\n"); |   out.append("\n"); | ||||||
|  |  | ||||||
|  |   for (const auto &it : this->supported_preset_modes) { | ||||||
|  |     out.append("  supported_preset_modes: "); | ||||||
|  |     out.append("'").append(it).append("'"); | ||||||
|  |     out.append("\n"); | ||||||
|  |   } | ||||||
|   out.append("}"); |   out.append("}"); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
| @@ -1480,6 +1493,16 @@ bool FanStateResponse::decode_varint(uint32_t field_id, ProtoVarInt value) { | |||||||
|       return false; |       return false; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | bool FanStateResponse::decode_length(uint32_t field_id, ProtoLengthDelimited value) { | ||||||
|  |   switch (field_id) { | ||||||
|  |     case 7: { | ||||||
|  |       this->preset_mode = value.as_string(); | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |     default: | ||||||
|  |       return false; | ||||||
|  |   } | ||||||
|  | } | ||||||
| bool FanStateResponse::decode_32bit(uint32_t field_id, Proto32Bit value) { | bool FanStateResponse::decode_32bit(uint32_t field_id, Proto32Bit value) { | ||||||
|   switch (field_id) { |   switch (field_id) { | ||||||
|     case 1: { |     case 1: { | ||||||
| @@ -1497,6 +1520,7 @@ void FanStateResponse::encode(ProtoWriteBuffer buffer) const { | |||||||
|   buffer.encode_enum<enums::FanSpeed>(4, this->speed); |   buffer.encode_enum<enums::FanSpeed>(4, this->speed); | ||||||
|   buffer.encode_enum<enums::FanDirection>(5, this->direction); |   buffer.encode_enum<enums::FanDirection>(5, this->direction); | ||||||
|   buffer.encode_int32(6, this->speed_level); |   buffer.encode_int32(6, this->speed_level); | ||||||
|  |   buffer.encode_string(7, this->preset_mode); | ||||||
| } | } | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
| void FanStateResponse::dump_to(std::string &out) const { | void FanStateResponse::dump_to(std::string &out) const { | ||||||
| @@ -1527,6 +1551,10 @@ void FanStateResponse::dump_to(std::string &out) const { | |||||||
|   sprintf(buffer, "%" PRId32, this->speed_level); |   sprintf(buffer, "%" PRId32, this->speed_level); | ||||||
|   out.append(buffer); |   out.append(buffer); | ||||||
|   out.append("\n"); |   out.append("\n"); | ||||||
|  |  | ||||||
|  |   out.append("  preset_mode: "); | ||||||
|  |   out.append("'").append(this->preset_mode).append("'"); | ||||||
|  |   out.append("\n"); | ||||||
|   out.append("}"); |   out.append("}"); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
| @@ -1572,6 +1600,20 @@ bool FanCommandRequest::decode_varint(uint32_t field_id, ProtoVarInt value) { | |||||||
|       this->speed_level = value.as_int32(); |       this->speed_level = value.as_int32(); | ||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
|  |     case 12: { | ||||||
|  |       this->has_preset_mode = value.as_bool(); | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |     default: | ||||||
|  |       return false; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | bool FanCommandRequest::decode_length(uint32_t field_id, ProtoLengthDelimited value) { | ||||||
|  |   switch (field_id) { | ||||||
|  |     case 13: { | ||||||
|  |       this->preset_mode = value.as_string(); | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|     default: |     default: | ||||||
|       return false; |       return false; | ||||||
|   } |   } | ||||||
| @@ -1598,6 +1640,8 @@ void FanCommandRequest::encode(ProtoWriteBuffer buffer) const { | |||||||
|   buffer.encode_enum<enums::FanDirection>(9, this->direction); |   buffer.encode_enum<enums::FanDirection>(9, this->direction); | ||||||
|   buffer.encode_bool(10, this->has_speed_level); |   buffer.encode_bool(10, this->has_speed_level); | ||||||
|   buffer.encode_int32(11, this->speed_level); |   buffer.encode_int32(11, this->speed_level); | ||||||
|  |   buffer.encode_bool(12, this->has_preset_mode); | ||||||
|  |   buffer.encode_string(13, this->preset_mode); | ||||||
| } | } | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
| void FanCommandRequest::dump_to(std::string &out) const { | void FanCommandRequest::dump_to(std::string &out) const { | ||||||
| @@ -1648,6 +1692,14 @@ void FanCommandRequest::dump_to(std::string &out) const { | |||||||
|   sprintf(buffer, "%" PRId32, this->speed_level); |   sprintf(buffer, "%" PRId32, this->speed_level); | ||||||
|   out.append(buffer); |   out.append(buffer); | ||||||
|   out.append("\n"); |   out.append("\n"); | ||||||
|  |  | ||||||
|  |   out.append("  has_preset_mode: "); | ||||||
|  |   out.append(YESNO(this->has_preset_mode)); | ||||||
|  |   out.append("\n"); | ||||||
|  |  | ||||||
|  |   out.append("  preset_mode: "); | ||||||
|  |   out.append("'").append(this->preset_mode).append("'"); | ||||||
|  |   out.append("\n"); | ||||||
|   out.append("}"); |   out.append("}"); | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -472,6 +472,7 @@ class ListEntitiesFanResponse : public ProtoMessage { | |||||||
|   bool disabled_by_default{false}; |   bool disabled_by_default{false}; | ||||||
|   std::string icon{}; |   std::string icon{}; | ||||||
|   enums::EntityCategory entity_category{}; |   enums::EntityCategory entity_category{}; | ||||||
|  |   std::vector<std::string> supported_preset_modes{}; | ||||||
|   void encode(ProtoWriteBuffer buffer) const override; |   void encode(ProtoWriteBuffer buffer) const override; | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
|   void dump_to(std::string &out) const override; |   void dump_to(std::string &out) const override; | ||||||
| @@ -490,6 +491,7 @@ class FanStateResponse : public ProtoMessage { | |||||||
|   enums::FanSpeed speed{}; |   enums::FanSpeed speed{}; | ||||||
|   enums::FanDirection direction{}; |   enums::FanDirection direction{}; | ||||||
|   int32_t speed_level{0}; |   int32_t speed_level{0}; | ||||||
|  |   std::string preset_mode{}; | ||||||
|   void encode(ProtoWriteBuffer buffer) const override; |   void encode(ProtoWriteBuffer buffer) const override; | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
|   void dump_to(std::string &out) const override; |   void dump_to(std::string &out) const override; | ||||||
| @@ -497,6 +499,7 @@ class FanStateResponse : public ProtoMessage { | |||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   bool decode_32bit(uint32_t field_id, Proto32Bit value) override; |   bool decode_32bit(uint32_t field_id, Proto32Bit value) override; | ||||||
|  |   bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override; | ||||||
|   bool decode_varint(uint32_t field_id, ProtoVarInt value) override; |   bool decode_varint(uint32_t field_id, ProtoVarInt value) override; | ||||||
| }; | }; | ||||||
| class FanCommandRequest : public ProtoMessage { | class FanCommandRequest : public ProtoMessage { | ||||||
| @@ -512,6 +515,8 @@ class FanCommandRequest : public ProtoMessage { | |||||||
|   enums::FanDirection direction{}; |   enums::FanDirection direction{}; | ||||||
|   bool has_speed_level{false}; |   bool has_speed_level{false}; | ||||||
|   int32_t speed_level{0}; |   int32_t speed_level{0}; | ||||||
|  |   bool has_preset_mode{false}; | ||||||
|  |   std::string preset_mode{}; | ||||||
|   void encode(ProtoWriteBuffer buffer) const override; |   void encode(ProtoWriteBuffer buffer) const override; | ||||||
| #ifdef HAS_PROTO_MESSAGE_DUMP | #ifdef HAS_PROTO_MESSAGE_DUMP | ||||||
|   void dump_to(std::string &out) const override; |   void dump_to(std::string &out) const override; | ||||||
| @@ -519,6 +524,7 @@ class FanCommandRequest : public ProtoMessage { | |||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   bool decode_32bit(uint32_t field_id, Proto32Bit value) override; |   bool decode_32bit(uint32_t field_id, Proto32Bit value) override; | ||||||
|  |   bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override; | ||||||
|   bool decode_varint(uint32_t field_id, ProtoVarInt value) override; |   bool decode_varint(uint32_t field_id, ProtoVarInt value) override; | ||||||
| }; | }; | ||||||
| class ListEntitiesLightResponse : public ProtoMessage { | class ListEntitiesLightResponse : public ProtoMessage { | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ void CopyFan::setup() { | |||||||
|     this->oscillating = source_->oscillating; |     this->oscillating = source_->oscillating; | ||||||
|     this->speed = source_->speed; |     this->speed = source_->speed; | ||||||
|     this->direction = source_->direction; |     this->direction = source_->direction; | ||||||
|  |     this->preset_mode = source_->preset_mode; | ||||||
|     this->publish_state(); |     this->publish_state(); | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
| @@ -19,6 +20,7 @@ void CopyFan::setup() { | |||||||
|   this->oscillating = source_->oscillating; |   this->oscillating = source_->oscillating; | ||||||
|   this->speed = source_->speed; |   this->speed = source_->speed; | ||||||
|   this->direction = source_->direction; |   this->direction = source_->direction; | ||||||
|  |   this->preset_mode = source_->preset_mode; | ||||||
|   this->publish_state(); |   this->publish_state(); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -33,6 +35,7 @@ fan::FanTraits CopyFan::get_traits() { | |||||||
|   traits.set_speed(base.supports_speed()); |   traits.set_speed(base.supports_speed()); | ||||||
|   traits.set_supported_speed_count(base.supported_speed_count()); |   traits.set_supported_speed_count(base.supported_speed_count()); | ||||||
|   traits.set_direction(base.supports_direction()); |   traits.set_direction(base.supports_direction()); | ||||||
|  |   traits.set_supported_preset_modes(base.supported_preset_modes()); | ||||||
|   return traits; |   return traits; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -46,6 +49,8 @@ void CopyFan::control(const fan::FanCall &call) { | |||||||
|     call2.set_speed(*call.get_speed()); |     call2.set_speed(*call.get_speed()); | ||||||
|   if (call.get_direction().has_value()) |   if (call.get_direction().has_value()) | ||||||
|     call2.set_direction(*call.get_direction()); |     call2.set_direction(*call.get_direction()); | ||||||
|  |   if (!call.get_preset_mode().empty()) | ||||||
|  |     call2.set_preset_mode(call.get_preset_mode()); | ||||||
|   call2.perform(); |   call2.perform(); | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ from esphome.const import ( | |||||||
|     CONF_ON_SPEED_SET, |     CONF_ON_SPEED_SET, | ||||||
|     CONF_ON_TURN_OFF, |     CONF_ON_TURN_OFF, | ||||||
|     CONF_ON_TURN_ON, |     CONF_ON_TURN_ON, | ||||||
|  |     CONF_ON_PRESET_SET, | ||||||
|     CONF_TRIGGER_ID, |     CONF_TRIGGER_ID, | ||||||
|     CONF_DIRECTION, |     CONF_DIRECTION, | ||||||
|     CONF_RESTORE_MODE, |     CONF_RESTORE_MODE, | ||||||
| @@ -57,6 +58,9 @@ CycleSpeedAction = fan_ns.class_("CycleSpeedAction", automation.Action) | |||||||
| FanTurnOnTrigger = fan_ns.class_("FanTurnOnTrigger", automation.Trigger.template()) | FanTurnOnTrigger = fan_ns.class_("FanTurnOnTrigger", automation.Trigger.template()) | ||||||
| FanTurnOffTrigger = fan_ns.class_("FanTurnOffTrigger", automation.Trigger.template()) | FanTurnOffTrigger = fan_ns.class_("FanTurnOffTrigger", automation.Trigger.template()) | ||||||
| FanSpeedSetTrigger = fan_ns.class_("FanSpeedSetTrigger", automation.Trigger.template()) | FanSpeedSetTrigger = fan_ns.class_("FanSpeedSetTrigger", automation.Trigger.template()) | ||||||
|  | FanPresetSetTrigger = fan_ns.class_( | ||||||
|  |     "FanPresetSetTrigger", automation.Trigger.template() | ||||||
|  | ) | ||||||
|  |  | ||||||
| FanIsOnCondition = fan_ns.class_("FanIsOnCondition", automation.Condition.template()) | FanIsOnCondition = fan_ns.class_("FanIsOnCondition", automation.Condition.template()) | ||||||
| FanIsOffCondition = fan_ns.class_("FanIsOffCondition", automation.Condition.template()) | FanIsOffCondition = fan_ns.class_("FanIsOffCondition", automation.Condition.template()) | ||||||
| @@ -101,9 +105,46 @@ FAN_SCHEMA = cv.ENTITY_BASE_SCHEMA.extend(cv.MQTT_COMMAND_COMPONENT_SCHEMA).exte | |||||||
|                 cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(FanSpeedSetTrigger), |                 cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(FanSpeedSetTrigger), | ||||||
|             } |             } | ||||||
|         ), |         ), | ||||||
|  |         cv.Optional(CONF_ON_PRESET_SET): automation.validate_automation( | ||||||
|  |             { | ||||||
|  |                 cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(FanPresetSetTrigger), | ||||||
|  |             } | ||||||
|  |         ), | ||||||
|     } |     } | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | _PRESET_MODES_SCHEMA = cv.All( | ||||||
|  |     cv.ensure_list(cv.string_strict), | ||||||
|  |     cv.Length(min=1), | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def validate_preset_modes(value): | ||||||
|  |     # Check against defined schema | ||||||
|  |     value = _PRESET_MODES_SCHEMA(value) | ||||||
|  |  | ||||||
|  |     # Ensure preset names are unique | ||||||
|  |     errors = [] | ||||||
|  |     presets = set() | ||||||
|  |     for i, preset in enumerate(value): | ||||||
|  |         # If name does not exist yet add it | ||||||
|  |         if preset not in presets: | ||||||
|  |             presets.add(preset) | ||||||
|  |             continue | ||||||
|  |  | ||||||
|  |         # Otherwise it's an error | ||||||
|  |         errors.append( | ||||||
|  |             cv.Invalid( | ||||||
|  |                 f"Found duplicate preset name '{preset}'. Presets must have unique names.", | ||||||
|  |                 [i], | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     if errors: | ||||||
|  |         raise cv.MultipleInvalid(errors) | ||||||
|  |  | ||||||
|  |     return value | ||||||
|  |  | ||||||
|  |  | ||||||
| async def setup_fan_core_(var, config): | async def setup_fan_core_(var, config): | ||||||
|     await setup_entity(var, config) |     await setup_entity(var, config) | ||||||
| @@ -154,6 +195,9 @@ async def setup_fan_core_(var, config): | |||||||
|     for conf in config.get(CONF_ON_SPEED_SET, []): |     for conf in config.get(CONF_ON_SPEED_SET, []): | ||||||
|         trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var) |         trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var) | ||||||
|         await automation.build_automation(trigger, [], conf) |         await automation.build_automation(trigger, [], conf) | ||||||
|  |     for conf in config.get(CONF_ON_PRESET_SET, []): | ||||||
|  |         trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var) | ||||||
|  |         await automation.build_automation(trigger, [], conf) | ||||||
|  |  | ||||||
|  |  | ||||||
| async def register_fan(var, config): | async def register_fan(var, config): | ||||||
|   | |||||||
| @@ -165,5 +165,23 @@ class FanSpeedSetTrigger : public Trigger<> { | |||||||
|   int last_speed_; |   int last_speed_; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | class FanPresetSetTrigger : public Trigger<> { | ||||||
|  |  public: | ||||||
|  |   FanPresetSetTrigger(Fan *state) { | ||||||
|  |     state->add_on_state_callback([this, state]() { | ||||||
|  |       auto preset_mode = state->preset_mode; | ||||||
|  |       auto should_trigger = preset_mode != this->last_preset_mode_; | ||||||
|  |       this->last_preset_mode_ = preset_mode; | ||||||
|  |       if (should_trigger) { | ||||||
|  |         this->trigger(); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     this->last_preset_mode_ = state->preset_mode; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |  protected: | ||||||
|  |   std::string last_preset_mode_; | ||||||
|  | }; | ||||||
|  |  | ||||||
| }  // namespace fan | }  // namespace fan | ||||||
| }  // namespace esphome | }  // namespace esphome | ||||||
|   | |||||||
| @@ -32,9 +32,12 @@ void FanCall::perform() { | |||||||
|   if (this->direction_.has_value()) { |   if (this->direction_.has_value()) { | ||||||
|     ESP_LOGD(TAG, "  Direction: %s", LOG_STR_ARG(fan_direction_to_string(*this->direction_))); |     ESP_LOGD(TAG, "  Direction: %s", LOG_STR_ARG(fan_direction_to_string(*this->direction_))); | ||||||
|   } |   } | ||||||
|  |   if (!this->preset_mode_.empty()) { | ||||||
|  |     ESP_LOGD(TAG, "  Preset Mode: %s", this->preset_mode_.c_str()); | ||||||
|  |   } | ||||||
|   this->parent_.control(*this); |   this->parent_.control(*this); | ||||||
| } | } | ||||||
|  |  | ||||||
| void FanCall::validate_() { | void FanCall::validate_() { | ||||||
|   auto traits = this->parent_.get_traits(); |   auto traits = this->parent_.get_traits(); | ||||||
|  |  | ||||||
| @@ -62,6 +65,15 @@ void FanCall::validate_() { | |||||||
|     ESP_LOGW(TAG, "'%s' - This fan does not support directions!", this->parent_.get_name().c_str()); |     ESP_LOGW(TAG, "'%s' - This fan does not support directions!", this->parent_.get_name().c_str()); | ||||||
|     this->direction_.reset(); |     this->direction_.reset(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   if (!this->preset_mode_.empty()) { | ||||||
|  |     const auto &preset_modes = traits.supported_preset_modes(); | ||||||
|  |     if (preset_modes.find(this->preset_mode_) == preset_modes.end()) { | ||||||
|  |       ESP_LOGW(TAG, "'%s' - This fan does not support preset mode '%s'!", this->parent_.get_name().c_str(), | ||||||
|  |                this->preset_mode_.c_str()); | ||||||
|  |       this->preset_mode_.clear(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| FanCall FanRestoreState::to_call(Fan &fan) { | FanCall FanRestoreState::to_call(Fan &fan) { | ||||||
| @@ -70,6 +82,14 @@ FanCall FanRestoreState::to_call(Fan &fan) { | |||||||
|   call.set_oscillating(this->oscillating); |   call.set_oscillating(this->oscillating); | ||||||
|   call.set_speed(this->speed); |   call.set_speed(this->speed); | ||||||
|   call.set_direction(this->direction); |   call.set_direction(this->direction); | ||||||
|  |  | ||||||
|  |   if (fan.get_traits().supports_preset_modes()) { | ||||||
|  |     // Use stored preset index to get preset name | ||||||
|  |     const auto &preset_modes = fan.get_traits().supported_preset_modes(); | ||||||
|  |     if (this->preset_mode < preset_modes.size()) { | ||||||
|  |       call.set_preset_mode(*std::next(preset_modes.begin(), this->preset_mode)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   return call; |   return call; | ||||||
| } | } | ||||||
| void FanRestoreState::apply(Fan &fan) { | void FanRestoreState::apply(Fan &fan) { | ||||||
| @@ -77,6 +97,14 @@ void FanRestoreState::apply(Fan &fan) { | |||||||
|   fan.oscillating = this->oscillating; |   fan.oscillating = this->oscillating; | ||||||
|   fan.speed = this->speed; |   fan.speed = this->speed; | ||||||
|   fan.direction = this->direction; |   fan.direction = this->direction; | ||||||
|  |  | ||||||
|  |   if (fan.get_traits().supports_preset_modes()) { | ||||||
|  |     // Use stored preset index to get preset name | ||||||
|  |     const auto &preset_modes = fan.get_traits().supported_preset_modes(); | ||||||
|  |     if (this->preset_mode < preset_modes.size()) { | ||||||
|  |       fan.preset_mode = *std::next(preset_modes.begin(), this->preset_mode); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   fan.publish_state(); |   fan.publish_state(); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -100,7 +128,9 @@ void Fan::publish_state() { | |||||||
|   if (traits.supports_direction()) { |   if (traits.supports_direction()) { | ||||||
|     ESP_LOGD(TAG, "  Direction: %s", LOG_STR_ARG(fan_direction_to_string(this->direction))); |     ESP_LOGD(TAG, "  Direction: %s", LOG_STR_ARG(fan_direction_to_string(this->direction))); | ||||||
|   } |   } | ||||||
|  |   if (traits.supports_preset_modes() && !this->preset_mode.empty()) { | ||||||
|  |     ESP_LOGD(TAG, "  Preset Mode: %s", this->preset_mode.c_str()); | ||||||
|  |   } | ||||||
|   this->state_callback_.call(); |   this->state_callback_.call(); | ||||||
|   this->save_state_(); |   this->save_state_(); | ||||||
| } | } | ||||||
| @@ -143,20 +173,36 @@ void Fan::save_state_() { | |||||||
|   state.oscillating = this->oscillating; |   state.oscillating = this->oscillating; | ||||||
|   state.speed = this->speed; |   state.speed = this->speed; | ||||||
|   state.direction = this->direction; |   state.direction = this->direction; | ||||||
|  |  | ||||||
|  |   if (this->get_traits().supports_preset_modes() && !this->preset_mode.empty()) { | ||||||
|  |     const auto &preset_modes = this->get_traits().supported_preset_modes(); | ||||||
|  |     // Store index of current preset mode | ||||||
|  |     auto preset_iterator = preset_modes.find(this->preset_mode); | ||||||
|  |     if (preset_iterator != preset_modes.end()) | ||||||
|  |       state.preset_mode = std::distance(preset_modes.begin(), preset_iterator); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   this->rtc_.save(&state); |   this->rtc_.save(&state); | ||||||
| } | } | ||||||
|  |  | ||||||
| void Fan::dump_traits_(const char *tag, const char *prefix) { | void Fan::dump_traits_(const char *tag, const char *prefix) { | ||||||
|   if (this->get_traits().supports_speed()) { |   auto traits = this->get_traits(); | ||||||
|  |  | ||||||
|  |   if (traits.supports_speed()) { | ||||||
|     ESP_LOGCONFIG(tag, "%s  Speed: YES", prefix); |     ESP_LOGCONFIG(tag, "%s  Speed: YES", prefix); | ||||||
|     ESP_LOGCONFIG(tag, "%s  Speed count: %d", prefix, this->get_traits().supported_speed_count()); |     ESP_LOGCONFIG(tag, "%s  Speed count: %d", prefix, traits.supported_speed_count()); | ||||||
|   } |   } | ||||||
|   if (this->get_traits().supports_oscillation()) { |   if (traits.supports_oscillation()) { | ||||||
|     ESP_LOGCONFIG(tag, "%s  Oscillation: YES", prefix); |     ESP_LOGCONFIG(tag, "%s  Oscillation: YES", prefix); | ||||||
|   } |   } | ||||||
|   if (this->get_traits().supports_direction()) { |   if (traits.supports_direction()) { | ||||||
|     ESP_LOGCONFIG(tag, "%s  Direction: YES", prefix); |     ESP_LOGCONFIG(tag, "%s  Direction: YES", prefix); | ||||||
|   } |   } | ||||||
|  |   if (traits.supports_preset_modes()) { | ||||||
|  |     ESP_LOGCONFIG(tag, "%s  Supported presets:", prefix); | ||||||
|  |     for (const std::string &s : traits.supported_preset_modes()) | ||||||
|  |       ESP_LOGCONFIG(tag, "%s    - %s", prefix, s.c_str()); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| }  // namespace fan | }  // namespace fan | ||||||
|   | |||||||
| @@ -72,6 +72,11 @@ class FanCall { | |||||||
|     return *this; |     return *this; | ||||||
|   } |   } | ||||||
|   optional<FanDirection> get_direction() const { return this->direction_; } |   optional<FanDirection> get_direction() const { return this->direction_; } | ||||||
|  |   FanCall &set_preset_mode(const std::string &preset_mode) { | ||||||
|  |     this->preset_mode_ = preset_mode; | ||||||
|  |     return *this; | ||||||
|  |   } | ||||||
|  |   std::string get_preset_mode() const { return this->preset_mode_; } | ||||||
|  |  | ||||||
|   void perform(); |   void perform(); | ||||||
|  |  | ||||||
| @@ -83,6 +88,7 @@ class FanCall { | |||||||
|   optional<bool> oscillating_; |   optional<bool> oscillating_; | ||||||
|   optional<int> speed_; |   optional<int> speed_; | ||||||
|   optional<FanDirection> direction_{}; |   optional<FanDirection> direction_{}; | ||||||
|  |   std::string preset_mode_{}; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| struct FanRestoreState { | struct FanRestoreState { | ||||||
| @@ -90,6 +96,7 @@ struct FanRestoreState { | |||||||
|   int speed; |   int speed; | ||||||
|   bool oscillating; |   bool oscillating; | ||||||
|   FanDirection direction; |   FanDirection direction; | ||||||
|  |   uint8_t preset_mode; | ||||||
|  |  | ||||||
|   /// Convert this struct to a fan call that can be performed. |   /// Convert this struct to a fan call that can be performed. | ||||||
|   FanCall to_call(Fan &fan); |   FanCall to_call(Fan &fan); | ||||||
| @@ -107,6 +114,8 @@ class Fan : public EntityBase { | |||||||
|   int speed{0}; |   int speed{0}; | ||||||
|   /// The current direction of the fan |   /// The current direction of the fan | ||||||
|   FanDirection direction{FanDirection::FORWARD}; |   FanDirection direction{FanDirection::FORWARD}; | ||||||
|  |   // The current preset mode of the fan | ||||||
|  |   std::string preset_mode{}; | ||||||
|  |  | ||||||
|   FanCall turn_on(); |   FanCall turn_on(); | ||||||
|   FanCall turn_off(); |   FanCall turn_off(); | ||||||
|   | |||||||
| @@ -1,3 +1,6 @@ | |||||||
|  | #include <set> | ||||||
|  | #include <utility> | ||||||
|  |  | ||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
| namespace esphome { | namespace esphome { | ||||||
| @@ -25,12 +28,19 @@ class FanTraits { | |||||||
|   bool supports_direction() const { return this->direction_; } |   bool supports_direction() const { return this->direction_; } | ||||||
|   /// Set whether this fan supports changing direction |   /// Set whether this fan supports changing direction | ||||||
|   void set_direction(bool direction) { this->direction_ = direction; } |   void set_direction(bool direction) { this->direction_ = direction; } | ||||||
|  |   /// Return the preset modes supported by the fan. | ||||||
|  |   std::set<std::string> supported_preset_modes() const { return this->preset_modes_; } | ||||||
|  |   /// Set the preset modes supported by the fan. | ||||||
|  |   void set_supported_preset_modes(const std::set<std::string> &preset_modes) { this->preset_modes_ = preset_modes; } | ||||||
|  |   /// Return if preset modes are supported | ||||||
|  |   bool supports_preset_modes() const { return !this->preset_modes_.empty(); } | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   bool oscillation_{false}; |   bool oscillation_{false}; | ||||||
|   bool speed_{false}; |   bool speed_{false}; | ||||||
|   bool direction_{false}; |   bool direction_{false}; | ||||||
|   int speed_count_{}; |   int speed_count_{}; | ||||||
|  |   std::set<std::string> preset_modes_{}; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| }  // namespace fan | }  // namespace fan | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ import esphome.config_validation as cv | |||||||
| from esphome import automation | from esphome import automation | ||||||
| from esphome.automation import maybe_simple_id | from esphome.automation import maybe_simple_id | ||||||
| from esphome.components import fan, output | from esphome.components import fan, output | ||||||
|  | from esphome.components.fan import validate_preset_modes | ||||||
| from esphome.const import ( | from esphome.const import ( | ||||||
|     CONF_ID, |     CONF_ID, | ||||||
|     CONF_DECAY_MODE, |     CONF_DECAY_MODE, | ||||||
| @@ -10,6 +11,7 @@ from esphome.const import ( | |||||||
|     CONF_PIN_A, |     CONF_PIN_A, | ||||||
|     CONF_PIN_B, |     CONF_PIN_B, | ||||||
|     CONF_ENABLE_PIN, |     CONF_ENABLE_PIN, | ||||||
|  |     CONF_PRESET_MODES, | ||||||
| ) | ) | ||||||
| from .. import hbridge_ns | from .. import hbridge_ns | ||||||
|  |  | ||||||
| @@ -28,7 +30,6 @@ DECAY_MODE_OPTIONS = { | |||||||
| # Actions | # Actions | ||||||
| BrakeAction = hbridge_ns.class_("BrakeAction", automation.Action) | BrakeAction = hbridge_ns.class_("BrakeAction", automation.Action) | ||||||
|  |  | ||||||
|  |  | ||||||
| CONFIG_SCHEMA = fan.FAN_SCHEMA.extend( | CONFIG_SCHEMA = fan.FAN_SCHEMA.extend( | ||||||
|     { |     { | ||||||
|         cv.GenerateID(CONF_ID): cv.declare_id(HBridgeFan), |         cv.GenerateID(CONF_ID): cv.declare_id(HBridgeFan), | ||||||
| @@ -39,6 +40,7 @@ CONFIG_SCHEMA = fan.FAN_SCHEMA.extend( | |||||||
|         ), |         ), | ||||||
|         cv.Optional(CONF_SPEED_COUNT, default=100): cv.int_range(min=1), |         cv.Optional(CONF_SPEED_COUNT, default=100): cv.int_range(min=1), | ||||||
|         cv.Optional(CONF_ENABLE_PIN): cv.use_id(output.FloatOutput), |         cv.Optional(CONF_ENABLE_PIN): cv.use_id(output.FloatOutput), | ||||||
|  |         cv.Optional(CONF_PRESET_MODES): validate_preset_modes, | ||||||
|     } |     } | ||||||
| ).extend(cv.COMPONENT_SCHEMA) | ).extend(cv.COMPONENT_SCHEMA) | ||||||
|  |  | ||||||
| @@ -69,3 +71,6 @@ async def to_code(config): | |||||||
|     if CONF_ENABLE_PIN in config: |     if CONF_ENABLE_PIN in config: | ||||||
|         enable_pin = await cg.get_variable(config[CONF_ENABLE_PIN]) |         enable_pin = await cg.get_variable(config[CONF_ENABLE_PIN]) | ||||||
|         cg.add(var.set_enable_pin(enable_pin)) |         cg.add(var.set_enable_pin(enable_pin)) | ||||||
|  |  | ||||||
|  |     if CONF_PRESET_MODES in config: | ||||||
|  |         cg.add(var.set_preset_modes(config[CONF_PRESET_MODES])) | ||||||
|   | |||||||
| @@ -33,7 +33,12 @@ void HBridgeFan::setup() { | |||||||
|     restore->apply(*this); |     restore->apply(*this); | ||||||
|     this->write_state_(); |     this->write_state_(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Construct traits | ||||||
|  |   this->traits_ = fan::FanTraits(this->oscillating_ != nullptr, true, true, this->speed_count_); | ||||||
|  |   this->traits_.set_supported_preset_modes(this->preset_modes_); | ||||||
| } | } | ||||||
|  |  | ||||||
| void HBridgeFan::dump_config() { | void HBridgeFan::dump_config() { | ||||||
|   LOG_FAN("", "H-Bridge Fan", this); |   LOG_FAN("", "H-Bridge Fan", this); | ||||||
|   if (this->decay_mode_ == DECAY_MODE_SLOW) { |   if (this->decay_mode_ == DECAY_MODE_SLOW) { | ||||||
| @@ -42,9 +47,7 @@ void HBridgeFan::dump_config() { | |||||||
|     ESP_LOGCONFIG(TAG, "  Decay Mode: Fast"); |     ESP_LOGCONFIG(TAG, "  Decay Mode: Fast"); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| fan::FanTraits HBridgeFan::get_traits() { |  | ||||||
|   return fan::FanTraits(this->oscillating_ != nullptr, true, true, this->speed_count_); |  | ||||||
| } |  | ||||||
| void HBridgeFan::control(const fan::FanCall &call) { | void HBridgeFan::control(const fan::FanCall &call) { | ||||||
|   if (call.get_state().has_value()) |   if (call.get_state().has_value()) | ||||||
|     this->state = *call.get_state(); |     this->state = *call.get_state(); | ||||||
| @@ -54,10 +57,12 @@ void HBridgeFan::control(const fan::FanCall &call) { | |||||||
|     this->oscillating = *call.get_oscillating(); |     this->oscillating = *call.get_oscillating(); | ||||||
|   if (call.get_direction().has_value()) |   if (call.get_direction().has_value()) | ||||||
|     this->direction = *call.get_direction(); |     this->direction = *call.get_direction(); | ||||||
|  |   this->preset_mode = call.get_preset_mode(); | ||||||
|  |  | ||||||
|   this->write_state_(); |   this->write_state_(); | ||||||
|   this->publish_state(); |   this->publish_state(); | ||||||
| } | } | ||||||
|  |  | ||||||
| void HBridgeFan::write_state_() { | void HBridgeFan::write_state_() { | ||||||
|   float speed = this->state ? static_cast<float>(this->speed) / static_cast<float>(this->speed_count_) : 0.0f; |   float speed = this->state ? static_cast<float>(this->speed) / static_cast<float>(this->speed_count_) : 0.0f; | ||||||
|   if (speed == 0.0f) {  // off means idle |   if (speed == 0.0f) {  // off means idle | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
|  | #include <set> | ||||||
|  |  | ||||||
| #include "esphome/core/automation.h" | #include "esphome/core/automation.h" | ||||||
| #include "esphome/components/output/binary_output.h" | #include "esphome/components/output/binary_output.h" | ||||||
| #include "esphome/components/output/float_output.h" | #include "esphome/components/output/float_output.h" | ||||||
| @@ -20,10 +22,11 @@ class HBridgeFan : public Component, public fan::Fan { | |||||||
|   void set_pin_a(output::FloatOutput *pin_a) { pin_a_ = pin_a; } |   void set_pin_a(output::FloatOutput *pin_a) { pin_a_ = pin_a; } | ||||||
|   void set_pin_b(output::FloatOutput *pin_b) { pin_b_ = pin_b; } |   void set_pin_b(output::FloatOutput *pin_b) { pin_b_ = pin_b; } | ||||||
|   void set_enable_pin(output::FloatOutput *enable) { enable_ = enable; } |   void set_enable_pin(output::FloatOutput *enable) { enable_ = enable; } | ||||||
|  |   void set_preset_modes(const std::set<std::string> &presets) { preset_modes_ = presets; } | ||||||
|  |  | ||||||
|   void setup() override; |   void setup() override; | ||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
|   fan::FanTraits get_traits() override; |   fan::FanTraits get_traits() override { return this->traits_; } | ||||||
|  |  | ||||||
|   fan::FanCall brake(); |   fan::FanCall brake(); | ||||||
|  |  | ||||||
| @@ -34,6 +37,8 @@ class HBridgeFan : public Component, public fan::Fan { | |||||||
|   output::BinaryOutput *oscillating_{nullptr}; |   output::BinaryOutput *oscillating_{nullptr}; | ||||||
|   int speed_count_{}; |   int speed_count_{}; | ||||||
|   DecayMode decay_mode_{DECAY_MODE_SLOW}; |   DecayMode decay_mode_{DECAY_MODE_SLOW}; | ||||||
|  |   fan::FanTraits traits_; | ||||||
|  |   std::set<std::string> preset_modes_{}; | ||||||
|  |  | ||||||
|   void control(const fan::FanCall &call) override; |   void control(const fan::FanCall &call) override; | ||||||
|   void write_state_(); |   void write_state_(); | ||||||
|   | |||||||
| @@ -1,14 +1,17 @@ | |||||||
| import esphome.codegen as cg | import esphome.codegen as cg | ||||||
| import esphome.config_validation as cv | import esphome.config_validation as cv | ||||||
| from esphome.components import fan, output | from esphome.components import fan, output | ||||||
|  | from esphome.components.fan import validate_preset_modes | ||||||
| from esphome.const import ( | from esphome.const import ( | ||||||
|  |     CONF_PRESET_MODES, | ||||||
|  |     CONF_DIRECTION_OUTPUT, | ||||||
|     CONF_OSCILLATION_OUTPUT, |     CONF_OSCILLATION_OUTPUT, | ||||||
|     CONF_OUTPUT, |     CONF_OUTPUT, | ||||||
|     CONF_DIRECTION_OUTPUT, |  | ||||||
|     CONF_OUTPUT_ID, |     CONF_OUTPUT_ID, | ||||||
|     CONF_SPEED, |     CONF_SPEED, | ||||||
|     CONF_SPEED_COUNT, |     CONF_SPEED_COUNT, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| from .. import speed_ns | from .. import speed_ns | ||||||
|  |  | ||||||
| SpeedFan = speed_ns.class_("SpeedFan", cg.Component, fan.Fan) | SpeedFan = speed_ns.class_("SpeedFan", cg.Component, fan.Fan) | ||||||
| @@ -23,6 +26,7 @@ CONFIG_SCHEMA = fan.FAN_SCHEMA.extend( | |||||||
|             "Configuring individual speeds is deprecated." |             "Configuring individual speeds is deprecated." | ||||||
|         ), |         ), | ||||||
|         cv.Optional(CONF_SPEED_COUNT, default=100): cv.int_range(min=1), |         cv.Optional(CONF_SPEED_COUNT, default=100): cv.int_range(min=1), | ||||||
|  |         cv.Optional(CONF_PRESET_MODES): validate_preset_modes, | ||||||
|     } |     } | ||||||
| ).extend(cv.COMPONENT_SCHEMA) | ).extend(cv.COMPONENT_SCHEMA) | ||||||
|  |  | ||||||
| @@ -40,3 +44,6 @@ async def to_code(config): | |||||||
|     if CONF_DIRECTION_OUTPUT in config: |     if CONF_DIRECTION_OUTPUT in config: | ||||||
|         direction_output = await cg.get_variable(config[CONF_DIRECTION_OUTPUT]) |         direction_output = await cg.get_variable(config[CONF_DIRECTION_OUTPUT]) | ||||||
|         cg.add(var.set_direction(direction_output)) |         cg.add(var.set_direction(direction_output)) | ||||||
|  |  | ||||||
|  |     if CONF_PRESET_MODES in config: | ||||||
|  |         cg.add(var.set_preset_modes(config[CONF_PRESET_MODES])) | ||||||
|   | |||||||
| @@ -12,11 +12,14 @@ void SpeedFan::setup() { | |||||||
|     restore->apply(*this); |     restore->apply(*this); | ||||||
|     this->write_state_(); |     this->write_state_(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   // Construct traits | ||||||
|  |   this->traits_ = fan::FanTraits(this->oscillating_ != nullptr, true, this->direction_ != nullptr, this->speed_count_); | ||||||
|  |   this->traits_.set_supported_preset_modes(this->preset_modes_); | ||||||
| } | } | ||||||
|  |  | ||||||
| void SpeedFan::dump_config() { LOG_FAN("", "Speed Fan", this); } | void SpeedFan::dump_config() { LOG_FAN("", "Speed Fan", this); } | ||||||
| fan::FanTraits SpeedFan::get_traits() { |  | ||||||
|   return fan::FanTraits(this->oscillating_ != nullptr, true, this->direction_ != nullptr, this->speed_count_); |  | ||||||
| } |  | ||||||
| void SpeedFan::control(const fan::FanCall &call) { | void SpeedFan::control(const fan::FanCall &call) { | ||||||
|   if (call.get_state().has_value()) |   if (call.get_state().has_value()) | ||||||
|     this->state = *call.get_state(); |     this->state = *call.get_state(); | ||||||
| @@ -26,10 +29,12 @@ void SpeedFan::control(const fan::FanCall &call) { | |||||||
|     this->oscillating = *call.get_oscillating(); |     this->oscillating = *call.get_oscillating(); | ||||||
|   if (call.get_direction().has_value()) |   if (call.get_direction().has_value()) | ||||||
|     this->direction = *call.get_direction(); |     this->direction = *call.get_direction(); | ||||||
|  |   this->preset_mode = call.get_preset_mode(); | ||||||
|  |  | ||||||
|   this->write_state_(); |   this->write_state_(); | ||||||
|   this->publish_state(); |   this->publish_state(); | ||||||
| } | } | ||||||
|  |  | ||||||
| void SpeedFan::write_state_() { | void SpeedFan::write_state_() { | ||||||
|   float speed = this->state ? static_cast<float>(this->speed) / static_cast<float>(this->speed_count_) : 0.0f; |   float speed = this->state ? static_cast<float>(this->speed) / static_cast<float>(this->speed_count_) : 0.0f; | ||||||
|   this->output_->set_level(speed); |   this->output_->set_level(speed); | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| #pragma once | #pragma once | ||||||
|  |  | ||||||
|  | #include <set> | ||||||
|  |  | ||||||
| #include "esphome/core/component.h" | #include "esphome/core/component.h" | ||||||
| #include "esphome/components/output/binary_output.h" | #include "esphome/components/output/binary_output.h" | ||||||
| #include "esphome/components/output/float_output.h" | #include "esphome/components/output/float_output.h" | ||||||
| @@ -15,7 +17,8 @@ class SpeedFan : public Component, public fan::Fan { | |||||||
|   void dump_config() override; |   void dump_config() override; | ||||||
|   void set_oscillating(output::BinaryOutput *oscillating) { this->oscillating_ = oscillating; } |   void set_oscillating(output::BinaryOutput *oscillating) { this->oscillating_ = oscillating; } | ||||||
|   void set_direction(output::BinaryOutput *direction) { this->direction_ = direction; } |   void set_direction(output::BinaryOutput *direction) { this->direction_ = direction; } | ||||||
|   fan::FanTraits get_traits() override; |   void set_preset_modes(const std::set<std::string> &presets) { this->preset_modes_ = presets; } | ||||||
|  |   fan::FanTraits get_traits() override { return this->traits_; } | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
|   void control(const fan::FanCall &call) override; |   void control(const fan::FanCall &call) override; | ||||||
| @@ -25,6 +28,8 @@ class SpeedFan : public Component, public fan::Fan { | |||||||
|   output::BinaryOutput *oscillating_{nullptr}; |   output::BinaryOutput *oscillating_{nullptr}; | ||||||
|   output::BinaryOutput *direction_{nullptr}; |   output::BinaryOutput *direction_{nullptr}; | ||||||
|   int speed_count_{}; |   int speed_count_{}; | ||||||
|  |   fan::FanTraits traits_; | ||||||
|  |   std::set<std::string> preset_modes_{}; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| }  // namespace speed | }  // namespace speed | ||||||
|   | |||||||
| @@ -504,6 +504,7 @@ CONF_ON_LOOP = "on_loop" | |||||||
| CONF_ON_MESSAGE = "on_message" | CONF_ON_MESSAGE = "on_message" | ||||||
| CONF_ON_MULTI_CLICK = "on_multi_click" | CONF_ON_MULTI_CLICK = "on_multi_click" | ||||||
| CONF_ON_OPEN = "on_open" | CONF_ON_OPEN = "on_open" | ||||||
|  | CONF_ON_PRESET_SET = "on_preset_set" | ||||||
| CONF_ON_PRESS = "on_press" | CONF_ON_PRESS = "on_press" | ||||||
| CONF_ON_RAW_VALUE = "on_raw_value" | CONF_ON_RAW_VALUE = "on_raw_value" | ||||||
| CONF_ON_RELEASE = "on_release" | CONF_ON_RELEASE = "on_release" | ||||||
| @@ -601,6 +602,7 @@ CONF_PRESET = "preset" | |||||||
| CONF_PRESET_BOOST = "preset_boost" | CONF_PRESET_BOOST = "preset_boost" | ||||||
| CONF_PRESET_COMMAND_TOPIC = "preset_command_topic" | CONF_PRESET_COMMAND_TOPIC = "preset_command_topic" | ||||||
| CONF_PRESET_ECO = "preset_eco" | CONF_PRESET_ECO = "preset_eco" | ||||||
|  | CONF_PRESET_MODES = "preset_modes" | ||||||
| CONF_PRESET_SLEEP = "preset_sleep" | CONF_PRESET_SLEEP = "preset_sleep" | ||||||
| CONF_PRESET_STATE_TOPIC = "preset_state_topic" | CONF_PRESET_STATE_TOPIC = "preset_state_topic" | ||||||
| CONF_PRESSURE = "pressure" | CONF_PRESSURE = "pressure" | ||||||
|   | |||||||
| @@ -2944,6 +2944,33 @@ fan: | |||||||
|     on_speed_set: |     on_speed_set: | ||||||
|       then: |       then: | ||||||
|         - logger.log: Fan speed was changed! |         - logger.log: Fan speed was changed! | ||||||
|  |   - platform: speed | ||||||
|  |     id: fan_speed_presets | ||||||
|  |     icon: mdi:weather-windy | ||||||
|  |     output: pca_6 | ||||||
|  |     speed_count: 10 | ||||||
|  |     name: Speed Fan w/ Presets | ||||||
|  |     oscillation_output: gpio_19 | ||||||
|  |     direction_output: gpio_26 | ||||||
|  |     preset_modes: | ||||||
|  |       - Preset 1 | ||||||
|  |       - Preset 2 | ||||||
|  |     on_preset_set: | ||||||
|  |       then: | ||||||
|  |         - logger.log: Preset mode was changed! | ||||||
|  |   - platform: hbridge | ||||||
|  |     id: fan_hbridge_presets | ||||||
|  |     icon: mdi:weather-windy | ||||||
|  |     speed_count: 4 | ||||||
|  |     name: H-bridge Fan w/ Presets | ||||||
|  |     pin_a: pca_6 | ||||||
|  |     pin_b: pca_7 | ||||||
|  |     preset_modes: | ||||||
|  |       - Preset 1 | ||||||
|  |       - Preset 2 | ||||||
|  |     on_preset_set: | ||||||
|  |       then: | ||||||
|  |         - logger.log: Preset mode was changed! | ||||||
|   - platform: bedjet |   - platform: bedjet | ||||||
|     name: My Bedjet fan |     name: My Bedjet fan | ||||||
|     bedjet_id: my_bedjet_client |     bedjet_id: my_bedjet_client | ||||||
| @@ -4193,4 +4220,3 @@ alarm_control_panel: | |||||||
|       then: |       then: | ||||||
|         - lambda: !lambda |- |         - lambda: !lambda |- | ||||||
|             ESP_LOGD("TEST", "State change %s", alarm_control_panel_state_to_string(id(alarmcontrolpanel1)->get_state())); |             ESP_LOGD("TEST", "State change %s", alarm_control_panel_state_to_string(id(alarmcontrolpanel1)->get_state())); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user