diff --git a/README.md b/README.md index 956557ec0ea..dbc530819c4 100644 --- a/README.md +++ b/README.md @@ -11,8 +11,9 @@ This app triggers restaurant pagers in a brute force manner, useful to test if d - Retekess TD174 ### Features -- Select range of stations (needs improving for full range) -- Select range of pagers (needs improving for full range) +- Send a Range of Signals +- Select range of stations +- Select range of pagers ## How to install on Flipper Zero - If you do not have one, download a firmware onto your PC via git
diff --git a/application.fam b/application.fam index a36b4ca43f9..01eb0d23379 100644 --- a/application.fam +++ b/application.fam @@ -7,7 +7,8 @@ App( fap_icon="icons/meal_pager_10px.png", fap_icon_assets="icons", fap_category="Sub-Ghz", - fap_version="1.0", + fap_version="1.1", + fap_libs=["assets"], fap_author="leedave", fap_weburl="https://github.com/leedave/flipper-zero-meal-pager", fap_description="This app triggers restaurant pagers in a brute force manner, useful to test if devices are still functional.", diff --git a/docs/changelog.md b/docs/changelog.md index 2c9adca3463..64f39d6e03d 100644 --- a/docs/changelog.md +++ b/docs/changelog.md @@ -1,3 +1,16 @@ +## v1.1 +- Created a new UI Input View as FW does not supply one for numbers +- New UI to Set First Station +- New UI to Set Last Station +- New UI to Set First Page +- New UI to Set Last Pager +- Removed Vibro/Sound settings as not used + +Known issues +- After setting last station, the settings view is weird. Some kindo of memory bug. But data remains correct. +- Extensive use can cause crashes, must be some memory leak left + + ## v1.0 - Added support for TD174 diff --git a/helpers/gui/int_input.c b/helpers/gui/int_input.c new file mode 100644 index 00000000000..b93ce3126af --- /dev/null +++ b/helpers/gui/int_input.c @@ -0,0 +1,390 @@ +#include "int_input.h" + +#include +#include +#include + +/** IntInput type */ +struct IntInput { + View* view; +}; + +typedef struct { + const char text; + const uint8_t x; + const uint8_t y; +} IntInputKey; + +typedef struct { + const char* header; + char* text_buffer; + size_t text_buffer_size; + bool clear_default_text; + + IntInputCallback callback; + void* callback_context; + + int8_t selected_row; + uint8_t selected_column; +} IntInputModel; + +static const uint8_t keyboard_origin_x = 7; +static const uint8_t keyboard_origin_y = 31; +static const uint8_t keyboard_row_count = 2; +static const uint8_t enter_symbol = '\r'; +static const uint8_t backspace_symbol = '\b'; + +static const IntInputKey keyboard_keys_row_1[] = { + {'0', 0, 12}, + {'1', 11, 12}, + {'2', 22, 12}, + {'3', 33, 12}, + {'4', 44, 12}, + {backspace_symbol, 103, 4}, +}; + +static const IntInputKey keyboard_keys_row_2[] = { + {'5', 0, 26}, + {'6', 11, 26}, + {'7', 22, 26}, + {'8', 33, 26}, + {'9', 44, 26}, + {enter_symbol, 95, 17}, +}; + +/** Get row size + * + * @param row_index Index of row + * + * @return uint8_t Row size + */ +static uint8_t int_input_get_row_size(uint8_t row_index) { + uint8_t row_size = 0; + + switch(row_index + 1) { + case 1: + row_size = COUNT_OF(keyboard_keys_row_1); + break; + case 2: + row_size = COUNT_OF(keyboard_keys_row_2); + break; + default: + furi_crash(); + } + + return row_size; +} + +/** Get row pointer + * + * @param row_index Index of row + * + * @return const IntInputKey* Row pointer + */ +static const IntInputKey* int_input_get_row(uint8_t row_index) { + const IntInputKey* row = NULL; + + switch(row_index + 1) { + case 1: + row = keyboard_keys_row_1; + break; + case 2: + row = keyboard_keys_row_2; + break; + default: + furi_crash(); + } + + return row; +} + +/** Draw input box (common view) + * + * @param canvas The canvas + * @param model The model + */ +static void int_input_draw_input(Canvas* canvas, IntInputModel* model) { + const uint8_t text_x = 8; + const uint8_t text_y = 25; + + elements_slightly_rounded_frame(canvas, 6, 14, 116, 15); + + const char* text = model->text_buffer; + canvas_draw_str(canvas, text_x, text_y, text); +} + +static void int_input_backspace_cb(IntInputModel* model) { + uint8_t text_length = model->clear_default_text ? 1 : strlen(model->text_buffer); + if(text_length > 0) { + model->text_buffer[text_length - 1] = 0; + } +} + +/** Handle up button + * + * @param model The model + */ +static void int_input_handle_up(IntInputModel* model) { + if(model->selected_row > 0) { + model->selected_row--; + } +} + +/** Handle down button + * + * @param model The model + */ +static void int_input_handle_down(IntInputModel* model) { + if(model->selected_row < keyboard_row_count - 1) { + model->selected_row += 1; + } +} + +/** Handle left button + * + * @param model The model + */ +static void int_input_handle_left(IntInputModel* model) { + if(model->selected_column > 0) { + model->selected_column--; + } else { + model->selected_column = int_input_get_row_size(model->selected_row) - 1; + } +} + +/** Handle right button + * + * @param model The model + */ +static void int_input_handle_right(IntInputModel* model) { + if(model->selected_column < int_input_get_row_size(model->selected_row) - 1) { + model->selected_column++; + } else { + model->selected_column = 0; + } +} + +/** Handle OK button + * + * @param model The model + */ +static void int_input_handle_ok(IntInputModel* model) { + char selected = int_input_get_row(model->selected_row)[model->selected_column].text; + size_t text_length = strlen(model->text_buffer); + if(selected == enter_symbol) { + model->callback(model->callback_context); + } else if(selected == backspace_symbol) { + int_input_backspace_cb(model); + } else { + if(model->clear_default_text) { + text_length = 0; + } + if(text_length < (model->text_buffer_size - 1)) { + model->text_buffer[text_length] = selected; + model->text_buffer[text_length + 1] = 0; + } + } + model->clear_default_text = false; +} + +/** Draw callback + * + * @param canvas The canvas + * @param _model The model + */ +static void int_input_view_draw_callback(Canvas* canvas, void* _model) { + IntInputModel* model = _model; + uint8_t text_length = model->text_buffer ? strlen(model->text_buffer) : 0; + UNUSED(text_length); + + canvas_clear(canvas); + canvas_set_color(canvas, ColorBlack); + + int_input_draw_input(canvas, model); + + canvas_set_font(canvas, FontSecondary); + canvas_draw_str(canvas, 2, 9, model->header); + canvas_set_font(canvas, FontKeyboard); + // Draw keyboard + for(uint8_t row = 0; row < keyboard_row_count; row++) { + const uint8_t column_count = int_input_get_row_size(row); + const IntInputKey* keys = int_input_get_row(row); + + for(size_t column = 0; column < column_count; column++) { + if(keys[column].text == enter_symbol) { + canvas_set_color(canvas, ColorBlack); + if(model->selected_row == row && model->selected_column == column) { + canvas_draw_icon( + canvas, + keyboard_origin_x + keys[column].x, + keyboard_origin_y + keys[column].y, + &I_KeySaveSelected_24x11); + } else { + canvas_draw_icon( + canvas, + keyboard_origin_x + keys[column].x, + keyboard_origin_y + keys[column].y, + &I_KeySave_24x11); + } + } else if(keys[column].text == backspace_symbol) { + canvas_set_color(canvas, ColorBlack); + if(model->selected_row == row && model->selected_column == column) { + canvas_draw_icon( + canvas, + keyboard_origin_x + keys[column].x, + keyboard_origin_y + keys[column].y, + &I_KeyBackspaceSelected_16x9); + } else { + canvas_draw_icon( + canvas, + keyboard_origin_x + keys[column].x, + keyboard_origin_y + keys[column].y, + &I_KeyBackspace_16x9); + } + } else { + if(model->selected_row == row && model->selected_column == column) { + canvas_set_color(canvas, ColorBlack); + canvas_draw_box( + canvas, + keyboard_origin_x + keys[column].x - 3, + keyboard_origin_y + keys[column].y - 10, + 11, + 13); + canvas_set_color(canvas, ColorWhite); + } else if(model->selected_row == -1 && row == 0 && model->selected_column == column) { + canvas_set_color(canvas, ColorBlack); + canvas_draw_frame( + canvas, + keyboard_origin_x + keys[column].x - 3, + keyboard_origin_y + keys[column].y - 10, + 11, + 13); + } else { + canvas_set_color(canvas, ColorBlack); + } + + canvas_draw_glyph( + canvas, + keyboard_origin_x + keys[column].x, + keyboard_origin_y + keys[column].y, + keys[column].text); + } + } + } +} + +/** Input callback + * + * @param event The event + * @param context The context + * + * @return true + * @return false + */ +static bool int_input_view_input_callback(InputEvent* event, void* context) { + IntInput* int_input = context; + furi_assert(int_input); + + bool consumed = false; + + // Fetch the model + IntInputModel* model = view_get_model(int_input->view); + + if(event->type == InputTypeShort || event->type == InputTypeLong || + event->type == InputTypeRepeat) { + consumed = true; + switch(event->key) { + case InputKeyLeft: + int_input_handle_left(model); + break; + case InputKeyRight: + int_input_handle_right(model); + break; + case InputKeyUp: + int_input_handle_up(model); + break; + case InputKeyDown: + int_input_handle_down(model); + break; + case InputKeyOk: + int_input_handle_ok(model); + break; + default: + consumed = false; + break; + } + } + + // commit view + view_commit_model(int_input->view, consumed); + + return consumed; +} + +void int_input_reset(IntInput* int_input) { + FURI_LOG_D("INT_INPUT", "Resetting Model"); + furi_assert(int_input); + with_view_model( + int_input->view, + IntInputModel * model, + { + model->header = ""; + model->selected_row = 0; + model->selected_column = 0; + model->clear_default_text = false; + model->text_buffer = ""; + model->text_buffer_size = 0; + model->callback = NULL; + model->callback_context = NULL; + }, + true); +} + +IntInput* int_input_alloc() { + IntInput* int_input = malloc(sizeof(IntInput)); + int_input->view = view_alloc(); + view_set_context(int_input->view, int_input); + view_allocate_model(int_input->view, ViewModelTypeLocking, sizeof(IntInputModel)); + view_set_draw_callback(int_input->view, int_input_view_draw_callback); + view_set_input_callback(int_input->view, int_input_view_input_callback); + + int_input_reset(int_input); + + return int_input; +} + +void int_input_free(IntInput* int_input) { + furi_assert(int_input); + view_free(int_input->view); + free(int_input); +} + +View* int_input_get_view(IntInput* int_input) { + furi_assert(int_input); + return int_input->view; +} + +void int_input_set_result_callback( + IntInput* int_input, + IntInputCallback callback, + void* callback_context, + char* text_buffer, + size_t text_buffer_size, + bool clear_default_text) { + with_view_model( + int_input->view, + IntInputModel * model, + { + model->callback = callback; + model->callback_context = callback_context; + model->text_buffer = text_buffer; + model->text_buffer_size = text_buffer_size; + model->clear_default_text = clear_default_text; + }, + true); +} + +void int_input_set_header_text(IntInput* int_input, const char* text) { + with_view_model( + int_input->view, IntInputModel * model, { model->header = text; }, true); +} diff --git a/helpers/gui/int_input.h b/helpers/gui/int_input.h new file mode 100644 index 00000000000..87d5bee5cad --- /dev/null +++ b/helpers/gui/int_input.h @@ -0,0 +1,71 @@ +/** + * @file int_input.h + * GUI: Integer keyboard view module API + */ + +#pragma once + +#include + +#ifdef __cplusplus +extern "C" { +#endif + +/** Int input anonymous structure */ +typedef struct IntInput IntInput; + +/** callback that is executed on save button press */ +typedef void (*IntInputCallback)(void* context); + +/** callback that is executed when byte buffer is changed */ +typedef void (*IntChangedCallback)(void* context); + +/** Allocate and initialize Int input. This Int input is used to enter Ints. + * + * @return IntInput instance pointer + */ +IntInput* int_input_alloc(); + +/** Deinitialize and free byte input + * + * @param int_input Int input instance + */ +void int_input_free(IntInput* int_input); + +/** Get byte input view + * + * @param int_input byte input instance + * + * @return View instance that can be used for embedding + */ +View* int_input_get_view(IntInput* int_input); + +/** Set byte input result callback + * + * @param int_input byte input instance + * @param input_callback input callback fn + * @param changed_callback changed callback fn + * @param callback_context callback context + * @param text_buffer buffer to use + * @param text_buffer_size buffer length + * @param clear_default_text clear previous entry + */ + +void int_input_set_result_callback( + IntInput* int_input, + IntInputCallback input_callback, + void* callback_context, + char* text_buffer, + size_t text_buffer_size, + bool clear_default_text); + +/** Set byte input header text + * + * @param int_input byte input instance + * @param text text to be shown + */ +void int_input_set_header_text(IntInput* int_input, const char* text); + +#ifdef __cplusplus +} +#endif diff --git a/helpers/meal_pager_custom_event.h b/helpers/meal_pager_custom_event.h index 4d0b73613c1..44d02f86739 100644 --- a/helpers/meal_pager_custom_event.h +++ b/helpers/meal_pager_custom_event.h @@ -23,6 +23,8 @@ typedef enum { Meal_PagerCustomEventViewTransmitterSendStart, Meal_PagerCustomEventViewTransmitterSendStop, Meal_PagerCustomEventViewTransmitterError, + Meal_PagerCustomerEventIntInput, + Meal_PagerCustomEventViewIntInputOk, } Meal_PagerCustomEvent; enum Meal_PagerCustomEventType { diff --git a/helpers/meal_pager_led.c b/helpers/meal_pager_led.c index 626ce71e137..68c449f8174 100644 --- a/helpers/meal_pager_led.c +++ b/helpers/meal_pager_led.c @@ -2,14 +2,18 @@ void meal_pager_blink_start_subghz(Meal_Pager* app) { furi_assert(app); - notification_message(app->notification, &sequence_blink_stop); - notification_message(app->notification, &sequence_blink_start_magenta); + if(app->led == 1) { + notification_message(app->notification, &sequence_blink_stop); + notification_message(app->notification, &sequence_blink_start_magenta); + } } void meal_pager_blink_start_compile(Meal_Pager* app) { furi_assert(app); - notification_message(app->notification, &sequence_blink_stop); - notification_message(app->notification, &sequence_blink_start_yellow); + if(app->led == 1) { + notification_message(app->notification, &sequence_blink_stop); + notification_message(app->notification, &sequence_blink_start_yellow); + } } void meal_pager_blink_stop(Meal_Pager* app) { diff --git a/helpers/meal_pager_storage.c b/helpers/meal_pager_storage.c index fbb4b5b56bf..83d9a1796bf 100644 --- a/helpers/meal_pager_storage.c +++ b/helpers/meal_pager_storage.c @@ -15,7 +15,11 @@ static void meal_pager_close_config_file(FlipperFormat* file) { flipper_format_free(file); } -bool meal_pager_save_subghz_buffer_file_start(void* context, FlipperFormat* ff, Storage* storage, char* frequency) { +bool meal_pager_save_subghz_buffer_file_start( + void* context, + FlipperFormat* ff, + Storage* storage, + char* frequency) { // SubGhz TXRX can only be loaded with files, makes sense as to save RAM Meal_Pager* app = context; UNUSED(app); @@ -53,12 +57,11 @@ bool meal_pager_save_subghz_buffer_file_start(void* context, FlipperFormat* ff, return success; } - success = - flipper_format_write_header_cstr( - ff, MEAL_PAGER_SUBGHZ_FILE_TYPE, MEAL_PAGER_SUBGHZ_FILE_VERSION) && - flipper_format_write_string_cstr(ff, "Frequency", frequency) && - flipper_format_write_string_cstr(ff, "Preset", MEAL_PAGER_SUBGHZ_FILE_PRESET) && - flipper_format_write_string_cstr(ff, "Protocol", MEAL_PAGER_SUBGHZ_FILE_Protocol); + success = flipper_format_write_header_cstr( + ff, MEAL_PAGER_SUBGHZ_FILE_TYPE, MEAL_PAGER_SUBGHZ_FILE_VERSION) && + flipper_format_write_string_cstr(ff, "Frequency", frequency) && + flipper_format_write_string_cstr(ff, "Preset", MEAL_PAGER_SUBGHZ_FILE_PRESET) && + flipper_format_write_string_cstr(ff, "Protocol", MEAL_PAGER_SUBGHZ_FILE_Protocol); return success; } @@ -178,6 +181,7 @@ void meal_pager_read_settings(void* context) { } flipper_format_read_uint32(fff_file, MEAL_PAGER_SETTINGS_KEY_PAGER_TYPE, &app->pager_type, 1); + meal_pager_set_max_values(app); flipper_format_read_uint32( fff_file, MEAL_PAGER_SETTINGS_KEY_FIRST_STATION, &app->first_station, 1); flipper_format_read_uint32( @@ -194,6 +198,27 @@ void meal_pager_read_settings(void* context) { flipper_format_rewind(fff_file); + furi_string_free(temp_str); + meal_pager_close_config_file(fff_file); meal_pager_close_storage(); +} + +void meal_pager_set_max_values(void* context) { + Meal_Pager* app = context; + switch(app->pager_type) { + case Meal_PagerPagerTypeT119: + case Meal_PagerPagerTypeTD165: + app->max_station = 8191; + app->max_pager = 999; + break; + case Meal_PagerPagerTypeTD174: + app->max_station = 8191; + app->max_pager = 10; + break; + case Meal_PagerPagerTypeTD157: + app->max_station = 1023; + app->max_pager = 999; + break; + } } \ No newline at end of file diff --git a/helpers/meal_pager_storage.h b/helpers/meal_pager_storage.h index 816b99b0230..748a961cbf4 100644 --- a/helpers/meal_pager_storage.h +++ b/helpers/meal_pager_storage.h @@ -29,9 +29,14 @@ #define MEAL_PAGER_SUBGHZ_FILE_PRESET "FuriHalSubGhzPresetOok650Async" #define MEAL_PAGER_SUBGHZ_FILE_Protocol "RAW" -bool meal_pager_save_subghz_buffer_file_start(void* context, FlipperFormat* ff, Storage* storage, char* frequency); +bool meal_pager_save_subghz_buffer_file_start( + void* context, + FlipperFormat* ff, + Storage* storage, + char* frequency); void meal_pager_save_subghz_buffer_stop(void* context, FlipperFormat* ff); void meal_pager_save_settings(void* context); -void meal_pager_read_settings(void* context); \ No newline at end of file +void meal_pager_read_settings(void* context); +void meal_pager_set_max_values(void* context); \ No newline at end of file diff --git a/helpers/retekess/meal_pager_retekess_t119.c b/helpers/retekess/meal_pager_retekess_t119.c index 79b9c70553d..fa6415537af 100644 --- a/helpers/retekess/meal_pager_retekess_t119.c +++ b/helpers/retekess/meal_pager_retekess_t119.c @@ -107,7 +107,8 @@ bool meal_pager_retekess_t119_generate_all(void* context) { Storage* storage = furi_record_open(RECORD_STORAGE); FlipperFormat* ff = flipper_format_file_alloc(storage); - bool success = meal_pager_save_subghz_buffer_file_start(app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); + bool success = meal_pager_save_subghz_buffer_file_start( + app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); if(!success) { FURI_LOG_D(TAG, "failed to save to buffer"); diff --git a/helpers/retekess/meal_pager_retekess_td157.c b/helpers/retekess/meal_pager_retekess_td157.c index d9c08cfc638..6467927ee0a 100644 --- a/helpers/retekess/meal_pager_retekess_td157.c +++ b/helpers/retekess/meal_pager_retekess_td157.c @@ -93,7 +93,8 @@ bool meal_pager_retekess_td157_generate_all(void* context) { Storage* storage = furi_record_open(RECORD_STORAGE); FlipperFormat* ff = flipper_format_file_alloc(storage); - bool success = meal_pager_save_subghz_buffer_file_start(app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); + bool success = meal_pager_save_subghz_buffer_file_start( + app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); if(!success) { FURI_LOG_D(TAG, "failed to save to buffer"); diff --git a/helpers/retekess/meal_pager_retekess_td165.c b/helpers/retekess/meal_pager_retekess_td165.c index f052f0e026c..0f81f281b96 100644 --- a/helpers/retekess/meal_pager_retekess_td165.c +++ b/helpers/retekess/meal_pager_retekess_td165.c @@ -95,7 +95,8 @@ bool meal_pager_retekess_td165_generate_all(void* context) { Storage* storage = furi_record_open(RECORD_STORAGE); FlipperFormat* ff = flipper_format_file_alloc(storage); - bool success = meal_pager_save_subghz_buffer_file_start(app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); + bool success = meal_pager_save_subghz_buffer_file_start( + app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_FREQUENCY); if(!success) { FURI_LOG_D(TAG, "failed to save to buffer"); diff --git a/helpers/retekess/meal_pager_retekess_td174.c b/helpers/retekess/meal_pager_retekess_td174.c index afc354b19c5..312bf900fdd 100644 --- a/helpers/retekess/meal_pager_retekess_td174.c +++ b/helpers/retekess/meal_pager_retekess_td174.c @@ -98,7 +98,8 @@ bool meal_pager_retekess_td174_generate_all(void* context) { Storage* storage = furi_record_open(RECORD_STORAGE); FlipperFormat* ff = flipper_format_file_alloc(storage); - bool success = meal_pager_save_subghz_buffer_file_start(app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_ALT_FREQUENCY); + bool success = meal_pager_save_subghz_buffer_file_start( + app, ff, storage, MEAL_PAGER_SUBGHZ_FILE_ALT_FREQUENCY); if(!success) { FURI_LOG_D(TAG, "failed to save to buffer"); diff --git a/meal_pager.c b/meal_pager.c index abbc715f642..5a66f3babd7 100644 --- a/meal_pager.c +++ b/meal_pager.c @@ -58,6 +58,10 @@ Meal_Pager* meal_pager_app_alloc() { app->stop_transmit = false; app->repeats = 1; app->repeats_char = "1"; + app->max_station = 8191; + app->max_pager = 999; + + snprintf(app->text_store[0], 32, "%lu", app->first_station); // Used for File Browser app->dialogs = furi_record_open(RECORD_DIALOGS); @@ -65,6 +69,9 @@ Meal_Pager* meal_pager_app_alloc() { app->subghz = subghz_alloc(); + // Custom made int keyboard + app->int_input = int_input_alloc(); + // Load configs meal_pager_read_settings(app); @@ -87,8 +94,17 @@ Meal_Pager* meal_pager_app_alloc() { Meal_PagerViewIdSettings, variable_item_list_get_view(app->variable_item_list)); + app->int_input = int_input_alloc(); + view_dispatcher_add_view( + app->view_dispatcher, Meal_PagerViewIdIntInput, int_input_get_view(app->int_input)); + //End Scene Additions + snprintf(app->text_store[0], 20, "%lu", app->first_station); + snprintf(app->text_store[1], 20, "%lu", app->last_station); + snprintf(app->text_store[2], 20, "%lu", app->first_pager); + snprintf(app->text_store[3], 20, "%lu", app->last_pager); + return app; } @@ -102,7 +118,9 @@ void meal_pager_app_free(Meal_Pager* app) { view_dispatcher_remove_view(app->view_dispatcher, Meal_PagerViewIdMenu); view_dispatcher_remove_view(app->view_dispatcher, Meal_PagerViewIdTransmit); view_dispatcher_remove_view(app->view_dispatcher, Meal_PagerViewIdSettings); + view_dispatcher_remove_view(app->view_dispatcher, Meal_PagerViewIdIntInput); submenu_free(app->submenu); + int_input_free(app->int_input); view_dispatcher_free(app->view_dispatcher); furi_record_close(RECORD_GUI); diff --git a/meal_pager.h b/meal_pager.h index ad7de499f7d..0e1fd95650b 100644 --- a/meal_pager.h +++ b/meal_pager.h @@ -1,2 +1,2 @@ #pragma once -#include "meal_pager_i.h" \ No newline at end of file +#include "meal_pager_i.h" diff --git a/meal_pager_i.h b/meal_pager_i.h index 659aca28a34..fab22e07bf8 100644 --- a/meal_pager_i.h +++ b/meal_pager_i.h @@ -19,6 +19,7 @@ #include "helpers/meal_pager_storage.h" #include "helpers/subghz/subghz_types.h" #include "helpers/subghz/subghz.h" +#include "helpers/gui/int_input.h" #define TAG "Meal_Pager" @@ -59,6 +60,11 @@ typedef struct { bool stop_transmit; uint32_t repeats; char* repeats_char; + IntInput* int_input; + char* text_buffer; + uint32_t max_station; + uint32_t max_pager; + char text_store[6][129]; } Meal_Pager; typedef enum { @@ -66,6 +72,7 @@ typedef enum { Meal_PagerViewIdMenu, Meal_PagerViewIdTransmit, Meal_PagerViewIdSettings, + Meal_PagerViewIdIntInput, } Meal_PagerViewId; typedef enum { diff --git a/scenes/meal_pager_scene_config.h b/scenes/meal_pager_scene_config.h index 9fb8580cfc2..7165afd98c6 100644 --- a/scenes/meal_pager_scene_config.h +++ b/scenes/meal_pager_scene_config.h @@ -1,4 +1,8 @@ ADD_SCENE(meal_pager, startscreen, Startscreen) ADD_SCENE(meal_pager, menu, Menu) ADD_SCENE(meal_pager, transmit, Transmit) -ADD_SCENE(meal_pager, settings, Settings) \ No newline at end of file +ADD_SCENE(meal_pager, settings, Settings) +ADD_SCENE(meal_pager, set_first_station, SetFirstStation) +ADD_SCENE(meal_pager, set_last_station, SetLastStation) +ADD_SCENE(meal_pager, set_first_pager, SetFirstPager) +ADD_SCENE(meal_pager, set_last_pager, SetLastPager) \ No newline at end of file diff --git a/scenes/meal_pager_scene_menu.c b/scenes/meal_pager_scene_menu.c index 3dbd7a26832..f30daac5a86 100644 --- a/scenes/meal_pager_scene_menu.c +++ b/scenes/meal_pager_scene_menu.c @@ -4,7 +4,10 @@ enum SubmenuIndex { SubmenuIndexTransmit = 10, - SubmenuIndexScene2, + SubmenuIndexSetFirstStation, + SubmenuIndexSetLastStation, + SubmenuIndexSetFirstPager, + SubmenuIndexSetLastPager, SubmenuIndexScene3, SubmenuIndexScene4, SubmenuIndexScene5, @@ -25,6 +28,30 @@ void meal_pager_scene_menu_on_enter(void* context) { SubmenuIndexTransmit, meal_pager_scene_menu_submenu_callback, app); + submenu_add_item( + app->submenu, + "Set First Station", + SubmenuIndexSetFirstStation, + meal_pager_scene_menu_submenu_callback, + app); + submenu_add_item( + app->submenu, + "Set Last Station", + SubmenuIndexSetLastStation, + meal_pager_scene_menu_submenu_callback, + app); + submenu_add_item( + app->submenu, + "Set First Pager", + SubmenuIndexSetFirstPager, + meal_pager_scene_menu_submenu_callback, + app); + submenu_add_item( + app->submenu, + "Set Last Pager", + SubmenuIndexSetLastPager, + meal_pager_scene_menu_submenu_callback, + app); submenu_add_item( app->submenu, "Settings", @@ -62,6 +89,26 @@ bool meal_pager_scene_menu_on_event(void* context, SceneManagerEvent event) { subghz_txrx_stop(app->subghz->txrx); FURI_LOG_D(TAG, "Stop Event from Menu"); return true; + } else if(event.event == SubmenuIndexSetFirstStation) { + scene_manager_set_scene_state( + app->scene_manager, Meal_PagerSceneSetFirstStation, SubmenuIndexSetFirstStation); + scene_manager_next_scene(app->scene_manager, Meal_PagerSceneSetFirstStation); + return true; + } else if(event.event == SubmenuIndexSetLastStation) { + scene_manager_set_scene_state( + app->scene_manager, Meal_PagerSceneSetLastStation, SubmenuIndexSetLastStation); + scene_manager_next_scene(app->scene_manager, Meal_PagerSceneSetLastStation); + return true; + } else if(event.event == SubmenuIndexSetFirstPager) { + scene_manager_set_scene_state( + app->scene_manager, Meal_PagerSceneSetFirstPager, SubmenuIndexSetFirstPager); + scene_manager_next_scene(app->scene_manager, Meal_PagerSceneSetFirstPager); + return true; + } else if(event.event == SubmenuIndexSetLastPager) { + scene_manager_set_scene_state( + app->scene_manager, Meal_PagerSceneSetLastPager, SubmenuIndexSetLastPager); + scene_manager_next_scene(app->scene_manager, Meal_PagerSceneSetLastPager); + return true; } } else if(event.type == SceneManagerEventTypeTick) { if(app->state_notifications == SubGhzNotificationStateTx) { diff --git a/scenes/meal_pager_scene_set_first_pager.c b/scenes/meal_pager_scene_set_first_pager.c new file mode 100644 index 00000000000..5ee2b0990b1 --- /dev/null +++ b/scenes/meal_pager_scene_set_first_pager.c @@ -0,0 +1,63 @@ +#include "../meal_pager_i.h" +#include "../helpers/meal_pager_custom_event.h" +#include "../helpers/meal_pager_led.h" +#include + +void meal_pager_set_first_pager_callback(void* context) { + furi_assert(context); + Meal_Pager* app = context; + view_dispatcher_send_custom_event(app->view_dispatcher, Meal_PagerCustomerEventIntInput); +} + +void meal_pager_scene_set_first_pager_on_enter(void* context) { + furi_assert(context); + Meal_Pager* app = context; + IntInput* int_input = app->int_input; + size_t enter_name_length = 5; + meal_pager_set_max_values(app); + char* str = "Set First Pager (0 - 999)"; + const char* constStr = str; + snprintf(str, 36, "Set First Pager (0 - %lu)", app->max_pager); + + int_input_set_header_text(int_input, constStr); + + int_input_set_result_callback( + int_input, + meal_pager_set_first_pager_callback, + context, + app->text_store[2], + enter_name_length, + false); + + view_dispatcher_switch_to_view(app->view_dispatcher, Meal_PagerViewIdIntInput); +} + +bool meal_pager_scene_set_first_pager_on_event(void* context, SceneManagerEvent event) { + Meal_Pager* app = context; + bool consumed = false; + + if(event.type == SceneManagerEventTypeBack) { + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeCustom) { + app->first_pager = atoi(app->text_store[2]); + app->first_pager_char = app->text_store[2]; + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeTick) { + if(app->state_notifications == SubGhzNotificationStateTx) { + app->state_notifications = SubGhzNotificationStateIDLE; + subghz_txrx_stop(app->subghz->txrx); + meal_pager_blink_stop(app); + meal_pager_transmit_model_set_sending(app->meal_pager_transmit, 0); + } + return true; + } + + return consumed; +} + +void meal_pager_scene_set_first_pager_on_exit(void* context) { + Meal_Pager* app = context; + UNUSED(app); +} \ No newline at end of file diff --git a/scenes/meal_pager_scene_set_first_station.c b/scenes/meal_pager_scene_set_first_station.c new file mode 100644 index 00000000000..0711823d050 --- /dev/null +++ b/scenes/meal_pager_scene_set_first_station.c @@ -0,0 +1,63 @@ +#include "../meal_pager_i.h" +#include "../helpers/meal_pager_custom_event.h" +#include "../helpers/meal_pager_led.h" +#include + +void meal_pager_set_first_station_callback(void* context) { + furi_assert(context); + Meal_Pager* app = context; + view_dispatcher_send_custom_event(app->view_dispatcher, Meal_PagerCustomerEventIntInput); +} + +void meal_pager_scene_set_first_station_on_enter(void* context) { + furi_assert(context); + Meal_Pager* app = context; + IntInput* int_input = app->int_input; + size_t enter_name_length = 5; + meal_pager_set_max_values(app); + char* str = "Set First Station (0 - 9999)"; + const char* constStr = str; + snprintf(str, 36, "Set First Station (0 - %lu)", app->max_station); + + int_input_set_header_text(int_input, constStr); + + int_input_set_result_callback( + int_input, + meal_pager_set_first_station_callback, + context, + app->text_store[0], + enter_name_length, + false); + + view_dispatcher_switch_to_view(app->view_dispatcher, Meal_PagerViewIdIntInput); +} + +bool meal_pager_scene_set_first_station_on_event(void* context, SceneManagerEvent event) { + Meal_Pager* app = context; + bool consumed = false; + + if(event.type == SceneManagerEventTypeBack) { + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeCustom) { + app->first_station = atoi(app->text_store[0]); + app->first_station_char = app->text_store[0]; + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeTick) { + if(app->state_notifications == SubGhzNotificationStateTx) { + app->state_notifications = SubGhzNotificationStateIDLE; + subghz_txrx_stop(app->subghz->txrx); + meal_pager_blink_stop(app); + meal_pager_transmit_model_set_sending(app->meal_pager_transmit, 0); + } + return true; + } + + return consumed; +} + +void meal_pager_scene_set_first_station_on_exit(void* context) { + Meal_Pager* app = context; + UNUSED(app); +} \ No newline at end of file diff --git a/scenes/meal_pager_scene_set_last_pager.c b/scenes/meal_pager_scene_set_last_pager.c new file mode 100644 index 00000000000..6cfd83f3990 --- /dev/null +++ b/scenes/meal_pager_scene_set_last_pager.c @@ -0,0 +1,63 @@ +#include "../meal_pager_i.h" +#include "../helpers/meal_pager_custom_event.h" +#include "../helpers/meal_pager_led.h" +#include + +void meal_pager_set_last_pager_callback(void* context) { + furi_assert(context); + Meal_Pager* app = context; + view_dispatcher_send_custom_event(app->view_dispatcher, Meal_PagerCustomerEventIntInput); +} + +void meal_pager_scene_set_last_pager_on_enter(void* context) { + furi_assert(context); + Meal_Pager* app = context; + IntInput* int_input = app->int_input; + size_t enter_name_length = 5; + meal_pager_set_max_values(app); + char* str = "Set Last Pager (0 - 999)"; + const char* constStr = str; + snprintf(str, 36, "Set Last Pager (%lu - %lu)", app->first_pager, app->max_pager); + + int_input_set_header_text(int_input, constStr); + + int_input_set_result_callback( + int_input, + meal_pager_set_last_pager_callback, + context, + app->text_store[3], + enter_name_length, + false); + + view_dispatcher_switch_to_view(app->view_dispatcher, Meal_PagerViewIdIntInput); +} + +bool meal_pager_scene_set_last_pager_on_event(void* context, SceneManagerEvent event) { + Meal_Pager* app = context; + bool consumed = false; + + if(event.type == SceneManagerEventTypeBack) { + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeCustom) { + app->last_pager = atoi(app->text_store[3]); + app->last_pager_char = app->text_store[3]; + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeTick) { + if(app->state_notifications == SubGhzNotificationStateTx) { + app->state_notifications = SubGhzNotificationStateIDLE; + subghz_txrx_stop(app->subghz->txrx); + meal_pager_blink_stop(app); + meal_pager_transmit_model_set_sending(app->meal_pager_transmit, 0); + } + return true; + } + + return consumed; +} + +void meal_pager_scene_set_last_pager_on_exit(void* context) { + Meal_Pager* app = context; + UNUSED(app); +} \ No newline at end of file diff --git a/scenes/meal_pager_scene_set_last_station.c b/scenes/meal_pager_scene_set_last_station.c new file mode 100644 index 00000000000..67eb633923f --- /dev/null +++ b/scenes/meal_pager_scene_set_last_station.c @@ -0,0 +1,63 @@ +#include "../meal_pager_i.h" +#include "../helpers/meal_pager_custom_event.h" +#include "../helpers/meal_pager_led.h" +#include + +void meal_pager_set_last_station_callback(void* context) { + furi_assert(context); + Meal_Pager* app = context; + view_dispatcher_send_custom_event(app->view_dispatcher, Meal_PagerCustomerEventIntInput); +} + +void meal_pager_scene_set_last_station_on_enter(void* context) { + furi_assert(context); + Meal_Pager* app = context; + IntInput* int_input = app->int_input; + size_t enter_name_length = 5; + meal_pager_set_max_values(app); + char* str = "Set Last Station (0 - 9999)"; + const char* constStr = str; + snprintf(str, 36, "Set Last Station (%lu - %lu)", app->last_station, app->max_station); + + int_input_set_header_text(int_input, constStr); + + int_input_set_result_callback( + int_input, + meal_pager_set_last_station_callback, + context, + app->text_store[1], + enter_name_length, + false); + + view_dispatcher_switch_to_view(app->view_dispatcher, Meal_PagerViewIdIntInput); +} + +bool meal_pager_scene_set_last_station_on_event(void* context, SceneManagerEvent event) { + Meal_Pager* app = context; + bool consumed = false; + + if(event.type == SceneManagerEventTypeBack) { + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeCustom) { + app->last_station = atoi(app->text_store[1]); + app->last_station_char = app->text_store[1]; + scene_manager_previous_scene(app->scene_manager); + return true; + } else if(event.type == SceneManagerEventTypeTick) { + if(app->state_notifications == SubGhzNotificationStateTx) { + app->state_notifications = SubGhzNotificationStateIDLE; + subghz_txrx_stop(app->subghz->txrx); + meal_pager_blink_stop(app); + meal_pager_transmit_model_set_sending(app->meal_pager_transmit, 0); + } + return true; + } + + return consumed; +} + +void meal_pager_scene_set_last_station_on_exit(void* context) { + Meal_Pager* app = context; + UNUSED(app); +} \ No newline at end of file diff --git a/scenes/meal_pager_scene_settings.c b/scenes/meal_pager_scene_settings.c index 082870160a8..d60efe52da4 100644 --- a/scenes/meal_pager_scene_settings.c +++ b/scenes/meal_pager_scene_settings.c @@ -66,21 +66,11 @@ static void meal_pager_scene_settings_set_pager_type(VariableItem* item) { } static void meal_pager_scene_settings_set_first_station(VariableItem* item) { - Meal_Pager* app = variable_item_get_context(item); - uint32_t index = variable_item_get_current_value_index(item); - - snprintf(app->first_station_char, 20, "%lu", index); - variable_item_set_current_value_text(item, app->first_station_char); - app->first_station = index; + UNUSED(item); } static void meal_pager_scene_settings_set_last_station(VariableItem* item) { - Meal_Pager* app = variable_item_get_context(item); - uint32_t index = variable_item_get_current_value_index(item); - - snprintf(app->last_station_char, 20, "%lu", index); - variable_item_set_current_value_text(item, app->last_station_char); - app->last_station = index; + UNUSED(item); } static void meal_pager_scene_settings_set_first_pager(VariableItem* item) { @@ -110,7 +100,7 @@ static void meal_pager_scene_settings_set_repeats(VariableItem* item) { app->repeats = index; } -static void meal_pager_scene_settings_set_haptic(VariableItem* item) { +/*static void meal_pager_scene_settings_set_haptic(VariableItem* item) { Meal_Pager* app = variable_item_get_context(item); uint8_t index = variable_item_get_current_value_index(item); @@ -123,7 +113,7 @@ static void meal_pager_scene_settings_set_speaker(VariableItem* item) { uint8_t index = variable_item_get_current_value_index(item); variable_item_set_current_value_text(item, speaker_text[index]); app->speaker = speaker_value[index]; -} +}*/ static void meal_pager_scene_settings_set_led(VariableItem* item) { Meal_Pager* app = variable_item_get_context(item); @@ -151,11 +141,7 @@ void meal_pager_scene_settings_on_enter(void* context) { // Pager Type item = variable_item_list_add( - app->variable_item_list, - "Pager Type:", - 4, - meal_pager_scene_settings_set_pager_type, - app); + app->variable_item_list, "Pager Type:", 4, meal_pager_scene_settings_set_pager_type, app); value_index = value_index_uint32(app->pager_type, pager_type_value, 4); variable_item_set_current_value_index(item, value_index); variable_item_set_current_value_text(item, pager_type_text[value_index]); @@ -164,10 +150,9 @@ void meal_pager_scene_settings_on_enter(void* context) { item = variable_item_list_add( app->variable_item_list, "First Station", - 255, + 1, meal_pager_scene_settings_set_first_station, app); - variable_item_set_current_value_index(item, app->first_station); snprintf(app->first_pager_char, 20, "%lu", app->first_station); variable_item_set_current_value_text(item, app->first_station_char); @@ -175,24 +160,21 @@ void meal_pager_scene_settings_on_enter(void* context) { item = variable_item_list_add( app->variable_item_list, "Last Station", - 255, + 1, meal_pager_scene_settings_set_last_station, app); - variable_item_set_current_value_index(item, app->last_station); snprintf(app->last_station_char, 20, "%lu", app->last_station); variable_item_set_current_value_text(item, app->last_station_char); // First Pager item = variable_item_list_add( - app->variable_item_list, "First Pager", 99, meal_pager_scene_settings_set_first_pager, app); - variable_item_set_current_value_index(item, app->first_pager); + app->variable_item_list, "First Pager", 1, meal_pager_scene_settings_set_first_pager, app); snprintf(app->first_pager_char, 20, "%lu", app->first_pager); variable_item_set_current_value_text(item, app->first_pager_char); // Last Pager item = variable_item_list_add( - app->variable_item_list, "Last Pager", 99, meal_pager_scene_settings_set_last_pager, app); - variable_item_set_current_value_index(item, app->last_pager); + app->variable_item_list, "Last Pager", 1, meal_pager_scene_settings_set_last_pager, app); snprintf(app->last_pager_char, 20, "%lu", app->last_pager); variable_item_set_current_value_text(item, app->last_pager_char); @@ -203,19 +185,19 @@ void meal_pager_scene_settings_on_enter(void* context) { snprintf(app->repeats_char, 20, "%lu", app->repeats); variable_item_set_current_value_text(item, app->repeats_char); - // Vibro on/off - item = variable_item_list_add( + // Vibro on/off Disabled until used + /*item = variable_item_list_add( app->variable_item_list, "Vibro/Haptic:", 2, meal_pager_scene_settings_set_haptic, app); value_index = value_index_uint32(app->haptic, haptic_value, 2); variable_item_set_current_value_index(item, value_index); - variable_item_set_current_value_text(item, haptic_text[value_index]); + variable_item_set_current_value_text(item, haptic_text[value_index]);*/ - // Sound on/off - item = variable_item_list_add( + // Sound on/off Disabled until used + /*item = variable_item_list_add( app->variable_item_list, "Sound:", 2, meal_pager_scene_settings_set_speaker, app); value_index = value_index_uint32(app->speaker, speaker_value, 2); variable_item_set_current_value_index(item, value_index); - variable_item_set_current_value_text(item, speaker_text[value_index]); + variable_item_set_current_value_text(item, speaker_text[value_index]);*/ // LED Effects on/off item = variable_item_list_add( diff --git a/scenes/meal_pager_scene_transmit.c b/scenes/meal_pager_scene_transmit.c index 142c35ff2d5..dc49d2911a7 100644 --- a/scenes/meal_pager_scene_transmit.c +++ b/scenes/meal_pager_scene_transmit.c @@ -93,7 +93,7 @@ bool meal_pager_scene_transmit_on_event(void* context, SceneManagerEvent event) break; } } else if(event.type == SceneManagerEventTypeTick) { - if(app->state_notifications == SubGhzNotificationStateTx) { + if(app->state_notifications == SubGhzNotificationStateTx && app->led == 1) { notification_message(app->notification, &sequence_blink_magenta_10); } return true;