Transition.cpp 4.2 KB

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