2014-07-08 03:11:14 -04:00
# pragma once
2015-03-21 18:27:37 -04:00
# include "Curves.h"
2014-07-08 03:11:14 -04:00
# include "Xform.h"
# include "PaletteList.h"
# include "SpatialFilter.h"
# include "TemporalFilter.h"
2015-07-07 00:36:46 -04:00
# include "EmberMotion.h"
2014-07-08 03:11:14 -04:00
/// <summary>
/// Ember class.
/// </summary>
namespace EmberNs
{
2016-02-18 21:58:24 -05:00
template < typename T > class Interpolater ;
2014-07-08 03:11:14 -04:00
/// <summary>
/// Bit position specifying the presence of each type of 3D parameter.
/// One, none, some or all of these can be present.
/// </summary>
2016-01-04 19:50:15 -05:00
enum class eProjBits : et
2014-07-08 03:11:14 -04:00
{
PROJBITS_ZPOS = 1 ,
PROJBITS_PERSP = 2 ,
PROJBITS_PITCH = 4 ,
PROJBITS_YAW = 8 ,
PROJBITS_BLUR = 16 ,
} ;
/// <summary>
/// Ember is the main class for holding all of the information required to render a fractal flame.
/// This includes a vector of xforms, a final xform, size and color information as well as an Xml edit
/// document that users can use to keep track of changes.
/// Operations will often desire operating on just the regular xforms, or the regular xforms plus the final one.
/// The word "total" is used to signify when the final xform is included in the operation.
/// Template argument expected to be float or double.
/// </summary>
template < typename T >
class EMBER_API Ember
{
public :
/// <summary>
/// Default constructor which calls Init() to set default values.
/// </summary>
Ember ( )
{
2016-04-03 21:55:12 -04:00
m_Background . Reset ( ) ;
m_Curves . Init ( ) ;
m_Xforms . reserve ( 12 ) ;
m_CamMat = m3T ( 0 ) ;
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectNone ;
2014-07-08 03:11:14 -04:00
}
2014-09-10 01:41:26 -04:00
2015-01-29 21:22:15 -05:00
/// <summary>
/// Copy constructor to copy an Ember object of the same type.
/// </summary>
/// <param name="ember">The Ember object to copy</param>
2014-07-08 03:11:14 -04:00
Ember ( const Ember < T > & ember )
{
Ember < T > : : operator = < T > ( ember ) ;
}
/// <summary>
/// Copy constructor to copy an Ember object of type U, where T is usually the same type as U.
/// </summary>
/// <param name="ember">The Ember object to copy</param>
template < typename U >
Ember ( const Ember < U > & ember )
{
Ember < T > : : operator = < U > ( ember ) ;
}
/// <summary>
/// Destructor which clears the Xml edits.
/// </summary>
~ Ember ( )
{
ClearEdit ( ) ;
}
2015-01-29 21:22:15 -05:00
/// <summary>
/// Assignment operator to copy an Ember object of the same type.
/// For some reason the compiler requires Xform to define two assignment operators,
/// however it gets confused when Ember has two.
/// </summary>
/// <param name="ember">The Ember object to copy</param>
2014-07-08 03:11:14 -04:00
Ember < T > & operator = ( const Ember < T > & ember )
{
if ( this ! = & ember )
Ember < T > : : operator = < T > ( ember ) ;
return * this ;
}
/// <summary>
/// Assignment operator to assign an Ember object of type U, where T is usually the same type as U.
/// </summary>
/// <param name="ember">The Ember object to copy.</param>
/// <returns>Reference to updated self</returns>
template < typename U >
Ember < T > & operator = ( const Ember < U > & ember )
{
m_FinalRasW = ember . m_FinalRasW ;
m_FinalRasH = ember . m_FinalRasH ;
m_OrigFinalRasW = ember . m_OrigFinalRasW ;
m_OrigFinalRasH = ember . m_OrigFinalRasH ;
2016-02-13 20:24:51 -05:00
m_OrigPixPerUnit = T ( ember . m_OrigPixPerUnit ) ;
2014-11-28 04:37:51 -05:00
m_SubBatchSize = ember . m_SubBatchSize ;
m_FuseCount = ember . m_FuseCount ;
2014-07-08 03:11:14 -04:00
m_Supersample = ember . m_Supersample ;
m_TemporalSamples = ember . m_TemporalSamples ;
m_Symmetry = ember . m_Symmetry ;
m_Quality = T ( ember . m_Quality ) ;
m_PixelsPerUnit = T ( ember . m_PixelsPerUnit ) ;
m_Zoom = T ( ember . m_Zoom ) ;
m_CamZPos = T ( ember . m_CamZPos ) ;
m_CamPerspective = T ( ember . m_CamPerspective ) ;
m_CamYaw = T ( ember . m_CamYaw ) ;
m_CamPitch = T ( ember . m_CamPitch ) ;
m_CamDepthBlur = T ( ember . m_CamDepthBlur ) ;
m_CamMat = ember . m_CamMat ;
m_CenterX = T ( ember . m_CenterX ) ;
2014-09-10 01:41:26 -04:00
m_CenterY = T ( ember . m_CenterY ) ;
2014-10-27 17:21:06 -04:00
m_RotCenterY = T ( ember . m_RotCenterY ) ;
2014-07-08 03:11:14 -04:00
m_Rotate = T ( ember . m_Rotate ) ;
m_Brightness = T ( ember . m_Brightness ) ;
m_Gamma = T ( ember . m_Gamma ) ;
m_Vibrancy = T ( ember . m_Vibrancy ) ;
m_GammaThresh = T ( ember . m_GammaThresh ) ;
m_HighlightPower = T ( ember . m_HighlightPower ) ;
m_Time = T ( ember . m_Time ) ;
m_Background = ember . m_Background ;
m_Interp = ember . m_Interp ;
m_AffineInterp = ember . m_AffineInterp ;
m_MinRadDE = T ( ember . m_MinRadDE ) ;
m_MaxRadDE = T ( ember . m_MaxRadDE ) ;
m_CurveDE = T ( ember . m_CurveDE ) ;
m_SpatialFilterType = ember . m_SpatialFilterType ;
m_SpatialFilterRadius = T ( ember . m_SpatialFilterRadius ) ;
m_TemporalFilterType = ember . m_TemporalFilterType ;
m_TemporalFilterExp = T ( ember . m_TemporalFilterExp ) ;
m_TemporalFilterWidth = T ( ember . m_TemporalFilterWidth ) ;
m_PaletteMode = ember . m_PaletteMode ;
m_PaletteInterp = ember . m_PaletteInterp ;
m_Name = ember . m_Name ;
m_ParentFilename = ember . m_ParentFilename ;
m_Index = ember . m_Index ;
2014-10-14 11:53:15 -04:00
m_ScaleType = ember . ScaleType ( ) ;
2014-07-08 03:11:14 -04:00
m_Palette = ember . m_Palette ;
2015-03-21 18:27:37 -04:00
m_Curves = ember . m_Curves ;
2014-07-08 03:11:14 -04:00
m_Xforms . clear ( ) ;
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < ember . XformCount ( ) ; i + + )
2014-07-08 03:11:14 -04:00
{
2014-08-03 19:16:10 -04:00
if ( Xform < U > * p = ember . GetXform ( i ) )
{
Xform < T > xform = * p ; //Will call assignment operator to convert between types T and U.
AddXform ( xform ) ;
}
2014-07-08 03:11:14 -04:00
}
Xform < T > finalXform = * ember . FinalXform ( ) ; //Will call assignment operator to convert between types T and U.
SetFinalXform ( finalXform ) ;
//Interpolated-against final xforms need animate & color speed set to 0.
if ( ! ember . UseFinalXform ( ) )
{
m_FinalXform . m_Motion . clear ( ) ;
m_FinalXform . m_Animate = 0 ;
m_FinalXform . m_ColorSpeed = 0 ;
}
SetProjFunc ( ) ;
ClearEdit ( ) ;
2014-09-10 01:41:26 -04:00
2015-10-27 00:31:35 -04:00
if ( ember . m_Edits )
2014-07-08 03:11:14 -04:00
m_Edits = xmlCopyDoc ( ember . m_Edits , 1 ) ;
2016-04-03 21:55:12 -04:00
CopyCont ( m_EmberMotionElements , ember . m_EmberMotionElements ) ;
2014-07-08 03:11:14 -04:00
return * this ;
}
/// <summary>
/// Add a copy of a new xform to the xforms vector.
/// </summary>
/// <param name="xform">The xform to copy and add</param>
void AddXform ( const Xform < T > & xform )
{
m_Xforms . push_back ( xform ) ;
m_Xforms [ m_Xforms . size ( ) - 1 ] . CacheColorVals ( ) ;
m_Xforms [ m_Xforms . size ( ) - 1 ] . ParentEmber ( this ) ;
}
/// <summary>
/// Add the specified number of empty xforms.
/// </summary>
/// <param name="count">The number of xforms to add</param>
2014-10-14 11:53:15 -04:00
void AddXforms ( size_t count )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < count ; i + + )
2014-07-08 03:11:14 -04:00
{
Xform < T > xform ;
AddXform ( xform ) ;
}
}
/// <summary>
/// Add empty padding xforms until the total xform count is xformPad.
/// </summary>
/// <param name="xformPad">The total number of xforms to finish with</param>
2014-10-14 11:53:15 -04:00
void PadXforms ( size_t xformPad )
2014-07-08 03:11:14 -04:00
{
if ( xformPad > XformCount ( ) )
AddXforms ( xformPad - XformCount ( ) ) ;
}
/// <summary>
/// Copy this ember with optional padding xforms added.
/// </summary>
/// <param name="xformPad">The total number of xforms if additional padding xforms are desired. Default: 0.</param>
/// <param name="doFinal">Whether to copy the final xform. Default: false.</param>
/// <returns>The newly constructed ember</returns>
2016-04-03 21:55:12 -04:00
Ember < T > Copy ( size_t xformPad = 0 , bool doFinal = false ) const
2014-07-08 03:11:14 -04:00
{
Ember < T > ember ( * this ) ;
ember . PadXforms ( xformPad ) ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
if ( doFinal )
{
if ( UseFinalXform ( ) ) //Caller wanted one and this ember has one.
{
ember . m_FinalXform = m_FinalXform ;
}
else //Caller wanted one and this ember doesn't have one.
{
//Interpolated-against final xforms need animate & color speed set to 0 and motion elements cleared.
ember . m_FinalXform . m_Animate = 0 ;
ember . m_FinalXform . m_ColorSpeed = 0 ;
ember . m_FinalXform . m_Motion . clear ( ) ;
ember . m_FinalXform . ClearAndDeleteVariations ( ) ;
2016-04-13 23:59:57 -04:00
ember . m_FinalXform . AddVariation ( m_VariationList - > GetVariationCopy ( eVariationId : : VAR_LINEAR , 0 ) ) ; //Do this so it doesn't appear empty.
2014-07-08 03:11:14 -04:00
}
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
return ember ;
}
/// <summary>
/// Delete an xform at the specified index.
/// Shuffle xaos if present to reflect the deleted xform.
/// </summary>
/// <param name="i">The index to delete</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool DeleteXform ( size_t i )
2014-07-08 03:11:14 -04:00
{
if ( i < XformCount ( ) )
{
m_Xforms . erase ( m_Xforms . begin ( ) + i ) ;
//Now shuffle xaos values from i on back by 1 for every xform.
2014-10-14 11:53:15 -04:00
for ( size_t x1 = 0 ; x1 < XformCount ( ) ; x1 + + )
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
if ( auto xform = GetXform ( x1 ) )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t x2 = i + 1 ; x2 < = XformCount ( ) ; x2 + + ) //Iterate from the position after the deletion index up to the old count.
2014-07-08 03:11:14 -04:00
xform - > SetXaos ( x2 - 1 , xform - > Xaos ( x2 ) ) ;
xform - > TruncateXaos ( ) ; //Make sure no old values are hanging around in case more xforms are added to this ember later.
}
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
return true ;
}
return false ;
}
/// <summary>
/// Delete the xform at the specified index, including the final one.
/// </summary>
/// <param name="i">The index to delete</param>
/// <returns>True if success, else false.</returns>
2014-10-14 11:53:15 -04:00
bool DeleteTotalXform ( size_t i )
2014-07-08 03:11:14 -04:00
{
if ( DeleteXform ( i ) )
2015-12-31 16:41:59 -05:00
{ }
2014-07-08 03:11:14 -04:00
else if ( i = = XformCount ( ) & & UseFinalXform ( ) )
m_FinalXform . Clear ( ) ;
else
return false ;
return true ;
}
/// <summary>
/// Get a pointer to the xform at the specified index, excluding the final one.
/// </summary>
/// <param name="i">The index to get</param>
2014-09-10 01:41:26 -04:00
/// <returns>A pointer to the xform at the index if successful, else nullptr.</returns>
2014-10-14 11:53:15 -04:00
Xform < T > * GetXform ( size_t i ) const
2014-07-08 03:11:14 -04:00
{
if ( i < XformCount ( ) )
2014-12-07 02:51:44 -05:00
return const_cast < Xform < T > * > ( & m_Xforms [ i ] ) ;
2014-07-08 03:11:14 -04:00
else
2014-09-10 01:41:26 -04:00
return nullptr ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Get a pointer to the xform at the specified index, including the final one.
/// </summary>
/// <param name="i">The index to get</param>
/// <param name="forceFinal">If true, return the final xform when its index is requested even if one is not present</param>
2014-09-10 01:41:26 -04:00
/// <returns>A pointer to the xform at the index if successful, else nullptr.</returns>
2014-10-14 11:53:15 -04:00
Xform < T > * GetTotalXform ( size_t i , bool forceFinal = false ) const
2014-07-08 03:11:14 -04:00
{
if ( i < XformCount ( ) )
2014-12-07 02:51:44 -05:00
return const_cast < Xform < T > * > ( & m_Xforms [ i ] ) ;
2014-07-08 03:11:14 -04:00
else if ( i = = XformCount ( ) | | forceFinal )
2014-12-07 02:51:44 -05:00
return const_cast < Xform < T > * > ( & m_FinalXform ) ;
2014-07-08 03:11:14 -04:00
else
2014-09-10 01:41:26 -04:00
return nullptr ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Search the xforms, excluding final, to find which one's address matches the address of the specified xform.
/// </summary>
/// <param name="xform">A pointer to the xform to find</param>
/// <returns>The index of the matched xform if found, else -1.</returns>
2014-10-14 11:53:15 -04:00
intmax_t GetXformIndex ( Xform < T > * xform ) const
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
intmax_t index = - 1 ;
2014-07-08 03:11:14 -04:00
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < m_Xforms . size ( ) ; i + + )
2014-07-08 03:11:14 -04:00
if ( GetXform ( i ) = = xform )
2014-12-07 02:51:44 -05:00
return intmax_t ( i ) ;
2014-07-08 03:11:14 -04:00
return index ;
}
/// <summary>
/// Search the xforms, including final, to find which one's address matches the address of the specified xform.
/// </summary>
/// <param name="xform">A pointer to the xform to find</param>
/// <returns>The index of the matched xform if found, else -1.</returns>
2014-10-14 11:53:15 -04:00
intmax_t GetTotalXformIndex ( Xform < T > * xform ) const
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
size_t totalXformCount = TotalXformCount ( ) ;
2014-09-10 01:41:26 -04:00
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < totalXformCount ; i + + )
2014-07-08 03:11:14 -04:00
if ( GetTotalXform ( i ) = = xform )
2014-12-07 02:51:44 -05:00
return intmax_t ( i ) ;
2014-07-08 03:11:14 -04:00
return - 1 ;
}
/// <summary>
/// Assign the final xform.
/// </summary>
/// <param name="xform">The xform to copy and assign to the final xform</param>
void SetFinalXform ( const Xform < T > & xform )
{
m_FinalXform = xform ;
m_FinalXform . CacheColorVals ( ) ;
m_FinalXform . ParentEmber ( this ) ;
}
/// <summary>
/// Delete the final xform.
/// </summary>
void DeleteFinalXform ( )
{
m_FinalXform . ClearAndDeleteVariations ( ) ;
}
/// <summary>
/// Determine whether the specified xform has the same address as the final xform.
/// </summary>
/// <param name="xform">A pointer to the xform to test</param>
/// <returns>True if matched, else false.</returns>
2016-04-03 21:55:12 -04:00
bool IsFinalXform ( Xform < T > * xform ) const
2014-07-08 03:11:14 -04:00
{
return & m_FinalXform = = xform ;
}
/// <summary>
/// Delete all motion elements from all xforms including final.
/// </summary>
void DeleteMotionElements ( )
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < TotalXformCount ( ) ; i + + )
2014-07-08 03:11:14 -04:00
GetTotalXform ( i ) - > DeleteMotionElements ( ) ;
2015-07-02 13:01:37 -04:00
2015-07-07 00:36:46 -04:00
m_EmberMotionElements . clear ( ) ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Call CacheColorVals() and SetPrecalcFlags() on all xforms including final.
/// </summary>
void CacheXforms ( )
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < TotalXformCount ( ) ; i + + )
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
auto xform = GetTotalXform ( i ) ;
2014-07-08 03:11:14 -04:00
xform - > CacheColorVals ( ) ;
xform - > SetPrecalcFlags ( ) ;
}
}
2014-10-14 11:53:15 -04:00
/// <summary>
/// Set the projection function pointer based on the
/// values of the 3D fields.
/// </summary>
2014-07-08 03:11:14 -04:00
void SetProjFunc ( )
{
2014-10-14 11:53:15 -04:00
size_t projBits = ProjBits ( ) ;
2014-07-08 03:11:14 -04:00
if ( ! projBits ) //No 3D at all, then do nothing.
{
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectNone ;
}
else
{
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
m_CamMat [ 0 ] [ 0 ] = std : : cos ( - m_CamYaw ) ;
m_CamMat [ 1 ] [ 0 ] = - std : : sin ( - m_CamYaw ) ;
2014-07-08 03:11:14 -04:00
m_CamMat [ 2 ] [ 0 ] = 0 ;
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
m_CamMat [ 0 ] [ 1 ] = std : : cos ( m_CamPitch ) * std : : sin ( - m_CamYaw ) ;
m_CamMat [ 1 ] [ 1 ] = std : : cos ( m_CamPitch ) * std : : cos ( - m_CamYaw ) ;
m_CamMat [ 2 ] [ 1 ] = - std : : sin ( m_CamPitch ) ;
m_CamMat [ 0 ] [ 2 ] = std : : sin ( m_CamPitch ) * std : : sin ( - m_CamYaw ) ;
m_CamMat [ 1 ] [ 2 ] = std : : sin ( m_CamPitch ) * std : : cos ( - m_CamYaw ) ;
m_CamMat [ 2 ] [ 2 ] = std : : cos ( m_CamPitch ) ;
2014-07-08 03:11:14 -04:00
2016-01-04 19:50:15 -05:00
if ( projBits & et ( eProjBits : : PROJBITS_BLUR ) )
2014-07-08 03:11:14 -04:00
{
2016-01-04 19:50:15 -05:00
if ( projBits & et ( eProjBits : : PROJBITS_YAW ) )
2014-07-08 03:11:14 -04:00
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectPitchYawDepthBlur ;
else
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectPitchDepthBlur ;
}
2016-01-04 19:50:15 -05:00
else if ( ( projBits & et ( eProjBits : : PROJBITS_PITCH ) ) | | ( projBits & et ( eProjBits : : PROJBITS_YAW ) ) )
2014-07-08 03:11:14 -04:00
{
2016-01-04 19:50:15 -05:00
if ( projBits & et ( eProjBits : : PROJBITS_YAW ) )
2014-07-08 03:11:14 -04:00
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectPitchYaw ;
else
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectPitch ;
}
else
{
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectZPerspective ;
}
}
m_BlurCoef = T ( 0.1 ) * m_CamDepthBlur ;
}
/// <summary>
/// Determine whether xaos is used in any xform, excluding final.
/// </summary>
/// <returns>True if any xaos found, else false.</returns>
2014-10-14 11:53:15 -04:00
bool XaosPresent ( ) const
2014-07-08 03:11:14 -04:00
{
bool b = false ;
2015-05-03 20:13:14 -04:00
for ( auto & xform : m_Xforms ) b | = xform . XaosPresent ( ) ; //If at least one entry is not equal to 1, then xaos is present.
2014-07-08 03:11:14 -04:00
return b ;
}
/// <summary>
/// Remove all xaos from this ember.
/// </summary>
void ClearXaos ( )
{
2015-05-03 20:13:14 -04:00
for ( auto & xform : m_Xforms ) xform . ClearXaos ( ) ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Change the size of the final output image and adjust the pixels per unit
/// so that the orientation of the image remains the same in the new size.
/// </summary>
/// <param name="width">New width</param>
/// <param name="height">New height</param>
/// <param name="onlyScaleIfNewIsSmaller">True to only scale if the new dimensions are smaller than the original, else always scale.</param>
2014-10-14 11:53:15 -04:00
void SetSizeAndAdjustScale ( size_t width , size_t height , bool onlyScaleIfNewIsSmaller , eScaleType scaleType )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
if ( ( onlyScaleIfNewIsSmaller & & ( width < m_OrigFinalRasW | | height < m_OrigFinalRasH ) ) | | ! onlyScaleIfNewIsSmaller )
2014-07-08 03:11:14 -04:00
{
2015-12-31 19:00:36 -05:00
if ( scaleType = = eScaleType : : SCALE_WIDTH )
2014-10-14 11:53:15 -04:00
m_PixelsPerUnit = m_OrigPixPerUnit * ( T ( width ) / T ( m_OrigFinalRasW ) ) ;
2015-12-31 19:00:36 -05:00
else if ( scaleType = = eScaleType : : SCALE_HEIGHT )
2014-10-14 11:53:15 -04:00
m_PixelsPerUnit = m_OrigPixPerUnit * ( T ( height ) / T ( m_OrigFinalRasH ) ) ;
2014-07-08 03:11:14 -04:00
}
2014-10-14 11:53:15 -04:00
m_ScaleType = scaleType ;
2014-07-08 03:11:14 -04:00
m_FinalRasW = width ;
m_FinalRasH = height ;
}
2014-10-14 11:53:15 -04:00
/// <summary>
/// Set the original final output image dimensions to be equal to the current ones.
/// </summary>
void SyncSize ( )
{
m_OrigFinalRasW = m_FinalRasW ;
m_OrigFinalRasH = m_FinalRasH ;
m_OrigPixPerUnit = m_PixelsPerUnit ;
}
/// <summary>
/// Set the current final output image dimensions to be equal to the original ones.
/// </summary>
void RestoreSize ( )
{
m_FinalRasW = m_OrigFinalRasW ;
m_FinalRasH = m_OrigFinalRasH ;
m_PixelsPerUnit = m_OrigPixPerUnit ;
}
2014-07-08 03:11:14 -04:00
/// <summary>
/// Set all xform weights to 1 / xform count.
/// </summary>
void EqualizeWeights ( )
{
T weight = T ( 1 ) / m_Xforms . size ( ) ;
2015-05-03 20:13:14 -04:00
for ( auto & xform : m_Xforms ) xform . m_Weight = weight ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Calculates the normalized weights of the xforms and places them in the passed in vector.
/// <param name="normalizedWeights">A vector to hold the normalized weights</param>
/// </summary>
void CalcNormalizedWeights ( vector < T > & normalizedWeights )
{
T norm = 0 ;
size_t i = 0 ;
if ( normalizedWeights . size ( ) ! = m_Xforms . size ( ) )
normalizedWeights . resize ( m_Xforms . size ( ) ) ;
2015-05-03 20:13:14 -04:00
for ( auto & xform : m_Xforms ) norm + = xform . m_Weight ;
2015-12-31 16:41:59 -05:00
2015-05-03 20:13:14 -04:00
for ( auto & weight : normalizedWeights ) { weight = ( norm = = T ( 0 ) ? T ( 0 ) : m_Xforms [ i ] . m_Weight / norm ) ; i + + ; }
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Get a vector of pointers to all variations present in all Xforms, with no duplicates.
/// Meaning, that if a variation is present in more than one Xform, only the first one encountered
/// will be added.
/// <param name="variations">A vector to hold pointers to the variations present. This will be cleared first.</param>
/// </summary>
void GetPresentVariations ( vector < Variation < T > * > & variations , bool baseOnly = true ) const
{
2014-10-14 11:53:15 -04:00
size_t i = 0 , xformIndex = 0 , totalVarCount = m_FinalXform . TotalVariationCount ( ) ;
2014-07-08 03:11:14 -04:00
variations . clear ( ) ;
2015-05-03 20:13:14 -04:00
2015-12-31 16:41:59 -05:00
while ( auto xform = GetTotalXform ( xformIndex + + ) )
totalVarCount + = xform - > TotalVariationCount ( ) ;
xformIndex = 0 ;
2014-07-08 03:11:14 -04:00
variations . reserve ( totalVarCount ) ;
2015-12-31 16:41:59 -05:00
while ( auto xform = GetTotalXform ( xformIndex + + ) )
2014-07-08 03:11:14 -04:00
{
i = 0 ;
totalVarCount = xform - > TotalVariationCount ( ) ;
while ( Variation < T > * tempVar = xform - > GetVariation ( i + + ) )
{
if ( baseOnly )
{
string tempVarBaseName = tempVar - > BaseName ( ) ;
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 ( ! FindIf ( variations , [ & ] ( const Variation < T > * var ) - > bool { return tempVar - > VariationId ( ) = = var - > VariationId ( ) ; } ) & &
2015-12-31 16:41:59 -05:00
! FindIf ( variations , [ & ] ( const Variation < T > * var ) - > bool { return tempVarBaseName = = var - > BaseName ( ) ; } ) )
2014-07-08 03:11:14 -04:00
variations . push_back ( tempVar ) ;
}
else
{
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 ( ! FindIf ( variations , [ & ] ( const Variation < T > * var ) - > bool { return tempVar - > VariationId ( ) = = var - > VariationId ( ) ; } ) )
2014-07-08 03:11:14 -04:00
variations . push_back ( tempVar ) ;
}
}
}
}
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>
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
/// Flatten all xforms by adding a flatten variation if none is present, and if none of the
/// variations or parameters in the vector are not present.
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>
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
/// <param name="names">Vector of variation and parameter names that inhibit flattening</param>
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
/// <returns>True if flatten was added to any of the xforms, false if it already was present or if none of the specified variations or parameters were present.</returns>
bool Flatten ( vector < string > & names )
{
bool flattened = false ;
2015-12-31 16:41:59 -05:00
for ( auto & xform : m_Xforms )
flattened | = xform . Flatten ( names ) ;
if ( UseFinalXform ( ) )
flattened | = m_FinalXform . Flatten ( 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
return flattened ;
}
/// <summary>
2015-12-31 16:41:59 -05:00
/// Unflatten all xforms by removing flatten, pre_flatten and post_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
/// </summary>
2015-12-31 16:41:59 -05:00
/// <returns>True if any flatten was removed, false if it wasn't 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 Unflatten ( )
{
2015-12-31 16:41:59 -05:00
size_t xformIndex = 0 ;
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 unflattened = false ;
2015-12-31 16:41:59 -05:00
while ( auto xform = GetTotalXform ( xformIndex + + ) )
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
unflattened | = xform - > DeleteVariationById ( eVariationId : : VAR_PRE_FLATTEN ) ;
unflattened | = xform - > DeleteVariationById ( eVariationId : : VAR_FLATTEN ) ;
unflattened | = xform - > DeleteVariationById ( eVariationId : : VAR_POST_FLATTEN ) ;
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
return unflattened ;
}
2014-07-08 03:11:14 -04:00
/// <summary>
/// Thin wrapper around Interpolate() which takes a vector of embers rather than a pointer and size.
/// All embers are expected to be aligned, including the final xform. If not the behavior is undefined.
/// </summary>
/// <param name="embers">Vector of embers</param>
/// <param name="coefs">Coefficients vector which must be the same length as the vector of embers</param>
/// <param name="stagger">Stagger if greater than 0</param>
2016-04-03 21:55:12 -04:00
void Interpolate ( const vector < Ember < T > > & embers , vector < T > & coefs , T stagger )
2014-07-08 03:11:14 -04:00
{
Interpolate ( embers . data ( ) , embers . size ( ) , coefs , stagger ) ;
}
/// <summary>
/// Interpolate the specified embers using the coefficients supplied.
/// All embers are expected to be aligned, including the final xform. If not the behavior is undefined.
/// </summary>
/// <param name="embers">Pointer to buffer of embers</param>
/// <param name="size">Number of elements in the buffer of embers</param>
/// <param name="coefs">Coefficients vector which must be the same length as the vector of embers</param>
/// <param name="stagger">Stagger if greater than 0</param>
2016-04-03 21:55:12 -04:00
void Interpolate ( const Ember < T > * embers , size_t size , vector < T > & coefs , T stagger )
2014-07-08 03:11:14 -04:00
{
if ( size ! = coefs . size ( ) | | size < 2 )
return ;
bool allID , final ;
2014-10-14 11:53:15 -04:00
size_t l , maxXformCount , totalXformCount ;
2014-07-08 03:11:14 -04:00
T bgAlphaSave = m_Background . a ;
2014-09-10 01:41:26 -04:00
T coefSave [ 2 ] { 0 , 0 } ;
2014-07-08 03:11:14 -04:00
vector < Xform < T > * > xformVec ;
2014-09-10 01:41:26 -04:00
//Palette and others
2015-12-31 19:00:36 -05:00
if ( embers [ 0 ] . m_PaletteInterp = = ePaletteInterp : : INTERP_HSV )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( glm : : length_t i = 0 ; i < 256 ; i + + )
2014-07-08 03:11:14 -04:00
{
T t [ 3 ] , s [ 4 ] = { 0 , 0 , 0 , 0 } ;
2014-09-10 01:41:26 -04:00
2014-10-14 11:53:15 -04:00
for ( glm : : length_t k = 0 ; k < size ; k + + )
2014-07-08 03:11:14 -04:00
{
Palette < T > : : RgbToHsv ( glm : : value_ptr ( embers [ k ] . m_Palette [ i ] ) , t ) ;
2014-10-14 11:53:15 -04:00
for ( size_t j = 0 ; j < 3 ; j + + )
2014-07-08 03:11:14 -04:00
s [ j ] + = coefs [ k ] * t [ j ] ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
s [ 3 ] + = coefs [ k ] * embers [ k ] . m_Palette [ i ] [ 3 ] ;
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
Palette < T > : : HsvToRgb ( s , glm : : value_ptr ( m_Palette [ i ] ) ) ;
m_Palette [ i ] [ 3 ] = s [ 3 ] ;
2014-09-10 01:41:26 -04:00
2014-10-14 11:53:15 -04:00
for ( glm : : length_t j = 0 ; j < 4 ; j + + )
2014-07-08 03:11:14 -04:00
Clamp < T > ( m_Palette [ i ] [ j ] , 0 , 1 ) ;
}
}
2015-12-31 19:00:36 -05:00
else if ( embers [ 0 ] . m_PaletteInterp = = ePaletteInterp : : INTERP_SWEEP )
2014-07-08 03:11:14 -04:00
{
//Sweep - not the best option for float indices.
2014-10-14 11:53:15 -04:00
for ( glm : : length_t i = 0 ; i < 256 ; i + + )
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
size_t j = ( i < ( 256 * coefs [ 0 ] ) ) ? 0 : 1 ;
2014-07-08 03:11:14 -04:00
m_Palette [ i ] = embers [ j ] . m_Palette [ i ] ;
}
}
m_Palette . m_Index = - 1 ; //Random.
m_Symmetry = 0 ;
m_SpatialFilterType = embers [ 0 ] . m_SpatialFilterType ;
m_TemporalFilterType = embers [ 0 ] . m_TemporalFilterType ;
m_PaletteMode = embers [ 0 ] . m_PaletteMode ;
m_AffineInterp = embers [ 0 ] . m_AffineInterp ;
2014-11-28 04:37:51 -05:00
//Interpolate ember parameters, these should be in the same order the members are declared.
2014-07-08 03:11:14 -04:00
InterpI < & Ember < T > : : m_FinalRasW > ( embers , coefs , size ) ;
InterpI < & Ember < T > : : m_FinalRasH > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpI < & Ember < T > : : m_SubBatchSize > ( embers , coefs , size ) ;
InterpI < & Ember < T > : : m_FuseCount > ( embers , coefs , size ) ;
2014-07-08 03:11:14 -04:00
InterpI < & Ember < T > : : m_Supersample > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpI < & Ember < T > : : m_TemporalSamples > ( embers , coefs , size ) ;
2014-07-08 03:11:14 -04:00
InterpT < & Ember < T > : : m_Quality > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpT < & Ember < T > : : m_PixelsPerUnit > ( embers , coefs , size ) ;
2014-07-08 03:11:14 -04:00
InterpT < & Ember < T > : : m_Zoom > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CamZPos > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CamPerspective > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CamYaw > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CamPitch > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CamDepthBlur > ( embers , coefs , size ) ;
InterpX < m3T , & Ember < T > : : m_CamMat > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpT < & Ember < T > : : m_CenterX > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CenterY > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_RotCenterY > ( embers , coefs , size ) ;
2014-07-08 03:11:14 -04:00
InterpT < & Ember < T > : : m_Rotate > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpT < & Ember < T > : : m_Brightness > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_Gamma > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_Vibrancy > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_GammaThresh > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_HighlightPower > ( embers , coefs , size ) ;
InterpX < Color < T > , & Ember < T > : : m_Background > ( embers , coefs , size ) ; m_Background . a = bgAlphaSave ; //Don't interp alpha.
InterpT < & Ember < T > : : m_TemporalFilterExp > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_TemporalFilterWidth > ( embers , coefs , size ) ;
2014-07-08 03:11:14 -04:00
InterpT < & Ember < T > : : m_MaxRadDE > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_MinRadDE > ( embers , coefs , size ) ;
InterpT < & Ember < T > : : m_CurveDE > ( embers , coefs , size ) ;
2014-11-28 04:37:51 -05:00
InterpT < & Ember < T > : : m_SpatialFilterRadius > ( embers , coefs , size ) ;
2015-03-21 18:27:37 -04:00
InterpX < Curves < T > , & Ember < T > : : m_Curves > ( embers , coefs , size ) ;
2015-06-20 14:35:08 -04:00
//Normally done in assignment, must manually do here.
SetProjFunc ( ) ;
2014-07-08 03:11:14 -04:00
//An extra step needed here due to the OOD that was not needed in the original.
//A small price to pay for the conveniences it affords us elsewhere.
//First clear the xforms, and find out the max number of xforms in all of the embers in the list.
m_Xforms . clear ( ) ;
maxXformCount = Interpolater < T > : : MaxXformCount ( embers , size ) ; //Max number of standard transforms in embers, excluding final.
//m_Xforms.reserve(maxXformCount);
final = Interpolater < T > : : AnyFinalPresent ( embers , size ) ; //Did any embers have a final xform?
totalXformCount = maxXformCount + ( final ? 1 : 0 ) ;
xformVec . reserve ( size ) ;
//Populate the xform list member such that each element is a merge of all of the xforms at that position in all of the embers.
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < totalXformCount ; i + + ) //For each xform to populate.
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
for ( size_t j = 0 ; j < size ; j + + ) //For each ember in the list.
2014-07-08 03:11:14 -04:00
{
if ( i < embers [ j ] . TotalXformCount ( ) ) //Xform in this position in this ember.
{
xformVec . push_back ( embers [ j ] . GetTotalXform ( i ) ) ; //Temporary list to pass to MergeXforms().
}
}
if ( i < maxXformCount ) //Working with standard xforms?
AddXform ( Interpolater < T > : : MergeXforms ( xformVec , true ) ) ; //Merge, set weights to zero, and add to the xform list.
else if ( final ) //Or is it the final xform (i will be == maxXformCount)?
m_FinalXform = Interpolater < T > : : MergeXforms ( xformVec , true ) ;
xformVec . clear ( ) ;
}
//Now have a merged list, so interpolate the weight values.
//This includes all xforms plus final.
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < totalXformCount ; i + + )
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
auto thisXform = GetTotalXform ( i ) ;
2014-07-08 03:11:14 -04:00
if ( size = = 2 & & stagger > 0 & & thisXform ! = & m_FinalXform )
{
coefSave [ 0 ] = coefs [ 0 ] ;
coefSave [ 1 ] = coefs [ 1 ] ;
coefs [ 0 ] = Interpolater < T > : : GetStaggerCoef ( coefSave [ 0 ] , stagger , XformCount ( ) , i ) ; //Standard xform count without final.
coefs [ 1 ] = 1 - coefs [ 0 ] ;
}
2014-10-14 11:53:15 -04:00
for ( size_t j = 0 ; j < thisXform - > TotalVariationCount ( ) ; j + + ) //For each variation in this xform.
2014-07-08 03:11:14 -04:00
{
Variation < T > * var = thisXform - > GetVariation ( j ) ;
ParametricVariation < T > * parVar = dynamic_cast < ParametricVariation < T > * > ( var ) ; //Will use below if it's parametric.
var - > m_Weight = 0 ;
2015-10-27 00:31:35 -04:00
if ( parVar )
2014-07-08 03:11:14 -04:00
parVar - > Clear ( ) ;
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < size ; k + + ) //For each ember in the list.
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
auto tempXform = embers [ k ] . GetTotalXform ( i ) ; //Xform in this position in this ember, including final.
2014-07-08 03:11:14 -04:00
if ( tempXform )
{
Variation < T > * tempVar = tempXform - > GetVariationById ( var - > VariationId ( ) ) ; //See if the variation at this xform index exists in that ember at this xform index.
2015-10-27 00:31:35 -04:00
if ( tempVar )
2014-07-08 03:11:14 -04:00
{
//Interp weight.
var - > m_Weight + = tempVar - > m_Weight * coefs [ k ] ;
//If it was a parametric variation, interp params.
2015-10-27 00:31:35 -04:00
if ( parVar )
2014-07-08 03:11:14 -04:00
{
ParametricVariation < T > * tempParVar = dynamic_cast < ParametricVariation < T > * > ( tempVar ) ;
2015-10-27 00:31:35 -04:00
if ( tempParVar & & ( parVar - > ParamCount ( ) = = tempParVar - > ParamCount ( ) ) ) //This check will should always be true, but just check to be absolutely sure to avoid clobbering memory.
2014-07-08 03:11:14 -04:00
{
2015-06-28 20:48:26 -04:00
auto params = parVar - > Params ( ) ;
auto tempParams = tempParVar - > Params ( ) ;
2014-07-08 03:11:14 -04:00
for ( l = 0 ; l < parVar - > ParamCount ( ) ; l + + )
{
if ( ! tempParams [ l ] . IsPrecalc ( ) )
* ( params [ l ] . Param ( ) ) + = tempParams [ l ] . ParamVal ( ) * coefs [ k ] ;
}
}
}
}
}
}
}
InterpXform < & Xform < T > : : m_Weight > ( thisXform , i , embers , coefs , size ) ;
InterpXform < & Xform < T > : : m_ColorX > ( thisXform , i , embers , coefs , size ) ;
InterpXform < & Xform < T > : : m_ColorSpeed > ( thisXform , i , embers , coefs , size ) ;
InterpXform < & Xform < T > : : m_Opacity > ( thisXform , i , embers , coefs , size ) ;
InterpXform < & Xform < T > : : m_Animate > ( thisXform , i , embers , coefs , size ) ;
ClampGte0Ref ( thisXform - > m_Weight ) ;
ClampRef < T > ( thisXform - > m_ColorX , 0 , 1 ) ;
ClampRef < T > ( thisXform - > m_ColorSpeed , - 1 , 1 ) ;
//Interp affine and post.
2015-12-31 19:00:36 -05:00
if ( m_AffineInterp = = eAffineInterp : : AFFINE_INTERP_LOG )
2014-07-08 03:11:14 -04:00
{
vector < v2T > cxMag ( size ) ;
vector < v2T > cxAng ( size ) ;
vector < v2T > cxTrn ( size ) ;
thisXform - > m_Affine . m_Mat = m23T ( 0 ) ;
//Affine part.
Interpolater < T > : : ConvertLinearToPolar ( embers , size , i , 0 , cxAng , cxMag , cxTrn ) ;
Interpolater < T > : : InterpAndConvertBack ( coefs , cxAng , cxMag , cxTrn , thisXform - > m_Affine ) ;
//Post part.
allID = true ;
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < size ; k + + ) //For each ember in the list.
2014-07-08 03:11:14 -04:00
{
if ( i < embers [ k ] . TotalXformCount ( ) ) //Xform in this position in this ember.
{
2016-02-02 20:51:58 -05:00
auto tempXform = embers [ k ] . GetTotalXform ( i ) ;
2014-07-08 03:11:14 -04:00
allID & = tempXform - > m_Post . IsID ( ) ;
}
}
thisXform - > m_Post . m_Mat = m23T ( 0 ) ;
if ( allID )
{
thisXform - > m_Post . A ( 1 ) ;
thisXform - > m_Post . E ( 1 ) ;
}
else
{
Interpolater < T > : : ConvertLinearToPolar ( embers , size , i , 1 , cxAng , cxMag , cxTrn ) ;
Interpolater < T > : : InterpAndConvertBack ( coefs , cxAng , cxMag , cxTrn , thisXform - > m_Post ) ;
2014-09-10 01:41:26 -04:00
}
2014-07-08 03:11:14 -04:00
}
2015-12-31 19:00:36 -05:00
else if ( m_AffineInterp = = eAffineInterp : : AFFINE_INTERP_LINEAR )
2014-07-08 03:11:14 -04:00
{
//Interpolate pre and post affine using coefs.
allID = true ;
thisXform - > m_Affine . m_Mat = m23T ( 0 ) ;
thisXform - > m_Post . m_Mat = m23T ( 0 ) ;
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < size ; k + + )
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
auto tempXform = embers [ k ] . GetTotalXform ( i ) ; //Xform in this position in this ember.
2014-07-08 03:11:14 -04:00
if ( tempXform )
{
allID & = tempXform - > m_Post . IsID ( ) ;
thisXform - > m_Affine . m_Mat + = ( coefs [ k ] * tempXform - > m_Affine . m_Mat ) ;
thisXform - > m_Post . m_Mat + = ( coefs [ k ] * tempXform - > m_Post . m_Mat ) ;
}
}
if ( allID )
thisXform - > m_Post . m_Mat = glm : : mat2x3 ( 1 ) ;
}
//Final stagger cleanup goes here.
if ( size = = 2 & & stagger > 0 & & thisXform ! = & m_FinalXform )
{
coefs [ 0 ] = coefSave [ 0 ] ;
coefs [ 1 ] = coefSave [ 1 ] ;
}
}
//Normally these functions are called automatically when
//adding variations to a xform, or when setting a parametric
//variation value via name lookup. However, since the values
//were directly written to, must manually call them here.
CacheXforms ( ) ;
2015-05-31 01:14:34 -04:00
//Need to merge xaos. Original does xaos all the time, but really only need to do it if at least one xform in at least one ember uses it, else skip.
//Omit final xform from xaos processing.
2014-07-08 03:11:14 -04:00
if ( Interpolater < T > : : AnyXaosPresent ( embers , size ) )
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < XformCount ( ) ; i + + )
2014-07-08 03:11:14 -04:00
{
2015-03-30 20:46:52 -04:00
for ( size_t k = 0 ; k < XformCount ( ) ; k + + ) //First make each xform xaos array be maxXformCount elements long and set them to zero.
m_Xforms [ i ] . SetXaos ( k , 0 ) ;
2014-07-08 03:11:14 -04:00
//Now fill them with interpolated values.
2014-10-14 11:53:15 -04:00
for ( size_t j = 0 ; j < size ; j + + ) //For each ember in the list.
2014-07-08 03:11:14 -04:00
{
2016-02-02 20:51:58 -05:00
auto tempXform = embers [ j ] . GetXform ( i ) ;
2014-07-08 03:11:14 -04:00
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < XformCount ( ) ; k + + ) //For each xaos entry in this xform's xaos array, sum it with the same entry in all of the embers multiplied by the coef for that ember.
2014-07-08 03:11:14 -04:00
{
m_Xforms [ i ] . SetXaos ( k , m_Xforms [ i ] . Xaos ( k ) + tempXform - > Xaos ( k ) * coefs [ j ] ) ;
}
}
//Make sure no xaos entries for this xform were less than zero.
2014-10-14 11:53:15 -04:00
for ( size_t k = 0 ; k < XformCount ( ) ; k + + )
2014-07-08 03:11:14 -04:00
if ( m_Xforms [ i ] . Xaos ( k ) < 0 )
m_Xforms [ i ] . SetXaos ( k , 0 ) ;
}
}
}
/// <summary>
/// Thin wrapper around InterpolateCatmullRom().
/// The ember vector is expected to be a length of 4 with all xforms aligned, including final.
/// </summary>
/// <param name="embers">Vector of embers</param>
/// <param name="t">Used in calculating Catmull-Rom coefficients</param>
2016-04-03 21:55:12 -04:00
void InterpolateCatmullRom ( const vector < Ember < T > > & embers , T t )
2014-07-08 03:11:14 -04:00
{
InterpolateCatmullRom ( embers . data ( ) , embers . size ( ) , t ) ;
}
/// <summary>
/// Use Catmull-Rom coefficients and pass to Interpolate().
/// The ember array is expected to be a length of 4 with all xforms aligned, including final.
/// </summary>
/// <param name="embers">Pointer to buffer of embers</param>
/// <param name="size">Number of elements in the buffer of embers</param>
/// <param name="t">Used in calculating Catmull-Rom coefficients</param>
2016-04-03 21:55:12 -04:00
void InterpolateCatmullRom ( const Ember < T > * embers , size_t size , T t )
2014-07-08 03:11:14 -04:00
{
T t2 = t * t ;
T t3 = t2 * t ;
vector < T > cmc ( 4 ) ;
cmc [ 0 ] = ( 2 * t2 - t - t3 ) / 2 ;
cmc [ 1 ] = ( 3 * t3 - 5 * t2 + 2 ) / 2 ;
cmc [ 2 ] = ( 4 * t2 - 3 * t3 + t ) / 2 ;
cmc [ 3 ] = ( t3 - t2 ) / 2 ;
Interpolate ( embers , size , cmc , 0 ) ;
}
/// <summary>
/// Rotate all pre-affine transforms in non-final xforms whose animate value is non-zero by
/// the specified angle in a counter-clockwise direction.
/// Omit padding xforms.
/// Do not rotate post affine transforms.
/// </summary>
/// <param name="angle">The angle to rotate by</param>
void RotateAffines ( T angle )
{
2014-10-14 11:53:15 -04:00
for ( size_t i = 0 ; i < XformCount ( ) ; i + + ) //Only look at normal xforms, exclude final.
2014-07-08 03:11:14 -04:00
{
//Don't rotate xforms with animate set to 0.
if ( m_Xforms [ i ] . m_Animate = = 0 )
continue ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Assume that if there are no variations, then it's a padding xform.
2015-12-31 19:00:36 -05:00
if ( m_Xforms [ i ] . Empty ( ) & & m_AffineInterp ! = eAffineInterp : : AFFINE_INTERP_LOG )
2014-07-08 03:11:14 -04:00
continue ;
2016-02-12 00:38:21 -05:00
m_Xforms [ i ] . m_Affine . Rotate ( angle * DEG_2_RAD_T ) ;
2014-07-08 03:11:14 -04:00
//Don't rotate post.
}
}
/// <summary>
/// Adds symmetry to this ember by adding additional xforms.
/// sym = 2 or more means rotational.
/// sym = 1 means identity, ie no symmetry.
/// sym = 0 means pick a random symmetry (maybe none).
/// sym = -1 means bilateral (reflection).
/// sym = -2 or less means rotational and reflective.
/// </summary>
/// <param name="sym">The type of symmetry to add</param>
/// <param name="rand">The random context to use for generating random symmetry</param>
--User changes
-Add support for multiple GPU devices.
--These options are present in the command line and in Fractorium.
-Change scheme of specifying devices from platform,device to just total device index.
--Single number on the command line.
--Change from combo boxes for device selection to a table of all devices in Fractorium.
-Temporal samples defaults to 100 instead of 1000 which was needless overkill.
--Bug fixes
-EmberAnimate, EmberRender, FractoriumSettings, FinalRenderDialog: Fix wrong order of arguments to Clamp() when assigning thread priority.
-VariationsDC.h: Fix NVidia OpenCL compilation error in DCTriangleVariation.
-FractoriumXformsColor.cpp: Checking for null pixmap pointer is not enough, must also check if the underlying buffer is null via call to QPixmap::isNull().
--Code changes
-Ember.h: Add case for FLAME_MOTION_NONE and default in ApplyFlameMotion().
-EmberMotion.h: Call base constructor.
-EmberPch.h: #pragma once only on Windows.
-EmberToXml.h:
--Handle different types of exceptions.
--Add default cases to ToString().
-Isaac.h: Remove unused variable in constructor.
-Point.h: Call base constructor in Color().
-Renderer.h/cpp:
--Add bool to Alloc() to only allocate memory for the histogram. Needed for multi-GPU.
--Make CoordMap() return a const ref, not a pointer.
-SheepTools.h:
--Use 64-bit types like the rest of the code already does.
--Fix some comment misspellings.
-Timing.h: Make BeginTime(), EndTime(), ElapsedTime() and Format() be const functions.
-Utils.h:
--Add new functions Equal() and Split().
--Handle more exception types in ReadFile().
--Get rid of most legacy blending of C and C++ argument parsing.
-XmlToEmber.h:
--Get rid of most legacy blending of C and C++ code from flam3.
--Remove some unused variables.
-EmberAnimate:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
--If a render fails, exit since there is no point in continuing an animation with a missing frame.
--Pass variables to threaded save better, which most likely fixes a very subtle bug that existed before.
--Remove some unused variables.
-EmberGenome, EmberRender:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
-EmberRender:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
--Only print values when not rendering with OpenCL, since they're always 0 in that case.
-EmberCLPch.h:
--#pragma once only on Windows.
--#include <atomic>.
-IterOpenCLKernelCreator.h: Add new kernel for summing two histograms. This is needed for multi-GPU.
-OpenCLWrapper.h:
--Move all OpenCL info related code into its own class OpenCLInfo.
--Add members to cache the values of global memory size and max allocation size.
-RendererCL.h/cpp:
--Redesign to accomodate multi-GPU.
--Constructor now takes a vector of devices.
--Remove DumpErrorReport() function, it's handled in the base.
--ClearBuffer(), ReadPoints(), WritePoints(), ReadHist() and WriteHist() now optionally take a device index as a parameter.
--MakeDmap() override and m_DmapCL member removed because it no longer applies since the histogram is always float since the last commit.
--Add new function SumDeviceHist() to sum histograms from two devices by first copying to a temporary on the host, then a temporary on the device, then summing.
--m_Calls member removed, as it's now per-device.
--OpenCLWrapper removed.
--m_Seeds member is now a vector of vector of seeds, to accomodate a separate and different array of seeds for each device.
--Added member m_Devices, a vector of unique_ptr of RendererCLDevice.
-EmberCommon.h
--Added Devices() function to convert from a vector of device indices to a vector of platform,device indices.
--Changed CreateRenderer() to accept a vector of devices to create a single RendererCL which will split work across multiple devices.
--Added CreateRenderers() function to accept a vector of devices to create multiple RendererCL, each which will render on a single device.
--Add more comments to some existing functions.
-EmberCommonPch.h: #pragma once only on Windows.
-EmberOptions.h:
--Remove --platform option, it's just sequential device number now with the --device option.
--Make --out be OPT_USE_RENDER instead of OPT_RENDER_ANIM since it's an error condition when animating. It makes no sense to write all frames to a single image.
--Add Devices() function to parse comma separated --device option string and return a vector of device indices.
--Make int and uint types be 64-bit, so intmax_t and size_t.
--Make better use of macros.
-JpegUtils.h: Make string parameters to WriteJpeg() and WritePng() be const ref.
-All project files: Turn off buffer security check option in Visual Studio (/Gs-)
-deployment.pri: Remove the line OTHER_FILES +=, it's pointless and was causing problems.
-Ember.pro, EmberCL.pro: Add CONFIG += plugin, otherwise it wouldn't link.
-EmberCL.pro: Add new files for multi-GPU support.
-build_all.sh: use -j4 and QMAKE=${QMAKE:/usr/bin/qmake}
-shared_settings.pri:
-Add version string.
-Remove old DESTDIR definitions.
-Add the following lines or else nothing would build:
CONFIG(release, debug|release) {
CONFIG += warn_off
DESTDIR = ../../../Bin/release
}
CONFIG(debug, debug|release) {
DESTDIR = ../../../Bin/debug
}
QMAKE_POST_LINK += $$quote(cp --update ../../../Data/flam3-palettes.xml $${DESTDIR}$$escape_expand(\n\t))
LIBS += -L/usr/lib -lpthread
-AboutDialog.ui: Another futile attempt to make it look correct on Linux.
-FinalRenderDialog.h/cpp:
--Add support for multi-GPU.
--Change from combo boxes for device selection to a table of all devices.
--Ensure device selection makes sense.
--Remove "FinalRender" prefix of various function names, it's implied given the context.
-FinalRenderEmberController.h/cpp:
--Add support for multi-GPU.
--Change m_FinishedImageCount to be atomic.
--Move CancelRender() from the base to FinalRenderEmberController<T>.
--Refactor RenderComplete() to omit any progress related functionality or image saving since it can be potentially ran in a thread.
--Consolidate setting various renderer fields into SyncGuiToRenderer().
-Fractorium.cpp: Allow for resizing of the options dialog to show the entire device table.
-FractoriumCommon.h: Add various functions to handle a table showing the available OpenCL devices on the system.
-FractoriumEmberController.h/cpp: Remove m_FinalImageIndex, it's no longer needed.
-FractoriumRender.cpp: Scale the interactive sub batch count and quality by the number of devices used.
-FractoriumSettings.h/cpp:
--Temporal samples defaults to 100 instead of 1000 which was needless overkill.
--Add multi-GPU support, remove old device,platform pair.
-FractoriumToolbar.cpp: Disable OpenCL toolbar button if there are no devices present on the system.
-FractoriumOptionsDialog.h/cpp:
--Add support for multi-GPU.
--Consolidate more assignments in DataToGui().
--Enable/disable CPU/OpenCL items in response to OpenCL checkbox event.
-Misc: Convert almost everything to size_t for unsigned, intmax_t for signed.
2015-09-12 21:33:45 -04:00
void AddSymmetry ( intmax_t sym , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
2014-07-08 03:11:14 -04:00
{
--User changes
-Add support for multiple GPU devices.
--These options are present in the command line and in Fractorium.
-Change scheme of specifying devices from platform,device to just total device index.
--Single number on the command line.
--Change from combo boxes for device selection to a table of all devices in Fractorium.
-Temporal samples defaults to 100 instead of 1000 which was needless overkill.
--Bug fixes
-EmberAnimate, EmberRender, FractoriumSettings, FinalRenderDialog: Fix wrong order of arguments to Clamp() when assigning thread priority.
-VariationsDC.h: Fix NVidia OpenCL compilation error in DCTriangleVariation.
-FractoriumXformsColor.cpp: Checking for null pixmap pointer is not enough, must also check if the underlying buffer is null via call to QPixmap::isNull().
--Code changes
-Ember.h: Add case for FLAME_MOTION_NONE and default in ApplyFlameMotion().
-EmberMotion.h: Call base constructor.
-EmberPch.h: #pragma once only on Windows.
-EmberToXml.h:
--Handle different types of exceptions.
--Add default cases to ToString().
-Isaac.h: Remove unused variable in constructor.
-Point.h: Call base constructor in Color().
-Renderer.h/cpp:
--Add bool to Alloc() to only allocate memory for the histogram. Needed for multi-GPU.
--Make CoordMap() return a const ref, not a pointer.
-SheepTools.h:
--Use 64-bit types like the rest of the code already does.
--Fix some comment misspellings.
-Timing.h: Make BeginTime(), EndTime(), ElapsedTime() and Format() be const functions.
-Utils.h:
--Add new functions Equal() and Split().
--Handle more exception types in ReadFile().
--Get rid of most legacy blending of C and C++ argument parsing.
-XmlToEmber.h:
--Get rid of most legacy blending of C and C++ code from flam3.
--Remove some unused variables.
-EmberAnimate:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
--If a render fails, exit since there is no point in continuing an animation with a missing frame.
--Pass variables to threaded save better, which most likely fixes a very subtle bug that existed before.
--Remove some unused variables.
-EmberGenome, EmberRender:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
-EmberRender:
--Support multi-GPU processing that alternates full frames between devices.
--Use OpenCLInfo instead of OpenCLWrapper for --openclinfo option.
--Remove bucketT template parameter, and hard code float in its place.
--Only print values when not rendering with OpenCL, since they're always 0 in that case.
-EmberCLPch.h:
--#pragma once only on Windows.
--#include <atomic>.
-IterOpenCLKernelCreator.h: Add new kernel for summing two histograms. This is needed for multi-GPU.
-OpenCLWrapper.h:
--Move all OpenCL info related code into its own class OpenCLInfo.
--Add members to cache the values of global memory size and max allocation size.
-RendererCL.h/cpp:
--Redesign to accomodate multi-GPU.
--Constructor now takes a vector of devices.
--Remove DumpErrorReport() function, it's handled in the base.
--ClearBuffer(), ReadPoints(), WritePoints(), ReadHist() and WriteHist() now optionally take a device index as a parameter.
--MakeDmap() override and m_DmapCL member removed because it no longer applies since the histogram is always float since the last commit.
--Add new function SumDeviceHist() to sum histograms from two devices by first copying to a temporary on the host, then a temporary on the device, then summing.
--m_Calls member removed, as it's now per-device.
--OpenCLWrapper removed.
--m_Seeds member is now a vector of vector of seeds, to accomodate a separate and different array of seeds for each device.
--Added member m_Devices, a vector of unique_ptr of RendererCLDevice.
-EmberCommon.h
--Added Devices() function to convert from a vector of device indices to a vector of platform,device indices.
--Changed CreateRenderer() to accept a vector of devices to create a single RendererCL which will split work across multiple devices.
--Added CreateRenderers() function to accept a vector of devices to create multiple RendererCL, each which will render on a single device.
--Add more comments to some existing functions.
-EmberCommonPch.h: #pragma once only on Windows.
-EmberOptions.h:
--Remove --platform option, it's just sequential device number now with the --device option.
--Make --out be OPT_USE_RENDER instead of OPT_RENDER_ANIM since it's an error condition when animating. It makes no sense to write all frames to a single image.
--Add Devices() function to parse comma separated --device option string and return a vector of device indices.
--Make int and uint types be 64-bit, so intmax_t and size_t.
--Make better use of macros.
-JpegUtils.h: Make string parameters to WriteJpeg() and WritePng() be const ref.
-All project files: Turn off buffer security check option in Visual Studio (/Gs-)
-deployment.pri: Remove the line OTHER_FILES +=, it's pointless and was causing problems.
-Ember.pro, EmberCL.pro: Add CONFIG += plugin, otherwise it wouldn't link.
-EmberCL.pro: Add new files for multi-GPU support.
-build_all.sh: use -j4 and QMAKE=${QMAKE:/usr/bin/qmake}
-shared_settings.pri:
-Add version string.
-Remove old DESTDIR definitions.
-Add the following lines or else nothing would build:
CONFIG(release, debug|release) {
CONFIG += warn_off
DESTDIR = ../../../Bin/release
}
CONFIG(debug, debug|release) {
DESTDIR = ../../../Bin/debug
}
QMAKE_POST_LINK += $$quote(cp --update ../../../Data/flam3-palettes.xml $${DESTDIR}$$escape_expand(\n\t))
LIBS += -L/usr/lib -lpthread
-AboutDialog.ui: Another futile attempt to make it look correct on Linux.
-FinalRenderDialog.h/cpp:
--Add support for multi-GPU.
--Change from combo boxes for device selection to a table of all devices.
--Ensure device selection makes sense.
--Remove "FinalRender" prefix of various function names, it's implied given the context.
-FinalRenderEmberController.h/cpp:
--Add support for multi-GPU.
--Change m_FinishedImageCount to be atomic.
--Move CancelRender() from the base to FinalRenderEmberController<T>.
--Refactor RenderComplete() to omit any progress related functionality or image saving since it can be potentially ran in a thread.
--Consolidate setting various renderer fields into SyncGuiToRenderer().
-Fractorium.cpp: Allow for resizing of the options dialog to show the entire device table.
-FractoriumCommon.h: Add various functions to handle a table showing the available OpenCL devices on the system.
-FractoriumEmberController.h/cpp: Remove m_FinalImageIndex, it's no longer needed.
-FractoriumRender.cpp: Scale the interactive sub batch count and quality by the number of devices used.
-FractoriumSettings.h/cpp:
--Temporal samples defaults to 100 instead of 1000 which was needless overkill.
--Add multi-GPU support, remove old device,platform pair.
-FractoriumToolbar.cpp: Disable OpenCL toolbar button if there are no devices present on the system.
-FractoriumOptionsDialog.h/cpp:
--Add support for multi-GPU.
--Consolidate more assignments in DataToGui().
--Enable/disable CPU/OpenCL items in response to OpenCL checkbox event.
-Misc: Convert almost everything to size_t for unsigned, intmax_t for signed.
2015-09-12 21:33:45 -04:00
intmax_t k ;
size_t i , result = 0 ;
2014-07-08 03:11:14 -04:00
T a ;
if ( sym = = 0 )
{
2015-12-31 16:41:59 -05:00
static intmax_t symDistrib [ ] =
{
2014-07-08 03:11:14 -04:00
- 4 , - 3 ,
- 2 , - 2 , - 2 ,
- 1 , - 1 , - 1 ,
2015-12-31 16:41:59 -05:00
2 , 2 , 2 ,
3 , 3 ,
4 , 4 ,
2014-07-08 03:11:14 -04:00
} ;
if ( rand . Rand ( ) & 1 )
sym = symDistrib [ rand . Rand ( ) % Vlen ( symDistrib ) ] ;
else if ( rand . Rand ( ) & 31 )
2015-09-13 14:53:14 -04:00
sym = intmax_t ( rand . Rand ( ) % 13 ) - 6 ;
2014-07-08 03:11:14 -04:00
else
2015-09-13 14:53:14 -04:00
sym = intmax_t ( rand . Rand ( ) % 51 ) - 25 ;
2014-07-08 03:11:14 -04:00
}
if ( sym = = 1 | | sym = = 0 )
return ;
m_Symmetry = sym ;
if ( sym < 0 )
{
i = XformCount ( ) ; //Don't apply sym to final.
Xform < T > xform ;
AddXform ( xform ) ;
m_Xforms [ i ] . m_Weight = 1 ;
m_Xforms [ i ] . m_ColorSpeed = 0 ;
2015-01-23 21:26:18 -05:00
m_Xforms [ i ] . m_Animate = 0 ;
2014-07-08 03:11:14 -04:00
m_Xforms [ i ] . m_ColorX = 1 ;
m_Xforms [ i ] . m_ColorY = 1 ; //Added in case 2D palette support is ever added.
m_Xforms [ i ] . m_Affine . A ( - 1 ) ;
m_Xforms [ i ] . m_Affine . B ( 0 ) ;
m_Xforms [ i ] . m_Affine . C ( 0 ) ;
m_Xforms [ i ] . m_Affine . D ( 0 ) ;
m_Xforms [ i ] . m_Affine . E ( 1 ) ;
m_Xforms [ i ] . m_Affine . F ( 0 ) ;
2016-04-13 23:59:57 -04:00
m_Xforms [ i ] . AddVariation ( m_VariationList - > GetVariationCopy ( eVariationId : : VAR_LINEAR ) ) ;
2014-07-08 03:11:14 -04:00
result + + ;
sym = - sym ;
}
a = T ( 2 * M_PI / sym ) ;
for ( k = 1 ; k < sym ; k + + )
{
i = XformCount ( ) ;
Xform < T > xform ;
AddXform ( xform ) ;
2015-01-23 21:26:18 -05:00
m_Xforms [ i ] . m_Weight = 1 ;
m_Xforms [ i ] . m_ColorSpeed = 0 ;
m_Xforms [ i ] . m_Animate = 0 ;
2014-07-08 03:11:14 -04:00
m_Xforms [ i ] . m_ColorX = m_Xforms [ i ] . m_ColorY = ( sym < 3 ) ? 0 : ( T ( k - 1 ) / T ( sym - 2 ) ) ; //Added Y.
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
m_Xforms [ i ] . m_Affine . A ( Round6 ( std : : cos ( k * a ) ) ) ;
m_Xforms [ i ] . m_Affine . D ( Round6 ( std : : sin ( k * a ) ) ) ;
2014-07-08 03:11:14 -04:00
m_Xforms [ i ] . m_Affine . B ( Round6 ( - m_Xforms [ i ] . m_Affine . D ( ) ) ) ;
m_Xforms [ i ] . m_Affine . E ( m_Xforms [ i ] . m_Affine . A ( ) ) ;
m_Xforms [ i ] . m_Affine . C ( 0 ) ;
m_Xforms [ i ] . m_Affine . F ( 0 ) ;
2016-04-13 23:59:57 -04:00
m_Xforms [ i ] . AddVariation ( m_VariationList - > GetVariationCopy ( eVariationId : : VAR_LINEAR ) ) ;
2014-07-08 03:11:14 -04:00
result + + ;
}
//Sort the newly added xforms, do not touch the original ones.
std : : sort ( m_Xforms . end ( ) - result , m_Xforms . end ( ) , & Interpolater < T > : : CompareXforms ) ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Return a uint with bits set to indicate which kind of projection should be done.
/// </summary>
/// <param name="onlyScaleIfNewIsSmaller">A uint with bits set for each kind of projection that is needed</param>
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
size_t ProjBits ( ) const
2014-07-08 03:11:14 -04:00
{
2014-10-14 11:53:15 -04:00
size_t val = 0 ;
2014-07-08 03:11:14 -04:00
2016-01-04 19:50:15 -05:00
if ( m_CamZPos ! = 0 ) val | = et ( eProjBits : : PROJBITS_ZPOS ) ;
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
if ( m_CamPerspective ! = 0 ) val | = et ( eProjBits : : PROJBITS_PERSP ) ;
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
if ( m_CamPitch ! = 0 ) val | = et ( eProjBits : : PROJBITS_PITCH ) ;
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
if ( m_CamYaw ! = 0 ) val | = et ( eProjBits : : PROJBITS_YAW ) ;
2015-12-31 16:41:59 -05:00
2016-01-04 19:50:15 -05:00
if ( m_CamDepthBlur ! = 0 ) val | = et ( eProjBits : : PROJBITS_BLUR ) ;
2014-07-08 03:11:14 -04:00
return val ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Call the appropriate projection function via function pointer.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">The Isaac object to pass to the projection functions</param>
2014-07-08 03:11:14 -04:00
inline void Proj ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
( this - > * m_ProjFunc ) ( point , rand ) ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Placeholder to do nothing.
/// </summary>
/// <param name="point">Unused</param>
/// <param name="rand">Unused</param>
2014-07-08 03:11:14 -04:00
void ProjectNone ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Project when only z is set, and not pitch, yaw, projection or depth blur.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">Unused</param>
2014-07-08 03:11:14 -04:00
void ProjectZPerspective ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
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
T zr = Zeps ( 1 - m_CamPerspective * ( point . m_Z - m_CamZPos ) ) ;
2014-07-08 03:11:14 -04:00
point . m_X / = zr ;
point . m_Y / = zr ;
point . m_Z - = m_CamZPos ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Project when pitch, and optionally z and perspective are set, but not depth blur or yaw.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">Unused</param>
2014-07-08 03:11:14 -04:00
void ProjectPitch ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
T z = point . m_Z - m_CamZPos ;
T y = m_CamMat [ 1 ] [ 1 ] * point . m_Y + m_CamMat [ 2 ] [ 1 ] * z ;
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
T zr = Zeps ( 1 - m_CamPerspective * ( m_CamMat [ 1 ] [ 2 ] * point . m_Y + m_CamMat [ 2 ] [ 2 ] * z ) ) ;
2014-07-08 03:11:14 -04:00
point . m_X / = zr ;
point . m_Y = y / zr ;
point . m_Z - = m_CamZPos ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Project when depth blur, and optionally pitch, perspective and z are set, but not yaw.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">Used for blurring</param>
2014-07-08 03:11:14 -04:00
void ProjectPitchDepthBlur ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
T y , z , zr ;
T dsin , dcos ;
T t = rand . Frand01 < T > ( ) * M_2PI ;
z = point . m_Z - m_CamZPos ;
y = m_CamMat [ 1 ] [ 1 ] * point . m_Y + m_CamMat [ 2 ] [ 1 ] * z ;
z = m_CamMat [ 1 ] [ 2 ] * point . m_Y + m_CamMat [ 2 ] [ 2 ] * z ;
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
zr = Zeps ( 1 - m_CamPerspective * z ) ;
2014-07-08 03:11:14 -04:00
sincos ( t , & dsin , & dcos ) ;
T dr = rand . Frand01 < T > ( ) * m_BlurCoef * z ;
point . m_X = ( point . m_X + dr * dcos ) / zr ;
point . m_Y = ( y + dr * dsin ) / zr ;
point . m_Z - = m_CamZPos ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Project when depth blur, yaw and optionally pitch are set, but not perspective and z.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">Used for blurring</param>
2014-07-08 03:11:14 -04:00
void ProjectPitchYawDepthBlur ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
T dsin , dcos ;
T t = rand . Frand01 < T > ( ) * M_2PI ;
T z = point . m_Z - m_CamZPos ;
T x = m_CamMat [ 0 ] [ 0 ] * point . m_X + m_CamMat [ 1 ] [ 0 ] * point . m_Y ;
T y = m_CamMat [ 0 ] [ 1 ] * point . m_X + m_CamMat [ 1 ] [ 1 ] * point . m_Y + m_CamMat [ 2 ] [ 1 ] * z ;
z = m_CamMat [ 0 ] [ 2 ] * point . m_X + m_CamMat [ 1 ] [ 2 ] * point . m_Y + m_CamMat [ 2 ] [ 2 ] * z ;
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
T zr = Zeps ( 1 - m_CamPerspective * z ) ;
2014-07-08 03:11:14 -04:00
T dr = rand . Frand01 < T > ( ) * m_BlurCoef * z ;
sincos ( t , & dsin , & dcos ) ;
point . m_X = ( x + dr * dcos ) / zr ;
point . m_Y = ( y + dr * dsin ) / zr ;
point . m_Z - = m_CamZPos ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Project when yaw and optionally pitch, z, and perspective are set, but not depth blur.
/// </summary>
/// <param name="point">The point to project</param>
/// <param name="rand">Unused</param>
2014-07-08 03:11:14 -04:00
void ProjectPitchYaw ( Point < T > & point , QTIsaac < ISAAC_SIZE , ISAAC_INT > & rand )
{
T z = point . m_Z - m_CamZPos ;
T x = m_CamMat [ 0 ] [ 0 ] * point . m_X + m_CamMat [ 1 ] [ 0 ] * point . m_Y ;
T y = m_CamMat [ 0 ] [ 1 ] * point . m_X + m_CamMat [ 1 ] [ 1 ] * point . m_Y + m_CamMat [ 2 ] [ 1 ] * z ;
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
T zr = Zeps ( 1 - m_CamPerspective * ( m_CamMat [ 0 ] [ 2 ] * point . m_X + m_CamMat [ 1 ] [ 2 ] * point . m_Y + m_CamMat [ 2 ] [ 2 ] * z ) ) ;
2014-07-08 03:11:14 -04:00
point . m_X = x / zr ;
point . m_Y = y / zr ;
point . m_Z - = m_CamZPos ;
}
2015-07-06 10:05:43 -04:00
# define APP_FMP(x) x += param.second * Interpolater<T>::MotionFuncs(motion.m_MotionFunc, motion.m_MotionFreq * (blend + motion.m_MotionOffset))
2015-07-07 00:36:46 -04:00
2015-07-02 13:01:37 -04:00
/// <summary>
/// Update ember parameters based on stored motion elements
/// </summary>
/// <param name="blend">The time percentage value which dictates how much of a percentage of 360 degrees it should be rotated and the time position for the motion elements</param>
void ApplyFlameMotion ( T blend )
{
2015-07-07 00:36:46 -04:00
for ( size_t i = 0 ; i < m_EmberMotionElements . size ( ) ; + + i )
2015-07-02 13:01:37 -04:00
{
2015-07-07 00:36:46 -04:00
auto & motion = m_EmberMotionElements [ i ] ;
2015-07-02 13:01:37 -04:00
for ( size_t j = 0 ; j < motion . m_MotionParams . size ( ) ; + + j )
{
2015-07-07 00:36:46 -04:00
auto & param = motion . m_MotionParams [ j ] ;
2015-07-02 13:01:37 -04:00
switch ( param . first )
{
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_ZOOM :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Zoom ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_ZPOS :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CamZPos ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_PERSPECTIVE :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CamPerspective ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_YAW :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CamYaw ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_PITCH :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CamPitch ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_DEPTH_BLUR :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CamDepthBlur ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_CENTER_X :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CenterX ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_CENTER_Y :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_CenterY ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_ROTATE :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Rotate ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_BRIGHTNESS :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Brightness ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_GAMMA :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Gamma ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_GAMMA_THRESH :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_GammaThresh ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_HIGHLIGHT_POWER :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_HighlightPower ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_BACKGROUND_R :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Background . r ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_BACKGROUND_G :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Background . g ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_BACKGROUND_B :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Background . b ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_VIBRANCY :
2015-12-31 16:41:59 -05:00
APP_FMP ( m_Vibrancy ) ;
break ;
2015-12-31 19:00:36 -05:00
case eEmberMotionParam : : FLAME_MOTION_NONE :
2015-12-31 16:41:59 -05:00
default :
break ;
2015-07-02 13:01:37 -04:00
}
}
}
}
2014-07-08 03:11:14 -04:00
/// <summary>
/// Clear this ember and set to either reasonable or unreasonable default values.
/// </summary>
/// <param name="useDefaults">Use reasonable default if true, else use out of bounds values.</param>
void Clear ( bool useDefaults = true )
{
m_Palette . m_Index = - 1 ;
m_CenterX = 0 ;
m_CenterY = 0 ;
2014-10-27 17:21:06 -04:00
m_RotCenterY = 0 ;
2014-07-08 03:11:14 -04:00
m_Gamma = 4 ;
m_Vibrancy = 1 ;
m_Brightness = 4 ;
m_Symmetry = 0 ;
m_Rotate = 0 ;
m_PixelsPerUnit = 50 ;
2016-03-12 17:42:56 -05:00
m_Interp = eInterp : : EMBER_INTERP_SMOOTH ;
2015-12-31 19:00:36 -05:00
m_PaletteInterp = ePaletteInterp : : INTERP_HSV ;
2014-07-08 03:11:14 -04:00
m_Index = 0 ;
m_ParentFilename = " " ;
2014-10-14 11:53:15 -04:00
m_ScaleType = eScaleType : : SCALE_NONE ;
2014-07-08 03:11:14 -04:00
if ( useDefaults )
{
//If defaults are on, set to reasonable values.
m_HighlightPower = - 1 ;
m_Background . Reset ( ) ;
m_FinalRasW = 100 ;
m_FinalRasH = 100 ;
m_Supersample = 1 ;
m_SpatialFilterRadius = T ( 0.5 ) ;
m_Zoom = 0 ;
m_ProjFunc = & EmberNs : : Ember < T > : : ProjectNone ;
m_CamZPos = 0 ;
m_CamPerspective = 0 ;
m_CamYaw = 0 ;
m_CamPitch = 0 ;
m_CamDepthBlur = 0 ;
m_BlurCoef = 0 ;
m_CamMat = m3T ( 0 ) ;
m_Quality = 1 ;
m_MaxRadDE = T ( 9.0 ) ;
m_MinRadDE = 0 ;
m_CurveDE = T ( 0.4 ) ;
m_GammaThresh = T ( 0.01 ) ;
--User changes
-Add new variations: bubbleT3D, crob, hexaplay3D, hexcrop, hexes, hexnix3D, loonie2, loonie3, nBlur, octapol and synth.
-Allow for pre/post versions of dc_bubble, dc_cylinder and dc_linear whereas before they were omitted.
-When saving a file with multiple embers in it, detect if time values are all the same and if so, start them at zero and increment by 1 for each ember.
-Allow for numerous quality increases to be coalesced into one. It will pick up at the end of the current render.
-Show selection highlight on variations tree in response to mouse hover. This makes it easier to see for which variation or param the current mouse wheel action will apply.
-Make default temporal samples be 100, whereas before it was 1000 which was overkill.
-Require the shift key to be held with delete for deleting an ember to prevent it from triggering when the user enters delete in the edit box.
-This wasn't otherwise fixable without writing a lot more code.
--Bug fixes
-EmberGenome was crashing when generating a sequence from a source file with more than 2 embers in it.
-EmberGenome was improperly handling the first frame of a merge after the last frame of the loop.
-These bugs were due to a previous commit. Revert parts of that commit.
-Prevent a zoom value of less than 0 when reading from xml.
-Slight optimization of the crescents, and mask variations, if the compiler wasn't doing it already.
-Unique file naming was broken because it was looking for _# and the default names ended with -#.
-Disallow renaming of an ember in the library tree to an empty string.
-Severe bug that prevented some variations from being read correctly from params generated outside this program.
-Severe OpenCL randomization bug. The first x coordinates of the first points in the first kernel call of the first ember of a render since the OpenCL renderer object was created were not random and were mostly -1.
-Severe bug when populating xform selection distributions that could sometimes cause a crash due to roundoff error. Fix by using double.
-Limit the max number of variations in a random ember to MAX_CL_VARS, which is 8. This ensures they'll look the same on CPU and GPU.
-Prevent user from saving stylesheet to default.qss, it's a special reserved filename.
--Code changes
-Generalize using the running sum output point inside of a variation for all cases: pre, reg and post.
-Allow for array variables in variations where the address of each element is stored in m_Params.
-Qualify all math functions with std::
-No longer use our own Clamp() in OpenCL, instead use the standard clamp().
-Redesign how functions are used in the variations OpenCL code.
-Add tests to EmberTester to verify some of the new functionality.
-Place more const and override qualifiers on functions where appropriate.
-Add a global rand with a lock to be used very sparingly.
-Use a map instead of a vector for bad param names in Xml parsing.
-Prefix affine interpolation mode defines with "AFFINE_" to make their purpose more clear.
-Allow for variations that change state during iteration by sending a separate copy of the ember to each rendering thread.
-Implement this same functionality with a local struct in OpenCL. It's members are the total of all variables that need to change state within an ember.
-Add Contains() function to Utils.h.
-EmberRender: print names of kernels being printed with --dump_kernel option.
-Clean up EmberTester to handle some of the recent changes.
-Fix various casts.
-Replace % 2 with & 1, even though the compiler was likely doing this already.
-Add new file Variations06.h to accommodate new variations.
-General cleanup.
2015-11-22 17:15:07 -05:00
m_TemporalSamples = 100 ;
2015-12-31 19:00:36 -05:00
m_SpatialFilterType = eSpatialFilterType : : GAUSSIAN_SPATIAL_FILTER ;
m_AffineInterp = eAffineInterp : : AFFINE_INTERP_LOG ;
m_TemporalFilterType = eTemporalFilterType : : BOX_TEMPORAL_FILTER ;
2014-07-08 03:11:14 -04:00
m_TemporalFilterWidth = 1 ;
m_TemporalFilterExp = 0 ;
2015-12-31 19:00:36 -05:00
m_PaletteMode = ePaletteMode : : PALETTE_STEP ;
2014-07-08 03:11:14 -04:00
}
else
{
//Defaults are off, so set to UN-reasonable values.
m_HighlightPower = - 1 ;
m_Background = Color < T > ( - 1 , - 1 , - 1 , 1 ) ;
m_FinalRasW = 0 ;
m_FinalRasH = 0 ;
m_Supersample = 0 ;
m_SpatialFilterRadius = - 1 ;
m_Zoom = 999999 ;
2014-09-10 01:41:26 -04:00
m_ProjFunc = nullptr ;
2014-07-08 03:11:14 -04:00
m_CamZPos = 999999 ;
m_CamPerspective = 999999 ;
m_CamYaw = 999999 ;
m_CamPitch = 999999 ;
m_CamDepthBlur = 999999 ;
m_BlurCoef = 999999 ;
m_CamMat = m3T ( 999999 ) ;
m_Quality = - 1 ;
m_MaxRadDE = - 1 ;
m_MinRadDE = - 1 ;
m_CurveDE = - 1 ;
m_GammaThresh = - 1 ;
m_TemporalSamples = 0 ;
2015-12-31 19:00:36 -05:00
m_SpatialFilterType = eSpatialFilterType : : GAUSSIAN_SPATIAL_FILTER ;
m_AffineInterp = eAffineInterp : : AFFINE_INTERP_LOG ;
m_TemporalFilterType = eTemporalFilterType : : BOX_TEMPORAL_FILTER ;
2014-07-08 03:11:14 -04:00
m_TemporalFilterWidth = - 1 ;
m_TemporalFilterExp = - 999 ;
2015-12-31 19:00:36 -05:00
m_PaletteMode = ePaletteMode : : PALETTE_STEP ;
2014-07-08 03:11:14 -04:00
}
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
m_Xforms . clear ( ) ;
m_FinalXform . Clear ( ) ;
2015-03-21 18:27:37 -04:00
m_Curves . Init ( ) ;
2014-07-08 03:11:14 -04:00
ClearEdit ( ) ;
}
/// <summary>
2014-09-10 01:41:26 -04:00
/// Thin wrapper to clear edit doc if not nullptr and set to nullptr.
2014-07-08 03:11:14 -04:00
/// </summary>
void ClearEdit ( )
{
2015-10-27 00:31:35 -04:00
if ( m_Edits )
2014-07-08 03:11:14 -04:00
xmlFreeDoc ( m_Edits ) ;
2014-09-10 01:41:26 -04:00
m_Edits = nullptr ;
2014-07-08 03:11:14 -04:00
}
/// <summary>
/// Return a string representation of this ember.
/// </summary>
/// <returns>The string representation of this ember</returns>
string ToString ( ) const
{
2014-10-14 11:53:15 -04:00
size_t i ;
2014-07-08 03:11:14 -04:00
ostringstream ss ;
2015-12-31 16:41:59 -05:00
ss < < " Final Raster Width: " < < m_FinalRasW < < " \n "
< < " Final Raster Height: " < < m_FinalRasH < < " \n "
< < " Original Raster Width: " < < m_OrigFinalRasW < < " \n "
< < " Original Raster Height: " < < m_OrigFinalRasH < < " \n "
< < " Supersample: " < < m_Supersample < < " \n "
< < " Temporal Samples: " < < m_TemporalSamples < < " \n "
< < " Symmetry: " < < m_Symmetry < < " \n "
< < " Quality: " < < m_Quality < < " \n "
< < " Pixels Per Unit: " < < m_PixelsPerUnit < < " \n "
< < " Original Pixels Per Unit: " < < m_OrigPixPerUnit < < " \n "
< < " Sub Batch Size: " < < m_SubBatchSize < < " \n "
< < " Fuse Count: " < < m_FuseCount < < " \n "
< < " Zoom: " < < m_Zoom < < " \n "
< < " ZPos: " < < m_CamZPos < < " \n "
< < " Perspective: " < < m_CamPerspective < < " \n "
< < " Yaw: " < < m_CamYaw < < " \n "
< < " Pitch: " < < m_CamPitch < < " \n "
< < " Depth Blur: " < < m_CamDepthBlur < < " \n "
< < " CenterX: " < < m_CenterX < < " \n "
< < " CenterY: " < < m_CenterY < < " \n "
< < " RotCenterY: " < < m_RotCenterY < < " \n "
< < " Rotate: " < < m_Rotate < < " \n "
< < " Brightness: " < < m_Brightness < < " \n "
< < " Gamma: " < < m_Gamma < < " \n "
< < " Vibrancy: " < < m_Vibrancy < < " \n "
< < " Gamma Threshold: " < < m_GammaThresh < < " \n "
< < " Highlight Power: " < < m_HighlightPower < < " \n "
< < " Time: " < < m_Time < < " \n "
< < " Background: " < < m_Background . r < < " , " < < m_Background . g < < " , " < < m_Background . b < < " , " < < m_Background . a < < " \n "
< < " Interp: " < < m_Interp < < " \n "
< < " Affine Interp Type: " < < m_AffineInterp < < " \n "
< < " Minimum DE Radius: " < < m_MinRadDE < < " \n "
< < " Maximum DE Radius: " < < m_MaxRadDE < < " \n "
< < " DE Curve: " < < m_CurveDE < < " \n "
< < " Spatial Filter Type: " < < m_SpatialFilterType < < " \n "
< < " Spatial Filter Radius: " < < m_SpatialFilterRadius < < " \n "
< < " Temporal Filter Type: " < < m_TemporalFilterType < < " \n "
< < " Temporal Filter Exp: " < < m_TemporalFilterExp < < " \n "
< < " Temporal Filter Width: " < < m_TemporalFilterWidth < < " \n "
< < " Palette Mode: " < < m_PaletteMode < < " \n "
< < " Palette Interp: " < < m_PaletteInterp < < " \n "
< < " Palette Index: " < < m_Palette . m_Index < < " \n "
2014-07-08 03:11:14 -04:00
//Add palette info here if needed.
2015-12-31 16:41:59 -05:00
< < " Name: " < < m_Name < < " \n "
< < " Index: " < < m_Index < < " \n "
< < " Scale Type: " < < m_ScaleType < < " \n "
< < " Parent Filename: " < < m_ParentFilename < < " \n "
< < " \n " ;
2014-07-08 03:11:14 -04:00
for ( i = 0 ; i < XformCount ( ) ; i + + )
{
2015-12-31 16:41:59 -05:00
ss < < " Xform " < < i < < " : \n " < < m_Xforms [ i ] . ToString ( ) < < " \n " ;
2014-07-08 03:11:14 -04:00
}
if ( UseFinalXform ( ) )
2015-12-31 16:41:59 -05:00
ss < < " Final Xform: " < < m_FinalXform . ToString ( ) < < " \n " ;
2014-07-08 03:11:14 -04:00
return ss . str ( ) ;
}
/// <summary>
/// Accessors.
/// </summary>
2016-03-28 21:49:10 -04:00
inline const Xform < T > * Xforms ( ) const { return m_Xforms . data ( ) ; }
inline Xform < T > * NonConstXforms ( ) { return m_Xforms . data ( ) ; }
2014-10-14 11:53:15 -04:00
inline size_t XformCount ( ) const { return m_Xforms . size ( ) ; }
2014-07-08 03:11:14 -04:00
inline const Xform < T > * FinalXform ( ) const { return & m_FinalXform ; }
inline Xform < T > * NonConstFinalXform ( ) { return & m_FinalXform ; }
inline bool UseFinalXform ( ) const { return ! m_FinalXform . Empty ( ) ; }
2014-10-14 11:53:15 -04:00
inline size_t TotalXformCount ( ) const { return XformCount ( ) + ( UseFinalXform ( ) ? 1 : 0 ) ; }
2014-07-08 03:11:14 -04:00
inline int PaletteIndex ( ) const { return m_Palette . m_Index ; }
inline T BlurCoef ( ) { return m_BlurCoef ; }
2014-10-14 11:53:15 -04:00
inline eScaleType ScaleType ( ) const { return m_ScaleType ; }
2014-07-08 03:11:14 -04:00
//The width and height in pixels of the final output image. The size of the histogram and DE filtering buffers will differ from this.
//Xml fields: "size".
2016-04-03 21:55:12 -04:00
size_t m_FinalRasW = 1920 ;
size_t m_FinalRasH = 1080 ;
size_t m_OrigFinalRasW = 1920 ; //Keep track of the originals read from the Xml, because...
size_t m_OrigFinalRasH = 1080 ; //the dimension may change in an editor and the originals are needed for the aspect ratio.
T m_OrigPixPerUnit = 240 ;
2014-07-08 03:11:14 -04:00
2014-11-28 04:37:51 -05:00
//The iteration depth. This was a rendering parameter in flam3 but has been made a member here
//so that it can be adjusted more easily.
2016-04-03 21:55:12 -04:00
size_t m_SubBatchSize = DEFAULT_SBS ;
2014-11-28 04:37:51 -05:00
//The number of iterations to disregard for each sub batch. This was a rendering parameter in flam3 but has been made a member here
//so that it can be adjusted more easily.
2016-04-03 21:55:12 -04:00
size_t m_FuseCount = 15 ;
2014-11-28 04:37:51 -05:00
2014-07-08 03:11:14 -04:00
//The multiplier in size of the histogram and DE filtering buffers. Must be at least one, preferrably never larger than 4, only useful at 2.
//Xml field: "supersample" or "overample (deprecated)".
2016-04-03 21:55:12 -04:00
size_t m_Supersample = 1 ;
2014-12-07 02:51:44 -05:00
2014-07-08 03:11:14 -04:00
//When animating, split each pass into this many pieces, each doing a fraction of the total iterations. Each temporal sample
//will render an interpolated instance of the ember that is a fraction of the current ember and the next one.
//When rendering a single image, this field is always set to 1.
//Xml field: "temporal_samples".
2016-04-03 21:55:12 -04:00
size_t m_TemporalSamples = 100 ;
2014-07-08 03:11:14 -04:00
//Whether or not any symmetry was added. This field is in a bit of a state of conflict right now as flam3 has a severe bug.
//Xml field: "symmetry".
2016-04-03 21:55:12 -04:00
intmax_t m_Symmetry = 0 ;
2014-07-08 03:11:14 -04:00
//The number of iterations per pixel of the final output image. Note this is not affected by the increase in pixels in the
//histogram and DE filtering buffer due to supersampling. It can be affected by a non-zero zoom value though.
//10 is a good value for interactive/real-time rendering, 100-200 is good for previewing, 1000 is good for final output. Above that is mostly a waste of energy.
//Xml field: "quality".
2016-04-03 21:55:12 -04:00
T m_Quality = 100 ;
2014-07-08 03:11:14 -04:00
//The number of pixels in the final output image that corresponds to the distance from 0-1 in the cartesian plane used for iterating.
//A larger value produces a more zoomed in imgage. A value of 240 is commonly used, but in practice it varies widely depending on the image.
//Note that increasing this value does not adjust the quality by a proportional amount, so an increased value may produce a degraded image.
//Xml field: "scale".
2016-04-03 21:55:12 -04:00
T m_PixelsPerUnit = 240 ;
2014-09-10 01:41:26 -04:00
//A value greater than 0 will zoom in the field of view, however it will also increase the quality by a proportional amount. This is used to
2014-07-08 03:11:14 -04:00
//overcome the shortcoming of scale by also adjusting the quality.
//Xml field: "zoom".
2016-04-03 21:55:12 -04:00
T m_Zoom = 0 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//3D fields.
2014-09-10 01:41:26 -04:00
private :
2014-07-08 03:11:14 -04:00
typedef void ( Ember < T > : : * ProjFuncPtr ) ( Point < T > & , QTIsaac < ISAAC_SIZE , ISAAC_INT > & ) ;
ProjFuncPtr m_ProjFunc ;
public :
//Xml field: "cam_zpos".
2016-04-03 21:55:12 -04:00
T m_CamZPos = 0 ;
2014-07-08 03:11:14 -04:00
//Xml field: "cam_persp".
2016-04-03 21:55:12 -04:00
T m_CamPerspective = 0 ;
2014-07-08 03:11:14 -04:00
//Xml field: "cam_yaw".
2016-04-03 21:55:12 -04:00
T m_CamYaw = 0 ;
2014-07-08 03:11:14 -04:00
//Xml field: "cam_pitch".
2016-04-03 21:55:12 -04:00
T m_CamPitch = 0 ;
2014-07-08 03:11:14 -04:00
//Xml field: "cam_dof".
2016-04-03 21:55:12 -04:00
T m_CamDepthBlur = 0 ;
2014-07-08 03:11:14 -04:00
private :
2016-04-03 21:55:12 -04:00
T m_BlurCoef = 0 ;
2014-07-08 03:11:14 -04:00
public :
m3T m_CamMat ;
//The camera offset from the center of the cartesian plane. Since this is the camera offset, the final output image will be moved in the opposite
2014-10-27 17:21:06 -04:00
//direction of the values specified. There is also a second copy of the Y coordinate needed because m_CenterY will be modified during strips rendering.
2014-07-08 03:11:14 -04:00
//Xml field: "center".
2016-04-03 21:55:12 -04:00
T m_CenterX = 0 ;
T m_CenterY = 0 ;
T m_RotCenterY = 0 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Rotate the camera by this many degrees. Since this is a camera rotation, the final output image will be rotated counter-clockwise.
//Xml field: "rotate".
2016-04-03 21:55:12 -04:00
T m_Rotate = 0 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Determine how bright to make the image during final accumulation.
//Xml field: "brightness".
2016-04-03 21:55:12 -04:00
T m_Brightness = 4 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Gamma level used in gamma correction during final accumulation.
//Xml field: "gamma".
2016-04-03 21:55:12 -04:00
T m_Gamma = 4 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Used in color correction during final accumulation.
//Xml field: "vibrancy".
2016-04-03 21:55:12 -04:00
T m_Vibrancy = 1 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Gamma threshold used in gamma correction during final accumulation.
//Xml field: "gamma_threshold".
2016-04-03 21:55:12 -04:00
T m_GammaThresh = T ( 0.01 ) ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Value to control saturation of some pixels in gamma correction during final accumulation.
//Xml field: "highlight_power".
2016-04-03 21:55:12 -04:00
T m_HighlightPower = - 1 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//When animating a file full of many embers, this value is used to specify the time in the animation
//that this ember should be rendered. They must all be sequential and increase by a default value of 1.
//Xml field: "time".
2016-04-03 21:55:12 -04:00
T m_Time = 0 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//The background color of the image used in final accumulation, ranged 0-1.
//Xml field: "background".
Color < T > m_Background ;
//Animation/interpolation.
//The type of interpolation to use when interpolating between embers for animation.
//Xml field: "interpolation".
2016-04-03 21:55:12 -04:00
eInterp m_Interp = eInterp : : EMBER_INTERP_SMOOTH ;
2014-07-08 03:11:14 -04:00
//The type of interpolation to use on affine transforms when interpolating embers for animation.
//Xml field: "interpolation_type" or "interpolation_space (deprecated)".
2016-04-03 21:55:12 -04:00
eAffineInterp m_AffineInterp = eAffineInterp : : AFFINE_INTERP_LOG ;
2014-07-08 03:11:14 -04:00
//The type of interpolation to use for the palette when interpolating embers for animation.
//Xml field: "palette_interpolation".
2016-04-03 21:55:12 -04:00
ePaletteInterp m_PaletteInterp = ePaletteInterp : : INTERP_HSV ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Temporal Filter.
//Only used if temporal filter type is exp, else unused.
//Xml field: "temporal_filter_exp".
2016-04-03 21:55:12 -04:00
T m_TemporalFilterExp = 0 ;
2014-07-08 03:11:14 -04:00
//The width of the temporal filter.
//Xml field: "temporal_filter_width".
2016-04-03 21:55:12 -04:00
T m_TemporalFilterWidth = 1 ;
2014-07-08 03:11:14 -04:00
//The type of the temporal filter: Gaussian, Box or Exp.
//Xml field: "temporal_filter_type".
2016-04-03 21:55:12 -04:00
eTemporalFilterType m_TemporalFilterType = eTemporalFilterType : : BOX_TEMPORAL_FILTER ;
2014-07-08 03:11:14 -04:00
//Density Estimation Filter.
//The minimum radius of the DE filter.
//Xml field: "estimator_minimum".
2016-04-03 21:55:12 -04:00
T m_MinRadDE = 0 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//The maximum radius of the DE filter.
//Xml field: "estimator_radius".
2016-04-03 21:55:12 -04:00
T m_MaxRadDE = 9 ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//The shape of the curve that governs how quickly or slowly the filter drops off as it moves away from the center point.
//Xml field: "estimator_curve".
2016-04-03 21:55:12 -04:00
T m_CurveDE = T ( 0.4 ) ;
2014-07-08 03:11:14 -04:00
//Spatial Filter.
//The radius of the spatial filter used in final accumulation.
//Xml field: "filter".
2016-04-03 21:55:12 -04:00
T m_SpatialFilterRadius = T ( 0.5 ) ;
2014-07-08 03:11:14 -04:00
//The type of spatial filter used in final accumulation:
//Gaussian, Hermite, Box, Triangle, Bell, Bspline, Lanczos3
//Lanczos2, Mitchell, Blackman, Catrom, Hamming, Hanning, Quadratic.
//Xml field: "filter_shape".
2016-04-03 21:55:12 -04:00
eSpatialFilterType m_SpatialFilterType = eSpatialFilterType : : GAUSSIAN_SPATIAL_FILTER ;
2014-09-10 01:41:26 -04:00
2014-07-08 03:11:14 -04:00
//Palette.
//The method used for retrieving a color from the palette when accumulating to the histogram: step, linear.
//Xml field: "palette_mode".
2016-04-03 21:55:12 -04:00
ePaletteMode m_PaletteMode = ePaletteMode : : PALETTE_LINEAR ;
2014-07-08 03:11:14 -04:00
//The color palette to use. Can be specified inline as Xml color fields, or as a hex buffer. Can also be specified
//as an index into the palette file with an optional hue rotation applied. Inserting as a hex buffer is the preferred method.
//Xml field: "color" or "colors" or "palette" .
2016-04-03 21:55:12 -04:00
Palette < T > m_Palette ; //Final palette that is actually used is a copy of this inside of render, which will be of type bucketT (float).
2014-07-08 03:11:14 -04:00
2015-03-21 18:27:37 -04:00
//Curves used to adjust the color during final accumulation.
Curves < T > m_Curves ;
2014-07-08 03:11:14 -04:00
//Strings.
//The name of this ember.
//Xml field: "name".
2016-04-03 21:55:12 -04:00
string m_Name = string ( " No name " ) ;
2014-07-08 03:11:14 -04:00
//The name of the file that this ember was contained in.
//Xml field: "".
2016-04-03 21:55:12 -04:00
string m_ParentFilename = string ( " No parent " ) ;
2014-07-08 03:11:14 -04:00
//An Xml edit document describing information about the author as well as an edit history of the ember.
//Xml field: "edit".
2016-04-03 21:55:12 -04:00
xmlDocPtr m_Edits = nullptr ;
2014-07-08 03:11:14 -04:00
//The 0-based position of this ember in the file it was contained in.
2016-04-03 21:55:12 -04:00
size_t m_Index = 0 ;
2014-07-08 03:11:14 -04:00
2015-06-24 06:23:17 -04:00
//The list of motion elements for the top-level flame params
2015-07-07 00:36:46 -04:00
vector < EmberMotion < T > > m_EmberMotionElements ;
2015-06-24 06:23:17 -04:00
2014-07-08 03:11:14 -04:00
private :
2014-10-14 11:53:15 -04:00
/// <summary>
/// The type of scaling used when resizing.
/// </summary>
2016-04-03 21:55:12 -04:00
eScaleType m_ScaleType = eScaleType : : SCALE_NONE ;
2014-10-14 11:53:15 -04:00
2015-03-21 18:27:37 -04:00
//The vector containing all of the xforms.
//Xml field: "xform".
vector < Xform < T > > m_Xforms ;
//Optional final xform. Default is empty.
//Discussed in section 3.2 of the paper, page 6.
//Xml field: "finalxform".
Xform < T > m_FinalXform ;
2016-02-18 21:58:24 -05:00
//Single global reference to create variations with.
2016-04-13 23:59:57 -04:00
shared_ptr < VariationList < T > > m_VariationList = VariationList < T > : : Instance ( ) ;
2016-02-18 21:58:24 -05:00
2014-07-08 03:11:14 -04:00
/// <summary>
2014-07-22 10:53:36 -04:00
/// Interpolation function that takes the address of a member variable of type T as a template parameter.
/// This is an alternative to using macros.
2014-07-08 03:11:14 -04:00
/// </summary>
2014-07-22 10:53:36 -04:00
/// <param name="embers">The list of embers to interpolate</param>
/// <param name="coefs">The list of coefficients to interpolate</param>
/// <param name="size">The size of the lists, both must match.</param>
2014-07-08 03:11:14 -04:00
template < T Ember < T > : : * m >
2016-04-03 21:55:12 -04:00
void InterpT ( const Ember < T > * embers , const vector < T > & coefs , size_t size )
2014-07-08 03:11:14 -04:00
{
this - > * m = 0 ;
for ( size_t k = 0 ; k < size ; k + + )
this - > * m + = coefs [ k ] * embers [ k ] . * m ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Interpolation function that takes the address of a member variable of any type as a template parameter.
/// </summary>
/// <param name="embers">The list of embers to interpolate</param>
/// <param name="coefs">The list of coefficients to interpolate</param>
/// <param name="size">The size of the lists, both must match.</param>
2014-07-08 03:11:14 -04:00
template < typename M , M Ember < T > : : * m >
2016-04-03 21:55:12 -04:00
void InterpX ( const Ember < T > * embers , const vector < T > & coefs , size_t size )
2014-07-08 03:11:14 -04:00
{
this - > * m = M ( ) ;
for ( size_t k = 0 ; k < size ; k + + )
this - > * m + = coefs [ k ] * embers [ k ] . * m ;
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Interpolation function that takes the address of a member variable of type integer as a template parameter.
/// </summary>
/// <param name="embers">The list of embers to interpolate</param>
/// <param name="coefs">The list of coefficients to interpolate</param>
/// <param name="size">The size of the lists, both must match.</param>
2014-10-14 11:53:15 -04:00
template < size_t Ember < T > : : * m >
2016-04-03 21:55:12 -04:00
void InterpI ( const Ember < T > * embers , const vector < T > & coefs , size_t size )
2014-07-08 03:11:14 -04:00
{
T t = 0 ;
for ( size_t k = 0 ; k < size ; k + + )
t + = coefs [ k ] * embers [ k ] . * m ;
--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
this - > * m = size_t ( std : : rint ( t ) ) ;
2014-07-08 03:11:14 -04:00
}
2014-07-22 10:53:36 -04:00
/// <summary>
/// Interpolation function that takes the address of an xform member variable of type T as a template parameter.
/// This is an alternative to using macros.
/// </summary>
/// <param name="xform">A pointer to a list of xforms to interpolate</param>
/// <param name="i">The xform index to interpolate</param>
/// <param name="embers">The list of embers to interpolate</param>
/// <param name="coefs">The list of coefficients to interpolate</param>
/// <param name="size">The size of the lists, both must match.</param>
2014-07-08 03:11:14 -04:00
template < T Xform < T > : : * m >
2016-04-03 21:55:12 -04:00
void InterpXform ( Xform < T > * xform , size_t i , const Ember < T > * embers , const vector < T > & coefs , size_t size )
2014-07-08 03:11:14 -04:00
{
xform - > * m = T ( 0 ) ;
for ( size_t k = 0 ; k < size ; k + + )
xform - > * m + = coefs [ k ] * embers [ k ] . GetTotalXform ( i ) - > * m ;
}
} ;
/// <summary>
/// Comparer for sorting embers based on time.
/// </summary>
/// <param name="av">Pointer to the first ember to compare</param>
/// <param name="bv">Pointer to the second ember to compare</param>
/// <returns>True if av's time is less than bv's time, else false.</returns>
template < typename T >
static inline bool CompareEmbers ( const Ember < T > & a , const Ember < T > & b )
{
return a . m_Time < b . m_Time ;
}
2014-09-10 01:41:26 -04:00
}