unleashed-firmware/applications/external/unitemp/unitemp.c
MX 35f95336ed
Heat index
by ClementGre
2023-06-25 00:38:26 +03:00

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;
}