bpm.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. #include <furi.h>
  2. #include <dialogs/dialogs.h>
  3. #include <gui/gui.h>
  4. #include <input/input.h>
  5. #include <stdlib.h>
  6. typedef enum {
  7. EventTypeTick,
  8. EventTypeKey,
  9. } EventType;
  10. typedef struct {
  11. EventType type;
  12. InputEvent input;
  13. } PluginEvent;
  14. typedef struct {
  15. int taps;
  16. float interval;
  17. float bpm;
  18. } BPMTapper;
  19. static void show_hello() {
  20. // BEGIN HELLO DIALOG
  21. DialogsApp* dialogs = furi_record_open(RECORD_DIALOGS);
  22. DialogMessage* message = dialog_message_alloc();
  23. const char* header_text = "BPM Tapper";
  24. const char* message_text = "Tap center to start";
  25. dialog_message_set_header(message, header_text, 63, 3, AlignCenter, AlignTop);
  26. dialog_message_set_text(message, message_text, 0, 17, AlignLeft, AlignTop);
  27. dialog_message_set_buttons(message, NULL, "Tap", NULL);
  28. dialog_message_set_icon(message, &I_DolphinCommon_56x48, 72, 17);
  29. dialog_message_show(dialogs, message);
  30. dialog_message_free(message);
  31. furi_record_close(RECORD_DIALOGS);
  32. // END HELLO DIALOG
  33. }
  34. static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
  35. furi_assert(event_queue);
  36. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  37. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  38. }
  39. static void render_callback(Canvas* const canvas, void* ctx) {
  40. const BPMTapper* bpm_state = acquire_mutex((ValueMutex*)ctx, 25);
  41. if (bpm_state == NULL) {
  42. return;
  43. }
  44. // border
  45. canvas_draw_frame(canvas, 0, 0, 128, 64);
  46. canvas_set_font(canvas, FontPrimary);
  47. char taps[8];
  48. //sprintf(taps, "%d", bpm_state->taps);
  49. itoa(bpm_state->taps, taps, 10);
  50. canvas_draw_str_aligned(canvas, 5, 5, AlignRight, AlignBottom, taps);
  51. release_mutex((ValueMutex*)ctx, bpm_state);
  52. }
  53. static void bpm_state_init(BPMTapper* const plugin_state) {
  54. plugin_state->taps = 0;
  55. plugin_state->bpm = 0.0;
  56. }
  57. int32_t bpm_tapper_app(void* p) {
  58. UNUSED(p);
  59. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  60. BPMTapper* bpm_state = malloc(sizeof(BPMTapper));
  61. // setup
  62. bpm_state_init(bpm_state);
  63. ValueMutex state_mutex;
  64. if (!init_mutex(&state_mutex, bpm_state, sizeof(bpm_state))) {
  65. FURI_LOG_E("BPM-Tapper", "cannot create mutex\r\n");
  66. free(bpm_state);
  67. return 255;
  68. }
  69. show_hello();
  70. // BEGIN IMPLEMENTATION
  71. // Set system callbacks
  72. ViewPort* view_port = view_port_alloc();
  73. view_port_draw_callback_set(view_port, render_callback, &state_mutex);
  74. view_port_input_callback_set(view_port, input_callback, event_queue);
  75. // Open GUI and register view_port
  76. Gui* gui = furi_record_open("gui");
  77. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  78. PluginEvent event;
  79. for (bool processing = true; processing;) {
  80. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  81. BPMTapper* bpm_state = (BPMTapper*)acquire_mutex_block(&state_mutex);
  82. if(event_status == FuriStatusOk) {
  83. // press events
  84. if(event.type == EventTypeKey) {
  85. if(event.input.type == InputTypePress) {
  86. switch(event.input.key) {
  87. case InputKeyUp:
  88. case InputKeyDown:
  89. case InputKeyRight:
  90. case InputKeyLeft:
  91. case InputKeyOk:
  92. bpm_state->taps++;
  93. break;
  94. case InputKeyBack:
  95. // Exit the plugin
  96. processing = false;
  97. break;
  98. }
  99. }
  100. }
  101. } else {
  102. FURI_LOG_D("BPM-Tapper", "FuriMessageQueue: event timeout");
  103. // event timeout
  104. }
  105. view_port_update(view_port);
  106. release_mutex(&state_mutex, bpm_state);
  107. }
  108. view_port_enabled_set(view_port, false);
  109. gui_remove_view_port(gui, view_port);
  110. furi_record_close("gui");
  111. view_port_free(view_port);
  112. furi_message_queue_free(event_queue);
  113. delete_mutex(&state_mutex);
  114. //free(bpm_state);
  115. return 0;
  116. }