stepcounter.c 5.1 KB

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