9static const char *
const TAG =
"hlk_fm22x";
12static constexpr size_t HLK_FM22X_MAX_RESPONSE_SIZE = 36;
15 ESP_LOGCONFIG(TAG,
"Setting up HLK-FM22X...");
38 if (name.length() > 31) {
39 ESP_LOGE(TAG,
"enroll_face(): name too long '%s'", name.c_str());
42 ESP_LOGI(TAG,
"Starting enrollment for %s", name.c_str());
43 std::array<uint8_t, 35> data{};
45 std::copy(name.begin(), name.end(), data.begin() + 1);
54 ESP_LOGI(TAG,
"Verify face");
55 static const uint8_t DATA[] = {0, 0};
60 ESP_LOGI(TAG,
"Deleting face in slot %d", face_id);
61 const uint8_t data[] = {(uint8_t) (face_id >> 8), (uint8_t) (face_id & 0xFF)};
66 ESP_LOGI(TAG,
"Deleting all stored faces");
71 ESP_LOGD(TAG,
"Getting face count");
76 ESP_LOGI(TAG,
"Resetting module");
84 ESP_LOGV(TAG,
"Send command: 0x%.2X", command);
93 this->
write((uint8_t) (START_CODE >> 8));
94 this->
write((uint8_t) (START_CODE & 0xFF));
95 this->
write((uint8_t) command);
96 uint16_t data_size = size;
97 this->
write((uint8_t) (data_size >> 8));
98 this->
write((uint8_t) (data_size & 0xFF));
104 for (
size_t i = 0; i < size; i++) {
105 this->
write(data[i]);
109 this->
write(checksum);
124 if ((this->
read() != (uint8_t) (START_CODE >> 8)) || (this->
read() != (uint8_t) (START_CODE & 0xFF))) {
125 ESP_LOGE(TAG,
"Invalid start code");
140 std::vector<uint8_t> data;
142 for (uint16_t idx = 0; idx <
length; ++idx) {
145 data.push_back(
byte);
148#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
150 ESP_LOGV(TAG,
"Recv type: 0x%.2X, data: %s", response_type,
format_hex_pretty_to(hex_buf, data.data(), data.size()));
154 if (
byte != checksum) {
155 ESP_LOGE(TAG,
"Invalid checksum for data. Calculated: 0x%.2X, Received: 0x%.2X",
checksum,
byte);
158 switch (response_type) {
166 ESP_LOGW(TAG,
"Unexpected response type: 0x%.2X", response_type);
174 if (data.size() < 17) {
175 ESP_LOGE(TAG,
"Invalid face note data size: %u", data.size());
181 for (int16_t &i : info) {
182 i = ((int16_t) data[offset + 1] << 8) | data[offset];
185 ESP_LOGV(TAG,
"Face state: status: %d, left: %d, top: %d, right: %d, bottom: %d, yaw: %d, pitch: %d, roll: %d",
186 info[0], info[1], info[2], info[3], info[4], info[5], info[6], info[7]);
187 this->
face_info_callback_.call(info[0], info[1], info[2], info[3], info[4], info[5], info[6], info[7]);
207 ESP_LOGW(TAG,
"Unhandled note: 0x%.2X", data[0]);
215 if (data[0] != (uint8_t) expected) {
216 ESP_LOGE(TAG,
"Unexpected response command. Expected: 0x%.2X, Received: 0x%.2X", expected, data[0]);
221 ESP_LOGE(TAG,
"Command <0x%.2X> failed. Error: 0x%.2X", data[0], data[1]);
241 int16_t face_id = ((int16_t) data[2] << 8) | data[3];
242 std::string name(data.begin() + 4, data.begin() + 36);
243 ESP_LOGD(TAG,
"Face verified. ID: %d, name: %s", face_id, name.c_str());
254 int16_t face_id = ((int16_t) data[2] << 8) | data[3];
256 ESP_LOGI(TAG,
"Face enrolled. ID: %d, Direction: 0x%.2X", face_id,
direction);
270 std::string version(data.begin() + 2, data.end());
281 ESP_LOGI(TAG,
"Deleted face");
284 ESP_LOGI(TAG,
"Deleted all faces");
287 ESP_LOGI(TAG,
"Module reset");
303 ESP_LOGCONFIG(TAG,
"HLK_FM22X:");
304 LOG_UPDATE_INTERVAL(
this);
virtual void mark_failed()
Mark this component as failed.
void defer(const std::string &name, std::function< void()> &&f)
Defer a callback to the next loop() call.
void publish_state(bool new_state)
Publish a new state to the front-end.
void delete_face(int16_t face_id)
sensor::Sensor * last_face_id_sensor_
text_sensor::TextSensor * last_face_name_text_sensor_
CallbackManager< void(int16_t, int16_t, int16_t, int16_t, int16_t, int16_t, int16_t, int16_t)> face_info_callback_
void enroll_face(const std::string &name, HlkFm22xFaceDirection direction)
void handle_reply_(const std::vector< uint8_t > &data)
void send_command_(HlkFm22xCommand command, const uint8_t *data=nullptr, size_t size=0)
CallbackManager< void(uint8_t)> face_scan_invalid_callback_
sensor::Sensor * face_count_sensor_
CallbackManager< void()> face_scan_unmatched_callback_
void set_enrolling_(bool enrolling)
void handle_note_(const std::vector< uint8_t > &data)
CallbackManager< void(int16_t, uint8_t)> enrollment_done_callback_
CallbackManager< void(uint8_t)> enrollment_failed_callback_
sensor::Sensor * status_sensor_
text_sensor::TextSensor * version_text_sensor_
CallbackManager< void(int16_t, std::string)> face_scan_matched_callback_
binary_sensor::BinarySensor * enrolling_binary_sensor_
HlkFm22xCommand active_command_
void dump_config() override
void publish_state(float state)
Publish a new state to the front-end.
float get_state() const
Getter-syntax for .state.
const std::string & get_state() const
Getter-syntax for .state.
void publish_state(const std::string &state)
size_t write(uint8_t data)
char * format_hex_pretty_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length, char separator)
Format byte array as uppercase hex to buffer (base implementation).
constexpr size_t format_hex_pretty_size(size_t byte_count)
Calculate buffer size needed for format_hex_pretty_to with separator: "XX:XX:...:XX\0".