diff --git a/.gitignore b/.gitignore
index 4eec902..78d4f18 100644
--- a/.gitignore
+++ b/.gitignore
@@ -262,3 +262,4 @@ Builds/include/GL
/Builds/QtCreator/Ember/.qtc_clangd/compile_commands.json
/Builds/QtCreator/Fractorium/.qtc_clangd/compile_commands.json
.DS_Store
+/.vs/VSWorkspaceState.json
diff --git a/Builds/MSVC/Installer/Product.wxs b/Builds/MSVC/Installer/Product.wxs
index e3892de..b48581d 100644
--- a/Builds/MSVC/Installer/Product.wxs
+++ b/Builds/MSVC/Installer/Product.wxs
@@ -124,19 +124,19 @@
-
+
-
+
-
+
-
+
-
+
diff --git a/Source/Ember/Ember.h b/Source/Ember/Ember.h
index 7c7c1ad..d72793e 100644
--- a/Source/Ember/Ember.h
+++ b/Source/Ember/Ember.h
@@ -163,6 +163,14 @@ public:
m_CurveDE = static_cast(ember.m_CurveDE);
m_SpatialFilterType = ember.m_SpatialFilterType;
m_SpatialFilterRadius = static_cast(ember.m_SpatialFilterRadius);
+ m_Stagger = ember.m_Stagger;
+ m_Rotations = ember.m_Rotations;
+ m_SecondsPerRotation = ember.m_SecondsPerRotation;
+ m_RotateXformsCw = ember.m_RotateXformsCw;
+ m_BlendSeconds = ember.m_BlendSeconds;
+ m_RotationsPerBlend = ember.m_RotationsPerBlend;
+ m_BlendRotateXformsCw = ember.m_BlendRotateXformsCw;
+ m_Linear = ember.m_Linear;
m_TemporalFilterType = ember.m_TemporalFilterType;
m_TemporalFilterExp = static_cast(ember.m_TemporalFilterExp);
m_TemporalFilterWidth = static_cast(ember.m_TemporalFilterWidth);
@@ -171,6 +179,7 @@ public:
m_Name = ember.m_Name;
m_ParentFilename = ember.m_ParentFilename;
m_Index = ember.m_Index;
+ //m_Animations = ember.m_Animations;
m_ScaleType = ember.ScaleType();
m_Palette = ember.m_Palette;
m_Curves = ember.m_Curves;
@@ -193,6 +202,7 @@ public:
{
m_FinalXform.m_Motion.clear();
m_FinalXform.m_Animate = 0;
+ m_FinalXform.m_AnimateOrigin = 0;
m_FinalXform.m_ColorSpeed = 0;
}
@@ -303,6 +313,7 @@ public:
ember.m_FinalXform.m_Affine.MakeID();
ember.m_FinalXform.m_Post.MakeID();
ember.m_FinalXform.m_Animate = 0;
+ ember.m_FinalXform.m_AnimateOrigin = 0;
ember.m_FinalXform.m_ColorSpeed = 0;
ember.m_FinalXform.m_Motion.clear();
ember.m_FinalXform.ClearAndDeleteVariations();
@@ -925,11 +936,12 @@ public:
}
}
- InterpXform<&Xform::m_Weight> (thisXform, i, embers, coefs, size);
- InterpXform<&Xform::m_ColorX> (thisXform, i, embers, coefs, size);
- InterpXform<&Xform::m_ColorSpeed>(thisXform, i, embers, coefs, size);
- InterpXform<&Xform::m_Opacity> (thisXform, i, embers, coefs, size);
- InterpXform<&Xform::m_Animate> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_Weight> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_ColorX> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_ColorSpeed> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_Opacity> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_Animate> (thisXform, i, embers, coefs, size);
+ InterpXform<&Xform::m_AnimateOrigin> (thisXform, i, embers, coefs, size);
ClampGte0Ref(thisXform->m_Weight);
ClampRef(thisXform->m_ColorX, 0, 1);
ClampRef(thisXform->m_ColorSpeed, -1, 1);
@@ -1077,15 +1089,17 @@ public:
while (auto xform = GetTotalXform(i++))//Flam3 only allowed animation with normal xforms. This has been changed to allow animations of final xforms.
{
- //Don't rotate xforms with animate set to 0.
- if (xform->m_Animate == 0)
- continue;
-
//Assume that if there are no variations, then it's a padding xform.
if (xform->Empty() && m_AffineInterp != eAffineInterp::AFFINE_INTERP_LOG)
continue;
- xform->m_Affine.Rotate(angle * DEG_2_RAD_T);
+ //Don't rotate xforms with animate set to 0.
+ if (xform->m_Animate != 0)
+ xform->m_Affine.Rotate(angle * DEG_2_RAD_T);
+
+ if (xform->m_AnimateOrigin != 0)
+ xform->m_Affine.RotateTrans(angle * DEG_2_RAD_T);
+
//Don't rotate post.
}
}
@@ -1139,6 +1153,7 @@ public:
m_Xforms[i].m_Weight = 1;
m_Xforms[i].m_ColorSpeed = 0;
m_Xforms[i].m_Animate = 0;
+ m_Xforms[i].m_AnimateOrigin = 0;
m_Xforms[i].m_ColorX = 1;
m_Xforms[i].m_ColorY = 1;//Added in case 2D palette support is ever added.
m_Xforms[i].m_Affine.A(-1);
@@ -1162,6 +1177,7 @@ public:
m_Xforms[i].m_Weight = 1;
m_Xforms[i].m_ColorSpeed = 0;
m_Xforms[i].m_Animate = 0;
+ m_Xforms[i].m_AnimateOrigin = 0;
m_Xforms[i].m_ColorX = m_Xforms[i].m_ColorY = (sym < 3) ? 0 : (static_cast(k - 1) / static_cast(sym - 2));//Added Y.
m_Xforms[i].m_Affine.A(Round6(std::cos(k * a)));
m_Xforms[i].m_Affine.D(Round6(std::sin(k * a)));
@@ -1409,6 +1425,10 @@ public:
APP_FMP(m_BlurCurve);
break;
+ case eEmberMotionParam::FLAME_MOTION_SCALE:
+ APP_FMP(m_Zoom);
+ break;
+
case eEmberMotionParam::FLAME_MOTION_NONE:
default:
break;
@@ -1445,8 +1465,8 @@ public:
m_HighlightPower = 1;
m_K2 = 0;
m_Background.Reset();
- m_FinalRasW = 100;
- m_FinalRasH = 100;
+ m_FinalRasW = 1920;
+ m_FinalRasH = 1080;
m_Supersample = 1;
m_SpatialFilterRadius = static_cast(0.5);
m_Zoom = 0;
@@ -1470,11 +1490,19 @@ public:
m_TemporalSamples = 100;
m_SpatialFilterType = eSpatialFilterType::GAUSSIAN_SPATIAL_FILTER;
m_AffineInterp = eAffineInterp::AFFINE_INTERP_LOG;
+ m_PaletteMode = ePaletteMode::PALETTE_LINEAR;
+ m_Interp = eInterp::EMBER_INTERP_SMOOTH;
+ m_Stagger = 0;
+ m_Rotations = 1;
+ m_SecondsPerRotation = 3;
+ m_RotateXformsCw = false;
+ m_BlendSeconds = 3;
+ m_RotationsPerBlend = 0;
+ m_BlendRotateXformsCw = false;
+ m_Linear = false;
m_TemporalFilterType = eTemporalFilterType::BOX_TEMPORAL_FILTER;
m_TemporalFilterWidth = 1;
m_TemporalFilterExp = 1;
- m_PaletteMode = ePaletteMode::PALETTE_LINEAR;
- m_Interp = eInterp::EMBER_INTERP_SMOOTH;
}
else
{
@@ -1507,11 +1535,19 @@ public:
m_TemporalSamples = 0;
m_SpatialFilterType = eSpatialFilterType::GAUSSIAN_SPATIAL_FILTER;
m_AffineInterp = eAffineInterp::AFFINE_INTERP_LOG;
+ m_PaletteMode = ePaletteMode::PALETTE_STEP;
+ m_Interp = eInterp::EMBER_INTERP_LINEAR;
+ m_Stagger = 999999;
+ m_Rotations = 999999;
+ m_SecondsPerRotation = 999999;
+ m_RotateXformsCw = false;
+ m_BlendSeconds = 999999;
+ m_RotationsPerBlend = 999999;
+ m_BlendRotateXformsCw = false;
+ m_Linear = false;
m_TemporalFilterType = eTemporalFilterType::BOX_TEMPORAL_FILTER;
m_TemporalFilterWidth = -1;
m_TemporalFilterExp = -999;
- m_PaletteMode = ePaletteMode::PALETTE_STEP;
- m_Interp = eInterp::EMBER_INTERP_LINEAR;
}
m_Xforms.clear();
@@ -1578,6 +1614,13 @@ public:
<< "DE Curve: " << m_CurveDE << "\n"
<< "Spatial Filter Type: " << m_SpatialFilterType << "\n"
<< "Spatial Filter Radius: " << m_SpatialFilterRadius << "\n"
+ << "Stagger: " << m_Stagger << "\n"
+ << "Rotations: " << m_Rotations << "\n"
+ << "Seconds Per Rotation: " << m_SecondsPerRotation << "\n"
+ << "Rotate Xforms Clockwise: " << m_RotateXformsCw << "\n"
+ << "Blend Seconds: " << m_BlendSeconds << "\n"
+ << "Rotations Per Blend: " << m_RotationsPerBlend << "\n"
+ << "Blend Rotate Xforms Clockwise: " << m_BlendRotateXformsCw << "\n"
<< "Temporal Filter Type: " << m_TemporalFilterType << "\n"
<< "Temporal Filter Width: " << m_TemporalFilterWidth << "\n"
<< "Temporal Filter Exp: " << m_TemporalFilterExp << "\n"
@@ -1754,20 +1797,6 @@ public:
//Xml field: "palette_interpolation".
ePaletteInterp m_PaletteInterp = ePaletteInterp::INTERP_HSV;
- //Temporal Filter.
-
- //Only used if temporal filter type is exp, else unused.
- //Xml field: "temporal_filter_exp".
- T m_TemporalFilterExp = 1;
-
- //The width of the temporal filter.
- //Xml field: "temporal_filter_width".
- T m_TemporalFilterWidth = 1;
-
- //The type of the temporal filter: Gaussian, Box or Exp.
- //Xml field: "temporal_filter_type".
- eTemporalFilterType m_TemporalFilterType = eTemporalFilterType::BOX_TEMPORAL_FILTER;
-
//Density Estimation Filter.
//The minimum radius of the DE filter.
@@ -1825,6 +1854,30 @@ public:
//The 0-based position of this ember in the file it was contained in.
size_t m_Index = 0;
+ //The parameters for animating.
+ double m_Stagger = 0;
+ double m_Rotations = 1;
+ double m_SecondsPerRotation = 3;
+ bool m_RotateXformsCw = false;
+ double m_BlendSeconds = 3;
+ size_t m_RotationsPerBlend = 0;//Could this ever be double?
+ bool m_BlendRotateXformsCw = false;
+ bool m_Linear = false;
+
+ //Temporal Filter.
+
+ //Only used if temporal filter type is exp, else unused.
+ //Xml field: "temporal_filter_exp".
+ T m_TemporalFilterExp = 1;
+
+ //The width of the temporal filter.
+ //Xml field: "temporal_filter_width".
+ T m_TemporalFilterWidth = 1;
+
+ //The type of the temporal filter: Gaussian, Box or Exp.
+ //Xml field: "temporal_filter_type".
+ eTemporalFilterType m_TemporalFilterType = eTemporalFilterType::BOX_TEMPORAL_FILTER;
+
//The list of motion elements for the top-level flame params
vector> m_EmberMotionElements;
diff --git a/Source/Ember/EmberDefines.h b/Source/Ember/EmberDefines.h
index a30a92e..83ca675 100644
--- a/Source/Ember/EmberDefines.h
+++ b/Source/Ember/EmberDefines.h
@@ -169,7 +169,8 @@ enum class eEmberMotionParam : et//These must remain in this order forever.
FLAME_MOTION_BACKGROUND_G,
FLAME_MOTION_BACKGROUND_B,
FLAME_MOTION_VIBRANCY,
- FLAME_MOTION_BLUR_CURVE
+ FLAME_MOTION_BLUR_CURVE,
+ FLAME_MOTION_SCALE
};
///
diff --git a/Source/Ember/EmberToXml.cpp b/Source/Ember/EmberToXml.cpp
index bba4534..a5a25d6 100644
--- a/Source/Ember/EmberToXml.cpp
+++ b/Source/Ember/EmberToXml.cpp
@@ -158,6 +158,14 @@ string EmberToXml::ToString(Ember& ember, const string& extraAttributes, s
os << " cam_pitch=\"" << ember.m_CamPitch << "\"";
os << " cam_dof=\"" << ember.m_CamDepthBlur << "\"";
os << " blur_curve=\"" << ember.m_BlurCurve << "\"";
+ os << " stagger=\"" << ember.m_Stagger << "\"";
+ os << " rotations=\"" << ember.m_Rotations << "\"";
+ os << " seconds_per_rotation=\"" << ember.m_SecondsPerRotation << "\"";
+ os << " rotate_xforms_cw=\"" << ember.m_RotateXformsCw << "\"";
+ os << " blend_seconds=\"" << ember.m_BlendSeconds << "\"";
+ os << " rotations_per_blend=\"" << ember.m_RotationsPerBlend << "\"";
+ os << " blend_rotate_xforms_cw=\"" << ember.m_BlendRotateXformsCw << "\"";
+ os << " linear_blend=\"" << ember.m_Linear << "\"";
if (ember.m_PaletteMode == ePaletteMode::PALETTE_STEP)
os << " palette_mode=\"step\"";
@@ -544,6 +552,7 @@ string EmberToXml::ToString(Xform& xform, size_t xformCount, bool isFinal,
std::replace(s.begin(), s.end(), ' ', '_');
os << "name=\"" << s << "\" ";//Flam3 didn't do this, but Apo does.
os << "animate=\"" << xform.m_Animate << "\" ";//Flam3 only did this for non-final. Ember supports animating final.
+ os << "animate_origin=\"" << xform.m_AnimateOrigin << "\" ";
}
//Variation writing order differs slightly from the original to make it a bit more readable.
@@ -872,6 +881,10 @@ string EmberToXml::ToString(const EmberMotion& motion)
os << " vibrancy=\"" << motion.m_MotionParams[i].second << "\"";
break;
+ case eEmberMotionParam::FLAME_MOTION_SCALE:
+ os << " scale=\"" << motion.m_MotionParams[i].second << "\"";
+ break;
+
case eEmberMotionParam::FLAME_MOTION_NONE:
default:
break;
diff --git a/Source/Ember/Interpolate.h b/Source/Ember/Interpolate.h
index c6fcea9..10de44d 100644
--- a/Source/Ember/Interpolate.h
+++ b/Source/Ember/Interpolate.h
@@ -832,8 +832,10 @@ public:
//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.
- 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));
+ auto xfk = embers[k].GetXform(xfi);
+ auto xfkm1 = embers[k - 1].GetXform(xfi);
+ sym0 = ((xfkm1->m_Animate == 0 && xfkm1->m_AnimateOrigin == 0) || (xfkm1->Empty() && padSymFlag));
+ sym1 = ((xfk->m_Animate == 0 && xfk->m_AnimateOrigin == 0) || (xfk->Empty() && padSymFlag));
if (sym1 && !sym0)
embers[k].GetXform(xfi)->m_Wind[col] = cxang[k - 1][col] + 2 * T(M_PI);
diff --git a/Source/Ember/SheepTools.h b/Source/Ember/SheepTools.h
index 0d5b7bf..cc7af05 100644
--- a/Source/Ember/SheepTools.h
+++ b/Source/Ember/SheepTools.h
@@ -656,6 +656,7 @@ public:
xform->m_ColorY = m_Rand.Frand01();//Will need to update this if 2D coordinates are ever supported.
xform->m_ColorSpeed = T(0.5);
xform->m_Animate = 1;
+ xform->m_AnimateOrigin = T(m_Rand.RandBit());
xform->m_Affine.A(m_Rand.Frand11());
xform->m_Affine.B(m_Rand.Frand11());
xform->m_Affine.C(m_Rand.Frand11());
@@ -865,8 +866,8 @@ public:
{
auto& p = m_FinalImage[i];
m_Hist[static_cast((p.r * res) +
- (p.g * res) * res +
- (p.b * res) * res * res)]++;//A specific histogram index representing the sum of R,G,B values.
+ (p.g * res) * res +
+ (p.b * res) * res * res)]++;//A specific histogram index representing the sum of R,G,B values.
}
for (i = 0; i < res3; i++)
diff --git a/Source/Ember/Xform.h b/Source/Ember/Xform.h
index 4c9151c..f0e5daf 100644
--- a/Source/Ember/Xform.h
+++ b/Source/Ember/Xform.h
@@ -154,6 +154,7 @@ public:
m_DirectColor = static_cast(xform.m_DirectColor);
m_ColorSpeed = static_cast(xform.m_ColorSpeed);
m_Animate = static_cast(xform.m_Animate);
+ m_AnimateOrigin = static_cast(xform.m_AnimateOrigin);
m_Opacity = static_cast(xform.m_Opacity);
CacheColorVals();
m_HasPre = xform.HasPre();
@@ -209,6 +210,7 @@ public:
m_Weight = 0;
m_ColorSpeed = static_cast(0.5);
m_Animate = 1;
+ m_AnimateOrigin = 0;
m_ColorX = static_cast(count & 1);
m_ColorY = 0;
m_DirectColor = 1;
@@ -235,6 +237,7 @@ public:
m_Weight = EMPTYFIELD;
m_ColorSpeed = EMPTYFIELD;
m_Animate = EMPTYFIELD;
+ m_AnimateOrigin = EMPTYFIELD;
m_ColorX = EMPTYFIELD;
m_ColorY = EMPTYFIELD;
m_DirectColor = EMPTYFIELD;
@@ -547,6 +550,7 @@ public:
m_OneMinusColorCache = 0;
m_Opacity = 1;
m_Animate = 0;
+ m_AnimateOrigin = 0;
m_Wind[0] = 0;
m_Wind[1] = 0;
m_Name = "";
@@ -800,6 +804,7 @@ public:
APPMOT(m_Opacity);
APPMOT(m_ColorSpeed);
APPMOT(m_Animate);
+ APPMOT(m_AnimateOrigin);
for (size_t j = 0; j < currentMot.TotalVariationCount(); j++)//For each variation in the motion xform.
{
@@ -811,8 +816,11 @@ public:
{
Variation* newVar = motVar->Copy();
newVar->m_Weight = motVar->m_Weight * Interpolater::MotionFuncs(func, freq * (blend + cleanOffset));
- AddVariation(newVar);
- var = newVar;//Use this below for params.
+
+ if (AddVariation(newVar))
+ var = newVar;//Use this below for params.
+ else
+ delete newVar;
}
else//It was present, so apply the motion func to the weight.
{
@@ -821,7 +829,7 @@ public:
//At this point, we've added if needed, or just applied the motion func to the weight.
//Now apply the motion func to the params if needed.
- if (motParVar)
+ if (var && motParVar)
{
auto parVar = dynamic_cast*>(var);
auto params = parVar->Params();
@@ -1213,7 +1221,8 @@ public:
ss << "\nColorY: " << m_ColorY;
ss << "\nDirect Color: " << m_DirectColor;
ss << "\nColor Speed: " << m_ColorSpeed;
- ss << "\nAnimate: " << m_Animate;
+ ss << "\nAnimate Local Rotation: " << m_Animate;
+ ss << "\nAnimate Origin Rotation: " << m_AnimateOrigin;
ss << "\nOpacity: " << m_Opacity;
ss << "\nWind: " << m_Wind[0] << ", " << m_Wind[1];
ss << "\nMotion Frequency: " << m_MotionFreq;
@@ -1290,7 +1299,8 @@ public:
//C = (C + Ci) * m_ColorSpeed.
T m_ColorSpeed;
T m_Opacity;//How much of this xform is seen. Range: 0.0 (invisible) - 1.0 (totally visible).
- T m_Animate;//Whether or not this xform rotates during animation. 0 means stationary, > 0 means rotate. Use T instead of bool so it can be interpolated.
+ T m_Animate;//Whether or not this xform rotates around its center during animation. 0 means stationary, > 0 means rotate. Use T instead of bool so it can be interpolated.
+ T m_AnimateOrigin;//Same, but rotate around the global origin.
T m_Wind[2];
eMotion m_MotionFunc;
T m_MotionFreq;
diff --git a/Source/Ember/XmlToEmber.cpp b/Source/Ember/XmlToEmber.cpp
index 2a78868..6425713 100644
--- a/Source/Ember/XmlToEmber.cpp
+++ b/Source/Ember/XmlToEmber.cpp
@@ -1389,6 +1389,7 @@ bool XmlToEmber::ParseEmberElementFromChaos(xmlNode* emberNode, Ember& cur
{
finalXform.m_Weight = 0;
finalXform.m_Animate = 0;//Do not animate final by default.
+ finalXform.m_AnimateOrigin = 0;
finalXform.m_ColorX = finalXform.m_ColorY = 0;//Chaotica does not support any kind of coloring for final xforms, opacity remains 1 though.
finalXform.m_ColorSpeed = 0;
currentEmber.SetFinalXform(finalXform);
@@ -1615,6 +1616,14 @@ bool XmlToEmber::ParseEmberElement(xmlNode* emberNode, Ember& currentEmber
else if (ParseAndAssign(curAtt->name, attStr, "rand_range", currentEmber.m_RandPointRange, ret)) {}
else if (ParseAndAssign(curAtt->name, attStr, "soloxform", soloXform, ret)) {}
else if (ParseAndAssign(curAtt->name, attStr, "new_linear", newLinear, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "stagger", currentEmber.m_Stagger, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "rotations", currentEmber.m_Rotations, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "seconds_per_rotation", currentEmber.m_SecondsPerRotation, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "rotate_xforms_cw", currentEmber.m_RotateXformsCw, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "blend_seconds", currentEmber.m_BlendSeconds, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "rotations_per_blend", currentEmber.m_RotationsPerBlend, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "blend_rotate_xforms_cw", currentEmber.m_BlendRotateXformsCw, ret)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "linear_blend", currentEmber.m_Linear, ret)) {}
//Parse more complicated reads that have multiple possible values.
else if (!Compare(curAtt->name, "interpolation"))
{
@@ -1818,6 +1827,9 @@ bool XmlToEmber::ParseEmberElement(xmlNode* emberNode, Ember& currentEmber
currentEmber.m_Curves.m_Points[3] = vals;
}
+ else if (!Compare(curAtt->name, "animations"))
+ {
+ }
xmlFree(attStr);
}
@@ -2023,6 +2035,7 @@ bool XmlToEmber::ParseEmberElement(xmlNode* emberNode, Ember& currentEmber
{
Xform finalXform;
finalXform.m_Animate = 0;//Do not animate final by default.
+ finalXform.m_AnimateOrigin = 0;
if (!ParseXform(childNode, finalXform, false, fromEmber))
{
@@ -2148,6 +2161,8 @@ bool XmlToEmber::ParseEmberElement(xmlNode* emberNode, Ember& currentEmber
ret = ret && AttToEmberMotionFloat(att, attStr, eEmberMotionParam::FLAME_MOTION_RAND_RANGE, motion);
else if (!Compare(curAtt->name, "vibrancy"))
ret = ret && AttToEmberMotionFloat(att, attStr, eEmberMotionParam::FLAME_MOTION_VIBRANCY, motion);
+ else if (!Compare(curAtt->name, "scale"))
+ ret = ret && AttToEmberMotionFloat(att, attStr, eEmberMotionParam::FLAME_MOTION_SCALE, motion);
else if (!Compare(curAtt->name, "background"))
{
double r = 0, g = 0, b = 0;
@@ -2258,6 +2273,7 @@ bool XmlToEmber::ParseXform(xmlNode* childNode, Xform& xform, bool motion,
if (ParseAndAssign(curAtt->name, attStr, "weight", xform.m_Weight, success)) {}
else if (ParseAndAssign(curAtt->name, attStr, "color_speed", xform.m_ColorSpeed, success)) {}
else if (ParseAndAssign(curAtt->name, attStr, "animate", xform.m_Animate, success)) {}
+ else if (ParseAndAssign(curAtt->name, attStr, "animate_origin", xform.m_AnimateOrigin, success)) {}
else if (ParseAndAssign(curAtt->name, attStr, "opacity", xform.m_Opacity, success)) {}
else if (ParseAndAssign(curAtt->name, attStr, "var_color", xform.m_DirectColor, success)) {}
else if (ParseAndAssign(curAtt->name, attStr, "motion_frequency", xform.m_MotionFreq, success)) {}
@@ -2275,6 +2291,7 @@ bool XmlToEmber::ParseXform(xmlNode* childNode, Xform& xform, bool motion,
Aton(attStr, temp);
xform.m_ColorSpeed = (1 - temp) / 2;
xform.m_Animate = T(temp > 0 ? 0 : 1);
+ //Ignore m_AnimateOrigin because it's new and symmetry is a legacy setting.
}
else if (!Compare(curAtt->name, "motion_function"))
{
diff --git a/Source/EmberGenome/EmberGenome.cpp b/Source/EmberGenome/EmberGenome.cpp
index c10d065..441d447 100644
--- a/Source/EmberGenome/EmberGenome.cpp
+++ b/Source/EmberGenome/EmberGenome.cpp
@@ -455,6 +455,8 @@ bool EmberGenome(int argc, _TCHAR* argv[], EmberOptions& opt)
return true;
}
+ //Note that in Fractorium, the sequence is created from the params stored in the flame file.
+ //Here, it's done using the values passed on the command line, which will override the values in the file.
if (opt.Sequence() != "")
{
Ember result;
diff --git a/Source/Fractorium/FinalRenderDialog.cpp b/Source/Fractorium/FinalRenderDialog.cpp
index 6948fc2..1175e4f 100644
--- a/Source/Fractorium/FinalRenderDialog.cpp
+++ b/Source/Fractorium/FinalRenderDialog.cpp
@@ -932,7 +932,8 @@ void FractoriumFinalRenderDialog::showEvent(QShowEvent* e)
if (CreateControllerFromGUI(true))//Create controller if it does not exist, or if it does and the renderer is not running.
{
- const auto index = static_cast(m_Fractorium->m_Controller->Index()) + 1;
+ //If animating, then always start at 0 because the entire sequence usually needs to be rendered and if they need to start at a specific frame, they can use the Start At checkbox.
+ const auto index = m_FromSequence ? 0 : static_cast(m_Fractorium->m_Controller->Index()) + 1;
#ifdef DO_DOUBLE
Ember ed;
EmberFile efi;
@@ -976,7 +977,7 @@ void FractoriumFinalRenderDialog::showEvent(QShowEvent* e)
ui.FinalRenderDoAllCheckBox->setChecked(true);
//ui.FinalRenderDoSequenceCheckBox->setChecked(true);
ui.FinalRenderApplyToAllCheckBox->setChecked(true);
- m_PrefixEdit->setText(firstfile + "_");
+ //m_PrefixEdit->setText(firstfile + "_");
}
ui.FinalRenderTextOutput->clear();
diff --git a/Source/Fractorium/Fractorium.cpp b/Source/Fractorium/Fractorium.cpp
index be8ec6a..76c8f18 100644
--- a/Source/Fractorium/Fractorium.cpp
+++ b/Source/Fractorium/Fractorium.cpp
@@ -1240,6 +1240,14 @@ void Fractorium::SetTabOrders()
w = SetTabOrder(this, w, m_SupersampleSpin);
w = SetTabOrder(this, w, m_InterpTypeCombo);//Flame animation.
w = SetTabOrder(this, w, m_AffineInterpTypeCombo);
+ w = SetTabOrder(this, w, m_RotationsSpin);
+ w = SetTabOrder(this, w, m_SecondsPerRotationSpin);
+ w = SetTabOrder(this, w, m_RotateXformsDirCombo);
+ w = SetTabOrder(this, w, m_BlendSecondsSpin);
+ w = SetTabOrder(this, w, m_RotationsPerBlendSpin);
+ w = SetTabOrder(this, w, m_BlendXformsRotateDirCombo);
+ w = SetTabOrder(this, w, m_BlendInterpTypeCombo);
+ w = SetTabOrder(this, w, m_StaggerSpin);
w = SetTabOrder(this, w, m_TemporalFilterWidthSpin);
w = SetTabOrder(this, w, m_TemporalFilterTypeCombo);
w = SetTabOrder(this, w, m_TemporalFilterExpSpin);
@@ -1249,24 +1257,6 @@ void Fractorium::SetTabOrders()
w = SetTabOrder(this, w, ui.SequenceStartFlameSpinBox);
w = SetTabOrder(this, w, ui.SequenceStopFlameSpinBox);
w = SetTabOrder(this, w, ui.SequenceAllButton);
- w = SetTabOrder(this, w, ui.SequenceRandomizeStaggerCheckBox);
- w = SetTabOrder(this, w, ui.SequenceStaggerSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomStaggerMaxSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomizeRotationsCheckBox);
- w = SetTabOrder(this, w, ui.SequenceRotationsSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRotationsCWCheckBox);
- w = SetTabOrder(this, w, ui.SequenceRandomRotationsMaxSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomizeFramesPerRotCheckBox);
- w = SetTabOrder(this, w, ui.SequenceFramesPerRotSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomFramesPerRotMaxSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomizeBlendFramesCheckBox);
- w = SetTabOrder(this, w, ui.SequenceBlendFramesSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomBlendMaxFramesSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRandomizeRotationsPerBlendCheckBox);
- w = SetTabOrder(this, w, ui.SequenceRotationsPerBlendSpinBox);
- w = SetTabOrder(this, w, ui.SequenceRotationsPerBlendCWCheckBox);
- w = SetTabOrder(this, w, ui.SequenceRotationsPerBlendMaxSpinBox);
- w = SetTabOrder(this, w, ui.SequenceLinearCheckBox);
w = SetTabOrder(this, w, ui.SequenceAnimationFpsSpinBox);
w = SetTabOrder(this, w, ui.SequenceGenerateButton);
w = SetTabOrder(this, w, ui.SequenceRenderButton);
diff --git a/Source/Fractorium/Fractorium.h b/Source/Fractorium/Fractorium.h
index eb698d3..085486d 100644
--- a/Source/Fractorium/Fractorium.h
+++ b/Source/Fractorium/Fractorium.h
@@ -210,23 +210,8 @@ public slots:
void OnSequenceOpenButtonClicked(bool checked);
void OnSequenceAnimateButtonClicked(bool checked);
void OnSequenceClearButtonClicked(bool checked);
- void OnSequenceRandomizeStaggerCheckBoxStateChanged(int state);
- void OnSequenceRandomizeFramesPerRotCheckBoxStateChanged(int state);
- void OnSequenceRandomizeRotationsCheckBoxStateChanged(int state);
- void OnSequenceRandomizeBlendFramesCheckBoxStateChanged(int state);
- void OnSequenceRandomizeRotationsPerBlendCheckBoxStateChanged(int state);
- void OnSequenceStaggerSpinBoxChanged(double d);
- void OnSequenceRandomStaggerMaxSpinBoxChanged(double d);
void OnSequenceStartFlameSpinBoxChanged(int d);
void OnSequenceStopFlameSpinBoxChanged(int d);
- void OnSequenceFramesPerRotSpinBoxChanged(int d);
- void OnSequenceRandomFramesPerRotMaxSpinBoxChanged(int d);
- void OnSequenceRotationsSpinBoxChanged(double d);
- void OnSequenceRandomRotationsMaxSpinBoxChanged(double d);
- void OnSequenceBlendFramesSpinBoxChanged(int d);
- void OnSequenceRandomBlendMaxFramesSpinBoxChanged(int d);
- void OnSequenceRandomRotationsPerBlendSpinBoxChanged(int d);
- void OnSequenceRandomRotationsPerBlendMaxSpinBoxChanged(int d);
//Params.
void OnBrightnessChanged(double d);//Color.
@@ -265,6 +250,14 @@ public slots:
void OnRandRangeChanged(double d);
void OnQualityChanged(double d);
void OnSupersampleChanged(int d);
+ void OnRotationsChanged(double d);
+ void OnSecondsPerRotationChanged(double d);
+ void OnRotateXformsDirComboCurrentIndexChanged(int index);
+ void OnBlendSecondsChanged(double d);
+ void OnRotationsPerBlendChanged(int d);
+ void OnBlendXformsRotateDirComboCurrentIndexChanged(int index);
+ void OnBlendInterpTypeComboCurrentIndexChanged(int index);
+ void OnStaggerChanged(double d);
void OnAffineInterpTypeComboCurrentIndexChanged(int index);
void OnInterpTypeComboCurrentIndexChanged(int index);
void OnExpChanged(double d);
@@ -280,7 +273,8 @@ public slots:
void OnXformWeightChanged(double d);
void OnEqualWeightButtonClicked(bool checked);
void OnXformNameChanged(const QString& s);
- void OnXformAnimateCheckBoxStateChanged(int state);
+ void OnXformAnimateLocalRotationCheckBoxStateChanged(int state);
+ void OnXformAnimateOriginRotationCheckBoxStateChanged(int state);
//Xforms Affine.
void OnPreAffineRowDoubleClicked(int logicalIndex);
@@ -523,8 +517,6 @@ private:
DoubleSpinBox* m_BlurCurveSpin;
DoubleSpinBox* m_SpatialFilterWidthSpin;//Filter.
StealthComboBox* m_SpatialFilterTypeCombo;
- DoubleSpinBox* m_TemporalFilterWidthSpin;
- StealthComboBox* m_TemporalFilterTypeCombo;
DoubleSpinBox* m_DEFilterMinRadiusSpin;
DoubleSpinBox* m_DEFilterMaxRadiusSpin;
DoubleSpinBox* m_DECurveSpin;
@@ -533,8 +525,18 @@ private:
DoubleSpinBox* m_RandRangeSpin;
DoubleSpinBox* m_QualitySpin;
SpinBox* m_SupersampleSpin;
+ StealthComboBox* m_InterpTypeCombo;//Animation.
StealthComboBox* m_AffineInterpTypeCombo;
- StealthComboBox* m_InterpTypeCombo;
+ DoubleSpinBox* m_RotationsSpin;
+ DoubleSpinBox* m_SecondsPerRotationSpin;
+ StealthComboBox* m_RotateXformsDirCombo;
+ DoubleSpinBox* m_BlendSecondsSpin;
+ SpinBox* m_RotationsPerBlendSpin;
+ StealthComboBox* m_BlendXformsRotateDirCombo;
+ StealthComboBox* m_BlendInterpTypeCombo;
+ DoubleSpinBox* m_StaggerSpin;
+ DoubleSpinBox* m_TemporalFilterWidthSpin;
+ StealthComboBox* m_TemporalFilterTypeCombo;
DoubleSpinBox* m_TemporalFilterExpSpin;
//Xforms.
diff --git a/Source/Fractorium/Fractorium.ui b/Source/Fractorium/Fractorium.ui
index f824e9b..11be37f 100644
--- a/Source/Fractorium/Fractorium.ui
+++ b/Source/Fractorium/Fractorium.ui
@@ -6,8 +6,8 @@
0
0
- 1709
- 1054
+ 2048
+ 1270
@@ -77,8 +77,8 @@
0
0
- 1277
- 985
+ 1687
+ 1201
@@ -105,7 +105,7 @@
0
0
240
- 981
+ 2000
@@ -170,7 +170,7 @@
0
0
230
- 926
+ 1945
@@ -1345,18 +1345,21 @@
0
- 112
+ 288
16777215
- 112
+ 288
Qt::NoFocus
+
+
+
QFrame::Panel
@@ -1387,6 +1390,9 @@
false
+
+ 13
+
2
@@ -1427,6 +1433,49 @@
Affine Interpolation
+
+
+ Rotations
+
+
+
+
+ Seconds/Rotation
+
+
+
+
+ Rotate Xforms Dir
+
+
+
+
+
+
+
+ Blend Seconds
+
+
+
+
+ Rotations/Blend
+
+
+
+
+ Blend Rotate Xforms Dir
+
+
+
+
+ Blend Interpolation
+
+
+
+
+ Stagger
+
+
Temporal Filter Width
@@ -1466,29 +1515,108 @@
-
- Temporal Filter Width
+ Rotations
- <html><head/><body><p>The width of the temporal filter used during animation.</p><p>When computing the temporal samples to render, boundary frames must be computed to know what to blend between. By default this is just the sequence frames before and after the one currently being rendered.</p><p>However, the time bounds of what’s being blended can be less than that, or even greater which means it will be blended across several sequence steps.</p><p>In practice, this will almost always have its default value of 1.</p></body></html>
+ <html><head/><body><p>The number of times the xforms of a keyframe will do a 360 degree rotation before moving on to the next interpolation step. Note that only xforms which have the animate flag set to true will be rotated.</p><p>Set to 0 to omit rotation before interpolating.</p></body></html>
-
- Temporal Filter Type
+ Seconds/Rotation
- <html><head/><body><p>The type of the temporal filter used during animation.</p><p>The filter is an array of values equal in length to the temporal samples used during animation.</p><p>For each temporal step used to create motion blur between sequence frames, the palette will be multiplied by the value in the filter array at that same step position.</p><p>Box: All filter elements are 1, which means the palette is unchanged from its original values.</p><p>Gaussian: The filter values follow the shape of a Gaussian distribution: starting close to zero, increasing up to 1 at the halfway point, then falling back down to zero. This can cause the output to be darker than the original keyframes since almost every filter value is less than 1, which might not be desirable.</p><p>Exp: Use an exponent to shape the filter values. A value of 0 sets all filter values to 1, making it the equivalent of Box. For Exp values other than 1, the output will usually be darker than the original keyframes since almost every filter value is less than 1, which might not be desirable.</p></body></html>
+ <html><head/><body><p>The number of seconds each rotation tales tp complete. Larger values give slower movement.</p></body></html>
-
+
+ Rotate Xforms Dir
+
+
+ <html><head/><body><p>Direction of loop rotation. Clockwise vs. counter clockwise.</p></body></html>
+
+
+ -
+
+ Blend Seconds
+
+
+ <html><head/><body><p>The number of seconds taken to interpolate from one keyframe to the next. Larger values give slower movement.</p></body></html>
+
+
+ -
+
+
+
+
+ -
+
+ Rotations/Blend
+
+
+ <html><head/><body><p>The number of times the xforms will do a 360 degree rotation while interpolating from one keyframe to the next. Note that only xforms which have the animate flag set to true will be rotated.</p><p>Set to 0 to omit rotation while interpolating.</p></body></html>
+
+
+ -
+
+
+
+
+ -
+
+ Blend Rotate Xforms Dir
+
+
+ <html><head/><body><p>Direction of rotation during interpolation. Clockwise vs. counter clockwise.</p></body></html>
+
+
+ -
+
+ Blend Interpolation
+
+
+ <html><head/><body><p>The method used to compute the blending distance for each step between keyframes.</p><p>Linear: Each step will be a linear interpolation. This option is the equivalend of the --unsmoother flag in EmberGenome.</p><p>Log: Steps in the beginning and end of the interpolation between keyframes will be smaller and those in the middle will be larger. This has the effect of slowing down the beginning and end portions, and speeding up the middle.</p></body></html>
+
+
+ -
+
+ Stagger
+
+
+ <html><head/><body><p>A decimal number between 0 and 1 which governs the xform interpolation behavior like so:</p><p> 0 (default): xforms will interpolate all at once.</p><p> 0.5: the interpolation of each xform is staggered so that when the first xform is half done, the second one starts, and so on.</p><p> 1: each xform interpolates consecutively with no overlap.</p></body></html>
+
+
+ -
+
+
+
+
+ -
+
+ Temporal Filter Width
+
+
+ <html><head/><body><p>The width of the temporal filter used during animation when Render as Animation Sequence is checked. This increases rendering time and is usually not needed when rendering at a higher frame rate such as 60fps.</p><p>When computing the temporal samples to render, boundary frames must be computed to know what to blend between. By default this is just the sequence frames before and after the one currently being rendered.</p><p>However, the time bounds of what’s being blended can be less than that, or even greater which means it will be blended across several sequence steps.</p><p>In practice, this will almost always have its default value of 1.</p></body></html>
+
+
+ -
+
+ Temporal Filter Type
+
+
+ <html><head/><body><p>The type of the temporal filter used during animation when Render as Animation Sequence is checked. This increases rendering time and is usually not needed when rendering at a higher frame rate such as 60fps.</p><p>The filter is an array of values equal in length to the temporal samples used during animation.</p><p>For each temporal step used to create motion blur between sequence frames, the palette will be multiplied by the value in the filter array at that same step position.</p><p>Box: All filter elements are 1, which means the palette is unchanged from its original values.</p><p>Gaussian: The filter values follow the shape of a Gaussian distribution: starting close to zero, increasing up to 1 at the halfway point, then falling back down to zero. This can cause the output to be darker than the original keyframes since almost every filter value is less than 1, which might not be desirable.</p><p>Exp: Use an exponent to shape the filter values. A value of 0 sets all filter values to 1, making it the equivalent of Box. For Exp values other than 1, the output will usually be darker than the original keyframes since almost every filter value is less than 1, which might not be desirable.</p></body></html>
+
+
+ -
Temporal Exp Value
- <html><head/><body><p>The exponent value when using the Exp temporal filter type.</p><p>0: The value for every temporal sample is 1, which makes this equivalent to the Box filter.</p><p>0.5: Convex curve from 0 to 1.</p><p>1: Straight line from 0 to 1.</p><p>> 1: Concave curve from 0 to 1, with a more gradual onset the larger the value is.</p><p>Otherwise unused.</p></body></html>
+ <html><head/><body><p>The exponent value when using the Exp temporal filter type when Render as Animation Sequence is checked. This increases rendering time and is usually not needed when rendering at a higher frame rate such as 60fps.</p><p>0: The value for every temporal sample is 1, which makes this equivalent to the Box filter.</p><p>0.5: Convex curve from 0 to 1.</p><p>1: Straight line from 0 to 1.</p><p>> 1: Concave curve from 0 to 1, with a more gradual onset the larger the value is.</p><p>Otherwise unused.</p></body></html>
- -
+
-
@@ -1880,7 +2008,7 @@
310
- 444
+ 445
@@ -2034,7 +2162,7 @@
0
- 2
+ 3
-
@@ -2407,7 +2535,7 @@
860
- 10
+ 0
295
761
@@ -3095,7 +3223,7 @@
240
0
301
- 901
+ 981
@@ -3150,7 +3278,7 @@
0
0
291
- 869
+ 949
@@ -3547,13 +3675,59 @@
-
-
-
- <html><head/><body><p>Rotate this xform during the rotation phase when creating a sequence for animation</p></body></html>
+
+
+
+ 0
+ 0
+
-
- Animate
+
+
+ 0
+ 0
+
+
+ Animation
+
+
+
+ 0
+
+
+ 9
+
+
+ 0
+
+
+ 0
+
+
+ 6
+
+
-
+
+
+ <html><head/><body><p>Rotate this xform around its local center during the rotation phase when creating a sequence for animation</p></body></html>
+
+
+ Local rotation
+
+
+
+ -
+
+
+ <html><head/><body><p>Rotate this xform around the world center during the rotation phase when creating a sequence for animation</p></body></html>
+
+
+ Origin rotation
+
+
+
+
-
@@ -6265,7 +6439,7 @@
0
0
269
- 650
+ 706
@@ -6325,7 +6499,7 @@
- 1020
+ 1160
0
301
881
@@ -7272,7 +7446,7 @@
0
0
- 1709
+ 2048
22
@@ -7511,8 +7685,8 @@
0
0
- 418
- 631
+ 347
+ 742
@@ -7521,7 +7695,7 @@
0
-
+
4
@@ -7542,162 +7716,6 @@
4
- -
-
-
-
- 0
- 0
-
-
-
-
- 33
- 0
-
-
-
-
- 16777215
- 16777215
-
-
-
- All
-
-
-
- -
-
-
- false
-
-
- QAbstractSpinBox::NoButtons
-
-
- Stagger max:
-
-
- 1.000000000000000
-
-
- 0.100000000000000
-
-
- 1.000000000000000
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
-
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- <html><head/><body><p>A decimal number between 0 and 1 which governs the xform interpolation behavior like so:</p><p> 0 (default): xforms will interpolate all at once.</p><p> 0.5: the interpolation of each xform is staggered so that when the first xform is half done, the second one starts, and so on.</p><p> 1: each xform interpolates consecutively with no overlap.</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
- Stagger:
-
-
- 1.000000000000000
-
-
- 0.100000000000000
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
-
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
-
-
-
-
- -
-
-
- QLayout::SetDefaultConstraint
-
-
- 4
-
-
-
-
-
-
- 0
- 0
-
-
-
- <html><head/><body><p>The number of times the xforms of a keyframe will do a 360 degree rotation before moving on to the next interpolation step. Note that only xforms which have the animate flag set to true will be rotated.</p><p>Set to 0 to omit rotation before interpolating.</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
- Rotations:
-
-
- 10000.000000000000000
-
-
- 3.000000000000000
-
-
-
- -
-
-
- <html><head/><body><p>Direction of loop rotation. Clockwise vs. counter clockwise.</p></body></html>
-
-
- CW
-
-
-
-
-
-
@@ -7729,103 +7747,17 @@
- -
-
-
- false
-
-
- QAbstractSpinBox::NoButtons
-
-
- Frames per rot max:
-
-
- 1
-
-
- 10000
-
-
- 30
-
-
-
- -
-
-
- false
-
-
- QAbstractSpinBox::NoButtons
-
-
- Rot per blend max:
-
-
- 0
-
-
-
- -
-
-
- 4
-
-
-
-
-
-
- 0
- 0
-
-
-
-
- 95
- 0
-
-
-
- <html><head/><body><p>The number of times the xforms will do a 360 degree rotation while interpolating from one keyframe to the next. Note that only xforms which have the animate flag set to true will be rotated.</p><p>Set to 0 to omit rotation while interpolating.</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
- Rot per blend:
-
-
- 100
-
-
- 1
-
-
-
- -
-
-
- <html><head/><body><p>Direction of rotation during interpolation. Clockwise vs. counter clockwise.</p></body></html>
-
-
- CW
-
-
-
-
-
- -
-
+
-
+
-
+
0
0
- 0
+ 33
0
@@ -7836,174 +7768,7 @@
-
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
-
- 16777215
- 16777215
-
-
-
- <html><head/><body><p>The 0-based index of the last flame in the file to end the sequence on</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
- Stop flame:
-
-
- 999999999
-
-
-
- -
-
-
- false
-
-
- QAbstractSpinBox::NoButtons
-
-
- Rotations max:
-
-
- 10000.000000000000000
-
-
- 3.000000000000000
-
-
-
- -
-
-
- false
-
-
- QAbstractSpinBox::NoButtons
-
-
- Blend frames max:
-
-
- 10000
-
-
- 120
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- Stop Previews
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- <html><head/><body><p>The number of frames used to interpolate from one keyframe to the next. Larger values give slower movement.</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
-
-
-
- Blend frames:
-
-
- 10000
-
-
- 120
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- <html><head/><body><p>Check any checkbox to have the value corresponding to its row be randomly selected between the values of first and second controls on that row</p></body></html>
-
-
- Qt::LeftToRight
-
-
- Random
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- <html><head/><body><p>The number of frames for each rotation specified by Rotations. Larger values give slower movement.</p></body></html>
-
-
- QAbstractSpinBox::NoButtons
-
-
- Frames per rot:
-
-
- 1
-
-
- 10000
-
-
- 30
-
-
-
- -
-
-
-
- 0
- 0
-
-
-
- Start Previews
+ All
@@ -8038,17 +7803,81 @@
- -
-
-
- <html><head/><body><p>The method used to compute the blending distance for each step between keyframes.</p><p>Checked: Each step will be a linear interpolation.</p><p>Unchecked: Steps in the beginning and end of the interpolation between keyframes will be smaller and those in the middle will be larger. This has the effect of slowing down the beginning and end portions, and speeding up the middle.</p><p>This option is the equivalend of the --unsmoother flag in EmberGenome.</p></body></html>
+
-
+
+
+
+ 0
+ 0
+
-
- Linear
+
+
+ 16777215
+ 16777215
+
+
+
+ <html><head/><body><p>The 0-based index of the last flame in the file to end the sequence on</p></body></html>
+
+
+ QAbstractSpinBox::NoButtons
+
+
+ Stop flame:
+
+
+ 999999999
- -
+
-
+
+
+
+ 0
+ 0
+
+
+
+ Stop Previews
+
+
+
+ -
+
+
+
+ 0
+ 0
+
+
+
+ Start Previews
+
+
+
+
+
+ -
+
+
+ 0
+
+
+ 0
+
+
+ 4
+
+
-
+
+
+ 0
+
+
+
+ -
@@ -8057,7 +7886,7 @@
- <html><head/><body><p>The number of frames per second for preview animation.</p></body></html>
+ <html><head/><body><p>The number of frames per second in the animation.</p></body></html>
QAbstractSpinBox::NoButtons
@@ -8069,23 +7898,16 @@
Animation FPS:
- 1
+ 10
- 200
+ 240
30
- -
-
-
- 0
-
-
-
-
diff --git a/Source/Fractorium/FractoriumEmberController.cpp b/Source/Fractorium/FractoriumEmberController.cpp
index acdf965..8668c1a 100644
--- a/Source/Fractorium/FractoriumEmberController.cpp
+++ b/Source/Fractorium/FractoriumEmberController.cpp
@@ -21,7 +21,6 @@ FractoriumEmberControllerBase::FractoriumEmberControllerBase(Fractorium* fractor
m_AnimateTimer = make_unique(m_Fractorium);
m_AnimateTimer->stop();
m_Fractorium->connect(m_RenderRestartTimer.get(), &QTimer::timeout, [&]() { m_Fractorium->StartRenderTimer(false); });//It's ok to pass false for the first shot because creating the controller will start the preview renders.
- // XXX: why not SLOT(SequenceAnimateNextFrame())?
m_Fractorium->connect(m_AnimateTimer.get(), &QTimer::timeout, [&]() { SequenceAnimateNextFrame(); });
}
diff --git a/Source/Fractorium/FractoriumEmberController.h b/Source/Fractorium/FractoriumEmberController.h
index 506c0b4..dc68af0 100644
--- a/Source/Fractorium/FractoriumEmberController.h
+++ b/Source/Fractorium/FractoriumEmberController.h
@@ -160,6 +160,7 @@ public:
virtual void VibrancyChanged(double d) { }
virtual void HighlightPowerChanged(double d) { }
virtual void K2Changed(double d) { }
+ virtual void BackgroundChanged(const QColor& color) { }
virtual void PaletteModeChanged(uint i) { }
virtual void WidthChanged(uint i) { }
virtual void HeightChanged(uint i) { }
@@ -177,8 +178,6 @@ public:
virtual void BlurCurveChanged(double d) { }
virtual void SpatialFilterWidthChanged(double d) { }
virtual void SpatialFilterTypeChanged(const QString& text) { }
- virtual void TemporalFilterWidthChanged(double d) { }
- virtual void TemporalFilterTypeChanged(const QString& text) { }
virtual void DEFilterMinRadiusWidthChanged(double d) { }
virtual void DEFilterMaxRadiusWidthChanged(double d) { }
virtual void DEFilterCurveWidthChanged(double d) { }
@@ -187,21 +186,30 @@ public:
virtual void RandRangeChanged(double d) { }
virtual void QualityChanged(double d) { }
virtual void SupersampleChanged(int d) { }
- virtual void AffineInterpTypeChanged(int i) { }
virtual void InterpTypeChanged(int i) { }
- virtual void BackgroundChanged(const QColor& color) { }
+ virtual void AffineInterpTypeChanged(int i) { }
+ virtual void RotationsChanged(double d) { }
+ virtual void SecondsPerRotationChanged(double d) { }
+ virtual void RotateXformsDirChanged(uint i) { }
+ virtual void BlendSecondsChanged(double d) { };
+ virtual void RotationsPerBlendChanged(uint i) { };
+ virtual void BlendXformsRotateDirChanged(uint i) { }
+ virtual void BlendInterpTypeChanged(uint i) { };
+ virtual void StaggerChanged(double d) { };
+ virtual void TemporalFilterWidthChanged(double d) { }
+ virtual void TemporalFilterTypeChanged(const QString& text) { }
+ virtual void ExpChanged(double d) { }
virtual void ClearColorCurves(int i) { }
virtual void ColorCurveChanged(int curveIndex, int pointInxed, const QPointF& point) { }
virtual void ColorCurvesPointAdded(size_t curveIndex, const QPointF& point) { }
virtual void ColorCurvesPointRemoved(size_t curveIndex, int pointIndex) { }
- virtual void ExpChanged(double d) { }
//Xforms.
virtual void CurrentXformComboChanged(int index) { }
virtual void XformWeightChanged(double d) { }
virtual void EqualizeWeights() { }
virtual void XformNameChanged(const QString& s) { }
- virtual void XformAnimateChanged(int state) { }
+ virtual void XformAnimateChangedHelper(int state, bool local) { }
virtual void FillXforms(int index = 0) { }
virtual void UpdateXformName(int index) { }
@@ -463,6 +471,7 @@ public:
void VibrancyChanged(double d) override;
void HighlightPowerChanged(double d) override;
void K2Changed(double d) override;
+ void BackgroundChanged(const QColor& col) override;
void PaletteModeChanged(uint i) override;
void WidthChanged(uint i) override;
void HeightChanged(uint i) override;
@@ -480,8 +489,6 @@ public:
void BlurCurveChanged(double d) override;
void SpatialFilterWidthChanged(double d) override;
void SpatialFilterTypeChanged(const QString& text) override;
- void TemporalFilterWidthChanged(double d) override;
- void TemporalFilterTypeChanged(const QString& text) override;
void DEFilterMinRadiusWidthChanged(double d) override;
void DEFilterMaxRadiusWidthChanged(double d) override;
void DEFilterCurveWidthChanged(double d) override;
@@ -490,21 +497,30 @@ public:
void RandRangeChanged(double d) override;
void QualityChanged(double d) override;
void SupersampleChanged(int d) override;
- void AffineInterpTypeChanged(int index) override;
void InterpTypeChanged(int index) override;
- void BackgroundChanged(const QColor& col) override;
+ void AffineInterpTypeChanged(int index) override;
+ void RotationsChanged(double d) override;
+ void SecondsPerRotationChanged(double d) override;
+ void RotateXformsDirChanged(uint i) override;
+ void BlendSecondsChanged(double d) override;
+ void RotationsPerBlendChanged(uint i) override;
+ void BlendXformsRotateDirChanged(uint i) override;
+ void BlendInterpTypeChanged(uint i) override;
+ void StaggerChanged(double d) override;
+ void TemporalFilterWidthChanged(double d) override;
+ void TemporalFilterTypeChanged(const QString& text) override;
+ void ExpChanged(double d) override;
void ClearColorCurves(int i) override;
void ColorCurveChanged(int curveIndex, int pointInxed, const QPointF& point) override;
void ColorCurvesPointAdded(size_t curveIndex, const QPointF& point) override;
void ColorCurvesPointRemoved(size_t curveIndex, int pointIndex) override;
- void ExpChanged(double d) override;
//Xforms.
void CurrentXformComboChanged(int index) override;
void XformWeightChanged(double d) override;
void EqualizeWeights() override;
void XformNameChanged(const QString& s) override;
- void XformAnimateChanged(int state) override;
+ void XformAnimateChangedHelper(int state, bool local) override;
void FillXforms(int index = 0) override;
void UpdateXformName(int index) override;
void FillWithXform(Xform* xform);
diff --git a/Source/Fractorium/FractoriumLibrary.cpp b/Source/Fractorium/FractoriumLibrary.cpp
index a761219..0e0cedc 100644
--- a/Source/Fractorium/FractoriumLibrary.cpp
+++ b/Source/Fractorium/FractoriumLibrary.cpp
@@ -8,63 +8,21 @@ void Fractorium::InitLibraryUI()
{
ui.LibraryTree->SetMainWindow(this);
//Making the TreeItemChanged() events use a direct connection is absolutely critical.
- connect(ui.LibraryTree, SIGNAL(itemChanged(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemChanged(QTreeWidgetItem*, int)), Qt::DirectConnection);
- connect(ui.LibraryTree, SIGNAL(itemDoubleClicked(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemDoubleClicked(QTreeWidgetItem*, int)), Qt::QueuedConnection);
- connect(ui.LibraryTree, SIGNAL(itemActivated(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemDoubleClicked(QTreeWidgetItem*, int)), Qt::QueuedConnection);
- connect(ui.SequenceTree, SIGNAL(itemChanged(QTreeWidgetItem*, int)), this, SLOT(OnSequenceTreeItemChanged(QTreeWidgetItem*, int)), Qt::DirectConnection);
- connect(ui.SequenceStartPreviewsButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceStartPreviewsButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceStopPreviewsButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceStopPreviewsButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceAllButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceAllButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceGenerateButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceGenerateButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceRenderButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceRenderButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceAnimateButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceAnimateButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceClearButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceClearButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceSaveButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceSaveButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceOpenButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceOpenButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.SequenceRandomizeStaggerCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnSequenceRandomizeStaggerCheckBoxStateChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomizeFramesPerRotCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnSequenceRandomizeFramesPerRotCheckBoxStateChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomizeRotationsCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnSequenceRandomizeRotationsCheckBoxStateChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomizeBlendFramesCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnSequenceRandomizeBlendFramesCheckBoxStateChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomizeRotationsPerBlendCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnSequenceRandomizeRotationsPerBlendCheckBoxStateChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceStaggerSpinBox, SIGNAL(valueChanged(double)), this, SLOT(OnSequenceStaggerSpinBoxChanged(double)), Qt::QueuedConnection);
- connect(ui.SequenceRandomStaggerMaxSpinBox, SIGNAL(valueChanged(double)), this, SLOT(OnSequenceRandomStaggerMaxSpinBoxChanged(double)), Qt::QueuedConnection);
- connect(ui.SequenceStartFlameSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceStartFlameSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceStopFlameSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceStopFlameSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceFramesPerRotSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceFramesPerRotSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomFramesPerRotMaxSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceRandomFramesPerRotMaxSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRotationsSpinBox, SIGNAL(valueChanged(double)), this, SLOT(OnSequenceRotationsSpinBoxChanged(double)), Qt::QueuedConnection);
- connect(ui.SequenceRandomRotationsMaxSpinBox, SIGNAL(valueChanged(double)), this, SLOT(OnSequenceRandomRotationsMaxSpinBoxChanged(double)), Qt::QueuedConnection);
- connect(ui.SequenceBlendFramesSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceBlendFramesSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRandomBlendMaxFramesSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceRandomBlendMaxFramesSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRotationsPerBlendSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceRandomRotationsPerBlendSpinBoxChanged(int)), Qt::QueuedConnection);
- connect(ui.SequenceRotationsPerBlendMaxSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceRandomRotationsPerBlendMaxSpinBoxChanged(int)), Qt::QueuedConnection);
- //Stagger
- ui.SequenceStaggerSpinBox->setValue(m_Settings->Stagger());//Lower.
- ui.SequenceStaggerSpinBox->setMaximum(std::numeric_limits::max());//Lower max = upper.
- ui.SequenceRandomStaggerMaxSpinBox->setValue(m_Settings->StaggerMax());//Upper.
- ui.SequenceRandomStaggerMaxSpinBox->setMinimum(m_Settings->Stagger());//Upper min = lower max.
- //Frames per rotation.
- ui.SequenceFramesPerRotSpinBox->setValue(m_Settings->FramesPerRot());//Lower.
- ui.SequenceFramesPerRotSpinBox->setMaximum(std::numeric_limits::max());//Lower max = upper.
- ui.SequenceRandomFramesPerRotMaxSpinBox->setValue(m_Settings->FramesPerRotMax());//Upper.
- ui.SequenceRandomFramesPerRotMaxSpinBox->setMinimum(m_Settings->FramesPerRot());//Upper min = lower max.
- //Rotations.
- ui.SequenceRotationsSpinBox->setValue(m_Settings->Rotations());//Lower.
- ui.SequenceRotationsSpinBox->setMaximum(std::numeric_limits::max());//Lower max = upper.
- ui.SequenceRandomRotationsMaxSpinBox->setValue(m_Settings->RotationsMax());//Upper.
- ui.SequenceRandomRotationsMaxSpinBox->setMinimum(m_Settings->Rotations());//Upper min = lower max.
- //Blend frames.
- ui.SequenceBlendFramesSpinBox->setValue(m_Settings->BlendFrames());//Lower.
- ui.SequenceBlendFramesSpinBox->setMaximum(std::numeric_limits::max());//Lower max = upper.
- ui.SequenceRandomBlendMaxFramesSpinBox->setValue(m_Settings->BlendFramesMax());//Upper.
- ui.SequenceRandomBlendMaxFramesSpinBox->setMinimum(m_Settings->BlendFrames());//Upper min = lower max.
- //Rotations per blend.
- ui.SequenceRotationsPerBlendSpinBox->setValue(m_Settings->RotationsPerBlend());//Lower.
- ui.SequenceRotationsPerBlendSpinBox->setMaximum(std::numeric_limits::max());//Lower max = upper.
- ui.SequenceRotationsPerBlendMaxSpinBox->setValue(m_Settings->RotationsPerBlendMax());//Upper.
- ui.SequenceRotationsPerBlendMaxSpinBox->setMinimum(m_Settings->RotationsPerBlend());//Upper min = lower max.
- //Linear.
- ui.SequenceLinearCheckBox->setChecked(m_Settings->Linear());
+ connect(ui.LibraryTree, SIGNAL(itemChanged(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemChanged(QTreeWidgetItem*, int)), Qt::DirectConnection);
+ connect(ui.LibraryTree, SIGNAL(itemDoubleClicked(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemDoubleClicked(QTreeWidgetItem*, int)), Qt::QueuedConnection);
+ connect(ui.LibraryTree, SIGNAL(itemActivated(QTreeWidgetItem*, int)), this, SLOT(OnEmberTreeItemDoubleClicked(QTreeWidgetItem*, int)), Qt::QueuedConnection);
+ connect(ui.SequenceTree, SIGNAL(itemChanged(QTreeWidgetItem*, int)), this, SLOT(OnSequenceTreeItemChanged(QTreeWidgetItem*, int)), Qt::DirectConnection);
+ connect(ui.SequenceStartPreviewsButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceStartPreviewsButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceStopPreviewsButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceStopPreviewsButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceAllButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceAllButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceGenerateButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceGenerateButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceRenderButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceRenderButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceAnimateButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceAnimateButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceClearButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceClearButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceSaveButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceSaveButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceOpenButton, SIGNAL(clicked(bool)), this, SLOT(OnSequenceOpenButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.SequenceStartFlameSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceStartFlameSpinBoxChanged(int)), Qt::QueuedConnection);
+ connect(ui.SequenceStopFlameSpinBox, SIGNAL(valueChanged(int)), this, SLOT(OnSequenceStopFlameSpinBoxChanged(int)), Qt::QueuedConnection);
//Animation FPS.
ui.SequenceAnimationFpsSpinBox->setValue(m_Settings->AnimationFps());
}
@@ -597,7 +555,6 @@ void FractoriumEmberController::RenderSequencePreviews(uint start, uint end)
{
RenderPreviews(m_Fractorium->ui.SequenceTree, m_SequencePreviewRenderer.get(), m_SequenceFile, start, end);
}
-
void Fractorium::OnSequenceStartPreviewsButtonClicked(bool checked) { m_Controller->RenderSequencePreviews(); }
///
@@ -630,7 +587,6 @@ void Fractorium::SyncFileCountToSequenceCount()
ui.SequenceStopFlameSpinBox->setValue(count);
}
}
-
void Fractorium::OnSequenceAllButtonClicked(bool checked) { SyncFileCountToSequenceCount(); }
///
@@ -649,36 +605,13 @@ void FractoriumEmberController::SequenceGenerateButtonClicked()
Ember result;
auto& ui = m_Fractorium->ui;
auto s = m_Fractorium->m_Settings;
- //Bools for determining whether to use hard coded vs. random values.
- const bool randStagger = ui.SequenceRandomizeStaggerCheckBox->isChecked();
- const bool randFramesRot = ui.SequenceRandomizeFramesPerRotCheckBox->isChecked();
- const bool randRot = ui.SequenceRandomizeRotationsCheckBox->isChecked();
- const bool randBlend = ui.SequenceRandomizeBlendFramesCheckBox->isChecked();
- const bool randBlendRot = ui.SequenceRandomizeRotationsPerBlendCheckBox->isChecked();
- //The direction to rotate the loops.
- const bool loopsCw = ui.SequenceRotationsCWCheckBox->isChecked();
- const bool loopsBlendCw = ui.SequenceRotationsPerBlendCWCheckBox->isChecked();
- //Whether to stagger, default is 1 which means no stagger.
- const double stagger = ui.SequenceStaggerSpinBox->value();
- const double staggerMax = ui.SequenceRandomStaggerMaxSpinBox->value();
- //Rotations on keyframes.
- const double rots = ui.SequenceRotationsSpinBox->value();
- const double rotsMax = ui.SequenceRandomRotationsMaxSpinBox->value();
- //Number of frames it takes to rotate a keyframe.
- const int framesPerRot = ui.SequenceFramesPerRotSpinBox->value();
- const int framesPerRotMax = ui.SequenceRandomFramesPerRotMaxSpinBox->value();
- //Number of frames it takes to interpolate.
- const int framesBlend = ui.SequenceBlendFramesSpinBox->value();
- const int framesBlendMax = ui.SequenceRandomBlendMaxFramesSpinBox->value();
//Number of rotations performed during interpolation.
- const int rotsPerBlend = ui.SequenceRotationsPerBlendSpinBox->value();
- const int rotsPerBlendMax = ui.SequenceRotationsPerBlendMaxSpinBox->value();
const size_t start = ui.SequenceStartFlameSpinBox->value();
const size_t stop = ui.SequenceStopFlameSpinBox->value();
const size_t startCount = ui.SequenceStartCountSpinBox->value();
- const bool linear = ui.SequenceLinearCheckBox->isChecked();
const size_t keyFrames = (stop - start) + 1;
size_t frameCount = 0;
+ size_t fps = ui.SequenceAnimationFpsSpinBox->value();
double frames = 0;
vector> devices;//Dummy.
EmberReport emberReport;
@@ -691,54 +624,24 @@ void FractoriumEmberController::SequenceGenerateButtonClicked()
return;
}
- if (!randRot && !randBlend)
+ SheepTools tools(palettePath, EmberCommon::CreateRenderer(eRendererType::CPU_RENDERER, devices, false, 0, emberReport));
+ auto it = Advance(m_EmberFile.m_Embers.begin(), start);
+
+ for (size_t i = start; i <= stop && it != m_EmberFile.m_Embers.end(); i++, ++it)
{
- if ((!rots || !framesPerRot) && !framesBlend)
+ if ((!it->m_Rotations || !it->m_SecondsPerRotation) && !it->m_BlendSeconds)
{
QMessageBox::critical(m_Fractorium, "Animation sequence parameters error",
- "Rotations and Frames per rot, or blend frames must be positive and non-zero");
+ "Rotations and seconds per rotation, or blend seconds must be positive and non-zero");
return;
}
- //if (framesPerRot > 1 && !rots)//Because framesPerRot control has a min value of 1, check greater than 1. Also don't need to check the inverse like in EmberGenome.
- //{
- //QMessageBox::critical(m_Fractorium, "Animation sequence parameters error",
- // "Frames per rot cannot be greater than one while Rotations is zero. Setting it to 1.");
- //ui.SequenceFramesPerRotSpinBox->setValue(1);
- //return;
- //}
+ frames += fps * it->m_SecondsPerRotation * it->m_Rotations;
+
+ if (i < stop)
+ frames += fps * it->m_BlendSeconds;
}
- SheepTools tools(palettePath, EmberCommon::CreateRenderer(eRendererType::CPU_RENDERER, devices, false, 0, emberReport));
- tools.SetSpinParams(!linear,
- stagger,//Will be set again below if random is used.
- 0,
- 0,
- s->Nick().toStdString(),
- s->Url().toStdString(),
- s->Id().toStdString(),
- "",
- 0,
- 0);
-
- if (randFramesRot)
- frames = ui.SequenceRandomFramesPerRotMaxSpinBox->value();
- else if (rots)
- frames = ui.SequenceFramesPerRotSpinBox->value();
- else
- frames = 1;
-
- if (randRot)
- frames *= ui.SequenceRandomRotationsMaxSpinBox->value();
- else
- frames *= ui.SequenceRotationsSpinBox->value();
-
- if (randBlend)
- frames += ui.SequenceRandomBlendMaxFramesSpinBox->value();
- else
- frames += ui.SequenceBlendFramesSpinBox->value();
-
- frames *= keyFrames;
frames += startCount;
os << setfill('0') << setprecision(0) << fixed;
m_SequenceFile.Clear();
@@ -747,16 +650,38 @@ void FractoriumEmberController::SequenceGenerateButtonClicked()
size_t frame;
Ember embers[2];//Spin needs contiguous array below, and this will also get modified, so a copy is needed to avoid modifying the embers in the original file.
const auto padding = streamsize(std::log10(frames)) + 1;
- auto it = Advance(m_EmberFile.m_Embers.begin(), start);
+ auto lastLoopsBlendCw = false;
+ it = Advance(m_EmberFile.m_Embers.begin(), start);
for (size_t i = start; i <= stop && it != m_EmberFile.m_Embers.end(); i++, ++it)
{
- const auto rotations = randRot ? m_Rand.Frand(rots, rotsMax) : rots;
+ //The direction to rotate the loops.
+ const auto loopsCw = it->m_RotateXformsCw;//ui.SequenceRotationsCWCheckBox->isChecked();
+ const auto loopsBlendCw = it->m_BlendRotateXformsCw;//ui.SequenceRotationsPerBlendCWCheckBox->isChecked();
+ //Whether to stagger, default is 1 which means no stagger.
+ const auto stagger = it->m_Stagger;//ui.SequenceStaggerSpinBox->value();
+ //Rotations on keyframes.
+ const auto rotations = it->m_Rotations;
+ //Number of frames it takes to rotate a keyframe.
+ const auto rotFrames = fps * (double)it->m_SecondsPerRotation;
+ //Number of frames it takes to interpolate.
+ const auto framesBlend = fps * it->m_BlendSeconds;
+ const auto rotsPerBlend = it->m_RotationsPerBlend;
+ const auto linear = it->m_Linear;
+ tools.SetSpinParams(!linear,
+ stagger,//Will be set again below if random is used.
+ 0,
+ 0,
+ s->Nick().toStdString(),
+ s->Url().toStdString(),
+ s->Id().toStdString(),
+ "",
+ 0,
+ 0);
embers[0] = *it;
if (rotations > 0)
{
- const auto rotFrames = randFramesRot ? m_Rand.Frand(framesPerRot, framesPerRotMax) : framesPerRot;
const auto roundFrames = size_t(std::round(rotFrames * rotations));
for (frame = 0; frame < roundFrames; frame++)
@@ -783,32 +708,29 @@ void FractoriumEmberController::SequenceGenerateButtonClicked()
auto it2 = it;//Need a quick temporary to avoid modifying it, which is used in the loop.
embers[1] = *(++it2);//Get the next ember to be used with blending below.
- const auto blendFrames = randBlend ? m_Rand.Frand(framesBlend, framesBlendMax) : framesBlend;
- const auto d = randBlendRot ? m_Rand.Frand(rotsPerBlend, rotsPerBlendMax) : double(rotsPerBlend);
+ const auto d = double(rotsPerBlend);
const auto rpb = size_t(std::round(d));
- if (randStagger)
- tools.Stagger(m_Rand.Frand(stagger, staggerMax));
-
- for (frame = 0; frame < blendFrames; frame++)
+ for (frame = 0; frame < framesBlend; frame++)
{
- const auto seqFlag = frame == 0 || (frame == blendFrames - 1);
- blend = frame / double(blendFrames);
+ const auto seqFlag = frame == 0 || (frame == framesBlend - 1);
+ blend = frame / framesBlend;
result.Clear();
tools.SpinInter(&embers[0], nullptr, result, startCount + frameCount++, seqFlag, blend, rpb, loopsBlendCw);
FormatName(result, os, padding);
m_SequenceFile.m_Embers.push_back(result);
}
}
+
+ lastLoopsBlendCw = loopsBlendCw;
}
it = Advance(m_EmberFile.m_Embers.begin(), stop);
- tools.Spin(*it, nullptr, result, startCount + frameCount, 0, loopsBlendCw);
+ tools.Spin(*it, nullptr, result, startCount + frameCount, 0, lastLoopsBlendCw);
FormatName(result, os, padding);
m_SequenceFile.m_Embers.push_back(result);
FillSequenceTree();//The sequence has been generated, now create preview thumbnails.
}
-
void Fractorium::OnSequenceGenerateButtonClicked(bool checked) { m_Controller->SequenceGenerateButtonClicked(); }
///
@@ -886,7 +808,6 @@ void FractoriumEmberController::SequenceAnimateButtonClicked()
}
}
}
-
void Fractorium::OnSequenceAnimateButtonClicked(bool checked) { m_Controller->SequenceAnimateButtonClicked(); }
///
@@ -898,7 +819,6 @@ void FractoriumEmberController::SequenceClearButtonClicked()
m_SequencePreviewRenderer->Stop();
m_Fractorium->ui.SequenceTree->clear();
}
-
void Fractorium::OnSequenceClearButtonClicked(bool checked) { m_Controller->SequenceClearButtonClicked(); }
///
@@ -924,7 +844,6 @@ void FractoriumEmberController::SequenceSaveButtonClicked()
m_Fractorium->ShowCritical("Save Failed", "Could not save sequence file, try saving to a different folder.");
}
}
-
void Fractorium::OnSequenceSaveButtonClicked(bool checked) { m_Controller->SequenceSaveButtonClicked(); }
///
@@ -977,79 +896,19 @@ void FractoriumEmberController::SequenceOpenButtonClicked()
}
}
}
-
void Fractorium::OnSequenceOpenButtonClicked(bool checked) { m_Controller->SequenceOpenButtonClicked(); }
-///
-/// Constrain all min/max spinboxes when the max spinboxes are enabled/disabled via the random checkbox.
-///
-void Fractorium::OnSequenceRandomizeStaggerCheckBoxStateChanged(int state)
-{
- ui.SequenceRandomStaggerMaxSpinBox->setMinimum(ui.SequenceStaggerSpinBox->value());
- ui.SequenceStaggerSpinBox->setMaximum(state ? ui.SequenceRandomStaggerMaxSpinBox->value() : std::numeric_limits::max());
- ui.SequenceRandomStaggerMaxSpinBox->setEnabled(state);
-}
-
-void Fractorium::OnSequenceRandomizeFramesPerRotCheckBoxStateChanged(int state)
-{
- ui.SequenceRandomFramesPerRotMaxSpinBox->setMinimum(ui.SequenceFramesPerRotSpinBox->value());
- ui.SequenceFramesPerRotSpinBox->setMaximum(state ? ui.SequenceRandomFramesPerRotMaxSpinBox->value() : std::numeric_limits::max());
- ui.SequenceRandomFramesPerRotMaxSpinBox->setEnabled(state);
-}
-
-void Fractorium::OnSequenceRandomizeRotationsCheckBoxStateChanged(int state)
-{
- ui.SequenceRandomRotationsMaxSpinBox->setMinimum(ui.SequenceRotationsSpinBox->value());
- ui.SequenceRotationsSpinBox->setMaximum(state ? ui.SequenceRandomRotationsMaxSpinBox->value() : std::numeric_limits::max());
- ui.SequenceRandomRotationsMaxSpinBox->setEnabled(state);
-}
-
-void Fractorium::OnSequenceRandomizeBlendFramesCheckBoxStateChanged(int state)
-{
- ui.SequenceRandomBlendMaxFramesSpinBox->setMinimum(ui.SequenceBlendFramesSpinBox->value());
- ui.SequenceBlendFramesSpinBox->setMaximum(state ? ui.SequenceRandomBlendMaxFramesSpinBox->value() : std::numeric_limits::max());
- ui.SequenceRandomBlendMaxFramesSpinBox->setEnabled(state);
-}
-
-void Fractorium::OnSequenceRandomizeRotationsPerBlendCheckBoxStateChanged(int state)
-{
- ui.SequenceRotationsPerBlendMaxSpinBox->setMinimum(ui.SequenceRotationsPerBlendSpinBox->value());
- ui.SequenceRotationsPerBlendSpinBox->setMaximum(state ? ui.SequenceRotationsPerBlendMaxSpinBox->value() : std::numeric_limits::max());
- ui.SequenceRotationsPerBlendMaxSpinBox->setEnabled(state);
-}
-
///
/// Constrain all min/max spinboxes.
///
-void Fractorium::OnSequenceStaggerSpinBoxChanged(double d) { if (ui.SequenceRandomizeStaggerCheckBox->isChecked()) ui.SequenceRandomStaggerMaxSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceRandomStaggerMaxSpinBoxChanged(double d) { if (ui.SequenceRandomStaggerMaxSpinBox->hasFocus()) ui.SequenceStaggerSpinBox->setMaximum(d); }
-void Fractorium::OnSequenceStartFlameSpinBoxChanged(int d) { ui.SequenceStopFlameSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceStopFlameSpinBoxChanged(int d) { if (ui.SequenceStopFlameSpinBox->hasFocus()) ui.SequenceStartFlameSpinBox->setMaximum(d); }
-void Fractorium::OnSequenceFramesPerRotSpinBoxChanged(int d) { if (ui.SequenceRandomizeFramesPerRotCheckBox->isChecked()) ui.SequenceRandomFramesPerRotMaxSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceRandomFramesPerRotMaxSpinBoxChanged(int d) { if (ui.SequenceRandomFramesPerRotMaxSpinBox->hasFocus()) ui.SequenceFramesPerRotSpinBox->setMaximum(d); }
-void Fractorium::OnSequenceRotationsSpinBoxChanged(double d) { if (ui.SequenceRandomizeRotationsCheckBox->isChecked()) ui.SequenceRandomRotationsMaxSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceRandomRotationsMaxSpinBoxChanged(double d) { if (ui.SequenceRandomRotationsMaxSpinBox->hasFocus()) ui.SequenceRotationsSpinBox->setMaximum(d); }
-void Fractorium::OnSequenceBlendFramesSpinBoxChanged(int d) { if (ui.SequenceRandomizeBlendFramesCheckBox->isChecked()) ui.SequenceRandomBlendMaxFramesSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceRandomBlendMaxFramesSpinBoxChanged(int d) { if (ui.SequenceRandomBlendMaxFramesSpinBox->hasFocus()) ui.SequenceBlendFramesSpinBox->setMaximum(d); }
-void Fractorium::OnSequenceRandomRotationsPerBlendSpinBoxChanged(int d) { if (ui.SequenceRandomizeRotationsPerBlendCheckBox->isChecked()) ui.SequenceRotationsPerBlendMaxSpinBox->setMinimum(d); }
-void Fractorium::OnSequenceRandomRotationsPerBlendMaxSpinBoxChanged(int d) { if (ui.SequenceRotationsPerBlendMaxSpinBox->hasFocus()) ui.SequenceRotationsPerBlendSpinBox->setMaximum(d); }
+void Fractorium::OnSequenceStartFlameSpinBoxChanged(int d) { ui.SequenceStopFlameSpinBox->setMinimum(d); }
+void Fractorium::OnSequenceStopFlameSpinBoxChanged(int d) { if (ui.SequenceStopFlameSpinBox->hasFocus()) ui.SequenceStartFlameSpinBox->setMaximum(d); }
///
/// Save all sequence settings to match the values in the controls.
///
void Fractorium::SyncSequenceSettings()
{
- m_Settings->Stagger(ui.SequenceStaggerSpinBox->value());
- m_Settings->StaggerMax(ui.SequenceRandomStaggerMaxSpinBox->value());
- m_Settings->FramesPerRot(ui.SequenceFramesPerRotSpinBox->value());
- m_Settings->FramesPerRotMax(ui.SequenceRandomFramesPerRotMaxSpinBox->value());
- m_Settings->Rotations(ui.SequenceRotationsSpinBox->value());
- m_Settings->RotationsMax(ui.SequenceRandomRotationsMaxSpinBox->value());
- m_Settings->BlendFrames(ui.SequenceBlendFramesSpinBox->value());
- m_Settings->BlendFramesMax(ui.SequenceRandomBlendMaxFramesSpinBox->value());
- m_Settings->RotationsPerBlend(ui.SequenceRotationsPerBlendSpinBox->value());
- m_Settings->RotationsPerBlendMax(ui.SequenceRotationsPerBlendMaxSpinBox->value());
- m_Settings->Linear(ui.SequenceLinearCheckBox->isChecked());
m_Settings->AnimationFps(ui.SequenceAnimationFpsSpinBox->value());
}
diff --git a/Source/Fractorium/FractoriumMenus.cpp b/Source/Fractorium/FractoriumMenus.cpp
index 64e2402..b58096c 100644
--- a/Source/Fractorium/FractoriumMenus.cpp
+++ b/Source/Fractorium/FractoriumMenus.cpp
@@ -1008,22 +1008,16 @@ void FractoriumEmberController::ClearFlame()
{
Update([&]()
{
- while (m_Ember.TotalXformCount() > 1)
- m_Ember.DeleteTotalXform(m_Ember.TotalXformCount() - 1);
-
- if (m_Ember.XformCount() == 1)
- {
- if (auto xform = m_Ember.GetXform(0))
- {
- xform->Clear();
- xform->AddVariation(m_VariationList->GetVariationCopy(eVariationId::VAR_LINEAR));
- xform->ParentEmber(&m_Ember);
- }
- }
-
- m_Ember.m_Curves.Init();
- FillXforms();
+ Xform newXform;
+ newXform.m_Weight = 0.25;
+ newXform.m_ColorX = m_Rand.Frand01();
+ newXform.AddVariation(m_VariationList->GetVariationCopy(eVariationId::VAR_LINEAR));
+ m_Ember.Clear();
+ m_Ember.AddXform(newXform);
+ FillXforms();//Must do this first because the palette setup in FillParamTablesAndPalette() uses the xforms combo.
+ FillParamTablesAndPalette();
FillCurvesControl();
+ FillSummary();
});
}
diff --git a/Source/Fractorium/FractoriumParams.cpp b/Source/Fractorium/FractoriumParams.cpp
index 9057086..a199944 100644
--- a/Source/Fractorium/FractoriumParams.cpp
+++ b/Source/Fractorium/FractoriumParams.cpp
@@ -9,7 +9,8 @@ void Fractorium::InitParamsUI()
int row = 0;
int spinHeight = 20;
double dmax = numeric_limits::max();
- vector comboVals;
+ int imax = numeric_limits::max();
+ vector comboVals, llComboVals;
QTableWidget* table = ui.ColorTable;
//Because QTableWidget does not allow for a single title bar/header
//at the top of a multi-column table, the workaround hack is to just
@@ -118,15 +119,29 @@ void Fractorium::InitParamsUI()
comboVals.clear();
comboVals.push_back("Linear");
comboVals.push_back("Log");
- SetupCombo( table, this, row, 1, m_AffineInterpTypeCombo, comboVals, SIGNAL(currentIndexChanged(int)), SLOT(OnAffineInterpTypeComboCurrentIndexChanged(int)));
+ llComboVals = comboVals;
+ SetupCombo(table, this, row, 1, m_AffineInterpTypeCombo, comboVals, SIGNAL(currentIndexChanged(int)), SLOT(OnAffineInterpTypeComboCurrentIndexChanged(int)));
m_AffineInterpTypeCombo->SetCurrentIndexStealth(static_cast(eAffineInterp::AFFINE_INTERP_LOG));
- SetupSpinner(table, this, row, 1, m_TemporalFilterWidthSpin, spinHeight, 1, 10, 1, SIGNAL(valueChanged(double)), SLOT(OnTemporalFilterWidthChanged(double)), true, 1, 1, 1);
+ SetupSpinner(table, this, row, 1, m_RotationsSpin, spinHeight, 0, dmax, 1, SIGNAL(valueChanged(double)), SLOT(OnRotationsChanged(double)), true, 1.0, 1.0, 0);
+ SetupSpinner(table, this, row, 1, m_SecondsPerRotationSpin, spinHeight, 0, dmax, 1, SIGNAL(valueChanged(double)), SLOT(OnSecondsPerRotationChanged(double)), true, 1.0, 1.0, 0);
+ comboVals.clear();
+ comboVals.push_back("Cw");
+ comboVals.push_back("Ccw");
+ SetupCombo(table, this, row, 1, m_RotateXformsDirCombo, comboVals, SIGNAL(currentIndexChanged(int)), SLOT(OnRotateXformsDirComboCurrentIndexChanged(int)));
+ m_RotateXformsDirCombo->SetCurrentIndexStealth(0);
+ SetupSpinner(table, this, row, 1, m_BlendSecondsSpin, spinHeight, 0, dmax, 1, SIGNAL(valueChanged(double)), SLOT(OnBlendSecondsChanged(double)), true, 1.0, 1.0, 0);
+ SetupSpinner( table, this, row, 1, m_RotationsPerBlendSpin, spinHeight, 0, imax, 1, SIGNAL(valueChanged(int)), SLOT(OnRotationsPerBlendChanged(int)), true, 0, 1, 0);
+ SetupCombo(table, this, row, 1, m_BlendXformsRotateDirCombo, comboVals, SIGNAL(currentIndexChanged(int)), SLOT(OnBlendXformsRotateDirComboCurrentIndexChanged(int)));
+ m_BlendXformsRotateDirCombo->SetCurrentIndexStealth(0);
+ SetupCombo(table, this, row, 1, m_BlendInterpTypeCombo, llComboVals, SIGNAL(currentIndexChanged(int)), SLOT(OnBlendInterpTypeComboCurrentIndexChanged(int)));
+ m_BlendInterpTypeCombo->SetCurrentIndexStealth(1);
+ SetupSpinner(table, this, row, 1, m_StaggerSpin, spinHeight, 0, 1, 0.1, SIGNAL(valueChanged(double)), SLOT(OnStaggerChanged(double)), true, 0, 1.0, 0);
+ SetupSpinner(table, this, row, 1, m_TemporalFilterWidthSpin, spinHeight, 1, 10, 1, SIGNAL(valueChanged(double)), SLOT(OnTemporalFilterWidthChanged(double)), true, 1, 1, 1);
comboVals = TemporalFilterCreator::FilterTypes();
- SetupCombo( table, this, row, 1, m_TemporalFilterTypeCombo, comboVals, SIGNAL(currentTextChanged(const QString&)), SLOT(OnTemporalFilterTypeComboCurrentIndexChanged(const QString&)));
+ SetupCombo(table, this, row, 1, m_TemporalFilterTypeCombo, comboVals, SIGNAL(currentTextChanged(const QString&)), SLOT(OnTemporalFilterTypeComboCurrentIndexChanged(const QString&)));
m_TemporalFilterTypeCombo->SetCurrentIndexStealth(static_cast(eTemporalFilterType::BOX_TEMPORAL_FILTER));
table->horizontalHeader()->setSectionResizeMode(0, QHeaderView::ResizeToContents);
- SetupSpinner(table, this, row, 1, m_TemporalFilterExpSpin, spinHeight, 0, 5, 0.1, SIGNAL(valueChanged(double)), SLOT(OnExpChanged(double)), true, 1, 1, 0);
- //
+ SetupSpinner(table, this, row, 1, m_TemporalFilterExpSpin, spinHeight, 0, 5, 0.1, SIGNAL(valueChanged(double)), SLOT(OnExpChanged(double)), true, 1, 1, 0);
AddSizePreset("HD", 1920, 1080);
AddSizePreset("QHD", 2560, 1440);
AddSizePreset("4K UHD", 3840, 2160);
@@ -198,7 +213,7 @@ bool Fractorium::ApplyAll()
///
/// Set the brightness to be used for calculating K1 and K2 for filtering and final accum.
-/// Called when brightness spinner is changed.
+/// Called when the brightness spinner is changed.
/// Resets the rendering process to the filtering stage.
///
/// The brightness
@@ -210,12 +225,11 @@ void FractoriumEmberController::BrightnessChanged(double d)
ember.m_Brightness = d;
}, true, eProcessAction::FILTER_AND_ACCUM, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnBrightnessChanged(double d) { m_Controller->BrightnessChanged(d); }
///
/// Set the gamma to be used for final accum.
-/// Called when gamma spinner is changed.
+/// Called when the gamma spinner is changed.
/// Resets the rendering process if temporal samples is greater than 1,
/// else if early clip is true, filter and accum, else final accum only.
///
@@ -231,7 +245,7 @@ void Fractorium::OnGammaChanged(double d) { m_Controller->GammaChanged(d); }
///
/// Set the gamma threshold to be used for final accum.
-/// Called when gamma threshold spinner is changed.
+/// Called when the gamma threshold spinner is changed.
/// Resets the rendering process to the final accumulation stage.
///
/// The gamma threshold
@@ -246,7 +260,7 @@ void Fractorium::OnGammaThresholdChanged(double d) { m_Controller->GammaThreshol
///
/// Set the vibrancy to be used for final accum.
-/// Called when vibrancy spinner is changed.
+/// Called when the vibrancy spinner is changed.
/// Resets the rendering process to the final accumulation stage if temporal samples is 1, else full reset.
///
/// The vibrancy
@@ -261,7 +275,7 @@ void Fractorium::OnVibrancyChanged(double d) { m_Controller->VibrancyChanged(d);
///
/// Set the highlight power to be used for final accum.
-/// Called when highlight power spinner is changed.
+/// Called when the highlight power spinner is changed.
/// Resets the rendering process to the final accumulation stage.
///
/// The highlight power
@@ -276,7 +290,7 @@ void Fractorium::OnHighlightPowerChanged(double d) { m_Controller->HighlightPowe
///
/// Set the k2 brightness value to be used for final accum.
-/// Called when k2 is changed.
+/// Called when the k2 spinner is changed.
/// Resets the rendering process to the final accumulation stage.
///
/// The k2 value
@@ -287,12 +301,11 @@ template void FractoriumEmberController::K2Changed(double d)
ember.m_K2 = d;
}, true, eProcessAction::FILTER_AND_ACCUM, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnK2Changed(double d) { m_Controller->K2Changed(d); }
///
/// Show the color selection dialog.
-/// Called when background color button is clicked.
+/// Called when the background color button is clicked.
///
/// Ignored
void Fractorium::OnBackgroundColorButtonClicked(bool checked)
@@ -326,12 +339,11 @@ void FractoriumEmberController::BackgroundChanged(const QColor& color)
ember.m_Background.b = color.blue() / 255.0;
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnColorSelected(const QColor& color) { m_Controller->BackgroundChanged(color); }
///
/// Set the palette index interpolation mode.
-/// Called when palette mode combo box index is changed.
+/// Called when the palette mode combo box index is changed.
/// Resets the rendering process.
///
/// The index of the palette mode combo box
@@ -361,7 +373,6 @@ template void FractoriumEmberController::WidthChanged(uint i)
ember.m_FinalRasW = i;
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnWidthChanged(int i) { m_Controller->WidthChanged(i); }
///
@@ -377,7 +388,6 @@ template void FractoriumEmberController::HeightChanged(uint i)
ember.m_FinalRasH = i;
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnHeightChanged(int i) { m_Controller->HeightChanged(i); }
///
@@ -476,7 +486,7 @@ void Fractorium::OnCenterYChanged(double d) { m_Controller->CenterYChanged(d); }
/// Set the scale (pixels per unit) value of the image.
/// Note this will not increase the number of iters ran, but will degrade quality.
/// To preserve quality, but exponentially increase iters, use zoom.
-/// Called when scale spinner is changed.
+/// Called when the scale spinner is changed.
/// Resets the rendering process.
///
/// The scale value
@@ -493,7 +503,7 @@ void Fractorium::OnScaleChanged(double d) { m_Controller->ScaleChanged(d); }
/// Set the zoom value of the image.
/// Note this will increase the number of iters ran exponentially.
/// To zoom in without increasing iters, but sacrifice quality, use scale.
-/// Called when zoom spinner is changed.
+/// Called when the zoom spinner is changed.
/// Resets the rendering process.
///
/// The zoom value
@@ -508,10 +518,10 @@ void Fractorium::OnZoomChanged(double d) { m_Controller->ZoomChanged(d); }
///
/// Set the angular rotation of the image.
-/// Called when rotate spinner is changed.
+/// Called when the rotate spinner is changed.
/// Resets the rendering process.
///
-/// The rotation in angles
+/// The rotation in degrees
template void FractoriumEmberController::RotateChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -519,11 +529,14 @@ template void FractoriumEmberController::RotateChanged(double d)
ember.m_Rotate = d;
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
}
-void Fractorium::OnRotateChanged(double d)
-{
- m_Controller->RotateChanged(d); // d is ever between -180 and +180
-}
+void Fractorium::OnRotateChanged(double d) { m_Controller->RotateChanged(d); }
+///
+/// Set the 3D z position of the image.
+/// Called when the 3D zpos spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D zpos in world space units
template void FractoriumEmberController::ZPosChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -533,6 +546,12 @@ template void FractoriumEmberController::ZPosChanged(double d)
}
void Fractorium::OnZPosChanged(double d) { m_Controller->ZPosChanged(d); }
+///
+/// Set the 3D persepctive of the image.
+/// Called when the 3D persepctive spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D perspective in world space units
template void FractoriumEmberController::PerspectiveChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -542,6 +561,12 @@ template void FractoriumEmberController::PerspectiveChanged(doub
}
void Fractorium::OnPerspectiveChanged(double d) { m_Controller->PerspectiveChanged(d); }
+///
+/// Set the 3D pitch of the image.
+/// Called when the 3D pitch spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D pitch in degrees
template void FractoriumEmberController::PitchChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -551,6 +576,12 @@ template void FractoriumEmberController::PitchChanged(double d)
}
void Fractorium::OnPitchChanged(double d) { m_Controller->PitchChanged(d); }
+///
+/// Set the 3D yaw of the image.
+/// Called when the 3D yaw spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D yaw in degrees
template void FractoriumEmberController::YawChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -560,6 +591,12 @@ template void FractoriumEmberController::YawChanged(double d)
}
void Fractorium::OnYawChanged(double d) { m_Controller->YawChanged(d); }
+///
+/// Set the 3D depth blur of the image.
+/// Called when the 3D depth blur spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D depth blur
template void FractoriumEmberController::DepthBlurChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -569,6 +606,12 @@ template void FractoriumEmberController::DepthBlurChanged(double
}
void Fractorium::OnDepthBlurChanged(double d) { m_Controller->DepthBlurChanged(d); }
+///
+/// Set the 3D blur curve used to calculate the 3D depth blur of the image.
+/// Called when the blur curve spinner is changed.
+/// Resets the rendering process.
+///
+/// The 3D blur curve
template void FractoriumEmberController::BlurCurveChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
@@ -595,7 +638,6 @@ template void FractoriumEmberController::SpatialFilterWidthChang
ember.m_SpatialFilterRadius = d;
}, true, m_Renderer->EarlyClip() ? eProcessAction::FILTER_AND_ACCUM : eProcessAction::ACCUM_ONLY, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnSpatialFilterWidthChanged(double d) { m_Controller->SpatialFilterWidthChanged(d); }
///
@@ -611,7 +653,6 @@ template void FractoriumEmberController::SpatialFilterTypeChange
ember.m_SpatialFilterType = SpatialFilterCreator::FromString(text.toStdString());
}, true, m_Renderer->EarlyClip() ? eProcessAction::FILTER_AND_ACCUM : eProcessAction::ACCUM_ONLY, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnSpatialFilterTypeComboCurrentIndexChanged(const QString& text) { m_Controller->SpatialFilterTypeChanged(text); }
///
@@ -672,7 +713,6 @@ template void FractoriumEmberController::DEFilterCurveWidthChang
ember.m_CurveDE = d;
}, true, eProcessAction::FILTER_AND_ACCUM, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnDEFilterCurveWidthChanged(double d) { m_Controller->DEFilterCurveWidthChanged(d); }
///
@@ -707,7 +747,6 @@ template void FractoriumEmberController::RandRangeChanged(double
ember.m_RandPointRange = d;
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnRandRangeChanged(double d) { m_Controller->RandRangeChanged(d); }
///
@@ -758,6 +797,35 @@ template void FractoriumEmberController::SupersampleChanged(int
}
void Fractorium::OnSupersampleChanged(int d) { m_Controller->SupersampleChanged(d); }
+///
+/// Set the interpolation type.
+/// Does not reset anything because this is only used for animation.
+/// Called when the interp type combo box index is changed.
+///
+/// The index
+template
+void FractoriumEmberController::InterpTypeChanged(int i)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ eInterp interp;
+
+ if (i == 0)
+ interp = eInterp::EMBER_INTERP_LINEAR;
+ else if (i == 1)
+ interp = eInterp::EMBER_INTERP_SMOOTH;
+ else
+ interp = eInterp::EMBER_INTERP_LINEAR;
+
+ ember.m_Interp = interp;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_Interp = interp;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnInterpTypeComboCurrentIndexChanged(int index) { m_Controller->InterpTypeChanged(index); }
+
///
/// Set the affine interpolation type.
/// Does not reset anything because this is only used for animation.
@@ -786,38 +854,159 @@ void FractoriumEmberController::AffineInterpTypeChanged(int i)
m_EmberFilePointer->m_AffineInterp = interp;
}, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnAffineInterpTypeComboCurrentIndexChanged(int index) { m_Controller->AffineInterpTypeChanged(index); }
///
-/// Set the interpolation type.
+/// Set the rotations to be used in animation.
+/// Called when the rotations spinner is changed.
/// Does not reset anything because this is only used for animation.
-/// Called when the interp type combo box index is changed.
///
-/// The index
-template
-void FractoriumEmberController::InterpTypeChanged(int i)
+/// The number of rotations to perform in a loop
+template void FractoriumEmberController::RotationsChanged(double d)
{
UpdateAll([&](Ember& ember, bool isMain)
{
- eInterp interp;
-
- if (i == 0)
- interp = eInterp::EMBER_INTERP_LINEAR;
- else if (i == 1)
- interp = eInterp::EMBER_INTERP_SMOOTH;
- else
- interp = eInterp::EMBER_INTERP_LINEAR;
-
- ember.m_Interp = interp;
+ ember.m_Rotations = d;
if (!m_Fractorium->ApplyAll())
if (m_EmberFilePointer)
- m_EmberFilePointer->m_Interp = interp;
+ m_EmberFilePointer->m_Rotations = d;
}, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
}
+void Fractorium::OnRotationsChanged(double d) { m_Controller->RotationsChanged(d); }
-void Fractorium::OnInterpTypeComboCurrentIndexChanged(int index) { m_Controller->InterpTypeChanged(index); }
+///
+/// Set the seconds each loop rotation takes in animation.
+/// Called when the seconds per rotation spinner is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The number of seconds each loop rotation should take
+template void FractoriumEmberController::SecondsPerRotationChanged(double d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_SecondsPerRotation = d;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_SecondsPerRotation = d;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnSecondsPerRotationChanged(double d) { m_Controller->SecondsPerRotationChanged(d); }
+
+///
+/// Set the direction loop rotations rotate in animation.
+/// Called when the xforms rotation direction combobox index is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The index
+template void FractoriumEmberController::RotateXformsDirChanged(uint d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_RotateXformsCw = d == 0;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_RotateXformsCw = ember.m_RotateXformsCw;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnRotateXformsDirComboCurrentIndexChanged(int i) { m_Controller->RotateXformsDirChanged(i); }
+
+///
+/// Set the seconds each blend takes in animation.
+/// Called when the blend seconds spinner is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The number of seconds each blend should take
+template void FractoriumEmberController::BlendSecondsChanged(double d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_BlendSeconds = d;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_BlendSeconds = ember.m_BlendSeconds;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnBlendSecondsChanged(double d) { m_Controller->BlendSecondsChanged(d); }
+
+///
+/// Set the rotations each blend performs in animation.
+/// Called when the rotations per blend spinner is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The number of rotations each blend should perform
+template void FractoriumEmberController::RotationsPerBlendChanged(uint d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_RotationsPerBlend = d;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_RotationsPerBlend = d;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnRotationsPerBlendChanged(int d) { m_Controller->RotationsPerBlendChanged(d); }
+
+///
+/// Set the direction blend rotations rotate in animation.
+/// Called when the blend xforms rotation direction combobox index is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The index
+template void FractoriumEmberController::BlendXformsRotateDirChanged(uint d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_BlendRotateXformsCw = d == 0;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_BlendRotateXformsCw = ember.m_BlendRotateXformsCw;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnBlendXformsRotateDirComboCurrentIndexChanged(int i) { m_Controller->BlendXformsRotateDirChanged(i); }
+
+///
+/// Set the blend interpolation type in animation.
+/// Called when the blend interpolation type combobox index is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The index
+template void FractoriumEmberController::BlendInterpTypeChanged(uint d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_Linear = d == 0;
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_Linear = ember.m_Linear;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnBlendInterpTypeComboCurrentIndexChanged(int i) { m_Controller->BlendInterpTypeChanged(i); }
+
+///
+/// Set the stagger amount in animation.
+/// Called when the stagger spinner is changed.
+/// Does not reset anything because this is only used for animation.
+///
+/// The amount to stagger the blending of xforms
+template void FractoriumEmberController::StaggerChanged(double d)
+{
+ UpdateAll([&](Ember& ember, bool isMain)
+ {
+ ember.m_Stagger = Clamp(d, 0.0, 1.0);
+
+ if (!m_Fractorium->ApplyAll())
+ if (m_EmberFilePointer)
+ m_EmberFilePointer->m_Stagger = ember.m_Stagger;
+ }, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
+}
+void Fractorium::OnStaggerChanged(double d) { m_Controller->StaggerChanged(d); }
///
/// Set the temporal filter width to be used with animation.
@@ -878,7 +1067,6 @@ void FractoriumEmberController::ExpChanged(double d)
m_EmberFilePointer->m_TemporalFilterExp = d;
}, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
}
-
void Fractorium::OnExpChanged(double d) { m_Controller->ExpChanged(d); }
///
@@ -932,9 +1120,6 @@ void FractoriumEmberController::FillParamTablesAndPalette()
m_Fractorium->m_BlurCurveSpin->SetValueStealth(m_Ember.m_BlurCurve);
m_Fractorium->m_SpatialFilterWidthSpin->SetValueStealth(m_Ember.m_SpatialFilterRadius);//Filter.
m_Fractorium->m_SpatialFilterTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_SpatialFilterType));
- m_Fractorium->m_TemporalFilterWidthSpin->SetValueStealth(m_Ember.m_TemporalFilterWidth);
- m_Fractorium->m_TemporalFilterTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_TemporalFilterType));
- m_Fractorium->m_TemporalFilterExpSpin->SetValueStealth(m_Ember.m_TemporalFilterExp);
m_Fractorium->m_DEFilterMinRadiusSpin->SetValueStealth(m_Ember.m_MinRadDE);
m_Fractorium->m_DEFilterMaxRadiusSpin->SetValueStealth(m_Ember.m_MaxRadDE);
m_Fractorium->m_DECurveSpin->SetValueStealth(m_Ember.m_CurveDE);
@@ -943,8 +1128,19 @@ void FractoriumEmberController::FillParamTablesAndPalette()
m_Fractorium->m_RandRangeSpin->SetValueStealth(m_Ember.m_RandPointRange);
m_Fractorium->m_QualitySpin->SetValueStealth(m_Ember.m_Quality);
m_Fractorium->m_SupersampleSpin->SetValueStealth(m_Ember.m_Supersample);
- m_Fractorium->m_AffineInterpTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_AffineInterp));
m_Fractorium->m_InterpTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_Interp));
+ m_Fractorium->m_AffineInterpTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_AffineInterp));
+ m_Fractorium->m_RotationsSpin->SetValueStealth(m_Ember.m_Rotations);
+ m_Fractorium->m_SecondsPerRotationSpin->SetValueStealth(m_Ember.m_SecondsPerRotation);
+ m_Fractorium->m_RotateXformsDirCombo->SetCurrentIndexStealth(m_Ember.m_RotateXformsCw ? 0 : 1);
+ m_Fractorium->m_BlendSecondsSpin->SetValueStealth(m_Ember.m_BlendSeconds);
+ m_Fractorium->m_RotationsPerBlendSpin->SetValueStealth(m_Ember.m_RotationsPerBlend);
+ m_Fractorium->m_BlendXformsRotateDirCombo->SetCurrentIndexStealth(m_Ember.m_BlendRotateXformsCw ? 0 : 1);
+ m_Fractorium->m_BlendInterpTypeCombo->SetCurrentIndexStealth(m_Ember.m_Linear ? 0 : 1);
+ m_Fractorium->m_StaggerSpin->SetValueStealth(m_Ember.m_Stagger);
+ m_Fractorium->m_TemporalFilterWidthSpin->SetValueStealth(m_Ember.m_TemporalFilterWidth);
+ m_Fractorium->m_TemporalFilterTypeCombo->SetCurrentIndexStealth(static_cast(m_Ember.m_TemporalFilterType));
+ m_Fractorium->m_TemporalFilterExpSpin->SetValueStealth(m_Ember.m_TemporalFilterExp);
auto temp = m_Ember.m_Palette.m_Filename;
if (temp.get())
@@ -986,9 +1182,6 @@ void FractoriumEmberController::ParamsToEmberPrivate(Ember& ember, bool im
if (imageParamsOnly)
return;
- ember.m_TemporalFilterWidth = m_Fractorium->m_TemporalFilterWidthSpin->value();
- ember.m_TemporalFilterType = static_cast(m_Fractorium->m_TemporalFilterTypeCombo->currentIndex());
- ember.m_TemporalFilterExp = m_Fractorium->m_TemporalFilterExpSpin->value();
const auto color = m_Fractorium->ui.ColorTable->item(5, 1)->background();
ember.m_Background.r = color.color().red() / 255.0;
ember.m_Background.g = color.color().green() / 255.0;
@@ -1012,8 +1205,19 @@ void FractoriumEmberController::ParamsToEmberPrivate(Ember& ember, bool im
ember.m_RandPointRange = m_Fractorium->m_RandRangeSpin->value();
ember.m_Quality = m_Fractorium->m_QualitySpin->value();
ember.m_Supersample = m_Fractorium->m_SupersampleSpin->value();
- ember.m_AffineInterp = static_cast(m_Fractorium->m_AffineInterpTypeCombo->currentIndex());
ember.m_Interp = static_cast(m_Fractorium->m_InterpTypeCombo->currentIndex());
+ ember.m_AffineInterp = static_cast(m_Fractorium->m_AffineInterpTypeCombo->currentIndex());
+ ember.m_Rotations = m_Fractorium->m_RotationsSpin->value();
+ ember.m_SecondsPerRotation = m_Fractorium->m_SecondsPerRotationSpin->value();
+ ember.m_RotateXformsCw = m_Fractorium->m_RotateXformsDirCombo->currentIndex() == 0;
+ ember.m_BlendSeconds = m_Fractorium->m_BlendSecondsSpin->value();
+ ember.m_RotationsPerBlend = m_Fractorium->m_RotationsPerBlendSpin->value();
+ ember.m_BlendRotateXformsCw = m_Fractorium->m_BlendXformsRotateDirCombo->currentIndex() == 0;
+ ember.m_Linear = m_Fractorium->m_BlendInterpTypeCombo->currentIndex() == 0;
+ ember.m_Stagger = m_Fractorium->m_StaggerSpin->value();
+ ember.m_TemporalFilterWidth = m_Fractorium->m_TemporalFilterWidthSpin->value();
+ ember.m_TemporalFilterType = static_cast(m_Fractorium->m_TemporalFilterTypeCombo->currentIndex());
+ ember.m_TemporalFilterExp = m_Fractorium->m_TemporalFilterExpSpin->value();
ember.SyncSize();
}
diff --git a/Source/Fractorium/FractoriumXforms.cpp b/Source/Fractorium/FractoriumXforms.cpp
index 4dd15c8..1093061 100644
--- a/Source/Fractorium/FractoriumXforms.cpp
+++ b/Source/Fractorium/FractoriumXforms.cpp
@@ -8,14 +8,15 @@ void Fractorium::InitXformsUI()
{
const int spinHeight = 20;
auto row = 0;
- connect(ui.AddXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.AddLinkedXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddLinkedXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.DuplicateXformButton, SIGNAL(clicked(bool)), this, SLOT(OnDuplicateXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.ClearXformButton, SIGNAL(clicked(bool)), this, SLOT(OnClearXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.DeleteXformButton, SIGNAL(clicked(bool)), this, SLOT(OnDeleteXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.AddFinalXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddFinalXformButtonClicked(bool)), Qt::QueuedConnection);
- connect(ui.CurrentXformCombo, SIGNAL(currentIndexChanged(int)), this, SLOT(OnCurrentXformComboChanged(int)), Qt::QueuedConnection);
- connect(ui.AnimateXformCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnXformAnimateCheckBoxStateChanged(int)), Qt::QueuedConnection);
+ connect(ui.AddXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.AddLinkedXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddLinkedXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.DuplicateXformButton, SIGNAL(clicked(bool)), this, SLOT(OnDuplicateXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.ClearXformButton, SIGNAL(clicked(bool)), this, SLOT(OnClearXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.DeleteXformButton, SIGNAL(clicked(bool)), this, SLOT(OnDeleteXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.AddFinalXformButton, SIGNAL(clicked(bool)), this, SLOT(OnAddFinalXformButtonClicked(bool)), Qt::QueuedConnection);
+ connect(ui.CurrentXformCombo, SIGNAL(currentIndexChanged(int)), this, SLOT(OnCurrentXformComboChanged(int)), Qt::QueuedConnection);
+ connect(ui.AnimateXformLocalRotationCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnXformAnimateLocalRotationCheckBoxStateChanged(int)), Qt::QueuedConnection);
+ connect(ui.AnimateXformOriginRotationCheckBox, SIGNAL(stateChanged(int)), this, SLOT(OnXformAnimateOriginRotationCheckBoxStateChanged(int)), Qt::QueuedConnection);
SetFixedTableHeader(ui.XformWeightNameTable->horizontalHeader(), QHeaderView::ResizeToContents);
//Use SetupSpinner() just to create the spinner, but use col of -1 to prevent it from being added to the table.
SetupSpinner(ui.XformWeightNameTable, this, row, -1, m_XformWeightSpin, spinHeight, 0, 1000, 0.05, SIGNAL(valueChanged(double)), SLOT(OnXformWeightChanged(double)), false, 0, 1, 0);
@@ -438,6 +439,7 @@ void FractoriumEmberController::AddFinalXform()
{
auto& final = m_Ember.m_CachedFinal;
final.m_Animate = 0;
+ final.m_AnimateOrigin = 0;
if (final.Empty())
final.AddVariation(m_VariationList->GetVariationCopy(eVariationId::VAR_LINEAR));//Just a placeholder so other parts of the code don't see it as being empty.
@@ -451,7 +453,7 @@ void FractoriumEmberController::AddFinalXform()
void Fractorium::OnAddFinalXformButtonClicked(bool checked) { m_Controller->AddFinalXform(); }
///
/// Set the weight of the selected xforms.
-/// Called when weight spinner changes.
+/// Called when the weight spinner changes.
/// Resets the rendering process.
///
/// The weight
@@ -518,8 +520,9 @@ void Fractorium::OnXformNameChanged(const QString& s)
/// Called when the user toggles the animate xform checkbox.
///
/// 1 for checked, else false
+/// true to rotate around the local center, else rotate around the origin
template
-void FractoriumEmberController::XformAnimateChanged(int state)
+void FractoriumEmberController::XformAnimateChangedHelper(int state, bool local)
{
T animate = state > 0 ? 1 : 0;
UpdateXform([&](Xform* xform, size_t xfindex, size_t selIndex)
@@ -532,26 +535,40 @@ void FractoriumEmberController::XformAnimateChanged(int state)
if (ember.UseFinalXform())
{
auto xform = ember.NonConstFinalXform();
- xform->m_Animate = animate;
+
+ if (local)
+ xform->m_Animate = animate;
+ else
+ xform->m_AnimateOrigin = animate;
}
if (!m_Fractorium->ApplyAll())
if (m_EmberFilePointer && m_EmberFilePointer->UseFinalXform())
- m_EmberFilePointer->NonConstFinalXform()->m_Animate = animate;
+ if (local)
+ m_EmberFilePointer->NonConstFinalXform()->m_Animate = animate;
+ else
+ m_EmberFilePointer->NonConstFinalXform()->m_AnimateOrigin = animate;
}
else//Current was not final, so apply to other embers which have a non-final xform at this index.
{
if (auto xform = ember.GetXform(xfindex))
- xform->m_Animate = animate;
+ if (local)
+ xform->m_Animate = animate;
+ else
+ xform->m_AnimateOrigin = animate;
if (!m_Fractorium->ApplyAll() && m_EmberFilePointer)
if (auto xform = m_EmberFilePointer->GetXform(xfindex))
- xform->m_Animate = animate;
+ if (local)
+ xform->m_Animate = animate;
+ else
+ xform->m_AnimateOrigin = animate;
}
}, false, eProcessAction::NOTHING, m_Fractorium->ApplyAll());
}, eXformUpdate::UPDATE_SELECTED, false);
}
-void Fractorium::OnXformAnimateCheckBoxStateChanged(int state) { m_Controller->XformAnimateChanged(state); }
+void Fractorium::OnXformAnimateLocalRotationCheckBoxStateChanged(int state) { m_Controller->XformAnimateChangedHelper(state, true); }
+void Fractorium::OnXformAnimateOriginRotationCheckBoxStateChanged(int state) { m_Controller->XformAnimateChangedHelper(state, false); }
///
/// Fill all GUI widgets with values from the passed in xform.
@@ -562,9 +579,12 @@ void FractoriumEmberController::FillWithXform(Xform* xform)
{
m_Fractorium->m_XformWeightSpin->SetValueStealth(xform->m_Weight);
SetNormalizedWeightText(xform);
- m_Fractorium->ui.AnimateXformCheckBox->blockSignals(true);
- m_Fractorium->ui.AnimateXformCheckBox->setChecked(xform->m_Animate > 0 ? true : false);
- m_Fractorium->ui.AnimateXformCheckBox->blockSignals(false);
+ m_Fractorium->ui.AnimateXformLocalRotationCheckBox->blockSignals(true);
+ m_Fractorium->ui.AnimateXformLocalRotationCheckBox->setChecked(xform->m_Animate > 0 ? true : false);
+ m_Fractorium->ui.AnimateXformLocalRotationCheckBox->blockSignals(false);
+ m_Fractorium->ui.AnimateXformOriginRotationCheckBox->blockSignals(true);
+ m_Fractorium->ui.AnimateXformOriginRotationCheckBox->setChecked(xform->m_AnimateOrigin > 0 ? true : false);
+ m_Fractorium->ui.AnimateXformOriginRotationCheckBox->blockSignals(false);
if (const auto item = m_Fractorium->ui.XformWeightNameTable->item(0, 1))
{
diff --git a/Source/Fractorium/FractoriumXformsColor.cpp b/Source/Fractorium/FractoriumXformsColor.cpp
index b20f09b..fd2352b 100644
--- a/Source/Fractorium/FractoriumXformsColor.cpp
+++ b/Source/Fractorium/FractoriumXformsColor.cpp
@@ -36,7 +36,7 @@ void Fractorium::InitXformsColorUI()
///
/// Set the color index of the selected xforms.
/// Update the color index scrollbar to match.
-/// Called when spinner in the color index cell in the palette ref table is changed.
+/// Called when the spinner in the color index cell in the palette ref table is changed.
/// Optionally resets the rendering process.
///
/// The color index, 0-1/
@@ -83,7 +83,7 @@ void Fractorium::OnXformColorIndexChanged(double d, bool updateRender, bool upda
///
/// Set the color index of the current xform.
/// Will trigger an update which will cause the color index cell in the palette ref table to match.
-/// Called when color index scrollbar is changed.
+/// Called when the color index scrollbar is changed.
/// Resets the rendering process.
///
/// The color index, 0-1.
@@ -148,7 +148,7 @@ void Fractorium::OnToggleColorSpeedsButtonClicked(bool b) { m_Controller->Toggle
///
/// Set the color speed of the selected xforms.
-/// Called when xform color speed spinner is changed.
+/// Called when the xform color speed spinner is changed.
/// Resets the rendering process.
///
/// The color speed, -1-1.
@@ -168,7 +168,7 @@ void Fractorium::OnXformOpacityChanged(double d) { m_Controller->XformOpacityCha
///
/// Set the direct color percentage of the selected xforms.
-/// Called when xform direct color spinner is changed.
+/// Called when the xform direct color spinner is changed.
/// Note this only affects xforms that include a dc_ variation.
/// Resets the rendering process.
///
@@ -184,7 +184,7 @@ void Fractorium::OnXformDirectColorChanged(double d) { m_Controller->XformDirect
/// set to zero while rendering so that only the effect of current xform is visible.
/// This will not permanently alter the opacities of ember, as the temporary opacity values will be applied
/// right before rendering and reset right after.
-/// Called when solo xform check box is checked.
+/// Called when the solo xform check box is checked.
/// Resets the rendering process.
///
/// The state of the checkbox
diff --git a/Source/Fractorium/FractoriumXformsVariations.cpp b/Source/Fractorium/FractoriumXformsVariations.cpp
index d6a4973..5902ae7 100644
--- a/Source/Fractorium/FractoriumXformsVariations.cpp
+++ b/Source/Fractorium/FractoriumXformsVariations.cpp
@@ -407,7 +407,7 @@ QIcon FractoriumEmberController::MakeVariationIcon(const Variation* var)
///
/// Change the sorting to be either by variation ID, or by weight.
/// If sorting by variation ID, repeated clicks will alternate ascending or descending.
-/// Called when user clicks the tree headers.
+/// Called when the user clicks the tree headers.
///
/// Column index of the header clicked. Sort by name if 0, sort by weight if 1.
void Fractorium::OnTreeHeaderSectionClicked(int logicalIndex)
@@ -435,7 +435,7 @@ void Fractorium::OnVariationsFilterLineEditTextChanged(const QString& text)
///
/// Clear the variation name filter, which will display all variations.
-/// Called when clear variations filter button is clicked.
+/// Called when the clear variations filter button is clicked.
///
/// Ignored
void Fractorium::OnVariationsFilterClearButtonClicked(bool checked)