[FL-1756, FL-1769, FL-1776, FL-1759] Gui: input events complementary V3, refactoring. SubGhz: read/emulate fixes. Cleanup. (#684)
* Gui: move rotation logic to ViewPort, replace delayed View switch in ViewDispatcher with event filtering and redirection to previous view. * SubGhz: add function description * Gui, Input: add event id to input events. * SubGhz: fix "crashing on ?" * SubGhz: add icon scanning * SubGhz: updated interface read scene, updated interface config scene * Assets: update subghz assets * SubGhz: replaced the picture in the read scene, changed the paths to additional files * SubGhz: fix deadlock in timer callback * SubGhz: fix icon read scene * SubGhz: fix icon read scene * SubGhz: fix duble text transmitter scene * SubGhz: correct spelling. Gui: bigger queue for ViewDispatcher. * SubGhz: fix creation and transmission of dynamic code without the presence of a manufactory key * SubGhz: fix keelog, setting a name in the absence of a manufactory key * SubGhz: fix load bad keelog key * Format sources * Furi: remove garbage from core. GpioTester: fix memory leak and cleanup * Accessor: remove obsolete notification code * MusicPlayer: remove input event injection * Input: rename id to sequence Co-authored-by: Aleksandr Kutuzov <alleteam@gmail.com>
This commit is contained in:
		| @@ -2,143 +2,134 @@ | ||||
| #include <furi-hal.h> | ||||
|  | ||||
| #include <gui/gui.h> | ||||
| #include <input/input.h> | ||||
| #include <notification/notification-messages.h> | ||||
|  | ||||
| typedef struct { | ||||
|     const char* name; | ||||
|     GpioPin pin; | ||||
|     const GpioPin* pin; | ||||
| } GpioItem; | ||||
|  | ||||
| const GpioItem GPIO_PINS[] = { | ||||
|     {"1.2: PA7", {GPIOA, GPIO_PIN_7}}, | ||||
|     {"1.3: PA6", {GPIOA, GPIO_PIN_6}}, | ||||
|     {"1.4: PA4", {GPIOA, GPIO_PIN_4}}, | ||||
|     {"1.5: PB3", {GPIOB, GPIO_PIN_3}}, | ||||
|     {"1.6: PB2", {GPIOB, GPIO_PIN_2}}, | ||||
|     {"1.7: PC3", {GPIOC, GPIO_PIN_3}}, | ||||
|  | ||||
|     {"2.7: PC1", {GPIOC, GPIO_PIN_1}}, | ||||
|     {"2.8: PC0", {GPIOC, GPIO_PIN_0}}, | ||||
| static const GpioItem GPIO_PINS[] = { | ||||
|     {"1.2: PA7", &gpio_ext_pa7}, | ||||
|     {"1.3: PA6", &gpio_ext_pa6}, | ||||
|     {"1.4: PA4", &gpio_ext_pa4}, | ||||
|     {"1.5: PB3", &gpio_ext_pb3}, | ||||
|     {"1.6: PB2", &gpio_ext_pb2}, | ||||
|     {"1.7: PC3", &gpio_ext_pc3}, | ||||
|     {"2.7: PC1", &gpio_ext_pc1}, | ||||
|     {"2.8: PC0", &gpio_ext_pc0}, | ||||
| }; | ||||
|  | ||||
| typedef enum { | ||||
|     EventTypeTick, | ||||
|     EventTypeKey, | ||||
| } EventType; | ||||
|  | ||||
| typedef struct { | ||||
|     union { | ||||
|         InputEvent input; | ||||
|     } value; | ||||
|     EventType type; | ||||
| } AppEvent; | ||||
| static const size_t GPIO_PINS_COUNT = sizeof(GPIO_PINS) / sizeof(GPIO_PINS[0]); | ||||
|  | ||||
| typedef struct { | ||||
|     osMessageQueueId_t input_queue; | ||||
|     uint8_t gpio_index; | ||||
| } State; | ||||
|     ViewPort* view_port; | ||||
|     Gui* gui; | ||||
|     NotificationApp* notification; | ||||
| } GpioTest; | ||||
|  | ||||
| static void render_callback(Canvas* canvas, void* ctx) { | ||||
|     State* state = (State*)acquire_mutex((ValueMutex*)ctx, 25); | ||||
| static void gpio_test_render_callback(Canvas* canvas, void* ctx) { | ||||
|     GpioTest* gpio_test = ctx; | ||||
|  | ||||
|     canvas_clear(canvas); | ||||
|     canvas_set_color(canvas, ColorBlack); | ||||
|     canvas_set_font(canvas, FontPrimary); | ||||
|     canvas_draw_str(canvas, 2, 10, "GPIO Control"); | ||||
|     canvas_set_font(canvas, FontSecondary); | ||||
|     canvas_draw_str(canvas, 2, 25, GPIO_PINS[state->gpio_index].name); | ||||
|  | ||||
|     release_mutex((ValueMutex*)ctx, state); | ||||
|     canvas_draw_str(canvas, 2, 25, GPIO_PINS[gpio_test->gpio_index].name); | ||||
| } | ||||
|  | ||||
| static void input_callback(InputEvent* input_event, void* ctx) { | ||||
|     osMessageQueueId_t event_queue = ctx; | ||||
| static void gpio_test_input_callback(InputEvent* input_event, void* ctx) { | ||||
|     GpioTest* gpio_test = ctx; | ||||
|  | ||||
|     AppEvent event; | ||||
|     event.type = EventTypeKey; | ||||
|     event.value.input = *input_event; | ||||
|     osMessageQueuePut(event_queue, &event, 0, 0); | ||||
|     osMessageQueuePut(gpio_test->input_queue, input_event, 0, 0); | ||||
| } | ||||
|  | ||||
| static void gpio_test_configure_pins(GpioMode mode) { | ||||
|     for(size_t i = 0; i < GPIO_PINS_COUNT; i++) { | ||||
|         hal_gpio_write(GPIO_PINS[i].pin, false); | ||||
|         hal_gpio_init(GPIO_PINS[i].pin, mode, GpioPullNo, GpioSpeedLow); | ||||
|     } | ||||
| } | ||||
|  | ||||
| GpioTest* gpio_test_alloc() { | ||||
|     GpioTest* instance = furi_alloc(sizeof(GpioTest)); | ||||
|  | ||||
|     gpio_test_configure_pins(GpioModeOutputPushPull); | ||||
|  | ||||
|     instance->input_queue = osMessageQueueNew(8, sizeof(InputEvent), NULL); | ||||
|     furi_check(instance->input_queue); | ||||
|  | ||||
|     instance->view_port = view_port_alloc(); | ||||
|     view_port_draw_callback_set(instance->view_port, gpio_test_render_callback, instance); | ||||
|     view_port_input_callback_set(instance->view_port, gpio_test_input_callback, instance); | ||||
|  | ||||
|     instance->gui = furi_record_open("gui"); | ||||
|     gui_add_view_port(instance->gui, instance->view_port, GuiLayerFullscreen); | ||||
|  | ||||
|     instance->notification = furi_record_open("notification"); | ||||
|  | ||||
|     return instance; | ||||
| } | ||||
|  | ||||
| void gpio_test_free(GpioTest* instance) { | ||||
|     furi_assert(instance); | ||||
|  | ||||
|     furi_record_close("notification"); | ||||
|  | ||||
|     view_port_enabled_set(instance->view_port, false); | ||||
|     gui_remove_view_port(instance->gui, instance->view_port); | ||||
|     furi_record_close("gui"); | ||||
|  | ||||
|     view_port_free(instance->view_port); | ||||
|  | ||||
|     osMessageQueueDelete(instance->input_queue); | ||||
|  | ||||
|     gpio_test_configure_pins(GpioModeAnalog); | ||||
|  | ||||
|     free(instance); | ||||
| } | ||||
|  | ||||
| int32_t gpio_test_app(void* p) { | ||||
|     osMessageQueueId_t event_queue = osMessageQueueNew(8, sizeof(AppEvent), NULL); | ||||
|     furi_check(event_queue); | ||||
|     GpioTest* gpio_test = gpio_test_alloc(); | ||||
|  | ||||
|     State _state; | ||||
|     _state.gpio_index = 0; | ||||
|     InputEvent event; | ||||
|     while(osMessageQueueGet(gpio_test->input_queue, &event, NULL, osWaitForever) == osOK) { | ||||
|         if(event.type == InputTypeShort) { | ||||
|             if(event.key == InputKeyBack) { | ||||
|                 notification_message(gpio_test->notification, &sequence_reset_green); | ||||
|                 break; | ||||
|             } | ||||
|  | ||||
|     ValueMutex state_mutex; | ||||
|     if(!init_mutex(&state_mutex, &_state, sizeof(State))) { | ||||
|         printf("[gpio-tester] cannot create mutex\r\n"); | ||||
|         return 255; | ||||
|     } | ||||
|  | ||||
|     ViewPort* view_port = view_port_alloc(); | ||||
|  | ||||
|     view_port_draw_callback_set(view_port, render_callback, &state_mutex); | ||||
|     view_port_input_callback_set(view_port, input_callback, event_queue); | ||||
|  | ||||
|     // Open GUI and register view_port | ||||
|     Gui* gui = furi_record_open("gui"); | ||||
|     gui_add_view_port(gui, view_port, GuiLayerFullscreen); | ||||
|  | ||||
|     NotificationApp* notification = furi_record_open("notification"); | ||||
|  | ||||
|     // configure pin | ||||
|     for(uint8_t i = 0; i < sizeof(GPIO_PINS) / sizeof(GPIO_PINS[0]); i++) { | ||||
|         hal_gpio_init( | ||||
|             (GpioPin*)&GPIO_PINS[i].pin, GpioModeOutputPushPull, GpioPullNo, GpioSpeedLow); | ||||
|     } | ||||
|  | ||||
|     AppEvent event; | ||||
|     while(1) { | ||||
|         osStatus_t event_status = osMessageQueueGet(event_queue, &event, NULL, osWaitForever); | ||||
|         State* state = (State*)acquire_mutex_block(&state_mutex); | ||||
|  | ||||
|         if(event_status == osOK) { | ||||
|             if(event.type == EventTypeKey) { | ||||
|                 if(event.value.input.type == InputTypeShort && | ||||
|                    event.value.input.key == InputKeyBack) { | ||||
|                     printf("[gpio-tester] bye!\r\n"); | ||||
|                     notification_message(notification, &sequence_reset_green); | ||||
|                     furi_record_close("notification"); | ||||
|  | ||||
|                     view_port_enabled_set(view_port, false); | ||||
|                     gui_remove_view_port(gui, view_port); | ||||
|                     view_port_free(view_port); | ||||
|  | ||||
|                     return 0; | ||||
|             if(event.key == InputKeyRight) { | ||||
|                 if(gpio_test->gpio_index < (GPIO_PINS_COUNT - 1)) { | ||||
|                     gpio_test->gpio_index++; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|                 if(event.value.input.type == InputTypeShort && | ||||
|                    event.value.input.key == InputKeyRight) { | ||||
|                     if(state->gpio_index < (sizeof(GPIO_PINS) / sizeof(GPIO_PINS[0]) - 1)) { | ||||
|                         state->gpio_index++; | ||||
|                     } | ||||
|             if(event.key == InputKeyLeft) { | ||||
|                 if(gpio_test->gpio_index > 0) { | ||||
|                     gpio_test->gpio_index--; | ||||
|                 } | ||||
|  | ||||
|                 if(event.value.input.type == InputTypeShort && | ||||
|                    event.value.input.key == InputKeyLeft) { | ||||
|                     if(state->gpio_index > 0) { | ||||
|                         state->gpio_index--; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
|                 if(event.value.input.key == InputKeyOk) { | ||||
|                     if(event.value.input.type == InputTypePress) { | ||||
|                         hal_gpio_write((GpioPin*)&GPIO_PINS[state->gpio_index].pin, true); | ||||
|                         notification_message(notification, &sequence_set_green_255); | ||||
|                     } else if(event.value.input.type == InputTypeRelease) { | ||||
|                         hal_gpio_write((GpioPin*)&GPIO_PINS[state->gpio_index].pin, false); | ||||
|                         notification_message(notification, &sequence_reset_green); | ||||
|                     } | ||||
|             } | ||||
|         } else { | ||||
|             if(event.key == InputKeyOk) { | ||||
|                 if(event.type == InputTypePress) { | ||||
|                     hal_gpio_write(GPIO_PINS[gpio_test->gpio_index].pin, true); | ||||
|                     notification_message(gpio_test->notification, &sequence_set_green_255); | ||||
|                 } else if(event.type == InputTypeRelease) { | ||||
|                     hal_gpio_write(GPIO_PINS[gpio_test->gpio_index].pin, false); | ||||
|                     notification_message(gpio_test->notification, &sequence_reset_green); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         release_mutex(&state_mutex, state); | ||||
|         view_port_update(view_port); | ||||
|         view_port_update(gpio_test->view_port); | ||||
|     } | ||||
|  | ||||
|     gpio_test_free(gpio_test); | ||||
|  | ||||
|     return 0; | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user