subbrute_main_view.c 13 KB

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