bpm.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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. typedef struct {
  17. int taps;
  18. double bpm;
  19. uint32_t last_stamp;
  20. uint32_t interval;
  21. } BPMTapper;
  22. // TOO SLOW!
  23. //uint64_t dolphin_state_timestamp() {
  24. // FuriHalRtcDateTime datetime;
  25. // furi_hal_rtc_get_datetime(&datetime);
  26. // return furi_hal_rtc_datetime_to_timestamp(&datetime);
  27. //}
  28. static void show_hello() {
  29. // BEGIN HELLO DIALOG
  30. DialogsApp* dialogs = furi_record_open(RECORD_DIALOGS);
  31. DialogMessage* message = dialog_message_alloc();
  32. const char* header_text = "BPM Tapper";
  33. const char* message_text = "Tap center to start";
  34. dialog_message_set_header(message, header_text, 63, 3, AlignCenter, AlignTop);
  35. dialog_message_set_text(message, message_text, 0, 17, AlignLeft, AlignTop);
  36. dialog_message_set_buttons(message, NULL, "Tap", NULL);
  37. dialog_message_set_icon(message, &I_DolphinCommon_56x48, 72, 17);
  38. dialog_message_show(dialogs, message);
  39. dialog_message_free(message);
  40. furi_record_close(RECORD_DIALOGS);
  41. // END HELLO DIALOG
  42. }
  43. static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
  44. furi_assert(event_queue);
  45. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  46. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  47. }
  48. static void render_callback(Canvas* const canvas, void* ctx) {
  49. string_t tempStr;
  50. const BPMTapper* bpm_state = acquire_mutex((ValueMutex*)ctx, 25);
  51. if (bpm_state == NULL) {
  52. return;
  53. }
  54. // border
  55. //canvas_draw_frame(canvas, 0, 0, 128, 64);
  56. canvas_set_font(canvas, FontPrimary);
  57. string_init(tempStr);
  58. string_printf(tempStr, "Taps: %d", bpm_state->taps);
  59. canvas_draw_str_aligned(canvas, 5, 15, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  60. string_reset(tempStr);
  61. string_printf(tempStr, "Interval: %dms", bpm_state->interval);
  62. canvas_draw_str_aligned(canvas, 5, 25, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  63. string_reset(tempStr);
  64. string_printf(tempStr, "%.2f", bpm_state->bpm);
  65. canvas_set_font(canvas, FontBigNumbers);
  66. canvas_draw_str_aligned(canvas, 20, 50, AlignLeft, AlignBottom, string_get_cstr(tempStr));
  67. string_reset(tempStr);
  68. string_clear(tempStr);
  69. release_mutex((ValueMutex*)ctx, bpm_state);
  70. }
  71. static void bpm_state_init(BPMTapper* const plugin_state) {
  72. plugin_state->taps = 0;
  73. plugin_state->bpm = 120.0;
  74. plugin_state->last_stamp = 0;
  75. plugin_state->interval = 500;
  76. }
  77. int32_t bpm_tapper_app(void* p) {
  78. UNUSED(p);
  79. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  80. BPMTapper* bpm_state = malloc(sizeof(BPMTapper));
  81. // setup
  82. bpm_state_init(bpm_state);
  83. ValueMutex state_mutex;
  84. if (!init_mutex(&state_mutex, bpm_state, sizeof(bpm_state))) {
  85. FURI_LOG_E("BPM-Tapper", "cannot create mutex\r\n");
  86. free(bpm_state);
  87. return 255;
  88. }
  89. show_hello();
  90. // BEGIN IMPLEMENTATION
  91. // Set system callbacks
  92. ViewPort* view_port = view_port_alloc();
  93. view_port_draw_callback_set(view_port, render_callback, &state_mutex);
  94. view_port_input_callback_set(view_port, input_callback, event_queue);
  95. // Open GUI and register view_port
  96. Gui* gui = furi_record_open("gui");
  97. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  98. PluginEvent event;
  99. for (bool processing = true; processing;) {
  100. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  101. BPMTapper* bpm_state = (BPMTapper*)acquire_mutex_block(&state_mutex);
  102. if(event_status == FuriStatusOk) {
  103. // press events
  104. if(event.type == EventTypeKey) {
  105. if(event.input.type == InputTypePress) {
  106. switch(event.input.key) {
  107. case InputKeyUp:
  108. case InputKeyDown:
  109. case InputKeyRight:
  110. case InputKeyLeft:
  111. case InputKeyOk:
  112. bpm_state->taps++;
  113. uint32_t new_stamp = furi_get_tick();
  114. bpm_state->interval = new_stamp - bpm_state->last_stamp;
  115. bpm_state->last_stamp = new_stamp;
  116. float bps = 1.0 / (bpm_state->interval / 1000.0);
  117. bpm_state->bpm = bps * 60.0;
  118. break;
  119. case InputKeyBack:
  120. // Exit the plugin
  121. processing = false;
  122. break;
  123. }
  124. }
  125. }
  126. } else {
  127. FURI_LOG_D("BPM-Tapper", "FuriMessageQueue: event timeout");
  128. // event timeout
  129. }
  130. view_port_update(view_port);
  131. release_mutex(&state_mutex, bpm_state);
  132. }
  133. view_port_enabled_set(view_port, false);
  134. gui_remove_view_port(gui, view_port);
  135. furi_record_close("gui");
  136. view_port_free(view_port);
  137. furi_message_queue_free(event_queue);
  138. delete_mutex(&state_mutex);
  139. //free(bpm_state);
  140. return 0;
  141. }