-//
-// Copyright (c) 2014 Samsung Electronics Co., Ltd.
-//
-// Licensed under the Flora License, Version 1.0 (the License);
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-//
-// http://floralicense.org/license/
-//
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an AS IS BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-//
+/*
+ * Copyright (c) 2014 Samsung Electronics Co., Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ */
// CLASS HEADER
#include <dali/internal/event/render-tasks/render-task-impl.h>
+// EXTERNAL INCLUDES
+#include <cstring> // for strcmp
+
// INTERNAL INCLUDES
#include <dali/public-api/common/dali-common.h>
-#include <dali/internal/common/event-to-update.h>
+#include <dali/public-api/object/type-registry.h>
+#include <dali/internal/event/common/event-thread-services.h>
#include <dali/internal/event/actors/actor-impl.h>
#include <dali/internal/event/actors/camera-actor-impl.h>
+#include <dali/internal/event/common/property-helper.h>
#include <dali/internal/event/common/stage-impl.h>
-#include <dali/internal/event/common/thread-local-storage.h>
+#include <dali/internal/event/common/projection.h>
#include <dali/internal/event/images/frame-buffer-image-impl.h>
-#include <dali/internal/update/render-tasks/scene-graph-render-task.h>
#include <dali/internal/update/nodes/node.h>
+#include <dali/internal/event/render-tasks/render-task-list-impl.h>
+#include <dali/internal/update/render-tasks/scene-graph-render-task.h>
#if defined(DEBUG_ENABLED)
namespace
namespace Dali
{
-const Property::Index RenderTask::VIEWPORT_POSITION = 0;
-const Property::Index RenderTask::VIEWPORT_SIZE = 1;
-const Property::Index RenderTask::CLEAR_COLOR = 2;
-
namespace Internal
{
namespace // For internal properties
{
-const std::string DEFAULT_PROPERTY_NAMES[] =
-{
- "viewport-position",
- "viewport-size",
- "clear-color"
-};
-const int DEFAULT_PROPERTY_COUNT = sizeof( DEFAULT_PROPERTY_NAMES ) / sizeof( std::string );
+// Properties
-const Property::Type DEFAULT_PROPERTY_TYPES[DEFAULT_PROPERTY_COUNT] =
-{
- Property::VECTOR2, // viewport-position
- Property::VECTOR2, // viewport-size
- Property::VECTOR4, // clear-color
-};
+// Name Type writable animatable constraint-input enum for index-checking
+DALI_PROPERTY_TABLE_BEGIN
+DALI_PROPERTY( "viewportPosition", VECTOR2, true, true, true, Dali::RenderTask::Property::VIEWPORT_POSITION )
+DALI_PROPERTY( "viewportSize", VECTOR2, true, true, true, Dali::RenderTask::Property::VIEWPORT_SIZE )
+DALI_PROPERTY( "clearColor", VECTOR4, true, true, true, Dali::RenderTask::Property::CLEAR_COLOR )
+DALI_PROPERTY_TABLE_END( DEFAULT_OBJECT_PROPERTY_START_INDEX )
-}// unnamed namespace
+// Signals
-RenderTask::DefaultPropertyLookup* RenderTask::sDefaultPropertyLookup = NULL;
+const char* const SIGNAL_FINISHED = "finished";
+
+TypeRegistration mType( typeid( Dali::RenderTask ), typeid( Dali::BaseHandle ), NULL );
+
+SignalConnectorType signalConnector1( mType, SIGNAL_FINISHED, &RenderTask::DoConnectSignal );
+
+} // Unnamed namespace
RenderTask* RenderTask::New( bool isSystemLevel )
{
- RenderTask* task( new RenderTask( ThreadLocalStorage::Get().GetEventToUpdate(), isSystemLevel ) );
+ RenderTask* task( new RenderTask( isSystemLevel ) );
return task;
}
void RenderTask::SetSourceActor( Actor* actor )
{
+ const Stage* stage = Stage::GetCurrent();
+ if ( stage )
+ {
+ stage->GetRenderTaskList().SetExclusive( this, mExclusive );
+ }
mSourceConnector.SetActor( actor );
}
{
mExclusive = exclusive;
+ const Stage* stage = Stage::GetCurrent();
+ if ( stage )
+ {
+ stage->GetRenderTaskList().SetExclusive( this, exclusive );
+ }
+
if ( mSceneObject )
{
// mSceneObject is being used in a separate thread; queue a message to set the value
- SetExclusiveMessage( mEventToUpdate, *mSceneObject, mExclusive );
+ SetExclusiveMessage( GetEventThreadServices(), *mSceneObject, mExclusive );
}
}
}
}
// mSceneObject is being used in a separate thread; queue a message to set the value
- SetFrameBufferIdMessage( mEventToUpdate, *mSceneObject, resourceId );
+ SetFrameBufferIdMessage( GetEventThreadServices(), *mSceneObject, resourceId );
}
else
{
void RenderTask::SetViewportPosition(const Vector2& value)
{
- StagePtr stage = Stage::GetCurrent();
- BakeViewportPositionMessage( stage->GetUpdateInterface(), *mSceneObject, value );
+ BakeViewportPositionMessage( GetEventThreadServices(), *mSceneObject, value );
}
Vector2 RenderTask::GetCurrentViewportPosition() const
{
- StagePtr stage = Stage::GetCurrent();
- return mSceneObject->GetViewportPosition(stage->GetEventBufferIndex());
+ return mSceneObject->GetViewportPosition( GetEventThreadServices().GetEventBufferIndex() );
}
void RenderTask::SetViewportSize(const Vector2& value)
{
- StagePtr stage = Stage::GetCurrent();
- BakeViewportSizeMessage( stage->GetUpdateInterface(), *mSceneObject, value );
+ BakeViewportSizeMessage( GetEventThreadServices(), *mSceneObject, value );
}
Vector2 RenderTask::GetCurrentViewportSize() const
{
- StagePtr stage = Stage::GetCurrent();
- return mSceneObject->GetViewportSize(stage->GetEventBufferIndex());
+ return mSceneObject->GetViewportSize( GetEventThreadServices().GetEventBufferIndex() );
}
void RenderTask::SetViewport( const Viewport& viewport )
void RenderTask::GetViewport( Viewport& viewPort ) const
{
- BufferIndex bufferIndex = Stage::GetCurrent()->GetEventBufferIndex();
+ BufferIndex bufferIndex = GetEventThreadServices().GetEventBufferIndex();
if(!mSceneObject->GetViewportEnabled( bufferIndex ))
{
}
else
{
- Vector2 size( Stage::GetCurrent()->GetSize() );
- viewPort.x = viewPort.y = 0;
- viewPort.width = size.width;
- viewPort.height = size.height;
+ Internal::Stage* stage = Internal::Stage::GetCurrent();
+ if ( stage )
+ {
+ Vector2 size( stage->GetSize() );
+ viewPort.x = viewPort.y = 0;
+ viewPort.width = size.width;
+ viewPort.height = size.height;
+ }
}
}
else
if ( mSceneObject )
{
// mSceneObject is being used in a separate thread; queue a message to set the value
- StagePtr stage = Stage::GetCurrent();
- BakeClearColorMessage( stage->GetUpdateInterface(), *mSceneObject, color );
+ BakeClearColorMessage( GetEventThreadServices(), *mSceneObject, color );
}
}
}
const Vector4& RenderTask::GetClearColor() const
{
- StagePtr stage = Stage::GetCurrent();
- return mSceneObject->GetClearColor(stage->GetEventBufferIndex());
+ return mSceneObject->GetClearColor( GetEventThreadServices().GetEventBufferIndex() );
}
void RenderTask::SetClearEnabled( bool enabled )
if ( mSceneObject )
{
// mSceneObject is being used in a separate thread; queue a message to set the value
- SetClearEnabledMessage( mEventToUpdate, *mSceneObject, mClearEnabled );
+ SetClearEnabledMessage( GetEventThreadServices(), *mSceneObject, mClearEnabled );
}
}
}
return mClearEnabled;
}
+void RenderTask::SetCullMode( bool mode )
+{
+ if ( mCullMode != mode )
+ {
+ mCullMode = mode;
+
+ if ( mSceneObject )
+ {
+ // mSceneObject is being used in a separate thread; queue a message to set the value
+ SetCullModeMessage( GetEventThreadServices(), *mSceneObject, mCullMode );
+ }
+ }
+}
+
+bool RenderTask::GetCullMode() const
+{
+ return mCullMode;
+}
+
void RenderTask::SetRefreshRate( unsigned int refreshRate )
{
DALI_LOG_TRACE_METHOD_FMT(gLogRender, "this:%p rate:%d\n", this, refreshRate);
if ( mSceneObject )
{
// mSceneObject is being used in a separate thread; queue a message to set the value
- SetRefreshRateMessage( mEventToUpdate, *mSceneObject, refreshRate );
+ SetRefreshRateMessage( GetEventThreadServices(), *mSceneObject, refreshRate );
}
}
{
CameraActor* localCamera = GetCameraActor();
StagePtr stage = Stage::GetCurrent();
- CameraActor& defaultCamera = stage->GetDefaultCameraActor();
- if( localCamera )
+ if( stage )
{
- Viewport viewport;
- Vector2 size( stage->GetSize() );
- viewport.x = viewport.y = 0;
- viewport.width = size.width;
- viewport.height = size.height;
-
- float localX, localY;
- inside = mMappingConnector.mActor->ScreenToLocal(defaultCamera.GetViewMatrix(), defaultCamera.GetProjectionMatrix(), viewport, localX, localY, screenCoords.x, screenCoords.y);
- Vector3 actorSize = mMappingConnector.mActor->GetCurrentSize();
- if( inside && localX >= 0.f && localX <= actorSize.x && localY >= 0.f && localY <= actorSize.y)
+ CameraActor& defaultCamera = stage->GetDefaultCameraActor();
+ if( localCamera )
{
- screenCoords.x = localX;
- screenCoords.y = localY;
- if( !localCamera->GetInvertYAxis() )
+ Viewport viewport;
+ Vector2 size( stage->GetSize() );
+ viewport.x = viewport.y = 0;
+ viewport.width = size.width;
+ viewport.height = size.height;
+
+ float localX, localY;
+ inside = mMappingConnector.mActor->ScreenToLocal(defaultCamera.GetViewMatrix(), defaultCamera.GetProjectionMatrix(), viewport, localX, localY, screenCoords.x, screenCoords.y);
+ Vector3 actorSize = mMappingConnector.mActor->GetCurrentSize();
+ if( inside && localX >= 0.f && localX <= actorSize.x && localY >= 0.f && localY <= actorSize.y)
{
- screenCoords.y = actorSize.y - localY;
+ screenCoords.x = localX;
+ screenCoords.y = localY;
+ }
+ else
+ {
+ inside = false;
}
}
else
inside = false;
}
}
- else
- {
- inside = false;
- }
}
else if ( mFrameBufferImage && mScreenToFrameBufferFunction )
{
return mIsSystemLevel;
}
+bool RenderTask::WorldToViewport(const Vector3 &position, float& viewportX, float& viewportY) const
+{
+ CameraActor* cam = GetCameraActor();
+
+ Vector4 pos(position);
+ pos.w = 1.0;
+
+ Vector4 viewportPosition;
+
+ Viewport viewport;
+ GetViewport( viewport );
+
+ bool ok = ProjectFull(pos,
+ cam->GetViewMatrix(),
+ cam->GetProjectionMatrix(),
+ viewport.x,
+ viewport.y,
+ viewport.width,
+ viewport.height,
+ viewportPosition);
+ if(ok)
+ {
+ viewportX = viewportPosition.x;
+ viewportY = viewportPosition.y;
+ }
+
+ return ok;
+}
+
+bool RenderTask::ViewportToLocal(Actor* actor, float viewportX, float viewportY, float &localX, float &localY) const
+{
+ return actor->ScreenToLocal( *this, localX, localY, viewportX, viewportY );
+}
+
SceneGraph::RenderTask* RenderTask::CreateSceneObject()
{
// This should only be called once, with no existing scene-object
}
// mSceneObject is being used in a separate thread; queue a message to set the value
- SetFrameBufferIdMessage( mEventToUpdate, *mSceneObject, resourceId );
+ SetFrameBufferIdMessage( GetEventThreadServices(), *mSceneObject, resourceId );
// Send messages to set other properties that may have changed since last time we were on stage
- SetExclusiveMessage( mEventToUpdate, *mSceneObject, mExclusive );
- SetClearColorMessage( mEventToUpdate, *mSceneObject, mClearColor );
- SetClearEnabledMessage( mEventToUpdate, *mSceneObject, mClearEnabled );
- SetRefreshRateMessage( mEventToUpdate, *mSceneObject, mRefreshRate );
+ SetExclusiveMessage( GetEventThreadServices(), *mSceneObject, mExclusive );
+ SetClearColorMessage( GetEventThreadServices(), *mSceneObject, mClearColor );
+ SetClearEnabledMessage( GetEventThreadServices(), *mSceneObject, mClearEnabled );
+ SetCullModeMessage( GetEventThreadServices(), *mSceneObject, mCullMode );
+ SetRefreshRateMessage( GetEventThreadServices(), *mSceneObject, mRefreshRate );
// Caller takes ownership
return mSceneObject;
}
/********************************************************************************
- ********************************************************************************
******************************** PROPERTY METHODS **************************
- ********************************************************************************
********************************************************************************/
-bool RenderTask::IsSceneObjectRemovable() const
+unsigned int RenderTask::GetDefaultPropertyCount() const
{
- return false; // The scene object is permanently "on stage" whilst this object is alive
+ return DEFAULT_PROPERTY_COUNT;
}
-unsigned int RenderTask::GetDefaultPropertyCount() const
+void RenderTask::GetDefaultPropertyIndices( Property::IndexContainer& indices ) const
{
- return DEFAULT_PROPERTY_COUNT;
+ indices.Reserve( DEFAULT_PROPERTY_COUNT );
+
+ for ( int i = 0; i < DEFAULT_PROPERTY_COUNT; ++i )
+ {
+ indices.PushBack( i );
+ }
}
-const std::string& RenderTask::GetDefaultPropertyName( Property::Index index ) const
+const char* RenderTask::GetDefaultPropertyName( Property::Index index ) const
{
- // ProxyObject guarantees that index is within range
- return DEFAULT_PROPERTY_NAMES[index];
+ if( index < DEFAULT_PROPERTY_COUNT )
+ {
+ return DEFAULT_PROPERTY_DETAILS[index].name;
+ }
+ else
+ {
+ return NULL;
+ }
}
Property::Index RenderTask::GetDefaultPropertyIndex(const std::string& name) const
{
Property::Index index = Property::INVALID_INDEX;
- // Lazy initialization of static sDefaultPropertyLookup
- if (!sDefaultPropertyLookup)
+ // Look for name in default properties
+ for( int i = 0; i < DEFAULT_PROPERTY_COUNT; ++i )
{
- sDefaultPropertyLookup = new DefaultPropertyLookup();
-
- for (int i=0; i<DEFAULT_PROPERTY_COUNT; ++i)
+ if( 0 == strcmp( name.c_str(), DEFAULT_PROPERTY_DETAILS[i].name ) ) // dont want to convert rhs to string
{
- (*sDefaultPropertyLookup)[DEFAULT_PROPERTY_NAMES[i]] = i;
+ index = i;
+ break;
}
}
- DALI_ASSERT_DEBUG( NULL != sDefaultPropertyLookup );
-
- // Look for name in default properties
- DefaultPropertyLookup::const_iterator result = sDefaultPropertyLookup->find( name );
- if ( sDefaultPropertyLookup->end() != result )
- {
- index = result->second;
- }
return index;
}
bool RenderTask::IsDefaultPropertyWritable(Property::Index index) const
{
- return true;
+ return DEFAULT_PROPERTY_DETAILS[ index ].writable;
}
bool RenderTask::IsDefaultPropertyAnimatable(Property::Index index) const
{
- return true;
+ return DEFAULT_PROPERTY_DETAILS[ index ].animatable;
+}
+
+bool RenderTask::IsDefaultPropertyAConstraintInput( Property::Index index ) const
+{
+ return DEFAULT_PROPERTY_DETAILS[ index ].constraintInput;
}
Property::Type RenderTask::GetDefaultPropertyType(Property::Index index) const
{
- // ProxyObject guarantees that index is within range
- return DEFAULT_PROPERTY_TYPES[index];
+ if( index < DEFAULT_PROPERTY_COUNT )
+ {
+ return DEFAULT_PROPERTY_DETAILS[index].type;
+ }
+
+ // index out of range...return Property::NONE
+ return Property::NONE;
}
void RenderTask::SetDefaultProperty( Property::Index index, const Property::Value& property )
{
- // ProxyObject guarantees the property is writable and index is in range
switch ( index )
{
- case Dali::RenderTask::VIEWPORT_POSITION:
+ case Dali::RenderTask::Property::VIEWPORT_POSITION:
{
SetViewportPosition( property.Get<Vector2>() );
break;
}
- case Dali::RenderTask::VIEWPORT_SIZE:
+ case Dali::RenderTask::Property::VIEWPORT_SIZE:
{
SetViewportSize( property.Get<Vector2>() );
break;
}
- case Dali::RenderTask::CLEAR_COLOR:
+ case Dali::RenderTask::Property::CLEAR_COLOR:
{
SetClearColor( property.Get<Vector4>() );
break;
}
-
default:
{
- DALI_ASSERT_ALWAYS(false && "RenderTask property index out of range"); // should not come here
+ // nothing to do
break;
}
}
}
-void RenderTask::SetCustomProperty( Property::Index /*index*/, const CustomProperty& /*entry*/, const Property::Value& /*value*/ )
-{
- // TODO: support them, it doesn't hurt.
- DALI_ASSERT_ALWAYS( 0 && "RenderTask does not support custom properties");
-}
-
Property::Value RenderTask::GetDefaultProperty(Property::Index index) const
{
Property::Value value;
- // ProxyObject guarantees that index is within range
switch ( index )
{
- case Dali::RenderTask::VIEWPORT_POSITION:
+ case Dali::RenderTask::Property::VIEWPORT_POSITION:
{
value = GetCurrentViewportPosition();
break;
}
- case Dali::RenderTask::VIEWPORT_SIZE:
+ case Dali::RenderTask::Property::VIEWPORT_SIZE:
{
value = GetCurrentViewportSize();
break;
}
- case Dali::RenderTask::CLEAR_COLOR:
+ case Dali::RenderTask::Property::CLEAR_COLOR:
{
value = GetClearColor();
break;
return value;
}
-void RenderTask::InstallSceneObjectProperty( SceneGraph::PropertyBase& newProperty, const std::string& name, unsigned int index )
-{
- // TODO: support them, it doesn't hurt.
- DALI_ASSERT_ALWAYS( 0 && "RenderTask does not support custom properties" );
-}
-
const SceneGraph::PropertyOwner* RenderTask::GetSceneObject() const
{
return mSceneObject;
{
switch ( index )
{
- case Dali::RenderTask::VIEWPORT_POSITION:
+ case Dali::RenderTask::Property::VIEWPORT_POSITION:
property = &mSceneObject->mViewportPosition;
break;
- case Dali::RenderTask::VIEWPORT_SIZE:
+ case Dali::RenderTask::Property::VIEWPORT_SIZE:
property = &mSceneObject->mViewportSize;
break;
- case Dali::RenderTask::CLEAR_COLOR:
+ case Dali::RenderTask::Property::CLEAR_COLOR:
property = &mSceneObject->mClearColor;
break;
{
switch ( index )
{
- case Dali::RenderTask::VIEWPORT_POSITION:
+ case Dali::RenderTask::Property::VIEWPORT_POSITION:
property = &mSceneObject->mViewportPosition;
break;
- case Dali::RenderTask::VIEWPORT_SIZE:
+ case Dali::RenderTask::Property::VIEWPORT_SIZE:
property = &mSceneObject->mViewportSize;
break;
- case Dali::RenderTask::CLEAR_COLOR:
+ case Dali::RenderTask::Property::CLEAR_COLOR:
property = &mSceneObject->mViewportSize;
break;
{
DALI_LOG_INFO(gLogRender, Debug::General, "RenderTask::EmitSignalFinish(this:%p)\n", this);
- if( !mSignalFinishedV2.Empty() )
+ if( !mSignalFinished.Empty() )
{
Dali::RenderTask handle( this );
- mSignalFinishedV2.Emit(handle );
+ mSignalFinished.Emit(handle );
}
}
-Dali::RenderTask::RenderTaskSignalV2& RenderTask::FinishedSignal()
+Dali::RenderTask::RenderTaskSignalType& RenderTask::FinishedSignal()
{
- return mSignalFinishedV2;
+ return mSignalFinished;
}
bool RenderTask::DoConnectSignal( BaseObject* object, ConnectionTrackerInterface* tracker, const std::string& signalName, FunctorDelegate* functor )
bool connected( true );
RenderTask* renderTask = dynamic_cast<RenderTask*>(object);
- if ( Dali::RenderTask::SIGNAL_FINISHED == signalName )
+ if ( 0 == strcmp( signalName.c_str(), SIGNAL_FINISHED ) )
{
renderTask->FinishedSignal().Connect( tracker, functor );
}
return connected;
}
-RenderTask::RenderTask( EventToUpdate& eventToUpdate, bool isSystemLevel )
-: mEventToUpdate( eventToUpdate ),
- mSceneObject( NULL ),
+RenderTask::RenderTask( bool isSystemLevel )
+: mSceneObject( NULL ),
mSourceConnector( Connector::SOURCE_CONNECTOR, *this ),
mCameraConnector( Connector::CAMERA_CONNECTOR, *this ),
mMappingConnector( Connector::MAPPING_CONNECTOR, *this ),
mExclusive( Dali::RenderTask::DEFAULT_EXCLUSIVE ),
mInputEnabled( Dali::RenderTask::DEFAULT_INPUT_ENABLED ),
mClearEnabled( Dali::RenderTask::DEFAULT_CLEAR_ENABLED ),
+ mCullMode( Dali::RenderTask::DEFAULT_CULL_MODE ),
mIsSystemLevel( isSystemLevel )
{
DALI_LOG_INFO(gLogRender, Debug::General, "RenderTask::RenderTask(this:%p)\n", this);
}
}
-void RenderTask::Connector::SceneObjectAdded( ProxyObject& proxy )
+void RenderTask::Connector::SceneObjectAdded( Object& object )
{
UpdateRenderTask();
}
-void RenderTask::Connector::SceneObjectRemoved( ProxyObject& proxy )
+void RenderTask::Connector::SceneObjectRemoved( Object& object )
{
UpdateRenderTask();
}
-void RenderTask::Connector::ProxyDestroyed( ProxyObject& proxy )
+void RenderTask::Connector::ObjectDestroyed( Object& object )
{
+ if ( SOURCE_CONNECTOR == mType )
+ {
+ const Stage* stage = Stage::GetCurrent();
+ if ( stage )
+ {
+ stage->GetRenderTaskList().SetExclusive( &mRenderTask, false );
+ }
+ }
+
mActor = NULL;
UpdateRenderTask();
//the mapping node is not used in the scene graph
if ( SOURCE_CONNECTOR == mType )
{
- SetSourceNodeMessage( mRenderTask.mEventToUpdate, *(mRenderTask.mSceneObject), node );
+ SetSourceNodeMessage( mRenderTask.GetEventThreadServices(), *(mRenderTask.mSceneObject), node );
}
else if( CAMERA_CONNECTOR == mType )
{
- SetCameraNodeMessage( mRenderTask.mEventToUpdate, *(mRenderTask.mSceneObject), node );
+ SetCameraNodeMessage( mRenderTask.GetEventThreadServices(), *(mRenderTask.mSceneObject), node );
}
}
}