subbrute_main_view.c 13 KB

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