Atm_encoderInt.cpp 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. #include "Atm_encoderInt.h"
  2. #include <limits.h>
  3. // Loosely based on https://www.circuitsathome.com/mcu/reading-rotary-encoder-on-arduino (Oleg Mazurov)
  4. const char Atm_encoderInt::enc_states[16] = {0, (char)-1, 1, 0, 1, 0, 0, (char)-1, (char)-1, 0, 0, 1, 0, 1, (char)-1, 0};
  5. Atm_encoderInt& Atm_encoderInt::begin( int pin1, int pin2, int divider /* = 1 */ ) {
  6. // clang-format off
  7. const static state_t state_table[] PROGMEM = {
  8. /* ON_ENTER ON_LOOP ON_EXIT EVT_UP EVT_DOWN ELSE */
  9. /* IDLE */ -1, LP_IDLE, -1, UP, DOWN, -1,
  10. /* UP */ ENT_UP, -1, -1, -1, -1, IDLE,
  11. /* DOWN */ ENT_DOWN, -1, -1, -1, -1, IDLE,
  12. };
  13. // clang-format on
  14. Machine::begin( state_table, ELSE );
  15. this->pin1 = pin1;
  16. this->pin2 = pin2;
  17. this->divider = divider;
  18. encoder = new Encoder(pin1, pin2);
  19. // pinMode( pin1, INPUT );
  20. // pinMode( pin2, INPUT );
  21. // digitalWrite( pin1, HIGH );
  22. // digitalWrite( pin2, HIGH );
  23. min = INT_MIN;
  24. max = INT_MAX;
  25. value = 0;
  26. return *this;
  27. }
  28. int Atm_encoderInt::event( int id ) {
  29. switch ( id ) {
  30. case EVT_UP:
  31. return enc_direction == +1 && ( enc_counter % divider == 0 );
  32. case EVT_DOWN:
  33. return enc_direction == -1 && ( enc_counter % divider == 0 );
  34. }
  35. return 0;
  36. }
  37. void Atm_encoderInt::action( int id ) {
  38. int8_t enc_counter_prev = enc_counter;
  39. switch ( id ) {
  40. case LP_IDLE:
  41. // enc_bits = ( ( enc_bits << 2 ) | ( digitalRead( pin1 ) << 1 ) | ( digitalRead( pin2 ) ) ) & 0x0f;
  42. // enc_direction = enc_states[enc_bits];
  43. // if ( enc_direction != 0 ) {
  44. // enc_counter = enc_counter + enc_direction;
  45. // if ( ( enc_counter != 0 ) && ( enc_counter % divider == 0 ) ) {
  46. // if ( !count( enc_direction ) ) {
  47. // enc_direction = 0;
  48. // }
  49. // } // enc_bits = ( ( enc_bits << 2 ) | ( digitalRead( pin1 ) << 1 ) | ( digitalRead( pin2 ) ) ) & 0x0f;
  50. // enc_direction = enc_states[enc_bits];
  51. // if ( enc_direction != 0 ) {
  52. // enc_counter = enc_counter + enc_direction;
  53. // if ( ( enc_counter != 0 ) && ( enc_counter % divider == 0 ) ) {
  54. // if ( !count( enc_direction ) ) {
  55. // enc_direction = 0;
  56. // }
  57. // }
  58. // }
  59. // }
  60. enc_counter = encoder->read();
  61. enc_direction = enc_counter > enc_counter_prev ? 1 : -1;
  62. enc_direction = enc_counter == enc_counter_prev ? 0 : enc_direction;
  63. count(enc_direction);
  64. return;
  65. case ENT_UP:
  66. onup.push( state(), 1 );
  67. return;
  68. case ENT_DOWN:
  69. ondown.push( state(), 0 );
  70. return;
  71. }
  72. }
  73. Atm_encoderInt& Atm_encoderInt::range( int min, int max, bool wrap /* = false */ ) {
  74. if ( min > max ) {
  75. range_invert = true;
  76. this->min = max;
  77. this->max = min;
  78. } else {
  79. range_invert = false;
  80. this->min = min;
  81. this->max = max;
  82. }
  83. this->wrap = wrap;
  84. if ( value < min || value > max ) {
  85. value = min;
  86. }
  87. return *this;
  88. }
  89. Atm_encoderInt& Atm_encoderInt::set( int value ) {
  90. this->value = range_invert ? map( value, min, max, max, min ) : value;
  91. return *this;
  92. }
  93. Atm_encoderInt& Atm_encoderInt::onChange( Machine& machine, int event /* = 0 */ ) {
  94. onup.set( &machine, event );
  95. ondown.set( &machine, event );
  96. return *this;
  97. }
  98. Atm_encoderInt& Atm_encoderInt::onChange( atm_cb_push_t callback, int idx /* = 0 */ ) {
  99. onup.set( callback, idx );
  100. ondown.set( callback, idx );
  101. return *this;
  102. }
  103. Atm_encoderInt& Atm_encoderInt::onChange( bool status, Machine& machine, int event /* = 0 */ ) {
  104. if ( status ) {
  105. onup.set( &machine, event );
  106. } else {
  107. ondown.set( &machine, event );
  108. }
  109. return *this;
  110. }
  111. Atm_encoderInt& Atm_encoderInt::onChange( bool status, atm_cb_push_t callback, int idx /* = 0 */ ) {
  112. if ( status ) {
  113. onup.set( callback, idx );
  114. } else {
  115. ondown.set( callback, idx );
  116. }
  117. return *this;
  118. }
  119. int Atm_encoderInt::state( void ) {
  120. return range_invert ? map( value, min, max, max, min ) : value;
  121. }
  122. bool Atm_encoderInt::count( int direction ) {
  123. if ( (long)value + direction > max ) {
  124. if ( wrap ) {
  125. value = min;
  126. } else {
  127. return false;
  128. }
  129. } else if ( (long)value + direction < min ) {
  130. if ( wrap ) {
  131. value = max;
  132. } else {
  133. return false;
  134. }
  135. } else {
  136. value += direction;
  137. }
  138. return true;
  139. }
  140. Atm_encoderInt& Atm_encoderInt::trace( Stream& stream ) {
  141. Machine::setTrace( &stream, atm_serial_debug::trace, "ENCODER\0EVT_UP\0EVT_DOWN\0ELSE\0IDLE\0UP\0DOWN" );
  142. return *this;
  143. }