mirror of
https://github.com/esphome/esphome.git
synced 2025-01-07 13:21:44 +01:00
Introduce a local coordinate drawing system to display and display_buffer
This commit is contained in:
parent
a6f1701902
commit
313cb2bff5
4 changed files with 85 additions and 0 deletions
|
@ -334,6 +334,35 @@ bool Display::clip(int x, int y) {
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Display::set_local_coordinates_relative_to_current(int x_offset, int y_offset) {
|
||||||
|
Point p(0, 0);
|
||||||
|
if (!this->local_coordinate_.empty()) {
|
||||||
|
p = this->local_coordinate_.back();
|
||||||
|
}
|
||||||
|
p.x += x_offset;
|
||||||
|
p.y += y_offset;
|
||||||
|
|
||||||
|
this->local_coordinate_.push_back(p);
|
||||||
|
|
||||||
|
}
|
||||||
|
void Display::pop_local_coordinates() {
|
||||||
|
if (this->local_coordinate_.empty()) {
|
||||||
|
ESP_LOGE(TAG, "pop_local_coordinates: No local coordinates set");
|
||||||
|
} else {
|
||||||
|
this->local_coordinate_.pop_back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Point Display::get_local_coordinates() {
|
||||||
|
Point p(0, 0);
|
||||||
|
if (!this->local_coordinate_.empty()) {
|
||||||
|
p = this->local_coordinate_.back();
|
||||||
|
}
|
||||||
|
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
bool Display::clamp_x_(int x, int w, int &min_x, int &max_x) {
|
bool Display::clamp_x_(int x, int w, int &min_x, int &max_x) {
|
||||||
min_x = std::max(x, 0);
|
min_x = std::max(x, 0);
|
||||||
max_x = std::min(x + w, this->get_width());
|
max_x = std::min(x + w, this->get_width());
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "rect.h"
|
#include "rect.h"
|
||||||
|
#include "point.h"
|
||||||
|
|
||||||
#include "esphome/core/color.h"
|
#include "esphome/core/color.h"
|
||||||
#include "esphome/core/automation.h"
|
#include "esphome/core/automation.h"
|
||||||
|
@ -480,6 +481,36 @@ class Display : public PollingComponent {
|
||||||
*/
|
*/
|
||||||
bool clip(int x, int y);
|
bool clip(int x, int y);
|
||||||
|
|
||||||
|
/** Changes the local coordinates to be relative to (x, y). After calling a pixel
|
||||||
|
* drawn at (10, 20) would be drawn to the screen at (x + 10, y + 20)
|
||||||
|
*
|
||||||
|
* @param[in] x: x coordinate as the new local. Absolute to the displays underlying 0
|
||||||
|
* @param[in] y: y coordinate as the new local. Absolute to the displays underlying 0
|
||||||
|
*/
|
||||||
|
void set_local_coordinate(int x, int y) { this->local_coordinate_.push_back(Point(x, y)); };
|
||||||
|
|
||||||
|
/** Changes the local coordinates to be to be (x_local + x_offset, y_local + y_offset)
|
||||||
|
* After calling a pixel drawn at (10, 20) would be drawn to the screen at
|
||||||
|
* (x_local + x_offset + 10, y_local + y_offset + 20)
|
||||||
|
*
|
||||||
|
* @param[in] x_offset: x offset from the current local. Relative to the local x
|
||||||
|
* @param[in] y_offset: y offset from the current local. Relative to the local y
|
||||||
|
*/
|
||||||
|
void set_local_coordinates_relative_to_current(int x_offset, int y_offset);
|
||||||
|
|
||||||
|
/** Removes the most recent local coordinate system from use
|
||||||
|
*/
|
||||||
|
void pop_local_coordinates();
|
||||||
|
|
||||||
|
/** Gets the current local coordinates in the displays absolute coordinate system
|
||||||
|
*/
|
||||||
|
Point get_local_coordinates();
|
||||||
|
|
||||||
|
/** Clears all the local coordinate systems and revers to the displays absolute coordinate
|
||||||
|
* system
|
||||||
|
*/
|
||||||
|
void clear_local_coordinates() { this->local_coordinate_.clear(); };
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool clamp_x_(int x, int w, int &min_x, int &max_x);
|
bool clamp_x_(int x, int w, int &min_x, int &max_x);
|
||||||
bool clamp_y_(int y, int h, int &min_y, int &max_y);
|
bool clamp_y_(int y, int h, int &min_y, int &max_y);
|
||||||
|
@ -495,6 +526,7 @@ class Display : public PollingComponent {
|
||||||
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_;
|
std::vector<Rect> clipping_rectangle_;
|
||||||
|
std::vector<Point> local_coordinate_;
|
||||||
};
|
};
|
||||||
|
|
||||||
class DisplayPage {
|
class DisplayPage {
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
|
|
||||||
#include "esphome/core/application.h"
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
#include "point.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace display {
|
namespace display {
|
||||||
|
@ -45,6 +46,11 @@ int DisplayBuffer::get_height() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void HOT DisplayBuffer::draw_pixel_at(int x, int y, Color color) {
|
void HOT DisplayBuffer::draw_pixel_at(int x, int y, Color color) {
|
||||||
|
// Convert (x, y) from the local coordinate system to the display absolute coordinate system
|
||||||
|
Point local_coord = this->get_local_coordinates();
|
||||||
|
x += local_coord.x;
|
||||||
|
y += local_coord.y;
|
||||||
|
|
||||||
if (!this->get_clipping().inside(x, y))
|
if (!this->get_clipping().inside(x, y))
|
||||||
return; // NOLINT
|
return; // NOLINT
|
||||||
|
|
||||||
|
|
18
esphome/components/display/point.h
Normal file
18
esphome/components/display/point.h
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace display {
|
||||||
|
|
||||||
|
class Point {
|
||||||
|
public:
|
||||||
|
int16_t x;
|
||||||
|
int16_t y;
|
||||||
|
|
||||||
|
Point() : x(VALUE_NO_SET), y(VALUE_NO_SET) {};
|
||||||
|
inline Point(int16_t x, int16_t y) ALWAYS_INLINE : x(x), y(y) {};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace display
|
||||||
|
} // namespace esphome
|
Loading…
Reference in a new issue