bad_usb_view.c 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. #include "bad_usb_view.h"
  2. #include "../helpers/ducky_script.h"
  3. #include <toolbox/path.h>
  4. #include <gui/elements.h>
  5. #include <assets_icons.h>
  6. #define MAX_NAME_LEN 64
  7. struct BadUsb {
  8. View* view;
  9. BadUsbButtonCallback callback;
  10. void* context;
  11. };
  12. typedef struct {
  13. char file_name[MAX_NAME_LEN];
  14. char layout[MAX_NAME_LEN];
  15. BadUsbState state;
  16. uint8_t anim_frame;
  17. } BadUsbModel;
  18. static void bad_usb_draw_callback(Canvas* canvas, void* _model) {
  19. BadUsbModel* model = _model;
  20. FuriString* disp_str;
  21. disp_str = furi_string_alloc_set(model->file_name);
  22. elements_string_fit_width(canvas, disp_str, 128 - 2);
  23. canvas_set_font(canvas, FontSecondary);
  24. canvas_draw_str(canvas, 2, 8, furi_string_get_cstr(disp_str));
  25. if(strlen(model->layout) == 0) {
  26. furi_string_set(disp_str, "(default)");
  27. } else {
  28. furi_string_reset(disp_str);
  29. furi_string_push_back(disp_str, '(');
  30. for(size_t i = 0; i < strlen(model->layout); i++)
  31. furi_string_push_back(disp_str, model->layout[i]);
  32. furi_string_push_back(disp_str, ')');
  33. }
  34. elements_string_fit_width(canvas, disp_str, 128 - 2);
  35. canvas_draw_str(
  36. canvas, 2, 8 + canvas_current_font_height(canvas), furi_string_get_cstr(disp_str));
  37. furi_string_reset(disp_str);
  38. canvas_draw_icon(canvas, 22, 24, &I_UsbTree_48x22);
  39. if((model->state.state == BadUsbStateIdle) || (model->state.state == BadUsbStateDone) ||
  40. (model->state.state == BadUsbStateNotConnected)) {
  41. elements_button_center(canvas, "Run");
  42. elements_button_left(canvas, "Config");
  43. } else if((model->state.state == BadUsbStateRunning) || (model->state.state == BadUsbStateDelay)) {
  44. elements_button_center(canvas, "Stop");
  45. } else if(model->state.state == BadUsbStateWaitForBtn) {
  46. elements_button_center(canvas, "Press to continue");
  47. } else if(model->state.state == BadUsbStateWillRun) {
  48. elements_button_center(canvas, "Cancel");
  49. }
  50. if(model->state.state == BadUsbStateNotConnected) {
  51. canvas_draw_icon(canvas, 4, 26, &I_Clock_18x18);
  52. canvas_set_font(canvas, FontPrimary);
  53. canvas_draw_str_aligned(canvas, 127, 31, AlignRight, AlignBottom, "Connect");
  54. canvas_draw_str_aligned(canvas, 127, 43, AlignRight, AlignBottom, "to USB");
  55. } else if(model->state.state == BadUsbStateWillRun) {
  56. canvas_draw_icon(canvas, 4, 26, &I_Clock_18x18);
  57. canvas_set_font(canvas, FontPrimary);
  58. canvas_draw_str_aligned(canvas, 127, 31, AlignRight, AlignBottom, "Will run");
  59. canvas_draw_str_aligned(canvas, 127, 43, AlignRight, AlignBottom, "on connect");
  60. } else if(model->state.state == BadUsbStateFileError) {
  61. canvas_draw_icon(canvas, 4, 26, &I_Error_18x18);
  62. canvas_set_font(canvas, FontPrimary);
  63. canvas_draw_str_aligned(canvas, 127, 31, AlignRight, AlignBottom, "File");
  64. canvas_draw_str_aligned(canvas, 127, 43, AlignRight, AlignBottom, "ERROR");
  65. } else if(model->state.state == BadUsbStateScriptError) {
  66. canvas_draw_icon(canvas, 4, 26, &I_Error_18x18);
  67. canvas_set_font(canvas, FontPrimary);
  68. canvas_draw_str_aligned(canvas, 127, 33, AlignRight, AlignBottom, "ERROR:");
  69. canvas_set_font(canvas, FontSecondary);
  70. furi_string_printf(disp_str, "line %u", model->state.error_line);
  71. canvas_draw_str_aligned(
  72. canvas, 127, 46, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  73. furi_string_reset(disp_str);
  74. furi_string_set_str(disp_str, model->state.error);
  75. elements_string_fit_width(canvas, disp_str, canvas_width(canvas));
  76. canvas_draw_str_aligned(
  77. canvas, 127, 56, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  78. furi_string_reset(disp_str);
  79. } else if(model->state.state == BadUsbStateIdle) {
  80. canvas_draw_icon(canvas, 4, 26, &I_Smile_18x18);
  81. canvas_set_font(canvas, FontBigNumbers);
  82. canvas_draw_str_aligned(canvas, 114, 40, AlignRight, AlignBottom, "0");
  83. canvas_draw_icon(canvas, 117, 26, &I_Percent_10x14);
  84. } else if(model->state.state == BadUsbStateRunning) {
  85. if(model->anim_frame == 0) {
  86. canvas_draw_icon(canvas, 4, 23, &I_EviSmile1_18x21);
  87. } else {
  88. canvas_draw_icon(canvas, 4, 23, &I_EviSmile2_18x21);
  89. }
  90. canvas_set_font(canvas, FontBigNumbers);
  91. furi_string_printf(
  92. disp_str, "%u", ((model->state.line_cur - 1) * 100) / model->state.line_nb);
  93. canvas_draw_str_aligned(
  94. canvas, 114, 40, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  95. furi_string_reset(disp_str);
  96. canvas_draw_icon(canvas, 117, 26, &I_Percent_10x14);
  97. } else if(model->state.state == BadUsbStateDone) {
  98. canvas_draw_icon(canvas, 4, 23, &I_EviSmile1_18x21);
  99. canvas_set_font(canvas, FontBigNumbers);
  100. canvas_draw_str_aligned(canvas, 114, 40, AlignRight, AlignBottom, "100");
  101. furi_string_reset(disp_str);
  102. canvas_draw_icon(canvas, 117, 26, &I_Percent_10x14);
  103. } else if(model->state.state == BadUsbStateDelay) {
  104. if(model->anim_frame == 0) {
  105. canvas_draw_icon(canvas, 4, 23, &I_EviWaiting1_18x21);
  106. } else {
  107. canvas_draw_icon(canvas, 4, 23, &I_EviWaiting2_18x21);
  108. }
  109. canvas_set_font(canvas, FontBigNumbers);
  110. furi_string_printf(
  111. disp_str, "%u", ((model->state.line_cur - 1) * 100) / model->state.line_nb);
  112. canvas_draw_str_aligned(
  113. canvas, 114, 40, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  114. furi_string_reset(disp_str);
  115. canvas_draw_icon(canvas, 117, 26, &I_Percent_10x14);
  116. canvas_set_font(canvas, FontSecondary);
  117. furi_string_printf(disp_str, "delay %lus", model->state.delay_remain);
  118. canvas_draw_str_aligned(
  119. canvas, 127, 50, AlignRight, AlignBottom, furi_string_get_cstr(disp_str));
  120. furi_string_reset(disp_str);
  121. } else {
  122. canvas_draw_icon(canvas, 4, 26, &I_Clock_18x18);
  123. }
  124. furi_string_free(disp_str);
  125. }
  126. static bool bad_usb_input_callback(InputEvent* event, void* context) {
  127. furi_assert(context);
  128. BadUsb* bad_usb = context;
  129. bool consumed = false;
  130. if(event->type == InputTypeShort) {
  131. if((event->key == InputKeyLeft) || (event->key == InputKeyOk)) {
  132. consumed = true;
  133. furi_assert(bad_usb->callback);
  134. bad_usb->callback(event->key, bad_usb->context);
  135. }
  136. }
  137. return consumed;
  138. }
  139. BadUsb* bad_usb_alloc() {
  140. BadUsb* bad_usb = malloc(sizeof(BadUsb));
  141. bad_usb->view = view_alloc();
  142. view_allocate_model(bad_usb->view, ViewModelTypeLocking, sizeof(BadUsbModel));
  143. view_set_context(bad_usb->view, bad_usb);
  144. view_set_draw_callback(bad_usb->view, bad_usb_draw_callback);
  145. view_set_input_callback(bad_usb->view, bad_usb_input_callback);
  146. return bad_usb;
  147. }
  148. void bad_usb_free(BadUsb* bad_usb) {
  149. furi_assert(bad_usb);
  150. view_free(bad_usb->view);
  151. free(bad_usb);
  152. }
  153. View* bad_usb_get_view(BadUsb* bad_usb) {
  154. furi_assert(bad_usb);
  155. return bad_usb->view;
  156. }
  157. void bad_usb_set_button_callback(BadUsb* bad_usb, BadUsbButtonCallback callback, void* context) {
  158. furi_assert(bad_usb);
  159. furi_assert(callback);
  160. with_view_model(
  161. bad_usb->view,
  162. BadUsbModel * model,
  163. {
  164. UNUSED(model);
  165. bad_usb->callback = callback;
  166. bad_usb->context = context;
  167. },
  168. true);
  169. }
  170. void bad_usb_set_file_name(BadUsb* bad_usb, const char* name) {
  171. furi_assert(name);
  172. with_view_model(
  173. bad_usb->view,
  174. BadUsbModel * model,
  175. { strlcpy(model->file_name, name, MAX_NAME_LEN); },
  176. true);
  177. }
  178. void bad_usb_set_layout(BadUsb* bad_usb, const char* layout) {
  179. furi_assert(layout);
  180. with_view_model(
  181. bad_usb->view,
  182. BadUsbModel * model,
  183. { strlcpy(model->layout, layout, MAX_NAME_LEN); },
  184. true);
  185. }
  186. void bad_usb_set_state(BadUsb* bad_usb, BadUsbState* st) {
  187. furi_assert(st);
  188. with_view_model(
  189. bad_usb->view,
  190. BadUsbModel * model,
  191. {
  192. memcpy(&(model->state), st, sizeof(BadUsbState));
  193. model->anim_frame ^= 1;
  194. },
  195. true);
  196. }
  197. bool bad_usb_is_idle_state(BadUsb* bad_usb) {
  198. bool is_idle = false;
  199. with_view_model(
  200. bad_usb->view,
  201. BadUsbModel * model,
  202. {
  203. if((model->state.state == BadUsbStateIdle) ||
  204. (model->state.state == BadUsbStateDone) ||
  205. (model->state.state == BadUsbStateNotConnected)) {
  206. is_idle = true;
  207. }
  208. },
  209. false);
  210. return is_idle;
  211. }