mirror of
				https://github.com/esphome/esphome.git
				synced 2025-11-04 09:01:49 +00:00 
			
		
		
		
	Compare commits
	
		
			96 Commits
		
	
	
		
			2023.2.0b2
			...
			2023.2.4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					c037e95861 | ||
| 
						 | 
					2e1b35959f | ||
| 
						 | 
					7f46d9e0f9 | ||
| 
						 | 
					069b5f81a0 | ||
| 
						 | 
					3a36d0b13f | ||
| 
						 | 
					f0760e99b7 | ||
| 
						 | 
					18fecf8c09 | ||
| 
						 | 
					414cf1b333 | ||
| 
						 | 
					d10f891f51 | ||
| 
						 | 
					b5927322e6 | ||
| 
						 | 
					1cf4107e1c | ||
| 
						 | 
					c12408326c | ||
| 
						 | 
					4434e59e5a | ||
| 
						 | 
					45180d98f6 | ||
| 
						 | 
					44494ad18e | ||
| 
						 | 
					1447536906 | ||
| 
						 | 
					27ec517084 | ||
| 
						 | 
					ce1f034bac | ||
| 
						 | 
					f1f96f16e9 | ||
| 
						 | 
					7665e9b076 | ||
| 
						 | 
					227d94f38d | ||
| 
						 | 
					b724ae9e0e | ||
| 
						 | 
					df6cc14201 | ||
| 
						 | 
					d981d7859d | ||
| 
						 | 
					0f1ec515c1 | ||
| 
						 | 
					78e18256f7 | ||
| 
						 | 
					4899dfe642 | ||
| 
						 | 
					d6b6e94059 | ||
| 
						 | 
					310355a00b | ||
| 
						 | 
					8cf26d6f3c | ||
| 
						 | 
					b15a10f905 | ||
| 
						 | 
					5dcf1debd7 | ||
| 
						 | 
					9b57e1ac1d | ||
| 
						 | 
					68683e3a50 | ||
| 
						 | 
					4af4649e23 | ||
| 
						 | 
					8bcddef39d | ||
| 
						 | 
					4ac96ccea2 | ||
| 
						 | 
					3c5de77ae9 | ||
| 
						 | 
					a2925b1d37 | ||
| 
						 | 
					73748e9e20 | ||
| 
						 | 
					75c9823899 | ||
| 
						 | 
					c8c0bd3351 | ||
| 
						 | 
					e1cdeb7c8f | ||
| 
						 | 
					7f97f42552 | ||
| 
						 | 
					aa7f3569ec | ||
| 
						 | 
					2d0a08442e | ||
| 
						 | 
					d2380756b2 | ||
| 
						 | 
					925e3cb6c9 | ||
| 
						 | 
					6757acba56 | ||
| 
						 | 
					5cc91cdd95 | ||
| 
						 | 
					2b41886819 | ||
| 
						 | 
					72c6efd6a0 | ||
| 
						 | 
					a1f1804112 | ||
| 
						 | 
					a8b1ceb4e9 | ||
| 
						 | 
					4fb0f7f8c6 | ||
| 
						 | 
					958cadeca8 | ||
| 
						 | 
					00f2655f1a | ||
| 
						 | 
					074f5029eb | ||
| 
						 | 
					1691976587 | ||
| 
						 | 
					60e6b4d21e | ||
| 
						 | 
					5750591df2 | ||
| 
						 | 
					a75da54455 | ||
| 
						 | 
					de7f6c3f5f | ||
| 
						 | 
					4245480656 | ||
| 
						 | 
					1824c8131e | ||
| 
						 | 
					4e9606d2e0 | ||
| 
						 | 
					78500fa933 | ||
| 
						 | 
					9c69b98a49 | ||
| 
						 | 
					e6d8ef98d3 | ||
| 
						 | 
					3f1af1690b | ||
| 
						 | 
					84374b6b1e | ||
| 
						 | 
					391316c9b5 | ||
| 
						 | 
					705c62ebd7 | ||
| 
						 | 
					7209dd8bae | ||
| 
						 | 
					ab736c89bb | ||
| 
						 | 
					6911639617 | ||
| 
						 | 
					b9720d0715 | ||
| 
						 | 
					47b3267ed4 | ||
| 
						 | 
					e16ba2adb5 | ||
| 
						 | 
					0a19b1e32c | ||
| 
						 | 
					bae9a950c0 | ||
| 
						 | 
					72b2943332 | ||
| 
						 | 
					4ec0ef7548 | ||
| 
						 | 
					25bc6761f6 | ||
| 
						 | 
					81b6562c25 | ||
| 
						 | 
					ae74189fc2 | ||
| 
						 | 
					9e516efe10 | ||
| 
						 | 
					366e29439e | ||
| 
						 | 
					1c9c700d7f | ||
| 
						 | 
					b2e6b9d31f | ||
| 
						 | 
					7623f63846 | ||
| 
						 | 
					2bfaf9dce3 | ||
| 
						 | 
					5c2c1560bb | ||
| 
						 | 
					f7096ab78e | ||
| 
						 | 
					98f8feb625 | ||
| 
						 | 
					9944ca414e | 
							
								
								
									
										2
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							@@ -150,6 +150,6 @@ jobs:
 | 
				
			|||||||
              ref: "main",
 | 
					              ref: "main",
 | 
				
			||||||
              inputs: {
 | 
					              inputs: {
 | 
				
			||||||
                version: "${{ github.event.release.tag_name }}",
 | 
					                version: "${{ github.event.release.tag_name }}",
 | 
				
			||||||
                content: "${{ toJSON(github.event.release.body) }}"
 | 
					                content: ${{ toJSON(github.event.release.body) }}
 | 
				
			||||||
              }
 | 
					              }
 | 
				
			||||||
            })
 | 
					            })
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -128,3 +128,5 @@ tests/.esphome/
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
sdkconfig.*
 | 
					sdkconfig.*
 | 
				
			||||||
!sdkconfig.defaults
 | 
					!sdkconfig.defaults
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.tests/
 | 
				
			||||||
@@ -1,8 +1,6 @@
 | 
				
			|||||||
include LICENSE
 | 
					include LICENSE
 | 
				
			||||||
include README.md
 | 
					include README.md
 | 
				
			||||||
include requirements.txt
 | 
					include requirements.txt
 | 
				
			||||||
include esphome/dashboard/templates/*.html
 | 
					recursive-include esphome *.cpp *.h *.tcc *.c
 | 
				
			||||||
recursive-include esphome/dashboard/static *.ico *.js *.css *.woff* LICENSE
 | 
					 | 
				
			||||||
recursive-include esphome *.cpp *.h *.tcc
 | 
					 | 
				
			||||||
recursive-include esphome *.py.script
 | 
					recursive-include esphome *.py.script
 | 
				
			||||||
recursive-include esphome LICENSE.txt
 | 
					recursive-include esphome LICENSE.txt
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,6 +9,7 @@ from esphome.const import (
 | 
				
			|||||||
    DEVICE_CLASS_POWER,
 | 
					    DEVICE_CLASS_POWER,
 | 
				
			||||||
    DEVICE_CLASS_VOLTAGE,
 | 
					    DEVICE_CLASS_VOLTAGE,
 | 
				
			||||||
    STATE_CLASS_MEASUREMENT,
 | 
					    STATE_CLASS_MEASUREMENT,
 | 
				
			||||||
 | 
					    STATE_CLASS_TOTAL_INCREASING,
 | 
				
			||||||
    UNIT_AMPERE,
 | 
					    UNIT_AMPERE,
 | 
				
			||||||
    UNIT_KILOWATT_HOURS,
 | 
					    UNIT_KILOWATT_HOURS,
 | 
				
			||||||
    UNIT_VOLT,
 | 
					    UNIT_VOLT,
 | 
				
			||||||
@@ -66,16 +67,19 @@ CONFIG_SCHEMA = (
 | 
				
			|||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
 | 
					                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            cv.Optional(CONF_ENERGY_2): sensor.sensor_schema(
 | 
					            cv.Optional(CONF_ENERGY_2): sensor.sensor_schema(
 | 
				
			||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
 | 
					                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            cv.Optional(CONF_ENERGY_TOTAL): sensor.sensor_schema(
 | 
					            cv.Optional(CONF_ENERGY_TOTAL): sensor.sensor_schema(
 | 
				
			||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
 | 
					                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -83,11 +83,30 @@ def import_config(
 | 
				
			|||||||
        raise FileExistsError
 | 
					        raise FileExistsError
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if project_name == "esphome.web":
 | 
					    if project_name == "esphome.web":
 | 
				
			||||||
 | 
					        if "esp32c3" in import_url:
 | 
				
			||||||
 | 
					            board = "esp32-c3-devkitm-1"
 | 
				
			||||||
 | 
					            platform = "ESP32"
 | 
				
			||||||
 | 
					        elif "esp32s2" in import_url:
 | 
				
			||||||
 | 
					            board = "esp32-s2-saola-1"
 | 
				
			||||||
 | 
					            platform = "ESP32"
 | 
				
			||||||
 | 
					        elif "esp32s3" in import_url:
 | 
				
			||||||
 | 
					            board = "esp32-s3-devkitc-1"
 | 
				
			||||||
 | 
					            platform = "ESP32"
 | 
				
			||||||
 | 
					        elif "esp32" in import_url:
 | 
				
			||||||
 | 
					            board = "esp32dev"
 | 
				
			||||||
 | 
					            platform = "ESP32"
 | 
				
			||||||
 | 
					        elif "esp8266" in import_url:
 | 
				
			||||||
 | 
					            board = "esp01_1m"
 | 
				
			||||||
 | 
					            platform = "ESP8266"
 | 
				
			||||||
 | 
					        elif "pico-w" in import_url:
 | 
				
			||||||
 | 
					            board = "pico-w"
 | 
				
			||||||
 | 
					            platform = "RP2040"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        kwargs = {
 | 
					        kwargs = {
 | 
				
			||||||
            "name": name,
 | 
					            "name": name,
 | 
				
			||||||
            "friendly_name": friendly_name,
 | 
					            "friendly_name": friendly_name,
 | 
				
			||||||
            "platform": "ESP32" if "esp32" in import_url else "ESP8266",
 | 
					            "platform": platform,
 | 
				
			||||||
            "board": "esp32dev" if "esp32" in import_url else "esp01_1m",
 | 
					            "board": board,
 | 
				
			||||||
            "ssid": "!secret wifi_ssid",
 | 
					            "ssid": "!secret wifi_ssid",
 | 
				
			||||||
            "psk": "!secret wifi_password",
 | 
					            "psk": "!secret wifi_password",
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -15,6 +15,84 @@ static const char *const TAG = "display";
 | 
				
			|||||||
const Color COLOR_OFF(0, 0, 0, 0);
 | 
					const Color COLOR_OFF(0, 0, 0, 0);
 | 
				
			||||||
const Color COLOR_ON(255, 255, 255, 255);
 | 
					const Color COLOR_ON(255, 255, 255, 255);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Rect::expand(int16_t horizontal, int16_t vertical) {
 | 
				
			||||||
 | 
					  if (this->is_set() && (this->w >= (-2 * horizontal)) && (this->h >= (-2 * vertical))) {
 | 
				
			||||||
 | 
					    this->x = this->x - horizontal;
 | 
				
			||||||
 | 
					    this->y = this->y - vertical;
 | 
				
			||||||
 | 
					    this->w = this->w + (2 * horizontal);
 | 
				
			||||||
 | 
					    this->h = this->h + (2 * vertical);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Rect::extend(Rect rect) {
 | 
				
			||||||
 | 
					  if (!this->is_set()) {
 | 
				
			||||||
 | 
					    this->x = rect.x;
 | 
				
			||||||
 | 
					    this->y = rect.y;
 | 
				
			||||||
 | 
					    this->w = rect.w;
 | 
				
			||||||
 | 
					    this->h = rect.h;
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    if (this->x > rect.x) {
 | 
				
			||||||
 | 
					      this->x = rect.x;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->y > rect.y) {
 | 
				
			||||||
 | 
					      this->y = rect.y;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->x2() < rect.x2()) {
 | 
				
			||||||
 | 
					      this->w = rect.x2() - this->x;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->y2() < rect.y2()) {
 | 
				
			||||||
 | 
					      this->h = rect.y2() - this->y;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					void Rect::shrink(Rect rect) {
 | 
				
			||||||
 | 
					  if (!this->inside(rect)) {
 | 
				
			||||||
 | 
					    (*this) = Rect();
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    if (this->x < rect.x) {
 | 
				
			||||||
 | 
					      this->x = rect.x;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->y < rect.y) {
 | 
				
			||||||
 | 
					      this->y = rect.y;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->x2() > rect.x2()) {
 | 
				
			||||||
 | 
					      this->w = rect.x2() - this->x;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (this->y2() > rect.y2()) {
 | 
				
			||||||
 | 
					      this->h = rect.y2() - this->y;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool Rect::inside(int16_t x, int16_t y, bool absolute) {  // NOLINT
 | 
				
			||||||
 | 
					  if (!this->is_set()) {
 | 
				
			||||||
 | 
					    return true;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (absolute) {
 | 
				
			||||||
 | 
					    return ((x >= 0) && (x <= this->w) && (y >= 0) && (y <= this->h));
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    return ((x >= this->x) && (x <= this->x2()) && (y >= this->y) && (y <= this->y2()));
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bool Rect::inside(Rect rect, bool absolute) {
 | 
				
			||||||
 | 
					  if (!this->is_set() || !rect.is_set()) {
 | 
				
			||||||
 | 
					    return true;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  if (absolute) {
 | 
				
			||||||
 | 
					    return ((rect.x <= this->w) && (rect.w >= 0) && (rect.y <= this->h) && (rect.h >= 0));
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    return ((rect.x <= this->x2()) && (rect.x2() >= this->x) && (rect.y <= this->y2()) && (rect.y2() >= this->y));
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Rect::info(const std::string &prefix) {
 | 
				
			||||||
 | 
					  if (this->is_set()) {
 | 
				
			||||||
 | 
					    ESP_LOGI(TAG, "%s [%3d,%3d,%3d,%3d]", prefix.c_str(), this->x, this->y, this->w, this->h);
 | 
				
			||||||
 | 
					  } else
 | 
				
			||||||
 | 
					    ESP_LOGI(TAG, "%s ** IS NOT SET **", prefix.c_str());
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void DisplayBuffer::init_internal_(uint32_t buffer_length) {
 | 
					void DisplayBuffer::init_internal_(uint32_t buffer_length) {
 | 
				
			||||||
  ExternalRAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
 | 
					  ExternalRAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
 | 
				
			||||||
  this->buffer_ = allocator.allocate(buffer_length);
 | 
					  this->buffer_ = allocator.allocate(buffer_length);
 | 
				
			||||||
@@ -24,6 +102,7 @@ void DisplayBuffer::init_internal_(uint32_t buffer_length) {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
  this->clear();
 | 
					  this->clear();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void DisplayBuffer::fill(Color color) { this->filled_rectangle(0, 0, this->get_width(), this->get_height(), color); }
 | 
					void DisplayBuffer::fill(Color color) { this->filled_rectangle(0, 0, this->get_width(), this->get_height(), color); }
 | 
				
			||||||
void DisplayBuffer::clear() { this->fill(COLOR_OFF); }
 | 
					void DisplayBuffer::clear() { this->fill(COLOR_OFF); }
 | 
				
			||||||
int DisplayBuffer::get_width() {
 | 
					int DisplayBuffer::get_width() {
 | 
				
			||||||
@@ -50,6 +129,9 @@ int DisplayBuffer::get_height() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
void DisplayBuffer::set_rotation(DisplayRotation rotation) { this->rotation_ = rotation; }
 | 
					void DisplayBuffer::set_rotation(DisplayRotation rotation) { this->rotation_ = rotation; }
 | 
				
			||||||
void HOT DisplayBuffer::draw_pixel_at(int x, int y, Color color) {
 | 
					void HOT DisplayBuffer::draw_pixel_at(int x, int y, Color color) {
 | 
				
			||||||
 | 
					  if (!this->get_clipping().inside(x, y))
 | 
				
			||||||
 | 
					    return;  // NOLINT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  switch (this->rotation_) {
 | 
					  switch (this->rotation_) {
 | 
				
			||||||
    case DISPLAY_ROTATION_0_DEGREES:
 | 
					    case DISPLAY_ROTATION_0_DEGREES:
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
@@ -368,6 +450,10 @@ void DisplayBuffer::do_update_() {
 | 
				
			|||||||
  } else if (this->writer_.has_value()) {
 | 
					  } else if (this->writer_.has_value()) {
 | 
				
			||||||
    (*this->writer_)(*this);
 | 
					    (*this->writer_)(*this);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					  // remove all not ended clipping regions
 | 
				
			||||||
 | 
					  while (is_clipping()) {
 | 
				
			||||||
 | 
					    end_clipping();
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
void DisplayOnPageChangeTrigger::process(DisplayPage *from, DisplayPage *to) {
 | 
					void DisplayOnPageChangeTrigger::process(DisplayPage *from, DisplayPage *to) {
 | 
				
			||||||
  if ((this->from_ == nullptr || this->from_ == from) && (this->to_ == nullptr || this->to_ == to))
 | 
					  if ((this->from_ == nullptr || this->from_ == from) && (this->to_ == nullptr || this->to_ == to))
 | 
				
			||||||
@@ -392,6 +478,41 @@ void DisplayBuffer::strftime(int x, int y, Font *font, const char *format, time:
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void DisplayBuffer::start_clipping(Rect rect) {
 | 
				
			||||||
 | 
					  if (!this->clipping_rectangle_.empty()) {
 | 
				
			||||||
 | 
					    Rect r = this->clipping_rectangle_.back();
 | 
				
			||||||
 | 
					    rect.shrink(r);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  this->clipping_rectangle_.push_back(rect);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					void DisplayBuffer::end_clipping() {
 | 
				
			||||||
 | 
					  if (this->clipping_rectangle_.empty()) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "clear: Clipping is not set.");
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    this->clipping_rectangle_.pop_back();
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					void DisplayBuffer::extend_clipping(Rect add_rect) {
 | 
				
			||||||
 | 
					  if (this->clipping_rectangle_.empty()) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "add: Clipping is not set.");
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    this->clipping_rectangle_.back().extend(add_rect);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					void DisplayBuffer::shrink_clipping(Rect add_rect) {
 | 
				
			||||||
 | 
					  if (this->clipping_rectangle_.empty()) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "add: Clipping is not set.");
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    this->clipping_rectangle_.back().shrink(add_rect);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					Rect DisplayBuffer::get_clipping() {
 | 
				
			||||||
 | 
					  if (this->clipping_rectangle_.empty()) {
 | 
				
			||||||
 | 
					    return Rect();
 | 
				
			||||||
 | 
					  } else {
 | 
				
			||||||
 | 
					    return this->clipping_rectangle_.back();
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
bool Glyph::get_pixel(int x, int y) const {
 | 
					bool Glyph::get_pixel(int x, int y) const {
 | 
				
			||||||
  const int x_data = x - this->glyph_data_->offset_x;
 | 
					  const int x_data = x - this->glyph_data_->offset_x;
 | 
				
			||||||
  const int y_data = y - this->glyph_data_->offset_y;
 | 
					  const int y_data = y - this->glyph_data_->offset_y;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -4,7 +4,6 @@
 | 
				
			|||||||
#include "esphome/core/defines.h"
 | 
					#include "esphome/core/defines.h"
 | 
				
			||||||
#include "esphome/core/automation.h"
 | 
					#include "esphome/core/automation.h"
 | 
				
			||||||
#include "display_color_utils.h"
 | 
					#include "display_color_utils.h"
 | 
				
			||||||
 | 
					 | 
				
			||||||
#include <cstdarg>
 | 
					#include <cstdarg>
 | 
				
			||||||
#include <vector>
 | 
					#include <vector>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -100,6 +99,32 @@ enum DisplayRotation {
 | 
				
			|||||||
  DISPLAY_ROTATION_270_DEGREES = 270,
 | 
					  DISPLAY_ROTATION_270_DEGREES = 270,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static const int16_t VALUE_NO_SET = 32766;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Rect {
 | 
				
			||||||
 | 
					 public:
 | 
				
			||||||
 | 
					  int16_t x;  ///< X coordinate of corner
 | 
				
			||||||
 | 
					  int16_t y;  ///< Y coordinate of corner
 | 
				
			||||||
 | 
					  int16_t w;  ///< Width of region
 | 
				
			||||||
 | 
					  int16_t h;  ///< Height of region
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Rect() : x(VALUE_NO_SET), y(VALUE_NO_SET), w(VALUE_NO_SET), h(VALUE_NO_SET) {}  // NOLINT
 | 
				
			||||||
 | 
					  inline Rect(int16_t x, int16_t y, int16_t w, int16_t h) ALWAYS_INLINE : x(x), y(y), w(w), h(h) {}
 | 
				
			||||||
 | 
					  inline int16_t x2() { return this->x + this->w; };  ///< X coordinate of corner
 | 
				
			||||||
 | 
					  inline int16_t y2() { return this->y + this->h; };  ///< Y coordinate of corner
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  inline bool is_set() ALWAYS_INLINE { return (this->h != VALUE_NO_SET) && (this->w != VALUE_NO_SET); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void expand(int16_t horizontal, int16_t vertical);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void extend(Rect rect);
 | 
				
			||||||
 | 
					  void shrink(Rect rect);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  bool inside(Rect rect, bool absolute = false);
 | 
				
			||||||
 | 
					  bool inside(int16_t x, int16_t y, bool absolute = false);
 | 
				
			||||||
 | 
					  void info(const std::string &prefix = "rect info:");
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class Font;
 | 
					class Font;
 | 
				
			||||||
class Image;
 | 
					class Image;
 | 
				
			||||||
class DisplayBuffer;
 | 
					class DisplayBuffer;
 | 
				
			||||||
@@ -126,6 +151,7 @@ class DisplayBuffer {
 | 
				
			|||||||
  int get_width();
 | 
					  int get_width();
 | 
				
			||||||
  /// Get the height of the image in pixels with rotation applied.
 | 
					  /// Get the height of the image in pixels with rotation applied.
 | 
				
			||||||
  int get_height();
 | 
					  int get_height();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /// Set a single pixel at the specified coordinates to the given color.
 | 
					  /// Set a single pixel at the specified coordinates to the given color.
 | 
				
			||||||
  void draw_pixel_at(int x, int y, Color color = COLOR_ON);
 | 
					  void draw_pixel_at(int x, int y, Color color = COLOR_ON);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -374,6 +400,49 @@ class DisplayBuffer {
 | 
				
			|||||||
   */
 | 
					   */
 | 
				
			||||||
  virtual DisplayType get_display_type() = 0;
 | 
					  virtual DisplayType get_display_type() = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /** Set the clipping rectangle for further drawing
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * @param[in]  rect:       Pointer to Rect for clipping (or NULL for entire screen)
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * return true if success, false if error
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  void start_clipping(Rect rect);
 | 
				
			||||||
 | 
					  void start_clipping(int16_t left, int16_t top, int16_t right, int16_t bottom) {
 | 
				
			||||||
 | 
					    start_clipping(Rect(left, top, right - left, bottom - top));
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /** Add a rectangular region to the invalidation region
 | 
				
			||||||
 | 
					   * - This is usually called when an element has been modified
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * @param[in]  rect: Rectangle to add to the invalidation region
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  void extend_clipping(Rect rect);
 | 
				
			||||||
 | 
					  void extend_clipping(int16_t left, int16_t top, int16_t right, int16_t bottom) {
 | 
				
			||||||
 | 
					    this->extend_clipping(Rect(left, top, right - left, bottom - top));
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /** substract a rectangular region to the invalidation region
 | 
				
			||||||
 | 
					   *  - This is usually called when an element has been modified
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * @param[in]  rect: Rectangle to add to the invalidation region
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  void shrink_clipping(Rect rect);
 | 
				
			||||||
 | 
					  void shrink_clipping(uint16_t left, uint16_t top, uint16_t right, uint16_t bottom) {
 | 
				
			||||||
 | 
					    this->shrink_clipping(Rect(left, top, right - left, bottom - top));
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /** Reset the invalidation region
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  void end_clipping();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  /** Get the current the clipping rectangle
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * return rect for active clipping region
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  Rect get_clipping();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  bool is_clipping() const { return !this->clipping_rectangle_.empty(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 protected:
 | 
					 protected:
 | 
				
			||||||
  void vprintf_(int x, int y, Font *font, Color color, TextAlign align, const char *format, va_list arg);
 | 
					  void vprintf_(int x, int y, Font *font, Color color, TextAlign align, const char *format, va_list arg);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -390,6 +459,7 @@ class DisplayBuffer {
 | 
				
			|||||||
  DisplayPage *previous_page_{nullptr};
 | 
					  DisplayPage *previous_page_{nullptr};
 | 
				
			||||||
  std::vector<DisplayOnPageChangeTrigger *> on_page_change_triggers_;
 | 
					  std::vector<DisplayOnPageChangeTrigger *> on_page_change_triggers_;
 | 
				
			||||||
  bool auto_clear_enabled_{true};
 | 
					  bool auto_clear_enabled_{true};
 | 
				
			||||||
 | 
					  std::vector<Rect> clipping_rectangle_;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class DisplayPage {
 | 
					class DisplayPage {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -12,6 +12,8 @@ CONF_BLE_ID = "ble_id"
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
NO_BLUTOOTH_VARIANTS = [const.VARIANT_ESP32S2]
 | 
					NO_BLUTOOTH_VARIANTS = [const.VARIANT_ESP32S2]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					NO_BLUTOOTH_VARIANTS = [const.VARIANT_ESP32S2]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
esp32_ble_ns = cg.esphome_ns.namespace("esp32_ble")
 | 
					esp32_ble_ns = cg.esphome_ns.namespace("esp32_ble")
 | 
				
			||||||
ESP32BLE = esp32_ble_ns.class_("ESP32BLE", cg.Component)
 | 
					ESP32BLE = esp32_ble_ns.class_("ESP32BLE", cg.Component)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,6 +8,20 @@
 | 
				
			|||||||
#endif
 | 
					#endif
 | 
				
			||||||
#include <driver/ledc.h>
 | 
					#include <driver/ledc.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define CLOCK_FREQUENCY 80e6f
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef USE_ARDUINO
 | 
				
			||||||
 | 
					#ifdef SOC_LEDC_SUPPORT_XTAL_CLOCK
 | 
				
			||||||
 | 
					#undef CLOCK_FREQUENCY
 | 
				
			||||||
 | 
					// starting with ESP32 Arduino 2.0.2, the 40MHz crystal is used as clock by default if supported
 | 
				
			||||||
 | 
					#define CLOCK_FREQUENCY 40e6f
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
					#define DEFAULT_CLK LEDC_USE_APB_CLK
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static const uint8_t SETUP_ATTEMPT_COUNT_MAX = 5;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace esphome {
 | 
					namespace esphome {
 | 
				
			||||||
namespace ledc {
 | 
					namespace ledc {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -26,11 +40,11 @@ inline ledc_mode_t get_speed_mode(uint8_t) { return LEDC_LOW_SPEED_MODE; }
 | 
				
			|||||||
#endif
 | 
					#endif
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
float ledc_max_frequency_for_bit_depth(uint8_t bit_depth) { return 80e6f / float(1 << bit_depth); }
 | 
					float ledc_max_frequency_for_bit_depth(uint8_t bit_depth) { return CLOCK_FREQUENCY / float(1 << bit_depth); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
float ledc_min_frequency_for_bit_depth(uint8_t bit_depth, bool low_frequency) {
 | 
					float ledc_min_frequency_for_bit_depth(uint8_t bit_depth, bool low_frequency) {
 | 
				
			||||||
  const float max_div_num = ((1 << MAX_RES_BITS) - 1) / (low_frequency ? 32.0f : 256.0f);
 | 
					  const float max_div_num = ((1 << MAX_RES_BITS) - 1) / (low_frequency ? 32.0f : 256.0f);
 | 
				
			||||||
  return 80e6f / (max_div_num * float(1 << bit_depth));
 | 
					  return CLOCK_FREQUENCY / (max_div_num * float(1 << bit_depth));
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
optional<uint8_t> ledc_bit_depth_for_frequency(float frequency) {
 | 
					optional<uint8_t> ledc_bit_depth_for_frequency(float frequency) {
 | 
				
			||||||
@@ -46,6 +60,38 @@ optional<uint8_t> ledc_bit_depth_for_frequency(float frequency) {
 | 
				
			|||||||
  return {};
 | 
					  return {};
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef USE_ESP_IDF
 | 
				
			||||||
 | 
					esp_err_t configure_timer_frequency(ledc_mode_t speed_mode, ledc_timer_t timer_num, ledc_channel_t chan_num,
 | 
				
			||||||
 | 
					                                    uint8_t channel, uint8_t &bit_depth, float frequency) {
 | 
				
			||||||
 | 
					  bit_depth = *ledc_bit_depth_for_frequency(frequency);
 | 
				
			||||||
 | 
					  if (bit_depth < 1) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "Frequency %f can't be achieved with any bit depth", frequency);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ledc_timer_config_t timer_conf{};
 | 
				
			||||||
 | 
					  timer_conf.speed_mode = speed_mode;
 | 
				
			||||||
 | 
					  timer_conf.duty_resolution = static_cast<ledc_timer_bit_t>(bit_depth);
 | 
				
			||||||
 | 
					  timer_conf.timer_num = timer_num;
 | 
				
			||||||
 | 
					  timer_conf.freq_hz = (uint32_t) frequency;
 | 
				
			||||||
 | 
					  timer_conf.clk_cfg = DEFAULT_CLK;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Configure the time with fallback in case of error
 | 
				
			||||||
 | 
					  int attempt_count_max = SETUP_ATTEMPT_COUNT_MAX;
 | 
				
			||||||
 | 
					  esp_err_t init_result = ESP_FAIL;
 | 
				
			||||||
 | 
					  while (attempt_count_max > 0 && init_result != ESP_OK) {
 | 
				
			||||||
 | 
					    init_result = ledc_timer_config(&timer_conf);
 | 
				
			||||||
 | 
					    if (init_result != ESP_OK) {
 | 
				
			||||||
 | 
					      ESP_LOGW(TAG, "Unable to initialize timer with frequency %.1f and bit depth of %u", frequency, bit_depth);
 | 
				
			||||||
 | 
					      // try again with a lower bit depth
 | 
				
			||||||
 | 
					      timer_conf.duty_resolution = static_cast<ledc_timer_bit_t>(--bit_depth);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    attempt_count_max--;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return init_result;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void LEDCOutput::write_state(float state) {
 | 
					void LEDCOutput::write_state(float state) {
 | 
				
			||||||
  if (!initialized_) {
 | 
					  if (!initialized_) {
 | 
				
			||||||
    ESP_LOGW(TAG, "LEDC output hasn't been initialized yet!");
 | 
					    ESP_LOGW(TAG, "LEDC output hasn't been initialized yet!");
 | 
				
			||||||
@@ -61,6 +107,7 @@ void LEDCOutput::write_state(float state) {
 | 
				
			|||||||
  auto duty = static_cast<uint32_t>(duty_rounded);
 | 
					  auto duty = static_cast<uint32_t>(duty_rounded);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef USE_ARDUINO
 | 
					#ifdef USE_ARDUINO
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "Setting duty: %u on channel %u", duty, this->channel_);
 | 
				
			||||||
  ledcWrite(this->channel_, duty);
 | 
					  ledcWrite(this->channel_, duty);
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
#ifdef USE_ESP_IDF
 | 
					#ifdef USE_ESP_IDF
 | 
				
			||||||
@@ -72,6 +119,7 @@ void LEDCOutput::write_state(float state) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void LEDCOutput::setup() {
 | 
					void LEDCOutput::setup() {
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "Entering setup...");
 | 
				
			||||||
#ifdef USE_ARDUINO
 | 
					#ifdef USE_ARDUINO
 | 
				
			||||||
  this->update_frequency(this->frequency_);
 | 
					  this->update_frequency(this->frequency_);
 | 
				
			||||||
  this->turn_off();
 | 
					  this->turn_off();
 | 
				
			||||||
@@ -83,19 +131,16 @@ void LEDCOutput::setup() {
 | 
				
			|||||||
  auto timer_num = static_cast<ledc_timer_t>((channel_ % 8) / 2);
 | 
					  auto timer_num = static_cast<ledc_timer_t>((channel_ % 8) / 2);
 | 
				
			||||||
  auto chan_num = static_cast<ledc_channel_t>(channel_ % 8);
 | 
					  auto chan_num = static_cast<ledc_channel_t>(channel_ % 8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  bit_depth_ = *ledc_bit_depth_for_frequency(frequency_);
 | 
					  esp_err_t timer_init_result =
 | 
				
			||||||
  if (bit_depth_ < 1) {
 | 
					      configure_timer_frequency(speed_mode, timer_num, chan_num, this->channel_, this->bit_depth_, this->frequency_);
 | 
				
			||||||
    ESP_LOGW(TAG, "Frequency %f can't be achieved with any bit depth", frequency_);
 | 
					
 | 
				
			||||||
    this->status_set_warning();
 | 
					  if (timer_init_result != ESP_OK) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "Frequency %f can't be achieved with computed bit depth %u", this->frequency_, this->bit_depth_);
 | 
				
			||||||
 | 
					    this->status_set_error();
 | 
				
			||||||
 | 
					    return;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ledc_timer_config_t timer_conf{};
 | 
					  ESP_LOGV(TAG, "Configured frequency %f with a bit depth of %u bits", this->frequency_, this->bit_depth_);
 | 
				
			||||||
  timer_conf.speed_mode = speed_mode;
 | 
					 | 
				
			||||||
  timer_conf.duty_resolution = static_cast<ledc_timer_bit_t>(bit_depth_);
 | 
					 | 
				
			||||||
  timer_conf.timer_num = timer_num;
 | 
					 | 
				
			||||||
  timer_conf.freq_hz = (uint32_t) frequency_;
 | 
					 | 
				
			||||||
  timer_conf.clk_cfg = LEDC_AUTO_CLK;
 | 
					 | 
				
			||||||
  ledc_timer_config(&timer_conf);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ledc_channel_config_t chan_conf{};
 | 
					  ledc_channel_config_t chan_conf{};
 | 
				
			||||||
  chan_conf.gpio_num = pin_->get_pin();
 | 
					  chan_conf.gpio_num = pin_->get_pin();
 | 
				
			||||||
@@ -107,6 +152,7 @@ void LEDCOutput::setup() {
 | 
				
			|||||||
  chan_conf.hpoint = 0;
 | 
					  chan_conf.hpoint = 0;
 | 
				
			||||||
  ledc_channel_config(&chan_conf);
 | 
					  ledc_channel_config(&chan_conf);
 | 
				
			||||||
  initialized_ = true;
 | 
					  initialized_ = true;
 | 
				
			||||||
 | 
					  this->status_clear_error();
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -114,36 +160,80 @@ void LEDCOutput::dump_config() {
 | 
				
			|||||||
  ESP_LOGCONFIG(TAG, "LEDC Output:");
 | 
					  ESP_LOGCONFIG(TAG, "LEDC Output:");
 | 
				
			||||||
  LOG_PIN("  Pin ", this->pin_);
 | 
					  LOG_PIN("  Pin ", this->pin_);
 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  LEDC Channel: %u", this->channel_);
 | 
					  ESP_LOGCONFIG(TAG, "  LEDC Channel: %u", this->channel_);
 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  Frequency: %.1f Hz", this->frequency_);
 | 
					  ESP_LOGCONFIG(TAG, "  PWM Frequency: %.1f Hz", this->frequency_);
 | 
				
			||||||
 | 
					  ESP_LOGCONFIG(TAG, "  Bit depth: %u", this->bit_depth_);
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Max frequency for bit depth: %f", ledc_max_frequency_for_bit_depth(this->bit_depth_));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Min frequency for bit depth: %f",
 | 
				
			||||||
 | 
					           ledc_min_frequency_for_bit_depth(this->bit_depth_, (this->frequency_ < 100)));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Max frequency for bit depth-1: %f", ledc_max_frequency_for_bit_depth(this->bit_depth_ - 1));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Min frequency for bit depth-1: %f",
 | 
				
			||||||
 | 
					           ledc_min_frequency_for_bit_depth(this->bit_depth_ - 1, (this->frequency_ < 100)));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Max frequency for bit depth+1: %f", ledc_max_frequency_for_bit_depth(this->bit_depth_ + 1));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Min frequency for bit depth+1: %f",
 | 
				
			||||||
 | 
					           ledc_min_frequency_for_bit_depth(this->bit_depth_ + 1, (this->frequency_ < 100)));
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Max res bits: %d", MAX_RES_BITS);
 | 
				
			||||||
 | 
					  ESP_LOGV(TAG, "  Clock frequency: %f", CLOCK_FREQUENCY);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void LEDCOutput::update_frequency(float frequency) {
 | 
					void LEDCOutput::update_frequency(float frequency) {
 | 
				
			||||||
  auto bit_depth_opt = ledc_bit_depth_for_frequency(frequency);
 | 
					  auto bit_depth_opt = ledc_bit_depth_for_frequency(frequency);
 | 
				
			||||||
  if (!bit_depth_opt.has_value()) {
 | 
					  if (!bit_depth_opt.has_value()) {
 | 
				
			||||||
    ESP_LOGW(TAG, "Frequency %f can't be achieved with any bit depth", frequency);
 | 
					    ESP_LOGE(TAG, "Frequency %f can't be achieved with any bit depth", this->frequency_);
 | 
				
			||||||
    this->status_set_warning();
 | 
					    this->status_set_error();
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  this->bit_depth_ = bit_depth_opt.value_or(8);
 | 
					  this->bit_depth_ = bit_depth_opt.value_or(8);
 | 
				
			||||||
  this->frequency_ = frequency;
 | 
					  this->frequency_ = frequency;
 | 
				
			||||||
#ifdef USE_ARDUINO
 | 
					#ifdef USE_ARDUINO
 | 
				
			||||||
  ledcSetup(this->channel_, frequency, this->bit_depth_);
 | 
					  ESP_LOGV(TAG, "Using Arduino API - Trying to define channel, frequency and bit depth...");
 | 
				
			||||||
  initialized_ = true;
 | 
					  u_int32_t configured_frequency = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Configure LEDC channel, frequency and bit depth with fallback
 | 
				
			||||||
 | 
					  int attempt_count_max = SETUP_ATTEMPT_COUNT_MAX;
 | 
				
			||||||
 | 
					  while (attempt_count_max > 0 && configured_frequency == 0) {
 | 
				
			||||||
 | 
					    ESP_LOGV(TAG, "Trying initialize channel %u with frequency %.1f and bit depth of %u...", this->channel_,
 | 
				
			||||||
 | 
					             this->frequency_, this->bit_depth_);
 | 
				
			||||||
 | 
					    configured_frequency = ledcSetup(this->channel_, frequency, this->bit_depth_);
 | 
				
			||||||
 | 
					    if (configured_frequency != 0) {
 | 
				
			||||||
 | 
					      initialized_ = true;
 | 
				
			||||||
 | 
					      this->status_clear_error();
 | 
				
			||||||
 | 
					      ESP_LOGV(TAG, "Configured frequency: %u with bit depth: %u", configured_frequency, this->bit_depth_);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					      ESP_LOGW(TAG, "Unable to initialize channel %u with frequency %.1f and bit depth of %u", this->channel_,
 | 
				
			||||||
 | 
					               this->frequency_, this->bit_depth_);
 | 
				
			||||||
 | 
					      // try again with a lower bit depth
 | 
				
			||||||
 | 
					      this->bit_depth_--;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    attempt_count_max--;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (configured_frequency == 0) {
 | 
				
			||||||
 | 
					    ESP_LOGE(TAG, "Permanently failed to initialize channel %u with frequency %.1f and bit depth of %u", this->channel_,
 | 
				
			||||||
 | 
					             this->frequency_, this->bit_depth_);
 | 
				
			||||||
 | 
					    this->status_set_error();
 | 
				
			||||||
 | 
					    return;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif  // USE_ARDUINO
 | 
					#endif  // USE_ARDUINO
 | 
				
			||||||
#ifdef USE_ESP_IDF
 | 
					#ifdef USE_ESP_IDF
 | 
				
			||||||
  if (!initialized_) {
 | 
					  if (!initialized_) {
 | 
				
			||||||
    ESP_LOGW(TAG, "LEDC output hasn't been initialized yet!");
 | 
					    ESP_LOGW(TAG, "LEDC output hasn't been initialized yet!");
 | 
				
			||||||
    return;
 | 
					    return;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  auto speed_mode = get_speed_mode(channel_);
 | 
					  auto speed_mode = get_speed_mode(channel_);
 | 
				
			||||||
  auto timer_num = static_cast<ledc_timer_t>((channel_ % 8) / 2);
 | 
					  auto timer_num = static_cast<ledc_timer_t>((channel_ % 8) / 2);
 | 
				
			||||||
 | 
					  auto chan_num = static_cast<ledc_channel_t>(channel_ % 8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ledc_timer_config_t timer_conf{};
 | 
					  esp_err_t timer_init_result =
 | 
				
			||||||
  timer_conf.speed_mode = speed_mode;
 | 
					      configure_timer_frequency(speed_mode, timer_num, chan_num, this->channel_, this->bit_depth_, this->frequency_);
 | 
				
			||||||
  timer_conf.duty_resolution = static_cast<ledc_timer_bit_t>(bit_depth_);
 | 
					
 | 
				
			||||||
  timer_conf.timer_num = timer_num;
 | 
					  if (timer_init_result != ESP_OK) {
 | 
				
			||||||
  timer_conf.freq_hz = (uint32_t) frequency_;
 | 
					    ESP_LOGE(TAG, "Frequency %f can't be achieved with computed bit depth %u", this->frequency_, this->bit_depth_);
 | 
				
			||||||
  timer_conf.clk_cfg = LEDC_AUTO_CLK;
 | 
					    this->status_set_error();
 | 
				
			||||||
  ledc_timer_config(&timer_conf);
 | 
					    return;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  this->status_clear_error();
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
  // re-apply duty
 | 
					  // re-apply duty
 | 
				
			||||||
  this->write_state(this->duty_);
 | 
					  this->write_state(this->duty_);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -84,18 +84,18 @@ ClimateTraits AirConditioner::traits() {
 | 
				
			|||||||
  traits.set_supported_custom_presets(this->supported_custom_presets_);
 | 
					  traits.set_supported_custom_presets(this->supported_custom_presets_);
 | 
				
			||||||
  traits.set_supported_custom_fan_modes(this->supported_custom_fan_modes_);
 | 
					  traits.set_supported_custom_fan_modes(this->supported_custom_fan_modes_);
 | 
				
			||||||
  /* + MINIMAL SET OF CAPABILITIES */
 | 
					  /* + MINIMAL SET OF CAPABILITIES */
 | 
				
			||||||
  traits.add_supported_mode(ClimateMode::CLIMATE_MODE_OFF);
 | 
					 | 
				
			||||||
  traits.add_supported_mode(ClimateMode::CLIMATE_MODE_FAN_ONLY);
 | 
					 | 
				
			||||||
  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_AUTO);
 | 
					  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_AUTO);
 | 
				
			||||||
  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_LOW);
 | 
					  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_LOW);
 | 
				
			||||||
  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_MEDIUM);
 | 
					  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_MEDIUM);
 | 
				
			||||||
  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_HIGH);
 | 
					  traits.add_supported_fan_mode(ClimateFanMode::CLIMATE_FAN_HIGH);
 | 
				
			||||||
  traits.add_supported_swing_mode(ClimateSwingMode::CLIMATE_SWING_OFF);
 | 
					 | 
				
			||||||
  traits.add_supported_swing_mode(ClimateSwingMode::CLIMATE_SWING_VERTICAL);
 | 
					 | 
				
			||||||
  traits.add_supported_preset(ClimatePreset::CLIMATE_PRESET_NONE);
 | 
					 | 
				
			||||||
  traits.add_supported_preset(ClimatePreset::CLIMATE_PRESET_SLEEP);
 | 
					 | 
				
			||||||
  if (this->base_.getAutoconfStatus() == dudanov::midea::AUTOCONF_OK)
 | 
					  if (this->base_.getAutoconfStatus() == dudanov::midea::AUTOCONF_OK)
 | 
				
			||||||
    Converters::to_climate_traits(traits, this->base_.getCapabilities());
 | 
					    Converters::to_climate_traits(traits, this->base_.getCapabilities());
 | 
				
			||||||
 | 
					  if (!traits.get_supported_modes().empty())
 | 
				
			||||||
 | 
					    traits.add_supported_mode(ClimateMode::CLIMATE_MODE_OFF);
 | 
				
			||||||
 | 
					  if (!traits.get_supported_swing_modes().empty())
 | 
				
			||||||
 | 
					    traits.add_supported_swing_mode(ClimateSwingMode::CLIMATE_SWING_OFF);
 | 
				
			||||||
 | 
					  if (!traits.get_supported_presets().empty())
 | 
				
			||||||
 | 
					    traits.add_supported_preset(ClimatePreset::CLIMATE_PRESET_NONE);
 | 
				
			||||||
  return traits;
 | 
					  return traits;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -72,7 +72,7 @@ void MQTTClimateComponent::send_discovery(JsonObject root, mqtt::SendDiscoveryCo
 | 
				
			|||||||
    // preset_mode_state_topic
 | 
					    // preset_mode_state_topic
 | 
				
			||||||
    root[MQTT_PRESET_MODE_STATE_TOPIC] = this->get_preset_state_topic();
 | 
					    root[MQTT_PRESET_MODE_STATE_TOPIC] = this->get_preset_state_topic();
 | 
				
			||||||
    // presets
 | 
					    // presets
 | 
				
			||||||
    JsonArray presets = root.createNestedArray("presets");
 | 
					    JsonArray presets = root.createNestedArray("preset_modes");
 | 
				
			||||||
    if (traits.supports_preset(CLIMATE_PRESET_HOME))
 | 
					    if (traits.supports_preset(CLIMATE_PRESET_HOME))
 | 
				
			||||||
      presets.add("home");
 | 
					      presets.add("home");
 | 
				
			||||||
    if (traits.supports_preset(CLIMATE_PRESET_AWAY)) {
 | 
					    if (traits.supports_preset(CLIMATE_PRESET_AWAY)) {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,10 @@
 | 
				
			|||||||
#include "pid_autotuner.h"
 | 
					#include "pid_autotuner.h"
 | 
				
			||||||
#include "esphome/core/log.h"
 | 
					#include "esphome/core/log.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef M_PI
 | 
				
			||||||
 | 
					#define M_PI 3.1415926535897932384626433
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace esphome {
 | 
					namespace esphome {
 | 
				
			||||||
namespace pid {
 | 
					namespace pid {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -73,7 +77,7 @@ PIDAutotuner::PIDAutotuneResult PIDAutotuner::update(float setpoint, float proce
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (!std::isnan(this->setpoint_) && this->setpoint_ != setpoint) {
 | 
					  if (!std::isnan(this->setpoint_) && this->setpoint_ != setpoint) {
 | 
				
			||||||
    ESP_LOGW(TAG, "Setpoint changed during autotune! The result will not be accurate!");
 | 
					    ESP_LOGW(TAG, "%s: Setpoint changed during autotune! The result will not be accurate!", this->id_.c_str());
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  this->setpoint_ = setpoint;
 | 
					  this->setpoint_ = setpoint;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -87,7 +91,7 @@ PIDAutotuner::PIDAutotuneResult PIDAutotuner::update(float setpoint, float proce
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  if (!this->frequency_detector_.has_enough_data() || !this->amplitude_detector_.has_enough_data()) {
 | 
					  if (!this->frequency_detector_.has_enough_data() || !this->amplitude_detector_.has_enough_data()) {
 | 
				
			||||||
    // not enough data for calculation yet
 | 
					    // not enough data for calculation yet
 | 
				
			||||||
    ESP_LOGV(TAG, "  Not enough data yet for aututuner");
 | 
					    ESP_LOGV(TAG, "%s:   Not enough data yet for autotuner", this->id_.c_str());
 | 
				
			||||||
    return res;
 | 
					    return res;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -97,12 +101,13 @@ PIDAutotuner::PIDAutotuneResult PIDAutotuner::update(float setpoint, float proce
 | 
				
			|||||||
    // The frequency/amplitude is not fully accurate yet, try to wait
 | 
					    // The frequency/amplitude is not fully accurate yet, try to wait
 | 
				
			||||||
    // until the fault clears, or terminate after a while anyway
 | 
					    // until the fault clears, or terminate after a while anyway
 | 
				
			||||||
    if (zc_symmetrical) {
 | 
					    if (zc_symmetrical) {
 | 
				
			||||||
      ESP_LOGVV(TAG, "  ZC is not symmetrical");
 | 
					      ESP_LOGVV(TAG, "%s:   ZC is not symmetrical", this->id_.c_str());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (amplitude_convergent) {
 | 
					    if (amplitude_convergent) {
 | 
				
			||||||
      ESP_LOGVV(TAG, "  Amplitude is not convergent");
 | 
					      ESP_LOGVV(TAG, "%s:   Amplitude is not convergent", this->id_.c_str());
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    uint32_t phase = this->relay_function_.phase_count;
 | 
					    uint32_t phase = this->relay_function_.phase_count;
 | 
				
			||||||
 | 
					    ESP_LOGVV(TAG, "%s: >", this->id_.c_str());
 | 
				
			||||||
    ESP_LOGVV(TAG, "  Phase %u, enough=%u", phase, enough_data_phase_);
 | 
					    ESP_LOGVV(TAG, "  Phase %u, enough=%u", phase, enough_data_phase_);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (this->enough_data_phase_ == 0) {
 | 
					    if (this->enough_data_phase_ == 0) {
 | 
				
			||||||
@@ -116,7 +121,7 @@ PIDAutotuner::PIDAutotuneResult PIDAutotuner::update(float setpoint, float proce
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ESP_LOGI(TAG, "PID Autotune finished!");
 | 
					  ESP_LOGI(TAG, "%s: PID Autotune finished!", this->id_.c_str());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  float osc_ampl = this->amplitude_detector_.get_mean_oscillation_amplitude();
 | 
					  float osc_ampl = this->amplitude_detector_.get_mean_oscillation_amplitude();
 | 
				
			||||||
  float d = (this->relay_function_.output_positive - this->relay_function_.output_negative) / 2.0f;
 | 
					  float d = (this->relay_function_.output_positive - this->relay_function_.output_negative) / 2.0f;
 | 
				
			||||||
@@ -131,12 +136,12 @@ PIDAutotuner::PIDAutotuneResult PIDAutotuner::update(float setpoint, float proce
 | 
				
			|||||||
  return res;
 | 
					  return res;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
void PIDAutotuner::dump_config() {
 | 
					void PIDAutotuner::dump_config() {
 | 
				
			||||||
  ESP_LOGI(TAG, "PID Autotune:");
 | 
					 | 
				
			||||||
  if (this->state_ == AUTOTUNE_SUCCEEDED) {
 | 
					  if (this->state_ == AUTOTUNE_SUCCEEDED) {
 | 
				
			||||||
 | 
					    ESP_LOGI(TAG, "%s: PID Autotune:", this->id_.c_str());
 | 
				
			||||||
    ESP_LOGI(TAG, "  State: Succeeded!");
 | 
					    ESP_LOGI(TAG, "  State: Succeeded!");
 | 
				
			||||||
    bool has_issue = false;
 | 
					    bool has_issue = false;
 | 
				
			||||||
    if (!this->amplitude_detector_.is_amplitude_convergent()) {
 | 
					    if (!this->amplitude_detector_.is_amplitude_convergent()) {
 | 
				
			||||||
      ESP_LOGW(TAG, "  Could not reliable determine oscillation amplitude, PID parameters may be inaccurate!");
 | 
					      ESP_LOGW(TAG, "  Could not reliably determine oscillation amplitude, PID parameters may be inaccurate!");
 | 
				
			||||||
      ESP_LOGW(TAG, "    Please make sure you eliminate all outside influences on the measured temperature.");
 | 
					      ESP_LOGW(TAG, "    Please make sure you eliminate all outside influences on the measured temperature.");
 | 
				
			||||||
      has_issue = true;
 | 
					      has_issue = true;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -173,10 +178,12 @@ void PIDAutotuner::dump_config() {
 | 
				
			|||||||
    print_rule_("Pessen Integral PID", 0.7f, 1.75f, 0.105f);
 | 
					    print_rule_("Pessen Integral PID", 0.7f, 1.75f, 0.105f);
 | 
				
			||||||
    print_rule_("Some Overshoot PID", 0.333f, 0.667f, 0.111f);
 | 
					    print_rule_("Some Overshoot PID", 0.333f, 0.667f, 0.111f);
 | 
				
			||||||
    print_rule_("No Overshoot PID", 0.2f, 0.4f, 0.0625f);
 | 
					    print_rule_("No Overshoot PID", 0.2f, 0.4f, 0.0625f);
 | 
				
			||||||
 | 
					    ESP_LOGI(TAG, "%s: Autotune completed", this->id_.c_str());
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (this->state_ == AUTOTUNE_RUNNING) {
 | 
					  if (this->state_ == AUTOTUNE_RUNNING) {
 | 
				
			||||||
    ESP_LOGI(TAG, "  Autotune is still running!");
 | 
					    ESP_LOGD(TAG, "%s: PID Autotune:", this->id_.c_str());
 | 
				
			||||||
 | 
					    ESP_LOGD(TAG, "  Autotune is still running!");
 | 
				
			||||||
    ESP_LOGD(TAG, "  Status: Trying to reach %.2f °C", setpoint_ - relay_function_.current_target_error());
 | 
					    ESP_LOGD(TAG, "  Status: Trying to reach %.2f °C", setpoint_ - relay_function_.current_target_error());
 | 
				
			||||||
    ESP_LOGD(TAG, "  Stats so far:");
 | 
					    ESP_LOGD(TAG, "  Stats so far:");
 | 
				
			||||||
    ESP_LOGD(TAG, "    Phases: %u", relay_function_.phase_count);
 | 
					    ESP_LOGD(TAG, "    Phases: %u", relay_function_.phase_count);
 | 
				
			||||||
@@ -221,7 +228,6 @@ float PIDAutotuner::RelayFunction::update(float error) {
 | 
				
			|||||||
  float output = state == RELAY_FUNCTION_POSITIVE ? output_positive : output_negative;
 | 
					  float output = state == RELAY_FUNCTION_POSITIVE ? output_positive : output_negative;
 | 
				
			||||||
  if (change) {
 | 
					  if (change) {
 | 
				
			||||||
    this->phase_count++;
 | 
					    this->phase_count++;
 | 
				
			||||||
    ESP_LOGV(TAG, "Autotune: Turning output to %.1f%%", output * 100);
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  return output;
 | 
					  return output;
 | 
				
			||||||
@@ -245,10 +251,8 @@ void PIDAutotuner::OscillationFrequencyDetector::update(uint32_t now, float erro
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  if (had_crossing) {
 | 
					  if (had_crossing) {
 | 
				
			||||||
    // Had crossing above hysteresis threshold, record
 | 
					    // Had crossing above hysteresis threshold, record
 | 
				
			||||||
    ESP_LOGV(TAG, "Autotune: Detected Zero-Cross at %u", now);
 | 
					 | 
				
			||||||
    if (this->last_zerocross != 0) {
 | 
					    if (this->last_zerocross != 0) {
 | 
				
			||||||
      uint32_t dt = now - this->last_zerocross;
 | 
					      uint32_t dt = now - this->last_zerocross;
 | 
				
			||||||
      ESP_LOGV(TAG, "  dt: %u", dt);
 | 
					 | 
				
			||||||
      this->zerocrossing_intervals.push_back(dt);
 | 
					      this->zerocrossing_intervals.push_back(dt);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    this->last_zerocross = now;
 | 
					    this->last_zerocross = now;
 | 
				
			||||||
@@ -297,13 +301,11 @@ void PIDAutotuner::OscillationAmplitudeDetector::update(float error,
 | 
				
			|||||||
      // The positive error peak must have been in previous segment (180° shifted)
 | 
					      // The positive error peak must have been in previous segment (180° shifted)
 | 
				
			||||||
      // record phase_max
 | 
					      // record phase_max
 | 
				
			||||||
      this->phase_maxs.push_back(phase_max);
 | 
					      this->phase_maxs.push_back(phase_max);
 | 
				
			||||||
      ESP_LOGV(TAG, "Autotune: Phase Max: %f", phase_max);
 | 
					 | 
				
			||||||
    } else if (last_relay_state == RelayFunction::RELAY_FUNCTION_NEGATIVE) {
 | 
					    } else if (last_relay_state == RelayFunction::RELAY_FUNCTION_NEGATIVE) {
 | 
				
			||||||
      // Transitioned from negative error to positive error.
 | 
					      // Transitioned from negative error to positive error.
 | 
				
			||||||
      // The negative error peak must have been in previous segment (180° shifted)
 | 
					      // The negative error peak must have been in previous segment (180° shifted)
 | 
				
			||||||
      // record phase_min
 | 
					      // record phase_min
 | 
				
			||||||
      this->phase_mins.push_back(phase_min);
 | 
					      this->phase_mins.push_back(phase_min);
 | 
				
			||||||
      ESP_LOGV(TAG, "Autotune: Phase Min: %f", phase_min);
 | 
					 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // reset phase values for next phase
 | 
					    // reset phase values for next phase
 | 
				
			||||||
    this->phase_min = error;
 | 
					    this->phase_min = error;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -31,6 +31,8 @@ class PIDAutotuner {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  void dump_config();
 | 
					  void dump_config();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void set_autotuner_id(std::string id) { this->id_ = std::move(id); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  void set_noiseband(float noiseband) {
 | 
					  void set_noiseband(float noiseband) {
 | 
				
			||||||
    relay_function_.noiseband = noiseband;
 | 
					    relay_function_.noiseband = noiseband;
 | 
				
			||||||
    // ZC detector uses 1/4 the noiseband of relay function (noise suppression)
 | 
					    // ZC detector uses 1/4 the noiseband of relay function (noise suppression)
 | 
				
			||||||
@@ -106,6 +108,7 @@ class PIDAutotuner {
 | 
				
			|||||||
  } state_ = AUTOTUNE_RUNNING;
 | 
					  } state_ = AUTOTUNE_RUNNING;
 | 
				
			||||||
  float ku_;
 | 
					  float ku_;
 | 
				
			||||||
  float pu_;
 | 
					  float pu_;
 | 
				
			||||||
 | 
					  std::string id_;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // namespace pid
 | 
					}  // namespace pid
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -130,9 +130,6 @@ void PIDClimate::update_pid_() {
 | 
				
			|||||||
        // keep autotuner instance so that subsequent dump_configs will print the long result message.
 | 
					        // keep autotuner instance so that subsequent dump_configs will print the long result message.
 | 
				
			||||||
      } else {
 | 
					      } else {
 | 
				
			||||||
        value = res.output;
 | 
					        value = res.output;
 | 
				
			||||||
        if (mode != climate::CLIMATE_MODE_HEAT_COOL) {
 | 
					 | 
				
			||||||
          ESP_LOGW(TAG, "For PID autotuner you need to set AUTO (also called heat/cool) mode!");
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@@ -151,10 +148,24 @@ void PIDClimate::start_autotune(std::unique_ptr<PIDAutotuner> &&autotune) {
 | 
				
			|||||||
  float min_value = this->supports_cool_() ? -1.0f : 0.0f;
 | 
					  float min_value = this->supports_cool_() ? -1.0f : 0.0f;
 | 
				
			||||||
  float max_value = this->supports_heat_() ? 1.0f : 0.0f;
 | 
					  float max_value = this->supports_heat_() ? 1.0f : 0.0f;
 | 
				
			||||||
  this->autotuner_->config(min_value, max_value);
 | 
					  this->autotuner_->config(min_value, max_value);
 | 
				
			||||||
 | 
					  this->autotuner_->set_autotuner_id(this->get_object_id());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ESP_LOGI(TAG,
 | 
				
			||||||
 | 
					           "%s: Autotune has started. This can take a long time depending on the "
 | 
				
			||||||
 | 
					           "responsiveness of your system. Your system "
 | 
				
			||||||
 | 
					           "output will be altered to deliberately oscillate above and below the setpoint multiple times. "
 | 
				
			||||||
 | 
					           "Until your sensor provides a reading, the autotuner may display \'nan\'",
 | 
				
			||||||
 | 
					           this->get_object_id().c_str());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  this->set_interval("autotune-progress", 10000, [this]() {
 | 
					  this->set_interval("autotune-progress", 10000, [this]() {
 | 
				
			||||||
    if (this->autotuner_ != nullptr && !this->autotuner_->is_finished())
 | 
					    if (this->autotuner_ != nullptr && !this->autotuner_->is_finished())
 | 
				
			||||||
      this->autotuner_->dump_config();
 | 
					      this->autotuner_->dump_config();
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (mode != climate::CLIMATE_MODE_HEAT_COOL) {
 | 
				
			||||||
 | 
					    ESP_LOGW(TAG, "%s: !!! For PID autotuner you need to set AUTO (also called heat/cool) mode!",
 | 
				
			||||||
 | 
					             this->get_object_id().c_str());
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void PIDClimate::reset_integral_term() { this->controller_.reset_accumulated_integral(); }
 | 
					void PIDClimate::reset_integral_term() { this->controller_.reset_accumulated_integral(); }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -79,7 +79,9 @@ def register_trigger(name, type, data_type):
 | 
				
			|||||||
    validator = automation.validate_automation(
 | 
					    validator = automation.validate_automation(
 | 
				
			||||||
        {
 | 
					        {
 | 
				
			||||||
            cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(type),
 | 
					            cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(type),
 | 
				
			||||||
            cv.GenerateID(CONF_RECEIVER_ID): cv.use_id(RemoteReceiverBase),
 | 
					            cv.Optional(CONF_RECEIVER_ID): cv.invalid(
 | 
				
			||||||
 | 
					                "This has been removed in ESPHome 2022.3.0 and the trigger attaches directly to the parent receiver."
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
    registerer = TRIGGER_REGISTRY.register(f"on_{name}", validator)
 | 
					    registerer = TRIGGER_REGISTRY.register(f"on_{name}", validator)
 | 
				
			||||||
@@ -87,7 +89,6 @@ def register_trigger(name, type, data_type):
 | 
				
			|||||||
    def decorator(func):
 | 
					    def decorator(func):
 | 
				
			||||||
        async def new_func(config):
 | 
					        async def new_func(config):
 | 
				
			||||||
            var = cg.new_Pvariable(config[CONF_TRIGGER_ID])
 | 
					            var = cg.new_Pvariable(config[CONF_TRIGGER_ID])
 | 
				
			||||||
            await register_listener(var, config)
 | 
					 | 
				
			||||||
            await coroutine(func)(var, config)
 | 
					            await coroutine(func)(var, config)
 | 
				
			||||||
            await automation.build_automation(var, [(data_type, "x")], config)
 | 
					            await automation.build_automation(var, [(data_type, "x")], config)
 | 
				
			||||||
            return var
 | 
					            return var
 | 
				
			||||||
@@ -223,10 +224,12 @@ async def build_binary_sensor(full_config):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def build_triggers(full_config):
 | 
					async def build_triggers(full_config):
 | 
				
			||||||
 | 
					    triggers = []
 | 
				
			||||||
    for key in TRIGGER_REGISTRY:
 | 
					    for key in TRIGGER_REGISTRY:
 | 
				
			||||||
        for config in full_config.get(key, []):
 | 
					        for config in full_config.get(key, []):
 | 
				
			||||||
            func = TRIGGER_REGISTRY[key][0]
 | 
					            func = TRIGGER_REGISTRY[key][0]
 | 
				
			||||||
            await func(config)
 | 
					            triggers.append(await func(config))
 | 
				
			||||||
 | 
					    return triggers
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def build_dumpers(config):
 | 
					async def build_dumpers(config):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -56,7 +56,9 @@ async def to_code(config):
 | 
				
			|||||||
    for dumper in dumpers:
 | 
					    for dumper in dumpers:
 | 
				
			||||||
        cg.add(var.register_dumper(dumper))
 | 
					        cg.add(var.register_dumper(dumper))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await remote_base.build_triggers(config)
 | 
					    triggers = await remote_base.build_triggers(config)
 | 
				
			||||||
 | 
					    for trigger in triggers:
 | 
				
			||||||
 | 
					        cg.add(var.register_listener(trigger))
 | 
				
			||||||
    await cg.register_component(var, config)
 | 
					    await cg.register_component(var, config)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cg.add(var.set_tolerance(config[CONF_TOLERANCE]))
 | 
					    cg.add(var.set_tolerance(config[CONF_TOLERANCE]))
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -38,8 +38,8 @@ void Wiegand::setup() {
 | 
				
			|||||||
bool check_eparity(uint64_t value, int start, int length) {
 | 
					bool check_eparity(uint64_t value, int start, int length) {
 | 
				
			||||||
  int parity = 0;
 | 
					  int parity = 0;
 | 
				
			||||||
  uint64_t mask = 1LL << start;
 | 
					  uint64_t mask = 1LL << start;
 | 
				
			||||||
  for (int i = 0; i <= length; i++, mask <<= 1) {
 | 
					  for (int i = 0; i < length; i++, mask <<= 1) {
 | 
				
			||||||
    if (value & i)
 | 
					    if (value & mask)
 | 
				
			||||||
      parity++;
 | 
					      parity++;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  return !(parity & 1);
 | 
					  return !(parity & 1);
 | 
				
			||||||
@@ -48,8 +48,8 @@ bool check_eparity(uint64_t value, int start, int length) {
 | 
				
			|||||||
bool check_oparity(uint64_t value, int start, int length) {
 | 
					bool check_oparity(uint64_t value, int start, int length) {
 | 
				
			||||||
  int parity = 0;
 | 
					  int parity = 0;
 | 
				
			||||||
  uint64_t mask = 1LL << start;
 | 
					  uint64_t mask = 1LL << start;
 | 
				
			||||||
  for (int i = 0; i <= length; i++, mask <<= 1) {
 | 
					  for (int i = 0; i < length; i++, mask <<= 1) {
 | 
				
			||||||
    if (value & i)
 | 
					    if (value & mask)
 | 
				
			||||||
      parity++;
 | 
					      parity++;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  return parity & 1;
 | 
					  return parity & 1;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,6 @@
 | 
				
			|||||||
"""Constants used by esphome."""
 | 
					"""Constants used by esphome."""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
__version__ = "2023.2.0b2"
 | 
					__version__ = "2023.2.4"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
 | 
					ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -545,6 +545,14 @@ class DownloadBinaryRequestHandler(BaseHandler):
 | 
				
			|||||||
        self.finish()
 | 
					        self.finish()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class EsphomeVersionHandler(BaseHandler):
 | 
				
			||||||
 | 
					    @authenticated
 | 
				
			||||||
 | 
					    def get(self):
 | 
				
			||||||
 | 
					        self.set_header("Content-Type", "application/json")
 | 
				
			||||||
 | 
					        self.write(json.dumps({"version": const.__version__}))
 | 
				
			||||||
 | 
					        self.finish()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def _list_dashboard_entries():
 | 
					def _list_dashboard_entries():
 | 
				
			||||||
    files = settings.list_yaml_files()
 | 
					    files = settings.list_yaml_files()
 | 
				
			||||||
    return [DashboardEntry(file) for file in files]
 | 
					    return [DashboardEntry(file) for file in files]
 | 
				
			||||||
@@ -1000,8 +1008,14 @@ class SafeLoaderIgnoreUnknown(yaml.SafeLoader):
 | 
				
			|||||||
    def ignore_unknown(self, node):
 | 
					    def ignore_unknown(self, node):
 | 
				
			||||||
        return f"{node.tag} {node.value}"
 | 
					        return f"{node.tag} {node.value}"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def construct_yaml_binary(self, node) -> str:
 | 
				
			||||||
 | 
					        return super().construct_yaml_binary(node).decode("ascii")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
SafeLoaderIgnoreUnknown.add_constructor(None, SafeLoaderIgnoreUnknown.ignore_unknown)
 | 
					SafeLoaderIgnoreUnknown.add_constructor(None, SafeLoaderIgnoreUnknown.ignore_unknown)
 | 
				
			||||||
 | 
					SafeLoaderIgnoreUnknown.add_constructor(
 | 
				
			||||||
 | 
					    "tag:yaml.org,2002:binary", SafeLoaderIgnoreUnknown.construct_yaml_binary
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class JsonConfigRequestHandler(BaseHandler):
 | 
					class JsonConfigRequestHandler(BaseHandler):
 | 
				
			||||||
@@ -1135,6 +1149,7 @@ def make_app(debug=get_bool_env(ENV_DEV)):
 | 
				
			|||||||
            (f"{rel}rename", EsphomeRenameHandler),
 | 
					            (f"{rel}rename", EsphomeRenameHandler),
 | 
				
			||||||
            (f"{rel}prometheus-sd", PrometheusServiceDiscoveryHandler),
 | 
					            (f"{rel}prometheus-sd", PrometheusServiceDiscoveryHandler),
 | 
				
			||||||
            (f"{rel}boards/([a-z0-9]+)", BoardsRequestHandler),
 | 
					            (f"{rel}boards/([a-z0-9]+)", BoardsRequestHandler),
 | 
				
			||||||
 | 
					            (f"{rel}version", EsphomeVersionHandler),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
        **app_settings,
 | 
					        **app_settings,
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -142,7 +142,10 @@ def get_ini_content():
 | 
				
			|||||||
    # Sort to avoid changing build flags order
 | 
					    # Sort to avoid changing build flags order
 | 
				
			||||||
    CORE.add_platformio_option("build_flags", sorted(CORE.build_flags))
 | 
					    CORE.add_platformio_option("build_flags", sorted(CORE.build_flags))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    content = f"[env:{CORE.name}]\n"
 | 
					    content = "[platformio]\n"
 | 
				
			||||||
 | 
					    content += f"description = ESPHome {__version__}\n"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    content += f"[env:{CORE.name}]\n"
 | 
				
			||||||
    content += format_ini(CORE.platformio_options)
 | 
					    content += format_ini(CORE.platformio_options)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return content
 | 
					    return content
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,7 +9,7 @@ pyserial==3.5
 | 
				
			|||||||
platformio==6.1.5  # When updating platformio, also update Dockerfile
 | 
					platformio==6.1.5  # When updating platformio, also update Dockerfile
 | 
				
			||||||
esptool==4.4
 | 
					esptool==4.4
 | 
				
			||||||
click==8.1.3
 | 
					click==8.1.3
 | 
				
			||||||
esphome-dashboard==20230120.0
 | 
					esphome-dashboard==20230214.0
 | 
				
			||||||
aioesphomeapi==13.1.0
 | 
					aioesphomeapi==13.1.0
 | 
				
			||||||
zeroconf==0.47.1
 | 
					zeroconf==0.47.1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user