#include <dali/public-api/common/dali-common.h>
#include <dali/public-api/common/constants.h>
-namespace //Unnamed namespace
+namespace
{
//Memory pool used to allocate new nodes. Memory used by this pool will be released when process dies
// or DALI library is unloaded
Dali::Internal::MemoryPoolObjectAllocator<Dali::Internal::SceneGraph::Node> gNodeMemoryPool;
#ifdef DEBUG_ENABLED
-// keep track of nodes created / deleted, to ensure we have 0 when the process exits or DALi library is unloaded
-int32_t gNodeCount =0;
+// keep track of nodes alive, to ensure we have 0 when the process exits or DALi library is unloaded
+int32_t gNodeCount = 0;
// Called when the process is about to exit, Node count should be zero at this point.
void __attribute__ ((destructor)) ShutDown(void)
DALI_ASSERT_DEBUG( (gNodeCount == 0) && "Node memory leak");
}
#endif
-}
+} // Unnamed namespace
namespace Dali
{
namespace SceneGraph
{
-const PositionInheritanceMode Node::DEFAULT_POSITION_INHERITANCE_MODE( INHERIT_PARENT_POSITION );
const ColorMode Node::DEFAULT_COLOR_MODE( USE_OWN_MULTIPLY_PARENT_ALPHA );
-Node* Node::New( uint32_t id )
+uint32_t Node::mNodeCounter = 0; ///< A counter to provide unique node ids, up-to 4 billion
+
+Node* Node::New()
{
- return new ( gNodeMemoryPool.AllocateRawThreadSafe() ) Node( id );
+ return new ( gNodeMemoryPool.AllocateRawThreadSafe() ) Node;
}
void Node::Delete( Node* node )
}
}
-Node::Node( uint32_t id )
+Node::Node()
: mTransformManager( NULL ),
mTransformId( INVALID_TRANSFORM_ID ),
mParentOrigin( TRANSFORM_PROPERTY_PARENT_ORIGIN ),
mPosition( TRANSFORM_PROPERTY_POSITION ), // Zero initialized by default
mOrientation(), // Initialized to identity by default
mScale( TRANSFORM_PROPERTY_SCALE ),
+ mUpdateSizeHint( TRANSFORM_PROPERTY_UPDATE_SIZE_HINT ),
mVisible( true ),
mCulled( false ),
mColor( Color::WHITE ),
mWorldMatrix(),
mWorldColor( Color::WHITE ),
mClippingSortModifier( 0u ),
- mId( id ),
+ mId( ++mNodeCounter ),
mParent( NULL ),
mExclusiveRenderTask( NULL ),
mChildren(),
{
mUniformMapChanged[0] = 0u;
mUniformMapChanged[1] = 0u;
+ mPropertyDirty = false;
#ifdef DEBUG_ENABLED
gNodeCount++;
PropertyOwner::Destroy();
}
+uint32_t Node::GetId() const
+{
+ return mId;
+}
+
void Node::CreateTransform( SceneGraph::TransformManager* transformManager )
{
//Create a new transform
//Initialize all the animatable properties
mPosition.Initialize( transformManager, mTransformId );
mScale.Initialize( transformManager, mTransformId );
+ mUpdateSizeHint.Initialize( transformManager, mTransformId );
mOrientation.Initialize( transformManager, mTransformId );
mSize.Initialize( transformManager, mTransformId );
mParentOrigin.Initialize( transformManager, mTransformId );
CollectedUniformMap& localMap = mCollectedUniformMap[ bufferIndex ];
localMap.Resize(0);
- for( uint32_t i = 0, count=mUniformMaps.Count(); i<count; ++i )
+ for( UniformMap::SizeType i = 0, count=mUniformMaps.Count(); i<count; ++i )
{
localMap.PushBack( &mUniformMaps[i] );
}
}
mRenderer.PushBack( renderer );
+ SetPropertyDirty( true );
}
-void Node::RemoveRenderer( Renderer* renderer )
+void Node::RemoveRenderer( const Renderer* renderer )
{
RendererContainer::SizeType rendererCount( mRenderer.Size() );
for( RendererContainer::SizeType i = 0; i < rendererCount; ++i )
{
if( mRenderer[i] == renderer )
{
+ SetPropertyDirty( true );
mRenderer.Erase( mRenderer.Begin()+i);
return;
}
void Node::ResetDirtyFlags( BufferIndex updateBufferIndex )
{
mDirtyFlags = NodePropertyFlags::NOTHING;
+
+ SetPropertyDirty( false );
+
}
void Node::SetParent( Node& parentNode )
}
}
+void Node::SetPropertyDirty( bool value )
+{
+ mPropertyDirty = value;
+
+ const NodeIter endIter = mChildren.End();
+ for ( NodeIter iter = mChildren.Begin(); iter != endIter; ++iter )
+ {
+ Node* current = *iter;
+ current->SetPropertyDirty( value );
+ }
+}
+
+bool Node::IsPropertyDirty() const
+{
+ for( auto&& existingRenderer : mRenderer )
+ {
+ if( existingRenderer->IsDirty() )
+ {
+ return true;
+ }
+ }
+
+ return mPropertyDirty;
+}
+
} // namespace SceneGraph
} // namespace Internal