stepcounter.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. #include <furi.h>
  2. #include <furi_hal.h>
  3. #include <furi_hal_gpio.h>
  4. #include <furi_hal_resources.h>
  5. #include <gui/gui.h>
  6. #include <input/input.h>
  7. #include <locale/locale.h>
  8. #include <notification/notification.h>
  9. #include <notification/notification_messages.h>
  10. #define TAG_MEMSIC "memsic_2125_app"
  11. #define TAG_COUNTER "step_counter_app"
  12. typedef struct {
  13. const GpioPin* pin;
  14. bool prevState;
  15. uint32_t stepCount;
  16. bool counting;
  17. uint32_t time_of_last_high_pulse;
  18. uint32_t time_of_high_to_high;
  19. } StepCounterData;
  20. typedef enum {
  21. StepCounterEventTypeStep,
  22. StepCounterEventTypeKey,
  23. } StepCounterEventType;
  24. typedef struct {
  25. StepCounterEventType type;
  26. InputEvent input_event;
  27. } StepCounterEvent;
  28. typedef struct {
  29. FuriMessageQueue* queue;
  30. StepCounterData* data;
  31. ViewPort* view_port;
  32. } StepCounterContext;
  33. const GpioPin* const gpio_accelerometer = &gpio_ext_pc0;
  34. void step_callback(void* ctx) {
  35. StepCounterContext* context = (StepCounterContext*)ctx;
  36. StepCounterData* stepData = context->data;
  37. FuriHalCortexTimer timer = furi_hal_cortex_timer_get(0);
  38. uint32_t now = timer.start;
  39. if (furi_hal_gpio_read(stepData->pin)) {
  40. if (stepData->time_of_last_high_pulse != 0) {
  41. stepData->time_of_high_to_high = now - stepData->time_of_last_high_pulse;
  42. }
  43. stepData->time_of_last_high_pulse = now;
  44. } else {
  45. uint32_t high_duration = now - stepData->time_of_last_high_pulse;
  46. bool current_state = high_duration < (stepData->time_of_high_to_high >> 1);
  47. if(current_state != stepData->prevState) {
  48. stepData->prevState = current_state;
  49. stepData->stepCount++;
  50. StepCounterEvent event = {.type = StepCounterEventTypeStep};
  51. furi_message_queue_put(context->queue, &event, 0);
  52. }
  53. }
  54. }
  55. static void input_callback(InputEvent* input_event, FuriMessageQueue* queue) {
  56. StepCounterEvent event = {
  57. .type = StepCounterEventTypeKey,
  58. .input_event.key = input_event->key,
  59. .input_event.type = input_event->type,
  60. };
  61. furi_message_queue_put(queue, &event, FuriWaitForever);
  62. }
  63. static void render_callback(Canvas* canvas, void* ctx) {
  64. StepCounterContext* stepContext = (StepCounterContext*)ctx;
  65. StepCounterData* stepData = stepContext->data;
  66. char stepText[20];
  67. snprintf(stepText, sizeof(stepText), "Steps: %ld", stepData->stepCount);
  68. canvas_draw_str_aligned(canvas, 1, 1, AlignLeft, AlignTop, stepText);
  69. char buttonText[10];
  70. snprintf(buttonText, sizeof(buttonText), stepData->counting ? "STOP" : "START");
  71. canvas_draw_str_aligned(
  72. canvas,
  73. canvas_width(canvas) / 2,
  74. canvas_height(canvas) - 1,
  75. AlignCenter,
  76. AlignBottom,
  77. buttonText);
  78. }
  79. int32_t step_counter_app(void* p) {
  80. UNUSED(p);
  81. StepCounterContext* stepContext = malloc(sizeof(StepCounterContext));
  82. stepContext->data = malloc(sizeof(StepCounterData));
  83. stepContext->data->pin = gpio_accelerometer;
  84. stepContext->data->prevState = furi_hal_gpio_read(stepContext->data->pin);
  85. stepContext->data->stepCount = 0;
  86. stepContext->data->counting = false;
  87. stepContext->queue = furi_message_queue_alloc(8, sizeof(StepCounterEvent));
  88. furi_hal_gpio_init(
  89. stepContext->data->pin, GpioModeInterruptRiseFall, GpioPullNo, GpioSpeedVeryHigh);
  90. furi_hal_gpio_add_int_callback(stepContext->data->pin, step_callback, stepContext);
  91. ViewPort* view_port = view_port_alloc();
  92. view_port_draw_callback_set(view_port, render_callback, stepContext);
  93. view_port_input_callback_set(view_port, input_callback, stepContext->queue);
  94. Gui* gui = furi_record_open(RECORD_GUI);
  95. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  96. StepCounterEvent event;
  97. bool processing = true;
  98. do {
  99. if(furi_message_queue_get(stepContext->queue, &event, FuriWaitForever) == FuriStatusOk) {
  100. switch(event.type) {
  101. case StepCounterEventTypeKey:
  102. if(event.input_event.type == InputTypeShort &&
  103. event.input_event.key == InputKeyBack) {
  104. processing = false;
  105. } else if(
  106. event.input_event.type == InputTypeShort &&
  107. event.input_event.key == InputKeyOk) {
  108. stepContext->data->counting = !stepContext->data->counting;
  109. view_port_update(view_port);
  110. }
  111. break;
  112. case StepCounterEventTypeStep:
  113. view_port_update(view_port);
  114. break;
  115. default:
  116. break;
  117. }
  118. view_port_update(view_port);
  119. } else {
  120. processing = false;
  121. }
  122. } while(processing);
  123. furi_hal_gpio_remove_int_callback(stepContext->data->pin);
  124. view_port_enabled_set(view_port, false);
  125. gui_remove_view_port(gui, view_port);
  126. view_port_free(view_port);
  127. furi_record_close(RECORD_GUI);
  128. furi_message_queue_free(stepContext->queue);
  129. free(stepContext->data);
  130. free(stepContext);
  131. return 0;
  132. }