bad_usb_view.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #include "bad_usb_view.h"
  2. #include "../bad_usb_script.h"
  3. #include <gui/elements.h>
  4. #include <assets_icons.h>
  5. #define MAX_NAME_LEN 64
  6. struct BadUsb {
  7. View* view;
  8. BadUsbOkCallback callback;
  9. void* context;
  10. };
  11. typedef struct {
  12. char file_name[MAX_NAME_LEN];
  13. BadUsbState state;
  14. uint8_t anim_frame;
  15. } BadUsbModel;
  16. static void bad_usb_draw_callback(Canvas* canvas, void* _model) {
  17. BadUsbModel* model = _model;
  18. FuriString* disp_str;
  19. disp_str = furi_string_alloc_set(model->file_name);
  20. elements_string_fit_width(canvas, disp_str, 128 - 2);
  21. canvas_set_font(canvas, FontSecondary);
  22. canvas_draw_str(canvas, 2, 8, furi_string_get_cstr(disp_str));
  23. furi_string_reset(disp_str);
  24. canvas_draw_icon(canvas, 22, 20, &I_UsbTree_48x22);
  25. if((model->state.state == BadUsbStateIdle) || (model->state.state == BadUsbStateDone)) {
  26. elements_button_center(canvas, "Run");
  27. } else if((model->state.state == BadUsbStateRunning) || (model->state.state == BadUsbStateDelay)) {
  28. elements_button_center(canvas, "Stop");
  29. }
  30. if(model->state.state == BadUsbStateNotConnected) {
  31. canvas_draw_icon(canvas, 4, 22, &I_Clock_18x18);
  32. canvas_set_font(canvas, FontPrimary);
  33. canvas_draw_str_aligned(canvas, 127, 27, AlignRight, AlignBottom, "Connect");
  34. canvas_draw_str_aligned(canvas, 127, 39, AlignRight, AlignBottom, "to USB");
  35. } else if(model->state.state == BadUsbStateFileError) {
  36. canvas_draw_icon(canvas, 4, 22, &I_Error_18x18);
  37. canvas_set_font(canvas, FontPrimary);
  38. canvas_draw_str_aligned(canvas, 127, 27, AlignRight, AlignBottom, "File");
  39. canvas_draw_str_aligned(canvas, 127, 39, AlignRight, AlignBottom, "ERROR");
  40. } else if(model->state.state == BadUsbStateScriptError) {
  41. canvas_draw_icon(canvas, 4, 22, &I_Error_18x18);
  42. canvas_set_font(canvas, FontPrimary);
  43. canvas_draw_str_aligned(canvas, 127, 33, AlignRight, AlignBottom, "ERROR:");
  44. canvas_set_font(canvas, FontSecondary);
  45. furi_string_printf(disp_str, "line %u", model->state.error_line);
  46. canvas_draw_str_aligned(
  47. canvas, 127, 46, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  48. furi_string_reset(disp_str);
  49. canvas_draw_str_aligned(canvas, 127, 56, AlignRight, AlignBottom, model->state.error);
  50. } else if(model->state.state == BadUsbStateIdle) {
  51. canvas_draw_icon(canvas, 4, 22, &I_Smile_18x18);
  52. canvas_set_font(canvas, FontBigNumbers);
  53. canvas_draw_str_aligned(canvas, 114, 36, AlignRight, AlignBottom, "0");
  54. canvas_draw_icon(canvas, 117, 22, &I_Percent_10x14);
  55. } else if(model->state.state == BadUsbStateRunning) {
  56. if(model->anim_frame == 0) {
  57. canvas_draw_icon(canvas, 4, 19, &I_EviSmile1_18x21);
  58. } else {
  59. canvas_draw_icon(canvas, 4, 19, &I_EviSmile2_18x21);
  60. }
  61. canvas_set_font(canvas, FontBigNumbers);
  62. furi_string_printf(
  63. disp_str, "%u", ((model->state.line_cur - 1) * 100) / model->state.line_nb);
  64. canvas_draw_str_aligned(
  65. canvas, 114, 36, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  66. furi_string_reset(disp_str);
  67. canvas_draw_icon(canvas, 117, 22, &I_Percent_10x14);
  68. } else if(model->state.state == BadUsbStateDone) {
  69. canvas_draw_icon(canvas, 4, 19, &I_EviSmile1_18x21);
  70. canvas_set_font(canvas, FontBigNumbers);
  71. canvas_draw_str_aligned(canvas, 114, 36, AlignRight, AlignBottom, "100");
  72. furi_string_reset(disp_str);
  73. canvas_draw_icon(canvas, 117, 22, &I_Percent_10x14);
  74. } else if(model->state.state == BadUsbStateDelay) {
  75. if(model->anim_frame == 0) {
  76. canvas_draw_icon(canvas, 4, 19, &I_EviWaiting1_18x21);
  77. } else {
  78. canvas_draw_icon(canvas, 4, 19, &I_EviWaiting2_18x21);
  79. }
  80. canvas_set_font(canvas, FontBigNumbers);
  81. furi_string_printf(
  82. disp_str, "%u", ((model->state.line_cur - 1) * 100) / model->state.line_nb);
  83. canvas_draw_str_aligned(
  84. canvas, 114, 36, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  85. furi_string_reset(disp_str);
  86. canvas_draw_icon(canvas, 117, 22, &I_Percent_10x14);
  87. canvas_set_font(canvas, FontSecondary);
  88. furi_string_printf(disp_str, "delay %lus", model->state.delay_remain);
  89. canvas_draw_str_aligned(
  90. canvas, 127, 46, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  91. furi_string_reset(disp_str);
  92. } else {
  93. canvas_draw_icon(canvas, 4, 22, &I_Clock_18x18);
  94. }
  95. furi_string_free(disp_str);
  96. }
  97. static bool bad_usb_input_callback(InputEvent* event, void* context) {
  98. furi_assert(context);
  99. BadUsb* bad_usb = context;
  100. bool consumed = false;
  101. if(event->type == InputTypeShort) {
  102. if(event->key == InputKeyOk) {
  103. consumed = true;
  104. furi_assert(bad_usb->callback);
  105. bad_usb->callback(InputTypeShort, bad_usb->context);
  106. }
  107. }
  108. return consumed;
  109. }
  110. BadUsb* bad_usb_alloc() {
  111. BadUsb* bad_usb = malloc(sizeof(BadUsb));
  112. bad_usb->view = view_alloc();
  113. view_allocate_model(bad_usb->view, ViewModelTypeLocking, sizeof(BadUsbModel));
  114. view_set_context(bad_usb->view, bad_usb);
  115. view_set_draw_callback(bad_usb->view, bad_usb_draw_callback);
  116. view_set_input_callback(bad_usb->view, bad_usb_input_callback);
  117. return bad_usb;
  118. }
  119. void bad_usb_free(BadUsb* bad_usb) {
  120. furi_assert(bad_usb);
  121. view_free(bad_usb->view);
  122. free(bad_usb);
  123. }
  124. View* bad_usb_get_view(BadUsb* bad_usb) {
  125. furi_assert(bad_usb);
  126. return bad_usb->view;
  127. }
  128. void bad_usb_set_ok_callback(BadUsb* bad_usb, BadUsbOkCallback callback, void* context) {
  129. furi_assert(bad_usb);
  130. furi_assert(callback);
  131. with_view_model(
  132. bad_usb->view,
  133. BadUsbModel * model,
  134. {
  135. UNUSED(model);
  136. bad_usb->callback = callback;
  137. bad_usb->context = context;
  138. },
  139. true);
  140. }
  141. void bad_usb_set_file_name(BadUsb* bad_usb, const char* name) {
  142. furi_assert(name);
  143. with_view_model(
  144. bad_usb->view,
  145. BadUsbModel * model,
  146. { strlcpy(model->file_name, name, MAX_NAME_LEN); },
  147. true);
  148. }
  149. void bad_usb_set_state(BadUsb* bad_usb, BadUsbState* st) {
  150. furi_assert(st);
  151. with_view_model(
  152. bad_usb->view,
  153. BadUsbModel * model,
  154. {
  155. memcpy(&(model->state), st, sizeof(BadUsbState));
  156. model->anim_frame ^= 1;
  157. },
  158. true);
  159. }