updater_main.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. #include <gui/gui_i.h>
  2. #include <gui/view.h>
  3. #include <gui/elements.h>
  4. #include <gui/canvas.h>
  5. #include <furi.h>
  6. #include <input/input.h>
  7. #include "../updater_i.h"
  8. #include "updater_main.h"
  9. struct UpdaterMainView {
  10. View* view;
  11. ViewDispatcher* view_dispatcher;
  12. FuriPubSubSubscription* subscription;
  13. void* context;
  14. };
  15. static const uint8_t PROGRESS_RENDER_STEP = 1; /* percent, to limit rendering rate */
  16. typedef struct {
  17. FuriString* status;
  18. uint8_t progress, rendered_progress;
  19. bool failed;
  20. } UpdaterProgressModel;
  21. void updater_main_model_set_state(
  22. UpdaterMainView* main_view,
  23. const char* message,
  24. uint8_t progress,
  25. bool failed) {
  26. bool update = false;
  27. with_view_model(
  28. main_view->view,
  29. UpdaterProgressModel * model,
  30. {
  31. model->failed = failed;
  32. model->progress = progress;
  33. if(furi_string_cmp_str(model->status, message)) {
  34. furi_string_set(model->status, message);
  35. model->rendered_progress = progress;
  36. update = true;
  37. } else if(
  38. (model->rendered_progress > progress) ||
  39. ((progress - model->rendered_progress) > PROGRESS_RENDER_STEP)) {
  40. model->rendered_progress = progress;
  41. update = true;
  42. }
  43. },
  44. update);
  45. }
  46. View* updater_main_get_view(UpdaterMainView* main_view) {
  47. furi_assert(main_view);
  48. return main_view->view;
  49. }
  50. bool updater_main_input(InputEvent* event, void* context) {
  51. furi_assert(event);
  52. furi_assert(context);
  53. UpdaterMainView* main_view = context;
  54. if(!main_view->view_dispatcher) {
  55. return true;
  56. }
  57. if((event->type == InputTypeShort) && (event->key == InputKeyOk)) {
  58. view_dispatcher_send_custom_event(
  59. main_view->view_dispatcher, UpdaterCustomEventRetryUpdate);
  60. } else if((event->type == InputTypeLong) && (event->key == InputKeyBack)) {
  61. view_dispatcher_send_custom_event(
  62. main_view->view_dispatcher, UpdaterCustomEventCancelUpdate);
  63. }
  64. return true;
  65. }
  66. static void updater_main_draw_callback(Canvas* canvas, void* _model) {
  67. UpdaterProgressModel* model = _model;
  68. canvas_set_font(canvas, FontPrimary);
  69. if(model->failed) {
  70. canvas_draw_str_aligned(canvas, 42, 16, AlignLeft, AlignTop, "Update Failed!");
  71. canvas_set_font(canvas, FontSecondary);
  72. canvas_draw_str_aligned(
  73. canvas, 42, 32, AlignLeft, AlignTop, furi_string_get_cstr(model->status));
  74. canvas_draw_icon(canvas, 7, 16, &I_Warning_30x23);
  75. canvas_draw_str_aligned(
  76. canvas, 18, 51, AlignLeft, AlignTop, "to retry, hold to abort");
  77. canvas_draw_icon(canvas, 7, 50, &I_Ok_btn_9x9);
  78. canvas_draw_icon(canvas, 75, 51, &I_Pin_back_arrow_10x8);
  79. } else {
  80. canvas_draw_str_aligned(canvas, 55, 14, AlignLeft, AlignTop, "UPDATING");
  81. canvas_set_font(canvas, FontSecondary);
  82. canvas_draw_str_aligned(
  83. canvas, 64, 51, AlignCenter, AlignTop, furi_string_get_cstr(model->status));
  84. canvas_draw_icon(canvas, 4, 5, &I_Updating_32x40);
  85. elements_progress_bar(canvas, 42, 29, 80, (float)model->progress / 100);
  86. }
  87. }
  88. UpdaterMainView* updater_main_alloc() {
  89. UpdaterMainView* main_view = malloc(sizeof(UpdaterMainView));
  90. main_view->view = view_alloc();
  91. view_allocate_model(main_view->view, ViewModelTypeLocking, sizeof(UpdaterProgressModel));
  92. with_view_model(
  93. main_view->view,
  94. UpdaterProgressModel * model,
  95. { model->status = furi_string_alloc_set("Waiting for SD card"); },
  96. true);
  97. view_set_context(main_view->view, main_view);
  98. view_set_input_callback(main_view->view, updater_main_input);
  99. view_set_draw_callback(main_view->view, updater_main_draw_callback);
  100. return main_view;
  101. }
  102. void updater_main_free(UpdaterMainView* main_view) {
  103. furi_assert(main_view);
  104. with_view_model(
  105. main_view->view, UpdaterProgressModel * model, { furi_string_free(model->status); }, false);
  106. view_free(main_view->view);
  107. free(main_view);
  108. }
  109. void updater_main_set_storage_pubsub(UpdaterMainView* main_view, FuriPubSubSubscription* sub) {
  110. main_view->subscription = sub;
  111. }
  112. FuriPubSubSubscription* updater_main_get_storage_pubsub(UpdaterMainView* main_view) {
  113. return main_view->subscription;
  114. }
  115. void updater_main_set_view_dispatcher(UpdaterMainView* main_view, ViewDispatcher* view_dispatcher) {
  116. main_view->view_dispatcher = view_dispatcher;
  117. }