view.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  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. view_unlock_model(view);
  86. if(view->dispatcher) {
  87. view_dispatcher_update(view->dispatcher, view);
  88. }
  89. }
  90. void view_unlock_model(View* view) {
  91. furi_assert(view);
  92. if(view->model_type == ViewModelTypeLocking) {
  93. ViewModelLocking* model = (ViewModelLocking*)(view->model);
  94. furi_check(osMutexRelease(model->mutex) == osOK);
  95. }
  96. }
  97. void view_draw(View* view, Canvas* canvas) {
  98. furi_assert(view);
  99. if(view->draw_callback) {
  100. void* data = view_get_model(view);
  101. view->draw_callback(canvas, data);
  102. view_unlock_model(view);
  103. }
  104. }
  105. bool view_input(View* view, InputEvent* event) {
  106. furi_assert(view);
  107. if(view->input_callback) {
  108. return view->input_callback(event, view->context);
  109. } else {
  110. return false;
  111. }
  112. }
  113. uint32_t view_previous(View* view) {
  114. furi_assert(view);
  115. if(view->previous_callback) {
  116. return view->previous_callback(view->context);
  117. } else {
  118. return VIEW_IGNORE;
  119. }
  120. }
  121. uint32_t view_next(View* view) {
  122. furi_assert(view);
  123. if(view->next_callback) {
  124. return view->next_callback(view->context);
  125. } else {
  126. return VIEW_IGNORE;
  127. }
  128. }