2 * Copyright (c) 2014 Samsung Electronics Co., Ltd.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
19 #include <dali/internal/event/animation/animation-impl.h>
20 #include <dali/public-api/object/property-map.h>
25 #include <dali/public-api/actors/actor.h>
26 #include <dali/public-api/animation/alpha-function.h>
27 #include <dali/public-api/animation/time-period.h>
28 #include <dali/public-api/common/dali-common.h>
29 #include <dali/public-api/object/type-registry.h>
30 #include <dali/public-api/math/vector2.h>
31 #include <dali/public-api/math/radian.h>
32 #include <dali/internal/event/actors/actor-impl.h>
33 #include <dali/internal/event/animation/animation-playlist.h>
34 #include <dali/internal/event/animation/animator-connector.h>
35 #include <dali/internal/event/common/notification-manager.h>
36 #include <dali/internal/event/common/property-helper.h>
37 #include <dali/internal/event/common/stage-impl.h>
38 #include <dali/internal/event/common/thread-local-storage.h>
39 #include <dali/internal/event/effects/shader-effect-impl.h>
40 #include <dali/internal/update/manager/update-manager.h>
42 using Dali::Internal::SceneGraph::UpdateManager;
43 using Dali::Internal::SceneGraph::AnimatorBase;
44 using Dali::Internal::SceneGraph::Shader;
52 static bool SHOW_VALUE = true;
53 static bool HIDE_VALUE = false;
60 const char* const SIGNAL_FINISHED = "finished";
64 const char* const ACTION_PLAY = "play";
65 const char* const ACTION_STOP = "stop";
66 const char* const ACTION_PAUSE = "pause";
70 return Dali::Animation::New(0.f);
73 TypeRegistration mType( typeid( Dali::Animation ), typeid( Dali::BaseHandle ), Create );
75 SignalConnectorType signalConnector1( mType, SIGNAL_FINISHED, &Animation::DoConnectSignal );
77 TypeAction action1( mType, ACTION_PLAY, &Animation::DoAction );
78 TypeAction action2( mType, ACTION_STOP, &Animation::DoAction );
79 TypeAction action3( mType, ACTION_PAUSE, &Animation::DoAction );
81 const Dali::Animation::EndAction DEFAULT_END_ACTION( Dali::Animation::Bake );
82 const Dali::Animation::EndAction DEFAULT_DISCONNECT_ACTION( Dali::Animation::BakeFinal );
83 const Dali::Animation::Interpolation DEFAULT_INTERPOLATION( Dali::Animation::Linear );
84 const Dali::AlphaFunction DEFAULT_ALPHA_FUNCTION( Dali::AlphaFunction::DEFAULT );
89 AnimationPtr Animation::New(float durationSeconds)
91 Stage* stage = Stage::GetCurrent();
95 AnimationPlaylist& playlist = stage->GetAnimationPlaylist();
97 if( durationSeconds < 0.0f )
99 DALI_LOG_WARNING("duration should be greater than 0.0f.\n");
100 durationSeconds = 0.0f;
103 AnimationPtr animation = new Animation( *stage, playlist, durationSeconds, DEFAULT_END_ACTION, DEFAULT_DISCONNECT_ACTION, DEFAULT_ALPHA_FUNCTION );
105 // Second-phase construction
106 animation->Initialize();
116 Animation::Animation( EventThreadServices& eventThreadServices, AnimationPlaylist& playlist, float durationSeconds, EndAction endAction, EndAction disconnectAction, AlphaFunction defaultAlpha )
117 : mEventThreadServices( eventThreadServices ),
118 mPlaylist( playlist ),
120 mNotificationCount( 0 ),
121 mFinishedCallback( NULL ),
122 mFinishedCallbackObject( NULL ),
123 mDurationSeconds( durationSeconds ),
126 mPlayRange( Vector2(0.0f,1.0f)),
127 mEndAction( endAction ),
128 mDisconnectAction( disconnectAction ),
129 mDefaultAlpha( defaultAlpha )
133 void Animation::Initialize()
135 // Connect to the animation playlist
136 mPlaylist.AnimationCreated( *this );
143 Animation::~Animation()
145 // Guard to allow handle destruction after Core has been destroyed
146 if ( Stage::IsInstalled() )
148 // Disconnect from the animation playlist
149 mPlaylist.AnimationDestroyed( *this );
151 DestroySceneObject();
157 void Animation::CreateSceneObject()
159 DALI_ASSERT_DEBUG( mAnimation == NULL );
161 // Create a new animation, temporarily owned
162 SceneGraph::Animation* animation = SceneGraph::Animation::New( mDurationSeconds, mSpeedFactor, mPlayRange, mIsLooping, mEndAction, mDisconnectAction );
164 // Keep a const pointer to the animation.
165 mAnimation = animation;
167 // Transfer animation ownership to the update manager through a message
168 AddAnimationMessage( mEventThreadServices.GetUpdateManager(), animation );
171 void Animation::DestroySceneObject()
173 if ( mAnimation != NULL )
175 // Remove animation using a message to the update manager
176 RemoveAnimationMessage( mEventThreadServices.GetUpdateManager(), *mAnimation );
181 void Animation::SetDuration(float seconds)
185 DALI_LOG_WARNING("duration should be greater than 0.0f.\n");
189 // Cache for public getters
190 mDurationSeconds = seconds;
192 // mAnimation is being used in a separate thread; queue a message to set the value
193 SetDurationMessage( mEventThreadServices, *mAnimation, seconds );
196 float Animation::GetDuration() const
198 // This is not animatable; the cached value is up-to-date.
199 return mDurationSeconds;
202 void Animation::SetLooping(bool looping)
204 // Cache for public getters
205 mIsLooping = looping;
207 // mAnimation is being used in a separate thread; queue a message to set the value
208 SetLoopingMessage( mEventThreadServices, *mAnimation, looping );
211 bool Animation::IsLooping() const
213 // This is not animatable; the cached value is up-to-date.
217 void Animation::SetEndAction(EndAction action)
219 // Cache for public getters
222 // mAnimation is being used in a separate thread; queue a message to set the value
223 SetEndActionMessage( mEventThreadServices, *mAnimation, action );
226 Dali::Animation::EndAction Animation::GetEndAction() const
228 // This is not animatable; the cached value is up-to-date.
232 void Animation::SetDisconnectAction(EndAction action)
234 // Cache for public getters
235 mDisconnectAction = action;
237 // mAnimation is being used in a separate thread; queue a message to set the value
238 SetDisconnectActionMessage( mEventThreadServices, *mAnimation, action );
241 Dali::Animation::EndAction Animation::GetDisconnectAction() const
243 // This is not animatable; the cached value is up-to-date.
244 return mDisconnectAction;
247 void Animation::Play()
249 // Update the current playlist
250 mPlaylist.OnPlay( *this );
252 // mAnimation is being used in a separate thread; queue a Play message
253 PlayAnimationMessage( mEventThreadServices, *mAnimation );
256 void Animation::PlayFrom( float progress )
258 if( progress >= mPlayRange.x && progress <= mPlayRange.y )
260 // Update the current playlist
261 mPlaylist.OnPlay( *this );
263 // mAnimation is being used in a separate thread; queue a Play message
264 PlayAnimationFromMessage( mEventThreadServices, *mAnimation, progress );
268 void Animation::Pause()
270 // mAnimation is being used in a separate thread; queue a Pause message
271 PauseAnimationMessage( mEventThreadServices, *mAnimation );
274 void Animation::Stop()
276 // mAnimation is being used in a separate thread; queue a Stop message
277 StopAnimationMessage( mEventThreadServices.GetUpdateManager(), *mAnimation );
280 void Animation::Clear()
282 DALI_ASSERT_DEBUG(mAnimation);
284 // Remove all the connectors
287 // Replace the old scene-object with a new one
288 DestroySceneObject();
291 // Reset the notification count, since the new scene-object has never been played
292 mNotificationCount = 0;
294 // Update the current playlist
295 mPlaylist.OnClear( *this );
298 void Animation::AnimateBy(Property& target, Property::Value& relativeValue)
300 AnimateBy(target, relativeValue, mDefaultAlpha, TimePeriod(mDurationSeconds));
303 void Animation::AnimateBy(Property& target, Property::Value& relativeValue, AlphaFunction alpha)
305 AnimateBy(target, relativeValue, alpha, TimePeriod(mDurationSeconds));
308 void Animation::AnimateBy(Property& target, Property::Value& relativeValue, TimePeriod period)
310 AnimateBy(target, relativeValue, mDefaultAlpha, period);
313 void Animation::AnimateBy(Property& target, Property::Value& relativeValue, AlphaFunction alpha, TimePeriod period)
315 Object& object = dynamic_cast<Object&>( GetImplementation(target.object) );
317 ExtendDuration( period );
319 switch ( relativeValue.GetType() )
321 case Property::BOOLEAN:
323 AddAnimatorConnector( AnimatorConnector<bool>::New( object,
324 target.propertyIndex,
325 target.componentIndex,
326 new AnimateByBoolean(relativeValue.Get<bool>()),
332 case Property::INTEGER:
334 AddAnimatorConnector( AnimatorConnector<int>::New( object,
335 target.propertyIndex,
336 target.componentIndex,
337 new AnimateByInteger(relativeValue.Get<int>()),
343 case Property::FLOAT:
345 AddAnimatorConnector( AnimatorConnector<float>::New( object,
346 target.propertyIndex,
347 target.componentIndex,
348 new AnimateByFloat(relativeValue.Get<float>()),
354 case Property::VECTOR2:
356 AddAnimatorConnector( AnimatorConnector<Vector2>::New( object,
357 target.propertyIndex,
358 target.componentIndex,
359 new AnimateByVector2(relativeValue.Get<Vector2>()),
365 case Property::VECTOR3:
367 AddAnimatorConnector( AnimatorConnector<Vector3>::New( object,
368 target.propertyIndex,
369 target.componentIndex,
370 new AnimateByVector3(relativeValue.Get<Vector3>()),
376 case Property::VECTOR4:
378 AddAnimatorConnector( AnimatorConnector<Vector4>::New( object,
379 target.propertyIndex,
380 target.componentIndex,
381 new AnimateByVector4(relativeValue.Get<Vector4>()),
387 case Property::ROTATION:
389 AngleAxis angleAxis = relativeValue.Get<AngleAxis>();
391 AddAnimatorConnector( AnimatorConnector<Quaternion>::New( object,
392 target.propertyIndex,
393 target.componentIndex,
394 new RotateByAngleAxis(angleAxis.angle, angleAxis.axis),
401 DALI_ASSERT_ALWAYS( false && "Property type enumeration out of bounds" ); // should never come here
406 void Animation::AnimateTo(Property& target, Property::Value& destinationValue)
408 AnimateTo(target, destinationValue, mDefaultAlpha, TimePeriod(mDurationSeconds));
411 void Animation::AnimateTo(Property& target, Property::Value& destinationValue, AlphaFunction alpha)
413 AnimateTo(target, destinationValue, alpha, TimePeriod(mDurationSeconds));
416 void Animation::AnimateTo(Property& target, Property::Value& destinationValue, TimePeriod period)
418 AnimateTo(target, destinationValue, mDefaultAlpha, period);
421 void Animation::AnimateTo(Property& target, Property::Value& destinationValue, AlphaFunction alpha, TimePeriod period)
423 Object& object = dynamic_cast<Object&>( GetImplementation(target.object) );
425 AnimateTo( object, target.propertyIndex, target.componentIndex, destinationValue, alpha, period );
428 void Animation::AnimateTo(Object& targetObject, Property::Index targetPropertyIndex, int componentIndex, Property::Value& destinationValue, AlphaFunction alpha, TimePeriod period)
430 Property::Type type = targetObject.GetPropertyType(targetPropertyIndex);
431 if(componentIndex != Property::INVALID_COMPONENT_INDEX)
433 if( type == Property::VECTOR2
434 || type == Property::VECTOR3
435 || type == Property::VECTOR4 )
437 type = Property::FLOAT;
440 DALI_ASSERT_ALWAYS( type == destinationValue.GetType() && "DestinationValue does not match Target Property type" );
442 ExtendDuration( period );
444 switch (destinationValue.GetType())
446 case Property::BOOLEAN:
448 AddAnimatorConnector( AnimatorConnector<bool>::New( targetObject,
451 new AnimateToBoolean( destinationValue.Get<bool>() ),
457 case Property::INTEGER:
459 AddAnimatorConnector( AnimatorConnector<int>::New( targetObject,
462 new AnimateToInteger( destinationValue.Get<int>() ),
468 case Property::FLOAT:
470 if ( ( Dali::Actor::Property::SIZE_WIDTH == targetPropertyIndex )||
471 ( Dali::Actor::Property::SIZE_HEIGHT == targetPropertyIndex ) )
473 // Test whether this is actually an Actor
474 Actor* maybeActor = dynamic_cast<Actor*>( &targetObject );
477 // Notify the actor that its size is being animated
478 maybeActor->NotifySizeAnimation( *this, destinationValue.Get<float>(), targetPropertyIndex );
481 AddAnimatorConnector( AnimatorConnector<float>::New( targetObject,
484 new AnimateToFloat( destinationValue.Get<float>() ),
490 case Property::VECTOR2:
492 AddAnimatorConnector( AnimatorConnector<Vector2>::New( targetObject,
495 new AnimateToVector2( destinationValue.Get<Vector2>() ),
501 case Property::VECTOR3:
503 if ( Dali::Actor::Property::SIZE == targetPropertyIndex )
505 // Test whether this is actually an Actor
506 Actor* maybeActor = dynamic_cast<Actor*>( &targetObject );
509 // Notify the actor that its size is being animated
510 maybeActor->NotifySizeAnimation( *this, destinationValue.Get<Vector3>() );
514 AddAnimatorConnector( AnimatorConnector<Vector3>::New( targetObject,
517 new AnimateToVector3( destinationValue.Get<Vector3>() ),
523 case Property::VECTOR4:
525 AddAnimatorConnector( AnimatorConnector<Vector4>::New( targetObject,
528 new AnimateToVector4( destinationValue.Get<Vector4>() ),
534 case Property::ROTATION:
536 AddAnimatorConnector( AnimatorConnector<Quaternion>::New( targetObject,
539 new RotateToQuaternion( destinationValue.Get<Quaternion>() ),
546 DALI_ASSERT_ALWAYS( false && "Property type enumeration out of bounds" ); // should never come here
551 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames)
553 AnimateBetween(target, keyFrames, mDefaultAlpha, TimePeriod(mDurationSeconds), DEFAULT_INTERPOLATION );
556 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, Interpolation interpolation )
558 AnimateBetween(target, keyFrames, mDefaultAlpha, TimePeriod(mDurationSeconds), interpolation );
561 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, TimePeriod period)
563 AnimateBetween(target, keyFrames, mDefaultAlpha, period, DEFAULT_INTERPOLATION);
566 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, TimePeriod period, Interpolation interpolation)
568 AnimateBetween(target, keyFrames, mDefaultAlpha, period, interpolation);
571 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, AlphaFunction alpha)
573 AnimateBetween(target, keyFrames, alpha, TimePeriod(mDurationSeconds), DEFAULT_INTERPOLATION);
576 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, AlphaFunction alpha, Interpolation interpolation)
578 AnimateBetween(target, keyFrames, alpha, TimePeriod(mDurationSeconds), interpolation);
581 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, AlphaFunction alpha, TimePeriod period)
583 AnimateBetween(target, keyFrames, alpha, period, DEFAULT_INTERPOLATION);
586 void Animation::AnimateBetween(Property target, const KeyFrames& keyFrames, AlphaFunction alpha, TimePeriod period, Interpolation interpolation)
588 Object& object = dynamic_cast<Object&>( GetImplementation(target.object) );
590 ExtendDuration( period );
592 switch(keyFrames.GetType())
594 case Dali::Property::BOOLEAN:
596 const KeyFrameBoolean* kf;
597 GetSpecialization(keyFrames, kf);
598 KeyFrameBooleanPtr kfCopy = KeyFrameBoolean::Clone(*kf);
599 AddAnimatorConnector( AnimatorConnector<bool>::New( object,
600 target.propertyIndex,
601 target.componentIndex,
602 new KeyFrameBooleanFunctor(kfCopy),
608 case Dali::Property::INTEGER:
610 const KeyFrameInteger* kf;
611 GetSpecialization(keyFrames, kf);
612 KeyFrameIntegerPtr kfCopy = KeyFrameInteger::Clone(*kf);
613 AddAnimatorConnector( AnimatorConnector<int>::New( object,
614 target.propertyIndex,
615 target.componentIndex,
616 new KeyFrameIntegerFunctor(kfCopy,interpolation),
622 case Dali::Property::FLOAT:
624 const KeyFrameNumber* kf;
625 GetSpecialization(keyFrames, kf);
626 KeyFrameNumberPtr kfCopy = KeyFrameNumber::Clone(*kf);
627 AddAnimatorConnector( AnimatorConnector<float>::New( object,
628 target.propertyIndex,
629 target.componentIndex,
630 new KeyFrameNumberFunctor(kfCopy,interpolation),
636 case Dali::Property::VECTOR2:
638 const KeyFrameVector2* kf;
639 GetSpecialization(keyFrames, kf);
640 KeyFrameVector2Ptr kfCopy = KeyFrameVector2::Clone(*kf);
641 AddAnimatorConnector( AnimatorConnector<Vector2>::New( object,
642 target.propertyIndex,
643 target.componentIndex,
644 new KeyFrameVector2Functor(kfCopy,interpolation),
650 case Dali::Property::VECTOR3:
652 const KeyFrameVector3* kf;
653 GetSpecialization(keyFrames, kf);
654 KeyFrameVector3Ptr kfCopy = KeyFrameVector3::Clone(*kf);
655 AddAnimatorConnector( AnimatorConnector<Vector3>::New( object,
656 target.propertyIndex,
657 target.componentIndex,
658 new KeyFrameVector3Functor(kfCopy,interpolation),
664 case Dali::Property::VECTOR4:
666 const KeyFrameVector4* kf;
667 GetSpecialization(keyFrames, kf);
668 KeyFrameVector4Ptr kfCopy = KeyFrameVector4::Clone(*kf);
669 AddAnimatorConnector( AnimatorConnector<Vector4>::New( object,
670 target.propertyIndex,
671 target.componentIndex,
672 new KeyFrameVector4Functor(kfCopy,interpolation),
678 case Dali::Property::ROTATION:
680 const KeyFrameQuaternion* kf;
681 GetSpecialization(keyFrames, kf);
682 KeyFrameQuaternionPtr kfCopy = KeyFrameQuaternion::Clone(*kf);
683 AddAnimatorConnector( AnimatorConnector<Quaternion>::New( object,
684 target.propertyIndex,
685 target.componentIndex,
686 new KeyFrameQuaternionFunctor(kfCopy),
692 default: // not all property types are animateable
697 bool Animation::HasFinished()
699 bool hasFinished(false);
700 const int playCount(mAnimation->GetPlayCount());
702 // If the play count has been incremented, then another notification is required
703 if (playCount > mNotificationCount)
705 // Note that only one signal is emitted, if the animation has been played repeatedly
706 mNotificationCount = playCount;
714 Dali::Animation::AnimationSignalType& Animation::FinishedSignal()
716 return mFinishedSignal;
719 void Animation::EmitSignalFinish()
721 if ( !mFinishedSignal.Empty() )
723 Dali::Animation handle( this );
724 mFinishedSignal.Emit( handle );
727 // This callback is used internally, to avoid the overhead of using a signal.
728 if ( mFinishedCallback )
730 mFinishedCallback( mFinishedCallbackObject );
734 bool Animation::DoConnectSignal( BaseObject* object, ConnectionTrackerInterface* tracker, const std::string& signalName, FunctorDelegate* functor )
736 bool connected( true );
737 Animation* animation = dynamic_cast<Animation*>(object);
739 if( 0 == signalName.compare( SIGNAL_FINISHED ) )
741 animation->FinishedSignal().Connect( tracker, functor );
745 // signalName does not match any signal
752 void Animation::SetFinishedCallback( FinishedCallback callback, Object* object )
754 mFinishedCallback = callback;
755 mFinishedCallbackObject = object;
758 void Animation::AddAnimatorConnector( AnimatorConnectorBase* connector )
760 DALI_ASSERT_DEBUG( NULL != connector );
762 connector->SetParent(*this);
764 mConnectors.PushBack( connector );
767 void Animation::Animate( Actor& actor, const Path& path, const Vector3& forward )
769 Animate( actor, path, forward, mDefaultAlpha, TimePeriod(mDurationSeconds) );
772 void Animation::Animate( Actor& actor, const Path& path, const Vector3& forward, AlphaFunction alpha )
774 Animate( actor, path, forward, alpha, TimePeriod(mDurationSeconds) );
777 void Animation::Animate( Actor& actor, const Path& path, const Vector3& forward, TimePeriod period )
779 Animate( actor, path, forward, mDefaultAlpha, period );
782 void Animation::Animate( Actor& actor, const Path& path, const Vector3& forward, AlphaFunction alpha, TimePeriod period)
784 ExtendDuration( period );
786 PathPtr pathCopy = Path::Clone(path);
789 AddAnimatorConnector( AnimatorConnector<Vector3>::New( actor,
790 Dali::Actor::Property::POSITION,
791 Property::INVALID_COMPONENT_INDEX,
792 new PathPositionFunctor( pathCopy ),
796 //If forward is zero, PathRotationFunctor will always return the unit quaternion
797 if( forward != Vector3::ZERO )
800 AddAnimatorConnector( AnimatorConnector<Quaternion>::New( actor,
801 Dali::Actor::Property::ORIENTATION,
802 Property::INVALID_COMPONENT_INDEX,
803 new PathRotationFunctor( pathCopy, forward ),
809 void Animation::Show(Actor& actor, float delaySeconds)
811 ExtendDuration( TimePeriod(delaySeconds, 0) );
813 AddAnimatorConnector( AnimatorConnector<bool>::New( actor,
814 Dali::Actor::Property::VISIBLE,
815 Property::INVALID_COMPONENT_INDEX,
816 new AnimateToBoolean(SHOW_VALUE),
818 TimePeriod(delaySeconds, 0.0f/*immediate*/) ) );
821 void Animation::Hide(Actor& actor, float delaySeconds)
823 ExtendDuration( TimePeriod(delaySeconds, 0) );
825 AddAnimatorConnector( AnimatorConnector<bool>::New( actor,
826 Dali::Actor::Property::VISIBLE,
827 Property::INVALID_COMPONENT_INDEX,
828 new AnimateToBoolean(HIDE_VALUE),
830 TimePeriod(delaySeconds, 0.0f/*immediate*/) ) );
833 bool Animation::DoAction( BaseObject* object, const std::string& actionName, const Property::Map& attributes )
836 Animation* animation = dynamic_cast<Animation*>( object );
840 if( 0 == actionName.compare( ACTION_PLAY ) )
842 if( Property::Value* value = attributes.Find("duration", Property::FLOAT) )
844 animation->SetDuration( value->Get<float>() );
850 else if( 0 == actionName.compare( ACTION_STOP ) )
855 else if( 0 == actionName.compare( ACTION_PAUSE ) )
865 void Animation::SetCurrentProgress(float progress)
867 if( mAnimation && progress >= mPlayRange.x && progress <= mPlayRange.y )
869 // mAnimation is being used in a separate thread; queue a message to set the current progress
870 SetCurrentProgressMessage( mEventThreadServices, *mAnimation, progress );
874 float Animation::GetCurrentProgress()
878 return mAnimation->GetCurrentProgress();
884 void Animation::ExtendDuration( const TimePeriod& timePeriod )
886 float duration = timePeriod.delaySeconds + timePeriod.durationSeconds;
888 if( duration > mDurationSeconds )
890 SetDuration( duration );
894 void Animation::SetSpeedFactor( float factor )
898 mSpeedFactor = factor;
899 SetSpeedFactorMessage( mEventThreadServices, *mAnimation, factor );
903 float Animation::GetSpeedFactor() const
908 void Animation::SetPlayRange( const Vector2& range)
910 //Make sure the range specified is between 0.0 and 1.0
911 if( range.x >= 0.0f && range.x <= 1.0f && range.y >= 0.0f && range.y <= 1.0f )
913 Vector2 orderedRange( range );
914 //If the range is not in order swap values
915 if( range.x > range.y )
917 orderedRange = Vector2(range.y, range.x);
920 // Cache for public getters
921 mPlayRange = orderedRange;
923 // mAnimation is being used in a separate thread; queue a message to set play range
924 SetPlayRangeMessage( mEventThreadServices, *mAnimation, orderedRange );
928 Vector2 Animation::GetPlayRange() const
934 } // namespace Internal