9static const char *
const TAG =
"hlk_fm22x";
12 ESP_LOGCONFIG(TAG,
"Setting up HLK-FM22X...");
35 if (name.length() > 31) {
36 ESP_LOGE(TAG,
"enroll_face(): name too long '%s'", name.c_str());
39 ESP_LOGI(TAG,
"Starting enrollment for %s", name.c_str());
40 std::array<uint8_t, 35> data{};
42 std::copy(name.begin(), name.end(), data.begin() + 1);
51 ESP_LOGI(TAG,
"Verify face");
52 static const uint8_t DATA[] = {0, 0};
57 ESP_LOGI(TAG,
"Deleting face in slot %d", face_id);
58 const uint8_t data[] = {(uint8_t) (face_id >> 8), (uint8_t) (face_id & 0xFF)};
63 ESP_LOGI(TAG,
"Deleting all stored faces");
68 ESP_LOGD(TAG,
"Getting face count");
73 ESP_LOGI(TAG,
"Resetting module");
81 ESP_LOGV(TAG,
"Send command: 0x%.2X", command);
90 this->
write((uint8_t) (START_CODE >> 8));
91 this->
write((uint8_t) (START_CODE & 0xFF));
92 this->
write((uint8_t) command);
93 uint16_t data_size = size;
94 this->
write((uint8_t) (data_size >> 8));
95 this->
write((uint8_t) (data_size & 0xFF));
101 for (
size_t i = 0; i < size; i++) {
102 this->
write(data[i]);
106 this->
write(checksum);
121 if ((this->
read() != (uint8_t) (START_CODE >> 8)) || (this->
read() != (uint8_t) (START_CODE & 0xFF))) {
122 ESP_LOGE(TAG,
"Invalid start code");
137 std::vector<uint8_t> data;
139 for (uint16_t idx = 0; idx <
length; ++idx) {
142 data.push_back(
byte);
145 ESP_LOGV(TAG,
"Recv type: 0x%.2X, data: %s", response_type,
format_hex_pretty(data).c_str());
148 if (
byte != checksum) {
149 ESP_LOGE(TAG,
"Invalid checksum for data. Calculated: 0x%.2X, Received: 0x%.2X",
checksum,
byte);
152 switch (response_type) {
160 ESP_LOGW(TAG,
"Unexpected response type: 0x%.2X", response_type);
168 if (data.size() < 17) {
169 ESP_LOGE(TAG,
"Invalid face note data size: %u", data.size());
175 for (int16_t &i : info) {
176 i = ((int16_t) data[offset + 1] << 8) | data[offset];
179 ESP_LOGV(TAG,
"Face state: status: %d, left: %d, top: %d, right: %d, bottom: %d, yaw: %d, pitch: %d, roll: %d",
180 info[0], info[1], info[2], info[3], info[4], info[5], info[6], info[7]);
181 this->
face_info_callback_.call(info[0], info[1], info[2], info[3], info[4], info[5], info[6], info[7]);
201 ESP_LOGW(TAG,
"Unhandled note: 0x%.2X", data[0]);
209 if (data[0] != (uint8_t) expected) {
210 ESP_LOGE(TAG,
"Unexpected response command. Expected: 0x%.2X, Received: 0x%.2X", expected, data[0]);
215 ESP_LOGE(TAG,
"Command <0x%.2X> failed. Error: 0x%.2X", data[0], data[1]);
235 int16_t face_id = ((int16_t) data[2] << 8) | data[3];
236 std::string name(data.begin() + 4, data.begin() + 36);
237 ESP_LOGD(TAG,
"Face verified. ID: %d, name: %s", face_id, name.c_str());
248 int16_t face_id = ((int16_t) data[2] << 8) | data[3];
250 ESP_LOGI(TAG,
"Face enrolled. ID: %d, Direction: 0x%.2X", face_id,
direction);
264 std::string version(data.begin() + 2, data.end());
275 ESP_LOGI(TAG,
"Deleted face");
278 ESP_LOGI(TAG,
"Deleted all faces");
281 ESP_LOGI(TAG,
"Module reset");
297 ESP_LOGCONFIG(TAG,
"HLK_FM22X:");
298 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.
std::string get_state() const
Getter-syntax for .state.
void publish_state(const std::string &state)
size_t write(uint8_t data)
std::string format_hex_pretty(const uint8_t *data, size_t length, char separator, bool show_length)
Format a byte array in pretty-printed, human-readable hex format.