unitemp.c 12 KB

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