subbrute_main_view.c 13 KB

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