Atm_encoderInt.cpp 4.5 KB

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