ESPHome  2023.5.5
midea_protocol.h
Go to the documentation of this file.
1 #pragma once
2 
4 #include "esphome/core/helpers.h"
5 #include "remote_base.h"
6 #include <array>
7 #include <utility>
8 #include <vector>
9 
10 namespace esphome {
11 namespace remote_base {
12 
13 class MideaData {
14  public:
15  // Make default
16  MideaData() {}
17  // Make from initializer_list
18  MideaData(std::initializer_list<uint8_t> data) {
19  std::copy_n(data.begin(), std::min(data.size(), this->data_.size()), this->data_.begin());
20  }
21  // Make from vector
22  MideaData(const std::vector<uint8_t> &data) {
23  std::copy_n(data.begin(), std::min(data.size(), this->data_.size()), this->data_.begin());
24  }
25  // Default copy constructor
26  MideaData(const MideaData &) = default;
27 
28  uint8_t *data() { return this->data_.data(); }
29  const uint8_t *data() const { return this->data_.data(); }
30  uint8_t size() const { return this->data_.size(); }
31  bool is_valid() const { return this->data_[OFFSET_CS] == this->calc_cs_(); }
32  void finalize() { this->data_[OFFSET_CS] = this->calc_cs_(); }
33  bool is_compliment(const MideaData &rhs) const;
34  std::string to_string() const { return format_hex_pretty(this->data_.data(), this->data_.size()); }
35  // compare only 40-bits
36  bool operator==(const MideaData &rhs) const {
37  return std::equal(this->data_.begin(), this->data_.begin() + OFFSET_CS, rhs.data_.begin());
38  }
39  enum MideaDataType : uint8_t {
43  };
44  MideaDataType type() const { return static_cast<MideaDataType>(this->data_[0]); }
45  template<typename T> T to() const { return T(*this); }
46  uint8_t &operator[](size_t idx) { return this->data_[idx]; }
47  const uint8_t &operator[](size_t idx) const { return this->data_[idx]; }
48 
49  protected:
50  uint8_t get_value_(uint8_t idx, uint8_t mask = 255, uint8_t shift = 0) const {
51  return (this->data_[idx] >> shift) & mask;
52  }
53  void set_value_(uint8_t idx, uint8_t value, uint8_t mask = 255, uint8_t shift = 0) {
54  this->data_[idx] &= ~(mask << shift);
55  this->data_[idx] |= (value << shift);
56  }
57  void set_mask_(uint8_t idx, bool state, uint8_t mask = 255) { this->set_value_(idx, state ? mask : 0, mask); }
58  static const uint8_t OFFSET_CS = 5;
59  // 48-bits data
60  std::array<uint8_t, 6> data_;
61  // Calculate checksum
62  uint8_t calc_cs_() const;
63 };
64 
65 class MideaProtocol : public RemoteProtocol<MideaData> {
66  public:
67  void encode(RemoteTransmitData *dst, const MideaData &src) override;
68  optional<MideaData> decode(RemoteReceiveData src) override;
69  void dump(const MideaData &data) override;
70 };
71 
73  public:
74  bool matches(RemoteReceiveData src) override {
75  auto data = MideaProtocol().decode(src);
76  return data.has_value() && data.value() == this->data_;
77  }
78  void set_code(const std::vector<uint8_t> &code) { this->data_ = code; }
79 
80  protected:
82 };
83 
86 
87 template<typename... Ts> class MideaAction : public RemoteTransmitterActionBase<Ts...> {
88  TEMPLATABLE_VALUE(std::vector<uint8_t>, code)
89  void set_code_static(std::vector<uint8_t> code) { code_static_ = std::move(code); }
90  void set_code_template(std::function<std::vector<uint8_t>(Ts...)> func) { this->code_func_ = func; }
91 
92  void encode(RemoteTransmitData *dst, Ts... x) override {
94  if (!this->code_static_.empty()) {
95  data = MideaData(this->code_static_);
96  } else {
97  data = MideaData(this->code_func_(x...));
98  }
99  data.finalize();
100  MideaProtocol().encode(dst, data);
101  }
102 
103  protected:
104  std::function<std::vector<uint8_t>(Ts...)> code_func_{};
105  std::vector<uint8_t> code_static_{};
106 };
107 
108 } // namespace remote_base
109 } // namespace esphome
MideaDataType type() const
std::string format_hex_pretty(const uint8_t *data, size_t length)
Format the byte array data of length len in pretty-printed, human-readable hex.
Definition: helpers.cpp:263
void set_code(const std::vector< uint8_t > &code)
bool operator==(const MideaData &rhs) const
uint8_t & operator[](size_t idx)
bool is_compliment(const MideaData &rhs) const
void encode(RemoteTransmitData *dst, const MideaData &src) override
void set_value_(uint8_t idx, uint8_t value, uint8_t mask=255, uint8_t shift=0)
static const uint8_t OFFSET_CS
uint8_t get_value_(uint8_t idx, uint8_t mask=255, uint8_t shift=0) const
bool matches(RemoteReceiveData src) override
MideaData(std::initializer_list< uint8_t > data)
std::array< uint8_t, 6 > data_
const uint8_t & operator[](size_t idx) const
std::string to_string() const
optional< MideaData > decode(RemoteReceiveData src) override
void set_mask_(uint8_t idx, bool state, uint8_t mask=255)
Definition: a4988.cpp:4
const uint8_t * data() const
MideaData(const std::vector< uint8_t > &data)
bool state
Definition: fan.h:34