/* * USB Host Keyboard Example with Display Support */ #include "pico/stdlib.h" #include "pico/time.h" #include "pico/multicore.h" #include "tusb.h" #include #include #include #include #include "display.h" #include "commands/echo.h" #include "keyboard_input.h" #include "command_processor.h" #include "epaper_manager.h" // Holds last echoed line for display static char g_last_echo[128] = ""; // Global DisplayManager pointer static DisplayManager* g_display_manager = nullptr; // Keyboard buffer #define MAX_INPUT_LEN 64 static char g_input_buffer[MAX_INPUT_LEN]; static int g_buffer_index = 0; static bool execute_command(CommandAction action) { switch (action) { case CMD_REFRESH: printf("Command: /refresh\n"); epaper_force_refresh(); return true; case CMD_CLEAR: printf("Command: /clear\n"); epaper_clear(); return true; case CMD_WIFI: printf("Command: /wifisetup (Not implemented)\n"); return true; default: return false; } } static void process_kbd_report(hid_keyboard_report_t const *report) { KeyEvent event; if (parse_keyboard_report(report, &event)) { if (event.is_backspace) { if (g_buffer_index > 0) { printf("\b \b"); g_buffer_index--; g_input_buffer[g_buffer_index] = '\0'; // Update OLED if (g_display_manager) g_display_manager->refresh(g_input_buffer, g_last_echo); // Update e-Paper (in-place) epaper_send_update(g_input_buffer, false); } } else if (event.is_enter) { printf("\n"); CommandAction action = parse_command(g_input_buffer); if (execute_command(action)) { // Command handled, clear buffer but don't commit line g_buffer_index = 0; g_input_buffer[0] = '\0'; if (g_display_manager) { g_display_manager->set_last_echo("Command Executed"); g_display_manager->refresh("", "Command Executed"); } } else { // Update e-Paper (commit line) epaper_send_update(g_input_buffer, true); // Save to last echo strncpy(g_last_echo, g_input_buffer, sizeof(g_last_echo) - 1); g_last_echo[sizeof(g_last_echo) - 1] = '\0'; // Clear buffer g_buffer_index = 0; g_input_buffer[0] = '\0'; // Update OLED if (g_display_manager) { g_display_manager->set_last_echo(g_last_echo); g_display_manager->refresh("", g_last_echo); } } } else if (event.is_printable) { printf("%c", event.ascii); if (g_buffer_index < MAX_INPUT_LEN - 1) { g_input_buffer[g_buffer_index++] = event.ascii; g_input_buffer[g_buffer_index] = '\0'; // Update OLED if (g_display_manager) g_display_manager->refresh(g_input_buffer, g_last_echo); // Update e-Paper on every keystroke epaper_send_update(g_input_buffer, false); } } } } //--------------------------------------------------------------------+ // TinyUSB Callbacks //--------------------------------------------------------------------+ static bool g_keyboard_mounted = false; // Invoked when device is suspended void tuh_device_suspend_cb(uint8_t dev_addr) { printf("Device address = %d suspended\r\n", dev_addr); } // Invoked when device is resumed void tuh_device_resume_cb(uint8_t dev_addr) { printf("Device address = %d resumed\r\n", dev_addr); } void tuh_hid_mount_cb(uint8_t dev_addr, uint8_t instance, uint8_t const* desc_report, uint16_t desc_len) { printf("HID device address = %d, instance = %d is mounted\r\n", dev_addr, instance); uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance); if (itf_protocol == HID_ITF_PROTOCOL_KEYBOARD) { printf("Keyboard mounted\r\n"); g_keyboard_mounted = true; if (g_display_manager) { g_display_manager->refresh("Keyboard Connected", nullptr); } if (!tuh_hid_receive_report(dev_addr, instance)) { printf("Error: cannot request to receive report\r\n"); } } } void tuh_hid_umount_cb(uint8_t dev_addr, uint8_t instance) { printf("HID device address = %d, instance = %d is unmounted\r\n", dev_addr, instance); g_keyboard_mounted = false; // Reset input buffer g_buffer_index = 0; g_input_buffer[0] = '\0'; if (g_display_manager) { g_display_manager->refresh("Waiting for Keyboard", nullptr); } } void tuh_hid_report_received_cb(uint8_t dev_addr, uint8_t instance, uint8_t const* report, uint16_t len) { uint8_t const itf_protocol = tuh_hid_interface_protocol(dev_addr, instance); if (itf_protocol == HID_ITF_PROTOCOL_KEYBOARD) { process_kbd_report((hid_keyboard_report_t const*) report); } // continue to request to receive report if (!tuh_hid_receive_report(dev_addr, instance)) { printf("Error: cannot request to receive report\r\n"); } } int main() { stdio_init_all(); sleep_ms(3000); // Give time for power to settle and serial to connect printf("System Booting...\n"); // Launch display initialization on core 1 epaper_start_background_thread(); DisplayManager display; display.init(); g_display_manager = &display; display.refresh("Waiting for Keyboard", nullptr); printf("Initializing TinyUSB Host...\n"); tuh_init(BOARD_TUH_RHPORT); printf("TinyUSB Host Initialized.\n"); uint32_t last_print = 0; while (true) { tuh_task(); uint32_t now = to_ms_since_boot(get_absolute_time()); if (now - last_print > 5000) { printf("Heartbeat: %u, Mounted: %d\n", now, g_keyboard_mounted); last_print = now; } } return 0; }