2024-11-05 20:15:00 +08:00
|
|
|
|
#include "ml307_board.h"
|
2024-12-06 11:08:49 +08:00
|
|
|
|
|
2024-11-05 20:15:00 +08:00
|
|
|
|
#include "application.h"
|
2024-12-06 11:08:49 +08:00
|
|
|
|
#include "display.h"
|
2025-02-19 23:54:59 +08:00
|
|
|
|
#include "assets/lang_config.h"
|
2024-10-29 00:22:29 +08:00
|
|
|
|
|
|
|
|
|
|
#include <esp_log.h>
|
|
|
|
|
|
#include <esp_timer.h>
|
2025-08-29 09:04:23 +08:00
|
|
|
|
#include <font_awesome.h>
|
2025-02-04 14:24:51 +08:00
|
|
|
|
#include <opus_encoder.h>
|
2024-10-29 00:22:29 +08:00
|
|
|
|
|
|
|
|
|
|
static const char *TAG = "Ml307Board";
|
|
|
|
|
|
|
2025-07-17 20:18:21 +08:00
|
|
|
|
Ml307Board::Ml307Board(gpio_num_t tx_pin, gpio_num_t rx_pin, gpio_num_t dtr_pin) : tx_pin_(tx_pin), rx_pin_(rx_pin), dtr_pin_(dtr_pin) {
|
2024-10-29 00:22:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-02-04 14:24:51 +08:00
|
|
|
|
std::string Ml307Board::GetBoardType() {
|
|
|
|
|
|
return "ml307";
|
2024-10-29 00:22:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-30 06:58:29 +08:00
|
|
|
|
void Ml307Board::StartNetwork() {
|
2025-07-17 20:18:21 +08:00
|
|
|
|
auto& application = Application::GetInstance();
|
2024-11-06 06:18:56 +08:00
|
|
|
|
auto display = Board::GetInstance().GetDisplay();
|
2025-02-19 23:54:59 +08:00
|
|
|
|
display->SetStatus(Lang::Strings::DETECTING_MODULE);
|
2024-11-06 06:18:56 +08:00
|
|
|
|
|
2025-07-17 20:18:21 +08:00
|
|
|
|
while (true) {
|
|
|
|
|
|
modem_ = AtModem::Detect(tx_pin_, rx_pin_, dtr_pin_, 921600);
|
|
|
|
|
|
if (modem_ != nullptr) {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(1000));
|
|
|
|
|
|
}
|
2024-11-06 06:18:56 +08:00
|
|
|
|
|
2025-07-17 20:18:21 +08:00
|
|
|
|
modem_->OnNetworkStateChanged([this, &application](bool network_ready) {
|
|
|
|
|
|
if (network_ready) {
|
|
|
|
|
|
ESP_LOGI(TAG, "Network is ready");
|
|
|
|
|
|
} else {
|
|
|
|
|
|
ESP_LOGE(TAG, "Network is down");
|
|
|
|
|
|
auto device_state = application.GetDeviceState();
|
|
|
|
|
|
if (device_state == kDeviceStateListening || device_state == kDeviceStateSpeaking) {
|
|
|
|
|
|
application.Schedule([this, &application]() {
|
|
|
|
|
|
application.SetDeviceState(kDeviceStateIdle);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2024-11-06 06:18:56 +08:00
|
|
|
|
|
2025-07-17 20:18:21 +08:00
|
|
|
|
// Wait for network ready
|
2025-02-19 23:54:59 +08:00
|
|
|
|
display->SetStatus(Lang::Strings::REGISTERING_NETWORK);
|
2025-06-29 05:40:27 +08:00
|
|
|
|
while (true) {
|
2025-07-17 20:18:21 +08:00
|
|
|
|
auto result = modem_->WaitForNetworkReady();
|
|
|
|
|
|
if (result == NetworkStatus::ErrorInsertPin) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
application.Alert(Lang::Strings::ERROR, Lang::Strings::PIN_ERROR, "triangle_exclamation", Lang::Sounds::OGG_ERR_PIN);
|
2025-07-17 20:18:21 +08:00
|
|
|
|
} else if (result == NetworkStatus::ErrorRegistrationDenied) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
application.Alert(Lang::Strings::ERROR, Lang::Strings::REG_ERROR, "triangle_exclamation", Lang::Sounds::OGG_ERR_REG);
|
2025-06-29 05:40:27 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(10000));
|
2024-10-30 06:58:29 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-29 00:22:29 +08:00
|
|
|
|
// Print the ML307 modem information
|
2025-07-17 20:18:21 +08:00
|
|
|
|
std::string module_revision = modem_->GetModuleRevision();
|
|
|
|
|
|
std::string imei = modem_->GetImei();
|
|
|
|
|
|
std::string iccid = modem_->GetIccid();
|
|
|
|
|
|
ESP_LOGI(TAG, "ML307 Revision: %s", module_revision.c_str());
|
2024-10-29 00:22:29 +08:00
|
|
|
|
ESP_LOGI(TAG, "ML307 IMEI: %s", imei.c_str());
|
|
|
|
|
|
ESP_LOGI(TAG, "ML307 ICCID: %s", iccid.c_str());
|
2024-11-14 23:15:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-17 20:18:21 +08:00
|
|
|
|
NetworkInterface* Ml307Board::GetNetwork() {
|
|
|
|
|
|
return modem_.get();
|
2024-11-14 23:15:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-18 06:17:39 +08:00
|
|
|
|
const char* Ml307Board::GetNetworkStateIcon() {
|
2025-07-17 20:18:21 +08:00
|
|
|
|
if (modem_ == nullptr || !modem_->network_ready()) {
|
2024-11-18 06:17:39 +08:00
|
|
|
|
return FONT_AWESOME_SIGNAL_OFF;
|
|
|
|
|
|
}
|
2025-07-17 20:18:21 +08:00
|
|
|
|
int csq = modem_->GetCsq();
|
2024-11-18 06:17:39 +08:00
|
|
|
|
if (csq == -1) {
|
|
|
|
|
|
return FONT_AWESOME_SIGNAL_OFF;
|
2025-02-01 23:08:41 +08:00
|
|
|
|
} else if (csq >= 0 && csq <= 14) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
return FONT_AWESOME_SIGNAL_WEAK;
|
2024-11-18 06:17:39 +08:00
|
|
|
|
} else if (csq >= 15 && csq <= 19) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
return FONT_AWESOME_SIGNAL_FAIR;
|
2024-11-18 06:17:39 +08:00
|
|
|
|
} else if (csq >= 20 && csq <= 24) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
return FONT_AWESOME_SIGNAL_GOOD;
|
2024-11-18 06:17:39 +08:00
|
|
|
|
} else if (csq >= 25 && csq <= 31) {
|
2025-08-29 09:04:23 +08:00
|
|
|
|
return FONT_AWESOME_SIGNAL_STRONG;
|
2024-11-18 06:17:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ESP_LOGW(TAG, "Invalid CSQ: %d", csq);
|
|
|
|
|
|
return FONT_AWESOME_SIGNAL_OFF;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-11-03 05:54:15 +08:00
|
|
|
|
std::string Ml307Board::GetBoardJson() {
|
2024-10-29 00:22:29 +08:00
|
|
|
|
// Set the board type for OTA
|
2025-03-03 07:29:22 +08:00
|
|
|
|
std::string board_json = std::string("{\"type\":\"" BOARD_TYPE "\",");
|
|
|
|
|
|
board_json += "\"name\":\"" BOARD_NAME "\",";
|
2025-07-17 20:18:21 +08:00
|
|
|
|
board_json += "\"revision\":\"" + modem_->GetModuleRevision() + "\",";
|
|
|
|
|
|
board_json += "\"carrier\":\"" + modem_->GetCarrierName() + "\",";
|
|
|
|
|
|
board_json += "\"csq\":\"" + std::to_string(modem_->GetCsq()) + "\",";
|
|
|
|
|
|
board_json += "\"imei\":\"" + modem_->GetImei() + "\",";
|
|
|
|
|
|
board_json += "\"iccid\":\"" + modem_->GetIccid() + "\",";
|
|
|
|
|
|
board_json += "\"cereg\":" + modem_->GetRegistrationState().ToString() + "}";
|
2024-10-29 00:22:29 +08:00
|
|
|
|
return board_json;
|
|
|
|
|
|
}
|
2024-11-15 23:07:20 +08:00
|
|
|
|
|
|
|
|
|
|
void Ml307Board::SetPowerSaveMode(bool enabled) {
|
|
|
|
|
|
// TODO: Implement power save mode for ML307
|
|
|
|
|
|
}
|
2025-05-22 19:19:36 +08:00
|
|
|
|
|
|
|
|
|
|
std::string Ml307Board::GetDeviceStatusJson() {
|
|
|
|
|
|
/*
|
|
|
|
|
|
* 返回设备状态JSON
|
|
|
|
|
|
*
|
|
|
|
|
|
* 返回的JSON结构如下:
|
|
|
|
|
|
* {
|
|
|
|
|
|
* "audio_speaker": {
|
|
|
|
|
|
* "volume": 70
|
|
|
|
|
|
* },
|
|
|
|
|
|
* "screen": {
|
|
|
|
|
|
* "brightness": 100,
|
|
|
|
|
|
* "theme": "light"
|
|
|
|
|
|
* },
|
|
|
|
|
|
* "battery": {
|
|
|
|
|
|
* "level": 50,
|
|
|
|
|
|
* "charging": true
|
|
|
|
|
|
* },
|
|
|
|
|
|
* "network": {
|
|
|
|
|
|
* "type": "cellular",
|
|
|
|
|
|
* "carrier": "CHINA MOBILE",
|
|
|
|
|
|
* "csq": 10
|
|
|
|
|
|
* }
|
|
|
|
|
|
* }
|
|
|
|
|
|
*/
|
|
|
|
|
|
auto& board = Board::GetInstance();
|
|
|
|
|
|
auto root = cJSON_CreateObject();
|
|
|
|
|
|
|
|
|
|
|
|
// Audio speaker
|
|
|
|
|
|
auto audio_speaker = cJSON_CreateObject();
|
|
|
|
|
|
auto audio_codec = board.GetAudioCodec();
|
|
|
|
|
|
if (audio_codec) {
|
|
|
|
|
|
cJSON_AddNumberToObject(audio_speaker, "volume", audio_codec->output_volume());
|
|
|
|
|
|
}
|
|
|
|
|
|
cJSON_AddItemToObject(root, "audio_speaker", audio_speaker);
|
|
|
|
|
|
|
|
|
|
|
|
// Screen brightness
|
|
|
|
|
|
auto backlight = board.GetBacklight();
|
|
|
|
|
|
auto screen = cJSON_CreateObject();
|
|
|
|
|
|
if (backlight) {
|
|
|
|
|
|
cJSON_AddNumberToObject(screen, "brightness", backlight->brightness());
|
|
|
|
|
|
}
|
|
|
|
|
|
auto display = board.GetDisplay();
|
|
|
|
|
|
if (display && display->height() > 64) { // For LCD display only
|
2025-09-10 18:43:47 +08:00
|
|
|
|
auto theme = display->GetTheme();
|
|
|
|
|
|
if (theme != nullptr) {
|
|
|
|
|
|
cJSON_AddStringToObject(screen, "theme", theme->name().c_str());
|
|
|
|
|
|
}
|
2025-05-22 19:19:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
cJSON_AddItemToObject(root, "screen", screen);
|
|
|
|
|
|
|
|
|
|
|
|
// Battery
|
|
|
|
|
|
int battery_level = 0;
|
|
|
|
|
|
bool charging = false;
|
|
|
|
|
|
bool discharging = false;
|
|
|
|
|
|
if (board.GetBatteryLevel(battery_level, charging, discharging)) {
|
|
|
|
|
|
cJSON* battery = cJSON_CreateObject();
|
|
|
|
|
|
cJSON_AddNumberToObject(battery, "level", battery_level);
|
|
|
|
|
|
cJSON_AddBoolToObject(battery, "charging", charging);
|
|
|
|
|
|
cJSON_AddItemToObject(root, "battery", battery);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Network
|
|
|
|
|
|
auto network = cJSON_CreateObject();
|
|
|
|
|
|
cJSON_AddStringToObject(network, "type", "cellular");
|
2025-07-17 20:18:21 +08:00
|
|
|
|
cJSON_AddStringToObject(network, "carrier", modem_->GetCarrierName().c_str());
|
|
|
|
|
|
int csq = modem_->GetCsq();
|
2025-05-22 19:19:36 +08:00
|
|
|
|
if (csq == -1) {
|
|
|
|
|
|
cJSON_AddStringToObject(network, "signal", "unknown");
|
|
|
|
|
|
} else if (csq >= 0 && csq <= 14) {
|
|
|
|
|
|
cJSON_AddStringToObject(network, "signal", "very weak");
|
|
|
|
|
|
} else if (csq >= 15 && csq <= 19) {
|
|
|
|
|
|
cJSON_AddStringToObject(network, "signal", "weak");
|
|
|
|
|
|
} else if (csq >= 20 && csq <= 24) {
|
|
|
|
|
|
cJSON_AddStringToObject(network, "signal", "medium");
|
|
|
|
|
|
} else if (csq >= 25 && csq <= 31) {
|
|
|
|
|
|
cJSON_AddStringToObject(network, "signal", "strong");
|
|
|
|
|
|
}
|
|
|
|
|
|
cJSON_AddItemToObject(root, "network", network);
|
|
|
|
|
|
|
|
|
|
|
|
auto json_str = cJSON_PrintUnformatted(root);
|
|
|
|
|
|
std::string json(json_str);
|
|
|
|
|
|
cJSON_free(json_str);
|
|
|
|
|
|
cJSON_Delete(root);
|
|
|
|
|
|
return json;
|
|
|
|
|
|
}
|