| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348 |
- #include "barcode_app.h"
- #include "barcode_app_icons.h"
- /**
- * Opens a file browser dialog and returns the filepath of the selected file
- *
- * @param folder the folder to view when the browser opens
- * @param file_path a string pointer for the file_path when a file is selected,
- * file_path will be the folder path is nothing is selected
- * @returns true if a file is selected
- */
- static bool select_file(const char* folder, FuriString* file_path) {
- DialogsApp* dialogs = furi_record_open(RECORD_DIALOGS);
- DialogsFileBrowserOptions browser_options;
- dialog_file_browser_set_basic_options(&browser_options, "", &I_barcode_10);
- browser_options.base_path = DEFAULT_USER_BARCODES;
- furi_string_set(file_path, folder);
- bool res = dialog_file_browser_show(dialogs, file_path, file_path, &browser_options);
- furi_record_close(RECORD_DIALOGS);
- return res;
- }
- /**
- * Reads the data from a file and stores them in the FuriStrings raw_type and raw_data
- */
- ErrorCode read_raw_data(FuriString* file_path, FuriString* raw_type, FuriString* raw_data) {
- //Open Storage
- Storage* storage = furi_record_open(RECORD_STORAGE);
- FlipperFormat* ff = flipper_format_file_alloc(storage);
- ErrorCode reason = OKCode;
- if(!flipper_format_file_open_existing(ff, furi_string_get_cstr(file_path))) {
- FURI_LOG_E(TAG, "Could not open file %s", furi_string_get_cstr(file_path));
- reason = FileOpening;
- } else {
- if(!flipper_format_read_string(ff, "Type", raw_type)) {
- FURI_LOG_E(TAG, "Could not read \"Type\" string");
- reason = InvalidFileData;
- }
- if(!flipper_format_read_string(ff, "Data", raw_data)) {
- FURI_LOG_E(TAG, "Could not read \"Data\" string");
- reason = InvalidFileData;
- }
- }
- //Close Storage
- flipper_format_free(ff);
- furi_record_close(RECORD_STORAGE);
- return reason;
- }
- /**
- * Gets the file name from a file path
- * @param file_path the file path
- * @param file_name the FuriString to store the file name
- * @param remove_extension true if the extension should be removed, otherwise false
- */
- bool get_file_name_from_path(FuriString* file_path, FuriString* file_name, bool remove_extension) {
- if(file_path == NULL || file_name == NULL) {
- return false;
- }
- uint32_t slash_index = furi_string_search_rchar(file_path, '/', 0);
- if(slash_index == FURI_STRING_FAILURE || slash_index >= (furi_string_size(file_path) - 1)) {
- return false;
- }
- furi_string_set(file_name, file_path);
- furi_string_right(file_name, slash_index + 1);
- if(remove_extension) {
- uint32_t ext_index = furi_string_search_rchar(file_name, '.', 0);
- if(ext_index != FURI_STRING_FAILURE && ext_index < (furi_string_size(file_path))) {
- furi_string_left(file_name, ext_index);
- }
- }
- return true;
- }
- /**
- * Creates the barcode folder
- */
- void init_folder() {
- Storage* storage = furi_record_open(RECORD_STORAGE);
- FURI_LOG_I(TAG, "Creating barcodes folder");
- if(storage_simply_mkdir(storage, DEFAULT_USER_BARCODES)) {
- FURI_LOG_I(TAG, "Barcodes folder successfully created!");
- } else {
- FURI_LOG_I(TAG, "Barcodes folder already exists.");
- }
- furi_record_close(RECORD_STORAGE);
- }
- void select_barcode_item(BarcodeApp* app) {
- FuriString* file_path = furi_string_alloc();
- FuriString* raw_type = furi_string_alloc();
- FuriString* raw_data = furi_string_alloc();
- //this determines if the data was read correctly or if the
- bool loaded_success = true;
- ErrorCode reason = OKCode;
- bool file_selected = select_file(DEFAULT_USER_BARCODES, file_path);
- if(file_selected) {
- FURI_LOG_I(TAG, "The file selected is %s", furi_string_get_cstr(file_path));
- Barcode* barcode = app->barcode_view;
- reason = read_raw_data(file_path, raw_type, raw_data);
- if(reason != OKCode) {
- loaded_success = false;
- FURI_LOG_E(TAG, "Could not read data correctly");
- }
- //Free the data from the previous barcode
- barcode_free_model(barcode);
- with_view_model(
- barcode->view,
- BarcodeModel * model,
- {
- model->file_path = furi_string_alloc_set(file_path);
- model->data = malloc(sizeof(BarcodeData));
- model->data->valid = loaded_success;
- if(loaded_success) {
- model->data->raw_data = furi_string_alloc_set(raw_data);
- model->data->correct_data = furi_string_alloc();
- model->data->type_obj = get_type(raw_type);
- barcode_loader(model->data);
- } else {
- model->data->reason = reason;
- }
- },
- true);
- view_dispatcher_switch_to_view(app->view_dispatcher, BarcodeView);
- }
- furi_string_free(raw_type);
- furi_string_free(raw_data);
- furi_string_free(file_path);
- }
- void edit_barcode_item(BarcodeApp* app) {
- FuriString* file_path = furi_string_alloc();
- FuriString* file_name = furi_string_alloc();
- FuriString* raw_type = furi_string_alloc();
- FuriString* raw_data = furi_string_alloc();
- //this determines if the data was read correctly or if the
- ErrorCode reason = OKCode;
- bool file_selected = select_file(DEFAULT_USER_BARCODES, file_path);
- if(file_selected) {
- FURI_LOG_I(TAG, "The file selected is %s", furi_string_get_cstr(file_path));
- CreateView* create_view_object = app->create_view;
- reason = read_raw_data(file_path, raw_type, raw_data);
- if(reason != OKCode) {
- FURI_LOG_E(TAG, "Could not read data correctly");
- with_view_model(
- app->message_view->view,
- MessageViewModel * model,
- { model->message = get_error_code_message(reason); },
- true);
- view_dispatcher_switch_to_view(
- create_view_object->barcode_app->view_dispatcher, MessageErrorView);
- } else {
- BarcodeTypeObj* type_obj = get_type(raw_type);
- if(type_obj->type == UNKNOWN) {
- type_obj = barcode_type_objs[0];
- }
- get_file_name_from_path(file_path, file_name, true);
- create_view_free_model(create_view_object);
- with_view_model(
- create_view_object->view,
- CreateViewModel * model,
- {
- model->selected_menu_item = 0;
- model->barcode_type = type_obj;
- model->file_path = furi_string_alloc_set(file_path);
- model->file_name = furi_string_alloc_set(file_name);
- model->barcode_data = furi_string_alloc_set(raw_data);
- model->mode = EditMode;
- },
- true);
- view_dispatcher_switch_to_view(app->view_dispatcher, CreateBarcodeView);
- }
- }
- furi_string_free(raw_type);
- furi_string_free(raw_data);
- furi_string_free(file_name);
- furi_string_free(file_path);
- }
- void create_barcode_item(BarcodeApp* app) {
- CreateView* create_view_object = app->create_view;
- create_view_free_model(create_view_object);
- with_view_model(
- create_view_object->view,
- CreateViewModel * model,
- {
- model->selected_menu_item = 0;
- model->barcode_type = barcode_type_objs[0];
- model->file_path = furi_string_alloc();
- model->file_name = furi_string_alloc();
- model->barcode_data = furi_string_alloc();
- model->mode = NewMode;
- },
- true);
- view_dispatcher_switch_to_view(app->view_dispatcher, CreateBarcodeView);
- }
- void submenu_callback(void* context, uint32_t index) {
- furi_assert(context);
- BarcodeApp* app = context;
- if(index == SelectBarcodeItem) {
- select_barcode_item(app);
- } else if(index == EditBarcodeItem) {
- edit_barcode_item(app);
- } else if(index == CreateBarcodeItem) {
- create_barcode_item(app);
- }
- }
- uint32_t create_view_callback(void* context) {
- UNUSED(context);
- return CreateBarcodeView;
- }
- uint32_t main_menu_callback(void* context) {
- UNUSED(context);
- return MainMenuView;
- }
- uint32_t exit_callback(void* context) {
- UNUSED(context);
- return VIEW_NONE;
- }
- void free_app(BarcodeApp* app) {
- FURI_LOG_I(TAG, "Freeing Data");
- init_folder();
- free_types();
- view_dispatcher_remove_view(app->view_dispatcher, TextInputView);
- text_input_free(app->text_input);
- view_dispatcher_remove_view(app->view_dispatcher, MessageErrorView);
- message_view_free(app->message_view);
- view_dispatcher_remove_view(app->view_dispatcher, MainMenuView);
- submenu_free(app->main_menu);
- view_dispatcher_remove_view(app->view_dispatcher, CreateBarcodeView);
- create_view_free(app->create_view);
- view_dispatcher_remove_view(app->view_dispatcher, BarcodeView);
- barcode_free(app->barcode_view);
- //free the dispatcher
- view_dispatcher_free(app->view_dispatcher);
- furi_message_queue_free(app->event_queue);
- furi_record_close(RECORD_GUI);
- app->gui = NULL;
- free(app);
- }
- int32_t barcode_main(void* p) {
- UNUSED(p);
- BarcodeApp* app = malloc(sizeof(BarcodeApp));
- init_types();
- app->event_queue = furi_message_queue_alloc(8, sizeof(InputEvent));
- // Register view port in GUI
- app->gui = furi_record_open(RECORD_GUI);
- app->view_dispatcher = view_dispatcher_alloc();
- view_dispatcher_enable_queue(app->view_dispatcher);
- view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen);
- app->main_menu = submenu_alloc();
- submenu_add_item(app->main_menu, "Load Barcode", SelectBarcodeItem, submenu_callback, app);
- view_set_previous_callback(submenu_get_view(app->main_menu), exit_callback);
- view_dispatcher_add_view(app->view_dispatcher, MainMenuView, submenu_get_view(app->main_menu));
- submenu_add_item(app->main_menu, "Edit Barcode", EditBarcodeItem, submenu_callback, app);
- /*****************************
- * Creating Text Input View
- ******************************/
- app->text_input = text_input_alloc();
- view_set_previous_callback(text_input_get_view(app->text_input), create_view_callback);
- view_dispatcher_add_view(
- app->view_dispatcher, TextInputView, text_input_get_view(app->text_input));
- /*****************************
- * Creating Message View
- ******************************/
- app->message_view = message_view_allocate(app);
- view_dispatcher_add_view(
- app->view_dispatcher, MessageErrorView, message_get_view(app->message_view));
- /*****************************
- * Creating Create View
- ******************************/
- app->create_view = create_view_allocate(app);
- submenu_add_item(app->main_menu, "Create Barcode", CreateBarcodeItem, submenu_callback, app);
- view_set_previous_callback(create_get_view(app->create_view), main_menu_callback);
- view_dispatcher_add_view(
- app->view_dispatcher, CreateBarcodeView, create_get_view(app->create_view));
- /*****************************
- * Creating Barcode View
- ******************************/
- app->barcode_view = barcode_view_allocate(app);
- view_set_previous_callback(barcode_get_view(app->barcode_view), main_menu_callback);
- view_dispatcher_add_view(
- app->view_dispatcher, BarcodeView, barcode_get_view(app->barcode_view));
- //switch view to submenu and run dispatcher
- view_dispatcher_switch_to_view(app->view_dispatcher, MainMenuView);
- view_dispatcher_run(app->view_dispatcher);
- free_app(app);
- return 0;
- }
|