 2daf65b62b
			
		
	
	
		2daf65b62b
		
			
		
	
	
	
	
		
			
			* Notification app: init * Notification app: separate message sequences * Notification app: rename notifications to notification * Notification app: rework api * Notification app: new sequences for charger * Power app: add state for better led handling * Power app: NotificationSequence type, notification led process * Blink app: use notifications * Notification app: sound and vibro notifications * Notification app: note messages * Notification app: more messages * Notification app: update note message generator * Blink app: fix state counter * Notification app: fix delay event * App sd-filesystem: notifications * App notifications: headers c++ compatibility * App notifications: Cmaj success chord sequence * App iButton: use notifications * App notification: display backlight notifications * App notification: add "display on" message to success and error sequences * App accessor: use notifications * App ibutton: guard onewire key read * Lib-RFAL: remove api_hal_light usage * App notification: add blocking mode, rework display api * Cli led command: use internal notification instead of direc access to leds. * App unit test: use notifications * App lfrfid: use notifications * Apps: close notification record * App subghz: rough use of notifications * App notificaton: ignore reset flag * App strobe: removed * Lib irda decoder: fix nec decoding * App irda: fix assert, use notifications * Apps: use notifications * Fix IRDA tests * Cli: better var naming * App notification: readable sources Co-authored-by: Albert Kharisov <albert@flipperdevices.com> Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			145 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <furi.h>
 | |
| #include <api-hal.h>
 | |
| 
 | |
| #include <gui/gui.h>
 | |
| #include <input/input.h>
 | |
| #include <notification/notification-messages.h>
 | |
| 
 | |
| typedef struct {
 | |
|     const char* name;
 | |
|     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}},
 | |
| };
 | |
| 
 | |
| typedef enum {
 | |
|     EventTypeTick,
 | |
|     EventTypeKey,
 | |
| } EventType;
 | |
| 
 | |
| typedef struct {
 | |
|     union {
 | |
|         InputEvent input;
 | |
|     } value;
 | |
|     EventType type;
 | |
| } AppEvent;
 | |
| 
 | |
| typedef struct {
 | |
|     uint8_t gpio_index;
 | |
| } State;
 | |
| 
 | |
| static void render_callback(Canvas* canvas, void* ctx) {
 | |
|     State* state = (State*)acquire_mutex((ValueMutex*)ctx, 25);
 | |
| 
 | |
|     canvas_clear(canvas);
 | |
|     canvas_set_color(canvas, ColorBlack);
 | |
|     canvas_set_font(canvas, FontPrimary);
 | |
|     canvas_draw_str(canvas, 2, 10, "GPIO demo");
 | |
|     canvas_set_font(canvas, FontSecondary);
 | |
|     canvas_draw_str(canvas, 2, 25, GPIO_PINS[state->gpio_index].name);
 | |
| 
 | |
|     release_mutex((ValueMutex*)ctx, state);
 | |
| }
 | |
| 
 | |
| static void input_callback(InputEvent* input_event, void* ctx) {
 | |
|     osMessageQueueId_t event_queue = ctx;
 | |
| 
 | |
|     AppEvent event;
 | |
|     event.type = EventTypeKey;
 | |
|     event.value.input = *input_event;
 | |
|     osMessageQueuePut(event_queue, &event, 0, 0);
 | |
| }
 | |
| 
 | |
| int32_t app_gpio_test(void* p) {
 | |
|     osMessageQueueId_t event_queue = osMessageQueueNew(8, sizeof(AppEvent), NULL);
 | |
|     furi_check(event_queue);
 | |
| 
 | |
|     State _state;
 | |
|     _state.gpio_index = 0;
 | |
| 
 | |
|     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.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.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);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         release_mutex(&state_mutex, state);
 | |
|         view_port_update(view_port);
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 |