hex_viewer_startscreen.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. #include "../hex_viewer.h"
  2. #include <furi.h>
  3. #include <furi_hal.h>
  4. #include <input/input.h>
  5. #include <gui/elements.h>
  6. struct HexViewerStartscreen {
  7. View* view;
  8. HexViewerStartscreenCallback callback;
  9. void* context;
  10. };
  11. typedef struct {
  12. uint8_t file_bytes[HEX_VIEWER_LINES_ON_SCREEN][HEX_VIEWER_BYTES_PER_LINE];
  13. uint32_t file_offset;
  14. uint32_t file_read_bytes;
  15. uint32_t file_size;
  16. bool mode;
  17. uint32_t dbg;
  18. } HexViewerStartscreenModel;
  19. void hex_viewer_startscreen_set_callback(
  20. HexViewerStartscreen* instance,
  21. HexViewerStartscreenCallback callback,
  22. void* context) {
  23. furi_assert(instance);
  24. furi_assert(callback);
  25. instance->callback = callback;
  26. instance->context = context;
  27. }
  28. void hex_viewer_startscreen_draw(Canvas* canvas, HexViewerStartscreenModel* model) {
  29. canvas_clear(canvas);
  30. if(!model->file_size) {
  31. canvas_set_color(canvas, ColorBlack);
  32. canvas_set_font(canvas, FontPrimary);
  33. canvas_draw_str_aligned(canvas, 64, 10, AlignCenter, AlignTop, "HexViewer v2.0");
  34. canvas_set_font(canvas, FontSecondary);
  35. canvas_draw_str_aligned(canvas, 64, 22, AlignCenter, AlignTop, "Basic hex viewer");
  36. canvas_draw_str_aligned(canvas, 64, 32, AlignCenter, AlignTop, "for your Flipper");
  37. elements_button_center(canvas, "Open");
  38. } else {
  39. canvas_set_color(canvas, ColorBlack);
  40. elements_button_left(canvas, model->mode ? "Addr" : "Text");
  41. //elements_button_right(canvas, "Info");
  42. elements_button_center(canvas, "Menu");
  43. int ROW_HEIGHT = 12;
  44. int TOP_OFFSET = 10;
  45. int LEFT_OFFSET = 3;
  46. uint32_t line_count = model->file_size / HEX_VIEWER_BYTES_PER_LINE;
  47. if(model->file_size % HEX_VIEWER_BYTES_PER_LINE != 0) line_count += 1;
  48. uint32_t first_line_on_screen = model->file_offset / HEX_VIEWER_BYTES_PER_LINE;
  49. if(line_count > HEX_VIEWER_LINES_ON_SCREEN) {
  50. uint8_t width = canvas_width(canvas);
  51. elements_scrollbar_pos(
  52. canvas,
  53. width,
  54. 0,
  55. ROW_HEIGHT * HEX_VIEWER_LINES_ON_SCREEN,
  56. first_line_on_screen, // TODO
  57. line_count - (HEX_VIEWER_LINES_ON_SCREEN - 1));
  58. }
  59. char temp_buf[32];
  60. uint32_t row_iters = model->file_read_bytes / HEX_VIEWER_BYTES_PER_LINE;
  61. if(model->file_read_bytes % HEX_VIEWER_BYTES_PER_LINE != 0) row_iters += 1;
  62. for(uint32_t i = 0; i < row_iters; ++i) {
  63. uint32_t bytes_left_per_row = model->file_read_bytes - i * HEX_VIEWER_BYTES_PER_LINE;
  64. bytes_left_per_row = MIN(bytes_left_per_row, HEX_VIEWER_BYTES_PER_LINE);
  65. if(model->mode) {
  66. memcpy(temp_buf, model->file_bytes[i], bytes_left_per_row);
  67. temp_buf[bytes_left_per_row] = '\0';
  68. for(uint32_t j = 0; j < bytes_left_per_row; ++j)
  69. if(!isprint((int)temp_buf[j])) temp_buf[j] = '.';
  70. canvas_set_font(canvas, FontKeyboard);
  71. canvas_draw_str(canvas, LEFT_OFFSET, TOP_OFFSET + i * ROW_HEIGHT, temp_buf);
  72. } else {
  73. uint32_t addr = model->file_offset + i * HEX_VIEWER_BYTES_PER_LINE;
  74. snprintf(temp_buf, 32, "%04lX", addr);
  75. canvas_set_font(canvas, FontKeyboard);
  76. canvas_draw_str(canvas, LEFT_OFFSET, TOP_OFFSET + i * ROW_HEIGHT, temp_buf);
  77. }
  78. char* p = temp_buf;
  79. for(uint32_t j = 0; j < bytes_left_per_row; ++j)
  80. p += snprintf(p, 32, "%02X ", model->file_bytes[i][j]);
  81. canvas_set_font(canvas, FontKeyboard);
  82. canvas_draw_str(canvas, LEFT_OFFSET + 41, TOP_OFFSET + i * ROW_HEIGHT, temp_buf);
  83. }
  84. // Poor man's debug
  85. // snprintf(temp_buf, 32, "D %02lX", model->dbg);
  86. // elements_button_right(canvas, temp_buf);
  87. }
  88. }
  89. static void hex_viewer_startscreen_model_init(HexViewerStartscreenModel* const model) {
  90. memset(model->file_bytes, 0, sizeof(model->file_bytes));
  91. model->file_offset = 0;
  92. model->file_read_bytes = 0;
  93. model->file_size = 0;
  94. model->mode = false;
  95. model->dbg = 0;
  96. }
  97. static void
  98. update_local_model_from_app(HexViewer* const app, HexViewerStartscreenModel* const model) {
  99. memcpy(model->file_bytes, app->model->file_bytes, sizeof(model->file_bytes));
  100. model->file_offset = app->model->file_offset;
  101. model->file_read_bytes = app->model->file_read_bytes;
  102. model->file_size = app->model->file_size;
  103. //model->mode = app->model->mode;
  104. }
  105. bool hex_viewer_startscreen_input(InputEvent* event, void* context) {
  106. furi_assert(context);
  107. HexViewerStartscreen* instance = context;
  108. HexViewer* app = instance->context; // TO so good, but works
  109. // TODO InputTypeShort?
  110. if(event->type == InputTypeRelease || event->type == InputTypeRepeat) {
  111. switch(event->key) {
  112. case InputKeyBack:
  113. with_view_model(
  114. instance->view,
  115. HexViewerStartscreenModel * model,
  116. {
  117. instance->callback(HexViewerCustomEventStartscreenBack, instance->context);
  118. update_local_model_from_app(instance->context, model);
  119. },
  120. true);
  121. break;
  122. case InputKeyLeft:
  123. with_view_model(
  124. instance->view,
  125. HexViewerStartscreenModel * model,
  126. { model->mode = !model->mode; },
  127. true);
  128. break;
  129. case InputKeyRight:
  130. with_view_model(
  131. instance->view,
  132. HexViewerStartscreenModel * model,
  133. {
  134. // instance->callback(HexViewerCustomEventStartscreenRight, instance->context);
  135. // update_local_model_from_app(instance->context, model);
  136. model->dbg = 0;
  137. },
  138. true);
  139. break;
  140. case InputKeyUp:
  141. with_view_model(
  142. instance->view,
  143. HexViewerStartscreenModel * model,
  144. {
  145. if(app->model->file_offset > 0) {
  146. app->model->file_offset -= HEX_VIEWER_BYTES_PER_LINE;
  147. if(!hex_viewer_read_file(app)) break; // TODO Do smth
  148. }
  149. update_local_model_from_app(instance->context, model);
  150. },
  151. true);
  152. break;
  153. case InputKeyDown:
  154. with_view_model(
  155. instance->view,
  156. HexViewerStartscreenModel * model,
  157. {
  158. uint32_t last_byte_on_screen =
  159. app->model->file_offset + app->model->file_read_bytes;
  160. if(app->model->file_size > last_byte_on_screen) {
  161. app->model->file_offset += HEX_VIEWER_BYTES_PER_LINE;
  162. if(!hex_viewer_read_file(app)) break; // TODO Do smth
  163. }
  164. update_local_model_from_app(instance->context, model);
  165. },
  166. true);
  167. break;
  168. case InputKeyOk:
  169. with_view_model(
  170. instance->view,
  171. HexViewerStartscreenModel * model,
  172. {
  173. instance->callback(HexViewerCustomEventStartscreenOk, instance->context);
  174. update_local_model_from_app(instance->context, model);
  175. },
  176. true);
  177. break;
  178. case InputKeyMAX:
  179. break;
  180. }
  181. }
  182. return true;
  183. }
  184. void hex_viewer_startscreen_exit(void* context) {
  185. furi_assert(context);
  186. }
  187. void hex_viewer_startscreen_enter(void* context) {
  188. furi_assert(context);
  189. HexViewerStartscreen* instance = (HexViewerStartscreen*)context;
  190. with_view_model(
  191. instance->view,
  192. HexViewerStartscreenModel * model,
  193. { update_local_model_from_app(instance->context, model); },
  194. true);
  195. }
  196. HexViewerStartscreen* hex_viewer_startscreen_alloc() {
  197. HexViewerStartscreen* instance = malloc(sizeof(HexViewerStartscreen));
  198. instance->view = view_alloc();
  199. view_allocate_model(instance->view, ViewModelTypeLocking, sizeof(HexViewerStartscreenModel));
  200. view_set_context(instance->view, instance);
  201. view_set_draw_callback(instance->view, (ViewDrawCallback)hex_viewer_startscreen_draw);
  202. view_set_input_callback(instance->view, hex_viewer_startscreen_input);
  203. view_set_enter_callback(instance->view, hex_viewer_startscreen_enter);
  204. view_set_exit_callback(instance->view, hex_viewer_startscreen_exit);
  205. with_view_model(
  206. instance->view,
  207. HexViewerStartscreenModel * model,
  208. { hex_viewer_startscreen_model_init(model); },
  209. true);
  210. return instance;
  211. }
  212. void hex_viewer_startscreen_free(HexViewerStartscreen* instance) {
  213. furi_assert(instance);
  214. with_view_model(
  215. instance->view, HexViewerStartscreenModel * model, { UNUSED(model); }, true);
  216. view_free(instance->view);
  217. free(instance);
  218. }
  219. View* hex_viewer_startscreen_get_view(HexViewerStartscreen* instance) {
  220. furi_assert(instance);
  221. return instance->view;
  222. }