hid_keynote.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. #include "hid_keynote.h"
  2. #include <gui/elements.h>
  3. #include "../hid.h"
  4. #include "hid_icons.h"
  5. #define TAG "HidKeynote"
  6. struct HidKeynote {
  7. View* view;
  8. Hid* hid;
  9. };
  10. typedef struct {
  11. bool left_pressed;
  12. bool up_pressed;
  13. bool right_pressed;
  14. bool down_pressed;
  15. bool ok_pressed;
  16. bool back_pressed;
  17. bool connected;
  18. HidTransport transport;
  19. } HidKeynoteModel;
  20. static void hid_keynote_draw_arrow(Canvas* canvas, uint8_t x, uint8_t y, CanvasDirection dir) {
  21. canvas_draw_triangle(canvas, x, y, 5, 3, dir);
  22. if(dir == CanvasDirectionBottomToTop) {
  23. canvas_draw_line(canvas, x, y + 6, x, y - 1);
  24. } else if(dir == CanvasDirectionTopToBottom) {
  25. canvas_draw_line(canvas, x, y - 6, x, y + 1);
  26. } else if(dir == CanvasDirectionRightToLeft) {
  27. canvas_draw_line(canvas, x + 6, y, x - 1, y);
  28. } else if(dir == CanvasDirectionLeftToRight) {
  29. canvas_draw_line(canvas, x - 6, y, x + 1, y);
  30. }
  31. }
  32. static void hid_keynote_draw_callback(Canvas* canvas, void* context) {
  33. furi_assert(context);
  34. HidKeynoteModel* model = context;
  35. // Header
  36. if(model->transport == HidTransportBle) {
  37. if(model->connected) {
  38. canvas_draw_icon(canvas, 0, 0, &I_Ble_connected_15x15);
  39. } else {
  40. canvas_draw_icon(canvas, 0, 0, &I_Ble_disconnected_15x15);
  41. }
  42. }
  43. canvas_set_font(canvas, FontPrimary);
  44. elements_multiline_text_aligned(canvas, 17, 3, AlignLeft, AlignTop, "Keynote");
  45. canvas_draw_icon(canvas, 68, 2, &I_Pin_back_arrow_10x8);
  46. canvas_set_font(canvas, FontSecondary);
  47. elements_multiline_text_aligned(canvas, 127, 3, AlignRight, AlignTop, "Hold to exit");
  48. // Up
  49. canvas_draw_icon(canvas, 21, 24, &I_Button_18x18);
  50. if(model->up_pressed) {
  51. elements_slightly_rounded_box(canvas, 24, 26, 13, 13);
  52. canvas_set_color(canvas, ColorWhite);
  53. }
  54. hid_keynote_draw_arrow(canvas, 30, 30, CanvasDirectionBottomToTop);
  55. canvas_set_color(canvas, ColorBlack);
  56. // Down
  57. canvas_draw_icon(canvas, 21, 45, &I_Button_18x18);
  58. if(model->down_pressed) {
  59. elements_slightly_rounded_box(canvas, 24, 47, 13, 13);
  60. canvas_set_color(canvas, ColorWhite);
  61. }
  62. hid_keynote_draw_arrow(canvas, 30, 55, CanvasDirectionTopToBottom);
  63. canvas_set_color(canvas, ColorBlack);
  64. // Left
  65. canvas_draw_icon(canvas, 0, 45, &I_Button_18x18);
  66. if(model->left_pressed) {
  67. elements_slightly_rounded_box(canvas, 3, 47, 13, 13);
  68. canvas_set_color(canvas, ColorWhite);
  69. }
  70. hid_keynote_draw_arrow(canvas, 7, 53, CanvasDirectionRightToLeft);
  71. canvas_set_color(canvas, ColorBlack);
  72. // Right
  73. canvas_draw_icon(canvas, 42, 45, &I_Button_18x18);
  74. if(model->right_pressed) {
  75. elements_slightly_rounded_box(canvas, 45, 47, 13, 13);
  76. canvas_set_color(canvas, ColorWhite);
  77. }
  78. hid_keynote_draw_arrow(canvas, 53, 53, CanvasDirectionLeftToRight);
  79. canvas_set_color(canvas, ColorBlack);
  80. // Ok
  81. canvas_draw_icon(canvas, 63, 25, &I_Space_65x18);
  82. if(model->ok_pressed) {
  83. elements_slightly_rounded_box(canvas, 66, 27, 60, 13);
  84. canvas_set_color(canvas, ColorWhite);
  85. }
  86. canvas_draw_icon(canvas, 74, 29, &I_Ok_btn_9x9);
  87. elements_multiline_text_aligned(canvas, 91, 36, AlignLeft, AlignBottom, "Space");
  88. canvas_set_color(canvas, ColorBlack);
  89. // Back
  90. canvas_draw_icon(canvas, 63, 45, &I_Space_65x18);
  91. if(model->back_pressed) {
  92. elements_slightly_rounded_box(canvas, 66, 47, 60, 13);
  93. canvas_set_color(canvas, ColorWhite);
  94. }
  95. canvas_draw_icon(canvas, 74, 49, &I_Pin_back_arrow_10x8);
  96. elements_multiline_text_aligned(canvas, 91, 57, AlignLeft, AlignBottom, "Back");
  97. }
  98. static void hid_keynote_process(HidKeynote* hid_keynote, InputEvent* event) {
  99. with_view_model(
  100. hid_keynote->view,
  101. HidKeynoteModel * model,
  102. {
  103. if(event->type == InputTypePress) {
  104. if(event->key == InputKeyUp) {
  105. model->up_pressed = true;
  106. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_UP_ARROW);
  107. } else if(event->key == InputKeyDown) {
  108. model->down_pressed = true;
  109. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_DOWN_ARROW);
  110. } else if(event->key == InputKeyLeft) {
  111. model->left_pressed = true;
  112. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_LEFT_ARROW);
  113. } else if(event->key == InputKeyRight) {
  114. model->right_pressed = true;
  115. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_RIGHT_ARROW);
  116. } else if(event->key == InputKeyOk) {
  117. model->ok_pressed = true;
  118. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_SPACEBAR);
  119. } else if(event->key == InputKeyBack) {
  120. model->back_pressed = true;
  121. }
  122. } else if(event->type == InputTypeRelease) {
  123. if(event->key == InputKeyUp) {
  124. model->up_pressed = false;
  125. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_UP_ARROW);
  126. } else if(event->key == InputKeyDown) {
  127. model->down_pressed = false;
  128. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_DOWN_ARROW);
  129. } else if(event->key == InputKeyLeft) {
  130. model->left_pressed = false;
  131. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_LEFT_ARROW);
  132. } else if(event->key == InputKeyRight) {
  133. model->right_pressed = false;
  134. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_RIGHT_ARROW);
  135. } else if(event->key == InputKeyOk) {
  136. model->ok_pressed = false;
  137. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_SPACEBAR);
  138. } else if(event->key == InputKeyBack) {
  139. model->back_pressed = false;
  140. }
  141. } else if(event->type == InputTypeShort) {
  142. if(event->key == InputKeyBack) {
  143. hid_hal_keyboard_press(hid_keynote->hid, HID_KEYBOARD_DELETE);
  144. hid_hal_keyboard_release(hid_keynote->hid, HID_KEYBOARD_DELETE);
  145. hid_hal_consumer_key_press(hid_keynote->hid, HID_CONSUMER_AC_BACK);
  146. hid_hal_consumer_key_release(hid_keynote->hid, HID_CONSUMER_AC_BACK);
  147. }
  148. }
  149. },
  150. true);
  151. }
  152. static bool hid_keynote_input_callback(InputEvent* event, void* context) {
  153. furi_assert(context);
  154. HidKeynote* hid_keynote = context;
  155. bool consumed = false;
  156. if(event->type == InputTypeLong && event->key == InputKeyBack) {
  157. hid_hal_keyboard_release_all(hid_keynote->hid);
  158. } else {
  159. hid_keynote_process(hid_keynote, event);
  160. consumed = true;
  161. }
  162. return consumed;
  163. }
  164. HidKeynote* hid_keynote_alloc(Hid* hid) {
  165. HidKeynote* hid_keynote = malloc(sizeof(HidKeynote));
  166. hid_keynote->view = view_alloc();
  167. hid_keynote->hid = hid;
  168. view_set_context(hid_keynote->view, hid_keynote);
  169. view_allocate_model(hid_keynote->view, ViewModelTypeLocking, sizeof(HidKeynoteModel));
  170. view_set_draw_callback(hid_keynote->view, hid_keynote_draw_callback);
  171. view_set_input_callback(hid_keynote->view, hid_keynote_input_callback);
  172. with_view_model(
  173. hid_keynote->view, HidKeynoteModel * model, { model->transport = hid->transport; }, true);
  174. return hid_keynote;
  175. }
  176. void hid_keynote_free(HidKeynote* hid_keynote) {
  177. furi_assert(hid_keynote);
  178. view_free(hid_keynote->view);
  179. free(hid_keynote);
  180. }
  181. View* hid_keynote_get_view(HidKeynote* hid_keynote) {
  182. furi_assert(hid_keynote);
  183. return hid_keynote->view;
  184. }
  185. void hid_keynote_set_connected_status(HidKeynote* hid_keynote, bool connected) {
  186. furi_assert(hid_keynote);
  187. with_view_model(
  188. hid_keynote->view, HidKeynoteModel * model, { model->connected = connected; }, true);
  189. }