subbrute_main_view.c 12 KB

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