bpm.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <dialogs/dialogs.h>
  4. #include <gui/gui.h>
  5. #include <input/input.h>
  6. #include <m-string.h>
  7. #include <stdlib.h>
  8. typedef enum {
  9. EventTypeTick,
  10. EventTypeKey,
  11. } EventType;
  12. typedef struct {
  13. EventType type;
  14. InputEvent input;
  15. } PluginEvent;
  16. //QUEUE
  17. struct node {
  18. int interval;
  19. struct node *next;
  20. };
  21. typedef struct node node;
  22. typedef struct {
  23. int size;
  24. int max_size;
  25. node *front;
  26. node *rear;
  27. } queue;
  28. static void init_queue(queue *q) {
  29. q->size = 0;
  30. q->max_size = 5;
  31. q->front = NULL;
  32. q->rear = NULL;
  33. }
  34. static void queue_remove(queue *q) {
  35. node *tmp;
  36. tmp = q->front;
  37. q->front = q->front->next;
  38. q->size--;
  39. free(tmp);
  40. }
  41. static void queue_add(queue *q, int value) {
  42. node *tmp = malloc(sizeof(node));
  43. tmp->interval = value;
  44. tmp->next = NULL;
  45. if (q->size == q->max_size) {
  46. queue_remove(q);
  47. }
  48. // check if empty
  49. if (q->rear == NULL) {
  50. q->front = tmp;
  51. q->rear = tmp;
  52. } else {
  53. q->rear->next = tmp;
  54. q->rear = tmp;
  55. }
  56. q->size++;
  57. }
  58. static float queue_avg(queue *q) {
  59. float avg = 0.0;
  60. if (q->size == 0){
  61. return avg;
  62. } else {
  63. node *tmp;
  64. float sum = 0.0;
  65. tmp = q->front;
  66. while (tmp != NULL) {
  67. sum = sum + tmp->interval;
  68. tmp = tmp->next;
  69. }
  70. avg = sum / q->size;
  71. return avg;
  72. }
  73. }
  74. // TOO SLOW!
  75. //uint64_t dolphin_state_timestamp() {
  76. // FuriHalRtcDateTime datetime;
  77. // furi_hal_rtc_get_datetime(&datetime);
  78. // return furi_hal_rtc_datetime_to_timestamp(&datetime);
  79. //}
  80. //
  81. typedef struct {
  82. int taps;
  83. double bpm;
  84. uint32_t last_stamp;
  85. uint32_t interval;
  86. queue *tap_queue;
  87. } BPMTapper;
  88. static void show_hello() {
  89. // BEGIN HELLO DIALOG
  90. DialogsApp* dialogs = furi_record_open(RECORD_DIALOGS);
  91. DialogMessage* message = dialog_message_alloc();
  92. const char* header_text = "BPM Tapper";
  93. const char* message_text = "Tap center to start";
  94. dialog_message_set_header(message, header_text, 63, 3, AlignCenter, AlignTop);
  95. dialog_message_set_text(message, message_text, 0, 17, AlignLeft, AlignTop);
  96. dialog_message_set_buttons(message, NULL, "Tap", NULL);
  97. dialog_message_set_icon(message, &I_DolphinCommon_56x48, 72, 17);
  98. dialog_message_show(dialogs, message);
  99. dialog_message_free(message);
  100. furi_record_close(RECORD_DIALOGS);
  101. // END HELLO DIALOG
  102. }
  103. static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
  104. furi_assert(event_queue);
  105. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  106. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  107. }
  108. static void render_callback(Canvas* const canvas, void* ctx) {
  109. string_t tempStr;
  110. const BPMTapper* bpm_state = acquire_mutex((ValueMutex*)ctx, 25);
  111. if (bpm_state == NULL) {
  112. return;
  113. }
  114. // border
  115. //canvas_draw_frame(canvas, 0, 0, 128, 64);
  116. canvas_set_font(canvas, FontPrimary);
  117. string_init(tempStr);
  118. string_printf(tempStr, "Taps: %d", bpm_state->taps);
  119. canvas_draw_str_aligned(canvas, 5, 15, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  120. string_reset(tempStr);
  121. string_printf(tempStr, "Queue: %d", bpm_state->tap_queue->size);
  122. canvas_draw_str_aligned(canvas, 50, 15, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  123. string_reset(tempStr);
  124. string_printf(tempStr, "Interval: %dms", bpm_state->interval);
  125. canvas_draw_str_aligned(canvas, 5, 25, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  126. string_reset(tempStr);
  127. string_printf(tempStr, "%.2f", bpm_state->bpm);
  128. canvas_set_font(canvas, FontBigNumbers);
  129. canvas_draw_str_aligned(canvas, 20, 50, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  130. string_reset(tempStr);
  131. string_clear(tempStr);
  132. release_mutex((ValueMutex*)ctx, bpm_state);
  133. }
  134. static void bpm_state_init(BPMTapper* const plugin_state) {
  135. plugin_state->taps = 0;
  136. plugin_state->bpm = 120.0;
  137. plugin_state->last_stamp = 0;
  138. plugin_state->interval = 500;
  139. queue *q;
  140. q = malloc(sizeof(queue));
  141. init_queue(q);
  142. plugin_state->tap_queue = q;
  143. }
  144. int32_t bpm_tapper_app(void* p) {
  145. UNUSED(p);
  146. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  147. BPMTapper* bpm_state = malloc(sizeof(BPMTapper));
  148. // setup
  149. bpm_state_init(bpm_state);
  150. ValueMutex state_mutex;
  151. if (!init_mutex(&state_mutex, bpm_state, sizeof(bpm_state))) {
  152. FURI_LOG_E("BPM-Tapper", "cannot create mutex\r\n");
  153. free(bpm_state);
  154. return 255;
  155. }
  156. show_hello();
  157. // BEGIN IMPLEMENTATION
  158. // Set system callbacks
  159. ViewPort* view_port = view_port_alloc();
  160. view_port_draw_callback_set(view_port, render_callback, &state_mutex);
  161. view_port_input_callback_set(view_port, input_callback, event_queue);
  162. // Open GUI and register view_port
  163. Gui* gui = furi_record_open("gui");
  164. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  165. PluginEvent event;
  166. for (bool processing = true; processing;) {
  167. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  168. BPMTapper* bpm_state = (BPMTapper*)acquire_mutex_block(&state_mutex);
  169. if(event_status == FuriStatusOk) {
  170. // press events
  171. if(event.type == EventTypeKey) {
  172. if(event.input.type == InputTypePress) {
  173. switch(event.input.key) {
  174. case InputKeyUp:
  175. case InputKeyDown:
  176. case InputKeyRight:
  177. case InputKeyLeft:
  178. case InputKeyOk:
  179. bpm_state->taps++;
  180. uint32_t new_stamp = furi_get_tick();
  181. bpm_state->interval = new_stamp - bpm_state->last_stamp;
  182. bpm_state->last_stamp = new_stamp;
  183. queue_add(bpm_state->tap_queue, bpm_state->interval);
  184. float avg = queue_avg(bpm_state->tap_queue);
  185. float bps = 1.0 / (avg / 1000.0);
  186. bpm_state->bpm = bps * 60.0;
  187. break;
  188. case InputKeyBack:
  189. // Exit the plugin
  190. processing = false;
  191. break;
  192. }
  193. }
  194. }
  195. } else {
  196. FURI_LOG_D("BPM-Tapper", "FuriMessageQueue: event timeout");
  197. // event timeout
  198. }
  199. view_port_update(view_port);
  200. release_mutex(&state_mutex, bpm_state);
  201. }
  202. view_port_enabled_set(view_port, false);
  203. gui_remove_view_port(gui, view_port);
  204. furi_record_close("gui");
  205. view_port_free(view_port);
  206. furi_message_queue_free(event_queue);
  207. delete_mutex(&state_mutex);
  208. queue *q = bpm_state->tap_queue;
  209. free(q);
  210. free(bpm_state);
  211. return 0;
  212. }