ESPHome  2025.3.3
touchscreen.cpp
Go to the documentation of this file.
1 #include "touchscreen.h"
2 
3 #include "esphome/core/log.h"
4 
5 namespace esphome {
6 namespace touchscreen {
7 
8 static const char *const TAG = "touchscreen";
9 
11 
13  irq_pin->attach_interrupt(TouchscreenInterrupt::gpio_intr, &this->store_, type);
14  this->store_.init = true;
15  this->store_.touched = false;
16  ESP_LOGD(TAG, "Attach Touch Interupt");
17 }
18 
20  if (this->display_ != nullptr) {
21  this->display_width_ = this->display_->get_width();
22  this->display_height_ = this->display_->get_height();
23  }
25 }
26 
28  if (!this->store_.init) {
29  this->store_.touched = true;
30  } else {
31  // no need to poll if we have interrupts.
32  ESP_LOGW(TAG, "Touch Polling Stopped. You can safely remove the 'update_interval:' variable from the YAML file.");
33  this->stop_poller();
34  }
35 }
36 
38  if (this->store_.touched) {
39  ESP_LOGVV(TAG, "<< Do Touch loop >>");
40  this->first_touch_ = this->touches_.empty();
41  this->need_update_ = false;
42  this->is_touched_ = false;
43  this->skip_update_ = false;
44  for (auto &tp : this->touches_) {
45  if (tp.second.state == STATE_PRESSED || tp.second.state == STATE_UPDATED) {
46  tp.second.state |= STATE_RELEASING;
47  } else {
48  tp.second.state = STATE_RELEASED;
49  }
50  tp.second.x_prev = tp.second.x;
51  tp.second.y_prev = tp.second.y;
52  }
53  this->update_touches();
54  if (this->skip_update_) {
55  for (auto &tp : this->touches_) {
56  tp.second.state &= ~STATE_RELEASING;
57  }
58  } else {
59  this->store_.touched = false;
60  this->defer([this]() { this->send_touches_(); });
61  if (this->touch_timeout_ > 0) {
62  // Simulate a touch after <this->touch_timeout_> ms. This will reset any existing timeout operation.
63  // This is to detect touch release.
64  if (this->is_touched_) {
65  this->set_timeout(TAG, this->touch_timeout_, [this]() { this->store_.touched = true; });
66  } else {
67  this->cancel_timeout(TAG);
68  }
69  }
70  }
71  }
72 }
73 
74 void Touchscreen::add_raw_touch_position_(uint8_t id, int16_t x_raw, int16_t y_raw, int16_t z_raw) {
75  TouchPoint tp;
76  uint16_t x, y;
77  if (this->swap_x_y_) {
78  std::swap(x_raw, y_raw);
79  }
80  if (this->touches_.count(id) == 0) {
81  tp.state = STATE_PRESSED;
82  tp.id = id;
83  } else {
84  tp = this->touches_[id];
85  tp.state = STATE_UPDATED;
86  tp.y_prev = tp.y;
87  tp.x_prev = tp.x;
88  }
89  tp.x_raw = x_raw;
90  tp.y_raw = y_raw;
91  tp.z_raw = z_raw;
92  if (this->x_raw_max_ != this->x_raw_min_ and this->y_raw_max_ != this->y_raw_min_) {
93  x = this->normalize_(x_raw, this->x_raw_min_, this->x_raw_max_, this->invert_x_);
94  y = this->normalize_(y_raw, this->y_raw_min_, this->y_raw_max_, this->invert_y_);
95 
96  tp.x = (uint16_t) ((int) x * this->display_width_ / 0x1000);
97  tp.y = (uint16_t) ((int) y * this->display_height_ / 0x1000);
98  } else {
99  tp.state |= STATE_CALIBRATE;
100  }
101  if (tp.state == STATE_PRESSED) {
102  tp.x_org = tp.x;
103  tp.y_org = tp.y;
104  }
105 
106  this->touches_[id] = tp;
107 
108  this->is_touched_ = true;
109  if ((tp.x != tp.x_prev) || (tp.y != tp.y_prev)) {
110  this->need_update_ = true;
111  }
112 }
113 
115  TouchPoints_t touches;
116  ESP_LOGV(TAG, "Touch status: is_touched=%d, was_touched=%d", this->is_touched_, this->was_touched_);
117  for (auto tp : this->touches_) {
118  ESP_LOGV(TAG, "Touch status: %d/%d: raw:(%4d,%4d,%4d) calc:(%3d,%4d)", tp.second.id, tp.second.state,
119  tp.second.x_raw, tp.second.y_raw, tp.second.z_raw, tp.second.x, tp.second.y);
120  touches.push_back(tp.second);
121  }
122  if (this->need_update_ || (!this->is_touched_ && this->was_touched_)) {
123  this->update_trigger_.trigger(touches);
124  for (auto *listener : this->touch_listeners_) {
125  listener->update(touches);
126  }
127  }
128  if (!this->is_touched_) {
129  if (this->was_touched_) {
130  this->release_trigger_.trigger();
131  for (auto *listener : this->touch_listeners_)
132  listener->release();
133  this->touches_.clear();
134  }
135  } else {
136  if (this->first_touch_) {
137  TouchPoint tp = this->touches_.begin()->second;
138  this->touch_trigger_.trigger(tp, touches);
139  for (auto *listener : this->touch_listeners_) {
140  listener->touch(tp);
141  }
142  }
143  }
144  this->was_touched_ = this->is_touched_;
145 }
146 
147 int16_t Touchscreen::normalize_(int16_t val, int16_t min_val, int16_t max_val, bool inverted) {
148  int16_t ret;
149 
150  if (val <= min_val) {
151  ret = 0;
152  } else if (val >= max_val) {
153  ret = 0xfff;
154  } else {
155  ret = (int16_t) ((int) 0xfff * (val - min_val) / (max_val - min_val));
156  }
157 
158  ret = (inverted) ? 0xfff - ret : ret;
159 
160  return ret;
161 }
162 
163 } // namespace touchscreen
164 } // namespace esphome
std::vector< TouchPoint > TouchPoints_t
Definition: touchscreen.h:30
uint16_t x
Definition: tt21100.cpp:17
static void gpio_intr(TouchscreenInterrupt *store)
Definition: touchscreen.cpp:10
T id(T value)
Helper function to make id(var) known from lambdas work in custom components.
Definition: helpers.h:798
mopeka_std_values val[4]
uint16_t y
Definition: tt21100.cpp:18
const char *const TAG
Definition: spi.cpp:8
void attach_interrupt_(InternalGPIOPin *irq_pin, esphome::gpio::InterruptType type)
Call this function to send touch points to the on_touch listener and the binary_sensors.
Definition: touchscreen.cpp:12
int16_t normalize_(int16_t val, int16_t min_val, int16_t max_val, bool inverted=false)
uint8_t type
void call_setup() override
Definition: component.cpp:210
InterruptType
Definition: gpio.h:40
void add_raw_touch_position_(uint8_t id, int16_t x_raw, int16_t y_raw, int16_t z_raw=0)
Definition: touchscreen.cpp:74
Implementation of SPI Controller mode.
Definition: a01nyub.cpp:7
void attach_interrupt(void(*func)(T *), T *arg, gpio::InterruptType type) const
Definition: gpio.h:88
void swap(optional< T > &x, optional< T > &y) noexcept
Definition: optional.h:209