view.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. #include "view_i.h"
  2. View* view_alloc() {
  3. View* view = furi_alloc(sizeof(View));
  4. return view;
  5. }
  6. void view_free(View* view) {
  7. furi_assert(view);
  8. view_free_model(view);
  9. free(view);
  10. }
  11. void view_set_dispatcher(View* view, ViewDispatcher* view_dispatcher) {
  12. furi_assert(view);
  13. furi_assert(view_dispatcher);
  14. furi_assert(view->dispatcher == NULL);
  15. view->dispatcher = view_dispatcher;
  16. }
  17. void view_set_draw_callback(View* view, ViewDrawCallback callback) {
  18. furi_assert(view);
  19. furi_assert(view->draw_callback == NULL);
  20. view->draw_callback = callback;
  21. }
  22. void view_set_input_callback(View* view, ViewInputCallback callback) {
  23. furi_assert(view);
  24. furi_assert(view->input_callback == NULL);
  25. view->input_callback = callback;
  26. }
  27. void view_set_previous_callback(View* view, ViewNavigationCallback callback) {
  28. furi_assert(view);
  29. view->previous_callback = callback;
  30. }
  31. void view_set_next_callback(View* view, ViewNavigationCallback callback) {
  32. furi_assert(view);
  33. view->next_callback = callback;
  34. }
  35. void view_set_context(View* view, void* context) {
  36. furi_assert(view);
  37. furi_assert(context);
  38. view->context = context;
  39. }
  40. void view_allocate_model(View* view, ViewModelType type, size_t size) {
  41. furi_assert(view);
  42. furi_assert(size > 0);
  43. furi_assert(view->model_type == ViewModelTypeNone);
  44. furi_assert(view->model == NULL);
  45. view->model_type = type;
  46. if(view->model_type == ViewModelTypeLockFree) {
  47. view->model = furi_alloc(size);
  48. } else if(view->model_type == ViewModelTypeLocking) {
  49. ViewModelLocking* model = furi_alloc(sizeof(ViewModelLocking));
  50. model->mutex = osMutexNew(NULL);
  51. furi_check(model->mutex);
  52. model->data = furi_alloc(size);
  53. view->model = model;
  54. } else {
  55. furi_assert(false);
  56. }
  57. }
  58. void view_free_model(View* view) {
  59. furi_assert(view);
  60. if(view->model_type == ViewModelTypeNone) {
  61. return;
  62. } else if(view->model_type == ViewModelTypeLockFree) {
  63. free(view->model);
  64. } else if(view->model_type == ViewModelTypeLocking) {
  65. ViewModelLocking* model = view->model;
  66. furi_check(osMutexDelete(model->mutex) == osOK);
  67. free(model->data);
  68. free(model);
  69. view->model = NULL;
  70. } else {
  71. furi_assert(false);
  72. }
  73. }
  74. void* view_get_model(View* view) {
  75. furi_assert(view);
  76. if(view->model_type == ViewModelTypeLocking) {
  77. ViewModelLocking* model = (ViewModelLocking*)(view->model);
  78. furi_check(osMutexAcquire(model->mutex, osWaitForever) == osOK);
  79. return model->data;
  80. }
  81. return view->model;
  82. }
  83. void view_commit_model(View* view) {
  84. furi_assert(view);
  85. if(view->model_type == ViewModelTypeLocking) {
  86. ViewModelLocking* model = (ViewModelLocking*)(view->model);
  87. furi_check(osMutexRelease(model->mutex) == osOK);
  88. }
  89. if(view->dispatcher) {
  90. view_dispatcher_update(view->dispatcher, view);
  91. }
  92. }
  93. void view_draw(View* view, Canvas* canvas) {
  94. furi_assert(view);
  95. if(view->draw_callback) {
  96. void* data = view_get_model(view);
  97. view->draw_callback(canvas, data);
  98. view_commit_model(view);
  99. }
  100. }
  101. bool view_input(View* view, InputEvent* event) {
  102. furi_assert(view);
  103. if(view->input_callback) {
  104. return view->input_callback(event, view->context);
  105. } else {
  106. return false;
  107. }
  108. }
  109. uint32_t view_previous(View* view) {
  110. furi_assert(view);
  111. if(view->previous_callback) {
  112. return view->previous_callback(view->context);
  113. } else {
  114. return VIEW_IGNORE;
  115. }
  116. }
  117. uint32_t view_next(View* view) {
  118. furi_assert(view);
  119. if(view->next_callback) {
  120. return view->next_callback(view->context);
  121. } else {
  122. return VIEW_IGNORE;
  123. }
  124. }