2014-07-08 03:11:14 -04:00
# pragma once
# include "VariationList.h"
2014-09-01 00:25:15 -04:00
# include "Interpolate.h"
2014-07-08 03:11:14 -04:00
/// <summary>
/// Xform class.
/// </summary>
namespace EmberNs
{
/// <summary>
2016-02-18 21:58:24 -05:00
/// Xform and Ember need each other, but each can't include the other.
/// So Ember includes this file, and Ember is declared as a forward declaration here.
2014-07-08 03:11:14 -04:00
/// </summary>
template < typename T > class Ember ;
/// <summary>
/// If both polymorphism and templating are needed, uncomment this, fill it out and derive from it.
/// </summary>
//class EMBER_API XformBase
//{
//};
/// <summary>
/// An xform is a pre affine transform, a list of variations, and an optional final affine transform.
/// This is what gets applied to a point for each iteration.
/// Template argument expected to be float or double.
/// </summary>
template < typename T >
class EMBER_API Xform
{
public :
/// <summary>
2015-03-25 23:47:57 -04:00
/// Default constructor which calls Init() to set default or out of bounds values.
/// When useDefaults is true, Pre and post affine are defaulted to the identity matrix.
2014-07-08 03:11:14 -04:00
/// </summary>
2015-03-25 23:47:57 -04:00
/// <param name="useDefaults">Use reasonable default if true, else use out of bounds values.</param>
Xform ( bool useDefaults = true )
2014-07-08 03:11:14 -04:00
{
2015-03-25 23:47:57 -04:00
Init ( useDefaults ) ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Constructor that takes default arguments. Mostly used for testing.
/// Post affine is defaulted to the identity matrix.
/// </summary>
/// <param name="density">The probability that this xform is chosen</param>
/// <param name="colorX">The color index</param>
/// <param name="colorSpeed">The color speed</param>
/// <param name="opacity">The opacity</param>
/// <param name="a">The a value of the pre affine transform</param>
/// <param name="d">The d value of the pre affine transform</param>
/// <param name="b">The b value of the pre affine transform</param>
/// <param name="e">The e value of the pre affine transform</param>
/// <param name="c">The c value of the pre affine transform</param>
/// <param name="f">The f value of the pre affine transform</param>
/// <param name="pa">The a value of the post affine transform. Default: 1.</param>
/// <param name="pd">The d value of the post affine transform. Default: 0.</param>
/// <param name="pb">The b value of the post affine transform. Default: 0.</param>
/// <param name="pe">The e value of the post affine transform. Default: 1.</param>
/// <param name="pc">The c value of the post affine transform. Default: 0.</param>
/// <param name="pf">The f value of the post affine transform. Default: 0.</param>
Xform ( T weight , T colorX , T colorSpeed , T opacity ,
T a , T d , T b , T e , T c , T f ,
T pa = 1 ,
T pd = 0 ,
T pb = 0 ,
T pe = 1 ,
T pc = 0 ,
T pf = 0 )
2016-04-03 21:55:12 -04:00
: Xform ( )
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
{
m_Weight = weight ;
m_ColorX = colorX ;
m_ColorSpeed = colorSpeed ;
m_Opacity = opacity ;
m_Affine . A ( a ) ;
m_Affine . B ( b ) ;
m_Affine . C ( c ) ;
m_Affine . D ( d ) ;
m_Affine . E ( e ) ;
m_Affine . F ( f ) ;
m_Post . A ( pa ) ;
m_Post . B ( pb ) ;
m_Post . C ( pc ) ;
m_Post . D ( pd ) ;
m_Post . E ( pe ) ;
m_Post . F ( pf ) ;
2018-09-18 23:49:38 -04:00
m_HasPre = ! m_Affine . IsID ( ) ;
2014-07-08 03:11:14 -04:00
m_HasPost = ! m_Post . IsID ( ) ;
m_HasPreOrRegularVars = PreVariationCount ( ) > 0 | | VariationCount ( ) > 0 ;
CacheColorVals ( ) ; //Init already called this, but must call again since color was assigned above.
}
/// <summary>
/// Default copy constructor.
/// </summary>
/// <param name="xform">The Xform object to copy</param>
Xform ( const Xform < T > & xform )
2014-09-10 01:41:26 -04:00
: m_ParentEmber ( nullptr ) //Hack.
2014-07-08 03:11:14 -04:00
{
Xform < T > : : operator = < T > ( xform ) ;
}
/// <summary>
/// Copy constructor to copy an Xform object of type U.
/// </summary>
/// <param name="xform">The Xform object to copy</param>
template < typename U >
Xform ( const Xform < U > & xform )
2014-09-10 01:41:26 -04:00
: m_ParentEmber ( nullptr ) //Hack.
2014-07-08 03:11:14 -04:00
{
Xform < T > : : operator = < U > ( xform ) ;
}
/// <summary>
/// Deletes each element of the variation vector and clears it.
/// </summary>
~ Xform ( )
{
ClearAndDeleteVariations ( ) ;
}
/// <summary>
/// Default assignment operator.
/// </summary>
/// <param name="xform">The Xform object to copy</param>
Xform < T > & operator = ( const Xform < T > & xform )
{
if ( this ! = & xform )
Xform < T > : : operator = < T > ( xform ) ;
return * this ;
}
/// <summary>
/// Assignment operator to assign a Xform object of type U.
/// This will delete all of the variations in the vector
/// and repopulate it with copes of the variation in xform's vector.
/// All other values are assigned directly.
/// </summary>
/// <param name="xform">The Xform object to copy.</param>
/// <returns>Reference to updated self</returns>
template < typename U >
Xform < T > & operator = ( const Xform < U > & xform )
{
m_Affine = xform . m_Affine ;
m_Post = xform . m_Post ;
m_Weight = T ( xform . m_Weight ) ;
m_ColorX = T ( xform . m_ColorX ) ;
m_ColorY = T ( xform . m_ColorY ) ;
m_DirectColor = T ( xform . m_DirectColor ) ;
m_ColorSpeed = T ( xform . m_ColorSpeed ) ;
m_Animate = T ( xform . m_Animate ) ;
m_Opacity = T ( xform . m_Opacity ) ;
CacheColorVals ( ) ;
2018-09-18 23:49:38 -04:00
m_HasPre = xform . HasPre ( ) ;
2014-07-08 03:11:14 -04:00
m_HasPost = xform . HasPost ( ) ;
m_HasPreOrRegularVars = xform . PreVariationCount ( ) > 0 | | xform . VariationCount ( ) > 0 ;
m_Wind [ 0 ] = T ( xform . m_Wind [ 0 ] ) ;
m_Wind [ 1 ] = T ( xform . m_Wind [ 1 ] ) ;
2016-02-13 20:24:51 -05:00
m_MotionFreq = T ( xform . m_MotionFreq ) ;
2014-07-08 03:11:14 -04:00
m_MotionFunc = xform . m_MotionFunc ;
2016-02-13 20:24:51 -05:00
m_MotionOffset = T ( xform . m_MotionOffset ) ;
2014-07-08 03:11:14 -04:00
ClearAndDeleteVariations ( ) ;
//Must manually add them via the AddVariation() function so that
//the variation's m_IndexInXform member gets properly set to this.
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < xform . TotalVariationCount ( ) ; i + + )
2014-07-08 03:11:14 -04:00
{
2014-09-10 01:41:26 -04:00
Variation < T > * var = nullptr ;
2014-08-03 19:16:10 -04:00
if ( Variation < U > * varOrig = xform . GetVariation ( i ) )
{
varOrig - > Copy ( var ) ; //Will convert from type U to type T.
AddVariation ( var ) ; //Will internally call SetPrecalcFlags().
}
2014-07-08 03:11:14 -04:00
}
if ( TotalVariationCount ( ) = = 0 )
SetPrecalcFlags ( ) ;
//If this xform was already part of a different ember, then do not assign, else do.
if ( ! m_ParentEmber & & ( typeid ( T ) = = typeid ( U ) ) )
2014-12-07 03:59:26 -05:00
m_ParentEmber = reinterpret_cast < Ember < T > * > ( xform . ParentEmber ( ) ) ;
2014-07-08 03:11:14 -04:00
2016-05-02 19:54:56 -04:00
CopyCont ( m_Xaos , xform . XaosVec ( ) ) ;
2016-04-03 21:55:12 -04:00
CopyCont ( m_Motion , xform . m_Motion ) ;
2014-07-08 03:11:14 -04:00
m_Name = xform . m_Name ;
return * this ;
}
/// <summary>
/// Init default values.
2015-12-31 16:41:59 -05:00
/// Non default values are used to signify an uninitialized state. This is useful for
2015-03-25 23:47:57 -04:00
/// doing motion interpolation where we don't want to apply motion to all fields. By setting
2015-12-31 16:41:59 -05:00
/// unreasonable values before parsing, then only assigning the ones the motion tags specified,
2015-03-25 23:47:57 -04:00
/// it is clear which fields are intended to have motion applied to them.
2014-07-08 03:11:14 -04:00
/// </summary>
2015-03-25 23:47:57 -04:00
/// <param name="useDefaults">Use reasonable default if true, else use out of bounds values.</param>
void Init ( bool useDefaults = true )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
static size_t count = 0 ;
2014-07-08 03:11:14 -04:00
2015-03-25 23:47:57 -04:00
if ( useDefaults )
{
m_Weight = 0 ;
m_ColorSpeed = T ( 0.5 ) ;
m_Animate = 1 ;
m_ColorX = T ( count & 1 ) ;
m_ColorY = 0 ;
m_DirectColor = 1 ;
m_Opacity = 1 ;
m_Affine . A ( 1 ) ;
m_Affine . B ( 0 ) ;
m_Affine . C ( 0 ) ;
m_Affine . D ( 0 ) ;
m_Affine . E ( 1 ) ;
m_Affine . F ( 0 ) ;
m_Post . A ( 1 ) ;
m_Post . B ( 0 ) ;
m_Post . C ( 0 ) ;
m_Post . D ( 0 ) ;
m_Post . E ( 1 ) ;
m_Post . F ( 0 ) ;
m_Wind [ 0 ] = 0 ;
m_Wind [ 1 ] = 0 ;
m_MotionFreq = 0 ;
2015-07-06 10:05:43 -04:00
m_MotionOffset = 0 ;
2015-03-25 23:47:57 -04:00
}
else
{
m_Weight = EMPTYFIELD ;
m_ColorSpeed = EMPTYFIELD ;
m_Animate = EMPTYFIELD ;
m_ColorX = EMPTYFIELD ;
m_ColorY = EMPTYFIELD ;
m_DirectColor = EMPTYFIELD ;
m_Opacity = EMPTYFIELD ;
m_Affine . A ( EMPTYFIELD ) ;
m_Affine . B ( EMPTYFIELD ) ;
m_Affine . C ( EMPTYFIELD ) ;
m_Affine . D ( EMPTYFIELD ) ;
m_Affine . E ( EMPTYFIELD ) ;
m_Affine . F ( EMPTYFIELD ) ;
m_Post . A ( EMPTYFIELD ) ;
m_Post . B ( EMPTYFIELD ) ;
m_Post . C ( EMPTYFIELD ) ;
m_Post . D ( EMPTYFIELD ) ;
m_Post . E ( EMPTYFIELD ) ;
m_Post . F ( EMPTYFIELD ) ;
m_Wind [ 0 ] = EMPTYFIELD ;
m_Wind [ 1 ] = EMPTYFIELD ;
m_MotionFreq = EMPTYFIELD ;
2015-07-06 10:05:43 -04:00
m_MotionOffset = EMPTYFIELD ;
2015-03-25 23:47:57 -04:00
}
2014-07-08 03:11:14 -04:00
2015-12-31 19:00:36 -05:00
m_MotionFunc = eMotion : : MOTION_SIN ;
2014-07-08 03:11:14 -04:00
m_Motion . clear ( ) ;
m_NeedPrecalcSumSquares = false ;
m_NeedPrecalcSqrtSumSquares = false ;
m_NeedPrecalcAngles = false ;
m_NeedPrecalcAtanXY = false ;
m_NeedPrecalcAtanYX = false ;
2018-09-18 23:49:38 -04:00
m_HasPre = false ;
2014-07-08 03:11:14 -04:00
m_HasPost = false ;
m_HasPreOrRegularVars = false ;
2014-09-10 01:41:26 -04:00
m_ParentEmber = nullptr ;
2014-07-08 03:11:14 -04:00
m_PreVariations . reserve ( MAX_VARS_PER_XFORM ) ;
m_Variations . reserve ( MAX_VARS_PER_XFORM ) ;
m_PostVariations . reserve ( MAX_VARS_PER_XFORM ) ;
CacheColorVals ( ) ;
count + + ;
}
/// <summary>
/// Add a pointer to a variation which will be deleted on destruction so the caller should not delete.
/// This checks if the total number of variations is less than or equal to MAX_VARS_PER_XFORM.
/// It also checks if the variation is already present, in which case it doesn't add.
/// If add, set all precalcs.
/// </summary>
/// <param name="variation">Pointer to a varation to add</param>
/// <returns>True if the successful, else false.</returns>
bool AddVariation ( Variation < T > * variation )
{
2014-09-10 01:41:26 -04:00
if ( variation & & ( GetVariationById ( variation - > VariationId ( ) ) = = nullptr ) )
2014-07-08 03:11:14 -04:00
{
string name = variation - > Name ( ) ;
bool pre = name . find ( " pre_ " ) = = 0 ;
bool post = name . find ( " post_ " ) = = 0 ;
vector < Variation < T > * > * vec ;
if ( pre )
vec = & m_PreVariations ;
else if ( post )
vec = & m_PostVariations ;
else
vec = & m_Variations ;
if ( vec - > size ( ) < MAX_VARS_PER_XFORM )
{
vec - > push_back ( variation ) ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
//Flatten must always be last.
for ( size_t i = 0 ; i < vec - > size ( ) ; i + + )
{
if ( ( i ! = vec - > size ( ) - 1 ) & & ( ( * vec ) [ i ] - > Name ( ) . find ( " flatten " ) ! = string : : npos ) )
{
std : : swap ( ( * vec ) [ i ] , ( * vec ) [ vec - > size ( ) - 1 ] ) ;
break ;
}
}
2014-07-08 03:11:14 -04:00
SetPrecalcFlags ( ) ;
return true ;
}
}
return false ;
}
/// <summary>
/// Get a pointer to the variation at the specified index.
/// </summary>
/// <param name="index">The index in the list to retrieve</param>
2014-09-10 01:41:26 -04:00
/// <returns>A pointer to the variation at the index if in range, else nullptr.</returns>
2014-07-08 03:11:14 -04:00
Variation < T > * GetVariation ( size_t index ) const
{
size_t count = 0 ;
2014-09-10 01:41:26 -04:00
Variation < T > * var = nullptr ;
2015-12-31 16:41:59 -05:00
const_cast < Xform < T > * > ( this ) - > AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < variations . size ( ) ; i + + , count + + )
2014-07-08 03:11:14 -04:00
{
if ( count = = index )
{
var = variations [ i ] ;
keepGoing = false ;
break ;
}
}
} ) ;
return var ;
}
/// <summary>
/// Get a pointer to the variation with the specified ID.
/// </summary>
/// <param name="id">The ID to search for</param>
2014-09-10 01:41:26 -04:00
/// <returns>A pointer to the variation if found, else nullptr.</returns>
2014-07-08 03:11:14 -04:00
Variation < T > * GetVariationById ( eVariationId id ) const
{
2014-09-10 01:41:26 -04:00
Variation < T > * var = nullptr ;
2015-12-31 16:41:59 -05:00
const_cast < Xform < T > * > ( this ) - > AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto v : variations )
2014-07-08 03:11:14 -04:00
{
2015-10-27 00:31:35 -04:00
if ( v & & v - > VariationId ( ) = = id )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
var = v ;
2014-07-08 03:11:14 -04:00
keepGoing = false ;
break ;
}
}
} ) ;
return var ;
}
/// <summary>
/// Get a pointer to the variation with the specified name.
/// </summary>
/// <param name="name">The name to search for</param>
2014-09-10 01:41:26 -04:00
/// <returns>A pointer to the variation if found, else nullptr.</returns>
2014-10-14 11:53:15 -04:00
Variation < T > * GetVariationByName ( const string & name ) const
2014-07-08 03:11:14 -04:00
{
2014-09-10 01:41:26 -04:00
Variation < T > * var = nullptr ;
2015-12-31 16:41:59 -05:00
const_cast < Xform < T > * > ( this ) - > AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto v : variations )
2014-07-08 03:11:14 -04:00
{
2015-10-27 00:31:35 -04:00
if ( v & & v - > Name ( ) = = name )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
var = v ;
2014-07-08 03:11:14 -04:00
keepGoing = false ;
break ;
}
}
} ) ;
return var ;
}
/// <summary>
/// Get the index in the list of the variation pointer.
/// Note this is searching for the exact pointer address and not the name or ID of the variation.
/// </summary>
/// <param name="var">A pointer to the variation to search for</param>
/// <returns>The index of the variation if found, else -1</returns>
2014-10-14 11:53:15 -04:00
intmax_t GetVariationIndex ( Variation < T > * var ) const
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
intmax_t count = 0 , index = - 1 ;
2015-12-31 16:41:59 -05:00
const_cast < Xform < T > * > ( this ) - > AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < variations . size ( ) ; i + + , count + + )
2014-07-08 03:11:14 -04:00
{
if ( variations [ i ] = = var )
{
index = count ;
keepGoing = false ;
break ;
}
}
} ) ;
return index ;
}
/// <summary>
/// Delete the variation with the matching ID.
/// Update precalcs if deletion successful.
/// </summary>
/// <param name="id">The ID to search for</param>
/// <returns>True if deletion successful, else false.</returns>
bool DeleteVariationById ( eVariationId id )
{
bool found = false ;
2015-12-31 16:41:59 -05:00
AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < variations . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
{
2015-10-27 00:31:35 -04:00
if ( variations [ i ] & & variations [ i ] - > VariationId ( ) = = id )
2014-07-08 03:11:14 -04:00
{
delete variations [ i ] ;
variations . erase ( variations . begin ( ) + i ) ;
found = true ;
}
}
} ) ;
if ( found )
SetPrecalcFlags ( ) ;
return found ;
}
/// <summary>
/// Delete the motion elements.
/// </summary>
void DeleteMotionElements ( )
{
m_Motion . clear ( ) ;
}
/// <summary>
/// Delete all variations, clear the list and update precalc flags.
/// </summary>
void ClearAndDeleteVariations ( )
{
2015-12-31 16:41:59 -05:00
AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing ) { ClearVec < Variation < T > > ( variations ) ; } ) ;
2014-07-08 03:11:14 -04:00
SetPrecalcFlags ( ) ;
}
/// <summary>
/// Reset this xform to be totally empty by clearing all variations, resetting both affines to the
/// identity matrix, clearing xaos, color, visibility, wind, animate and setting name
/// to the empty string.
2014-09-10 01:41:26 -04:00
/// Note that this also sets the parent ember to nullptr, so if this xform is reused after calling Clear(),
2014-07-08 03:11:14 -04:00
/// the caller must reset the parent ember to whatever ember they add it to again.
/// </summary>
void Clear ( )
{
ClearAndDeleteVariations ( ) ;
DeleteMotionElements ( ) ;
m_Affine . MakeID ( ) ;
m_Post . MakeID ( ) ;
m_Xaos . clear ( ) ;
2014-09-10 01:41:26 -04:00
m_ParentEmber = nullptr ;
2014-07-08 03:11:14 -04:00
m_ColorSpeedCache = 0 ;
m_OneMinusColorCache = 0 ;
2017-04-09 13:10:45 -04:00
m_Opacity = 1 ;
2014-07-08 03:11:14 -04:00
m_Animate = 0 ;
m_Wind [ 0 ] = 0 ;
m_Wind [ 1 ] = 0 ;
m_Name = " " ;
}
/// <summary>
2017-03-27 21:05:06 -04:00
/// Compute color cache values: color speed and one minus color speed.
2014-07-08 03:11:14 -04:00
/// </summary>
void CacheColorVals ( )
{
//Figure out which is right. //TODO.
//m_ColorSpeedCache = m_ColorX * (1 - m_ColorSpeed) / 2;//Apo style.
//m_OneMinusColorCache = (1 + m_ColorSpeed) / 2;
m_ColorSpeedCache = m_ColorSpeed * m_ColorX ; //Flam3 style.
m_OneMinusColorCache = T ( 1.0 ) - m_ColorSpeed ;
}
/// <summary>
/// Return the xaos value at the specified index.
/// If the index is out of range, return 1.
/// This has the convenient effect that xaos is not present
/// by default and only has a value if explicitly added.
/// </summary>
/// <param name="i">The xaos index to retrieve</param>
/// <returns>The value at the index if in range, else 1.</returns>
T Xaos ( size_t i ) const
{
return i < m_Xaos . size ( ) ? m_Xaos [ i ] : 1 ;
}
/// <summary>
/// Set the xaos value for a given xform index.
/// If the index is out of range, a 1 value will be added
/// to the xaos vector repeatedly until it's one less than the
/// requested index in length, then finally add the specified value.
/// </summary>
/// <param name="i">The index to set</param>
/// <param name="val">The xaos value to set it to</param>
2014-10-14 11:53:15 -04:00
void SetXaos ( size_t i , T val )
2014-07-08 03:11:14 -04:00
{
if ( i < m_Xaos . size ( ) )
{
m_Xaos [ i ] = val ;
}
else
{
while ( m_Xaos . size ( ) < = i )
m_Xaos . push_back ( 1 ) ;
m_Xaos [ i ] = val ;
}
}
/// <summary>
/// Determine if any xaos value in the vector up to the xform count
/// of the parent ember is anything other than 1.
/// </summary>
/// <returns>True if found, else false.</returns>
bool XaosPresent ( ) const
{
if ( m_ParentEmber )
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < m_Xaos . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( i < m_ParentEmber - > XformCount ( ) )
if ( ! IsClose < T > ( m_Xaos [ i ] , 1 ) )
return true ; //If at least one entry is not equal to 1, then xaos is present.
return false ;
}
/// <summary>
/// Truncate the xaos vector to match the xform count of the parent ember.
/// </summary>
void TruncateXaos ( )
{
if ( m_ParentEmber )
while ( m_Xaos . size ( ) > m_ParentEmber - > XformCount ( ) )
m_Xaos . pop_back ( ) ;
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
/// <summary>
/// Remove all xaos from this xform.
/// </summary>
void ClearXaos ( )
{
m_Xaos . clear ( ) ;
}
/// <summary>
/// Normalize the variation weights.
/// </summary>
void NormalizeVariationWeights ( )
{
2015-12-31 16:41:59 -05:00
AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
T norm = 0 ;
2015-05-03 20:13:14 -04:00
for ( auto var : variations ) norm + = var - > m_Weight ;
2015-12-31 16:41:59 -05:00
--User changes
-No longer constrain pitch, yaw or depth spinners to -180 - 180.
--Bug fixes
-Properly set color index on padded xforms.
-Adding a padding final xform included a linear variation with a weight of zero to not appear empty. Made it have a weight of 1.
-Always write animate tag on final xform when saving to Xml.
-Motion was being applied to the wrong flame in SheepTools::Edge(), so apply it to the correct one.
-Prevent divide by zero when normalizing variation weights.
-Was accidentally adding the placeholder value of -9999 for motion_offset to varation weights and parameters when applying motion. Set to zero if no value present.
-Clamp flame rotation values to -180 - 180 when reading a flame from Xml.
-Events were not properly wired for user changes in the random rotations per blend controls in the sequencer.
-Fix major UI bugs with sequencer min/max random controls which made it nearly impossible to hand type values.
-Values from rotations per blend and rotations per blend max were not being saved to file between program runs.
-Checking animate for an xform was not applied to all flames even if Apply All was checked.
-Changing interpolation type, temporal filter width, temporal type, and affine interpolation type were not actually saving to the flame when changed.
-Grid on the main window was not being drawn at the right scale initially due to some OpenGL initialization occurring in the wrong order.
-Severe bugs in sequence generation code:
--Improperly detected padding xforms.
--When looking for specific variations during xform aligning, only presence was detected, when it should have been presence plus a weight greater than zero.
--When adding specific variations during xform aligning, must first remove any variations of that type.
--Two variables were unsigned when they should have been signed. This prevented large blocks of code from ever executing.
--When interpolating affines, an EPS that was too small was used, causing affine values to interpolate incorrectly. Instead use 1e-10 to ensure results equal to flam3.
--Code changes
-Modify FractoriumEmberController::UpdateXform() to pass the selected xform index as well as the absolute index to func().
2018-06-13 00:20:15 -04:00
for ( auto var : variations ) var - > m_Weight / = Zeps ( norm ) ; //Ensure a divide by zero never happens.
2014-07-08 03:11:14 -04:00
} ) ;
}
/// <summary>
/// Applies this xform to the point passed in and saves the result in the out point.
/// It's important to understand what happens here since it's the inner core of the algorithm.
/// See the internal comments for step by step details.
/// </summary>
/// <param name="inPoint">The initial point from the previous iteration</param>
/// <param name="outPoint">The output point</param>
/// <param name="rand">The random context to use</param>
/// <returns>True if a bad value was calculated, else false.</returns>
bool Apply ( Point < T > * inPoint , Point < T > * outPoint , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
size_t i ;
//This must be local, rather than a member, because this function can be called
//from multiple threads. If it were a member, they'd be clobbering each others' values.
IteratorHelper < T > iterHelper ;
//Calculate the color coordinate/index in the palette to look up later when accumulating the output point
//to the histogram. Calculate this value by interpolating between the index value of the
//last iteration with the one specified in this xform. Note that some cached values are used
//to reduce the amount of processing.
2017-03-27 21:05:06 -04:00
outPoint - > m_Opacity = m_Opacity ;
2014-07-08 03:11:14 -04:00
iterHelper . m_Color . x = outPoint - > m_ColorX = m_ColorSpeedCache + ( m_OneMinusColorCache * inPoint - > m_ColorX ) ;
if ( m_HasPreOrRegularVars )
{
//Compute the pre affine portion of the transform.
//These x, y values are what get passed to the variations below.
//Note that they are not changed after this, except in the case of pre_ variations.
2018-09-18 23:49:38 -04:00
if ( m_HasPre )
{
iterHelper . m_TransX = ( m_Affine . A ( ) * inPoint - > m_X ) + ( m_Affine . B ( ) * inPoint - > m_Y ) + m_Affine . C ( ) ;
iterHelper . m_TransY = ( m_Affine . D ( ) * inPoint - > m_X ) + ( m_Affine . E ( ) * inPoint - > m_Y ) + m_Affine . F ( ) ;
}
else
{
iterHelper . m_TransX = inPoint - > m_X ;
iterHelper . m_TransY = inPoint - > m_Y ;
}
2014-07-08 03:11:14 -04:00
iterHelper . m_TransZ = inPoint - > m_Z ;
//Apply pre_ variations, these don't affect outPoint, only iterHelper.m_TransX, Y, Z.
for ( i = 0 ; i < PreVariationCount ( ) ; i + + )
{
iterHelper . In . x = iterHelper . m_TransX ; //Read must be done before every pre variation because transX/Y are changing.
iterHelper . In . y = iterHelper . m_TransY ;
iterHelper . In . z = iterHelper . m_TransZ ;
--User changes
-Allow for pausing the renderer in the main window. This makes is more efficient when entering many parameters, such as when following a tutorial.
-Add support for new variations: erf, gamma, jac_cn, jac_dn, jac_sn, logDB, pressure_wave, pRose3D, splits3D, w, waves2b, x, xerf, y, z.
-Inform user of the start and stop of file parsing in EmberAnimate because the files could potentially be very large.
-Move the follwing fields to a new table called Animation: Interpolation, Affine Interpolation, Temporal Samples, Temporal Filter Width, Temporal Filter Type.
-These currently have no effect on the interactive renderer and instead are used when running flames through EmberGenome to generate sequences, and then animating them in Fractorium or EmberAnimate.
-Add new parameter overrides for EmberRender and EmberAnimate which directly assign values to all flames being rendered, rather than scale:
--quality
--demin
--demax
--Bug fixes
-Left pad instead of right pad names of sequence outputs from EmberGenome.
-Unique file naming was broken for files which already had an underscore in them.
-Properly report that png is the default format of EmberRender and EmberAnimate output instead of erroneously claiming it was jpg.
-Make command line programs search these folders in this order for the palette file:
./
~/.fractorium
~/.config/fractorium
/usr/share/fractorium
/usr/local/share/fractorium
-Fix possible bad values in hexes.
-Better assignment of Z variables.
-Fix boarders due to use of poorly implemented rint() function from flam3. Use std::rint() now.
-wedge_sph was completely wrong due to having accidentally swapped the mapping of two parameters.
-Make juliascope juliascope_power parameter be of type REAL_NONZERO since it's used as a denominator.
-Make Z assignment compatible with the originals in:
-arch, bcircle, bCollide, bent, bent2, bisplit, blob, blur_linear, blur_square, bMod, boarders, boarders2, bSwirl, bTransform, butterfly, cardioid, cell, circleblur, circlize, circlize2, circus, collideoscope, cos, cosine, cosh, coth, cpow, cpow2, crescents, cropn, csc, csch, curl, curve, dc_gridout, deltaa, diamond, disc2, eclipse, eCollide, edisc, eJulia, elliptic, eMod, eMotion, ennepers, epispiral, ePush, eRotate, eScale, eSwirl, ex, exp, expo, exponential, fan, fdisc, fibonacci, fibonacci2, fisheye, flipcircle, flipy, flower, flux, funnel, glynnia, GlynnSim1, GlynnSim2, GlynnSim3, gridout, handkerchief, heart, hole, idisc, julia, julian2, juliaNab, kaleidoscope, lazyTravis, Lissajous, mask, MobiusN, mobius_strip, modulus, murl, murl2, npolar, ortho, oscilloscope, parabola, perspective, petal, phoenix_julia, pie (was also inconsistent between cpu and gpu), poincare, popcorn, popcorn2, power, pow_block, rational3, rays, rblur, rings, rippled, roundspher, sec, secant2, sigmoid, sin, sineblur, sinh, sinusgrid, sphericaln, spiralwing, spirograph, split, squarize, squirrel, squish, sschecks, starblur, stripes, stwin, super_shape, tan, tancos, tangent, tanh, TwinTrian, twoface, unpolar, waves, wavesn, wedge_julia, whorl, xheart, zblur, zscale.
--Code changes
-Generalize Variation::PrecalcHelper() and rename to PrePostPrecalcHelper().
--Do the same for the OpenCL version and rename it PrePostPrecalcOpenCLString().
-Rename Variation::m_AssignType to m_PrePostAssignType since it's only relevant to pre/post variations.
2016-01-29 20:02:15 -05:00
m_PreVariations [ i ] - > PrePostPrecalcHelper ( iterHelper ) ; //Apply per-variation precalc, the second parameter is unused for pre variations.
2014-07-08 03:11:14 -04:00
m_PreVariations [ i ] - > Func ( iterHelper , * outPoint , rand ) ;
WritePre ( iterHelper , m_PreVariations [ i ] - > AssignType ( ) ) ;
}
if ( VariationCount ( ) > 0 )
{
//The original calculates sumsq and sumsqrt every time, regardless if they're used or not.
//With Precalc(), only calculate those values if they're needed.
Precalc ( iterHelper ) ; //Only need per-xform precalc with regular variations.
iterHelper . In . x = iterHelper . m_TransX ; //Only need to read once with regular variations, because transX/Y are fixed.
iterHelper . In . y = iterHelper . m_TransY ;
iterHelper . In . z = iterHelper . m_TransZ ;
//Since these get summed, initialize them to zero.
outPoint - > m_X = outPoint - > m_Y = outPoint - > m_Z = 0 ;
//Apply variations to the transformed points, accumulating each time, and store the final value in outPoint.
//Using a virtual function is about 3% faster than using a large case statement like the original did.
//Although research says that using virtual functions is slow, experience says otherwise. They execute
//with the exact same speed as both regular and static member functions.
for ( i = 0 ; i < VariationCount ( ) ; i + + )
{
m_Variations [ i ] - > Func ( iterHelper , * outPoint , rand ) ;
outPoint - > m_X + = iterHelper . Out . x ;
outPoint - > m_Y + = iterHelper . Out . y ;
outPoint - > m_Z + = iterHelper . Out . z ;
}
}
else //Only pre variations are present, no regular ones, so assign the affine transformed points directly to the output points.
{
outPoint - > m_X = iterHelper . m_TransX ;
outPoint - > m_Y = iterHelper . m_TransY ;
outPoint - > m_Z = iterHelper . m_TransZ ;
}
}
2014-11-03 02:16:34 -05:00
//Return the affine transformed points if no variations are present.
//Note this differs from flam3, which would just return zero in that scenario.
2014-07-08 03:11:14 -04:00
else
{
//There are no variations, so the affine transformed points can be assigned directly to the output points.
T inX = inPoint - > m_X ;
--User changes
-Support 4k monitors, and in general, properly scale any monitor that is not HD.
-Allow for a spatial filter of radius zero, which means do not use a spatial filter.
-Add new variations: concentric, cpow3, helicoid, helix, rand_cubes, sphereblur.
-Use a new method for computing elliptic which is more precise. Developed by Discord user Claude.
-Remove the 8 variation per xform limitation on the GPU.
-Allow for loading the last flame file on startup, rather than randoms.
-Use two different default quality values in the interactive renderer, one each for CPU and GPU.
-Creating linked xforms was using non-standard behavior. Make it match Apo and also support creating multiple linked xforms at once.
--Bug fixes
-No variations in an xform used to have the same behavior as a single linear variation with weight 1. While sensible, this breaks backward compatibility. No variations now sets the output point to zeroes.
-Prevent crashing the program when adjusting a value on the main window while a final render is in progress.
-The xaos table was inverted.
--Code changes
-Convert projects to Visual Studio 2017.
-Change bad vals from +- 1e10 to +-1e20.
-Reintroduce the symmetry tag in xforms for legacy support in programs that do not use color_speed.
-Compiler will not let us use default values in templated member functions anymore.
2017-11-26 20:27:00 -05:00
outPoint - > m_X = 0 ; //(m_Affine.A() * inX) + (m_Affine.B() * inPoint->m_Y) + m_Affine.C();
outPoint - > m_Y = 0 ; //(m_Affine.D() * inX) + (m_Affine.E() * inPoint->m_Y) + m_Affine.F();
outPoint - > m_Z = 0 ; //inPoint->m_Z;
2014-07-08 03:11:14 -04:00
}
//Apply post variations, these will modify outPoint.
for ( i = 0 ; i < PostVariationCount ( ) ; i + + )
{
iterHelper . In . x = outPoint - > m_X ; //Read must be done before every post variation because the out point is changing.
iterHelper . In . y = outPoint - > m_Y ;
iterHelper . In . z = outPoint - > m_Z ;
--User changes
-Allow for pausing the renderer in the main window. This makes is more efficient when entering many parameters, such as when following a tutorial.
-Add support for new variations: erf, gamma, jac_cn, jac_dn, jac_sn, logDB, pressure_wave, pRose3D, splits3D, w, waves2b, x, xerf, y, z.
-Inform user of the start and stop of file parsing in EmberAnimate because the files could potentially be very large.
-Move the follwing fields to a new table called Animation: Interpolation, Affine Interpolation, Temporal Samples, Temporal Filter Width, Temporal Filter Type.
-These currently have no effect on the interactive renderer and instead are used when running flames through EmberGenome to generate sequences, and then animating them in Fractorium or EmberAnimate.
-Add new parameter overrides for EmberRender and EmberAnimate which directly assign values to all flames being rendered, rather than scale:
--quality
--demin
--demax
--Bug fixes
-Left pad instead of right pad names of sequence outputs from EmberGenome.
-Unique file naming was broken for files which already had an underscore in them.
-Properly report that png is the default format of EmberRender and EmberAnimate output instead of erroneously claiming it was jpg.
-Make command line programs search these folders in this order for the palette file:
./
~/.fractorium
~/.config/fractorium
/usr/share/fractorium
/usr/local/share/fractorium
-Fix possible bad values in hexes.
-Better assignment of Z variables.
-Fix boarders due to use of poorly implemented rint() function from flam3. Use std::rint() now.
-wedge_sph was completely wrong due to having accidentally swapped the mapping of two parameters.
-Make juliascope juliascope_power parameter be of type REAL_NONZERO since it's used as a denominator.
-Make Z assignment compatible with the originals in:
-arch, bcircle, bCollide, bent, bent2, bisplit, blob, blur_linear, blur_square, bMod, boarders, boarders2, bSwirl, bTransform, butterfly, cardioid, cell, circleblur, circlize, circlize2, circus, collideoscope, cos, cosine, cosh, coth, cpow, cpow2, crescents, cropn, csc, csch, curl, curve, dc_gridout, deltaa, diamond, disc2, eclipse, eCollide, edisc, eJulia, elliptic, eMod, eMotion, ennepers, epispiral, ePush, eRotate, eScale, eSwirl, ex, exp, expo, exponential, fan, fdisc, fibonacci, fibonacci2, fisheye, flipcircle, flipy, flower, flux, funnel, glynnia, GlynnSim1, GlynnSim2, GlynnSim3, gridout, handkerchief, heart, hole, idisc, julia, julian2, juliaNab, kaleidoscope, lazyTravis, Lissajous, mask, MobiusN, mobius_strip, modulus, murl, murl2, npolar, ortho, oscilloscope, parabola, perspective, petal, phoenix_julia, pie (was also inconsistent between cpu and gpu), poincare, popcorn, popcorn2, power, pow_block, rational3, rays, rblur, rings, rippled, roundspher, sec, secant2, sigmoid, sin, sineblur, sinh, sinusgrid, sphericaln, spiralwing, spirograph, split, squarize, squirrel, squish, sschecks, starblur, stripes, stwin, super_shape, tan, tancos, tangent, tanh, TwinTrian, twoface, unpolar, waves, wavesn, wedge_julia, whorl, xheart, zblur, zscale.
--Code changes
-Generalize Variation::PrecalcHelper() and rename to PrePostPrecalcHelper().
--Do the same for the OpenCL version and rename it PrePostPrecalcOpenCLString().
-Rename Variation::m_AssignType to m_PrePostAssignType since it's only relevant to pre/post variations.
2016-01-29 20:02:15 -05:00
m_PostVariations [ i ] - > PrePostPrecalcHelper ( iterHelper ) ; //Apply per-variation precalc.
2014-07-08 03:11:14 -04:00
m_PostVariations [ i ] - > Func ( iterHelper , * outPoint , rand ) ;
WritePost ( iterHelper , * outPoint , m_PostVariations [ i ] - > AssignType ( ) ) ;
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Optionally apply the post affine transform if it's present.
if ( m_HasPost )
{
T postX = outPoint - > m_X ;
outPoint - > m_X = ( m_Post . A ( ) * postX ) + ( m_Post . B ( ) * outPoint - > m_Y ) + m_Post . C ( ) ;
outPoint - > m_Y = ( m_Post . D ( ) * postX ) + ( m_Post . E ( ) * outPoint - > m_Y ) + m_Post . F ( ) ;
}
2015-12-31 16:41:59 -05:00
outPoint - > m_ColorX = iterHelper . m_Color . x + m_DirectColor * ( outPoint - > m_ColorX - iterHelper . m_Color . x ) ;
06/04/2017
--User changes
-Make all fonts be MS Shell Dlg 2. This will require reloading dark.qss if users are already using it.
-Limit size of the left side of the palette editor.
-Disable create from image buttons in the palette editor when working on a fixed palette.
--Bug fixes
-The following variations were wrong: coshq, cothq.
-During iteration, the color index could become nan if all xform color speeds were negative. This could lead to bad results on the GPU. Fix to check for nan. Minimal speed difference.
--Code changes
-Make the following variations safer by using Zeps(): sinq, sinhq, secq, sechq, tanq, tanhq, cosq, coshq, cotq, cothq, cscq, cschq, estiq.
-Always pass -cl-no-signed-zeros -cl-denorms-are-zero to kernel compiles for both single and double.
-Flush all denormals to zero for all executable programs. This will likely lead to a speedup for badly behaving programs.
2017-06-04 20:37:29 -04:00
if ( std : : isnan ( outPoint - > m_ColorX ) )
outPoint - > m_ColorX = 0 ;
2014-07-08 03:11:14 -04:00
//Has the trajectory of x or y gone either to infinity, or too close to zero?
return BadVal ( outPoint - > m_X ) | | BadVal ( outPoint - > m_Y ) /* || BadVal(outPoint->m_Z)*/ ;
}
//Why are we not using template with member var addr as arg here?//TODO
2015-03-25 23:47:57 -04:00
# define APPMOT(x) \
do \
{ \
if ( currentMot . x ! = EMPTYFIELD ) \
2015-07-06 10:05:43 -04:00
x + = currentMot . x * Interpolater < T > : : MotionFuncs ( func , freq * ( blend + offset ) ) ; \
2015-03-25 23:47:57 -04:00
} while ( 0 )
2014-07-08 03:11:14 -04:00
/// <summary>
/// Apply the motion functions from the passed in xform to this xform.
/// </summary>
/// <param name="xform">The xform containing the motion functions</param>
/// <param name="blend">The time blending value 0-1</param>
void ApplyMotion ( Xform < T > & xform , T blend )
{
//Loop over the motion elements and add their contribution to the original vals.
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < xform . m_Motion . size ( ) ; i + + )
2014-09-10 01:41:26 -04:00
{
2014-07-08 03:11:14 -04:00
//Original only pulls these from the first motion xform which is a bug. Want to pull it from each one.
--User changes
-No longer constrain pitch, yaw or depth spinners to -180 - 180.
--Bug fixes
-Properly set color index on padded xforms.
-Adding a padding final xform included a linear variation with a weight of zero to not appear empty. Made it have a weight of 1.
-Always write animate tag on final xform when saving to Xml.
-Motion was being applied to the wrong flame in SheepTools::Edge(), so apply it to the correct one.
-Prevent divide by zero when normalizing variation weights.
-Was accidentally adding the placeholder value of -9999 for motion_offset to varation weights and parameters when applying motion. Set to zero if no value present.
-Clamp flame rotation values to -180 - 180 when reading a flame from Xml.
-Events were not properly wired for user changes in the random rotations per blend controls in the sequencer.
-Fix major UI bugs with sequencer min/max random controls which made it nearly impossible to hand type values.
-Values from rotations per blend and rotations per blend max were not being saved to file between program runs.
-Checking animate for an xform was not applied to all flames even if Apply All was checked.
-Changing interpolation type, temporal filter width, temporal type, and affine interpolation type were not actually saving to the flame when changed.
-Grid on the main window was not being drawn at the right scale initially due to some OpenGL initialization occurring in the wrong order.
-Severe bugs in sequence generation code:
--Improperly detected padding xforms.
--When looking for specific variations during xform aligning, only presence was detected, when it should have been presence plus a weight greater than zero.
--When adding specific variations during xform aligning, must first remove any variations of that type.
--Two variables were unsigned when they should have been signed. This prevented large blocks of code from ever executing.
--When interpolating affines, an EPS that was too small was used, causing affine values to interpolate incorrectly. Instead use 1e-10 to ensure results equal to flam3.
--Code changes
-Modify FractoriumEmberController::UpdateXform() to pass the selected xform index as well as the absolute index to func().
2018-06-13 00:20:15 -04:00
auto & currentMot = xform . m_Motion [ i ] ;
auto freq = currentMot . m_MotionFreq ;
auto func = currentMot . m_MotionFunc ;
auto offset = currentMot . m_MotionOffset ;
auto cleanOffset = offset ! = EMPTYFIELD ? offset : 0 ;
2014-07-08 03:11:14 -04:00
//Clamp these to the appropriate range after all are applied.
APPMOT ( m_Weight ) ;
APPMOT ( m_ColorX ) ;
//APPMOT(m_ColorY);
2014-09-10 01:41:26 -04:00
APPMOT ( m_DirectColor ) ;
APPMOT ( m_Opacity ) ;
2014-07-08 03:11:14 -04:00
APPMOT ( m_ColorSpeed ) ;
APPMOT ( m_Animate ) ;
2014-09-10 01:41:26 -04:00
2015-03-25 23:47:57 -04:00
for ( size_t j = 0 ; j < currentMot . TotalVariationCount ( ) ; j + + ) //For each variation in the motion xform.
2014-07-08 03:11:14 -04:00
{
2015-03-25 23:47:57 -04:00
Variation < T > * motVar = currentMot . GetVariation ( j ) ; //Get the variation, which may or may not be present in this xform.
2014-07-08 03:11:14 -04:00
ParametricVariation < T > * motParVar = dynamic_cast < ParametricVariation < T > * > ( motVar ) ;
Variation < T > * var = GetVariationById ( motVar - > VariationId ( ) ) ; //See if the variation in the motion xform was present in the xform.
if ( ! var ) //It wasn't present, so add it and set the weight.
{
Variation < T > * newVar = motVar - > Copy ( ) ;
--User changes
-No longer constrain pitch, yaw or depth spinners to -180 - 180.
--Bug fixes
-Properly set color index on padded xforms.
-Adding a padding final xform included a linear variation with a weight of zero to not appear empty. Made it have a weight of 1.
-Always write animate tag on final xform when saving to Xml.
-Motion was being applied to the wrong flame in SheepTools::Edge(), so apply it to the correct one.
-Prevent divide by zero when normalizing variation weights.
-Was accidentally adding the placeholder value of -9999 for motion_offset to varation weights and parameters when applying motion. Set to zero if no value present.
-Clamp flame rotation values to -180 - 180 when reading a flame from Xml.
-Events were not properly wired for user changes in the random rotations per blend controls in the sequencer.
-Fix major UI bugs with sequencer min/max random controls which made it nearly impossible to hand type values.
-Values from rotations per blend and rotations per blend max were not being saved to file between program runs.
-Checking animate for an xform was not applied to all flames even if Apply All was checked.
-Changing interpolation type, temporal filter width, temporal type, and affine interpolation type were not actually saving to the flame when changed.
-Grid on the main window was not being drawn at the right scale initially due to some OpenGL initialization occurring in the wrong order.
-Severe bugs in sequence generation code:
--Improperly detected padding xforms.
--When looking for specific variations during xform aligning, only presence was detected, when it should have been presence plus a weight greater than zero.
--When adding specific variations during xform aligning, must first remove any variations of that type.
--Two variables were unsigned when they should have been signed. This prevented large blocks of code from ever executing.
--When interpolating affines, an EPS that was too small was used, causing affine values to interpolate incorrectly. Instead use 1e-10 to ensure results equal to flam3.
--Code changes
-Modify FractoriumEmberController::UpdateXform() to pass the selected xform index as well as the absolute index to func().
2018-06-13 00:20:15 -04:00
newVar - > m_Weight = motVar - > m_Weight * Interpolater < T > : : MotionFuncs ( func , freq * ( blend + cleanOffset ) ) ;
2014-07-08 03:11:14 -04:00
AddVariation ( newVar ) ;
var = newVar ; //Use this below for params.
}
else //It was present, so apply the motion func to the weight.
{
--User changes
-No longer constrain pitch, yaw or depth spinners to -180 - 180.
--Bug fixes
-Properly set color index on padded xforms.
-Adding a padding final xform included a linear variation with a weight of zero to not appear empty. Made it have a weight of 1.
-Always write animate tag on final xform when saving to Xml.
-Motion was being applied to the wrong flame in SheepTools::Edge(), so apply it to the correct one.
-Prevent divide by zero when normalizing variation weights.
-Was accidentally adding the placeholder value of -9999 for motion_offset to varation weights and parameters when applying motion. Set to zero if no value present.
-Clamp flame rotation values to -180 - 180 when reading a flame from Xml.
-Events were not properly wired for user changes in the random rotations per blend controls in the sequencer.
-Fix major UI bugs with sequencer min/max random controls which made it nearly impossible to hand type values.
-Values from rotations per blend and rotations per blend max were not being saved to file between program runs.
-Checking animate for an xform was not applied to all flames even if Apply All was checked.
-Changing interpolation type, temporal filter width, temporal type, and affine interpolation type were not actually saving to the flame when changed.
-Grid on the main window was not being drawn at the right scale initially due to some OpenGL initialization occurring in the wrong order.
-Severe bugs in sequence generation code:
--Improperly detected padding xforms.
--When looking for specific variations during xform aligning, only presence was detected, when it should have been presence plus a weight greater than zero.
--When adding specific variations during xform aligning, must first remove any variations of that type.
--Two variables were unsigned when they should have been signed. This prevented large blocks of code from ever executing.
--When interpolating affines, an EPS that was too small was used, causing affine values to interpolate incorrectly. Instead use 1e-10 to ensure results equal to flam3.
--Code changes
-Modify FractoriumEmberController::UpdateXform() to pass the selected xform index as well as the absolute index to func().
2018-06-13 00:20:15 -04:00
var - > m_Weight + = motVar - > m_Weight * Interpolater < T > : : MotionFuncs ( func , freq * ( blend + cleanOffset ) ) ;
2014-07-08 03:11:14 -04:00
}
//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.
2015-10-27 00:31:35 -04:00
if ( motParVar )
2014-07-08 03:11:14 -04:00
{
2015-06-28 20:48:26 -04:00
auto parVar = dynamic_cast < ParametricVariation < T > * > ( var ) ;
auto params = parVar - > Params ( ) ;
auto motParams = motParVar - > Params ( ) ;
2014-07-08 03:11:14 -04:00
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < motParVar - > ParamCount ( ) ; k + + )
2014-07-08 03:11:14 -04:00
{
if ( ! motParams [ k ] . IsPrecalc ( ) )
--User changes
-No longer constrain pitch, yaw or depth spinners to -180 - 180.
--Bug fixes
-Properly set color index on padded xforms.
-Adding a padding final xform included a linear variation with a weight of zero to not appear empty. Made it have a weight of 1.
-Always write animate tag on final xform when saving to Xml.
-Motion was being applied to the wrong flame in SheepTools::Edge(), so apply it to the correct one.
-Prevent divide by zero when normalizing variation weights.
-Was accidentally adding the placeholder value of -9999 for motion_offset to varation weights and parameters when applying motion. Set to zero if no value present.
-Clamp flame rotation values to -180 - 180 when reading a flame from Xml.
-Events were not properly wired for user changes in the random rotations per blend controls in the sequencer.
-Fix major UI bugs with sequencer min/max random controls which made it nearly impossible to hand type values.
-Values from rotations per blend and rotations per blend max were not being saved to file between program runs.
-Checking animate for an xform was not applied to all flames even if Apply All was checked.
-Changing interpolation type, temporal filter width, temporal type, and affine interpolation type were not actually saving to the flame when changed.
-Grid on the main window was not being drawn at the right scale initially due to some OpenGL initialization occurring in the wrong order.
-Severe bugs in sequence generation code:
--Improperly detected padding xforms.
--When looking for specific variations during xform aligning, only presence was detected, when it should have been presence plus a weight greater than zero.
--When adding specific variations during xform aligning, must first remove any variations of that type.
--Two variables were unsigned when they should have been signed. This prevented large blocks of code from ever executing.
--When interpolating affines, an EPS that was too small was used, causing affine values to interpolate incorrectly. Instead use 1e-10 to ensure results equal to flam3.
--Code changes
-Modify FractoriumEmberController::UpdateXform() to pass the selected xform index as well as the absolute index to func().
2018-06-13 00:20:15 -04:00
* ( params [ k ] . Param ( ) ) + = motParams [ k ] . ParamVal ( ) * Interpolater < T > : : MotionFuncs ( func , freq * ( blend + cleanOffset ) ) ;
2014-07-08 03:11:14 -04:00
}
}
}
2014-10-14 11:53:15 -04:00
for ( glm : : length_t j = 0 ; j < 2 ; j + + )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( glm : : length_t k = 0 ; k < 3 ; k + + )
2014-07-08 03:11:14 -04:00
{
APPMOT ( m_Affine . m_Mat [ j ] [ k ] ) ;
APPMOT ( m_Post . m_Mat [ j ] [ k ] ) ;
}
}
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Make sure certain params are within reasonable bounds.
ClampRef < T > ( m_ColorX , 0 , 1 ) ;
//ClampRef<T>(m_ColorY, 0, 1);
ClampRef < T > ( m_DirectColor , 0 , 1 ) ;
ClampRef < T > ( m_Opacity , 0 , 1 ) ; //Original didn't clamp these, but do it here for correctness.
2015-03-25 23:47:57 -04:00
ClampRef < T > ( m_ColorSpeed , - 1 , 1 ) ;
2014-07-08 03:11:14 -04:00
ClampGte0Ref < T > ( m_Weight ) ;
}
/// <summary>
/// Accessors.
/// The precalc flags are duplicated in each variation. Each value here
/// is true if any of the variations need it precalculated.
/// </summary>
2014-10-14 11:53:15 -04:00
inline bool NeedPrecalcSumSquares ( ) const { return m_NeedPrecalcSumSquares ; }
2014-07-08 03:11:14 -04:00
inline bool NeedPrecalcSqrtSumSquares ( ) const { return m_NeedPrecalcSqrtSumSquares ; }
2014-10-14 11:53:15 -04:00
inline bool NeedPrecalcAngles ( ) const { return m_NeedPrecalcAngles ; }
inline bool NeedPrecalcAtanXY ( ) const { return m_NeedPrecalcAtanXY ; }
inline bool NeedPrecalcAtanYX ( ) const { return m_NeedPrecalcAtanYX ; }
2014-07-08 03:11:14 -04:00
inline bool NeedAnyPrecalc ( ) const { return NeedPrecalcSumSquares ( ) | | NeedPrecalcSqrtSumSquares ( ) | | NeedPrecalcAngles ( ) | | NeedPrecalcAtanXY ( ) | | NeedPrecalcAtanYX ( ) ; }
2018-09-18 23:49:38 -04:00
bool HasPre ( ) const { return m_HasPre ; }
2014-07-08 03:11:14 -04:00
bool HasPost ( ) const { return m_HasPost ; }
2014-10-14 11:53:15 -04:00
size_t PreVariationCount ( ) const { return m_PreVariations . size ( ) ; }
size_t VariationCount ( ) const { return m_Variations . size ( ) ; }
size_t PostVariationCount ( ) const { return m_PostVariations . size ( ) ; }
size_t TotalVariationCount ( ) const { return PreVariationCount ( ) + VariationCount ( ) + PostVariationCount ( ) ; }
2014-07-08 03:11:14 -04:00
bool Empty ( ) const { return TotalVariationCount ( ) = = 0 & & m_Affine . IsID ( ) ; } //Use this instead of padding like the original did.
T ColorSpeedCache ( ) const { return m_ColorSpeedCache ; }
T OneMinusColorCache ( ) const { return m_OneMinusColorCache ; }
const vector < T > & XaosVec ( ) const { return m_Xaos ; }
Ember < T > * ParentEmber ( ) const { return m_ParentEmber ; }
void ParentEmber ( Ember < T > * ember ) { m_ParentEmber = ember ; }
2015-06-29 23:13:53 -04:00
intmax_t IndexInParentEmber ( ) const { return m_ParentEmber ? m_ParentEmber - > GetTotalXformIndex ( const_cast < Xform < T > * > ( this ) ) : - 1 ; }
2014-07-08 03:11:14 -04:00
/// <summary>
/// Set the precalc flags based on whether any variation in the vector needs them.
/// Also call Precalc() virtual function on each variation, which will setup any needed
/// precalcs in parametric variations.
/// Set the parent xform of each variation to this.
/// </summary>
void SetPrecalcFlags ( )
{
m_NeedPrecalcSumSquares = false ;
m_NeedPrecalcSqrtSumSquares = false ;
m_NeedPrecalcAngles = false ;
m_NeedPrecalcAtanXY = false ;
m_NeedPrecalcAtanYX = false ;
2018-09-18 23:49:38 -04:00
m_HasPre = ! m_Affine . IsID ( ) ;
2014-07-08 03:11:14 -04:00
m_HasPost = ! m_Post . IsID ( ) ;
m_HasPreOrRegularVars = PreVariationCount ( ) > 0 | | VariationCount ( ) > 0 ;
//Only set precalcs for regular variations, they work differently for pre and post.
2015-05-03 20:13:14 -04:00
for ( auto var : m_Variations )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
if ( var - > NeedPrecalcSumSquares ( ) )
2014-07-08 03:11:14 -04:00
m_NeedPrecalcSumSquares = true ;
2015-05-03 20:13:14 -04:00
if ( var - > NeedPrecalcSqrtSumSquares ( ) )
2014-07-08 03:11:14 -04:00
m_NeedPrecalcSqrtSumSquares = true ;
2015-05-03 20:13:14 -04:00
if ( var - > NeedPrecalcAngles ( ) )
2014-07-08 03:11:14 -04:00
m_NeedPrecalcAngles = true ;
2015-05-03 20:13:14 -04:00
if ( var - > NeedPrecalcAtanXY ( ) )
2014-07-08 03:11:14 -04:00
m_NeedPrecalcAtanXY = true ;
2015-05-03 20:13:14 -04:00
if ( var - > NeedPrecalcAtanYX ( ) )
2014-07-08 03:11:14 -04:00
m_NeedPrecalcAtanYX = true ;
}
2015-12-31 16:41:59 -05:00
AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto var : variations )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
var - > ParentXform ( this ) ;
var - > Precalc ( ) ;
2014-07-08 03:11:14 -04:00
}
} ) ;
}
/// <summary>
/// Based on the precalc flags determined in SetPrecalcFlags(), do the appropriate precalcs.
/// </summary>
/// <param name="helper">The iterator helper to store the precalculated values in</param>
void Precalc ( IteratorHelper < T > & helper )
{
if ( m_NeedPrecalcSumSquares )
{
helper . m_PrecalcSumSquares = SQR ( helper . m_TransX ) + SQR ( helper . m_TransY ) ;
if ( m_NeedPrecalcSqrtSumSquares )
{
2015-04-13 07:32:58 -04:00
helper . m_PrecalcSqrtSumSquares = std : : sqrt ( helper . m_PrecalcSumSquares ) ;
2014-07-08 03:11:14 -04:00
if ( m_NeedPrecalcAngles )
{
2019-12-28 00:04:41 -05:00
helper . m_PrecalcCosa = helper . m_TransX / Zeps ( helper . m_PrecalcSqrtSumSquares ) ;
helper . m_PrecalcSina = helper . m_TransY / Zeps ( helper . m_PrecalcSqrtSumSquares ) ;
2014-07-08 03:11:14 -04:00
}
}
}
if ( m_NeedPrecalcAtanXY )
06/09/2017
--User changes
-dark.qss is now per-OS.
-Properly set/reload palette when coming from the palette editor. The latter must be done if they've modified the current palette even if they've clicked cancel.
--Bug fixes
-Make the following variations safer by using Zeps(): conic, bipolar, edisc, whorl, tan, csc, cot, tanh, sech, csch, coth, auger, bwraps, hypertile3d, hypertile3d1, ortho, poincare, rational3, barycentroid, sschecks, cscq, cschq, scry_3D, splitbrdr, hexcrop, nblur, crob.
-Fix bug enabling/disabling overwrite button in palette editor.
-Small optimization for gdoffs, use precalcAtanYX.
-Properly propagate z through circlesplit, cylinder2 and tile_log variations.
-Some values in truchet_fill could've been NaN.
--Code changes
-Make most installation files read only.
-Qualify many calls with std:: to ensure they're not colliding with glm::
-Use auto in more places.
2017-06-09 22:38:06 -04:00
helper . m_PrecalcAtanxy = std : : atan2 ( helper . m_TransX , helper . m_TransY ) ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
if ( m_NeedPrecalcAtanYX )
06/09/2017
--User changes
-dark.qss is now per-OS.
-Properly set/reload palette when coming from the palette editor. The latter must be done if they've modified the current palette even if they've clicked cancel.
--Bug fixes
-Make the following variations safer by using Zeps(): conic, bipolar, edisc, whorl, tan, csc, cot, tanh, sech, csch, coth, auger, bwraps, hypertile3d, hypertile3d1, ortho, poincare, rational3, barycentroid, sschecks, cscq, cschq, scry_3D, splitbrdr, hexcrop, nblur, crob.
-Fix bug enabling/disabling overwrite button in palette editor.
-Small optimization for gdoffs, use precalcAtanYX.
-Properly propagate z through circlesplit, cylinder2 and tile_log variations.
-Some values in truchet_fill could've been NaN.
--Code changes
-Make most installation files read only.
-Qualify many calls with std:: to ensure they're not colliding with glm::
-Use auto in more places.
2017-06-09 22:38:06 -04:00
helper . m_PrecalcAtanyx = std : : atan2 ( helper . m_TransY , helper . m_TransX ) ;
2014-07-08 03:11:14 -04:00
}
2014-09-10 01:41:26 -04:00
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
/// <summary>
2014-11-04 20:38:20 -05:00
/// Flatten this xform by adding a flatten variation if none is present, and if none of the
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
/// variations or parameters in the vector are present.
/// </summary>
2014-11-04 20:38:20 -05:00
/// <param name="names">Vector of variation and parameter names that inhibit flattening</param>
/// <returns>True if flatten was added, false if it already was present or if at least one of the specified variations or parameters were present.</returns>
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
bool Flatten ( vector < string > & names )
{
bool shouldFlatten = true ;
2016-04-13 23:59:57 -04:00
auto vl = VariationList < T > : : Instance ( ) ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
2016-01-04 19:50:15 -05:00
if ( GetVariationById ( eVariationId : : VAR_FLATTEN ) = = nullptr )
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
2015-12-31 16:41:59 -05:00
AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto var : variations )
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
if ( var - > m_Weight ! = 0 ) //This should never happen, but just to be safe.
{
2015-12-31 16:41:59 -05:00
if ( FindIf ( names , [ & ] ( const string & s ) - > bool { return ! _stricmp ( s . c_str ( ) , var - > Name ( ) . c_str ( ) ) ; } ) ) //If any variation is present, don't flatten.
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
shouldFlatten = false ;
keepGoing = false ;
break ;
}
}
//Now traverse the parameters for this variation.
2016-04-13 23:59:57 -04:00
if ( auto parVar = dynamic_cast < ParametricVariation < T > * > ( var ) ) //If any parametric variation parameter is present and non-zero, don't flatten.
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto & s : names )
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
if ( parVar - > GetParamVal ( s . c_str ( ) ) ! = 0 )
{
shouldFlatten = false ;
keepGoing = false ;
2015-05-03 20:13:14 -04:00
break ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
}
2015-05-03 20:13:14 -04:00
}
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
}
}
} ) ;
2014-11-04 20:38:20 -05:00
if ( shouldFlatten ) //Flatten was not present and neither was any variation name or parameter in the list.
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
--User changes
-Support 4k monitors, and in general, properly scale any monitor that is not HD.
-Allow for a spatial filter of radius zero, which means do not use a spatial filter.
-Add new variations: concentric, cpow3, helicoid, helix, rand_cubes, sphereblur.
-Use a new method for computing elliptic which is more precise. Developed by Discord user Claude.
-Remove the 8 variation per xform limitation on the GPU.
-Allow for loading the last flame file on startup, rather than randoms.
-Use two different default quality values in the interactive renderer, one each for CPU and GPU.
-Creating linked xforms was using non-standard behavior. Make it match Apo and also support creating multiple linked xforms at once.
--Bug fixes
-No variations in an xform used to have the same behavior as a single linear variation with weight 1. While sensible, this breaks backward compatibility. No variations now sets the output point to zeroes.
-Prevent crashing the program when adjusting a value on the main window while a final render is in progress.
-The xaos table was inverted.
--Code changes
-Convert projects to Visual Studio 2017.
-Change bad vals from +- 1e10 to +-1e20.
-Reintroduce the symmetry tag in xforms for legacy support in programs that do not use color_speed.
-Compiler will not let us use default values in templated member functions anymore.
2017-11-26 20:27:00 -05:00
auto varflatten = vl - > GetVariationCopy ( eVariationId : : VAR_FLATTEN ) ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
--User changes
-Support 4k monitors, and in general, properly scale any monitor that is not HD.
-Allow for a spatial filter of radius zero, which means do not use a spatial filter.
-Add new variations: concentric, cpow3, helicoid, helix, rand_cubes, sphereblur.
-Use a new method for computing elliptic which is more precise. Developed by Discord user Claude.
-Remove the 8 variation per xform limitation on the GPU.
-Allow for loading the last flame file on startup, rather than randoms.
-Use two different default quality values in the interactive renderer, one each for CPU and GPU.
-Creating linked xforms was using non-standard behavior. Make it match Apo and also support creating multiple linked xforms at once.
--Bug fixes
-No variations in an xform used to have the same behavior as a single linear variation with weight 1. While sensible, this breaks backward compatibility. No variations now sets the output point to zeroes.
-Prevent crashing the program when adjusting a value on the main window while a final render is in progress.
-The xaos table was inverted.
--Code changes
-Convert projects to Visual Studio 2017.
-Change bad vals from +- 1e10 to +-1e20.
-Reintroduce the symmetry tag in xforms for legacy support in programs that do not use color_speed.
-Compiler will not let us use default values in templated member functions anymore.
2017-11-26 20:27:00 -05:00
if ( ! AddVariation ( varflatten ) )
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
{
--User changes
-Support 4k monitors, and in general, properly scale any monitor that is not HD.
-Allow for a spatial filter of radius zero, which means do not use a spatial filter.
-Add new variations: concentric, cpow3, helicoid, helix, rand_cubes, sphereblur.
-Use a new method for computing elliptic which is more precise. Developed by Discord user Claude.
-Remove the 8 variation per xform limitation on the GPU.
-Allow for loading the last flame file on startup, rather than randoms.
-Use two different default quality values in the interactive renderer, one each for CPU and GPU.
-Creating linked xforms was using non-standard behavior. Make it match Apo and also support creating multiple linked xforms at once.
--Bug fixes
-No variations in an xform used to have the same behavior as a single linear variation with weight 1. While sensible, this breaks backward compatibility. No variations now sets the output point to zeroes.
-Prevent crashing the program when adjusting a value on the main window while a final render is in progress.
-The xaos table was inverted.
--Code changes
-Convert projects to Visual Studio 2017.
-Change bad vals from +- 1e10 to +-1e20.
-Reintroduce the symmetry tag in xforms for legacy support in programs that do not use color_speed.
-Compiler will not let us use default values in templated member functions anymore.
2017-11-26 20:27:00 -05:00
delete varflatten ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
return false ;
}
--User changes
-Support 4k monitors, and in general, properly scale any monitor that is not HD.
-Allow for a spatial filter of radius zero, which means do not use a spatial filter.
-Add new variations: concentric, cpow3, helicoid, helix, rand_cubes, sphereblur.
-Use a new method for computing elliptic which is more precise. Developed by Discord user Claude.
-Remove the 8 variation per xform limitation on the GPU.
-Allow for loading the last flame file on startup, rather than randoms.
-Use two different default quality values in the interactive renderer, one each for CPU and GPU.
-Creating linked xforms was using non-standard behavior. Make it match Apo and also support creating multiple linked xforms at once.
--Bug fixes
-No variations in an xform used to have the same behavior as a single linear variation with weight 1. While sensible, this breaks backward compatibility. No variations now sets the output point to zeroes.
-Prevent crashing the program when adjusting a value on the main window while a final render is in progress.
-The xaos table was inverted.
--Code changes
-Convert projects to Visual Studio 2017.
-Change bad vals from +- 1e10 to +-1e20.
-Reintroduce the symmetry tag in xforms for legacy support in programs that do not use color_speed.
-Compiler will not let us use default values in templated member functions anymore.
2017-11-26 20:27:00 -05:00
return true ;
Numerous fixes
0.4.0.5 Beta 07/18/2014
--User Changes
Allow for vibrancy values > 1.
Add flatten and unflatten menu items.
Automatically flatten like Apophysis does.
Add plugin and new_linear tags to Xml to be compatible with Apophysis.
--Bug Fixes
Fix blur, blur3d, bubble, cropn, cross, curl, curl3d, epispiral, ho,
julia3d, julia3dz, loonie, mirror_x, mirror_y, mirror_z, rotate_x,
sinusoidal, spherical, spherical3d, stripes.
Unique filename on final render was completely broken.
Two severe OpenCL bugs. Random seeds were biased and fusing was being
reset too often leading to results that differ from the CPU.
Subtle, but sometimes severe bug in the setup of the xaos weights.
Use properly defined epsilon by getting the value from
std::numeric_limits, rather than hard coding 1e-6 or 1e-10.
Omit incorrect usage of epsilon everywhere. It should not be
automatically added to denominators. Rather, it should only be used if
the denominator is zero.
Force final render progress bars to 100 on completion. Sometimes they
didn't seem to make it there.
Make variation name and params comparisons be case insensitive.
--Code Changes
Make ForEach and FindIf wrappers around std::for_each and std::find_if.
2014-07-19 02:33:18 -04:00
}
}
return false ;
}
2014-07-08 03:11:14 -04:00
/// <summary>
/// Generate the OpenCL string for reading input values to
/// be passed to a variation.
/// </summary>
/// <param name="varType">Type of the variation these values will be passed to.</param>
/// <returns>The OpenCL string</returns>
string ReadOpenCLString ( eVariationType varType )
{
string s ;
switch ( varType )
{
2016-01-04 19:50:15 -05:00
case eVariationType : : VARTYPE_REG :
case eVariationType : : VARTYPE_PRE :
2015-12-31 16:41:59 -05:00
s =
" \t vIn.x = transX; \n "
" \t vIn.y = transY; \n "
" \t vIn.z = transZ; \n " ;
break ;
2016-01-04 19:50:15 -05:00
case eVariationType : : VARTYPE_POST :
2015-12-31 16:41:59 -05:00
default :
s =
" \t vIn.x = outPoint->m_X; \n "
" \t vIn.y = outPoint->m_Y; \n "
" \t vIn.z = outPoint->m_Z; \n " ;
break ;
2014-07-08 03:11:14 -04:00
}
return s ;
}
/// <summary>
/// Assing output values from the result of a pre variation.
/// </summary>
/// <param name="helper">The helper to store the output values in</param>
/// <param name="assignType">The type of assignment this variation uses, assign or sum.</param>
inline void WritePre ( IteratorHelper < T > & helper , eVariationAssignType assignType )
{
switch ( assignType )
{
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SET :
2014-07-08 03:11:14 -04:00
{
helper . m_TransX = helper . Out . x ;
helper . m_TransY = helper . Out . y ;
helper . m_TransZ = helper . Out . z ;
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SUM :
2014-09-10 01:41:26 -04:00
default :
2014-07-08 03:11:14 -04:00
{
helper . m_TransX + = helper . Out . x ;
helper . m_TransY + = helper . Out . y ;
helper . m_TransZ + = helper . Out . z ;
break ;
}
}
}
/// <summary>
/// Assing output values from the result of a post variation.
/// </summary>
/// <param name="helper">The helper to store the output values in</param>
/// <param name="assignType">The type of assignment this variation uses, assign or sum.</param>
inline void WritePost ( IteratorHelper < T > & helper , Point < T > & outPoint , eVariationAssignType assignType )
{
switch ( assignType )
{
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SET :
2014-07-08 03:11:14 -04:00
{
outPoint . m_X = helper . Out . x ;
outPoint . m_Y = helper . Out . y ;
outPoint . m_Z = helper . Out . z ;
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SUM :
2014-09-10 01:41:26 -04:00
default :
2014-07-08 03:11:14 -04:00
{
outPoint . m_X + = helper . Out . x ;
outPoint . m_Y + = helper . Out . y ;
outPoint . m_Z + = helper . Out . z ;
break ;
}
}
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
/// <summary>
/// Generate the OpenCL string for writing output values from a call to a variation.
/// </summary>
/// <param name="varType">The type of variation these values were calculated from, pre, reg or post.</param>
/// <param name="assignType">The type of assignment used by the variation these values were calculated from, assign or sum.</param>
/// <returns>The OpenCL string</returns>
string WriteOpenCLString ( eVariationType varType , eVariationAssignType assignType )
{
string s ;
switch ( varType )
{
2016-01-04 19:50:15 -05:00
case eVariationType : : VARTYPE_REG :
2014-07-08 03:11:14 -04:00
{
s =
2015-12-31 16:41:59 -05:00
" \t outPoint->m_X += vOut.x; \n "
" \t outPoint->m_Y += vOut.y; \n "
" \t outPoint->m_Z += vOut.z; \n " ;
2014-07-08 03:11:14 -04:00
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationType : : VARTYPE_PRE :
2014-07-08 03:11:14 -04:00
{
switch ( assignType )
{
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SET :
2014-07-08 03:11:14 -04:00
{
s =
2015-12-31 16:41:59 -05:00
" \t transX = vOut.x; \n "
" \t transY = vOut.y; \n "
" \t transZ = vOut.z; \n " ;
2014-07-08 03:11:14 -04:00
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SUM :
2014-09-10 01:41:26 -04:00
default :
2014-07-08 03:11:14 -04:00
{
s =
2015-12-31 16:41:59 -05:00
" \t transX += vOut.x; \n "
" \t transY += vOut.y; \n "
" \t transZ += vOut.z; \n " ;
2014-07-08 03:11:14 -04:00
break ;
}
}
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationType : : VARTYPE_POST :
2014-09-10 01:41:26 -04:00
default :
2014-07-08 03:11:14 -04:00
{
switch ( assignType )
{
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SET :
2014-07-08 03:11:14 -04:00
{
s =
2015-12-31 16:41:59 -05:00
" \t outPoint->m_X = vOut.x; \n "
" \t outPoint->m_Y = vOut.y; \n "
" \t outPoint->m_Z = vOut.z; \n " ;
2014-07-08 03:11:14 -04:00
break ;
}
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
case eVariationAssignType : : ASSIGNTYPE_SUM :
2014-09-10 01:41:26 -04:00
default :
2014-07-08 03:11:14 -04:00
{
s =
2015-12-31 16:41:59 -05:00
" \t outPoint->m_X += vOut.x; \n "
" \t outPoint->m_Y += vOut.y; \n "
" \t outPoint->m_Z += vOut.z; \n " ;
2014-07-08 03:11:14 -04:00
break ;
}
}
break ;
}
}
return s ;
}
/// <summary>
/// Return a string representation of this xform.
/// It will include all pre affine values, and optionally post affine values if present.
/// Various variables, all variations as strings and xaos values if present.
/// </summary>
/// <returns>The string representation of this xform</returns>
string ToString ( ) const
{
ostringstream ss ;
ss < < " A: " < < m_Affine . A ( ) < < " "
< < " B: " < < m_Affine . B ( ) < < " "
< < " C: " < < m_Affine . C ( ) < < " "
< < " D: " < < m_Affine . D ( ) < < " "
< < " E: " < < m_Affine . E ( ) < < " "
2016-02-12 00:38:21 -05:00
< < " F: " < < m_Affine . F ( ) < < " \n " ;
2014-07-08 03:11:14 -04:00
if ( m_HasPost )
{
ss < < " Post A: " < < m_Post . A ( ) < < " "
< < " Post B: " < < m_Post . B ( ) < < " "
< < " Post C: " < < m_Post . C ( ) < < " "
< < " Post D: " < < m_Post . D ( ) < < " "
< < " Post E: " < < m_Post . E ( ) < < " "
2016-02-12 00:38:21 -05:00
< < " Post F: " < < m_Post . F ( ) < < " \n " ;
2014-07-08 03:11:14 -04:00
}
2016-02-12 00:38:21 -05:00
ss < < " Weight: " < < m_Weight ;
ss < < " \n ColorX: " < < m_ColorX ;
ss < < " \n ColorY: " < < m_ColorY ;
ss < < " \n Direct Color: " < < m_DirectColor ;
ss < < " \n Color Speed: " < < m_ColorSpeed ;
ss < < " \n Animate: " < < m_Animate ;
ss < < " \n Opacity: " < < m_Opacity ;
ss < < " \n Wind: " < < m_Wind [ 0 ] < < " , " < < m_Wind [ 1 ] ;
ss < < " \n Motion Frequency: " < < m_MotionFreq ;
ss < < " \n Motion Func: " < < m_MotionFunc ;
ss < < " \n Motion Offset: " < < m_MotionOffset ;
2015-12-31 16:41:59 -05:00
const_cast < Xform < T > * > ( this ) - > AllVarsFunc ( [ & ] ( vector < Variation < T > * > & variations , bool & keepGoing )
2014-07-08 03:11:14 -04:00
{
2015-05-03 20:13:14 -04:00
for ( auto var : variations )
2016-02-12 00:38:21 -05:00
ss < < var - > ToString ( ) < < " \n " ;
2014-07-08 03:11:14 -04:00
2016-02-12 00:38:21 -05:00
ss < < " \n " ;
2014-07-08 03:11:14 -04:00
} ) ;
if ( XaosPresent ( ) )
{
2015-05-03 20:13:14 -04:00
for ( auto xaos : m_Xaos )
ss < < xaos < < " " ;
2014-07-08 03:11:14 -04:00
2016-02-12 00:38:21 -05:00
ss < < " \n " ;
2014-07-08 03:11:14 -04:00
}
return ss . str ( ) ;
}
/// <summary>
/// Members are listed in the exact order they are used in Apply() to make them
/// as cache efficient as possible. Not all are public, so there is repeated public/private
/// access specifiers.
/// </summary>
private :
bool m_HasPreOrRegularVars ; //Whethere there are any pre or regular variations present.
public :
//Color coordinates for this function. This is the index into the palette used to look up a color and add to the histogram for each iter.
//The original only allows for an x coord. Will eventually allow for a y coord like Fractron for 2D palettes.
T m_ColorX , m_ColorY ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
private :
T m_ColorSpeedCache ; //Cache of m_ColorSpeed * m_ColorX. Need to recalc cache values whenever anything relating to color is set. Made private because one affects the other.
T m_OneMinusColorCache ; //Cache of 1 - m_ColorSpeedCache.
public :
//Coefficients for the affine portion of the transform.
//Discussed on page 3 of the paper:
//Fi(x, y) = (aix + biy + ci, dix + eiy + fi)
Affine2D < T > m_Affine ;
private :
vector < Variation < T > * > m_PreVariations ; //The list of pre variations to call when applying this xform.
vector < Variation < T > * > m_Variations ; //The list of variations to call when applying this xform.
2018-09-18 23:49:38 -04:00
bool m_HasPre ; //Whether a pre affine transform is present.
2014-07-08 03:11:14 -04:00
bool m_HasPost ; //Whether a post affine transform is present.
public :
//Coefficients for the affine portion of the post transform.
//Discussed on page 5 of the paper:
//Pi(x, y) = (α ix + βiy + γ i, δix + ǫiy + ζi).
Affine2D < T > m_Post ;
private :
vector < Variation < T > * > m_PostVariations ; //The list of post variations to call when applying this xform.
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
public :
T m_DirectColor ; //Used with direct color variations.
//Probability that this function is chosen. Can be greater than 1.
//Discussed on page 4 of the paper:
//Probability wi.
T m_Weight ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Scaling factor on color added to current iteration, also known as color weight. Normally defaults to 0.5.
//Discussed on page 9 of the paper with a hard coded default value of 0.5:
//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_Wind [ 2 ] ;
eMotion m_MotionFunc ;
2015-06-24 05:52:21 -04:00
T m_MotionFreq ;
2015-07-06 10:05:43 -04:00
T m_MotionOffset ;
2014-07-08 03:11:14 -04:00
vector < Xform < T > > m_Motion ;
string m_Name ;
private :
/// <summary>
/// Perform an operation on all variation vectors.
/// The operation is supplied in the func parameter.
/// To stop performing the operation on vectors after the current one,
/// set the keepGoing parameter to false;
/// </summary>
/// <param name="func">The function to call for each variation vector.</param>
void AllVarsFunc ( std : : function < void ( vector < Variation < T > * > & , bool & ) > func )
{
bool keepGoing = true ;
func ( m_PreVariations , keepGoing ) ;
if ( keepGoing )
func ( m_Variations , keepGoing ) ;
if ( keepGoing )
func ( m_PostVariations , keepGoing ) ;
}
vector < T > m_Xaos ; //Xaos vector which affects the probability that this xform is chosen. Usually empty.
Ember < T > * m_ParentEmber ; //The parent ember that contains this xform.
bool m_NeedPrecalcSumSquares ; //Whether any variation uses the precalc sum squares value in its calculations.
bool m_NeedPrecalcSqrtSumSquares ; //Whether any variation uses the sqrt precalc sum squares value in its calculations.
bool m_NeedPrecalcAngles ; //Whether any variation uses the precalc sin and cos values in its calculations.
bool m_NeedPrecalcAtanXY ; //Whether any variation uses the precalc atan XY value in its calculations.
bool m_NeedPrecalcAtanYX ; //Whether any variation uses the precalc atan YX value in its calculations.
} ;
2014-09-10 01:41:26 -04:00
}