bt_hid_media.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. #include "bt_hid_media.h"
  2. #include <furi.h>
  3. #include <furi_hal_bt_hid.h>
  4. #include <furi_hal_usb_hid.h>
  5. #include <gui/elements.h>
  6. struct BtHidMedia {
  7. View* view;
  8. };
  9. typedef struct {
  10. bool left_pressed;
  11. bool up_pressed;
  12. bool right_pressed;
  13. bool down_pressed;
  14. bool ok_pressed;
  15. bool connected;
  16. } BtHidMediaModel;
  17. static void bt_hid_media_draw_arrow(Canvas* canvas, uint8_t x, uint8_t y, CanvasDirection dir) {
  18. canvas_draw_triangle(canvas, x, y, 5, 3, dir);
  19. if(dir == CanvasDirectionBottomToTop) {
  20. canvas_draw_dot(canvas, x, y - 1);
  21. } else if(dir == CanvasDirectionTopToBottom) {
  22. canvas_draw_dot(canvas, x, y + 1);
  23. } else if(dir == CanvasDirectionRightToLeft) {
  24. canvas_draw_dot(canvas, x - 1, y);
  25. } else if(dir == CanvasDirectionLeftToRight) {
  26. canvas_draw_dot(canvas, x + 1, y);
  27. }
  28. }
  29. static void bt_hid_media_draw_callback(Canvas* canvas, void* context) {
  30. furi_assert(context);
  31. BtHidMediaModel* model = context;
  32. // Header
  33. canvas_set_font(canvas, FontPrimary);
  34. elements_multiline_text_aligned(canvas, 9, 3, AlignLeft, AlignTop, "Media player");
  35. canvas_set_font(canvas, FontSecondary);
  36. // Connected status
  37. if(model->connected) {
  38. canvas_draw_icon(canvas, 23, 17, &I_Ble_connected_38x34);
  39. elements_multiline_text_aligned(canvas, 35, 61, AlignCenter, AlignBottom, "Connected");
  40. } else {
  41. canvas_draw_icon(canvas, 23, 17, &I_Ble_disconnected_24x34);
  42. elements_multiline_text_aligned(canvas, 35, 61, AlignCenter, AlignBottom, "Disconnected");
  43. }
  44. // Keypad circles
  45. canvas_draw_icon(canvas, 76, 8, &I_Circles_47x47);
  46. // Up
  47. if(model->up_pressed) {
  48. canvas_draw_icon(canvas, 93, 9, &I_Pressed_Button_13x13);
  49. canvas_set_color(canvas, ColorWhite);
  50. }
  51. canvas_draw_icon(canvas, 96, 12, &I_Volup_8x6);
  52. canvas_set_color(canvas, ColorBlack);
  53. // Down
  54. if(model->down_pressed) {
  55. canvas_draw_icon(canvas, 93, 41, &I_Pressed_Button_13x13);
  56. canvas_set_color(canvas, ColorWhite);
  57. }
  58. canvas_draw_icon(canvas, 96, 45, &I_Voldwn_6x6);
  59. canvas_set_color(canvas, ColorBlack);
  60. // Left
  61. if(model->left_pressed) {
  62. canvas_draw_icon(canvas, 77, 25, &I_Pressed_Button_13x13);
  63. canvas_set_color(canvas, ColorWhite);
  64. }
  65. bt_hid_media_draw_arrow(canvas, 82, 31, CanvasDirectionRightToLeft);
  66. bt_hid_media_draw_arrow(canvas, 86, 31, CanvasDirectionRightToLeft);
  67. canvas_set_color(canvas, ColorBlack);
  68. // Right
  69. if(model->right_pressed) {
  70. canvas_draw_icon(canvas, 109, 25, &I_Pressed_Button_13x13);
  71. canvas_set_color(canvas, ColorWhite);
  72. }
  73. bt_hid_media_draw_arrow(canvas, 112, 31, CanvasDirectionLeftToRight);
  74. bt_hid_media_draw_arrow(canvas, 116, 31, CanvasDirectionLeftToRight);
  75. canvas_set_color(canvas, ColorBlack);
  76. // Ok
  77. if(model->ok_pressed) {
  78. canvas_draw_icon(canvas, 93, 25, &I_Pressed_Button_13x13);
  79. canvas_set_color(canvas, ColorWhite);
  80. }
  81. bt_hid_media_draw_arrow(canvas, 96, 31, CanvasDirectionLeftToRight);
  82. canvas_draw_line(canvas, 100, 29, 100, 33);
  83. canvas_draw_line(canvas, 102, 29, 102, 33);
  84. }
  85. static void bt_hid_media_process_press(BtHidMedia* bt_hid_media, InputEvent* event) {
  86. with_view_model(
  87. bt_hid_media->view, (BtHidMediaModel * model) {
  88. if(event->key == InputKeyUp) {
  89. model->up_pressed = true;
  90. furi_hal_bt_hid_media_press(FuriHalBtHidMediaVolumeUp);
  91. } else if(event->key == InputKeyDown) {
  92. model->down_pressed = true;
  93. furi_hal_bt_hid_media_press(FuriHalBtHidMediaVolumeDown);
  94. } else if(event->key == InputKeyLeft) {
  95. model->left_pressed = true;
  96. furi_hal_bt_hid_media_press(FuriHalBtHidMediaScanPrevious);
  97. } else if(event->key == InputKeyRight) {
  98. model->right_pressed = true;
  99. furi_hal_bt_hid_media_press(FuriHalBtHidMediaScanNext);
  100. } else if(event->key == InputKeyOk) {
  101. model->ok_pressed = true;
  102. furi_hal_bt_hid_media_press(FuriHalBtHidMediaPlayPause);
  103. }
  104. return true;
  105. });
  106. }
  107. static void bt_hid_media_process_release(BtHidMedia* bt_hid_media, InputEvent* event) {
  108. with_view_model(
  109. bt_hid_media->view, (BtHidMediaModel * model) {
  110. if(event->key == InputKeyUp) {
  111. model->up_pressed = false;
  112. furi_hal_bt_hid_media_release(FuriHalBtHidMediaVolumeUp);
  113. } else if(event->key == InputKeyDown) {
  114. model->down_pressed = false;
  115. furi_hal_bt_hid_media_release(FuriHalBtHidMediaVolumeDown);
  116. } else if(event->key == InputKeyLeft) {
  117. model->left_pressed = false;
  118. furi_hal_bt_hid_media_release(FuriHalBtHidMediaScanPrevious);
  119. } else if(event->key == InputKeyRight) {
  120. model->right_pressed = false;
  121. furi_hal_bt_hid_media_release(FuriHalBtHidMediaScanNext);
  122. } else if(event->key == InputKeyOk) {
  123. model->ok_pressed = false;
  124. furi_hal_bt_hid_media_release(FuriHalBtHidMediaPlayPause);
  125. }
  126. return true;
  127. });
  128. }
  129. static bool bt_hid_media_input_callback(InputEvent* event, void* context) {
  130. furi_assert(context);
  131. BtHidMedia* bt_hid_media = context;
  132. bool consumed = false;
  133. if(event->type == InputTypePress) {
  134. bt_hid_media_process_press(bt_hid_media, event);
  135. consumed = true;
  136. } else if(event->type == InputTypeRelease) {
  137. bt_hid_media_process_release(bt_hid_media, event);
  138. consumed = true;
  139. } else if(event->type == InputTypeShort) {
  140. if(event->key == InputKeyBack) {
  141. furi_hal_bt_hid_media_release_all();
  142. }
  143. }
  144. return consumed;
  145. }
  146. BtHidMedia* bt_hid_media_alloc() {
  147. BtHidMedia* bt_hid_media = malloc(sizeof(BtHidMedia));
  148. bt_hid_media->view = view_alloc();
  149. view_set_context(bt_hid_media->view, bt_hid_media);
  150. view_allocate_model(bt_hid_media->view, ViewModelTypeLocking, sizeof(BtHidMediaModel));
  151. view_set_draw_callback(bt_hid_media->view, bt_hid_media_draw_callback);
  152. view_set_input_callback(bt_hid_media->view, bt_hid_media_input_callback);
  153. return bt_hid_media;
  154. }
  155. void bt_hid_media_free(BtHidMedia* bt_hid_media) {
  156. furi_assert(bt_hid_media);
  157. view_free(bt_hid_media->view);
  158. free(bt_hid_media);
  159. }
  160. View* bt_hid_media_get_view(BtHidMedia* bt_hid_media) {
  161. furi_assert(bt_hid_media);
  162. return bt_hid_media->view;
  163. }
  164. void bt_hid_media_set_connected_status(BtHidMedia* bt_hid_media, bool connected) {
  165. furi_assert(bt_hid_media);
  166. with_view_model(
  167. bt_hid_media->view, (BtHidMediaModel * model) {
  168. model->connected = connected;
  169. return true;
  170. });
  171. }