unitemp.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. #include "unitemp.h"
  2. #include "interfaces/SingleWireSensor.h"
  3. #include "Sensors.h"
  4. #include "./views/UnitempViews.h"
  5. #include <furi_hal_power.h>
  6. #include <m-string.h>
  7. //TODO: Реализовать ограничение на добавление датчиков если интерфейс недоступен
  8. //TODO: Исключить добавление датчиков DS18x2x с одинаковыми ID
  9. //TODO: Проверка корректности параметров датчика перед аллокацией
  10. //TODO: Не выкидывать датчик в ошибку при первом же неудачном опросе
  11. //TODO: Запускать преобразование DS18x2x разом, затем поочерёдно считывать
  12. //TODO: Запрет сохранения DS18x2x с пустым адресом
  13. /* Переменные */
  14. //Данные приложения
  15. Unitemp* app;
  16. void uintemp_celsiumToFarengate(Sensor* sensor) {
  17. sensor->temp = sensor->temp * (9.0 / 5.0) + 32;
  18. }
  19. bool unitemp_saveSettings(void) {
  20. //Выделение памяти для потока
  21. app->file_stream = file_stream_alloc(app->storage);
  22. //Переменная пути к файлу
  23. FuriString* filepath = furi_string_alloc();
  24. //Составление пути к файлу
  25. furi_string_printf(filepath, "%s/%s", APP_PATH_FOLDER, APP_FILENAME_SETTINGS);
  26. //Создание папки плагина
  27. storage_common_mkdir(app->storage, APP_PATH_FOLDER);
  28. //Открытие потока
  29. if(!file_stream_open(
  30. app->file_stream, furi_string_get_cstr(filepath), FSAM_READ_WRITE, FSOM_CREATE_ALWAYS)) {
  31. FURI_LOG_E(
  32. APP_NAME,
  33. "An error occurred while saving the settings file: %d\r\n",
  34. file_stream_get_error(app->file_stream));
  35. //Закрытие потока и освобождение памяти
  36. file_stream_close(app->file_stream);
  37. stream_free(app->file_stream);
  38. return false;
  39. }
  40. //Сохранение настроек
  41. stream_write_format(
  42. app->file_stream, "INFINITY_BACKLIGHT %d\n", app->settings.infinityBacklight);
  43. stream_write_format(app->file_stream, "UNIT %d\n", app->settings.unit);
  44. //Закрытие потока и освобождение памяти
  45. file_stream_close(app->file_stream);
  46. stream_free(app->file_stream);
  47. FURI_LOG_I(APP_NAME, "Settings have been successfully saved\r\n");
  48. return true;
  49. }
  50. bool unitemp_loadSettings(void) {
  51. FURI_LOG_D(APP_NAME, "Loading settings...");
  52. //Выделение памяти на поток
  53. app->file_stream = file_stream_alloc(app->storage);
  54. //Переменная пути к файлу
  55. FuriString* filepath = furi_string_alloc();
  56. //Составление пути к файлу
  57. furi_string_printf(filepath, "%s/%s", APP_PATH_FOLDER, APP_FILENAME_SETTINGS);
  58. //Открытие потока к файлу настроек
  59. if(!file_stream_open(
  60. app->file_stream, furi_string_get_cstr(filepath), FSAM_READ_WRITE, FSOM_OPEN_EXISTING)) {
  61. //Сохранение настроек по умолчанию в случае отсутствия файла
  62. if(file_stream_get_error(app->file_stream) == FSE_NOT_EXIST) {
  63. FURI_LOG_W(APP_NAME, "Missing settings file. Setting defaults and saving...\r\n");
  64. //Закрытие потока и освобождение памяти
  65. file_stream_close(app->file_stream);
  66. stream_free(app->file_stream);
  67. //Сохранение стандартного конфига
  68. unitemp_saveSettings();
  69. return false;
  70. } else {
  71. FURI_LOG_E(
  72. APP_NAME,
  73. "An error occurred while loading the settings file: %d. Standard values have been applied\r\n",
  74. file_stream_get_error(app->file_stream));
  75. //Закрытие потока и освобождение памяти
  76. file_stream_close(app->file_stream);
  77. stream_free(app->file_stream);
  78. return false;
  79. }
  80. }
  81. //Вычисление размера файла
  82. uint8_t file_size = stream_size(app->file_stream);
  83. FURI_LOG_D(APP_NAME, "Settings file size: %d bytes\r\n", file_size);
  84. //Если файл пустой, то:
  85. if(file_size == (uint8_t)0) {
  86. FURI_LOG_W(APP_NAME, "Settings file is empty\r\n");
  87. //Закрытие потока и освобождение памяти
  88. file_stream_close(app->file_stream);
  89. stream_free(app->file_stream);
  90. //Сохранение стандартного конфига
  91. unitemp_saveSettings();
  92. return false;
  93. }
  94. //Выделение памяти под загрузку файла
  95. uint8_t* file_buf = malloc(file_size);
  96. //Опустошение буфера файла
  97. memset(file_buf, 0, file_size);
  98. //Загрузка файла
  99. if(stream_read(app->file_stream, file_buf, file_size) != file_size) {
  100. //Выход при ошибке чтения
  101. FURI_LOG_E(APP_NAME, "Error reading settings file\r\n");
  102. //Закрытие потока и освобождение памяти
  103. file_stream_close(app->file_stream);
  104. stream_free(app->file_stream);
  105. free(file_buf);
  106. return false;
  107. }
  108. //Построчное чтение файла
  109. //Указатель на начало строки
  110. FuriString* file = furi_string_alloc_set_str((char*)file_buf);
  111. //Сколько байт до конца строки
  112. size_t line_end = 0;
  113. while(line_end != STRING_FAILURE) {
  114. char buff[20] = {0};
  115. sscanf(((char*)(file_buf + line_end)), "%s", buff);
  116. if(!strcmp(buff, "INFINITY_BACKLIGHT")) {
  117. //Чтение значения параметра
  118. int p = 0;
  119. sscanf(((char*)(file_buf + line_end)), "INFINITY_BACKLIGHT %d", &p);
  120. if(p == 0) {
  121. app->settings.infinityBacklight = false;
  122. } else {
  123. app->settings.infinityBacklight = true;
  124. }
  125. } else if(!strcmp(buff, "UNIT")) {
  126. //Чтение значения параметра
  127. int p = 0;
  128. sscanf(((char*)(file_buf + line_end)), "\nUNIT %d", &p);
  129. if(p == CELSIUS) {
  130. app->settings.unit = CELSIUS;
  131. } else {
  132. app->settings.unit = FAHRENHEIT;
  133. }
  134. } else {
  135. FURI_LOG_W(APP_NAME, "Unknown settings parameter: %s\r\n", buff);
  136. }
  137. //Вычисление конца строки
  138. line_end = furi_string_search_char(file, '\n', line_end + 1);
  139. }
  140. free(file_buf);
  141. file_stream_close(app->file_stream);
  142. stream_free(app->file_stream);
  143. FURI_LOG_I(APP_NAME, "Settings have been successfully loaded\r\n");
  144. return true;
  145. }
  146. /**
  147. * @brief Выделение места под переменные плагина
  148. *
  149. * @return true Если всё прошло успешно
  150. * @return false Если в процессе загрузки произошла ошибка
  151. */
  152. static bool unitemp_alloc(void) {
  153. //Выделение памяти под данные приложения
  154. app = malloc(sizeof(Unitemp));
  155. //Разрешение работы приложения
  156. app->processing = true;
  157. //Открытие хранилища (?)
  158. app->storage = furi_record_open(RECORD_STORAGE);
  159. //Уведомления
  160. app->notifications = furi_record_open(RECORD_NOTIFICATION);
  161. //Установка значений по умолчанию
  162. app->settings.infinityBacklight = true; //Подсветка горит всегда
  163. app->settings.unit = CELSIUS; //Единица измерения - градусы Цельсия
  164. app->gui = furi_record_open(RECORD_GUI);
  165. //Диспетчер окон
  166. app->view_dispatcher = view_dispatcher_alloc();
  167. unitemp_Summary_alloc();
  168. unitemp_MainMenu_alloc();
  169. unitemp_Settings_alloc();
  170. unitemp_SensorsList_alloc();
  171. unitemp_SensorEdit_alloc();
  172. unitemp_SensorNameEdit_alloc();
  173. view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen);
  174. return true;
  175. }
  176. /**
  177. * @brief Освыбождение памяти после работы приложения
  178. */
  179. static void unitemp_free(void) {
  180. unitemp_SensorNameEdit_free();
  181. unitemp_SensorEdit_free();
  182. unitemp_SensorsList_free();
  183. unitemp_Settings_free();
  184. unitemp_MainMenu_free();
  185. unitemp_Summary_free();
  186. view_dispatcher_free(app->view_dispatcher);
  187. furi_record_close(RECORD_GUI);
  188. //Очистка датчиков
  189. //Высвыбождение данных датчиков
  190. unitemp_sensors_free();
  191. //Закрытие уведомлений
  192. furi_record_close(RECORD_NOTIFICATION);
  193. //Закрытие хранилища
  194. furi_record_close(RECORD_STORAGE);
  195. //Удаление в самую последнюю очередь
  196. free(app);
  197. }
  198. /**
  199. * @brief Точка входа в приложение
  200. *
  201. * @return Код ошибки
  202. */
  203. int32_t unitemp_app() {
  204. //Выделение памяти под переменные
  205. //Выход если произошла ошибка
  206. if(unitemp_alloc() == false) {
  207. //Освобождение памяти
  208. unitemp_free();
  209. //Выход
  210. return 0;
  211. }
  212. //Загрузка настроек из SD-карты
  213. unitemp_loadSettings();
  214. //Применение настроек
  215. if(app->settings.infinityBacklight == true) {
  216. //Постоянное свечение подсветки
  217. notification_message(app->notifications, &sequence_display_backlight_enforce_on);
  218. }
  219. app->settings.lastOTGState = furi_hal_power_is_otg_enabled();
  220. //Загрузка датчиков из SD-карты
  221. unitemp_sensors_load();
  222. //Инициализация датчиков
  223. unitemp_sensors_init();
  224. unitemp_Summary_switch();
  225. while(app->processing) {
  226. if(app->sensors_ready) unitemp_sensors_updateValues();
  227. furi_delay_ms(100);
  228. }
  229. //Деинициализация датчиков
  230. unitemp_sensors_deInit();
  231. //Автоматическое управление подсветкой
  232. if(app->settings.infinityBacklight == true)
  233. notification_message(app->notifications, &sequence_display_backlight_enforce_auto);
  234. //Освобождение памяти
  235. unitemp_free();
  236. //Выход
  237. return 0;
  238. }