SetRect animation with new animation system

This commit is contained in:
Daria Volvenkova 2016-04-18 22:55:18 +03:00 committed by r.kuznetsov
parent 9beccf2fd2
commit 05c5f5d766
4 changed files with 336 additions and 151 deletions

View file

@ -1,6 +1,11 @@
#include "animation_system.h"
#include "animation/interpolations.hpp"
#include "base/logging.hpp"
#include "boost/variant/variant.hpp"
#include "boost/variant/get.hpp"
namespace df
{
@ -19,6 +24,35 @@ double CalcAnimSpeedDuration(double pxDiff, double pxSpeed)
}
bool Animation::CouldBeMixedWith(TObject object, TObjectProperties const & properties)
{
if (!m_couldBeMixed)
return false;
ASSERT(HasObject(object), ());
TObjectProperties const & selfProperties = GetProperties(object);
TObjectProperties intersection;
set_intersection(selfProperties.begin(), selfProperties.end(),
properties.begin(), properties.end(),
inserter(intersection, intersection.end()));
return intersection.empty();
}
bool Animation::CouldBeMixedWith(Animation const & animation)
{
if (!m_couldBeMixed || animation.m_couldBeMixed)
return false;
for (auto const & object : animation.GetObjects())
{
if (!HasObject(object))
continue;
if (!CouldBeMixedWith(object, animation.GetProperties(object)))
return false;
}
return true;
}
Interpolator::Interpolator(double duration, double delay)
: m_elapsedTime(0.0)
, m_duration(duration)
@ -102,7 +136,7 @@ PositionInterpolator::PositionInterpolator(double delay, m2::PointD const & star
void PositionInterpolator::Advance(double elapsedSeconds)
{
TBase::Advance(elapsedSeconds);
InterpolatePoint(m_startPosition, m_endPosition, GetT());
m_position = InterpolatePoint(m_startPosition, m_endPosition, GetT());
}
// static
@ -164,35 +198,57 @@ void ScaleInterpolator::Advance(double elapsedSeconds)
FollowAnimation::FollowAnimation(m2::PointD const & startPos, m2::PointD const & endPos,
double startAngle, double endAngle,
double startScale, double endScale, ScreenBase const & convertor)
: Animation(false, false)
: Animation(true /* couldBeInterrupted */, false /* couldBeMixed */)
{
m_objects.insert(Animation::MapPlane);
SetMove(startPos, endPos, convertor);
SetRotate(startAngle, endAngle);
SetScale(startScale, endScale);
}
FollowAnimation::FollowAnimation()
: Animation(false, false)
: Animation(true /* couldBeInterrupted */, false /* couldBeMixed */)
{
m_objects.insert(Animation::MapPlane);
}
void FollowAnimation::SetMove(m2::PointD const & startPos, m2::PointD const & endPos,
ScreenBase const & convertor)
{
if (startPos != endPos)
{
m_positionInterpolator = make_unique_dp<PositionInterpolator>(startPos, endPos, convertor);
m_properties.insert(Animation::Position);
}
}
void FollowAnimation::SetRotate(double startAngle, double endAngle)
{
if (startAngle != endAngle)
{
m_angleInterpolator = make_unique_dp<AngleInterpolator>(startAngle, endAngle);
m_properties.insert(Animation::Angle);
}
}
void FollowAnimation::SetScale(double startScale, double endScale)
{
if (startScale != endScale)
{
m_scaleInterpolator = make_unique_dp<ScaleInterpolator>(startScale, endScale);
m_properties.insert(Animation::Scale);
}
}
Animation::TObjectProperties const & FollowAnimation::GetProperties(TObject object) const
{
ASSERT(object == Animation::MapPlane, ());
return m_properties;
}
bool FollowAnimation::HasProperty(TObject object, TProperty property) const
{
return HasObject(object) && m_properties.find(property) != m_properties.end();
}
void FollowAnimation::Advance(double elapsedSeconds)
@ -234,58 +290,58 @@ bool FollowAnimation::IsFinished() const
&& (m_positionInterpolator == nullptr || m_positionInterpolator->IsFinished()));
}
double FollowAnimation::GetScale(uint32_t object) const
Animation::TPropValue FollowAnimation::GetProperty(TObject object, TProperty property) const
{
ASSERT(object & GetObjects(), ());
ASSERT(m_scaleInterpolator != nullptr, ());
ASSERT(object == Animation::MapPlane, ());
if (m_scaleInterpolator != nullptr)
return m_scaleInterpolator->GetScale();
switch (property)
{
case Animation::Position:
ASSERT(m_positionInterpolator != nullptr, ());
if (m_positionInterpolator != nullptr)
return m_positionInterpolator->GetPosition();
break;
case Animation::Scale:
ASSERT(m_scaleInterpolator != nullptr, ());
if (m_scaleInterpolator != nullptr)
return m_scaleInterpolator->GetScale();
break;
case Animation::Angle:
ASSERT(m_angleInterpolator != nullptr, ());
if (m_angleInterpolator != nullptr)
return m_angleInterpolator->GetAngle();
break;
default:
ASSERT(!"Wrong property", ());
}
return 0.0;
}
double FollowAnimation::GetAngle(uint32_t object) const
Animation::TObjectProperties const & ParallelAnimation::GetProperties(TObject object) const
{
ASSERT(object & GetObjects(), ());
ASSERT(m_angleInterpolator != nullptr, ());
if (m_angleInterpolator != nullptr)
return m_angleInterpolator->GetAngle();
return 0.0;
ASSERT(HasObject(object), ());
return m_properties.find(object)->second;
}
m2::PointD FollowAnimation::GetPosition(uint32_t object) const
bool ParallelAnimation::HasProperty(TObject object, TProperty property) const
{
ASSERT(object & GetObjects(), ());
ASSERT(m_positionInterpolator != nullptr, ());
if (m_positionInterpolator != nullptr)
return m_positionInterpolator->GetPosition();
return m2::PointD();
if (!HasObject(object))
return false;
TObjectProperties properties = GetProperties(object);
return properties.find(property) != properties.end();
}
uint32_t ParallelAnimation::GetMask(uint32_t object) const
void ParallelAnimation::AddAnimation(drape_ptr<Animation> && animation)
{
int mask = 0;
for (auto const & anim : m_animations)
mask |= anim->GetMask(object);
return mask;
}
uint32_t ParallelAnimation::GetObjects() const
{
int objects = 0;
for (auto const & anim : m_animations)
objects |= anim->GetObjects();
return objects;
}
void ParallelAnimation::AddAnimation(ref_ptr<Animation> animation)
{
m_animations.push_back(animation);
TAnimObjects const & objects = animation->GetObjects();
m_objects.insert(objects.begin(), objects.end());
for (auto const & object : objects)
{
TObjectProperties const & properties = animation->GetProperties(object);
m_properties[object].insert(properties.begin(), properties.end());
}
m_animations.push_back(move(animation));
}
void ParallelAnimation::OnStart()
@ -315,25 +371,33 @@ void ParallelAnimation::Advance(double elapsedSeconds)
}
}
uint32_t SequenceAnimation::GetMask(uint32_t object) const
Animation::TAnimObjects const & SequenceAnimation::GetObjects() const
{
int mask = 0;
if (!m_animations.empty())
mask = m_animations.front()->GetMask(object);
return mask;
ASSERT(!m_animations.empty(), ());
return m_animations.front()->GetObjects();
}
uint32_t SequenceAnimation::GetObjects() const
bool SequenceAnimation::HasObject(TObject object) const
{
int objects = 0;
if (!m_animations.empty())
objects = m_animations.front()->GetObjects();
return objects;
ASSERT(!m_animations.empty(), ());
return m_animations.front()->HasObject(object);
}
void SequenceAnimation::AddAnimation(ref_ptr<Animation> animation)
Animation::TObjectProperties const & SequenceAnimation::GetProperties(TObject object) const
{
m_animations.push_back(animation);
ASSERT(!m_animations.empty(), ());
return m_animations.front()->GetProperties(object);
}
bool SequenceAnimation::HasProperty(TObject object, TProperty property) const
{
ASSERT(!m_animations.empty(), ());
return m_animations.front()->HasProperty(object, property);
}
void SequenceAnimation::AddAnimation(drape_ptr<Animation> && animation)
{
m_animations.push_back(move(animation));
}
void SequenceAnimation::OnStart()
@ -360,35 +424,38 @@ void SequenceAnimation::Advance(double elapsedSeconds)
}
}
AnimationSystem::AnimationSystem()
{
}
m2::AnyRectD AnimationSystem::GetRect(ScreenBase const & currentScreen)
{
double scale = currentScreen.GetScale();
double angle = currentScreen.GetAngle();
m2::PointD pos = currentScreen.GlobalRect().GlobalZero();
for (auto const & anim : m_animations)
{
if (anim->GetObjects() & kPlaneAnimObjBit)
{
int mask = anim->GetMask(kPlaneAnimObjBit);
if (mask & kScaleAnimationBit)
scale = anim->GetScale(kPlaneAnimObjBit);
if (mask & kMoveAnimationBit)
pos = anim->GetPosition(kPlaneAnimObjBit);
if (mask & kRotateAnimationBit)
angle = anim->GetAngle(kPlaneAnimObjBit);
}
}
const Animation::TObject obj = Animation::MapPlane;
double scale = boost::get<double>(
GetProperty(obj, Animation::Scale, currentScreen.GetScale()));
double angle = boost::get<double>(
GetProperty(obj, Animation::Angle, currentScreen.GetAngle()));
m2::PointD pos = boost::get<m2::PointD>(
GetProperty(obj, Animation::Position, currentScreen.GlobalRect().GlobalZero()));
m2::RectD rect = currentScreen.PixelRect();
rect.Offset(-rect.Center());
rect.Scale(scale);
return m2::AnyRectD(pos, angle, rect);
}
bool AnimationSystem::AnimationExists(Animation::Object object)
bool AnimationSystem::AnimationExists(Animation::TObject object) const
{
for (auto const & anim : m_animations)
if (m_animationChain.empty())
return false;
for (auto const & anim : m_animationChain.front())
{
if (anim->GetObjects() & (1 << object))
if (anim->HasObject(object))
return true;
}
for (auto it = m_propertyCache.begin(); it != m_propertyCache.end(); ++it)
{
if (it->first.first == object)
return true;
}
return false;
@ -400,25 +467,101 @@ AnimationSystem & AnimationSystem::Instance()
return animSystem;
}
void AnimationSystem::AddAnimation(drape_ptr<Animation> && animation)
void AnimationSystem::AddAnimation(drape_ptr<Animation> && animation, bool force)
{
for (auto & lst : m_animationChain)
{
bool couldBeMixed = true;
for (auto it = lst.begin(); it != lst.end();)
{
auto & anim = *it;
if (!anim->CouldBeMixedWith(*animation))
{
if (!force || !anim->CouldBeInterrupted())
{
couldBeMixed = false;
break;
}
// TODO: do not interrupt anything until it's not clear that we can mix
anim->Interrupt();
SaveAnimationResult(*anim);
it = lst.erase(it);
}
else
{
++it;
}
}
if (couldBeMixed)
{
animation->OnStart();
lst.emplace_back(move(animation));
return;
}
}
PushAnimation(move(animation));
}
void AnimationSystem::PushAnimation(drape_ptr<Animation> && animation)
{
animation->OnStart();
m_animations.insert(move(animation));
TAnimationList list;
list.emplace_back(move(animation));
m_animationChain.emplace_back(move(list));
}
void AnimationSystem::Advance(double elapsedSeconds)
{
auto iter = m_animations.begin();
while (iter != m_animations.end())
if (m_animationChain.empty())
return;
TAnimationList & frontList = m_animationChain.front();
for (auto it = frontList.begin(); it != frontList.end();)
{
(*iter)->Advance(elapsedSeconds);
if ((*iter)->IsFinished())
auto & anim = *it;
anim->Advance(elapsedSeconds);
if (anim->IsFinished())
{
(*iter)->OnFinish();
iter = m_animations.erase(iter);
anim->OnFinish();
SaveAnimationResult(*anim);
it = frontList.erase(it);
}
else
++iter;
{
++it;
}
}
}
Animation::TPropValue AnimationSystem::GetProperty(Animation::TObject object, Animation::TProperty property, Animation::TPropValue current) const
{
if (!m_animationChain.empty())
{
for (auto const & anim : m_animationChain.front())
{
if (anim->HasProperty(object, property))
return anim->GetProperty(object, property);
}
}
auto it = m_propertyCache.find(make_pair(object, property));
if (it != m_propertyCache.end())
{
Animation::TPropValue value(it->second);
m_propertyCache.erase(it);
return value;
}
return current;
}
void AnimationSystem::SaveAnimationResult(Animation const & animation)
{
for (auto const & object : animation.GetObjects())
{
for (auto const & property : animation.GetProperties(object))
{
m_propertyCache[make_pair(object, property)] = animation.GetProperty(object, property);
}
}
}

View file

@ -6,21 +6,15 @@
#include "geometry/screenbase.hpp"
#include "std/set.hpp"
#include "std/deque.hpp"
#include "std/noncopyable.hpp"
#include "boost/variant/variant_fwd.hpp"
namespace df
{
uint32_t constexpr kMoveAnimationBit = 1;
uint32_t constexpr kScaleAnimationBit = 1 << 1;
uint32_t constexpr kRotateAnimationBit = 1 << 2;
uint32_t constexpr kPerspectiveAnimationBit = 1 << 3;
uint32_t constexpr kArrowAnimObjBit = 1;
uint32_t constexpr kPlaneAnimObjBit = 1 << 1;
uint32_t constexpr kSelectionAnimBit = 1 << 2;
class Animation
{
public:
@ -40,6 +34,19 @@ public:
Selection
};
enum ObjectProperty
{
Position,
Scale,
Angle
};
using TObject = uint32_t;
using TProperty = uint32_t;
using TPropValue = boost::variant<double, m2::PointD>;
using TAnimObjects = set<TObject>;
using TObjectProperties = set<TProperty>;
Animation(bool couldBeInterrupted, bool couldBeMixed)
: m_couldBeInterrupted(couldBeInterrupted)
, m_couldBeMixed(couldBeMixed)
@ -47,11 +54,14 @@ public:
virtual void OnStart() {}
virtual void OnFinish() {}
virtual void Interrupt() {}
virtual Type GetType() const = 0;
virtual uint32_t GetMask(uint32_t object) const = 0;
virtual uint32_t GetObjects() const = 0;
virtual TAnimObjects const & GetObjects() const = 0;
virtual bool HasObject(TObject object) const = 0;
virtual TObjectProperties const & GetProperties(TObject object) const = 0;
virtual bool HasProperty(TObject object, TProperty property) const = 0;
virtual void SetMaxDuration(double maxDuration) = 0;
virtual double GetDuration() const = 0;
@ -59,16 +69,12 @@ public:
virtual void Advance(double elapsedSeconds) = 0;
virtual double GetScale(uint32_t object) const = 0;
virtual double GetAngle(uint32_t object) const = 0;
virtual m2::PointD GetPosition(uint32_t object) const = 0;
virtual TPropValue GetProperty(TObject object, TProperty property) const = 0;
bool CouldBeInterrupted() const { return m_couldBeInterrupted; }
bool CouldBeMixed() const { return m_couldBeMixed; }
bool CouldBeMixedWith(uint32_t object, int mask)
{
return m_couldBeMixed && !(mask & GetMask(object));
}
bool CouldBeMixedWith(TObject object, TObjectProperties const & properties);
bool CouldBeMixedWith(Animation const & animation);
protected:
bool m_couldBeInterrupted;
@ -163,27 +169,31 @@ public:
{
m_positionInterpolator.reset(new PositionInterpolator(startPos, endPos, convertor));
m_angleInterpolator.reset(new AngleInterpolator(startAngle, endAngle));
m_objects.insert(Animation::MyPositionArrow);
m_properties.insert(Animation::Position);
m_properties.insert(Animation::Angle);
}
Animation::Type GetType() const override { return Animation::Arrow; }
uint32_t GetMask(uint32_t object) const override
TAnimObjects const & GetObjects() const override
{
return (object & kArrowAnimObjBit) &&
((m_angleInterpolator != nullptr ? kRotateAnimationBit : 0) |
(m_positionInterpolator != nullptr ? kMoveAnimationBit : 0));
return m_objects;
}
uint32_t GetObjects() const override
bool HasObject(TObject object) const override
{
return kArrowAnimObjBit;
return object == Animation::MyPositionArrow;
}
TObjectProperties const & GetProperties(TObject object) const override;
bool HasProperty(TObject object, TProperty property) const override;
void Advance(double elapsedSeconds) override;
private:
drape_ptr<PositionInterpolator> m_positionInterpolator;
drape_ptr<AngleInterpolator> m_angleInterpolator;
TAnimObjects m_objects;
TObjectProperties m_properties;
};
class PerspectiveSwitchAnimation : public Animation
@ -194,21 +204,23 @@ class PerspectiveSwitchAnimation : public Animation
Animation::Type GetType() const override { return Animation::Perspective; }
uint32_t GetMask(uint32_t object) const override
TAnimObjects const & GetObjects() const override
{
return (object & kPlaneAnimObjBit) &&
(m_angleInterpolator != nullptr ? kPerspectiveAnimationBit : 0);
return m_objects;
}
uint32_t GetObjects() const override
bool HasObject(TObject object) const override
{
return kPlaneAnimObjBit;
return m_objects.find(object) != m_objects.end();
}
TObjectProperties const & GetProperties(TObject object) const override;
bool HasProperty(TObject object, TProperty property) const override;
void Advance(double elapsedSeconds) override;
private:
drape_ptr<AngleInterpolator> m_angleInterpolator;
TAnimObjects m_objects;
TObjectProperties m_properties;
};
class FollowAnimation : public Animation
@ -225,18 +237,16 @@ public:
Animation::Type GetType() const override { return Animation::ModelView; }
uint32_t GetMask(uint32_t object) const override
TAnimObjects const & GetObjects() const override
{
return (object & kPlaneAnimObjBit) ?
((m_angleInterpolator != nullptr ? kPerspectiveAnimationBit : 0) |
(m_positionInterpolator != nullptr ? kMoveAnimationBit : 0) |
(m_scaleInterpolator != nullptr ? kScaleAnimationBit : 0)) : 0;
return m_objects;
}
uint32_t GetObjects() const override
bool HasObject(TObject object) const override
{
return kPlaneAnimObjBit;
return object == Animation::MapPlane;
}
TObjectProperties const & GetProperties(TObject object) const override;
bool HasProperty(TObject object, TProperty property) const override;
void Advance(double elapsedSeconds) override;
@ -244,26 +254,26 @@ public:
double GetDuration() const override;
bool IsFinished() const override;
double GetScale(uint32_t object) const override;
double GetAngle(uint32_t object) const override;
m2::PointD GetPosition(uint32_t object) const override;
TPropValue GetProperty(TObject object, TProperty property) const override;
private:
drape_ptr<AngleInterpolator> m_angleInterpolator;
drape_ptr<PositionInterpolator> m_positionInterpolator;
drape_ptr<ScaleInterpolator> m_scaleInterpolator;
TObjectProperties m_properties;
TAnimObjects m_objects;
};
using TAnimations = vector<ref_ptr<Animation>>;
class SequenceAnimation : public Animation
{
public:
Animation::Type GetType() const override { return Animation::Sequence; }
uint32_t GetMask(uint32_t object) const override;
uint32_t GetObjects() const override;
TAnimObjects const & GetObjects() const override;
bool HasObject(TObject object) const override;
TObjectProperties const & GetProperties(TObject object) const override;
bool HasProperty(TObject object, TProperty property) const override;
void AddAnimation(ref_ptr<Animation> animation);
void AddAnimation(drape_ptr<Animation> && animation);
void OnStart() override;
void OnFinish() override;
@ -271,17 +281,25 @@ public:
void Advance(double elapsedSeconds) override;
private:
deque<ref_ptr<Animation>> m_animations;
deque<drape_ptr<Animation>> m_animations;
};
class ParallelAnimation : public Animation
{
public:
Animation::Type GetType() const override { return Animation::Parallel; }
uint32_t GetMask(uint32_t object) const override;
uint32_t GetObjects() const override;
TAnimObjects const & GetObjects() const override
{
return m_objects;
}
bool HasObject(TObject object) const override
{
return m_objects.find(object) != m_objects.end();
}
TObjectProperties const & GetProperties(TObject object) const override;
bool HasProperty(TObject object, TProperty property) const override;
void AddAnimation(ref_ptr<Animation> animation);
void AddAnimation(drape_ptr<Animation> && animation);
void OnStart() override;
void OnFinish() override;
@ -289,7 +307,9 @@ public:
void Advance(double elapsedSeconds) override;
private:
TAnimations m_animations;
list<drape_ptr<Animation>> m_animations;
TAnimObjects m_objects;
map<TObject, TObjectProperties> m_properties;
};
class AnimationSystem : private noncopyable
@ -299,18 +319,25 @@ public:
m2::AnyRectD GetRect(ScreenBase const & currentScreen);
bool AnimationExists(Animation::Object object);
bool AnimationExists(Animation::TObject object) const;
void AddAnimation(drape_ptr<Animation> && animation);
void AddAnimation(drape_ptr<Animation> && animation, bool force);
void PushAnimation(drape_ptr<Animation> && animation);
void Advance(double elapsedSeconds);
private:
AnimationSystem() {}
Animation::TPropValue GetProperty(Animation::TObject object, Animation::TProperty property, Animation::TPropValue current) const;
void SaveAnimationResult(Animation const & animation);
AnimationSystem();
private:
set<drape_ptr<Animation>> m_animations;
using TAnimationList = list<drape_ptr<Animation>>;
using TAnimationChain = deque<TAnimationList>;
using TPropertyCache = map<pair<Animation::TObject, Animation::TProperty>, Animation::TPropValue>;
TAnimationChain m_animationChain;
mutable TPropertyCache m_propertyCache;
};
}

View file

@ -568,9 +568,9 @@ void MyPositionController::CheckAnimFinished() const
void MyPositionController::AnimationStarted(ref_ptr<BaseModelViewAnimation> anim)
{
if (m_isPendingAnimation && m_animCreator != nullptr && anim != nullptr &&
(anim->GetType() == ModelViewAnimationType::FollowAndRotate ||
anim->GetType() == ModelViewAnimationType::Default))
if (m_isPendingAnimation && m_animCreator != nullptr)// && anim != nullptr &&
// (anim->GetType() == ModelViewAnimationType::FollowAndRotate ||
// anim->GetType() == ModelViewAnimationType::Default))
{
m_isPendingAnimation = false;
m_animCreator();

View file

@ -340,7 +340,7 @@ bool UserEventStream::SetScale(m2::PointD const & pxScaleCenter, double factor,
anim->SetMaxDuration(kMaxAnimationTimeSec);
if (df::IsAnimationAllowed(anim->GetDuration(), startScreen))
{
AnimationSystem::Instance().AddAnimation(move(anim));
AnimationSystem::Instance().AddAnimation(move(anim), true /* force */);
return false;
}
@ -463,7 +463,16 @@ bool UserEventStream::SetRect(m2::AnyRectD const & rect, bool isAnim)
return SetRect(rect, isAnim, [this](m2::AnyRectD const & startRect, m2::AnyRectD const & endRect,
double aDuration, double mDuration, double sDuration)
{
m_animation.reset(new ModelViewAnimation(startRect, endRect, aDuration, mDuration, sDuration));
drape_ptr<FollowAnimation> anim = make_unique_dp<FollowAnimation>();
anim->SetRotate(startRect.Angle().val(), endRect.Angle().val());
anim->SetMove(startRect.GlobalCenter(), endRect.GlobalCenter(), GetCurrentScreen());
m2::RectD pixelRect = GetCurrentScreen().PixelRect();
anim->SetScale(max(startRect.GetLocalRect().SizeX() / pixelRect.SizeX(),
startRect.GetLocalRect().SizeY() / pixelRect.SizeY()),
max(endRect.GetLocalRect().SizeX() / pixelRect.SizeX(),
endRect.GetLocalRect().SizeY() / pixelRect.SizeY()));
AnimationSystem::Instance().AddAnimation(move(anim), true /* force */);
//m_animation.reset(new ModelViewAnimation(startRect, endRect, aDuration, mDuration, sDuration));
if (m_listener)
m_listener->OnAnimationStarted(make_ref(m_animation));
});
@ -525,15 +534,21 @@ bool UserEventStream::SetFollowAndRotate(m2::PointD const & userPos, m2::PointD
m2::PointD const newCenter = FollowAndRotateAnimation::CalculateCenter(screen, userPos, pixelPos, -azimuth);
m2::AnyRectD const startRect = GetCurrentRect();
double const angleDuration = ModelViewAnimation::GetRotateDuration(startRect.Angle().val(), -azimuth);
double const moveDuration = ModelViewAnimation::GetMoveDuration(startRect.GlobalZero(), newCenter, screen);
double const duration = max(angleDuration, moveDuration);
if (df::IsAnimationAllowed(duration, screen))
//double const angleDuration = ModelViewAnimation::GetRotateDuration(startRect.Angle().val(), -azimuth);
//double const moveDuration = ModelViewAnimation::GetMoveDuration(startRect.GlobalZero(), newCenter, screen);
//double const duration = max(angleDuration, moveDuration);
drape_ptr<FollowAnimation> anim = make_unique_dp<FollowAnimation>();
anim->SetRotate(startRect.Angle().val(), -azimuth);
anim->SetMove(startRect.GlobalZero(), newCenter, screen);
if (df::IsAnimationAllowed(anim->GetDuration(), screen))
{
m_animation.reset(new FollowAndRotateAnimation(startRect, targetLocalRect, userPos,
screen.GtoP(userPos), pixelPos, azimuth, duration));
AnimationSystem::Instance().AddAnimation(move(anim), true /* force */);
//m_animation.reset(new FollowAndRotateAnimation(startRect, targetLocalRect, userPos,
// screen.GtoP(userPos), pixelPos, azimuth, duration));
if (m_listener)
m_listener->OnAnimationStarted(make_ref(m_animation));
m_listener->OnAnimationStarted(make_ref<FollowAndRotateAnimation>(nullptr/*m_animation*/));
return false;
}
}