mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2025-01-02 10:17:10 +03:00
35f95336ed
by ClementGre
343 lines
13 KiB
C
343 lines
13 KiB
C
/*
|
|
Unitemp - Universal temperature reader
|
|
Copyright (C) 2022-2023 Victor Nikitchuk (https://github.com/quen0n)
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "unitemp.h"
|
|
#include "interfaces/SingleWireSensor.h"
|
|
#include "Sensors.h"
|
|
#include "./views/UnitempViews.h"
|
|
|
|
#include <furi_hal_power.h>
|
|
|
|
/* Переменные */
|
|
//Данные приложения
|
|
Unitemp* app;
|
|
|
|
void uintemp_celsiumToFarengate(Sensor* sensor) {
|
|
sensor->temp = sensor->temp * (9.0 / 5.0) + 32;
|
|
sensor->heat_index = sensor->heat_index * (9.0 / 5.0) + 32;
|
|
}
|
|
|
|
static float heat_index_consts[9] = {
|
|
-42.379f,
|
|
2.04901523f,
|
|
10.14333127f,
|
|
-0.22475541f,
|
|
-0.00683783f,
|
|
-0.05481717f,
|
|
0.00122874f,
|
|
0.00085282f,
|
|
-0.00000199f};
|
|
void unitemp_calculate_heat_index(Sensor* sensor) {
|
|
// temp should be in Celsius, heat index will be in Celsius
|
|
float temp = sensor->temp * (9.0 / 5.0) + 32.0f;
|
|
float hum = sensor->hum;
|
|
sensor->heat_index =
|
|
(heat_index_consts[0] + heat_index_consts[1] * temp + heat_index_consts[2] * hum +
|
|
heat_index_consts[3] * temp * hum + heat_index_consts[4] * temp * temp +
|
|
heat_index_consts[5] * hum * hum + heat_index_consts[6] * temp * temp * hum +
|
|
heat_index_consts[7] * temp * hum * hum + heat_index_consts[8] * temp * temp * hum * hum -
|
|
32.0f) *
|
|
(5.0 / 9.0);
|
|
}
|
|
void unitemp_pascalToMmHg(Sensor* sensor) {
|
|
sensor->pressure = sensor->pressure * 0.007500638;
|
|
}
|
|
void unitemp_pascalToKPa(Sensor* sensor) {
|
|
sensor->pressure = sensor->pressure / 1000.0f;
|
|
}
|
|
void unitemp_pascalToHPa(Sensor* sensor) {
|
|
sensor->pressure = sensor->pressure / 100.0f;
|
|
}
|
|
void unitemp_pascalToInHg(Sensor* sensor) {
|
|
sensor->pressure = sensor->pressure * 0.0002953007;
|
|
}
|
|
|
|
bool unitemp_saveSettings(void) {
|
|
//Выделение памяти для потока
|
|
app->file_stream = file_stream_alloc(app->storage);
|
|
|
|
//Переменная пути к файлу
|
|
FuriString* filepath = furi_string_alloc();
|
|
//Составление пути к файлу
|
|
furi_string_printf(filepath, "%s/%s", APP_PATH_FOLDER, APP_FILENAME_SETTINGS);
|
|
//Создание папки плагина
|
|
storage_common_mkdir(app->storage, APP_PATH_FOLDER);
|
|
//Открытие потока
|
|
if(!file_stream_open(
|
|
app->file_stream, furi_string_get_cstr(filepath), FSAM_READ_WRITE, FSOM_CREATE_ALWAYS)) {
|
|
FURI_LOG_E(
|
|
APP_NAME,
|
|
"An error occurred while saving the settings file: %d",
|
|
file_stream_get_error(app->file_stream));
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
return false;
|
|
}
|
|
|
|
//Сохранение настроек
|
|
stream_write_format(
|
|
app->file_stream, "INFINITY_BACKLIGHT %d\n", app->settings.infinityBacklight);
|
|
stream_write_format(app->file_stream, "TEMP_UNIT %d\n", app->settings.temp_unit);
|
|
stream_write_format(app->file_stream, "PRESSURE_UNIT %d\n", app->settings.pressure_unit);
|
|
stream_write_format(app->file_stream, "HEAT_INDEX %d\n", app->settings.heat_index);
|
|
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
|
|
FURI_LOG_I(APP_NAME, "Settings have been successfully saved");
|
|
return true;
|
|
}
|
|
|
|
bool unitemp_loadSettings(void) {
|
|
UNITEMP_DEBUG("Loading settings...");
|
|
|
|
//Выделение памяти на поток
|
|
app->file_stream = file_stream_alloc(app->storage);
|
|
|
|
//Переменная пути к файлу
|
|
FuriString* filepath = furi_string_alloc();
|
|
//Составление пути к файлу
|
|
furi_string_printf(filepath, "%s/%s", APP_PATH_FOLDER, APP_FILENAME_SETTINGS);
|
|
|
|
//Открытие потока к файлу настроек
|
|
if(!file_stream_open(
|
|
app->file_stream, furi_string_get_cstr(filepath), FSAM_READ_WRITE, FSOM_OPEN_EXISTING)) {
|
|
//Сохранение настроек по умолчанию в случае отсутствия файла
|
|
if(file_stream_get_error(app->file_stream) == FSE_NOT_EXIST) {
|
|
FURI_LOG_W(APP_NAME, "Missing settings file. Setting defaults and saving...");
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
//Сохранение стандартного конфига
|
|
unitemp_saveSettings();
|
|
return false;
|
|
} else {
|
|
FURI_LOG_E(
|
|
APP_NAME,
|
|
"An error occurred while loading the settings file: %d. Standard values have been applied",
|
|
file_stream_get_error(app->file_stream));
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
//Вычисление размера файла
|
|
uint8_t file_size = stream_size(app->file_stream);
|
|
//Если файл пустой, то:
|
|
if(file_size == (uint8_t)0) {
|
|
FURI_LOG_W(APP_NAME, "Settings file is empty");
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
//Сохранение стандартного конфига
|
|
unitemp_saveSettings();
|
|
return false;
|
|
}
|
|
//Выделение памяти под загрузку файла
|
|
uint8_t* file_buf = malloc(file_size);
|
|
//Опустошение буфера файла
|
|
memset(file_buf, 0, file_size);
|
|
//Загрузка файла
|
|
if(stream_read(app->file_stream, file_buf, file_size) != file_size) {
|
|
//Выход при ошибке чтения
|
|
FURI_LOG_E(APP_NAME, "Error reading settings file");
|
|
//Закрытие потока и освобождение памяти
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
free(file_buf);
|
|
return false;
|
|
}
|
|
//Построчное чтение файла
|
|
//Указатель на начало строки
|
|
FuriString* file = furi_string_alloc_set_str((char*)file_buf);
|
|
//Сколько байт до конца строки
|
|
size_t line_end = 0;
|
|
|
|
while(line_end != ((size_t)-1) && line_end != (size_t)(file_size - 1)) {
|
|
char buff[20] = {0};
|
|
sscanf(((char*)(file_buf + line_end)), "%s", buff);
|
|
|
|
if(!strcmp(buff, "INFINITY_BACKLIGHT")) {
|
|
//Чтение значения параметра
|
|
int p = 0;
|
|
sscanf(((char*)(file_buf + line_end)), "INFINITY_BACKLIGHT %d", &p);
|
|
app->settings.infinityBacklight = p;
|
|
} else if(!strcmp(buff, "TEMP_UNIT")) {
|
|
//Чтение значения параметра
|
|
int p = 0;
|
|
sscanf(((char*)(file_buf + line_end)), "\nTEMP_UNIT %d", &p);
|
|
app->settings.temp_unit = p;
|
|
} else if(!strcmp(buff, "PRESSURE_UNIT")) {
|
|
//Чтение значения параметра
|
|
int p = 0;
|
|
sscanf(((char*)(file_buf + line_end)), "\nPRESSURE_UNIT %d", &p);
|
|
app->settings.pressure_unit = p;
|
|
} else if(!strcmp(buff, "HEAT_INDEX")) {
|
|
//Чтение значения параметра
|
|
int p = 0;
|
|
sscanf(((char*)(file_buf + line_end)), "\nHEAT_INDEX %d", &p);
|
|
app->settings.heat_index = p;
|
|
} else {
|
|
FURI_LOG_W(APP_NAME, "Unknown settings parameter: %s", buff);
|
|
}
|
|
|
|
//Вычисление конца строки
|
|
line_end = furi_string_search_char(file, '\n', line_end + 1);
|
|
}
|
|
free(file_buf);
|
|
file_stream_close(app->file_stream);
|
|
stream_free(app->file_stream);
|
|
|
|
FURI_LOG_I(APP_NAME, "Settings have been successfully loaded");
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief Выделение места под переменные плагина
|
|
*
|
|
* @return true Если всё прошло успешно
|
|
* @return false Если в процессе загрузки произошла ошибка
|
|
*/
|
|
static bool unitemp_alloc(void) {
|
|
//Выделение памяти под данные приложения
|
|
app = malloc(sizeof(Unitemp));
|
|
//Разрешение работы приложения
|
|
app->processing = true;
|
|
|
|
//Открытие хранилища (?)
|
|
app->storage = furi_record_open(RECORD_STORAGE);
|
|
|
|
//Уведомления
|
|
app->notifications = furi_record_open(RECORD_NOTIFICATION);
|
|
|
|
//Установка значений по умолчанию
|
|
app->settings.infinityBacklight = true; //Подсветка горит всегда
|
|
app->settings.temp_unit = UT_TEMP_CELSIUS; //Единица измерения температуры - градусы Цельсия
|
|
app->settings.pressure_unit = UT_PRESSURE_MM_HG; //Единица измерения давления - мм рт. ст.
|
|
app->settings.heat_index = false;
|
|
|
|
app->gui = furi_record_open(RECORD_GUI);
|
|
//Диспетчер окон
|
|
app->view_dispatcher = view_dispatcher_alloc();
|
|
|
|
app->sensors = NULL;
|
|
|
|
app->buff = malloc(BUFF_SIZE);
|
|
|
|
unitemp_General_alloc();
|
|
|
|
unitemp_MainMenu_alloc();
|
|
unitemp_Settings_alloc();
|
|
unitemp_SensorsList_alloc();
|
|
unitemp_SensorEdit_alloc();
|
|
unitemp_SensorNameEdit_alloc();
|
|
unitemp_SensorActions_alloc();
|
|
unitemp_widgets_alloc();
|
|
|
|
//Всплывающее окно
|
|
app->popup = popup_alloc();
|
|
view_dispatcher_add_view(app->view_dispatcher, UnitempViewPopup, popup_get_view(app->popup));
|
|
|
|
view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen);
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* @brief Освыбождение памяти после работы приложения
|
|
*/
|
|
static void unitemp_free(void) {
|
|
popup_free(app->popup);
|
|
//Удаление вида после обработки
|
|
view_dispatcher_remove_view(app->view_dispatcher, UnitempViewPopup);
|
|
unitemp_widgets_free();
|
|
|
|
unitemp_SensorActions_free();
|
|
unitemp_SensorNameEdit_free();
|
|
unitemp_SensorEdit_free();
|
|
unitemp_SensorsList_free();
|
|
unitemp_Settings_free();
|
|
unitemp_MainMenu_free();
|
|
unitemp_General_free();
|
|
|
|
free(app->buff);
|
|
|
|
view_dispatcher_free(app->view_dispatcher);
|
|
furi_record_close(RECORD_GUI);
|
|
//Очистка датчиков
|
|
//Высвыбождение данных датчиков
|
|
unitemp_sensors_free();
|
|
free(app->sensors);
|
|
|
|
//Закрытие уведомлений
|
|
furi_record_close(RECORD_NOTIFICATION);
|
|
//Закрытие хранилища
|
|
furi_record_close(RECORD_STORAGE);
|
|
//Удаление в самую последнюю очередь
|
|
free(app);
|
|
}
|
|
|
|
/**
|
|
* @brief Точка входа в приложение
|
|
*
|
|
* @return Код ошибки
|
|
*/
|
|
int32_t unitemp_app() {
|
|
//Выделение памяти под переменные
|
|
//Выход если произошла ошибка
|
|
if(unitemp_alloc() == false) {
|
|
//Освобождение памяти
|
|
unitemp_free();
|
|
//Выход
|
|
return 0;
|
|
}
|
|
|
|
//Загрузка настроек из SD-карты
|
|
unitemp_loadSettings();
|
|
//Применение настроек
|
|
if(app->settings.infinityBacklight == true) {
|
|
//Постоянное свечение подсветки
|
|
notification_message(app->notifications, &sequence_display_backlight_enforce_on);
|
|
}
|
|
app->settings.lastOTGState = furi_hal_power_is_otg_enabled();
|
|
//Загрузка датчиков из SD-карты
|
|
unitemp_sensors_load();
|
|
//Инициализация датчиков
|
|
unitemp_sensors_init();
|
|
|
|
unitemp_General_switch();
|
|
|
|
while(app->processing) {
|
|
if(app->sensors_ready) unitemp_sensors_updateValues();
|
|
furi_delay_ms(100);
|
|
}
|
|
|
|
//Деинициализация датчиков
|
|
unitemp_sensors_deInit();
|
|
//Автоматическое управление подсветкой
|
|
if(app->settings.infinityBacklight == true)
|
|
notification_message(app->notifications, &sequence_display_backlight_enforce_auto);
|
|
//Освобождение памяти
|
|
unitemp_free();
|
|
//Выход
|
|
return 0;
|
|
}
|