updater_main.c 4.4 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. string_t 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. with_view_model(
  27. main_view->view, (UpdaterProgressModel * model) {
  28. model->failed = failed;
  29. model->progress = progress;
  30. if(string_cmp_str(model->status, message)) {
  31. string_set(model->status, message);
  32. model->rendered_progress = progress;
  33. return true;
  34. }
  35. if((model->rendered_progress > progress) ||
  36. ((progress - model->rendered_progress) > PROGRESS_RENDER_STEP)) {
  37. model->rendered_progress = progress;
  38. return true;
  39. }
  40. return false;
  41. });
  42. }
  43. View* updater_main_get_view(UpdaterMainView* main_view) {
  44. furi_assert(main_view);
  45. return main_view->view;
  46. }
  47. bool updater_main_input(InputEvent* event, void* context) {
  48. furi_assert(event);
  49. furi_assert(context);
  50. UpdaterMainView* main_view = context;
  51. if(!main_view->view_dispatcher) {
  52. return true;
  53. }
  54. if((event->type == InputTypeShort) && (event->key == InputKeyOk)) {
  55. view_dispatcher_send_custom_event(
  56. main_view->view_dispatcher, UpdaterCustomEventRetryUpdate);
  57. } else if((event->type == InputTypeLong) && (event->key == InputKeyBack)) {
  58. view_dispatcher_send_custom_event(
  59. main_view->view_dispatcher, UpdaterCustomEventCancelUpdate);
  60. }
  61. return true;
  62. }
  63. static void updater_main_draw_callback(Canvas* canvas, void* _model) {
  64. UpdaterProgressModel* model = _model;
  65. canvas_set_font(canvas, FontPrimary);
  66. if(model->failed) {
  67. canvas_draw_str_aligned(canvas, 42, 16, AlignLeft, AlignTop, "Update Failed!");
  68. canvas_set_font(canvas, FontSecondary);
  69. canvas_draw_str_aligned(
  70. canvas, 42, 32, AlignLeft, AlignTop, string_get_cstr(model->status));
  71. canvas_draw_icon(canvas, 7, 16, &I_Warning_30x23);
  72. canvas_draw_str_aligned(
  73. canvas, 18, 51, AlignLeft, AlignTop, "to retry, hold to abort");
  74. canvas_draw_icon(canvas, 7, 50, &I_Ok_btn_9x9);
  75. canvas_draw_icon(canvas, 75, 51, &I_Pin_back_arrow_10x8);
  76. } else {
  77. canvas_draw_str_aligned(canvas, 55, 14, AlignLeft, AlignTop, "UPDATING");
  78. canvas_set_font(canvas, FontSecondary);
  79. canvas_draw_str_aligned(
  80. canvas, 64, 51, AlignCenter, AlignTop, string_get_cstr(model->status));
  81. canvas_draw_icon(canvas, 4, 5, &I_Updating_32x40);
  82. elements_progress_bar(canvas, 42, 29, 80, (float)model->progress / 100);
  83. }
  84. }
  85. UpdaterMainView* updater_main_alloc() {
  86. UpdaterMainView* main_view = malloc(sizeof(UpdaterMainView));
  87. main_view->view = view_alloc();
  88. view_allocate_model(main_view->view, ViewModelTypeLocking, sizeof(UpdaterProgressModel));
  89. with_view_model(
  90. main_view->view, (UpdaterProgressModel * model) {
  91. string_init_set(model->status, "Waiting for SD card");
  92. return true;
  93. });
  94. view_set_context(main_view->view, main_view);
  95. view_set_input_callback(main_view->view, updater_main_input);
  96. view_set_draw_callback(main_view->view, updater_main_draw_callback);
  97. return main_view;
  98. }
  99. void updater_main_free(UpdaterMainView* main_view) {
  100. furi_assert(main_view);
  101. with_view_model(
  102. main_view->view, (UpdaterProgressModel * model) {
  103. string_clear(model->status);
  104. return false;
  105. });
  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. }