air_mouse_app.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <gui/gui.h>
  4. #include <dialogs/dialogs.h>
  5. #include "imu_mouse.h"
  6. #include "air_mouse_ofw_icons.h"
  7. #define TAG "SensorModule"
  8. typedef struct {
  9. Gui* gui;
  10. ViewPort* view_port;
  11. FuriMessageQueue* input_queue;
  12. FuriHalSpiBusHandle* icm42688p_device;
  13. ICM42688P* icm42688p;
  14. bool icm42688p_valid;
  15. ImuThread* imu_thread;
  16. } SensorModuleApp;
  17. static void render_callback(Canvas* canvas, void* ctx) {
  18. UNUSED(ctx);
  19. canvas_clear(canvas);
  20. canvas_set_color(canvas, ColorBlack);
  21. canvas_draw_icon(canvas, 64 + 14, 8, &I_Circles_47x47);
  22. canvas_draw_icon(canvas, 83 + 14, 27, &I_Left_mouse_icon_9x9);
  23. canvas_draw_icon(canvas, 83 + 14, 11, &I_Right_mouse_icon_9x9);
  24. canvas_set_font(canvas, FontPrimary);
  25. canvas_draw_str(canvas, 0, 14, "Air Mouse");
  26. canvas_set_font(canvas, FontSecondary);
  27. canvas_draw_str(canvas, 0, 56, "Press Back to exit");
  28. }
  29. static void input_callback(InputEvent* input_event, void* ctx) {
  30. SensorModuleApp* app = ctx;
  31. furi_message_queue_put(app->input_queue, input_event, 0);
  32. }
  33. static SensorModuleApp* sensor_module_alloc(void) {
  34. SensorModuleApp* app = malloc(sizeof(SensorModuleApp));
  35. app->icm42688p_device = malloc(sizeof(FuriHalSpiBusHandle));
  36. memcpy(app->icm42688p_device, &furi_hal_spi_bus_handle_external, sizeof(FuriHalSpiBusHandle));
  37. app->icm42688p_device->cs = &gpio_ext_pc3;
  38. app->icm42688p = icm42688p_alloc(app->icm42688p_device, &gpio_ext_pb2);
  39. app->icm42688p_valid = icm42688p_init(app->icm42688p);
  40. app->input_queue = furi_message_queue_alloc(8, sizeof(InputEvent));
  41. app->view_port = view_port_alloc();
  42. view_port_draw_callback_set(app->view_port, render_callback, app);
  43. view_port_input_callback_set(app->view_port, input_callback, app);
  44. app->gui = furi_record_open(RECORD_GUI);
  45. gui_add_view_port(app->gui, app->view_port, GuiLayerFullscreen);
  46. return app;
  47. }
  48. static void sensor_module_free(SensorModuleApp* app) {
  49. gui_remove_view_port(app->gui, app->view_port);
  50. furi_record_close(RECORD_GUI);
  51. view_port_free(app->view_port);
  52. furi_message_queue_free(app->input_queue);
  53. if(app->imu_thread) {
  54. imu_stop(app->imu_thread);
  55. app->imu_thread = NULL;
  56. }
  57. if(!icm42688p_deinit(app->icm42688p)) {
  58. FURI_LOG_E(TAG, "Failed to deinitialize ICM42688P");
  59. }
  60. icm42688p_free(app->icm42688p);
  61. free(app->icm42688p_device);
  62. free(app);
  63. }
  64. int32_t air_mouse_app(void* arg) {
  65. UNUSED(arg);
  66. SensorModuleApp* app = sensor_module_alloc();
  67. if(!app->icm42688p_valid) {
  68. DialogsApp* dialogs = furi_record_open(RECORD_DIALOGS);
  69. DialogMessage* message = dialog_message_alloc();
  70. dialog_message_set_header(message, "Sensor Module error", 63, 0, AlignCenter, AlignTop);
  71. dialog_message_set_text(message, "Module not conntected", 63, 30, AlignCenter, AlignTop);
  72. dialog_message_show(dialogs, message);
  73. dialog_message_free(message);
  74. furi_record_close(RECORD_DIALOGS);
  75. sensor_module_free(app);
  76. return 0;
  77. }
  78. view_port_update(app->view_port);
  79. app->imu_thread = imu_start(app->icm42688p);
  80. while(1) {
  81. InputEvent input;
  82. if(furi_message_queue_get(app->input_queue, &input, FuriWaitForever) == FuriStatusOk) {
  83. if((input.key == InputKeyBack) && (input.type == InputTypeShort)) {
  84. break;
  85. } else if(input.key == InputKeyOk) {
  86. if(input.type == InputTypePress) {
  87. imu_mouse_key_press(app->imu_thread, ImuMouseKeyLeft, true);
  88. } else if(input.type == InputTypeRelease) {
  89. imu_mouse_key_press(app->imu_thread, ImuMouseKeyLeft, false);
  90. }
  91. } else if(input.key == InputKeyUp) {
  92. if(input.type == InputTypePress) {
  93. imu_mouse_key_press(app->imu_thread, ImuMouseKeyRight, true);
  94. } else if(input.type == InputTypeRelease) {
  95. imu_mouse_key_press(app->imu_thread, ImuMouseKeyRight, false);
  96. }
  97. }
  98. }
  99. }
  100. sensor_module_free(app);
  101. return 0;
  102. }