bpm.c 7.6 KB

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