#include using namespace std::placeholders; // for `_1` etc NetworkSettingsService::NetworkSettingsService(AsyncWebServer * server, FS * fs, SecurityManager * securityManager) : _httpEndpoint(NetworkSettings::read, NetworkSettings::update, this, server, NETWORK_SETTINGS_SERVICE_PATH, securityManager) , _fsPersistence(NetworkSettings::read, NetworkSettings::update, this, fs, NETWORK_SETTINGS_FILE) , _lastConnectionAttempt(0) { addUpdateHandler([&](const String & originId) { reconfigureWiFiConnection(); }, false); } void NetworkSettingsService::begin() { // We want the device to come up in opmode=0 (WIFI_OFF), when erasing the flash this is not the default. // If needed, we save opmode=0 before disabling persistence so the device boots with WiFi disabled in the future. if (WiFi.getMode() != WIFI_OFF) { WiFi.mode(WIFI_OFF); } // Disable WiFi config persistance and auto reconnect WiFi.persistent(false); WiFi.setAutoReconnect(false); WiFi.mode(WIFI_MODE_MAX); WiFi.mode(WIFI_MODE_NULL); WiFi.setScanMethod(WIFI_ALL_CHANNEL_SCAN); // default is FAST_SCAN WiFi.setSortMethod(WIFI_CONNECT_AP_BY_SIGNAL); // is default, no need to set WiFi.onEvent(std::bind(&NetworkSettingsService::WiFiEvent, this, _1)); _fsPersistence.readFromFS(); reconfigureWiFiConnection(); } void NetworkSettingsService::reconfigureWiFiConnection() { // disconnect and de-configure wifi if (WiFi.disconnect(true)) { _stopping = true; } } void NetworkSettingsService::loop() { unsigned long currentMillis = millis(); if (!_lastConnectionAttempt || (uint32_t)(currentMillis - _lastConnectionAttempt) >= WIFI_RECONNECTION_DELAY) { _lastConnectionAttempt = currentMillis; manageSTA(); } } void NetworkSettingsService::manageSTA() { // Abort if already connected, or if we have no SSID if (WiFi.isConnected() || _state.ssid.length() == 0) { return; } // Connect or reconnect as required if ((WiFi.getMode() & WIFI_STA) == 0) { if (_state.staticIPConfig) { WiFi.config(_state.localIP, _state.gatewayIP, _state.subnetMask, _state.dnsIP1, _state.dnsIP2); // configure for static IP } WiFi.setHostname(_state.hostname.c_str()); // set hostname // www.esp32.com/viewtopic.php?t=12055 if (_state.bandwidth20) { esp_wifi_set_bandwidth((wifi_interface_t)ESP_IF_WIFI_STA, WIFI_BW_HT20); } else { esp_wifi_set_bandwidth((wifi_interface_t)ESP_IF_WIFI_STA, WIFI_BW_HT40); } if (_state.nosleep) { WiFi.setSleep(false); // turn off sleep - WIFI_PS_NONE } // attempt to connect to the network uint mac[6]; if (!_state.bssid.isEmpty() && sscanf(_state.bssid.c_str(), "%X:%X:%X:%X:%X:%X", &mac[0], &mac[1], &mac[2], &mac[3], &mac[4], &mac[5]) == 6) { uint8_t mac1[6]; for (uint8_t i = 0; i < 6; i++) { mac1[i] = (uint8_t)mac[i]; } WiFi.begin(_state.ssid.c_str(), _state.password.c_str(), 0, mac1); } else { WiFi.begin(_state.ssid.c_str(), _state.password.c_str()); } // set power after wifi is startet, fixed value for C3_V1 if (WiFi.isConnected()) { #ifdef BOARD_C3_MINI_V1 // v1 needs this value, see https://github.com/emsesp/EMS-ESP32/pull/620#discussion_r993173979 WiFi.setTxPower(WIFI_POWER_8_5dBm); // https://www.wemos.cc/en/latest/c3/c3_mini_1_0_0.html#about-wifi #else // esp_wifi_set_max_tx_power(_state.tx_power * 4); WiFi.setTxPower((wifi_power_t)(_state.tx_power * 4)); #endif } } else { // not connected but STA-mode active => disconnect reconfigureWiFiConnection(); } } // handles if wifi stopped void NetworkSettingsService::WiFiEvent(WiFiEvent_t event) { if (event == ARDUINO_EVENT_WIFI_STA_STOP) { if (_stopping) { _lastConnectionAttempt = 0; _stopping = false; } } if (!_stopping && (event == ARDUINO_EVENT_WIFI_STA_LOST_IP || event == ARDUINO_EVENT_WIFI_STA_DISCONNECTED)) { reconfigureWiFiConnection(); } }