desktop_locked.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. #include <furi.h>
  2. #include "../desktop_i.h"
  3. #include "desktop_locked.h"
  4. void desktop_locked_set_callback(
  5. DesktopLockedView* locked_view,
  6. DesktopLockedViewCallback callback,
  7. void* context) {
  8. furi_assert(locked_view);
  9. furi_assert(callback);
  10. locked_view->callback = callback;
  11. locked_view->context = context;
  12. }
  13. void locked_view_timer_callback(void* context) {
  14. DesktopLockedView* locked_view = context;
  15. locked_view->callback(DesktopLockedEventUpdate, locked_view->context);
  16. }
  17. // temporary locked screen animation managment
  18. void desktop_locked_set_dolphin_animation(DesktopLockedView* locked_view) {
  19. with_view_model(
  20. locked_view->view, (DesktopLockedViewModel * model) {
  21. if(model->animation) icon_animation_free(model->animation);
  22. model->animation = icon_animation_alloc(desktop_get_icon());
  23. view_tie_icon_animation(locked_view->view, model->animation);
  24. return true;
  25. });
  26. }
  27. void desktop_locked_update_hint_timeout(DesktopLockedView* locked_view) {
  28. with_view_model(
  29. locked_view->view, (DesktopLockedViewModel * model) {
  30. model->hint_expire_at = osKernelGetTickCount() + osKernelGetTickFreq();
  31. return true;
  32. });
  33. }
  34. void desktop_locked_reset_door_pos(DesktopLockedView* locked_view) {
  35. with_view_model(
  36. locked_view->view, (DesktopLockedViewModel * model) {
  37. model->animation_seq_end = false;
  38. model->door_left_x = DOOR_L_POS;
  39. model->door_right_x = DOOR_R_POS;
  40. return true;
  41. });
  42. }
  43. void desktop_locked_manage_redraw(DesktopLockedView* locked_view) {
  44. bool animation_seq_end;
  45. with_view_model(
  46. locked_view->view, (DesktopLockedViewModel * model) {
  47. model->animation_seq_end = !model->door_left_x;
  48. animation_seq_end = model->animation_seq_end;
  49. if(!model->animation_seq_end) {
  50. model->door_left_x = CLAMP(model->door_left_x + 5, DOOR_L_POS_MAX, DOOR_L_POS);
  51. model->door_right_x = CLAMP(model->door_right_x - 5, DOOR_R_POS, DOOR_R_POS_MIN);
  52. } else {
  53. model->hint_expire_at = !model->hint_expire_at;
  54. }
  55. return true;
  56. });
  57. if(animation_seq_end) {
  58. osTimerStop(locked_view->timer);
  59. }
  60. }
  61. void desktop_locked_reset_counter(DesktopLockedView* locked_view) {
  62. locked_view->lock_count = 0;
  63. locked_view->lock_lastpress = 0;
  64. with_view_model(
  65. locked_view->view, (DesktopLockedViewModel * model) {
  66. model->hint_expire_at = 0;
  67. return true;
  68. });
  69. }
  70. void desktop_locked_render(Canvas* canvas, void* model) {
  71. DesktopLockedViewModel* m = model;
  72. uint32_t now = osKernelGetTickCount();
  73. canvas_clear(canvas);
  74. canvas_set_color(canvas, ColorBlack);
  75. if(!m->animation_seq_end) {
  76. canvas_draw_icon(canvas, m->door_left_x, 0, &I_DoorLeft_70x55);
  77. canvas_draw_icon(canvas, m->door_right_x, 0, &I_DoorRight_70x55);
  78. }
  79. if(m->animation && m->animation_seq_end) {
  80. canvas_draw_icon_animation(canvas, 0, -3, m->animation);
  81. }
  82. if(now < m->hint_expire_at) {
  83. if(!m->animation_seq_end) {
  84. canvas_set_font(canvas, FontPrimary);
  85. elements_multiline_text_framed(canvas, 42, 30, "Locked");
  86. } else {
  87. canvas_set_font(canvas, FontSecondary);
  88. canvas_draw_icon(canvas, 13, 5, &I_LockPopup_100x49);
  89. elements_multiline_text(canvas, 65, 20, "To unlock\npress:");
  90. }
  91. }
  92. }
  93. View* desktop_locked_get_view(DesktopLockedView* locked_view) {
  94. furi_assert(locked_view);
  95. return locked_view->view;
  96. }
  97. bool desktop_locked_input(InputEvent* event, void* context) {
  98. furi_assert(event);
  99. furi_assert(context);
  100. DesktopLockedView* locked_view = context;
  101. if(event->type == InputTypeShort) {
  102. desktop_locked_update_hint_timeout(locked_view);
  103. if(event->key == InputKeyBack) {
  104. uint32_t press_time = osKernelGetTickCount();
  105. // check if pressed sequentially
  106. if(press_time - locked_view->lock_lastpress > UNLOCK_RST_TIMEOUT) {
  107. locked_view->lock_lastpress = press_time;
  108. locked_view->lock_count = 0;
  109. } else if(press_time - locked_view->lock_lastpress < UNLOCK_RST_TIMEOUT) {
  110. locked_view->lock_lastpress = press_time;
  111. locked_view->lock_count++;
  112. }
  113. if(locked_view->lock_count == UNLOCK_CNT) {
  114. locked_view->lock_count = 0;
  115. locked_view->callback(DesktopLockedEventUnlock, locked_view->context);
  116. }
  117. }
  118. }
  119. // All events consumed
  120. return true;
  121. }
  122. void desktop_locked_enter(void* context) {
  123. DesktopLockedView* locked_view = context;
  124. with_view_model(
  125. locked_view->view, (DesktopLockedViewModel * model) {
  126. if(model->animation) icon_animation_start(model->animation);
  127. return false;
  128. });
  129. }
  130. void desktop_locked_exit(void* context) {
  131. DesktopLockedView* locked_view = context;
  132. with_view_model(
  133. locked_view->view, (DesktopLockedViewModel * model) {
  134. if(model->animation) icon_animation_stop(model->animation);
  135. return false;
  136. });
  137. }
  138. DesktopLockedView* desktop_locked_alloc() {
  139. DesktopLockedView* locked_view = furi_alloc(sizeof(DesktopLockedView));
  140. locked_view->view = view_alloc();
  141. locked_view->timer =
  142. osTimerNew(locked_view_timer_callback, osTimerPeriodic, locked_view, NULL);
  143. view_allocate_model(locked_view->view, ViewModelTypeLocking, sizeof(DesktopLockedViewModel));
  144. view_set_context(locked_view->view, locked_view);
  145. view_set_draw_callback(locked_view->view, (ViewDrawCallback)desktop_locked_render);
  146. view_set_input_callback(locked_view->view, desktop_locked_input);
  147. view_set_enter_callback(locked_view->view, desktop_locked_enter);
  148. view_set_exit_callback(locked_view->view, desktop_locked_exit);
  149. return locked_view;
  150. }
  151. void desktop_locked_free(DesktopLockedView* locked_view) {
  152. furi_assert(locked_view);
  153. osTimerDelete(locked_view->timer);
  154. view_free(locked_view->view);
  155. free(locked_view);
  156. }