bad_usb_view.c 6.2 KB

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