mirror of
https://github.com/meshtastic/firmware.git
synced 2026-04-07 00:43:18 -04:00
* Rename LED_PIN to LED_POWER, move handling out of main to dedicated module * Misc * Remove errant endif
224 lines
6.4 KiB
C++
224 lines
6.4 KiB
C++
#include "StatusLEDModule.h"
|
|
#include "MeshService.h"
|
|
#include "configuration.h"
|
|
#include <Arduino.h>
|
|
|
|
/*
|
|
StatusLEDModule manages the device's status LEDs, updating their states based on power and Bluetooth status.
|
|
It reflects charging, charged, discharging, and Bluetooth connection states using the appropriate LEDs.
|
|
*/
|
|
StatusLEDModule *statusLEDModule;
|
|
|
|
StatusLEDModule::StatusLEDModule() : concurrency::OSThread("StatusLEDModule")
|
|
{
|
|
bluetoothStatusObserver.observe(&bluetoothStatus->onNewStatus);
|
|
powerStatusObserver.observe(&powerStatus->onNewStatus);
|
|
#if !MESHTASTIC_EXCLUDE_INPUTBROKER
|
|
if (inputBroker)
|
|
inputObserver.observe(inputBroker);
|
|
#endif
|
|
}
|
|
|
|
int StatusLEDModule::handleStatusUpdate(const meshtastic::Status *arg)
|
|
{
|
|
switch (arg->getStatusType()) {
|
|
case STATUS_TYPE_POWER: {
|
|
meshtastic::PowerStatus *powerStatus = (meshtastic::PowerStatus *)arg;
|
|
if (powerStatus->getHasUSB() || powerStatus->getIsCharging()) {
|
|
power_state = charging;
|
|
if (powerStatus->getBatteryChargePercent() >= 100) {
|
|
power_state = charged;
|
|
}
|
|
} else {
|
|
if (powerStatus->getBatteryChargePercent() > 5) {
|
|
power_state = discharging;
|
|
} else {
|
|
power_state = critical;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case STATUS_TYPE_BLUETOOTH: {
|
|
meshtastic::BluetoothStatus *bluetoothStatus = (meshtastic::BluetoothStatus *)arg;
|
|
switch (bluetoothStatus->getConnectionState()) {
|
|
case meshtastic::BluetoothStatus::ConnectionState::DISCONNECTED: {
|
|
ble_state = unpaired;
|
|
PAIRING_LED_starttime = millis();
|
|
break;
|
|
}
|
|
case meshtastic::BluetoothStatus::ConnectionState::PAIRING: {
|
|
ble_state = pairing;
|
|
PAIRING_LED_starttime = millis();
|
|
break;
|
|
}
|
|
case meshtastic::BluetoothStatus::ConnectionState::CONNECTED: {
|
|
if (ble_state != connected) {
|
|
ble_state = connected;
|
|
PAIRING_LED_starttime = millis();
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
return 0;
|
|
};
|
|
#if !MESHTASTIC_EXCLUDE_INPUTBROKER
|
|
int StatusLEDModule::handleInputEvent(const InputEvent *event)
|
|
{
|
|
lastUserbuttonTime = millis();
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
int32_t StatusLEDModule::runOnce()
|
|
{
|
|
my_interval = 1000;
|
|
|
|
if (power_state == charging) {
|
|
#ifndef POWER_LED_HARDWARE_BLINKS_WHILE_CHARGING
|
|
CHARGE_LED_state = !CHARGE_LED_state;
|
|
#endif
|
|
} else if (power_state == charged) {
|
|
CHARGE_LED_state = LED_STATE_ON;
|
|
} else if (power_state == critical) {
|
|
if (POWER_LED_starttime + 30000 < millis() && !doing_fast_blink) {
|
|
doing_fast_blink = true;
|
|
POWER_LED_starttime = millis();
|
|
}
|
|
if (doing_fast_blink) {
|
|
PAIRING_LED_state = LED_STATE_OFF;
|
|
CHARGE_LED_state = !CHARGE_LED_state;
|
|
my_interval = 250;
|
|
if (POWER_LED_starttime + 2000 < millis()) {
|
|
doing_fast_blink = false;
|
|
}
|
|
} else {
|
|
CHARGE_LED_state = LED_STATE_OFF;
|
|
}
|
|
|
|
} else {
|
|
if (doing_fast_blink) {
|
|
CHARGE_LED_state = LED_STATE_OFF;
|
|
doing_fast_blink = false;
|
|
my_interval = 999;
|
|
} else {
|
|
CHARGE_LED_state = LED_STATE_ON;
|
|
doing_fast_blink = true;
|
|
my_interval = 1;
|
|
}
|
|
}
|
|
|
|
if (!config.bluetooth.enabled || PAIRING_LED_starttime + 30 * 1000 < millis() || doing_fast_blink) {
|
|
PAIRING_LED_state = LED_STATE_OFF;
|
|
} else if (ble_state == unpaired) {
|
|
if (slowTrack) {
|
|
PAIRING_LED_state = !PAIRING_LED_state;
|
|
slowTrack = false;
|
|
} else {
|
|
slowTrack = true;
|
|
}
|
|
} else if (ble_state == pairing) {
|
|
PAIRING_LED_state = !PAIRING_LED_state;
|
|
} else {
|
|
PAIRING_LED_state = LED_STATE_ON;
|
|
}
|
|
|
|
// Override if disabled in config
|
|
if (config.device.led_heartbeat_disabled) {
|
|
CHARGE_LED_state = LED_STATE_OFF;
|
|
}
|
|
#ifdef Battery_LED_1
|
|
bool chargeIndicatorLED1 = LED_STATE_OFF;
|
|
bool chargeIndicatorLED2 = LED_STATE_OFF;
|
|
bool chargeIndicatorLED3 = LED_STATE_OFF;
|
|
bool chargeIndicatorLED4 = LED_STATE_OFF;
|
|
if (lastUserbuttonTime + 10 * 1000 > millis() || CHARGE_LED_state == LED_STATE_ON) {
|
|
// should this be off at very low percentages?
|
|
chargeIndicatorLED1 = LED_STATE_ON;
|
|
if (powerStatus && powerStatus->getBatteryChargePercent() >= 25)
|
|
chargeIndicatorLED2 = LED_STATE_ON;
|
|
if (powerStatus && powerStatus->getBatteryChargePercent() >= 50)
|
|
chargeIndicatorLED3 = LED_STATE_ON;
|
|
if (powerStatus && powerStatus->getBatteryChargePercent() >= 75)
|
|
chargeIndicatorLED4 = LED_STATE_ON;
|
|
}
|
|
#endif
|
|
|
|
#if defined(HAS_PMU)
|
|
if (pmu_found && PMU) {
|
|
// blink the axp led
|
|
PMU->setChargingLedMode(CHARGE_LED_state ? XPOWERS_CHG_LED_ON : XPOWERS_CHG_LED_OFF);
|
|
}
|
|
#endif
|
|
|
|
#ifdef PCA_LED_POWER
|
|
io.digitalWrite(PCA_LED_POWER, CHARGE_LED_state);
|
|
#endif
|
|
#ifdef PCA_LED_ENABLE
|
|
io.digitalWrite(PCA_LED_ENABLE, CHARGE_LED_state);
|
|
#endif
|
|
#ifdef LED_POWER
|
|
digitalWrite(LED_POWER, CHARGE_LED_state);
|
|
#endif
|
|
#ifdef LED_PAIRING
|
|
digitalWrite(LED_PAIRING, PAIRING_LED_state);
|
|
#endif
|
|
|
|
#ifdef Battery_LED_1
|
|
digitalWrite(Battery_LED_1, chargeIndicatorLED1);
|
|
#endif
|
|
#ifdef Battery_LED_2
|
|
digitalWrite(Battery_LED_2, chargeIndicatorLED2);
|
|
#endif
|
|
#ifdef Battery_LED_3
|
|
digitalWrite(Battery_LED_3, chargeIndicatorLED3);
|
|
#endif
|
|
#ifdef Battery_LED_4
|
|
digitalWrite(Battery_LED_4, chargeIndicatorLED4);
|
|
#endif
|
|
|
|
return (my_interval);
|
|
}
|
|
|
|
void StatusLEDModule::setPowerLED(bool LEDon)
|
|
{
|
|
|
|
#if defined(HAS_PMU)
|
|
if (pmu_found && PMU) {
|
|
// blink the axp led
|
|
PMU->setChargingLedMode(LEDon ? XPOWERS_CHG_LED_ON : XPOWERS_CHG_LED_OFF);
|
|
}
|
|
#endif
|
|
if (LEDon)
|
|
LEDon = LED_STATE_ON;
|
|
else
|
|
LEDon = LED_STATE_OFF;
|
|
#ifdef PCA_LED_POWER
|
|
io.digitalWrite(PCA_LED_POWER, LEDon);
|
|
#endif
|
|
#ifdef PCA_LED_ENABLE
|
|
io.digitalWrite(PCA_LED_ENABLE, LEDon);
|
|
#endif
|
|
#ifdef LED_POWER
|
|
digitalWrite(LED_POWER, LEDon);
|
|
#endif
|
|
#ifdef LED_PAIRING
|
|
digitalWrite(LED_PAIRING, LEDon);
|
|
#endif
|
|
|
|
#ifdef Battery_LED_1
|
|
digitalWrite(Battery_LED_1, LEDon);
|
|
#endif
|
|
#ifdef Battery_LED_2
|
|
digitalWrite(Battery_LED_2, LEDon);
|
|
#endif
|
|
#ifdef Battery_LED_3
|
|
digitalWrite(Battery_LED_3, LEDon);
|
|
#endif
|
|
#ifdef Battery_LED_4
|
|
digitalWrite(Battery_LED_4, LEDon);
|
|
#endif
|
|
}
|