7170864fe4
* 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;
|
|
}
|