Transition.cpp 4.2 KB

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