Files
Momentum-Firmware/applications/services/input/input_cli.c
Willy-JL 197fef54e2 (Experimental) External CLI plugins -29kb DFU
Idea and plugin loader implementation by @akopachov, thanks!
All commands done except storage and rpc for latency reasons
2024-03-16 03:47:32 +00:00

237 lines
7.2 KiB
C

#include "input_i.h"
#include <furi.h>
#include <cli/cli.h>
#include <toolbox/args.h>
static void input_cli_usage() {
printf("Usage:\r\n");
printf("input <cmd> <args>\r\n");
printf("Cmd list:\r\n");
printf("\tdump\t\t\t - dump input events\r\n");
printf("\tkeyboard\t\t - use keyboard feedback to control flipper\r\n");
printf("\tsend <key> <type>\t - send input event\r\n");
}
static void input_cli_dump_events_callback(const void* value, void* ctx) {
furi_assert(value);
furi_assert(ctx);
FuriMessageQueue* input_queue = ctx;
furi_message_queue_put(input_queue, value, FuriWaitForever);
}
static void input_cli_dump(Cli* cli, FuriString* args, Input* input) {
UNUSED(args);
FuriMessageQueue* input_queue = furi_message_queue_alloc(8, sizeof(InputEvent));
FuriPubSubSubscription* input_subscription =
furi_pubsub_subscribe(input->event_pubsub, input_cli_dump_events_callback, input_queue);
InputEvent input_event;
printf("Press CTRL+C to stop\r\n");
while(!cli_cmd_interrupt_received(cli)) {
if(furi_message_queue_get(input_queue, &input_event, 100) == FuriStatusOk) {
printf(
"key: %s type: %s\r\n",
input_get_key_name(input_event.key),
input_get_type_name(input_event.type));
}
}
furi_pubsub_unsubscribe(input->event_pubsub, input_subscription);
furi_message_queue_free(input_queue);
}
static void fake_input(Input* input, InputKey key, InputType type) {
bool wrap = type == InputTypeShort || type == InputTypeLong;
InputEvent event;
event.key = key;
if(wrap) {
event.type = InputTypePress;
furi_pubsub_publish(input->event_pubsub, &event);
}
event.type = type;
furi_pubsub_publish(input->event_pubsub, &event);
if(wrap) {
event.type = InputTypeRelease;
furi_pubsub_publish(input->event_pubsub, &event);
}
}
static void input_cli_keyboard(Cli* cli, FuriString* args, Input* input) {
UNUSED(args);
printf("Using console keyboard feedback for flipper input\r\n");
printf("\r\nUsage:\r\n");
printf("\tMove = Arrows\r\n");
printf("\tOk = Enter\r\n");
printf("\tBack = Backspace/Ctrl + Q\r\n");
printf("\tEnable hold for next key = Space (press twice to send space key)\r\n");
printf("\r\nPress CTRL+C to stop\r\n");
bool hold = false;
while(cli_is_connected(cli)) {
char in_chr = cli_getc(cli);
if(in_chr == CliSymbolAsciiETX) break;
InputKey send_key = InputKeyMAX;
uint8_t send_ascii = AsciiValueNUL;
switch(in_chr) {
case CliSymbolAsciiEsc: // Escape code for arrows
if(!cli_read(cli, (uint8_t*)&in_chr, 1) || in_chr != '[') break;
if(!cli_read(cli, (uint8_t*)&in_chr, 1)) break;
if(in_chr >= 'A' && in_chr <= 'D') { // Arrows = Dpad
if(hold) {
send_key = InputKeyUp + (in_chr - 'A'); // Same order as InputKey
} else {
send_ascii = AsciiValueDC1 + (in_chr - 'A'); // Same order as DC
}
}
break;
case CliSymbolAsciiBackspace: // (minicom) Backspace = Back
case CliSymbolAsciiDel: // (putty/picocom) Backspace = Back
if(hold) {
send_key = InputKeyBack;
} else {
send_ascii = AsciiValueBS;
}
break;
case 0x11: // Ctrl Q = Escape (no Esc key over CLI)
if(hold) {
send_key = InputKeyBack;
} else {
send_ascii = AsciiValueESC;
}
break;
case CliSymbolAsciiCR: // Enter = Ok
if(hold) {
send_key = InputKeyOk;
} else {
send_ascii = AsciiValueCR;
}
break;
case CliSymbolAsciiSpace: // Space = Toggle hold next key
if(hold) {
send_ascii = ' ';
} else {
hold = true;
}
break;
default:
send_ascii = in_chr;
break;
}
if(send_key != InputKeyMAX) {
fake_input(input, send_key, hold ? InputTypeLong : InputTypeShort);
hold = false;
}
if(send_ascii != AsciiValueNUL) {
AsciiEvent event = {.value = send_ascii};
furi_pubsub_publish(input->ascii_pubsub, &event);
hold = false;
}
}
}
static void input_cli_send_print_usage() {
printf("Invalid arguments. Usage:\r\n");
printf("\tinput send <key> <type>\r\n");
printf("\t\t <key>\t - one of 'up', 'down', 'left', 'right', 'back', 'ok'\r\n");
printf("\t\t <type>\t - one of 'press', 'release', 'short', 'long'\r\n");
}
static void input_cli_send(Cli* cli, FuriString* args, Input* input) {
UNUSED(cli);
InputKey key;
InputType type;
FuriString* key_str;
key_str = furi_string_alloc();
bool parsed = false;
do {
// Parse Key
if(!args_read_string_and_trim(args, key_str)) {
break;
}
if(!furi_string_cmp(key_str, "up")) {
key = InputKeyUp;
} else if(!furi_string_cmp(key_str, "down")) {
key = InputKeyDown;
} else if(!furi_string_cmp(key_str, "left")) {
key = InputKeyLeft;
} else if(!furi_string_cmp(key_str, "right")) {
key = InputKeyRight;
} else if(!furi_string_cmp(key_str, "ok")) {
key = InputKeyOk;
} else if(!furi_string_cmp(key_str, "back")) {
key = InputKeyBack;
} else {
break;
}
// Parse Type
if(!furi_string_cmp(args, "press")) {
type = InputTypePress;
} else if(!furi_string_cmp(args, "release")) {
type = InputTypeRelease;
} else if(!furi_string_cmp(args, "short")) {
type = InputTypeShort;
} else if(!furi_string_cmp(args, "long")) {
type = InputTypeLong;
} else {
break;
}
parsed = true;
} while(false);
if(parsed) { //-V547
fake_input(input, key, type);
} else {
input_cli_send_print_usage();
}
furi_string_free(key_str);
}
void input_cli(Cli* cli, FuriString* args, void* context) {
furi_assert(cli);
furi_assert(context);
Input* input = context;
FuriString* cmd;
cmd = furi_string_alloc();
do {
if(!args_read_string_and_trim(args, cmd)) {
input_cli_usage();
break;
}
if(furi_string_cmp_str(cmd, "dump") == 0) {
input_cli_dump(cli, args, input);
break;
}
if(furi_string_cmp_str(cmd, "keyboard") == 0) {
input_cli_keyboard(cli, args, input);
break;
}
if(furi_string_cmp_str(cmd, "send") == 0) {
input_cli_send(cli, args, input);
break;
}
input_cli_usage();
} while(false);
furi_string_free(cmd);
}
#include <flipper_application/flipper_application.h>
static const FlipperAppPluginDescriptor plugin_descriptor = {
.appid = "input_cli",
.ep_api_version = 1,
.entry_point = &input_cli,
};
const FlipperAppPluginDescriptor* input_cli_plugin_ep() {
return &plugin_descriptor;
}