hid_ptt.c 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. #include "hid_ptt.h"
  2. #include <gui/elements.h>
  3. #include <notification/notification_messages.h>
  4. #include "../hid.h"
  5. #include "../views.h"
  6. #include "hid_icons.h"
  7. #define TAG "HidPtt"
  8. struct HidPtt {
  9. View* view;
  10. Hid* hid;
  11. };
  12. typedef struct {
  13. bool left_pressed;
  14. bool up_pressed;
  15. bool right_pressed;
  16. bool down_pressed;
  17. bool ok_pressed;
  18. bool back_pressed;
  19. bool connected;
  20. bool muted;
  21. HidTransport transport;
  22. } HidPttModel;
  23. static void hid_ptt_draw_arrow(Canvas* canvas, uint8_t x, uint8_t y, CanvasDirection dir) {
  24. canvas_draw_triangle(canvas, x, y, 5, 3, dir);
  25. if(dir == CanvasDirectionBottomToTop) {
  26. canvas_draw_line(canvas, x, y + 6, x, y - 1);
  27. } else if(dir == CanvasDirectionTopToBottom) {
  28. canvas_draw_line(canvas, x, y - 6, x, y + 1);
  29. } else if(dir == CanvasDirectionRightToLeft) {
  30. canvas_draw_line(canvas, x + 6, y, x - 1, y);
  31. } else if(dir == CanvasDirectionLeftToRight) {
  32. canvas_draw_line(canvas, x - 6, y, x + 1, y);
  33. }
  34. }
  35. static void hid_ptt_draw_callback(Canvas* canvas, void* context) {
  36. furi_assert(context);
  37. HidPttModel* model = context;
  38. // Header
  39. canvas_set_font(canvas, FontPrimary);
  40. if(model->transport == HidTransportBle) {
  41. if(model->connected) {
  42. canvas_draw_icon(canvas, 0, 0, &I_Ble_connected_15x15);
  43. } else {
  44. canvas_draw_icon(canvas, 0, 0, &I_Ble_disconnected_15x15);
  45. }
  46. }
  47. canvas_draw_icon(canvas, 2, 18, &I_Pin_back_arrow_10x8);
  48. canvas_set_font(canvas, FontSecondary);
  49. elements_multiline_text_aligned(canvas, 15, 19, AlignLeft, AlignTop, "Hold to exit");
  50. const uint8_t x_2 = 23;
  51. const uint8_t x_1 = 2;
  52. const uint8_t x_3 = 44;
  53. const uint8_t y_1 = 44;
  54. const uint8_t y_2 = 65;
  55. // Up
  56. canvas_draw_icon(canvas, x_2, y_1, &I_Button_18x18);
  57. if(model->up_pressed) {
  58. elements_slightly_rounded_box(canvas, x_2 + 3, y_1 + 2, 13, 13);
  59. canvas_set_color(canvas, ColorWhite);
  60. }
  61. hid_ptt_draw_arrow(canvas, x_2 + 9, y_1 + 6, CanvasDirectionBottomToTop);
  62. canvas_set_color(canvas, ColorBlack);
  63. // Down
  64. canvas_draw_icon(canvas, x_2, y_2, &I_Button_18x18);
  65. if(model->down_pressed) {
  66. elements_slightly_rounded_box(canvas, x_2 + 3, y_2 + 2, 13, 13);
  67. canvas_set_color(canvas, ColorWhite);
  68. }
  69. hid_ptt_draw_arrow(canvas, x_2 + 9, y_2 + 10, CanvasDirectionTopToBottom);
  70. canvas_set_color(canvas, ColorBlack);
  71. // Left
  72. canvas_draw_icon(canvas, x_1, y_2, &I_Button_18x18);
  73. if(model->left_pressed) {
  74. elements_slightly_rounded_box(canvas, x_1 + 3, y_2 + 2, 13, 13);
  75. canvas_set_color(canvas, ColorWhite);
  76. }
  77. hid_ptt_draw_arrow(canvas, x_1 + 7, y_2 + 8, CanvasDirectionRightToLeft);
  78. canvas_set_color(canvas, ColorBlack);
  79. // Right
  80. canvas_draw_icon(canvas, x_3, y_2, &I_Button_18x18);
  81. if(model->right_pressed) {
  82. elements_slightly_rounded_box(canvas, x_3 + 3, y_2 + 2, 13, 13);
  83. canvas_set_color(canvas, ColorWhite);
  84. }
  85. hid_ptt_draw_arrow(canvas, x_3 + 11, y_2 + 8, CanvasDirectionLeftToRight);
  86. canvas_set_color(canvas, ColorBlack);
  87. // Ok
  88. canvas_draw_icon(canvas, 2, 86, &I_Space_60x18);
  89. if(model->ok_pressed) {
  90. elements_slightly_rounded_box(canvas, 5, 88, 55, 13);
  91. canvas_set_color(canvas, ColorWhite);
  92. }
  93. canvas_draw_icon(canvas, 11, 90, &I_Ok_btn_9x9);
  94. elements_multiline_text_aligned(canvas, 26, 98, AlignLeft, AlignBottom, "Space");
  95. canvas_set_color(canvas, ColorBlack);
  96. // Back
  97. canvas_draw_icon(canvas, 2, 107, &I_Space_60x18);
  98. if(model->back_pressed) {
  99. elements_slightly_rounded_box(canvas, 5, 109, 55, 13);
  100. canvas_set_color(canvas, ColorWhite);
  101. }
  102. canvas_draw_icon(canvas, 11, 111, &I_Pin_back_arrow_10x8);
  103. elements_multiline_text_aligned(canvas, 26, 119, AlignLeft, AlignBottom, "Back");
  104. }
  105. static void hid_ptt_process(HidPtt* hid_ptt, InputEvent* event) {
  106. with_view_model(
  107. hid_ptt->view,
  108. HidPttModel * model,
  109. {
  110. if(event->type == InputTypePress) {
  111. if(event->key == InputKeyUp) {
  112. model->up_pressed = true;
  113. hid_hal_consumer_key_press(hid_ptt->hid, HID_CONSUMER_VOLUME_INCREMENT);
  114. } else if(event->key == InputKeyDown) {
  115. model->down_pressed = true;
  116. hid_hal_consumer_key_press(hid_ptt->hid, HID_CONSUMER_VOLUME_DECREMENT);
  117. } else if(event->key == InputKeyLeft) {
  118. model->left_pressed = true;
  119. } else if(event->key == InputKeyRight) {
  120. model->right_pressed = true;
  121. // } else if(event->key == InputKeyOk) {
  122. // model->ok_pressed = true;
  123. } else if(event->key == InputKeyBack) {
  124. hid_hal_keyboard_press(hid_ptt->hid, HID_KEYBOARD_SPACEBAR);
  125. model->back_pressed = true;
  126. }
  127. } else if(event->type == InputTypeRelease) {
  128. if(event->key == InputKeyUp) {
  129. model->up_pressed = false;
  130. hid_hal_consumer_key_release(hid_ptt->hid, HID_CONSUMER_VOLUME_INCREMENT);
  131. } else if(event->key == InputKeyDown) {
  132. model->down_pressed = false;
  133. hid_hal_consumer_key_release(hid_ptt->hid, HID_CONSUMER_VOLUME_DECREMENT);
  134. } else if(event->key == InputKeyLeft) {
  135. model->left_pressed = false;
  136. } else if(event->key == InputKeyRight) {
  137. model->right_pressed = false;
  138. // } else if(event->key == InputKeyOk) {
  139. // model->ok_pressed = false;
  140. } else if(event->key == InputKeyBack) {
  141. model->back_pressed = false;
  142. hid_hal_keyboard_release(hid_ptt->hid, HID_KEYBOARD_SPACEBAR);
  143. }
  144. } else if(event->type == InputTypeShort) {
  145. if(event->key == InputKeyOk) {
  146. //  + d
  147. hid_hal_keyboard_press(hid_ptt->hid, KEY_MOD_LEFT_GUI | HID_KEYBOARD_D);
  148. hid_hal_keyboard_release(hid_ptt->hid, KEY_MOD_LEFT_GUI | HID_KEYBOARD_D);
  149. }
  150. if(event->key == InputKeyRight) {
  151. //  + d
  152. hid_hal_keyboard_press(hid_ptt->hid, KEY_MOD_LEFT_GUI | HID_KEYBOARD_E);
  153. hid_hal_keyboard_release(hid_ptt->hid, KEY_MOD_LEFT_GUI | HID_KEYBOARD_E);
  154. }
  155. } else if(event->type == InputTypeLong && event->key == InputKeyLeft) {
  156. model->left_pressed = false;
  157. hid_hal_keyboard_release_all(hid_ptt->hid);
  158. view_dispatcher_switch_to_view(hid_ptt->hid->view_dispatcher, HidViewSubmenu);
  159. // sequence_double_vibro to notify that we quit PTT
  160. notification_message(hid_ptt->hid->notifications, &sequence_double_vibro);
  161. }
  162. },
  163. true);
  164. }
  165. static bool hid_ptt_input_callback(InputEvent* event, void* context) {
  166. furi_assert(context);
  167. HidPtt* hid_ptt = context;
  168. bool consumed = true;
  169. hid_ptt_process(hid_ptt, event);
  170. return consumed;
  171. }
  172. HidPtt* hid_ptt_alloc(Hid* hid) {
  173. HidPtt* hid_ptt = malloc(sizeof(HidPtt));
  174. hid_ptt->view = view_alloc();
  175. hid_ptt->hid = hid;
  176. view_set_context(hid_ptt->view, hid_ptt);
  177. view_allocate_model(hid_ptt->view, ViewModelTypeLocking, sizeof(HidPttModel));
  178. view_set_draw_callback(hid_ptt->view, hid_ptt_draw_callback);
  179. view_set_input_callback(hid_ptt->view, hid_ptt_input_callback);
  180. view_set_orientation(hid_ptt->view, ViewOrientationVerticalFlip);
  181. with_view_model(
  182. hid_ptt->view, HidPttModel * model, {
  183. model->transport = hid->transport;
  184. model->muted = false; // assume we're not muted
  185. }, true);
  186. return hid_ptt;
  187. }
  188. void hid_ptt_free(HidPtt* hid_ptt) {
  189. furi_assert(hid_ptt);
  190. view_free(hid_ptt->view);
  191. free(hid_ptt);
  192. }
  193. View* hid_ptt_get_view(HidPtt* hid_ptt) {
  194. furi_assert(hid_ptt);
  195. return hid_ptt->view;
  196. }
  197. void hid_ptt_set_connected_status(HidPtt* hid_ptt, bool connected) {
  198. furi_assert(hid_ptt);
  199. with_view_model(
  200. hid_ptt->view, HidPttModel * model, { model->connected = connected; }, true);
  201. }