13static const char *
const TAG =
"nextion";
16static constexpr uint8_t COMMAND_DELIMITER[3] = {0xFF, 0xFF, 0xFF};
17static constexpr size_t DELIMITER_SIZE =
sizeof(COMMAND_DELIMITER);
38#ifdef USE_NEXTION_COMMAND_SPACING
40 if (!this->
connection_state_.ignore_is_setup_ && !this->command_pacer_.can_send(now)) {
41 ESP_LOGN(TAG,
"Command spacing: delaying '%s'", command.c_str());
46 ESP_LOGN(TAG,
"cmd: %s", command.c_str());
49 const uint8_t to_send[3] = {0xFF, 0xFF, 0xFF};
52#ifdef USE_NEXTION_COMMAND_SPACING
69#ifdef USE_NEXTION_CONFIG_SKIP_CONNECTION_HANDSHAKE
70 ESP_LOGW(TAG,
"Connected (no handshake)");
79#ifdef USE_NEXTION_CONFIG_EXIT_REPARSE_ON_START
96 if (!response.empty() && response[0] == 0x1A) {
98 ESP_LOGV(TAG,
"0x1A error ignored (setup)");
101 if (response.empty() || response.find(
"comok") == std::string::npos) {
102#ifdef NEXTION_PROTOCOL_LOG
103 ESP_LOGN(TAG,
"Bad connect: %s", response.c_str());
104 for (
size_t i = 0; i < response.length(); i++) {
105 ESP_LOGN(TAG,
"resp: %s %d %d %c", response.c_str(), i, response[i], response[i]);
109 ESP_LOGW(TAG,
"Not connected");
115 ESP_LOGI(TAG,
"Connected");
118 ESP_LOGN(TAG,
"connect: %s", response.c_str());
122 size_t field_count = 0;
125 auto copy_field = [&](
char *dst,
size_t cap) {
126 size_t len = (
end == std::string::npos ? response.size() :
end) - start;
127 size_t n =
len < cap ?
len : cap;
128 std::memcpy(dst, response.data() + start, n);
131 while ((start = response.find_first_not_of(
',',
end)) != std::string::npos) {
132 end = response.find(
',', start);
133 switch (field_count) {
144 this->
flash_size_ =
static_cast<uint32_t>(std::strtoul(response.data() + start,
nullptr, 10));
154 ESP_LOGN(TAG,
"Connect info: %zu fields", field_count);
156 ESP_LOGE(TAG,
"Bad connect value: '%s'", response.c_str());
173 delete entry->component;
177 this->nextion_queue_.clear();
178#ifdef USE_NEXTION_WAVEFORM
182 this->waveform_queue_.clear();
187 ESP_LOGCONFIG(TAG,
"Nextion:");
189#ifdef USE_NEXTION_CONFIG_SKIP_CONNECTION_HANDSHAKE
190 ESP_LOGCONFIG(TAG,
" Skip handshake: YES");
194 " Device Model: %s\n"
196 " Serial Number: %s\n"
197 " Flash Size: %" PRIu32
" bytes",
200 ESP_LOGCONFIG(TAG,
" Device info: not yet detected");
203#ifdef USE_NEXTION_CONFIG_EXIT_REPARSE_ON_START
204 " Exit reparse: YES\n"
206 " Max queue age: %u ms\n"
207 " Startup override: %u ms\n"
208 " Wake On Touch: %s\n"
209 " Touch Timeout: %" PRIu16,
211 this->touch_sleep_timeout_);
214#ifdef USE_NEXTION_MAX_COMMANDS_PER_LOOP
219 ESP_LOGCONFIG(TAG,
" Wake Up Page: %u", this->
wake_up_page_);
222#ifdef USE_NEXTION_CONF_START_UP_PAGE
228#ifdef USE_NEXTION_COMMAND_SPACING
232#ifdef USE_NEXTION_MAX_QUEUE_SIZE
235#ifdef USE_NEXTION_TFT_UPLOAD
238 " TFT upload HTTP timeout: %" PRIu16
"ms\n"
239 " TFT upload HTTP retries: %u",
240 this->
tft_url_.c_str(), this->tft_upload_http_timeout_, this->tft_upload_http_retries_);
263 binarysensortype->update_component();
266 sensortype->update_component();
269 switchtype->update_component();
272 textsensortype->update_component();
291 int ret = vsnprintf(buffer,
sizeof(buffer),
format, arg);
294 ESP_LOGW(TAG,
"Bad cmd format: '%s'",
format);
302#ifdef NEXTION_PROTOCOL_LOG
304 ESP_LOGN(TAG,
"print_queue_members_ (top 10) size %zu", this->nextion_queue_.size());
305 ESP_LOGN(TAG,
"*******************************************");
307 for (
auto *i : this->nextion_queue_) {
312 ESP_LOGN(TAG,
"Queue null");
314 ESP_LOGN(TAG,
"Queue type: %d:%s, name: %s", i->component->get_queue_type(),
315 i->component->get_queue_type_string(), i->component->get_variable_name().c_str());
318 ESP_LOGN(TAG,
"*******************************************");
326 if (this->
connection_state_.nextion_reports_is_setup_ && !this->connection_state_.sent_setup_commands_) {
335#ifdef USE_NEXTION_CONF_START_UP_PAGE
364 ESP_LOGV(TAG,
"Manual ready set");
369#ifdef USE_NEXTION_COMMAND_SPACING
371#ifdef USE_NEXTION_WAVEFORM
372 if (!this->waveform_queue_.empty()) {
379#ifdef USE_NEXTION_COMMAND_SPACING
381#ifdef USE_NEXTION_MAX_COMMANDS_PER_LOOP
382 size_t commands_sent = 0;
385 for (
auto *item : this->nextion_queue_) {
386 if (item ==
nullptr || item->pending_command.empty()) {
390#ifdef USE_NEXTION_MAX_COMMANDS_PER_LOOP
392 ESP_LOGV(TAG,
"Pending cmds: loop limit reached, deferring");
405 item->pending_command.clear();
406 ESP_LOGVV(TAG,
"Pending cmd sent: %s", item->component->get_variable_name().c_str());
412 if (this->nextion_queue_.empty()) {
414 ESP_LOGE(TAG,
"Queue empty");
419 NextionQueue *nb = this->nextion_queue_.front();
420 if (!nb || !nb->component) {
421 ESP_LOGE(TAG,
"Invalid queue");
422 this->nextion_queue_.pop_front();
425 NextionComponentBase *
component = nb->component;
427 ESP_LOGN(TAG,
"Removed: %s",
component->get_variable_name().c_str());
430 if (
component->get_variable_name() ==
"sleep_wake") {
436 this->nextion_queue_.pop_front();
445 size_t to_read = std::min(avail,
sizeof(buf));
451 this->
command_data_.append(
reinterpret_cast<const char *
>(buf), to_read);
460#ifdef USE_NEXTION_MAX_COMMANDS_PER_LOOP
461 size_t commands_processed = 0;
464 size_t to_process_length = 0;
465 std::string to_process;
467 ESP_LOGN(TAG,
"command_data_ %s len %d", this->
command_data_.c_str(), this->command_data_.length());
468#ifdef NEXTION_PROTOCOL_LOG
471 while ((to_process_length = this->
command_data_.find(
reinterpret_cast<const char *
>(COMMAND_DELIMITER), 0,
472 DELIMITER_SIZE)) != std::string::npos) {
473#ifdef USE_NEXTION_MAX_COMMANDS_PER_LOOP
475 ESP_LOGV(TAG,
"Command limit reached, deferring");
479 ESP_LOGN(TAG,
"queue size: %zu", this->nextion_queue_.size());
480 while (to_process_length + DELIMITER_SIZE < this->
command_data_.length() &&
481 static_cast<uint8_t
>(this->command_data_[to_process_length + DELIMITER_SIZE]) == 0xFF) {
483 ESP_LOGN(TAG,
"Add 0xFF");
488 to_process_length -= 1;
489 to_process = this->
command_data_.substr(1, to_process_length);
491 switch (nextion_event) {
493 ESP_LOGW(TAG,
"Invalid instruction");
499 ESP_LOGVV(TAG,
"Cmd OK");
500 ESP_LOGN(TAG,
"this->nextion_queue_.empty() %s", YESNO(this->nextion_queue_.empty()));
504 if (this->nextion_queue_.empty()) {
511 ESP_LOGW(TAG,
"Invalid component ID/name");
515 ESP_LOGW(TAG,
"Invalid page ID");
519 ESP_LOGW(TAG,
"Invalid picture ID");
523 ESP_LOGW(TAG,
"Invalid font ID");
527 ESP_LOGW(TAG,
"File operation failed");
530 ESP_LOGW(TAG,
"CRC validation failed");
533 ESP_LOGW(TAG,
"Invalid baud rate");
536#ifdef USE_NEXTION_WAVEFORM
537 if (this->waveform_queue_.empty()) {
538 ESP_LOGW(TAG,
"Waveform ID/ch used but no sensor queued");
540 auto &nb = this->waveform_queue_.front();
541 NextionComponentBase *
component = nb->component;
542 ESP_LOGW(TAG,
"Invalid waveform ID %d/ch %d",
component->get_component_id(),
544 ESP_LOGN(TAG,
"Remove waveform ID %d/ch %d",
component->get_component_id(),
component->get_wave_channel_id());
546 this->waveform_queue_.pop();
549 ESP_LOGW(TAG,
"Waveform ID/ch error but waveform not enabled");
553 ESP_LOGW(TAG,
"Invalid variable name");
557 ESP_LOGW(TAG,
"Invalid variable operation");
561 ESP_LOGW(TAG,
"Variable assign failed");
565 ESP_LOGW(TAG,
"EEPROM operation failed");
568 ESP_LOGW(TAG,
"Invalid parameter count");
572 ESP_LOGW(TAG,
"Invalid component I/O");
575 ESP_LOGW(TAG,
"Undefined escape chars");
579 ESP_LOGW(TAG,
"Variable name too long");
584 ESP_LOGE(TAG,
"Serial buffer overflow");
588 if (to_process_length != 3) {
589 ESP_LOGW(TAG,
"Incorrect touch len: %zu (need 3)", to_process_length);
593 uint8_t page_id = to_process[0];
594 uint8_t component_id = to_process[1];
595 uint8_t touch_event = to_process[2];
596 ESP_LOGV(TAG,
"Touch %s: page %u comp %u", touch_event ?
"PRESS" :
"RELEASE", page_id, component_id);
597 for (
auto *touch : this->
touch_) {
598 touch->process_touch(page_id, component_id, touch_event != 0);
605 if (to_process_length != 1) {
606 ESP_LOGW(TAG,
"Page event: expect 1, got %zu", to_process_length);
610 uint8_t page_id = to_process[0];
611 ESP_LOGV(TAG,
"New page: %u", page_id);
620 if (to_process_length != 5) {
621 ESP_LOGW(TAG,
"Touch coordinate: expect 5, got %zu", to_process_length);
622 ESP_LOGW(TAG,
"%s", to_process.c_str());
626 const uint16_t
x = (uint16_t(to_process[0]) << 8) | to_process[1];
627 const uint16_t
y = (uint16_t(to_process[2]) << 8) | to_process[3];
628 const uint8_t touch_event = to_process[4];
629 ESP_LOGV(TAG,
"Touch %s at %u,%u", touch_event ?
"PRESS" :
"RELEASE",
x,
y);
639 if (this->nextion_queue_.empty()) {
640 ESP_LOGW(TAG,
"String return but queue is empty");
644 NextionQueue *nb = this->nextion_queue_.front();
645 if (!nb || !nb->component) {
646 ESP_LOGE(TAG,
"Invalid queue entry");
647 this->nextion_queue_.pop_front();
650 NextionComponentBase *
component = nb->component;
653 ESP_LOGE(TAG,
"String return but '%s' not text sensor",
component->get_variable_name().c_str());
655 ESP_LOGN(TAG,
"String resp: '%s' id: %s type: %s", to_process.c_str(),
component->get_variable_name().c_str(),
657 component->set_state_from_string(to_process,
true,
false);
661 this->nextion_queue_.pop_front();
672 if (this->nextion_queue_.empty()) {
673 ESP_LOGE(TAG,
"Numeric return but queue empty");
677 if (to_process_length < 4) {
678 ESP_LOGE(TAG,
"Numeric return but insufficient data (need 4, got %zu)", to_process_length);
682 int value =
static_cast<int>(
encode_uint32(to_process[3], to_process[2], to_process[1], to_process[0]));
684 NextionQueue *nb = this->nextion_queue_.front();
685 if (!nb || !nb->component) {
686 ESP_LOGE(TAG,
"Invalid queue");
687 this->nextion_queue_.pop_front();
690 NextionComponentBase *
component = nb->component;
695 ESP_LOGE(TAG,
"Numeric return but '%s' invalid type %d",
component->get_variable_name().c_str(),
698 ESP_LOGN(TAG,
"Numeric: %s type %d:%s val %d",
component->get_variable_name().c_str(),
700 component->set_state_from_int(value,
true,
false);
704 this->nextion_queue_.pop_front();
710 ESP_LOGVV(TAG,
"Auto sleep");
717 ESP_LOGVV(TAG,
"Auto wake");
725 ESP_LOGV(TAG,
"System start: %zu", to_process_length);
739 std::string variable_name;
742 auto index = to_process.find(
'\0');
743 if (index == std::string::npos || (to_process_length - index - 1) < 1) {
744 ESP_LOGE(TAG,
"Bad switch data (0x90)");
745 ESP_LOGN(TAG,
"proc: %s %zu %zu", to_process.c_str(), to_process_length, index);
749 variable_name = to_process.substr(0, index);
752 ESP_LOGN(TAG,
"Switch %s: %s", ONOFF(to_process[index] != 0), variable_name.c_str());
754#ifdef USE_NEXTION_TRIGGER_CUSTOM_SWITCH
758 for (
auto *switchtype : this->switchtype_) {
759 switchtype->process_bool(variable_name, to_process[index] != 0);
770 std::string variable_name;
772 auto index = to_process.find(
'\0');
773 if (index == std::string::npos || (to_process_length - index - 1) != 4) {
774 ESP_LOGE(TAG,
"Bad sensor data (0x91)");
775 ESP_LOGN(TAG,
"proc: %s %zu %zu", to_process.c_str(), to_process_length, index);
779 index = to_process.find(
'\0');
780 variable_name = to_process.substr(0, index);
782 int value =
static_cast<int>(
783 encode_uint32(to_process[index + 4], to_process[index + 3], to_process[index + 2], to_process[index + 1]));
785 ESP_LOGN(TAG,
"Sensor: %s=%d", variable_name.c_str(), value);
787#ifdef USE_NEXTION_TRIGGER_CUSTOM_SENSOR
791 for (
auto *sensor : this->sensortype_) {
792 sensor->process_sensor(variable_name, value);
805 std::string variable_name;
806 std::string text_value;
809 auto index = to_process.find(
'\0');
810 if (index == std::string::npos || (to_process_length - index - 1) < 1) {
811 ESP_LOGE(TAG,
"Bad text data (0x92)");
812 ESP_LOGN(TAG,
"proc: %s %zu %zu", to_process.c_str(), to_process_length, index);
816 variable_name = to_process.substr(0, index);
820 text_value = to_process.substr(index, to_process_length - index - 1);
822 ESP_LOGN(TAG,
"Text sensor: %s='%s'", variable_name.c_str(), text_value.c_str());
829#ifdef USE_NEXTION_TRIGGER_CUSTOM_TEXT_SENSOR
833 for (
auto *textsensortype : this->textsensortype_) {
834 textsensortype->process_text(variable_name, text_value);
845 std::string variable_name;
848 auto index = to_process.find(
'\0');
849 if (index == std::string::npos || (to_process_length - index - 1) < 1) {
850 ESP_LOGE(TAG,
"Bad binary data (0x93)");
851 ESP_LOGN(TAG,
"proc: %s %zu %zu", to_process.c_str(), to_process_length, index);
855 variable_name = to_process.substr(0, index);
858 ESP_LOGN(TAG,
"Binary sensor: %s=%s", variable_name.c_str(), ONOFF(to_process[index] != 0));
860#ifdef USE_NEXTION_TRIGGER_CUSTOM_BINARY_SENSOR
864 for (
auto *binarysensortype : this->binarysensortype_) {
865 binarysensortype->process_bool(&variable_name[0], to_process[index] != 0);
870 ESP_LOGVV(TAG,
"Data transmit done");
871#ifdef USE_NEXTION_WAVEFORM
877 ESP_LOGVV(TAG,
"Ready for transmit");
878#ifdef USE_NEXTION_WAVEFORM
879 if (this->waveform_queue_.empty()) {
880 ESP_LOGE(TAG,
"No waveforms queued");
883 auto &nb = this->waveform_queue_.front();
885 size_t buffer_to_send =
component->get_wave_buffer_size() < 255 ?
component->get_wave_buffer_size() : 255;
886 this->
write_array(component->get_wave_buffer().data(),
static_cast<int>(buffer_to_send));
887 ESP_LOGN(TAG,
"Send waveform: component id %d, waveform id %d, size %zu",
component->get_component_id(),
888 component->get_wave_channel_id(), buffer_to_send);
889 component->clear_wave_buffer(buffer_to_send);
891 this->waveform_queue_.pop();
893 ESP_LOGW(TAG,
"Waveform transmit ready but waveform not enabled");
898 ESP_LOGW(TAG,
"Unknown event: 0x%02X", nextion_event);
902 this->
command_data_.erase(0, to_process_length + DELIMITER_SIZE + 1);
907 if (this->
max_q_age_ms_ > 0 && !this->nextion_queue_.empty() &&
908 ms - this->nextion_queue_.front()->queue_time > this->max_q_age_ms_) {
909 for (
auto it = this->nextion_queue_.begin(); it != this->nextion_queue_.end();) {
910 if (ms - (*it)->queue_time > this->max_q_age_ms_) {
911 NextionComponentBase *
component = (*it)->component;
912 ESP_LOGV(TAG,
"Remove old queue '%s':'%s'",
component->get_queue_type_string(),
916 if (
component->get_variable_name() ==
"sleep_wake") {
923 it = this->nextion_queue_.erase(it);
930 ESP_LOGN(TAG,
"Loop end");
940 ESP_LOGN(TAG,
"State: %s=%lf (type %d)", name.c_str(),
state, queue_type);
942 switch (queue_type) {
944 for (
auto *sensor : this->sensortype_) {
945 if (name == sensor->get_variable_name()) {
946 sensor->set_state(
state,
true,
true);
953 for (
auto *sensor : this->binarysensortype_) {
954 if (name == sensor->get_variable_name()) {
955 sensor->set_state(
state != 0,
true,
true);
962 for (
auto *sensor : this->switchtype_) {
963 if (name == sensor->get_variable_name()) {
964 sensor->set_state(
state != 0,
true,
true);
971 ESP_LOGW(TAG,
"set_sensor_state: bad type %d", queue_type);
977 ESP_LOGV(TAG,
"State: %s='%s'", name.c_str(),
state.c_str());
979 for (
auto *sensor : this->textsensortype_) {
980 if (name == sensor->get_variable_name()) {
981 sensor->set_state(
state,
true,
true);
988 ESP_LOGV(TAG,
"Send states");
989 for (
auto *binarysensortype : this->binarysensortype_) {
990 if (force_update || binarysensortype->get_needs_to_send_update())
991 binarysensortype->send_state_to_nextion();
993 for (
auto *sensortype : this->sensortype_) {
994#ifdef USE_NEXTION_WAVEFORM
995 if ((force_update || sensortype->get_needs_to_send_update()) && sensortype->get_wave_channel_id() == UINT8_MAX) {
997 if (force_update || sensortype->get_needs_to_send_update()) {
999 sensortype->send_state_to_nextion();
1002 for (
auto *switchtype : this->switchtype_) {
1003 if (force_update || switchtype->get_needs_to_send_update())
1004 switchtype->send_state_to_nextion();
1006 for (
auto *textsensortype : this->textsensortype_) {
1007 if (force_update || textsensortype->get_needs_to_send_update())
1008 textsensortype->send_state_to_nextion();
1013 for (
auto *binarysensortype : this->binarysensortype_) {
1014 if (binarysensortype->get_variable_name().find(prefix, 0) != std::string::npos)
1015 binarysensortype->update_component_settings(
true);
1017 for (
auto *sensortype : this->sensortype_) {
1018 if (sensortype->get_variable_name().find(prefix, 0) != std::string::npos)
1019 sensortype->update_component_settings(
true);
1021 for (
auto *switchtype : this->switchtype_) {
1022 if (switchtype->get_variable_name().find(prefix, 0) != std::string::npos)
1023 switchtype->update_component_settings(
true);
1025 for (
auto *textsensortype : this->textsensortype_) {
1026 if (textsensortype->get_variable_name().find(prefix, 0) != std::string::npos)
1027 textsensortype->update_component_settings(
true);
1033 uint8_t nr_of_ff_bytes = 0;
1034 bool exit_flag =
false;
1035 bool ff_flag =
false;
1039 while ((timeout == 0 && this->
available()) ||
millis() - start <= timeout) {
1054 if (nr_of_ff_bytes >= 3)
1057 response += (char) c;
1059 if (response.find(0x05) != std::string::npos) {
1066 if (exit_flag || ff_flag) {
1072 response = response.substr(0, response.length() - 3);
1074 return response.length();
1088#ifdef USE_NEXTION_MAX_QUEUE_SIZE
1089 if (this->
max_queue_size_ > 0 && this->nextion_queue_.size() >= this->max_queue_size_) {
1090 ESP_LOGW(TAG,
"Queue full (%zu), drop: %s", this->nextion_queue_.size(), variable_name.c_str());
1095 RAMAllocator<nextion::NextionQueue> allocator;
1096 nextion::NextionQueue *nextion_queue = allocator.allocate(1);
1097 if (nextion_queue ==
nullptr) {
1098 ESP_LOGW(TAG,
"Queue alloc failed");
1101 new (nextion_queue) nextion::NextionQueue();
1104 nextion_queue->component =
new nextion::NextionComponentBase;
1105 nextion_queue->component->set_variable_name(variable_name);
1109 this->nextion_queue_.push_back(nextion_queue);
1111 ESP_LOGN(TAG,
"Queue NORESULT: %s", nextion_queue->component->get_variable_name().c_str());
1134#ifdef USE_NEXTION_COMMAND_SPACING
1142#ifdef USE_NEXTION_COMMAND_SPACING
1144 const std::string &command) {
1145#ifdef USE_NEXTION_MAX_QUEUE_SIZE
1146 if (this->
max_queue_size_ > 0 && this->nextion_queue_.size() >= this->max_queue_size_) {
1147 ESP_LOGW(TAG,
"Queue full (%zu), drop: %s", this->nextion_queue_.size(), variable_name.c_str());
1152 RAMAllocator<nextion::NextionQueue> allocator;
1153 nextion::NextionQueue *nextion_queue = allocator.allocate(1);
1154 if (nextion_queue ==
nullptr) {
1155 ESP_LOGW(TAG,
"Queue alloc failed");
1158 new (nextion_queue) nextion::NextionQueue();
1160 nextion_queue->component =
new nextion::NextionComponentBase;
1161 nextion_queue->component->set_variable_name(variable_name);
1163 nextion_queue->pending_command = command;
1165 this->nextion_queue_.push_back(nextion_queue);
1166 ESP_LOGVV(TAG,
"Queue with pending command: %s", variable_name.c_str());
1178 int ret = vsnprintf(buffer,
sizeof(buffer),
format, arg);
1181 ESP_LOGW(TAG,
"Bad cmd format: '%s'",
format);
1203 int ret = vsnprintf(buffer,
sizeof(buffer),
format, arg);
1206 ESP_LOGW(TAG,
"Bad cmd format: '%s'",
format);
1229 const std::string &variable_name_to_send, int32_t state_value) {
1234 const std::string &variable_name_to_send, int32_t state_value,
1235 bool is_sleep_safe) {
1256 const std::string &variable_name_to_send,
1257 const std::string &state_value) {
1262 const std::string &variable_name_to_send,
1263 const std::string &state_value,
bool is_sleep_safe) {
1268 state_value.c_str());
1284#ifdef USE_NEXTION_MAX_QUEUE_SIZE
1285 if (this->
max_queue_size_ > 0 && this->nextion_queue_.size() >= this->max_queue_size_) {
1286 ESP_LOGW(TAG,
"Queue full (%zu), drop GET: %s", this->nextion_queue_.size(),
1287 component->get_variable_name().c_str());
1292 RAMAllocator<nextion::NextionQueue> allocator;
1293 nextion::NextionQueue *nextion_queue = allocator.allocate(1);
1294 if (nextion_queue ==
nullptr) {
1295 ESP_LOGW(TAG,
"Queue alloc failed");
1298 new (nextion_queue) nextion::NextionQueue();
1303 ESP_LOGN(TAG,
"Queue %s: %s",
component->get_queue_type_string(),
component->get_variable_name().c_str());
1305 std::string command =
"get " +
component->get_variable_name_to_send();
1307#ifdef USE_NEXTION_COMMAND_SPACING
1311 nextion_queue->pending_command = command;
1312 this->nextion_queue_.push_back(nextion_queue);
1314 nextion_queue->pending_command.clear();
1318 this->nextion_queue_.push_back(nextion_queue);
1320 delete nextion_queue;
1325#ifdef USE_NEXTION_WAVEFORM
1335 RAMAllocator<nextion::NextionQueue> allocator;
1336 nextion::NextionQueue *nextion_queue = allocator.allocate(1);
1337 if (nextion_queue ==
nullptr) {
1338 ESP_LOGW(TAG,
"Queue alloc failed");
1341 new (nextion_queue) nextion::NextionQueue();
1346 if (!this->waveform_queue_.push(nextion_queue)) {
1347 ESP_LOGW(TAG,
"Waveform queue full, drop");
1348 delete nextion_queue;
1351 if (this->waveform_queue_.size() == 1)
1356 if (this->waveform_queue_.empty())
1359 auto *nb = this->waveform_queue_.front();
1361 size_t buffer_to_send =
component->get_wave_buffer_size() < 255 ?
component->get_wave_buffer_size() : 255;
1364 buf_append_printf(command,
sizeof(command), 0,
"addt %u,%u,%zu",
component->get_component_id(),
1365 component->get_wave_channel_id(), buffer_to_send);
void feed_wdt()
Feed the task watchdog.
uint32_t IRAM_ATTR HOT get_loop_component_start_time() const
Get the cached time in milliseconds from when the current component started its loop execution.
uint8_t get_spacing() const
Get current command spacing.
bool can_send(uint32_t now) const
Check if enough time has passed to send the next command.
void mark_sent(uint32_t now)
Record the transmit timestamp for the most recently sent command.
std::vector< NextionComponentBase * > touch_
std::vector< NextionComponentBase * > switchtype_
std::vector< NextionComponentBase * > binarysensortype_
StaticRingBuffer< NextionQueue *, 4 > waveform_queue_
Fixed-size ring buffer for waveform queue.
uint16_t max_q_age_ms_
Maximum age for queue items in ms.
bool send_command_(const std::string &command)
Manually send a raw command to the display and don't wait for an acknowledgement packet.
static constexpr size_t NEXTION_MODEL_MAX
Max observed ~18 chars from product numbering rules.
std::vector< NextionComponentBase * > textsensortype_
char firmware_version_[NEXTION_FW_MAX+1]
CallbackManager< void(uint8_t)> page_callback_
void check_pending_waveform_()
struct esphome::nextion::Nextion::@144 connection_state_
Status flags for Nextion display state management.
CallbackManager< void(StringRef, int32_t)> custom_sensor_callback_
void set_wake_up_page(uint8_t wake_up_page=255)
Sets which page Nextion loads when exiting sleep mode.
void all_components_send_state_(bool force_update=false)
void set_nextion_sensor_state(int queue_type, const std::string &name, float state)
Set the nextion sensor state object.
char serial_number_[NEXTION_SERIAL_MAX+1]
void add_addt_command_to_queue(NextionComponentBase *component) override
uint32_t flash_size_
Flash size in bytes — plain integer, no string needed.
uint16_t max_commands_per_loop_
uint16_t startup_override_ms_
Timeout before forcing setup complete.
void process_nextion_commands_()
void add_to_get_queue(NextionComponentBase *component) override
bool send_command_printf(const char *format,...) __attribute__((format(printf
Manually send a raw formatted command to the display.
std::list< NextionQueue * > nextion_queue_
void set_auto_wake_on_touch(bool auto_wake_on_touch)
Sets if Nextion should auto-wake from sleep when touch press occurs.
bool remove_from_q_(bool report_empty=true)
std::vector< NextionComponentBase * > sensortype_
uint16_t touch_sleep_timeout_
bool add_no_result_to_queue_with_printf_(const std::string &variable_name, const char *format,...) __attribute__((format(printf
void set_touch_sleep_timeout(uint16_t touch_sleep_timeout=0)
Set the touch sleep timeout of the display using the thsp command.
CallbackManager< void()> setup_callback_
std::string command_data_
optional< float > brightness_
CallbackManager< void()> wake_callback_
bool is_updating() override
Check if the TFT update process is currently running.
uint16_t recv_ret_string_(std::string &response, uint32_t timeout, bool recv_flag)
void goto_page(const char *page)
Show the page with a given name.
bool void add_no_result_to_queue_with_set_internal_(const std::string &variable_name, const std::string &variable_name_to_send, int32_t state_value, bool is_sleep_safe=false)
bool void add_no_result_to_queue_with_command_(const std::string &variable_name, const std::string &command)
CallbackManager< void(StringRef, bool)> custom_binary_sensor_callback_
bool send_command(const char *command)
Manually send a raw command to the display.
CallbackManager< void()> sleep_callback_
void reset_(bool reset_nextion=true)
void update_all_components()
void process_pending_in_queue_()
Process any commands in the queue that are pending due to command spacing.
static constexpr size_t NEXTION_FW_MAX
'S' prefix + integer (e.g. 'S99' or 123)
void add_no_result_to_queue_(const std::string &variable_name)
CallbackManager< void(uint8_t, uint8_t, bool)> touch_callback_
void print_queue_members_()
void dump_config() override
NextionCommandPacer command_pacer_
void set_nextion_text_state(const std::string &name, const std::string &state)
bool add_no_result_to_queue_with_ignore_sleep_printf_(const std::string &variable_name, const char *format,...) __attribute__((format(printf
CallbackManager< void()> buffer_overflow_callback_
void set_writer(const nextion_writer_t &writer)
char device_model_[NEXTION_MODEL_MAX+1]
CallbackManager< void(StringRef, StringRef)> custom_text_sensor_callback_
void add_no_result_to_queue_with_pending_command_(const std::string &variable_name, const std::string &command)
Add a command to the Nextion queue with a pending command for retry.
void add_no_result_to_queue_with_set(NextionComponentBase *component, int32_t state_value) override
static constexpr size_t NEXTION_SERIAL_MAX
Consistently 16 hex chars across all documented examples.
CallbackManager< void(StringRef, bool)> custom_switch_callback_
void update_components_by_prefix(const std::string &prefix)
uint32_t tft_upload_watchdog_timeout_
WDT timeout in ms (0 = no adjustment)
void set_backlight_brightness(float brightness)
Set the brightness of the backlight.
optional< std::array< uint8_t, N > > read_array()
void write_str(const char *str)
bool read_byte(uint8_t *data)
void write_array(const uint8_t *data, size_t len)
const Component * component
display::DisplayWriter< Nextion > nextion_writer_t
const char int const __FlashStringHelper * format
constexpr uint32_t encode_uint32(uint8_t byte1, uint8_t byte2, uint8_t byte3, uint8_t byte4)
Encode a 32-bit value given four bytes in most to least significant byte order.
size_t size_t const char va_start(args, fmt)
void HOT delay(uint32_t ms)
uint32_t IRAM_ATTR HOT millis()
Application App
Global storage of Application pointer - only one Application can exist.