* ble: prototype ble hid * ble: add HID service and characteristics * debug tools: add ble keyboard app * ble: change appearance * ble: working keyboard * bt: introduce furi-hal-bt-hid * bt: restart hid service on each keyboard app enter * bt: introduce switch profile * bt: add profile to ble glue * bt: working profile switch * bt: introduce bt serial profile, rework API * bt: rewotk HID profile * bt: rework gap with profile configuration * bt: move change profile routine to furi hal bt * bt: change switch profile API to blocking * bt: move battery update to furi hal bt * bt: cleanup * bt: add support for f6 target * bt: update documentation * bt: clean up code * bt: remove NO OUTPUT setting * bt: set numeric comparison pairing in BLE HID * bt: support f6 target * bt: set mac address in profile configuration * bt: set advertise name in profile config * bt: rework with furi thread * bt: support f6 target * bt: clear hci command buffer on core2 restart * bt: correct thread kill sequence * bt: fix freertos functions calls * bt: add some enterprise delays fo correct memory free * bt: code cleanup * bt: change terminate -> stop * bt: fix memory leakage Co-authored-by: あく <alleteam@gmail.com>
		
			
				
	
	
		
			139 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			139 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
#include <furi.h>
 | 
						|
#include <gui/gui.h>
 | 
						|
#include <input/input.h>
 | 
						|
#include <bt/bt_service/bt.h>
 | 
						|
#include <furi-hal-bt.h>
 | 
						|
#include <furi-hal-bt-hid.h>
 | 
						|
#include <furi-hal-usb-hid.h>
 | 
						|
 | 
						|
#define TAG "BleKeyboardApp"
 | 
						|
 | 
						|
typedef enum {
 | 
						|
    EventTypeInput,
 | 
						|
} EventType;
 | 
						|
 | 
						|
typedef struct {
 | 
						|
    union {
 | 
						|
        InputEvent input;
 | 
						|
    };
 | 
						|
    EventType type;
 | 
						|
} BleKeyboardEvent;
 | 
						|
 | 
						|
static void ble_keyboard_render_callback(Canvas* canvas, void* ctx) {
 | 
						|
    canvas_clear(canvas);
 | 
						|
 | 
						|
    canvas_set_font(canvas, FontPrimary);
 | 
						|
    canvas_draw_str(canvas, 0, 10, "BLE keypad demo");
 | 
						|
 | 
						|
    canvas_set_font(canvas, FontSecondary);
 | 
						|
    canvas_draw_str(canvas, 0, 63, "Hold [back] to exit");
 | 
						|
}
 | 
						|
 | 
						|
static void ble_keyboard_input_callback(InputEvent* input_event, void* ctx) {
 | 
						|
    osMessageQueueId_t event_queue = ctx;
 | 
						|
 | 
						|
    BleKeyboardEvent event;
 | 
						|
    event.type = EventTypeInput;
 | 
						|
    event.input = *input_event;
 | 
						|
    osMessageQueuePut(event_queue, &event, 0, osWaitForever);
 | 
						|
}
 | 
						|
 | 
						|
int32_t ble_keyboard_app(void* p) {
 | 
						|
    Bt* bt = furi_record_open("bt");
 | 
						|
    if(!bt_set_profile(bt, BtProfileHidKeyboard)) {
 | 
						|
        FURI_LOG_E(TAG, "Failed to switch profile");
 | 
						|
        furi_record_close("bt");
 | 
						|
        return -1;
 | 
						|
    }
 | 
						|
    bool bt_turned_on = furi_hal_bt_is_active();
 | 
						|
    if(!bt_turned_on) {
 | 
						|
        furi_hal_bt_start_advertising();
 | 
						|
    }
 | 
						|
 | 
						|
    osMessageQueueId_t event_queue = osMessageQueueNew(8, sizeof(BleKeyboardEvent), NULL);
 | 
						|
    furi_check(event_queue);
 | 
						|
    ViewPort* view_port = view_port_alloc();
 | 
						|
 | 
						|
    view_port_draw_callback_set(view_port, ble_keyboard_render_callback, NULL);
 | 
						|
    view_port_input_callback_set(view_port, ble_keyboard_input_callback, event_queue);
 | 
						|
 | 
						|
    // Open GUI and register view_port
 | 
						|
    Gui* gui = furi_record_open("gui");
 | 
						|
    gui_add_view_port(gui, view_port, GuiLayerFullscreen);
 | 
						|
 | 
						|
    BleKeyboardEvent event;
 | 
						|
    while(1) {
 | 
						|
        osStatus_t event_status = osMessageQueueGet(event_queue, &event, NULL, osWaitForever);
 | 
						|
 | 
						|
        if(event_status == osOK) {
 | 
						|
            if(event.type == EventTypeInput) {
 | 
						|
                if(event.input.type == InputTypeLong && event.input.key == InputKeyBack) {
 | 
						|
                    furi_hal_bt_hid_kb_release_all();
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyBack) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_ESC);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_ESC);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyOk) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_ENTER);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_ENTER);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyRight) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_RIGHT_ARROW);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_RIGHT_ARROW);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyLeft) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_LEFT_ARROW);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_LEFT_ARROW);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyDown) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_DOWN_ARROW);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_DOWN_ARROW);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if(event.input.key == InputKeyUp) {
 | 
						|
                    if(event.input.type == InputTypePress) {
 | 
						|
                        furi_hal_bt_hid_kb_press(KEY_UP_ARROW);
 | 
						|
                    } else if(event.input.type == InputTypeRelease) {
 | 
						|
                        furi_hal_bt_hid_kb_release(KEY_UP_ARROW);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
        view_port_update(view_port);
 | 
						|
    }
 | 
						|
 | 
						|
    if(bt_turned_on) {
 | 
						|
        furi_hal_bt_stop_advertising();
 | 
						|
    }
 | 
						|
    // remove & free all stuff created by app
 | 
						|
    gui_remove_view_port(gui, view_port);
 | 
						|
    view_port_free(view_port);
 | 
						|
    osMessageQueueDelete(event_queue);
 | 
						|
    furi_record_close("gui");
 | 
						|
    bt_set_profile(bt, BtProfileSerial);
 | 
						|
    furi_record_close("bt");
 | 
						|
    return 0;
 | 
						|
}
 |