307 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			307 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "wled.h"
 | |
| 
 | |
| /*
 | |
|  * Physical IO
 | |
|  */
 | |
| 
 | |
| #define WLED_DEBOUNCE_THRESHOLD      50 // only consider button input of at least 50ms as valid (debouncing)
 | |
| #define WLED_LONG_PRESS             600 // long press if button is released after held for at least 600ms
 | |
| #define WLED_DOUBLE_PRESS           350 // double press if another press within 350ms after a short press
 | |
| #define WLED_LONG_REPEATED_ACTION   300 // how often a repeated action (e.g. dimming) is fired on long press on button IDs >0
 | |
| #define WLED_LONG_AP               5000 // how long button 0 needs to be held to activate WLED-AP
 | |
| #define WLED_LONG_FACTORY_RESET   10000 // how long button 0 needs to be held to trigger a factory reset
 | |
| 
 | |
| static const char _mqtt_topic_button[] PROGMEM = "%s/button/%d";  // optimize flash usage
 | |
| 
 | |
| void shortPressAction(uint8_t b)
 | |
| {
 | |
|   if (!macroButton[b]) {
 | |
|     switch (b) {
 | |
|       case 0: toggleOnOff(); stateUpdated(CALL_MODE_BUTTON); break;
 | |
|       case 1: ++effectCurrent %= strip.getModeCount(); colorUpdated(CALL_MODE_BUTTON); break;
 | |
|     }
 | |
|   } else {
 | |
|     applyPreset(macroButton[b], CALL_MODE_BUTTON_PRESET);
 | |
|   }
 | |
| 
 | |
|   // publish MQTT message
 | |
|   if (buttonPublishMqtt && WLED_MQTT_CONNECTED) {
 | |
|     char subuf[64];
 | |
|     sprintf_P(subuf, _mqtt_topic_button, mqttDeviceTopic, (int)b);
 | |
|     mqtt->publish(subuf, 0, false, "short");
 | |
|   }
 | |
| }
 | |
| 
 | |
| void longPressAction(uint8_t b)
 | |
| {
 | |
|   if (!macroLongPress[b]) {
 | |
|     switch (b) {
 | |
|       case 0: setRandomColor(col); colorUpdated(CALL_MODE_BUTTON); break;
 | |
|       case 1: bri += 8; stateUpdated(CALL_MODE_BUTTON); buttonPressedTime[b] = millis(); break; // repeatable action
 | |
|     }
 | |
|   } else {
 | |
|     applyPreset(macroLongPress[b], CALL_MODE_BUTTON_PRESET);
 | |
|   }
 | |
| 
 | |
|   // publish MQTT message
 | |
|   if (buttonPublishMqtt && WLED_MQTT_CONNECTED) {
 | |
|     char subuf[64];
 | |
|     sprintf_P(subuf, _mqtt_topic_button, mqttDeviceTopic, (int)b);
 | |
|     mqtt->publish(subuf, 0, false, "long");
 | |
|   }
 | |
| }
 | |
| 
 | |
| void doublePressAction(uint8_t b)
 | |
| {
 | |
|   if (!macroDoublePress[b]) {
 | |
|     switch (b) {
 | |
|       //case 0: toggleOnOff(); colorUpdated(CALL_MODE_BUTTON); break; //instant short press on button 0 if no macro set
 | |
|       case 1: ++effectPalette %= strip.getPaletteCount(); colorUpdated(CALL_MODE_BUTTON); break;
 | |
|     }
 | |
|   } else {
 | |
|     applyPreset(macroDoublePress[b], CALL_MODE_BUTTON_PRESET);
 | |
|   }
 | |
| 
 | |
|   // publish MQTT message
 | |
|   if (buttonPublishMqtt && WLED_MQTT_CONNECTED) {
 | |
|     char subuf[64];
 | |
|     sprintf_P(subuf, _mqtt_topic_button, mqttDeviceTopic, (int)b);
 | |
|     mqtt->publish(subuf, 0, false, "double");
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool isButtonPressed(uint8_t i)
 | |
| {
 | |
|   if (btnPin[i]<0) return false;
 | |
|   uint8_t pin = btnPin[i];
 | |
| 
 | |
|   switch (buttonType[i]) {
 | |
|     case BTN_TYPE_NONE:
 | |
|     case BTN_TYPE_RESERVED:
 | |
|       break;
 | |
|     case BTN_TYPE_PUSH:
 | |
|     case BTN_TYPE_SWITCH:
 | |
|       if (digitalRead(pin) == LOW) return true;
 | |
|       break;
 | |
|     case BTN_TYPE_PUSH_ACT_HIGH:
 | |
|     case BTN_TYPE_PIR_SENSOR:
 | |
|       if (digitalRead(pin) == HIGH) return true;
 | |
|       break;
 | |
|     case BTN_TYPE_TOUCH:
 | |
|       #if defined(ARDUINO_ARCH_ESP32) && !defined(CONFIG_IDF_TARGET_ESP32C3)
 | |
|       if (touchRead(pin) <= touchThreshold) return true;
 | |
|       #endif
 | |
|       break;
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| void handleSwitch(uint8_t b)
 | |
| {
 | |
|   // isButtonPressed() handles inverted/noninverted logic
 | |
|   if (buttonPressedBefore[b] != isButtonPressed(b)) {
 | |
|     buttonPressedTime[b] = millis();
 | |
|     buttonPressedBefore[b] = !buttonPressedBefore[b];
 | |
|   }
 | |
| 
 | |
|   if (buttonLongPressed[b] == buttonPressedBefore[b]) return;
 | |
|     
 | |
|   if (millis() - buttonPressedTime[b] > WLED_DEBOUNCE_THRESHOLD) { //fire edge event only after 50ms without change (debounce)
 | |
|     if (!buttonPressedBefore[b]) { // on -> off
 | |
|       if (macroButton[b]) applyPreset(macroButton[b], CALL_MODE_BUTTON_PRESET);
 | |
|       else { //turn on
 | |
|         if (!bri) {toggleOnOff(); stateUpdated(CALL_MODE_BUTTON);}
 | |
|       } 
 | |
|     } else {  // off -> on
 | |
|       if (macroLongPress[b]) applyPreset(macroLongPress[b], CALL_MODE_BUTTON_PRESET);
 | |
|       else { //turn off
 | |
|         if (bri) {toggleOnOff(); stateUpdated(CALL_MODE_BUTTON);}
 | |
|       } 
 | |
|     }
 | |
| 
 | |
|     // publish MQTT message
 | |
|     if (buttonPublishMqtt && WLED_MQTT_CONNECTED) {
 | |
|       char subuf[64];
 | |
|       if (buttonType[b] == BTN_TYPE_PIR_SENSOR) sprintf_P(subuf, PSTR("%s/motion/%d"), mqttDeviceTopic, (int)b);
 | |
|       else sprintf_P(subuf, _mqtt_topic_button, mqttDeviceTopic, (int)b);
 | |
|       mqtt->publish(subuf, 0, false, !buttonPressedBefore[b] ? "off" : "on");
 | |
|     }
 | |
| 
 | |
|     buttonLongPressed[b] = buttonPressedBefore[b]; //save the last "long term" switch state
 | |
|   }
 | |
| }
 | |
| 
 | |
| void handleAnalog(uint8_t b)
 | |
| {
 | |
|   static uint8_t oldRead[WLED_MAX_BUTTONS];
 | |
|   #ifdef ESP8266
 | |
|   uint16_t aRead = analogRead(A0) >> 2; // convert 10bit read to 8bit
 | |
|   #else
 | |
|   uint16_t aRead = analogRead(btnPin[b]) >> 4; // convert 12bit read to 8bit
 | |
|   #endif
 | |
| 
 | |
|   if (buttonType[b] == BTN_TYPE_ANALOG_INVERTED) aRead = 255 - aRead;
 | |
| 
 | |
|   // remove noise & reduce frequency of UI updates
 | |
|   aRead &= 0xFC;
 | |
| 
 | |
|   if (oldRead[b] == aRead) return;  // no change in reading
 | |
|   oldRead[b] = aRead;
 | |
| 
 | |
|   // if no macro for "short press" and "long press" is defined use brightness control
 | |
|   if (!macroButton[b] && !macroLongPress[b]) {
 | |
|     // if "double press" macro defines which option to change
 | |
|     if (macroDoublePress[b] >= 250) {
 | |
|       // global brightness
 | |
|       if (aRead == 0) {
 | |
|         briLast = bri;
 | |
|         bri = 0;
 | |
|       } else{
 | |
|         bri = aRead;
 | |
|       }
 | |
|     } else if (macroDoublePress[b] == 249) {
 | |
|       // effect speed
 | |
|       effectSpeed = aRead;
 | |
|     } else if (macroDoublePress[b] == 248) {
 | |
|       // effect intensity
 | |
|       effectIntensity = aRead;
 | |
|     } else if (macroDoublePress[b] == 247) {
 | |
|       // selected palette
 | |
|       effectPalette = map(aRead, 0, 252, 0, strip.getPaletteCount()-1);
 | |
|     } else if (macroDoublePress[b] == 200) {
 | |
|       // primary color, hue, full saturation
 | |
|       colorHStoRGB(aRead*256,255,col);
 | |
|     } else {
 | |
|       // otherwise use "double press" for segment selection
 | |
|       WS2812FX::Segment& seg = strip.getSegment(macroDoublePress[b]);
 | |
|       if (aRead == 0) {
 | |
|         seg.setOption(SEG_OPTION_ON, 0); // off
 | |
|       } else {
 | |
|         seg.setOpacity(aRead, macroDoublePress[b]);
 | |
|         seg.setOption(SEG_OPTION_ON, 1);
 | |
|       }
 | |
|       // this will notify clients of update (websockets,mqtt,etc)
 | |
|       updateInterfaces(CALL_MODE_BUTTON);
 | |
|     }
 | |
|   } else {
 | |
|     //TODO:
 | |
|     // we can either trigger a preset depending on the level (between short and long entries)
 | |
|     // or use it for RGBW direct control
 | |
|   }
 | |
|   colorUpdated(CALL_MODE_BUTTON);
 | |
| }
 | |
| 
 | |
| void handleButton()
 | |
| {
 | |
|   static unsigned long lastRead = 0UL;
 | |
|   bool analog = false;
 | |
| 
 | |
|   for (uint8_t b=0; b<WLED_MAX_BUTTONS; b++) {
 | |
|     #ifdef ESP8266
 | |
|     if ((btnPin[b]<0 && !(buttonType[b] == BTN_TYPE_ANALOG || buttonType[b] == BTN_TYPE_ANALOG_INVERTED)) || buttonType[b] == BTN_TYPE_NONE) continue;
 | |
|     #else
 | |
|     if (btnPin[b]<0 || buttonType[b] == BTN_TYPE_NONE) continue;
 | |
|     #endif
 | |
| 
 | |
|     if (usermods.handleButton(b)) continue; // did usermod handle buttons
 | |
| 
 | |
|     if ((buttonType[b] == BTN_TYPE_ANALOG || buttonType[b] == BTN_TYPE_ANALOG_INVERTED) && millis() - lastRead > 250) {   // button is not a button but a potentiometer
 | |
|       analog = true;
 | |
|       handleAnalog(b); continue;
 | |
|     }
 | |
| 
 | |
|     //button is not momentary, but switch. This is only suitable on pins whose on-boot state does not matter (NOT gpio0)
 | |
|     if (buttonType[b] == BTN_TYPE_SWITCH || buttonType[b] == BTN_TYPE_PIR_SENSOR) {
 | |
|       handleSwitch(b); continue;
 | |
|     }
 | |
| 
 | |
|     //momentary button logic
 | |
|     if (isButtonPressed(b)) { //pressed
 | |
| 
 | |
|       if (!buttonPressedBefore[b]) buttonPressedTime[b] = millis();
 | |
|       buttonPressedBefore[b] = true;
 | |
| 
 | |
|       if (millis() - buttonPressedTime[b] > WLED_LONG_PRESS) { //long press
 | |
|         if (!buttonLongPressed[b]) longPressAction(b);
 | |
|         else if (b) { //repeatable action (~3 times per s) on button > 0
 | |
|           longPressAction(b);
 | |
|           buttonPressedTime[b] = millis() - WLED_LONG_REPEATED_ACTION; //300ms
 | |
|         }
 | |
|         buttonLongPressed[b] = true;
 | |
|       }
 | |
| 
 | |
|     } else if (!isButtonPressed(b) && buttonPressedBefore[b]) { //released
 | |
| 
 | |
|       long dur = millis() - buttonPressedTime[b];
 | |
|       if (dur < WLED_DEBOUNCE_THRESHOLD) {buttonPressedBefore[b] = false; continue;} //too short "press", debounce
 | |
|       bool doublePress = buttonWaitTime[b]; //did we have a short press before?
 | |
|       buttonWaitTime[b] = 0;
 | |
| 
 | |
|       if (b == 0 && dur > WLED_LONG_AP) { // long press on button 0 (when released)
 | |
|         if (dur > WLED_LONG_FACTORY_RESET) { // factory reset if pressed > 10 seconds
 | |
|           WLED_FS.format();
 | |
|           clearEEPROM();
 | |
|           doReboot = true;
 | |
|         } else {
 | |
|           WLED::instance().initAP(true);
 | |
|         }
 | |
|       } else if (!buttonLongPressed[b]) { //short press
 | |
|         if (b != 1 && !macroDoublePress[b]) { //don't wait for double press on buttons without a default action if no double press macro set
 | |
|           shortPressAction(b);
 | |
|         } else { //double press if less than 350 ms between current press and previous short press release (buttonWaitTime!=0)
 | |
|           if (doublePress) {
 | |
|             doublePressAction(b);
 | |
|           } else {
 | |
|             buttonWaitTime[b] = millis();
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|       buttonPressedBefore[b] = false;
 | |
|       buttonLongPressed[b] = false;
 | |
|     }
 | |
| 
 | |
|     //if 350ms elapsed since last short press release it is a short press
 | |
|     if (buttonWaitTime[b] && millis() - buttonWaitTime[b] > WLED_DOUBLE_PRESS && !buttonPressedBefore[b]) {
 | |
|       buttonWaitTime[b] = 0;
 | |
|       shortPressAction(b);
 | |
|     }
 | |
|   }
 | |
|   if (analog) lastRead = millis();
 | |
| }
 | |
| 
 | |
| void handleIO()
 | |
| {
 | |
|   handleButton();
 | |
|   
 | |
|   //set relay when LEDs turn on
 | |
|   if (strip.getBrightness())
 | |
|   {
 | |
|     lastOnTime = millis();
 | |
|     if (offMode)
 | |
|     {
 | |
|       if (rlyPin>=0) {
 | |
|         pinMode(rlyPin, OUTPUT);
 | |
|         digitalWrite(rlyPin, rlyMde);
 | |
|       }
 | |
|       offMode = false;
 | |
|     }
 | |
|   } else if (millis() - lastOnTime > 600)
 | |
|   {
 | |
|     if (!offMode) {
 | |
|       #ifdef ESP8266
 | |
|       // turn off built-in LED if strip is turned off
 | |
|       // this will break digital bus so will need to be reinitialised on On
 | |
|       PinOwner ledPinOwner = pinManager.getPinOwner(LED_BUILTIN);
 | |
|       if (!strip.isOffRefreshRequired() && (ledPinOwner == PinOwner::None || ledPinOwner == PinOwner::BusDigital)) {
 | |
|         pinMode(LED_BUILTIN, OUTPUT);
 | |
|         digitalWrite(LED_BUILTIN, HIGH);
 | |
|       }
 | |
|       #endif
 | |
|       if (rlyPin>=0) {
 | |
|         pinMode(rlyPin, OUTPUT);
 | |
|         digitalWrite(rlyPin, !rlyMde);
 | |
|       }
 | |
|     }
 | |
|     offMode = true;
 | |
|   }
 | |
| } | 
