unleashed-firmware/applications/plugins/unitemp/views/SensorEdit_view.c

374 lines
No EOL
15 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
Unitemp - Universal temperature reader
Copyright (C) 2022 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 "UnitempViews.h"
#include <gui/modules/variable_item_list.h>
#include "../interfaces/SingleWireSensor.h"
#include "../interfaces/OneWireSensor.h"
#include "../interfaces/I2CSensor.h"
//Текущий вид
static View* view;
//Список
static VariableItemList* variable_item_list;
//Текущий редактируемый датчик
static Sensor* editable_sensor;
//Изначальный GPIO датчика
static const GPIO* initial_gpio = NULL;
//Элемент списка - имя датчика
static VariableItem* sensor_name_item;
//Элемент списка - адрес датчика one wire
static VariableItem* onewire_addr_item;
//Элемент списка - адрес датчика one wire
static VariableItem* onewire_type_item;
//Элемент списка - смещение температуры
VariableItem* temp_offset_item;
#define OFFSET_BUFF_SIZE 5
//Буффер для текста смещения
static char* offset_buff;
extern uint8_t generalview_sensor_index;
#define VIEW_ID UnitempViewSensorEdit
bool _onewire_id_exist(uint8_t* id) {
if(id == NULL) return false;
for(uint8_t i = 0; i < unitemp_sensors_getActiveCount(); i++) {
if(unitemp_sensor_getActive(i)->type == &Dallas) {
if(unitemp_onewire_id_compare(
id, ((OneWireSensor*)(unitemp_sensor_getActive(i)->instance))->deviceID)) {
return true;
}
}
}
return false;
}
static void _onewire_scan(void) {
OneWireSensor* ow_sensor = editable_sensor->instance;
UNITEMP_DEBUG(
"devices on wire %d: %d", ow_sensor->bus->gpio->num, ow_sensor->bus->device_count);
//Сканирование шины one wire
unitemp_onewire_bus_init(ow_sensor->bus);
uint8_t* id = NULL;
do {
id = unitemp_onewire_bus_enum_next(ow_sensor->bus);
} while(_onewire_id_exist(id));
if(id == NULL) {
unitemp_onewire_bus_enum_init();
id = unitemp_onewire_bus_enum_next(ow_sensor->bus);
if(_onewire_id_exist(id)) {
do {
id = unitemp_onewire_bus_enum_next(ow_sensor->bus);
} while(_onewire_id_exist(id) && id != NULL);
}
if(id == NULL) {
memset(ow_sensor->deviceID, 0, 8);
ow_sensor->familyCode = 0;
unitemp_onewire_bus_deinit(ow_sensor->bus);
variable_item_set_current_value_text(onewire_addr_item, "empty");
variable_item_set_current_value_text(
onewire_type_item, unitemp_onewire_sensor_getModel(editable_sensor));
return;
}
}
unitemp_onewire_bus_deinit(ow_sensor->bus);
memcpy(ow_sensor->deviceID, id, 8);
ow_sensor->familyCode = id[0];
UNITEMP_DEBUG(
"Found sensor's ID: %02X%02X%02X%02X%02X%02X%02X%02X",
id[0],
id[1],
id[2],
id[3],
id[4],
id[5],
id[6],
id[7]);
if(ow_sensor->familyCode != 0) {
char id_buff[10];
snprintf(
id_buff,
10,
"%02X%02X%02X",
ow_sensor->deviceID[1],
ow_sensor->deviceID[2],
ow_sensor->deviceID[3]);
//А больше не лезет(
variable_item_set_current_value_text(onewire_addr_item, id_buff);
} else {
variable_item_set_current_value_text(onewire_addr_item, "empty");
}
variable_item_set_current_value_text(
onewire_type_item, unitemp_onewire_sensor_getModel(editable_sensor));
}
/**
* @brief Функция обработки нажатия кнопки "Назад"
*
* @param context Указатель на данные приложения
* @return ID вида в который нужно переключиться
*/
static uint32_t _exit_callback(void* context) {
UNUSED(context);
editable_sensor->status = UT_SENSORSTATUS_TIMEOUT;
if(!unitemp_sensor_isContains(editable_sensor)) unitemp_sensor_free(editable_sensor);
unitemp_sensors_reload();
//Возврат предыдущий вид
return UnitempViewGeneral;
}
/**
* @brief Функция обработки нажатия средней кнопки
*
* @param context Указатель на данные приложения
* @param index На каком элементе списка была нажата кнопка
*/
static void _enter_callback(void* context, uint32_t index) {
UNUSED(context);
//Смена имени
if(index == 0) {
unitemp_SensorNameEdit_switch(editable_sensor);
}
//Сохранение
if((index == 4 && editable_sensor->type->interface != &ONE_WIRE) ||
(index == 5 && editable_sensor->type->interface == &ONE_WIRE)) {
//Выход если датчик one wire не имеет ID
if(editable_sensor->type->interface == &ONE_WIRE &&
((OneWireSensor*)(editable_sensor->instance))->familyCode == 0) {
return;
}
if(initial_gpio != NULL) {
unitemp_gpio_unlock(initial_gpio);
initial_gpio = NULL;
}
editable_sensor->status = UT_SENSORSTATUS_TIMEOUT;
if(!unitemp_sensor_isContains(editable_sensor)) unitemp_sensors_add(editable_sensor);
unitemp_sensors_save();
unitemp_sensors_reload();
generalview_sensor_index = unitemp_sensors_getActiveCount() - 1;
unitemp_General_switch();
}
//Адрес устройства на шине one wire
if(index == 4 && editable_sensor->type->interface == &ONE_WIRE) {
_onewire_scan();
}
}
/**
* @brief Функция обработки изменения значения GPIO
*
* @param item Указатель на элемент списка
*/
static void _gpio_change_callback(VariableItem* item) {
uint8_t index = variable_item_get_current_value_index(item);
if(editable_sensor->type->interface == &SINGLE_WIRE) {
SingleWireSensor* instance = editable_sensor->instance;
instance->gpio =
unitemp_gpio_getAviablePort(editable_sensor->type->interface, index, initial_gpio);
variable_item_set_current_value_text(item, instance->gpio->name);
}
if(editable_sensor->type->interface == &ONE_WIRE) {
OneWireSensor* instance = editable_sensor->instance;
instance->bus->gpio =
unitemp_gpio_getAviablePort(editable_sensor->type->interface, index, NULL);
variable_item_set_current_value_text(item, instance->bus->gpio->name);
}
}
/**
* @brief Функция обработки изменения значения GPIO
*
* @param item Указатель на элемент списка
*/
static void _i2caddr_change_callback(VariableItem* item) {
uint8_t index = variable_item_get_current_value_index(item);
((I2CSensor*)editable_sensor->instance)->currentI2CAdr =
((I2CSensor*)editable_sensor->instance)->minI2CAdr + index * 2;
char buff[5];
snprintf(buff, 5, "0x%2X", ((I2CSensor*)editable_sensor->instance)->currentI2CAdr >> 1);
variable_item_set_current_value_text(item, buff);
}
/**
* @brief Функция обработки изменения значения имени датчика
*
* @param item Указатель на элемент списка
*/
static void _name_change_callback(VariableItem* item) {
variable_item_set_current_value_index(item, 0);
unitemp_SensorNameEdit_switch(editable_sensor);
}
/**
* @brief Функция обработки изменения значения адреса датчика one wire
*
* @param item Указатель на элемент списка
*/
static void _onwire_addr_change_callback(VariableItem* item) {
variable_item_set_current_value_index(item, 0);
_onewire_scan();
}
/**
* @brief Функция обработки изменения значения смещения температуры
*
* @param item Указатель на элемент списка
*/
static void _offset_change_callback(VariableItem* item) {
editable_sensor->temp_offset = variable_item_get_current_value_index(item) - 20;
snprintf(
offset_buff, OFFSET_BUFF_SIZE, "%+1.1f", (double)(editable_sensor->temp_offset / 10.0));
variable_item_set_current_value_text(item, offset_buff);
}
/**
* @brief Создание меню редактирования датчка
*/
void unitemp_SensorEdit_alloc(void) {
variable_item_list = variable_item_list_alloc();
//Сброс всех элементов меню
variable_item_list_reset(variable_item_list);
//Добавление колбека на нажатие средней кнопки
variable_item_list_set_enter_callback(variable_item_list, _enter_callback, app);
//Создание вида из списка
view = variable_item_list_get_view(variable_item_list);
//Добавление колбека на нажатие кнопки "Назад"
view_set_previous_callback(view, _exit_callback);
//Добавление вида в диспетчер
view_dispatcher_add_view(app->view_dispatcher, VIEW_ID, view);
offset_buff = malloc(OFFSET_BUFF_SIZE);
}
void unitemp_SensorEdit_switch(Sensor* sensor) {
editable_sensor = sensor;
editable_sensor->status = UT_SENSORSTATUS_INACTIVE;
//Сброс всех элементов меню
variable_item_list_reset(variable_item_list);
//Обнуление последнего выбранного пункта
variable_item_list_set_selected_item(variable_item_list, 0);
//Имя датчика
sensor_name_item = variable_item_list_add(
variable_item_list, "Name", strlen(sensor->name) > 7 ? 1 : 2, _name_change_callback, NULL);
variable_item_set_current_value_index(sensor_name_item, 0);
variable_item_set_current_value_text(sensor_name_item, sensor->name);
//Тип датчика (не редактируется)
onewire_type_item = variable_item_list_add(variable_item_list, "Type", 1, NULL, NULL);
variable_item_set_current_value_index(onewire_type_item, 0);
variable_item_set_current_value_text(
onewire_type_item,
(sensor->type->interface == &ONE_WIRE ? unitemp_onewire_sensor_getModel(editable_sensor) :
sensor->type->typename));
//Смещение температуры
temp_offset_item = variable_item_list_add(
variable_item_list, "Temp. offset", 41, _offset_change_callback, NULL);
variable_item_set_current_value_index(temp_offset_item, sensor->temp_offset + 20);
snprintf(
offset_buff, OFFSET_BUFF_SIZE, "%+1.1f", (double)(editable_sensor->temp_offset / 10.0));
variable_item_set_current_value_text(temp_offset_item, offset_buff);
//Порт подключения датчка (для one wire и single wire)
if(sensor->type->interface == &ONE_WIRE || sensor->type->interface == &SINGLE_WIRE) {
if(sensor->type->interface == &ONE_WIRE) {
initial_gpio = ((OneWireSensor*)editable_sensor->instance)->bus->gpio;
} else {
initial_gpio = ((SingleWireSensor*)editable_sensor->instance)->gpio;
}
uint8_t aviable_gpio_count =
unitemp_gpio_getAviablePortsCount(sensor->type->interface, initial_gpio);
VariableItem* item = variable_item_list_add(
variable_item_list, "GPIO", aviable_gpio_count, _gpio_change_callback, app);
uint8_t gpio_index = 0;
if(unitemp_sensor_isContains(editable_sensor)) {
for(uint8_t i = 0; i < aviable_gpio_count; i++) {
if(unitemp_gpio_getAviablePort(sensor->type->interface, i, initial_gpio) ==
initial_gpio) {
gpio_index = i;
break;
}
}
}
variable_item_set_current_value_index(item, gpio_index);
variable_item_set_current_value_text(
item,
unitemp_gpio_getAviablePort(sensor->type->interface, gpio_index, initial_gpio)->name);
}
//Адрес устройства на шине I2C (для датчиков I2C)
if(sensor->type->interface == &I2C) {
VariableItem* item = variable_item_list_add(
variable_item_list,
"I2C address",
(((I2CSensor*)sensor->instance)->maxI2CAdr >> 1) -
(((I2CSensor*)sensor->instance)->minI2CAdr >> 1) + 1,
_i2caddr_change_callback,
app);
snprintf(app->buff, 5, "0x%2X", ((I2CSensor*)sensor->instance)->currentI2CAdr >> 1);
variable_item_set_current_value_index(
item,
(((I2CSensor*)sensor->instance)->currentI2CAdr >> 1) -
(((I2CSensor*)sensor->instance)->minI2CAdr >> 1));
variable_item_set_current_value_text(item, app->buff);
}
//Адрес устройства на шине one wire (для датчиков one wire)
if(sensor->type->interface == &ONE_WIRE) {
onewire_addr_item = variable_item_list_add(
variable_item_list, "Address", 2, _onwire_addr_change_callback, NULL);
OneWireSensor* ow_sensor = sensor->instance;
if(ow_sensor->familyCode == 0) {
variable_item_set_current_value_text(onewire_addr_item, "Scan");
} else {
snprintf(
app->buff,
10,
"%02X%02X%02X",
ow_sensor->deviceID[1],
ow_sensor->deviceID[2],
ow_sensor->deviceID[3]);
variable_item_set_current_value_text(onewire_addr_item, app->buff);
}
}
variable_item_list_add(variable_item_list, "Save", 1, NULL, NULL);
view_dispatcher_switch_to_view(app->view_dispatcher, VIEW_ID);
}
void unitemp_SensorEdit_free(void) {
//Очистка списка элементов
variable_item_list_free(variable_item_list);
//Очистка вида
view_free(view);
//Удаление вида после обработки
view_dispatcher_remove_view(app->view_dispatcher, VIEW_ID);
free(offset_buff);
}