subbrute_main_view.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. #include "subbrute_main_view.h"
  2. #include "../subbrute_i.h"
  3. #include "../subbrute_protocols.h"
  4. #include <input/input.h>
  5. #include <gui/elements.h>
  6. #include <gui/icon.h>
  7. #include <assets_icons.h>
  8. #define STATUS_BAR_Y_SHIFT 14
  9. #define TAG "SubBruteMainView"
  10. #define ITEMS_ON_SCREEN 3
  11. struct SubBruteMainView {
  12. View* view;
  13. SubBruteMainViewCallback callback;
  14. void* context;
  15. uint8_t index;
  16. bool is_select_byte;
  17. const char* key_field;
  18. uint8_t repeats_count;
  19. uint8_t window_position;
  20. };
  21. typedef struct {
  22. uint8_t index;
  23. uint8_t extra_repeats;
  24. uint8_t window_position;
  25. bool is_select_byte;
  26. const char* key_field;
  27. uint8_t repeats_count;
  28. } SubBruteMainViewModel;
  29. void subbrute_main_view_set_callback(
  30. SubBruteMainView* instance,
  31. SubBruteMainViewCallback callback,
  32. void* context) {
  33. furi_assert(instance);
  34. furi_assert(callback);
  35. instance->callback = callback;
  36. instance->context = context;
  37. }
  38. FuriString* center_displayed_key(const char* key_cstr, uint8_t index) {
  39. uint8_t str_index = (index * 3);
  40. char display_menu[] = {
  41. 'X', 'X', ' ', 'X', 'X', ' ', '<', 'X', 'X', '>', ' ', 'X', 'X', ' ', 'X', 'X', '\0'};
  42. if(key_cstr != NULL) {
  43. if(index > 1) {
  44. display_menu[0] = key_cstr[str_index - 6];
  45. display_menu[1] = key_cstr[str_index - 5];
  46. } else {
  47. display_menu[0] = ' ';
  48. display_menu[1] = ' ';
  49. }
  50. if(index > 0) {
  51. display_menu[3] = key_cstr[str_index - 3];
  52. display_menu[4] = key_cstr[str_index - 2];
  53. } else {
  54. display_menu[3] = ' ';
  55. display_menu[4] = ' ';
  56. }
  57. display_menu[7] = key_cstr[str_index];
  58. display_menu[8] = key_cstr[str_index + 1];
  59. if((str_index + 4) <= (uint8_t)strlen(key_cstr)) {
  60. display_menu[11] = key_cstr[str_index + 3];
  61. display_menu[12] = key_cstr[str_index + 4];
  62. } else {
  63. display_menu[11] = ' ';
  64. display_menu[12] = ' ';
  65. }
  66. if((str_index + 8) <= (uint8_t)strlen(key_cstr)) {
  67. display_menu[14] = key_cstr[str_index + 6];
  68. display_menu[15] = key_cstr[str_index + 7];
  69. } else {
  70. display_menu[14] = ' ';
  71. display_menu[15] = ' ';
  72. }
  73. }
  74. return furi_string_alloc_set(display_menu);
  75. }
  76. void subbrute_main_view_draw(Canvas* canvas, SubBruteMainViewModel* model) {
  77. SubBruteMainViewModel* m = model;
  78. // Title
  79. canvas_set_font(canvas, FontPrimary);
  80. canvas_draw_box(canvas, 0, 0, canvas_width(canvas), STATUS_BAR_Y_SHIFT);
  81. canvas_invert_color(canvas);
  82. canvas_draw_str_aligned(canvas, 64, 3, AlignCenter, AlignTop, "Sub-GHz BruteForcer 3.1");
  83. canvas_invert_color(canvas);
  84. if(m->is_select_byte) {
  85. #ifdef FURI_DEBUG
  86. //FURI_LOG_D(TAG, "key_field: %s", model->key_field);
  87. #endif
  88. char msg_index[18];
  89. snprintf(msg_index, sizeof(msg_index), "Field index : %d", m->index);
  90. canvas_draw_str_aligned(canvas, 64, 26, AlignCenter, AlignTop, msg_index);
  91. FuriString* menu_items;
  92. menu_items = center_displayed_key(m->key_field, m->index);
  93. canvas_set_font(canvas, FontSecondary);
  94. canvas_draw_str_aligned(
  95. canvas, 64, 40, AlignCenter, AlignTop, furi_string_get_cstr(menu_items));
  96. elements_button_center(canvas, "Select");
  97. elements_button_left(canvas, "<");
  98. elements_button_right(canvas, ">");
  99. furi_string_reset(menu_items);
  100. furi_string_free(menu_items);
  101. } else {
  102. // Menu
  103. canvas_set_color(canvas, ColorBlack);
  104. canvas_set_font(canvas, FontSecondary);
  105. const uint8_t item_height = 16;
  106. #ifdef FURI_DEBUG
  107. //FURI_LOG_D(TAG, "window_position: %d, index: %d", model->window_position, model->index);
  108. #endif
  109. for(uint8_t position = 0; position < SubBruteAttackTotalCount; ++position) {
  110. uint8_t item_position = position - model->window_position;
  111. if(item_position < items_on_screen) {
  112. if(m->index == position) {
  113. canvas_draw_str_aligned(
  114. canvas,
  115. 4,
  116. 9 + (item_position * item_height) + STATUS_BAR_Y_SHIFT,
  117. AlignLeft,
  118. AlignCenter,
  119. subbrute_protocol_name(position));
  120. elements_frame(
  121. canvas, 1, 1 + (item_position * item_height) + STATUS_BAR_Y_SHIFT, 124, 15);
  122. } else {
  123. canvas_draw_str_aligned(
  124. canvas,
  125. 4,
  126. 9 + (item_position * item_height) + STATUS_BAR_Y_SHIFT,
  127. AlignLeft,
  128. AlignCenter,
  129. subbrute_protocol_name(position));
  130. }
  131. }
  132. }
  133. elements_scrollbar_pos(
  134. canvas,
  135. canvas_width(canvas),
  136. STATUS_BAR_Y_SHIFT + 2,
  137. canvas_height(canvas) - STATUS_BAR_Y_SHIFT,
  138. m->index,
  139. SubBruteAttackTotalCount);
  140. }
  141. }
  142. bool subbrute_main_view_input(InputEvent* event, void* context) {
  143. furi_assert(event);
  144. furi_assert(context);
  145. #ifdef FURI_DEBUG
  146. FURI_LOG_D(TAG, "InputKey: %d", event->key);
  147. #endif
  148. if(event->key == InputKeyBack && event->type == InputTypeShort) {
  149. return false;
  150. }
  151. SubBruteMainView* instance = context;
  152. const uint8_t min_value = 0;
  153. const uint8_t correct_total = SubBruteAttackTotalCount - 1;
  154. uint8_t index = 0;
  155. bool updated = false;
  156. bool consumed = false;
  157. bool is_short = (event->type == InputTypeShort) || (event->type == InputTypeRepeat);
  158. if(!instance->is_select_byte) {
  159. if(event->key == InputKeyUp && is_short) {
  160. if(instance->index == min_value) {
  161. instance->index = correct_total;
  162. } else {
  163. instance->index = CLAMP(instance->index - 1, correct_total, min_value);
  164. }
  165. updated = true;
  166. consumed = true;
  167. } else if(event->key == InputKeyDown && is_short) {
  168. if(instance->index == correct_total) {
  169. instance->index = min_value;
  170. } else {
  171. instance->index = CLAMP(instance->index + 1, correct_total, min_value);
  172. }
  173. updated = true;
  174. consumed = true;
  175. } else if(event->key == InputKeyOk && is_short) {
  176. if(index == SubBruteAttackLoadFile) {
  177. instance->callback(SubBruteCustomEventTypeLoadFile, instance->context);
  178. } else {
  179. instance->callback(SubBruteCustomEventTypeMenuSelected, instance->context);
  180. }
  181. consumed = true;
  182. updated = true;
  183. }
  184. if(updated) {
  185. instance->window_position = instance->index;
  186. if(instance->window_position > 0) {
  187. instance->window_position -= 1;
  188. }
  189. if(SubBruteAttackTotalCount <= ITEMS_ON_SCREEN) {
  190. instance->window_position = 0;
  191. } else {
  192. if(instance->window_position >= (SubBruteAttackTotalCount - ITEMS_ON_SCREEN)) {
  193. instance->window_position = (SubBruteAttackTotalCount - ITEMS_ON_SCREEN);
  194. }
  195. }
  196. }
  197. } else {
  198. if(event->key == InputKeyLeft && is_short) {
  199. if(instance->index > 0) {
  200. instance->index--;
  201. }
  202. updated = true;
  203. } else if(event->key == InputKeyRight && is_short) {
  204. if(instance->index < 7) {
  205. instance->index++;
  206. }
  207. updated = true;
  208. } else if(event->key == InputKeyOk && is_short) {
  209. instance->callback(SubBruteCustomEventTypeIndexSelected, instance->context);
  210. consumed = true;
  211. updated = true;
  212. }
  213. }
  214. if(updated) {
  215. with_view_model(
  216. instance->view,
  217. SubBruteMainViewModel * model,
  218. {
  219. model->index = instance->index;
  220. model->window_position = instance->window_position;
  221. model->key_field = instance->key_field;
  222. model->is_select_byte = instance->is_select_byte;
  223. model->repeats_count = instance->repeats_count;
  224. },
  225. true);
  226. }
  227. return consumed;
  228. }
  229. void subbrute_main_view_enter(void* context) {
  230. furi_assert(context);
  231. SubBruteMainView* instance = context;
  232. with_view_model(
  233. instance->view,
  234. SubBruteMainViewModel * model,
  235. {
  236. model->key_field = NULL;
  237. model->is_select_byte = false;
  238. },
  239. true);
  240. #ifdef FURI_DEBUG
  241. FURI_LOG_D(TAG, "subbrute_main_view_enter");
  242. #endif
  243. }
  244. void subbrute_main_view_exit(void* context) {
  245. furi_assert(context);
  246. #ifdef FURI_DEBUG
  247. FURI_LOG_D(TAG, "subbrute_main_view_exit");
  248. #endif
  249. }
  250. SubBruteMainView* subbrute_main_view_alloc() {
  251. SubBruteMainView* instance = malloc(sizeof(SubBruteMainView));
  252. instance->view = view_alloc();
  253. view_allocate_model(instance->view, ViewModelTypeLocking, sizeof(SubBruteMainViewModel));
  254. view_set_context(instance->view, instance);
  255. view_set_draw_callback(instance->view, (ViewDrawCallback)subbrute_main_view_draw);
  256. view_set_input_callback(instance->view, subbrute_main_view_input);
  257. view_set_enter_callback(instance->view, subbrute_main_view_enter);
  258. view_set_exit_callback(instance->view, subbrute_main_view_exit);
  259. with_view_model(
  260. instance->view,
  261. SubBruteMainViewModel * model,
  262. {
  263. model->index = 0;
  264. model->window_position = 0;
  265. model->extra_repeats = 0;
  266. model->key_field = NULL;
  267. model->is_select_byte = false;
  268. model->repeats_count = 0;
  269. },
  270. true);
  271. instance->index = 0;
  272. instance->window_position = 0;
  273. instance->key_field = NULL;
  274. instance->is_select_byte = false;
  275. instance->repeats_count = 0;
  276. return instance;
  277. }
  278. void subbrute_main_view_free(SubBruteMainView* instance) {
  279. furi_assert(instance);
  280. view_free(instance->view);
  281. free(instance);
  282. }
  283. View* subbrute_main_view_get_view(SubBruteMainView* instance) {
  284. furi_assert(instance);
  285. return instance->view;
  286. }
  287. void subbrute_main_view_set_index(
  288. SubBruteMainView* instance,
  289. uint8_t idx,
  290. bool is_select_byte,
  291. const char* key_field) {
  292. furi_assert(instance);
  293. furi_assert(idx < SubBruteAttackTotalCount);
  294. #ifdef FURI_DEBUG
  295. FURI_LOG_I(TAG, "Set index: %d", idx);
  296. #endif
  297. instance->is_select_byte = is_select_byte;
  298. instance->key_field = key_field;
  299. instance->index = idx;
  300. instance->window_position = idx;
  301. if(!is_select_byte) {
  302. if(instance->window_position > 0) {
  303. instance->window_position -= 1;
  304. }
  305. if(SubBruteAttackTotalCount <= ITEMS_ON_SCREEN) {
  306. instance->window_position = 0;
  307. } else {
  308. if(instance->window_position >= (SubBruteAttackTotalCount - ITEMS_ON_SCREEN)) {
  309. instance->window_position = (SubBruteAttackTotalCount - ITEMS_ON_SCREEN);
  310. }
  311. }
  312. }
  313. with_view_model(
  314. instance->view,
  315. SubBruteMainViewModel * model,
  316. {
  317. model->index = instance->index;
  318. model->window_position = instance->window_position;
  319. model->key_field = instance->key_field;
  320. model->is_select_byte = instance->is_select_byte;
  321. model->repeats_count = instance->repeats_count;
  322. },
  323. true);
  324. }
  325. SubBruteAttacks subbrute_main_view_get_index(SubBruteMainView* instance) {
  326. furi_assert(instance);
  327. return instance->index;
  328. }
  329. uint8_t subbrute_main_view_get_extra_repeats(SubBruteMainView* instance) {
  330. furi_assert(instance);
  331. uint8_t extra_repeats = 0;
  332. with_view_model(
  333. instance->view,
  334. SubBruteMainViewModel * model,
  335. { extra_repeats = model->extra_repeats; },
  336. false);
  337. return extra_repeats;
  338. }