ESPHome  2023.11.6
graph.cpp
Go to the documentation of this file.
1 #include "graph.h"
3 #include "esphome/core/color.h"
4 #include "esphome/core/log.h"
5 #include "esphome/core/hal.h"
6 #include <algorithm>
7 #include <sstream>
8 #include <iostream> // std::cout, std::fixed
9 #include <iomanip>
10 namespace esphome {
11 namespace graph {
12 
13 using namespace display;
14 
15 static const char *const TAG = "graph";
16 static const char *const TAGL = "graphlegend";
17 
19  this->length_ = length;
20  this->samples_.resize(length, NAN);
21  this->last_sample_ = millis();
22 }
23 
24 void HistoryData::take_sample(float data) {
25  uint32_t tm = millis();
26  uint32_t dt = tm - last_sample_;
27  last_sample_ = tm;
28 
29  // Step data based on time
30  this->period_ += dt;
31  while (this->period_ >= this->update_time_) {
32  this->samples_[this->count_] = data;
33  this->period_ -= this->update_time_;
34  this->count_ = (this->count_ + 1) % this->length_;
35  ESP_LOGV(TAG, "Updating trace with value: %f", data);
36  }
37  if (!std::isnan(data)) {
38  // Recalc recent max/min
39  this->recent_min_ = data;
40  this->recent_max_ = data;
41  for (int i = 0; i < this->length_; i++) {
42  if (!std::isnan(this->samples_[i])) {
43  if (this->recent_max_ < this->samples_[i])
44  this->recent_max_ = this->samples_[i];
45  if (this->recent_min_ > this->samples_[i])
46  this->recent_min_ = this->samples_[i];
47  }
48  }
49  }
50 }
51 
53  ESP_LOGI(TAG, "Init trace for sensor %s", this->get_name().c_str());
54  this->data_.init(g->get_width());
55  sensor_->add_on_state_callback([this](float state) { this->data_.take_sample(state); });
56  this->data_.set_update_time_ms(g->get_duration() * 1000 / g->get_width());
57 }
58 
59 void Graph::draw(Display *buff, uint16_t x_offset, uint16_t y_offset, Color color) {
61  if (this->border_) {
62  buff->horizontal_line(x_offset, y_offset, this->width_, color);
63  buff->horizontal_line(x_offset, y_offset + this->height_ - 1, this->width_, color);
64  buff->vertical_line(x_offset, y_offset, this->height_, color);
65  buff->vertical_line(x_offset + this->width_ - 1, y_offset, this->height_, color);
66  }
68  float ymin = NAN;
69  float ymax = NAN;
70  for (auto *trace : traces_) {
71  float mx = trace->get_tracedata()->get_recent_max();
72  float mn = trace->get_tracedata()->get_recent_min();
73  if (std::isnan(ymax) || (ymax < mx))
74  ymax = mx;
75  if (std::isnan(ymin) || (ymin > mn))
76  ymin = mn;
77  }
78  // Adjust if manually overridden
79  if (!std::isnan(this->min_value_))
80  ymin = this->min_value_;
81  if (!std::isnan(this->max_value_))
82  ymax = this->max_value_;
83 
84  float yrange = ymax - ymin;
85  if (yrange > this->max_range_) {
86  // Look back in trace data to best-fit into local range
87  float mx = NAN;
88  float mn = NAN;
89  for (uint32_t i = 0; i < this->width_; i++) {
90  for (auto *trace : traces_) {
91  float v = trace->get_tracedata()->get_value(i);
92  if (!std::isnan(v)) {
93  if ((v - mn) > this->max_range_)
94  break;
95  if ((mx - v) > this->max_range_)
96  break;
97  if (std::isnan(mx) || (v > mx))
98  mx = v;
99  if (std::isnan(mn) || (v < mn))
100  mn = v;
101  }
102  }
103  }
104  yrange = this->max_range_;
105  if (!std::isnan(mn)) {
106  ymin = mn;
107  ymax = ymin + this->max_range_;
108  }
109  ESP_LOGV(TAG, "Graphing at max_range. Using local min %f, max %f", mn, mx);
110  }
111 
112  float y_per_div = this->min_range_;
113  if (!std::isnan(this->gridspacing_y_)) {
114  y_per_div = this->gridspacing_y_;
115  }
116  // Restrict drawing too many gridlines
117  if (yrange > 10 * y_per_div) {
118  while (yrange > 10 * y_per_div) {
119  y_per_div *= 2;
120  }
121  ESP_LOGW(TAG, "Graphing reducing y-scale to prevent too many gridlines");
122  }
123 
124  // Adjust limits to nice y_per_div boundaries
125  int yn = 0;
126  int ym = 1;
127  if (!std::isnan(ymin) && !std::isnan(ymax)) {
128  yn = (int) floorf(ymin / y_per_div);
129  ym = (int) ceilf(ymax / y_per_div);
130  if (yn == ym) {
131  ym++;
132  }
133  ymin = yn * y_per_div;
134  ymax = ym * y_per_div;
135  yrange = ymax - ymin;
136  }
137 
139  if (!std::isnan(this->gridspacing_y_)) {
140  for (int y = yn; y <= ym; y++) {
141  int16_t py = (int16_t) roundf((this->height_ - 1) * (1.0 - (float) (y - yn) / (ym - yn)));
142  for (uint32_t x = 0; x < this->width_; x += 2) {
143  buff->draw_pixel_at(x_offset + x, y_offset + py, color);
144  }
145  }
146  }
147  if (!std::isnan(this->gridspacing_x_) && (this->gridspacing_x_ > 0)) {
148  int n = this->duration_ / this->gridspacing_x_;
149  // Restrict drawing too many gridlines
150  if (n > 20) {
151  while (n > 20) {
152  n /= 2;
153  }
154  ESP_LOGW(TAG, "Graphing reducing x-scale to prevent too many gridlines");
155  }
156  for (int i = 0; i <= n; i++) {
157  for (uint32_t y = 0; y < this->height_; y += 2) {
158  buff->draw_pixel_at(x_offset + i * (this->width_ - 1) / n, y_offset + y, color);
159  }
160  }
161  }
162 
164  ESP_LOGV(TAG, "Updating graph. ymin %f, ymax %f", ymin, ymax);
165  for (auto *trace : traces_) {
166  Color c = trace->get_line_color();
167  uint16_t thick = trace->get_line_thickness();
168  for (uint32_t i = 0; i < this->width_; i++) {
169  float v = (trace->get_tracedata()->get_value(i) - ymin) / yrange;
170  if (!std::isnan(v) && (thick > 0)) {
171  int16_t x = this->width_ - 1 - i;
172  uint8_t b = (i % (thick * LineType::PATTERN_LENGTH)) / thick;
173  if (((uint8_t) trace->get_line_type() & (1 << b)) == (1 << b)) {
174  int16_t y = (int16_t) roundf((this->height_ - 1) * (1.0 - v)) - thick / 2;
175  for (uint16_t t = 0; t < thick; t++) {
176  buff->draw_pixel_at(x_offset + x, y_offset + y + t, c);
177  }
178  }
179  }
180  }
181  }
182 }
183 
186  parent_ = g;
187 
188  // Determine maximum expected text and value width / height
189  int txtw = 0, txth = 0;
190  int valw = 0, valh = 0;
191  int lt = 0;
192  for (auto *trace : g->traces_) {
193  std::string txtstr = trace->get_name();
194  int fw, fos, fbl, fh;
195  this->font_label_->measure(txtstr.c_str(), &fw, &fos, &fbl, &fh);
196  if (fw > txtw)
197  txtw = fw;
198  if (fh > txth)
199  txth = fh;
200  if (trace->get_line_thickness() > lt)
201  lt = trace->get_line_thickness();
202  ESP_LOGI(TAGL, " %s %d %d", txtstr.c_str(), fw, fh);
203 
204  if (this->values_ != VALUE_POSITION_TYPE_NONE) {
205  std::stringstream ss;
206  ss << std::fixed << std::setprecision(trace->sensor_->get_accuracy_decimals()) << trace->sensor_->get_state();
207  std::string valstr = ss.str();
208  if (this->units_) {
209  valstr += trace->sensor_->get_unit_of_measurement();
210  }
211  this->font_value_->measure(valstr.c_str(), &fw, &fos, &fbl, &fh);
212  if (fw > valw)
213  valw = fw;
214  if (fh > valh)
215  valh = fh;
216  ESP_LOGI(TAGL, " %s %d %d", valstr.c_str(), fw, fh);
217  }
218  }
219  // Add extra margin
220  txtw *= 1.2;
221  valw *= 1.2;
222 
223  uint8_t n = g->traces_.size();
224  uint16_t w = this->width_;
225  uint16_t h = this->height_;
226  DirectionType dir = this->direction_;
227  ValuePositionType valpos = this->values_;
228  if (!this->font_value_) {
229  valpos = VALUE_POSITION_TYPE_NONE;
230  }
231  // Line sample always goes below text for compactness
232  this->yl_ = txth + (txth / 4) + lt / 2;
233 
234  if (dir == DIRECTION_TYPE_AUTO) {
235  dir = DIRECTION_TYPE_HORIZONTAL; // as default
236  if (h > 0) {
238  }
239  }
240 
241  if (valpos == VALUE_POSITION_TYPE_AUTO) {
242  // TODO: do something smarter?? - fit to w and h?
243  valpos = VALUE_POSITION_TYPE_BELOW;
244  }
245 
246  if (valpos == VALUE_POSITION_TYPE_BELOW) {
247  this->yv_ = txth + (txth / 4);
248  if (this->lines_)
249  this->yv_ += txth / 4 + lt;
250  } else if (valpos == VALUE_POSITION_TYPE_BESIDE) {
251  this->xv_ = (txtw + valw) / 2;
252  }
253 
254  // If width or height is specified we divide evenly within, else we do tight-fit
255  if (w == 0) {
256  this->x0_ = txtw / 2;
257  this->xs_ = txtw;
258  if (valpos == VALUE_POSITION_TYPE_BELOW) {
259  this->xs_ = std::max(txtw, valw);
260  ;
261  this->x0_ = this->xs_ / 2;
262  } else if (valpos == VALUE_POSITION_TYPE_BESIDE) {
263  this->xs_ = txtw + valw;
264  }
265  if (dir == DIRECTION_TYPE_VERTICAL) {
266  this->width_ = this->xs_;
267  } else {
268  this->width_ = this->xs_ * n;
269  }
270  } else {
271  this->xs_ = w / n;
272  this->x0_ = this->xs_ / 2;
273  }
274 
275  if (h == 0) {
276  this->ys_ = txth;
277  if (valpos == VALUE_POSITION_TYPE_BELOW) {
278  this->ys_ = txth + txth / 2 + valh;
279  if (this->lines_) {
280  this->ys_ += lt;
281  }
282  } else if (valpos == VALUE_POSITION_TYPE_BESIDE) {
283  if (this->lines_) {
284  this->ys_ = std::max(txth + txth / 4 + lt + txth / 4, valh + valh / 4);
285  } else {
286  this->ys_ = std::max(txth + txth / 4, valh + valh / 4);
287  }
288  this->height_ = this->ys_ * n;
289  }
290  if (dir == DIRECTION_TYPE_HORIZONTAL) {
291  this->height_ = this->ys_;
292  } else {
293  this->height_ = this->ys_ * n;
294  }
295  } else {
296  this->ys_ = h / n;
297  }
298 
299  if (dir == DIRECTION_TYPE_HORIZONTAL) {
300  this->ys_ = 0;
301  } else {
302  this->xs_ = 0;
303  }
304 }
305 
306 void Graph::draw_legend(display::Display *buff, uint16_t x_offset, uint16_t y_offset, Color color) {
307  if (!legend_)
308  return;
309 
311  if (this->border_) {
312  int w = legend_->width_;
313  int h = legend_->height_;
314  buff->horizontal_line(x_offset, y_offset, w, color);
315  buff->horizontal_line(x_offset, y_offset + h - 1, w, color);
316  buff->vertical_line(x_offset, y_offset, h, color);
317  buff->vertical_line(x_offset + w - 1, y_offset, h, color);
318  }
319 
320  int x = x_offset + legend_->x0_;
321  int y = y_offset;
322  for (auto *trace : traces_) {
323  std::string txtstr = trace->get_name();
324  ESP_LOGV(TAG, " %s", txtstr.c_str());
325 
326  buff->printf(x, y, legend_->font_label_, trace->get_line_color(), TextAlign::TOP_CENTER, "%s", txtstr.c_str());
327 
328  if (legend_->lines_) {
329  uint16_t thick = trace->get_line_thickness();
330  for (int i = 0; i < legend_->x0_ * 4 / 3; i++) {
331  uint8_t b = (i % (thick * LineType::PATTERN_LENGTH)) / thick;
332  if (((uint8_t) trace->get_line_type() & (1 << b)) == (1 << b)) {
333  buff->vertical_line(x - legend_->x0_ * 2 / 3 + i, y + legend_->yl_ - thick / 2, thick,
334  trace->get_line_color());
335  }
336  }
337  }
338 
339  if (legend_->values_ != VALUE_POSITION_TYPE_NONE) {
340  int xv = x + legend_->xv_;
341  int yv = y + legend_->yv_;
342  std::stringstream ss;
343  ss << std::fixed << std::setprecision(trace->sensor_->get_accuracy_decimals()) << trace->sensor_->get_state();
344  std::string valstr = ss.str();
345  if (legend_->units_) {
346  valstr += trace->sensor_->get_unit_of_measurement();
347  }
348  buff->printf(xv, yv, legend_->font_value_, trace->get_line_color(), TextAlign::TOP_CENTER, "%s", valstr.c_str());
349  ESP_LOGV(TAG, " value: %s", valstr.c_str());
350  }
351  x += legend_->xs_;
352  y += legend_->ys_;
353  }
354 }
355 
356 void Graph::setup() {
357  for (auto *trace : traces_) {
358  trace->init(this);
359  }
360 }
361 
363  for (auto *trace : traces_) {
364  ESP_LOGCONFIG(TAG, "Graph for sensor %s", trace->get_name().c_str());
365  }
366 }
367 
368 } // namespace graph
369 } // namespace esphome
void horizontal_line(int x, int y, int width, Color color=COLOR_ON)
Draw a horizontal line from the point [x,y] to [x+width,y] with the given color.
Definition: display.cpp:38
uint32_t get_width()
Definition: graph.h:159
void take_sample(float data)
Definition: graph.cpp:24
uint16_t x
Definition: tt21100.cpp:17
void printf(int x, int y, BaseFont *font, Color color, TextAlign align, const char *format,...) __attribute__((format(printf
Evaluate the printf-format format and print the result with the anchor point at [x,y] with font.
Definition: display.cpp:216
void setup() override
Definition: graph.cpp:356
uint32_t IRAM_ATTR HOT millis()
Definition: core.cpp:25
uint16_t y
Definition: tt21100.cpp:18
std::vector< GraphTrace * > traces_
Definition: graph.h:173
void dump_config() override
Definition: graph.cpp:362
void draw_legend(display::Display *buff, uint16_t x_offset, uint16_t y_offset, Color color)
Definition: graph.cpp:306
void init(int length)
Definition: graph.cpp:18
void init(Graph *g)
Determine the best coordinates of drawing text + lines.
Definition: graph.cpp:185
uint32_t get_duration()
Definition: graph.h:158
void vertical_line(int x, int y, int height, Color color=COLOR_ON)
Draw a vertical line from the point [x,y] to [x,y+width] with the given color.
Definition: display.cpp:43
ValuePositionType
Definition: graph.h:38
uint8_t h
Definition: bl0939.h:21
void init(Graph *g)
Definition: graph.cpp:52
uint16_t length
Definition: tt21100.cpp:12
void draw_pixel_at(int x, int y)
Set a single pixel at the specified coordinates to default color.
Definition: display.h:179
Implementation of SPI Controller mode.
Definition: a01nyub.cpp:7
void draw(display::Display *buff, uint16_t x_offset, uint16_t y_offset, Color color)
Definition: graph.cpp:59
bool state
Definition: fan.h:34