#pragma once #include "Ember.h" /// /// Interpolater class. /// namespace EmberNs { /// /// g++ needs a forward declaration here. /// template class Ember; /// /// Contains many static functions for handling interpolation and other miscellaneous operations on /// embers and vectors of embers. This class is similar to, and used in conjunction with SheepTools. /// Template argument expected to be float or double. /// template class EMBER_API Interpolater { public: /// /// Aligns the specified array of embers and stores in the output array. /// This is used to prepare embers before interpolating them. /// Alignment means that every ember in a list will have the same number of xforms. /// Each xform at a given position will have mostly the same variations as the xform /// in the same position in the rest of the embers. However some /// intelligence is applied to add or remove variations that wouldn't look good with /// the others present. /// After this function completes, sourceEmbers will remain unchanged and destEmbers /// will contain the aligned list of embers from sourceEmbers. /// /// The array of embers to align /// The array which will contain the aligned embers /// The number of elements in sourceEmbers static void Align(Ember* sourceEmbers, Ember* destEmbers, size_t count) { bool aligned = true; bool currentFinal, final = sourceEmbers[0].UseFinalXform(); size_t i, xf, currentCount, maxCount = sourceEmbers[0].XformCount(); Xform* destXform; Xform* destOtherXform; //Determine the max number of xforms present in sourceEmbers. //Also check if final xforms are used in any of them. for (i = 1; i < count; i++) { currentCount = sourceEmbers[i].XformCount(); if (currentCount != maxCount)//Any difference, less or more, means unaligned. { aligned = false; if (currentCount > maxCount) maxCount = currentCount; } currentFinal = sourceEmbers[i].UseFinalXform(); if (final != currentFinal)//Check if any used final. { aligned = false; final |= currentFinal; } } //Copy them using the max xform count, and do final if any had final. for (i = 0; i < count; i++) destEmbers[i] = sourceEmbers[i].Copy(maxCount, final); if (final) maxCount++; //Check to see if there's a parametric variation present in one xform //but not in an aligned xform. If this is the case, use the parameters //from the xform with the variation as the defaults for the blank one. //All embers will have the same number of xforms at this point. for (i = 0; i < count; i++) { size_t ii; for (xf = 0; xf < maxCount; xf++)//This will include both normal xforms and the final. { destXform = destEmbers[i].GetTotalXform(xf, final); //Ensure every parametric variation contained in every xform at either position i - 1 or i + 1 is also contained in the dest xform. if (i > 0) destOtherXform = destEmbers[i - 1].GetTotalXform(xf); else if (i < count - 1) destOtherXform = destEmbers[i + 1].GetTotalXform(xf); else destOtherXform = nullptr;//Should never happen if (destOtherXform) MergeXformVariations1Way(destOtherXform, destXform, true, true); //This is a new xform. Let's see if it's possible to choose a better 'identity' xform. //Check the neighbors to see if any of these variations are used: //rings2, fan2, blob, perspective, julian, juliascope, ngon, curl, super_shape, split //If so, can use a better starting point for these. //If the current xform index is greater than what the original xform count was for this ember, then it's a padding xform. if (xf >= sourceEmbers[i].TotalXformCount() && !aligned) { size_t found = 0; //Remove linear. destXform->DeleteVariationById(eVariationId::VAR_LINEAR); //Only do the next substitution for log interpolation. if ((i == 0 && destEmbers[i].m_AffineInterp == eAffineInterp::AFFINE_INTERP_LOG) || (i > 0 && destEmbers[i - 1].m_AffineInterp == eAffineInterp::AFFINE_INTERP_LOG)) { for (ii = -1; ii <= 1; ii += 2) { //Skip if out of bounds. if (i + ii < 0 || i + ii >= count) continue; //Skip if this is also padding. if (xf >= sourceEmbers[i + ii].TotalXformCount()) continue; destOtherXform = destEmbers[i + ii].GetTotalXform(xf); //Spherical / Ngon (trumps all others due to holes) //Interpolate these against a 180 degree rotated identity //with weight -1. //Added JULIAN/JULIASCOPE to get rid of black wedges. if (destOtherXform->GetVariationById(eVariationId::VAR_SPHERICAL) || destOtherXform->GetVariationById(eVariationId::VAR_NGON) || destOtherXform->GetVariationById(eVariationId::VAR_JULIAN) || destOtherXform->GetVariationById(eVariationId::VAR_JULIASCOPE) || destOtherXform->GetVariationById(eVariationId::VAR_POLAR) || destOtherXform->GetVariationById(eVariationId::VAR_WEDGE_SPH) || destOtherXform->GetVariationById(eVariationId::VAR_WEDGE_JULIA)) { destXform->AddVariation(new LinearVariation(-1)); //Set the coefs appropriately. destXform->m_Affine.A(-1); destXform->m_Affine.D(0); destXform->m_Affine.B(0); destXform->m_Affine.E(-1); destXform->m_Affine.C(0); destXform->m_Affine.F(0); found = -1; } } } if (found == 0) { for (ii = -1; ii <= 1; ii += 2) { //Skip if out of bounds. if (i + ii < 0 || i + ii >= count) continue; //Skip if this is also padding. if (xf >= sourceEmbers[i + ii].TotalXformCount()) continue; destOtherXform = destEmbers[i + ii].GetTotalXform(xf); if (destOtherXform->GetVariationById(eVariationId::VAR_RECTANGLES)) { RectanglesVariation* var = new RectanglesVariation(); var->SetParamVal("rectangles_x", 0); var->SetParamVal("rectangles_y", 0); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_RINGS2)) { Rings2Variation* var = new Rings2Variation(); var->SetParamVal("rings2_val", 0); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_FAN2)) { Fan2Variation* var = new Fan2Variation(); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_BLOB)) { BlobVariation* var = new BlobVariation(); var->SetParamVal("blob_low", 1); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_PERSPECTIVE)) { PerspectiveVariation* var = new PerspectiveVariation(); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_CURL)) { CurlVariation* var = new CurlVariation(); var->SetParamVal("curl_c1", 0); destXform->AddVariation(var); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_SUPER_SHAPE)) { SuperShapeVariation* var = new SuperShapeVariation(); var->SetParamVal("super_shape_n1", 2); var->SetParamVal("super_shape_n2", 2); var->SetParamVal("super_shape_n3", 2); destXform->AddVariation(var); found++; } } } //If none matched those, try the affine ones, fan and rings. if (found == 0) { for (ii = -1; ii <= 1; ii += 2) { //Skip if out of bounds. if (i + ii < 0 || i + ii >= count) continue; //Skip if this is also padding. if (xf >= sourceEmbers[i + ii].TotalXformCount()) continue; destOtherXform = destEmbers[i + ii].GetTotalXform(xf); if (destOtherXform->GetVariationById(eVariationId::VAR_FAN)) { destXform->AddVariation(new FanVariation()); found++; } if (destOtherXform->GetVariationById(eVariationId::VAR_RINGS)) { destXform->AddVariation(new RingsVariation()); found++; } } if (found > 0) { //Set the coefs appropriately. destXform->m_Affine.A(0); destXform->m_Affine.B(1);//This will be swapping x and y, seems strange, but it's what the original did. destXform->m_Affine.C(0); destXform->m_Affine.D(1); destXform->m_Affine.E(0); destXform->m_Affine.F(0); } } //If there still are no matches, switch back to linear. if (found == 0) { destXform->AddVariation(new LinearVariation()); } else if (found > 0) { //Otherwise, normalize the weights. destXform->NormalizeVariationWeights(); } } }//Xforms. }//Embers. } /// /// Thin wrapper around AnyXaosPresent(). /// /// The vector of embers to inspect for xaos /// True if at least one ember contained xaos, else false. static bool AnyXaosPresent(vector>& embers) { return AnyXaosPresent(embers.data(), embers.size()); } /// /// Determine whether at least one ember in the array contained xaos. /// /// The array of embers to inspect /// The size of the embers array /// True if at least one ember contained xaos, else false. static bool AnyXaosPresent(Ember* embers, size_t size) { for (size_t i = 0; i < size; i++) if (embers[i].XaosPresent()) return true; return false; } /// /// Thin wrapper around MaxXformCount(). /// /// The vector of embers to inspect for the greatest xform count /// The greatest non-final xform count in any of the embers static size_t MaxXformCount(vector>& embers) { return MaxXformCount(embers.data(), embers.size()); } /// /// Find the maximum number of non-final xforms present in the array of embers. /// /// The array of embers to inspect /// The size of the embers array /// The greatest non-final xform count in any of the embers static size_t MaxXformCount(Ember* embers, size_t size) { size_t i, maxCount = 0; for (i = 0; i < size; i++) if (embers[i].XformCount() > maxCount) maxCount = embers[i].XformCount(); return maxCount; } /// /// Thin wrapper around AnyFinalPresent(). /// /// The vector of embers to inspect the presence of a final xform /// True if any contained a non-empty final xform, else false. static bool AnyFinalPresent(vector>& embers) { return AnyFinalPresent(embers.data(), embers.size()); } /// /// Determine whether at least one ember in the array contained a non-empty final xform. /// /// The array of embers to inspect the presence of a final xform /// The size of the embers array /// True if any contained a final xform, else false. static bool AnyFinalPresent(Ember* embers, size_t size) { for (size_t i = 0; i < size; i++) if (embers[i].UseFinalXform()) return true; return false; } /// /// Thin wrapper around Interpolate(). /// /// The vector of embers to interpolate /// The time position in the vector specifying the point of interpolation /// Stagger if > 0 /// The interpolated result static void Interpolate(vector>& embers, T time, T stagger, Ember& result) { Interpolate(embers.data(), embers.size(), time, stagger, result); } /// /// Interpolates the array of embers at a specified time and stores the result. /// /// The embers array /// The size of the embers array /// The time position in the vector specifying the point of interpolation /// Stagger if > 0 /// The interpolated result static void Interpolate(Ember* embers, size_t size, T time, T stagger, Ember& result) { if (size == 1) { result = embers[0];//Deep copy. return; } size_t i1, i2; vector c(2); Ember localEmbers[4]; bool smoothFlag = false; if (embers[0].m_Time >= time) { i1 = 0; i2 = 1; } else if (embers[size - 1].m_Time <= time) { i1 = size - 2; i2 = size - 1; } else { i1 = 0; while (embers[i1].m_Time < time) i1++; i1--; i2 = i1 + 1; } c[0] = (embers[i2].m_Time - time) / (embers[i2].m_Time - embers[i1].m_Time); c[1] = 1 - c[0]; //To interpolate the xforms, make copies of the source embers //and ensure that they both have the same number of xforms before progressing. if (embers[i1].m_Interp == eInterp::EMBER_INTERP_LINEAR) { Align(&embers[i1], &localEmbers[0], 2); smoothFlag = false; } else { if (i1 == 0) { //fprintf(stderr, "error: cannot use smooth interpolation on first segment.\n"); //fprintf(stderr, "reverting to linear interpolation.\n"); Align(&embers[i1], &localEmbers[0], 2); smoothFlag = false; } if (i2 == size - 1) { //fprintf(stderr, "error: cannot use smooth interpolation on last segment.\n"); //fprintf(stderr, "reverting to linear interpolation.\n"); Align(&embers[i1], &localEmbers[0], 2); smoothFlag = false; } Align(&embers[i1 - 1], &localEmbers[0], 4);//Should really be doing some sort of checking here to ensure the ember vectors have 4 elements. smoothFlag = true; } result.m_Time = time; result.m_Interp = eInterp::EMBER_INTERP_LINEAR; result.m_AffineInterp = embers[0].m_AffineInterp; result.m_PaletteInterp = ePaletteInterp::INTERP_HSV; if (!smoothFlag) result.Interpolate(&localEmbers[0], 2, c, stagger); else result.InterpolateCatmullRom(&localEmbers[0], 4, c[1]); } /// /// Merge the variations in a vector of xforms into a single xform so that /// it contains one variation for each variation type that was present in the /// vector of xforms. /// /// The xforms to merge /// Clear weights if true, else copy weights /// The xform whose variations are a result of the merge static Xform MergeXforms(vector*>& xforms, bool clearWeights = false) { Xform xform; for (auto xf : xforms) MergeXformVariations1Way(xf, &xform, false, clearWeights); return xform; } /// /// Merges the xform variations from one xform to another, but not back. /// /// The source xform to merge from /// The destination xform to merge to /// If true, only merge parametric variations, else merge all /// If true, set variation weights in dest to 0, else copy weights static void MergeXformVariations1Way(Xform* source, Xform* dest, bool parVarsOnly, bool clearWeights) { for (size_t i = 0; i < source->TotalVariationCount(); i++)//Iterate through the first xform's variations. { Variation* var = source->GetVariation(i);//Grab the variation at index in in the first xform. Variation* var2 = dest->GetVariationById(var->VariationId());//See if the same variation exists in the second xform. ParametricVariation* parVar = dynamic_cast*>(var);//Parametric cast of the first var for later. if (!var2)//Only take action if the second xform did not contain this variation. { if (parVarsOnly)//Only add if parametric. { if (parVar) { Variation* parVarCopy = parVar->Copy(); if (clearWeights) parVarCopy->m_Weight = 0; dest->AddVariation(parVarCopy); } } else//Add regardless of type. { Variation* varCopy = var->Copy(); if (clearWeights) varCopy->m_Weight = 0; dest->AddVariation(varCopy); } } } } /// /// Merges the xform variations from one xform to another, and back. /// After this function completes, both xforms will have the same variations. /// /// The source xform to merge from, and to /// The destination xform to merge to, and from /// If true, only merge parametric variations, else merge all /// If true, set variation weights in dest to 0, else copy weights static void MergeXformVariations2Way(Xform* source, Xform* dest, bool parVarsOnly, bool clearWeights) { MergeXformVariations1Way(source, dest, parVarsOnly, clearWeights); MergeXformVariations1Way(dest, source, parVarsOnly, clearWeights); } /// /// Interpolate a vector of parametric variations by a vector of coefficients and store the ouput in a new parametric variation. /// Elements in first which are not the same variation type as second will be ignored. /// /// The vector of parametric variations to interpolate /// The parametric variation to store the output. This must be initialized first to the desired type. /// The vector of coefficients used to interpolate static void InterpParametricVar(vector*>& first, ParametricVariation* second, vector& c) { //First, make sure the variation vector is the same size as the coefficient vector. if (second && first.size() == c.size()) { second->Clear(); auto secondParams = second->Params(); //Iterate through each of the source variations. for (size_t i = 0; i < first.size(); i++) { auto firstVar = first[i]; //Make sure the source variation at this index is the same type as the variation being written to. if (firstVar->VariationId() == second->VariationId()) { size_t size = firstVar->ParamCount(); auto firstParams = firstVar->Params(); //Multiply each parameter of the variation at this index by the coefficient at this index, and add //the result to the corresponding parameter in second. for (size_t j = 0; j < size; j++) { if (!firstParams[j].IsPrecalc()) *(secondParams[j].Param()) += c[i] * firstParams[j].ParamVal(); } } } second->Precalc(); } } /// /// Thin wrapper around ConvertLinearToPolar(). /// /// The vector of embers whose affine transforms will be copied and converted /// The xform index in each ember to convert /// If 0 convert pre affine, else post affine. /// The vec2 vector to store the polar angular values /// The vec2 vector to store the polar magnitude values /// The vec2 vector to store the polar translation values static void ConvertLinearToPolar(vector>& embers, size_t xfi, size_t cflag, vector& cxAng, vector& cxMag, vector& cxTrn) { ConvertLinearToPolar(embers.data(), embers.size(), xfi, cflag, cxAng, cxMag, cxTrn); } /// /// Convert pre or post affine coordinates of the xform at a specific index in each ember from linear to polar and store as separate /// vec2 components in the vector parameters cxAng, cxMag and cxTrn. /// /// The array of embers whose affine transforms will be copied and converted /// The size of the embers array /// The xform index in each ember to convert /// If 0 convert pre affine, else post affine. /// The vec2 vector to store the polar angular values /// The vec2 vector to store the polar magnitude values /// The vec2 vector to store the polar translation values static void ConvertLinearToPolar(Ember* embers, size_t size, size_t xfi, size_t cflag, vector& cxAng, vector& cxMag, vector& cxTrn) { if (size == cxAng.size() && size == cxMag.size() && size == cxTrn.size()) { T c1[2], d, t, refang; glm::length_t col, k; int zlm[2]; const char* loc = __FUNCTION__; for (k = 0; k < size; k++) { //Establish the angles and magnitudes for each component. //Keep translation linear. zlm[0] = zlm[1] = 0; if (Xform* xform = embers[k].GetTotalXform(xfi)) { for (col = 0; col < 2; col++) { if (cflag == 0) { c1[0] = xform->m_Affine.m_Mat[0][col];//a or b. c1[1] = xform->m_Affine.m_Mat[1][col];//d or e. t = xform->m_Affine.m_Mat[col][2];//c or f. } else { c1[0] = xform->m_Post.m_Mat[0][col]; c1[1] = xform->m_Post.m_Mat[1][col]; t = xform->m_Post.m_Mat[col][2]; } cxAng[k][col] = atan2(c1[1], c1[0]); cxMag[k][col] = std::sqrt(c1[0] * c1[0] + c1[1] * c1[1]); if (cxMag[k][col] == 0) zlm[col] = 1; cxTrn[k][col] = t; } if (zlm[0] == 1 && zlm[1] == 0) cxAng[k][0] = cxAng[k][1]; else if (zlm[0] == 0 && zlm[1] == 1) cxAng[k][1] = cxAng[k][0]; } else { cout << loc << ": xform " << xfi << " is missing when it was expected, something is severely wrong." << endl; } } //Make sure the rotation is the shorter direction around the circle //by adjusting each angle in succession, and rotate clockwise if 180 degrees. for (col = 0; col < 2; col++) { for (k = 1; k < size; k++) { if (Xform* xform = embers[k].GetTotalXform(xfi)) { //Adjust angles differently if an asymmetric case. if (xform->m_Wind[col] > 0 && cflag == 0) { //Adjust the angles to make sure that it's within wind : wind + 2pi. refang = xform->m_Wind[col] - M_2PI; //Make sure both angles are within [refang refang + 2 * pi]. while (cxAng[k - 1][col] < refang) cxAng[k - 1][col] += M_2PI; while (cxAng[k - 1][col] > refang + M_2PI) cxAng[k - 1][col] -= M_2PI; while (cxAng[k][col] < refang) cxAng[k][col] += M_2PI; while (cxAng[k][col] > refang + M_2PI) cxAng[k][col] -= M_2PI; } else { //Normal way of adjusting angles. d = cxAng[k][col] - cxAng[k - 1][col]; //Adjust to avoid the -pi/pi discontinuity. if (d > M_PI + EPS) cxAng[k][col] -= M_2PI; else if (d < -(M_PI - EPS))//Forces clockwise rotation at 180. cxAng[k][col] += M_2PI; } } else { cout << loc << ": xform " << xfi << " is missing when it was expected, something is severely wrong." << endl; } } } } } /// /// Never really understood what this did, but it has to do with winding. /// /// The array of embers /// The size of the embers array static void AsymmetricRefAngles(Ember* embers, size_t count) { size_t k, xfi; T cxang[4][2], c1[2], d; for (xfi = 0; xfi < embers[0].XformCount(); xfi++)//Final xforms don't rotate regardless of their symmetry. { for (k = 0; k < count; k++) { //Establish the angle for each component. //Should potentially functionalize. for (glm::length_t col = 0; col < 2; col++) { c1[0] = embers[k].GetXform(xfi)->m_Affine.m_Mat[0][col];//A,D then B,E. c1[1] = embers[k].GetXform(xfi)->m_Affine.m_Mat[1][col]; cxang[k][col] = atan2(c1[1], c1[0]); } } for (k = 1; k < count; k++) { for (size_t col = 0; col < 2; col++) { int sym0, sym1; int padSymFlag; d = cxang[k][col] - cxang[k - 1][col]; //Adjust to avoid the -pi/pi discontinuity. if (d > T(M_PI + EPS)) cxang[k][col] -= 2 * T(M_PI); else if (d < -T(M_PI - EPS) ) cxang[k][col] += 2 * T(M_PI); //If this is an asymmetric case, store the NON-symmetric angle //Check them pairwise and store the reference angle in the second //to avoid overwriting if asymmetric on both sides. padSymFlag = 0; sym0 = (embers[k - 1].GetXform(xfi)->m_Animate == 0 || (embers[k - 1].GetXform(xfi)->Empty() && padSymFlag)); sym1 = (embers[k ].GetXform(xfi)->m_Animate == 0 || (embers[k ].GetXform(xfi)->Empty() && padSymFlag)); if (sym1 && !sym0) embers[k].GetXform(xfi)->m_Wind[col] = cxang[k - 1][col] + 2 * T(M_PI); else if (sym0 && !sym1) embers[k].GetXform(xfi)->m_Wind[col] = cxang[k][col] + 2 * T(M_PI); } } } } /// /// Never really understood what this did. /// /// The coefficients vector /// The vec2 vector to store the polar angular values /// The vec2 vector to store the polar magnitude values /// The vec2 vector to store the polar translation values /// The Affine2D to store the inerpolated values in static void InterpAndConvertBack(vector& coefs, vector& cxAng, vector& cxMag, vector& cxTrn, Affine2D& store) { size_t size = coefs.size(); glm::length_t i, col, accmode[2] = { 0, 0 }; T expmag, accang[2] = { 0, 0 }, accmag[2] = { 0, 0 }; //Accumulation mode defaults to logarithmic, but in special //cases switch to linear accumulation. for (col = 0; col < 2; col++) { for (i = 0; i < size; i++) { if (log(cxMag[i][col]) < -10) accmode[col] = 1;//Mode set to linear interp. } } for (i = 0; i < size; i++) { for (col = 0; col < 2; col++) { accang[col] += coefs[i] * cxAng[i][col]; if (accmode[col] == 0) accmag[col] += coefs[i] * std::log(cxMag[i][col]); else accmag[col] += coefs[i] * (cxMag[i][col]); //Translation is ready to go. store.m_Mat[col][2] += coefs[i] * cxTrn[i][col]; } } //Convert the angle back to rectangular. for (col = 0; col < 2; col++) { if (accmode[col] == 0) expmag = std::exp(accmag[col]); else expmag = accmag[col]; store.m_Mat[0][col] = expmag * std::cos(accang[col]); store.m_Mat[1][col] = expmag * std::sin(accang[col]); } } /// /// Smooths the time values for animations. /// /// The time value to smooth /// the smoothed time value static inline T Smoother(T t) { return 3 * t * t - 2 * t * t * t; } /// /// Gets the stagger coef based on the position of the current xform among the others. /// Never really understood what this did. /// /// The time value /// The stagger percentage /// The number xforms in the ember /// The index of this xform within the ember /// The stagger coefficient static inline T GetStaggerCoef(T t, T staggerPercent, size_t numXforms, size_t thisXform) { //maxStag is the spacing between xform start times if staggerPercent = 1.0. T maxStag = T(numXforms - 1) / numXforms; //Scale the spacing by staggerPercent. T stagScaled = staggerPercent * maxStag; //t ranges from 1 to 0 (the contribution of cp[0] to the blend). //The first line below makes the first xform interpolate first. //The second line makes the last xform interpolate first. T st = stagScaled * (numXforms - 1 - thisXform) / (numXforms - 1); T et = st + (1 - stagScaled); if (t <= st) return 0; else if (t >= et) return 1; else return Smoother((t - st) / (1 - stagScaled)); } /// /// Apply the specified motion function to a value. /// /// The function type to apply, sin, triangle, hill or saw. /// The time value to apply the motion function to /// The new time value computed by applying the specified motion function to the time value static T MotionFuncs(eMotion funcNum, T timeVal) { //Motion funcs should be cyclic, and equal to 0 at integral time values //abs peak values should be not be greater than 1. switch (funcNum) { case EmberNs::eMotion::MOTION_SIN: { return std::sin(T(2.0) * T(M_PI) * timeVal); } break; case EmberNs::eMotion::MOTION_TRIANGLE: { T fr = fmod(timeVal, T(1.0)); if (fr < 0) fr += 1; if (fr <= T(0.25)) fr *= 4; else if (fr <= T(0.75)) fr = -4 * fr + 2; else fr = 4 * fr - 4; return fr; } break; case EmberNs::eMotion::MOTION_HILL: { return ((1 - std::cos(T(2.0) * T(M_PI) * timeVal)) * T(0.5)); } break; case EmberNs::eMotion::MOTION_SAW: { return (T(2.0) * fmod(timeVal - T(0.5), T(1.0)) - T(1.0)); } break; default: return timeVal; break; } } /* //Will need to alter this to handle 2D palettes. static bool InterpMissingColors(vector>& palette) { //Check for a non-full palette. int minIndex, maxIndex; int colorli, colorri; int wrapMin, wrapMax; int intl, intr; int str, enr; int i, j, k; double prcr; if (palette.size() != 256) return false; for (i = 0; i < 256; i++) { if (palette[i].m_Index >= 0) { minIndex = i; break; } } if (i == 256) { //No colors. Set all indices properly. for (i = 0; i < 256; i++) palette[i].m_Index = (T)i; return false; } wrapMin = minIndex + 256; for (i = 255; i >= 0; i--)//Moving backwards, ouch! { if (palette[i].m_Index >= 0) { maxIndex = i; break; } } wrapMax = maxIndex - 256; //Loop over the indices looking for negs, i = 0; while (i < 256) { if (palette[i].m_Index < 0) { //Start of a range of negs. str = i; intl = i - 1; colorli = intl; while (palette[i].m_Index < 0 && i < 256) { enr = i; intr = i + 1; colorri = intr; i++; } if (intl == -1) { intl = wrapMax; colorli = maxIndex; } if (intr == 256) { intr = wrapMin; colorri = minIndex; } for (j = str; j <= enr; j++) { prcr = (j - intl) / T(intr - intl); for (k = 0; k <= 3; k++) palette[j].Channels[k] = T(palette[colorli].Channels[k] * (1 - prcr) + palette[colorri].Channels[k] * prcr); palette[j].m_Index = T(j); } i = colorri + 1; } else i++; } return true; } */ /// /// Compare xforms for sorting based first on color speed and second on determinants if /// color speeds are equal. /// /// The first xform to compare /// The second xform to compare /// true if a > b, else false. static inline bool CompareXforms(const Xform& a, const Xform& b) { if (a.m_ColorSpeed > b.m_ColorSpeed) return true; if (a.m_ColorSpeed < b.m_ColorSpeed) return false; //Original did this every time, even though it's only needed if the color speeds are equal. m2T aMat2 = a.m_Affine.ToMat2ColMajor(); m2T bMat2 = b.m_Affine.ToMat2ColMajor(); T ad = glm::determinant(aMat2); T bd = glm::determinant(bMat2); if (a.m_ColorSpeed > 0) { if (ad < 0) return false; if (bd < 0) return true; ad = atan2(a.m_Affine.A(), a.m_Affine.D()); bd = atan2(b.m_Affine.A(), b.m_Affine.D()); } return ad > bd; } }; }