mirror of
https://github.com/Next-Flip/Momentum-Firmware.git
synced 2026-04-24 03:29:57 -07:00
285 lines
10 KiB
C
285 lines
10 KiB
C
#include "../subghz_i.h"
|
||
#include "subghz/types.h"
|
||
#include "../helpers/subghz_custom_event.h"
|
||
#include <lib/toolbox/value_index.h>
|
||
#include <machine/endian.h>
|
||
#include <toolbox/strint.h>
|
||
#include <lib/subghz/blocks/generic.h>
|
||
|
||
#define TAG "SubGhzSceneSignalSettings"
|
||
|
||
static uint32_t counter_mode = 0xff;
|
||
static uint32_t counter32 = 0x0;
|
||
static uint16_t counter16 = 0x0;
|
||
static uint8_t byte_count = 0;
|
||
static uint8_t* byte_ptr = NULL;
|
||
static FuriString* byte_input_text;
|
||
|
||
#define COUNTER_MODE_COUNT 7
|
||
static const char* const counter_mode_text[COUNTER_MODE_COUNT] = {
|
||
"System",
|
||
"Mode 1",
|
||
"Mode 2",
|
||
"Mode 3",
|
||
"Mode 4",
|
||
"Mode 5",
|
||
"Mode 6",
|
||
};
|
||
|
||
static const int32_t counter_mode_value[COUNTER_MODE_COUNT] = {
|
||
0,
|
||
1,
|
||
2,
|
||
3,
|
||
4,
|
||
5,
|
||
6,
|
||
};
|
||
|
||
typedef struct {
|
||
char* name;
|
||
uint8_t mode_count;
|
||
} Protocols;
|
||
|
||
// List of protocols names and appropriate CounterMode counts
|
||
static Protocols protocols[] = {
|
||
{"Nice FloR-S", 3},
|
||
{"CAME Atomo", 4},
|
||
{"Alutech AT-4N", 3},
|
||
{"KeeLoq", 7},
|
||
};
|
||
|
||
#define PROTOCOLS_COUNT (sizeof(protocols) / sizeof(Protocols));
|
||
|
||
void subghz_scene_signal_settings_counter_mode_changed(VariableItem* item) {
|
||
uint8_t index = variable_item_get_current_value_index(item);
|
||
variable_item_set_current_value_text(item, counter_mode_text[index]);
|
||
counter_mode = counter_mode_value[index];
|
||
}
|
||
|
||
void subghz_scene_signal_settings_byte_input_callback(void* context) {
|
||
SubGhz* subghz = context;
|
||
view_dispatcher_send_custom_event(subghz->view_dispatcher, SubGhzCustomEventByteInputDone);
|
||
}
|
||
|
||
void subghz_scene_signal_settings_variable_item_list_enter_callback(void* context, uint32_t index) {
|
||
SubGhz* subghz = context;
|
||
|
||
// when we click OK on "Edit counter" item
|
||
if(index == 1) {
|
||
furi_string_cat_printf(byte_input_text, "%i", subghz_block_generic_global.cnt_lenght_bit);
|
||
furi_string_cat_str(byte_input_text, "-bits counter in HEX");
|
||
|
||
// Setup byte_input view
|
||
ByteInput* byte_input = subghz->byte_input;
|
||
byte_input_set_header_text(byte_input, furi_string_get_cstr(byte_input_text));
|
||
|
||
byte_input_set_result_callback(
|
||
byte_input,
|
||
subghz_scene_signal_settings_byte_input_callback,
|
||
NULL,
|
||
subghz,
|
||
byte_ptr,
|
||
byte_count);
|
||
view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewIdByteInput);
|
||
}
|
||
}
|
||
|
||
void subghz_scene_signal_settings_on_enter(void* context) {
|
||
SubGhz* subghz = context;
|
||
|
||
// ### Counter mode section ###
|
||
|
||
// When we open saved file we do some check and fill up subghz->file_path.
|
||
// So now we use it to check is there CounterMode in file or not
|
||
const char* file_path = furi_string_get_cstr(subghz->file_path);
|
||
|
||
furi_assert(subghz);
|
||
furi_assert(file_path);
|
||
|
||
Storage* storage = furi_record_open(RECORD_STORAGE);
|
||
FlipperFormat* fff_data_file = flipper_format_file_alloc(storage);
|
||
FuriString* tmp_text = furi_string_alloc_set_str("");
|
||
|
||
uint32_t tmp_counter_mode = 0;
|
||
counter_mode = 0xff;
|
||
uint8_t mode_count = 1;
|
||
|
||
// Open file and check is it contains allowed protocols and CounterMode variable - if not then CounterMode will stay 0xff
|
||
// if file contain allowed protocol but not contain CounterMode value then setup default CounterMode value = 0 and available CounterMode count for this protocol
|
||
// if file contain CounterMode value then load it
|
||
if(!flipper_format_file_open_existing(fff_data_file, file_path)) {
|
||
FURI_LOG_E(TAG, "Error open file %s", file_path);
|
||
} else {
|
||
flipper_format_read_string(fff_data_file, "Protocol", tmp_text);
|
||
// compare available protocols names, load CounterMode value from file and setup variable_item_list values_count
|
||
for(uint8_t i = 0; i < PROTOCOLS_COUNT i++) {
|
||
if(!strcmp(furi_string_get_cstr(tmp_text), protocols[i].name)) {
|
||
mode_count = protocols[i].mode_count;
|
||
if(flipper_format_read_uint32(fff_data_file, "CounterMode", &tmp_counter_mode, 1)) {
|
||
counter_mode = (uint8_t)tmp_counter_mode;
|
||
} else {
|
||
counter_mode = 0;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
FURI_LOG_D(TAG, "Loaded CounterMode value %li", counter_mode);
|
||
|
||
flipper_format_file_close(fff_data_file);
|
||
flipper_format_free(fff_data_file);
|
||
furi_record_close(RECORD_STORAGE);
|
||
|
||
// ### Counter edit section ###
|
||
byte_input_text = furi_string_alloc_set_str("Enter ");
|
||
bool counter_not_available = true;
|
||
SubGhzProtocolDecoderBase* decoder = subghz_txrx_get_decoder(subghz->txrx);
|
||
|
||
// deserialaze and decode loaded sugbhz file and push data to subghz_block_generic_global variable
|
||
if(subghz_protocol_decoder_base_deserialize(decoder, subghz_txrx_get_fff_data(subghz->txrx)) ==
|
||
SubGhzProtocolStatusOk) {
|
||
subghz_protocol_decoder_base_get_string(decoder, tmp_text);
|
||
} else {
|
||
FURI_LOG_E(TAG, "Cant deserialize this subghz file");
|
||
}
|
||
|
||
if(!subghz_block_generic_global.cnt_is_available) {
|
||
counter_mode = 0xff;
|
||
FURI_LOG_D(TAG, "Counter mode and edit not available for this protocol");
|
||
} else {
|
||
counter_not_available = false;
|
||
|
||
// Check is there byte_count more than 2 hex bytes long or not
|
||
// To show hex value we must correct revert bytes for ByteInput view with __bswapХХ
|
||
if(subghz_block_generic_global.cnt_lenght_bit > 16) {
|
||
counter32 = subghz_block_generic_global.current_cnt;
|
||
furi_string_printf(tmp_text, "%lX", counter32);
|
||
counter32 = __bswap32(counter32);
|
||
byte_ptr = (uint8_t*)&counter32;
|
||
byte_count = 4;
|
||
} else {
|
||
counter16 = subghz_block_generic_global.current_cnt;
|
||
furi_string_printf(tmp_text, "%X", counter16);
|
||
counter16 = __bswap16(counter16);
|
||
byte_ptr = (uint8_t*)&counter16;
|
||
byte_count = 2;
|
||
}
|
||
}
|
||
|
||
furi_assert(byte_ptr);
|
||
furi_assert(byte_count > 0);
|
||
|
||
//Create and Enable/Disable variable_item_list depending on current values
|
||
VariableItemList* variable_item_list = subghz->variable_item_list;
|
||
int32_t value_index;
|
||
VariableItem* item;
|
||
|
||
variable_item_list_set_enter_callback(
|
||
variable_item_list,
|
||
subghz_scene_signal_settings_variable_item_list_enter_callback,
|
||
subghz);
|
||
|
||
item = variable_item_list_add(
|
||
variable_item_list,
|
||
"Counter Mode",
|
||
mode_count,
|
||
subghz_scene_signal_settings_counter_mode_changed,
|
||
subghz);
|
||
value_index = value_index_int32(counter_mode, counter_mode_value, mode_count);
|
||
|
||
variable_item_set_current_value_index(item, value_index);
|
||
variable_item_set_current_value_text(item, counter_mode_text[value_index]);
|
||
variable_item_set_locked(item, (counter_mode == 0xff), "Not available\nfor this\nprotocol !");
|
||
|
||
item = variable_item_list_add(variable_item_list, "Edit Counter", 1, NULL, subghz);
|
||
variable_item_set_current_value_index(item, 0);
|
||
variable_item_set_current_value_text(item, furi_string_get_cstr(tmp_text));
|
||
variable_item_set_locked(item, (counter_not_available), "Not available\nfor this\nprotocol !");
|
||
|
||
furi_string_free(tmp_text);
|
||
|
||
view_dispatcher_switch_to_view(subghz->view_dispatcher, SubGhzViewIdVariableItemList);
|
||
}
|
||
|
||
bool subghz_scene_signal_settings_on_event(void* context, SceneManagerEvent event) {
|
||
SubGhz* subghz = context;
|
||
|
||
if(event.type == SceneManagerEventTypeCustom) {
|
||
if(event.event == SubGhzCustomEventByteInputDone) {
|
||
switch(byte_count) {
|
||
case 2:
|
||
// set new cnt value and override_flag to global variable and call transmit to generate and save subghz signal
|
||
counter16 = __bswap16(counter16);
|
||
subghz_block_generic_global_counter_override_set(counter16);
|
||
subghz_tx_start(subghz, subghz_txrx_get_fff_data(subghz->txrx));
|
||
subghz_txrx_stop(subghz->txrx);
|
||
break;
|
||
case 4:
|
||
// the same for 32 bit Counter
|
||
counter32 = __bswap32(counter32);
|
||
subghz_block_generic_global_counter_override_set(counter32);
|
||
subghz_tx_start(subghz, subghz_txrx_get_fff_data(subghz->txrx));
|
||
subghz_txrx_stop(subghz->txrx);
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
scene_manager_previous_scene(subghz->scene_manager);
|
||
return true;
|
||
|
||
} else {
|
||
if(event.type == SceneManagerEventTypeBack) {
|
||
scene_manager_previous_scene(subghz->scene_manager);
|
||
return true;
|
||
}
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
void subghz_scene_signal_settings_on_exit(void* context) {
|
||
SubGhz* subghz = context;
|
||
const char* file_path = furi_string_get_cstr(subghz->file_path);
|
||
|
||
furi_assert(subghz);
|
||
furi_assert(file_path);
|
||
|
||
// if ConterMode was changed from 0xff then we must update or write new value to file
|
||
if(counter_mode != 0xff) {
|
||
Storage* storage = furi_record_open(RECORD_STORAGE);
|
||
FlipperFormat* fff_data_file = flipper_format_file_alloc(storage);
|
||
|
||
// check is the file available for update/insert CounterMode value
|
||
if(flipper_format_file_open_existing(fff_data_file, file_path)) {
|
||
if(flipper_format_insert_or_update_uint32(
|
||
fff_data_file, "CounterMode", &counter_mode, 1)) {
|
||
FURI_LOG_D(
|
||
TAG, "Successfully updated/inserted CounterMode value %li", counter_mode);
|
||
} else {
|
||
FURI_LOG_E(TAG, "Error update/insert CounterMode value");
|
||
}
|
||
} else {
|
||
FURI_LOG_E(TAG, "Error open file %s for writing", file_path);
|
||
}
|
||
|
||
flipper_format_file_close(fff_data_file);
|
||
flipper_format_free(fff_data_file);
|
||
furi_record_close(RECORD_STORAGE);
|
||
|
||
// we need to reload file after editing when we exit from Signal Settings menu.
|
||
if(subghz_key_load(subghz, file_path, false)) {
|
||
FURI_LOG_D(TAG, "Subghz file was successfully reloaded");
|
||
} else {
|
||
FURI_LOG_E(TAG, "Error reloading subghz file");
|
||
}
|
||
}
|
||
|
||
// Clear views
|
||
variable_item_list_set_selected_item(subghz->variable_item_list, 0);
|
||
variable_item_list_reset(subghz->variable_item_list);
|
||
byte_input_set_result_callback(subghz->byte_input, NULL, NULL, NULL, NULL, 0);
|
||
byte_input_set_header_text(subghz->byte_input, "");
|
||
furi_string_free(byte_input_text);
|
||
}
|