Transition.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. #include <Transition.h>
  2. #include <Arduino.h>
  3. #include <cmath>
  4. Transition::Builder::Builder(size_t id, const BulbId& bulbId, TransitionFn callback, size_t maxSteps)
  5. : id(id)
  6. , bulbId(bulbId)
  7. , callback(callback)
  8. , duration(0)
  9. , period(0)
  10. , numPeriods(0)
  11. , maxSteps(maxSteps)
  12. { }
  13. Transition::Builder& Transition::Builder::setDuration(float duration) {
  14. this->duration = duration * DURATION_UNIT_MULTIPLIER;
  15. return *this;
  16. }
  17. void Transition::Builder::setDurationRaw(size_t duration) {
  18. this->duration = duration;
  19. }
  20. Transition::Builder& Transition::Builder::setPeriod(size_t period) {
  21. this->period = period;
  22. return *this;
  23. }
  24. Transition::Builder& Transition::Builder::setDurationAwarePeriod(size_t period, size_t duration, size_t maxSteps) {
  25. if ((period * maxSteps) < duration) {
  26. setPeriod(std::ceil(duration / static_cast<float>(maxSteps)));
  27. } else {
  28. setPeriod(period);
  29. }
  30. return *this;
  31. }
  32. size_t Transition::Builder::getNumPeriods() const {
  33. return this->numPeriods;
  34. }
  35. size_t Transition::Builder::getDuration() const {
  36. return this->duration;
  37. }
  38. size_t Transition::Builder::getPeriod() const {
  39. return this->period;
  40. }
  41. size_t Transition::Builder::getMaxSteps() const {
  42. return this->maxSteps;
  43. }
  44. bool Transition::Builder::isSetDuration() const {
  45. return this->duration > 0;
  46. }
  47. bool Transition::Builder::isSetPeriod() const {
  48. return this->period > 0;
  49. }
  50. bool Transition::Builder::isSetNumPeriods() const {
  51. return this->numPeriods > 0;
  52. }
  53. size_t Transition::Builder::numSetParams() const {
  54. size_t setCount = 0;
  55. if (isSetDuration()) { ++setCount; }
  56. if (isSetPeriod()) { ++setCount; }
  57. if (isSetNumPeriods()) { ++setCount; }
  58. return setCount;
  59. }
  60. size_t Transition::Builder::getOrComputePeriod() const {
  61. if (period > 0) {
  62. return period;
  63. } else if (duration > 0 && numPeriods > 0) {
  64. size_t computed = floor(duration / static_cast<float>(numPeriods));
  65. return max(MIN_PERIOD, computed);
  66. } else {
  67. return 0;
  68. }
  69. }
  70. size_t Transition::Builder::getOrComputeDuration() const {
  71. if (duration > 0) {
  72. return duration;
  73. } else if (period > 0 && numPeriods > 0) {
  74. return period * numPeriods;
  75. } else {
  76. return 0;
  77. }
  78. }
  79. size_t Transition::Builder::getOrComputeNumPeriods() const {
  80. if (numPeriods > 0) {
  81. return numPeriods;
  82. } else if (period > 0 && duration > 0) {
  83. size_t _numPeriods = ceil(duration / static_cast<float>(period));
  84. return max(static_cast<size_t>(1), _numPeriods);
  85. } else {
  86. return 0;
  87. }
  88. }
  89. std::shared_ptr<Transition> Transition::Builder::build() {
  90. // Set defaults for underspecified transitions
  91. size_t numSet = numSetParams();
  92. if (numSet == 0) {
  93. setDuration(DEFAULT_DURATION);
  94. setDurationAwarePeriod(DEFAULT_PERIOD, duration, maxSteps);
  95. } else if (numSet == 1) {
  96. // If duration is unbound, bind it
  97. if (! isSetDuration()) {
  98. setDurationRaw(DEFAULT_DURATION);
  99. // Otherwise, bind the period
  100. } else {
  101. setDurationAwarePeriod(DEFAULT_PERIOD, duration, maxSteps);
  102. }
  103. }
  104. return _build();
  105. }
  106. Transition::Transition(
  107. size_t id,
  108. const BulbId& bulbId,
  109. size_t period,
  110. TransitionFn callback
  111. ) : id(id)
  112. , bulbId(bulbId)
  113. , callback(callback)
  114. , period(period)
  115. , lastSent(0)
  116. { }
  117. void Transition::tick() {
  118. unsigned long now = millis();
  119. if ((lastSent + period) <= now
  120. && ((!isFinished() || lastSent == 0))) { // always send at least once
  121. step();
  122. lastSent = now;
  123. }
  124. }
  125. size_t Transition::calculatePeriod(int16_t distance, size_t stepSize, size_t duration) {
  126. float fPeriod =
  127. distance != 0
  128. ? (duration / (distance / static_cast<float>(stepSize)))
  129. : 0;
  130. return static_cast<size_t>(round(fPeriod));
  131. }
  132. void Transition::stepValue(int16_t& current, int16_t end, int16_t stepSize) {
  133. int16_t delta = end - current;
  134. if (std::abs(delta) < std::abs(stepSize)) {
  135. current += delta;
  136. } else {
  137. current += stepSize;
  138. }
  139. }
  140. void Transition::serialize(JsonObject& json) {
  141. json[F("id")] = id;
  142. json[F("period")] = period;
  143. json[F("last_sent")] = lastSent;
  144. JsonObject bulbParams = json.createNestedObject("bulb");
  145. bulbId.serialize(bulbParams);
  146. childSerialize(json);
  147. }