Fork of the espurna firmware for `mhsw` switches
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

259 lines
9.1 KiB

6 years ago
6 years ago
7 years ago
  1. /*
  2. BUTTON MODULE
  3. Copyright (C) 2016-2018 by Xose Pérez <xose dot perez at gmail dot com>
  4. */
  5. // -----------------------------------------------------------------------------
  6. // BUTTON
  7. // -----------------------------------------------------------------------------
  8. #include <DebounceEvent.h>
  9. #include <vector>
  10. typedef struct {
  11. DebounceEvent * button;
  12. unsigned long actions;
  13. unsigned int relayID;
  14. } button_t;
  15. std::vector<button_t> _buttons;
  16. #if MQTT_SUPPORT
  17. void buttonMQTT(unsigned char id, uint8_t event) {
  18. if (id >= _buttons.size()) return;
  19. char payload[2];
  20. itoa(event, payload, 10);
  21. mqttSend(MQTT_TOPIC_BUTTON, id, payload, false, false); // 1st bool = force, 2nd = retain
  22. }
  23. #endif
  24. #if WEB_SUPPORT
  25. bool _buttonWebSocketOnReceive(const char * key, JsonVariant& value) {
  26. return (strncmp(key, "btn", 3) == 0);
  27. }
  28. #endif
  29. int buttonFromRelay(unsigned int relayID) {
  30. for (unsigned int i=0; i < _buttons.size(); i++) {
  31. if (_buttons[i].relayID == relayID) return i;
  32. }
  33. return -1;
  34. }
  35. bool buttonState(unsigned char id) {
  36. if (id >= _buttons.size()) return false;
  37. return _buttons[id].button->pressed();
  38. }
  39. unsigned char buttonAction(unsigned char id, unsigned char event) {
  40. if (id >= _buttons.size()) return BUTTON_MODE_NONE;
  41. unsigned long actions = _buttons[id].actions;
  42. if (event == BUTTON_EVENT_PRESSED) return (actions) & 0x0F;
  43. if (event == BUTTON_EVENT_CLICK) return (actions >> 4) & 0x0F;
  44. if (event == BUTTON_EVENT_DBLCLICK) return (actions >> 8) & 0x0F;
  45. if (event == BUTTON_EVENT_LNGCLICK) return (actions >> 12) & 0x0F;
  46. if (event == BUTTON_EVENT_LNGLNGCLICK) return (actions >> 16) & 0x0F;
  47. return BUTTON_MODE_NONE;
  48. }
  49. unsigned long buttonStore(unsigned long pressed, unsigned long click, unsigned long dblclick, unsigned long lngclick, unsigned long lnglngclick) {
  50. unsigned int value;
  51. value = pressed;
  52. value += click << 4;
  53. value += dblclick << 8;
  54. value += lngclick << 12;
  55. value += lnglngclick << 16;
  56. return value;
  57. }
  58. uint8_t mapEvent(uint8_t event, uint8_t count, uint16_t length) {
  59. if (event == EVENT_PRESSED) return BUTTON_EVENT_PRESSED;
  60. if (event == EVENT_CHANGED) return BUTTON_EVENT_CLICK;
  61. if (event == EVENT_RELEASED) {
  62. if (count == 1) {
  63. if (length > BUTTON_LNGLNGCLICK_DELAY) return BUTTON_EVENT_LNGLNGCLICK;
  64. if (length > BUTTON_LNGCLICK_DELAY) return BUTTON_EVENT_LNGCLICK;
  65. return BUTTON_EVENT_CLICK;
  66. }
  67. if (count == 2) return BUTTON_EVENT_DBLCLICK;
  68. }
  69. return BUTTON_EVENT_NONE;
  70. }
  71. void buttonEvent(unsigned int id, unsigned char event) {
  72. DEBUG_MSG_P(PSTR("[BUTTON] Button #%u event %u\n"), id, event);
  73. if (event == 0) return;
  74. #if MQTT_SUPPORT
  75. buttonMQTT(id, event);
  76. #endif
  77. unsigned char action = buttonAction(id, event);
  78. if (action == BUTTON_MODE_TOGGLE) {
  79. if (_buttons[id].relayID > 0) {
  80. relayToggle(_buttons[id].relayID - 1);
  81. }
  82. }
  83. if (action == BUTTON_MODE_ON) {
  84. if (_buttons[id].relayID > 0) {
  85. relayStatus(_buttons[id].relayID - 1, true);
  86. }
  87. }
  88. if (action == BUTTON_MODE_OFF) {
  89. if (_buttons[id].relayID > 0) {
  90. relayStatus(_buttons[id].relayID - 1, false);
  91. }
  92. }
  93. if (action == BUTTON_MODE_AP) createAP();
  94. if (action == BUTTON_MODE_RESET) {
  95. deferredReset(100, CUSTOM_RESET_HARDWARE);
  96. }
  97. if (action == BUTTON_MODE_FACTORY) {
  98. DEBUG_MSG_P(PSTR("\n\nFACTORY RESET\n\n"));
  99. resetSettings();
  100. deferredReset(100, CUSTOM_RESET_FACTORY);
  101. }
  102. }
  103. void buttonSetup() {
  104. #ifdef ITEAD_SONOFF_DUAL
  105. unsigned int actions = buttonStore(BUTTON_MODE_NONE, BUTTON_MODE_TOGGLE, BUTTON_MODE_NONE, BUTTON_MODE_NONE, BUTTON_MODE_NONE);
  106. _buttons.push_back({new DebounceEvent(0, BUTTON_PUSHBUTTON), actions, 1});
  107. _buttons.push_back({new DebounceEvent(0, BUTTON_PUSHBUTTON), actions, 2});
  108. _buttons.push_back({new DebounceEvent(0, BUTTON_PUSHBUTTON), actions, BUTTON3_RELAY});
  109. #else
  110. unsigned long btnDelay = getSetting("btnDelay", BUTTON_DBLCLICK_DELAY).toInt();
  111. #if BUTTON1_PIN != GPIO_NONE
  112. {
  113. unsigned int actions = buttonStore(BUTTON1_PRESS, BUTTON1_CLICK, BUTTON1_DBLCLICK, BUTTON1_LNGCLICK, BUTTON1_LNGLNGCLICK);
  114. _buttons.push_back({new DebounceEvent(BUTTON1_PIN, BUTTON1_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON1_RELAY});
  115. }
  116. #endif
  117. #if BUTTON2_PIN != GPIO_NONE
  118. {
  119. unsigned int actions = buttonStore(BUTTON2_PRESS, BUTTON2_CLICK, BUTTON2_DBLCLICK, BUTTON2_LNGCLICK, BUTTON2_LNGLNGCLICK);
  120. _buttons.push_back({new DebounceEvent(BUTTON2_PIN, BUTTON2_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON2_RELAY});
  121. }
  122. #endif
  123. #if BUTTON3_PIN != GPIO_NONE
  124. {
  125. unsigned int actions = buttonStore(BUTTON3_PRESS, BUTTON3_CLICK, BUTTON3_DBLCLICK, BUTTON3_LNGCLICK, BUTTON3_LNGLNGCLICK);
  126. _buttons.push_back({new DebounceEvent(BUTTON3_PIN, BUTTON3_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON3_RELAY});
  127. }
  128. #endif
  129. #if BUTTON4_PIN != GPIO_NONE
  130. {
  131. unsigned int actions = buttonStore(BUTTON4_PRESS, BUTTON4_CLICK, BUTTON4_DBLCLICK, BUTTON4_LNGCLICK, BUTTON4_LNGLNGCLICK);
  132. _buttons.push_back({new DebounceEvent(BUTTON4_PIN, BUTTON4_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON4_RELAY});
  133. }
  134. #endif
  135. #if BUTTON5_PIN != GPIO_NONE
  136. {
  137. unsigned int actions = buttonStore(BUTTON5_PRESS, BUTTON5_CLICK, BUTTON5_DBLCLICK, BUTTON5_LNGCLICK, BUTTON5_LNGLNGCLICK);
  138. _buttons.push_back({new DebounceEvent(BUTTON5_PIN, BUTTON5_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON5_RELAY});
  139. }
  140. #endif
  141. #if BUTTON6_PIN != GPIO_NONE
  142. {
  143. unsigned int actions = buttonStore(BUTTON6_PRESS, BUTTON6_CLICK, BUTTON6_DBLCLICK, BUTTON6_LNGCLICK, BUTTON6_LNGLNGCLICK);
  144. _buttons.push_back({new DebounceEvent(BUTTON6_PIN, BUTTON6_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON6_RELAY});
  145. }
  146. #endif
  147. #if BUTTON7_PIN != GPIO_NONE
  148. {
  149. unsigned int actions = buttonStore(BUTTON7_PRESS, BUTTON7_CLICK, BUTTON7_DBLCLICK, BUTTON7_LNGCLICK, BUTTON7_LNGLNGCLICK);
  150. _buttons.push_back({new DebounceEvent(BUTTON7_PIN, BUTTON7_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON7_RELAY});
  151. }
  152. #endif
  153. #if BUTTON8_PIN != GPIO_NONE
  154. {
  155. unsigned int actions = buttonStore(BUTTON8_PRESS, BUTTON8_CLICK, BUTTON8_DBLCLICK, BUTTON8_LNGCLICK, BUTTON8_LNGLNGCLICK);
  156. _buttons.push_back({new DebounceEvent(BUTTON8_PIN, BUTTON8_MODE, BUTTON_DEBOUNCE_DELAY, btnDelay), actions, BUTTON8_RELAY});
  157. }
  158. #endif
  159. #endif
  160. DEBUG_MSG_P(PSTR("[BUTTON] Number of buttons: %u\n"), _buttons.size());
  161. // Websocket Callbacks
  162. #if WEB_SUPPORT
  163. wsOnReceiveRegister(_buttonWebSocketOnReceive);
  164. #endif
  165. // Register loop
  166. espurnaRegisterLoop(buttonLoop);
  167. }
  168. void buttonLoop() {
  169. #ifdef ITEAD_SONOFF_DUAL
  170. if (Serial.available() >= 4) {
  171. if (Serial.read() == 0xA0) {
  172. if (Serial.read() == 0x04) {
  173. unsigned char value = Serial.read();
  174. if (Serial.read() == 0xA1) {
  175. // RELAYs and BUTTONs are synchonized in the SIL F330
  176. // The on-board BUTTON2 should toggle RELAY0 value
  177. // Since we are not passing back RELAY2 value
  178. // (in the relayStatus method) it will only be present
  179. // here if it has actually been pressed
  180. if ((value & 4) == 4) {
  181. buttonEvent(2, BUTTON_EVENT_CLICK);
  182. return;
  183. }
  184. // Otherwise check if any of the other two BUTTONs
  185. // (in the header) has been pressed, but we should
  186. // ensure that we only toggle one of them to avoid
  187. // the synchronization going mad
  188. // This loop is generic for any PSB-04 module
  189. for (unsigned int i=0; i<relayCount(); i++) {
  190. bool status = (value & (1 << i)) > 0;
  191. // Check if the status for that relay has changed
  192. if (relayStatus(i) != status) {
  193. buttonEvent(i, BUTTON_EVENT_CLICK);
  194. break;
  195. }
  196. }
  197. }
  198. }
  199. }
  200. }
  201. #else
  202. for (unsigned int i=0; i < _buttons.size(); i++) {
  203. if (unsigned char event = _buttons[i].button->loop()) {
  204. unsigned char count = _buttons[i].button->getEventCount();
  205. unsigned long length = _buttons[i].button->getEventLength();
  206. unsigned char mapped = mapEvent(event, count, length);
  207. buttonEvent(i, mapped);
  208. }
  209. }
  210. #endif
  211. }