2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2013 cocos2d-x.org
|
|
|
|
|
2022-07-10 09:47:41 +08:00
|
|
|
https://axis-project.github.io/
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2020-10-21 10:12:00 +08:00
|
|
|
#include "ActionTimeline/CCTimeLine.h"
|
|
|
|
#include "ActionTimeline/CCActionTimeline.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
USING_NS_AX;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
NS_TIMELINE_BEGIN
|
|
|
|
|
|
|
|
Timeline* Timeline::create()
|
|
|
|
{
|
2021-12-08 00:11:53 +08:00
|
|
|
Timeline* ret = new Timeline();
|
|
|
|
ret->autorelease();
|
|
|
|
return ret;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Timeline::Timeline()
|
|
|
|
: _currentKeyFrame(nullptr)
|
|
|
|
, _currentKeyFrameIndex(0)
|
|
|
|
, _fromIndex(0)
|
|
|
|
, _toIndex(0)
|
|
|
|
, _betweenDuration(0)
|
|
|
|
, _actionTag(0)
|
|
|
|
, _ActionTimeline(nullptr)
|
|
|
|
, _node(nullptr)
|
2021-12-25 10:04:45 +08:00
|
|
|
{}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
Timeline::~Timeline() {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
void Timeline::gotoFrame(int frameIndex)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_frames.size() == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
binarySearchKeyFrame(frameIndex);
|
|
|
|
apply(frameIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::stepToFrame(int frameIndex)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_frames.size() == 0)
|
2019-11-23 20:27:39 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
updateCurrentKeyFrame(frameIndex);
|
|
|
|
apply(frameIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
Timeline* Timeline::clone()
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
Timeline* timeline = Timeline::create();
|
2019-11-23 20:27:39 +08:00
|
|
|
timeline->_actionTag = _actionTag;
|
|
|
|
|
|
|
|
for (auto frame : _frames)
|
|
|
|
{
|
|
|
|
Frame* newFrame = frame->clone();
|
|
|
|
timeline->addFrame(newFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
return timeline;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::addFrame(Frame* frame)
|
|
|
|
{
|
|
|
|
_frames.pushBack(frame);
|
|
|
|
frame->setTimeline(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::insertFrame(Frame* frame, int index)
|
|
|
|
{
|
|
|
|
_frames.insert(index, frame);
|
|
|
|
frame->setTimeline(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::removeFrame(Frame* frame)
|
|
|
|
{
|
|
|
|
_frames.eraseObject(frame);
|
|
|
|
frame->setTimeline(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::setNode(Node* node)
|
|
|
|
{
|
|
|
|
for (auto frame : _frames)
|
|
|
|
{
|
|
|
|
frame->setNode(node);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Node* Timeline::getNode() const
|
|
|
|
{
|
|
|
|
return _node;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::apply(unsigned int frameIndex)
|
|
|
|
{
|
|
|
|
if (_currentKeyFrame)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
float currentPercent =
|
|
|
|
_betweenDuration == 0 ? 0 : (frameIndex - _currentKeyFrameIndex) / (float)_betweenDuration;
|
2019-11-23 20:27:39 +08:00
|
|
|
_currentKeyFrame->apply(currentPercent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::binarySearchKeyFrame(unsigned int frameIndex)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
Frame* from = nullptr;
|
|
|
|
Frame* to = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
long length = _frames.size();
|
2019-11-23 20:27:39 +08:00
|
|
|
bool needEnterFrame = false;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
do
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (frameIndex < _frames.at(0)->getFrameIndex())
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_currentKeyFrameIndex >= _frames.at(0)->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
needEnterFrame = true;
|
|
|
|
|
|
|
|
_fromIndex = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
_toIndex = 0;
|
|
|
|
|
|
|
|
from = to = _frames.at(0);
|
2019-11-23 20:27:39 +08:00
|
|
|
_currentKeyFrameIndex = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
_betweenDuration = _frames.at(0)->getFrameIndex();
|
2019-11-23 20:27:39 +08:00
|
|
|
break;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
else if (frameIndex >= _frames.at(length - 1)->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_fromIndex = (int)(length - 1);
|
2021-12-25 10:04:45 +08:00
|
|
|
_toIndex = 0;
|
|
|
|
|
|
|
|
from = to = _frames.at(length - 1);
|
2019-11-23 20:27:39 +08:00
|
|
|
if (from->isEnterWhenPassed())
|
|
|
|
needEnterFrame = true;
|
|
|
|
|
|
|
|
_currentKeyFrameIndex = _frames.at(length - 1)->getFrameIndex();
|
2021-12-25 10:04:45 +08:00
|
|
|
_betweenDuration = 0;
|
2019-11-23 20:27:39 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
long target = -1;
|
2021-12-25 10:04:45 +08:00
|
|
|
long low = 0, high = length - 1, mid = 0;
|
|
|
|
while (low <= high)
|
|
|
|
{
|
|
|
|
mid = (low + high) / 2;
|
|
|
|
if (frameIndex >= _frames.at(mid)->getFrameIndex() && frameIndex < _frames.at(mid + 1)->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
target = mid;
|
|
|
|
break;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_frames.at(mid)->getFrameIndex() > frameIndex)
|
|
|
|
high = mid - 1;
|
2019-11-23 20:27:39 +08:00
|
|
|
else
|
2021-12-25 10:04:45 +08:00
|
|
|
low = mid + 1;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
_fromIndex = (int)target;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (length > 1)
|
2019-11-23 20:27:39 +08:00
|
|
|
_toIndex = (int)(target + 1);
|
|
|
|
else
|
|
|
|
_toIndex = (int)target;
|
|
|
|
|
|
|
|
from = _frames.at(_fromIndex);
|
|
|
|
to = _frames.at(_toIndex);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (target == 0 && _currentKeyFrameIndex < from->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
needEnterFrame = true;
|
|
|
|
|
|
|
|
_currentKeyFrameIndex = from->getFrameIndex();
|
2021-12-25 10:04:45 +08:00
|
|
|
_betweenDuration = to->getFrameIndex() - from->getFrameIndex();
|
2019-11-23 20:27:39 +08:00
|
|
|
} while (0);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (needEnterFrame || _currentKeyFrame != from)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_currentKeyFrame = from;
|
|
|
|
_currentKeyFrame->onEnter(to, frameIndex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Timeline::updateCurrentKeyFrame(unsigned int frameIndex)
|
|
|
|
{
|
|
|
|
//! If play to current frame's front or back, then find current frame again
|
|
|
|
if (frameIndex < _currentKeyFrameIndex || frameIndex >= _currentKeyFrameIndex + _betweenDuration)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
Frame* from = nullptr;
|
|
|
|
Frame* to = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
do
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
long length = _frames.size();
|
|
|
|
|
|
|
|
if (frameIndex < _frames.at(0)->getFrameIndex())
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
from = to = _frames.at(0);
|
2019-11-23 20:27:39 +08:00
|
|
|
_currentKeyFrameIndex = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
_betweenDuration = _frames.at(0)->getFrameIndex();
|
2019-11-23 20:27:39 +08:00
|
|
|
break;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
else if (frameIndex >= _frames.at(length - 1)->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned int lastFrameIndex = _frames.at(length - 1)->getFrameIndex();
|
|
|
|
if (_currentKeyFrameIndex >= lastFrameIndex)
|
2019-11-23 20:27:39 +08:00
|
|
|
return;
|
|
|
|
frameIndex = lastFrameIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
_fromIndex = _toIndex;
|
|
|
|
from = _frames.at(_fromIndex);
|
|
|
|
_currentKeyFrameIndex = from->getFrameIndex();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
_toIndex = _fromIndex + 1;
|
|
|
|
if ((ssize_t)_toIndex >= length)
|
|
|
|
{
|
|
|
|
_toIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
to = _frames.at(_toIndex);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (frameIndex == from->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
break;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (frameIndex > from->getFrameIndex() && frameIndex < to->getFrameIndex())
|
2019-11-23 20:27:39 +08:00
|
|
|
break;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (from->isEnterWhenPassed())
|
2019-11-23 20:27:39 +08:00
|
|
|
from->onEnter(to, from->getFrameIndex());
|
2021-12-25 10:04:45 +08:00
|
|
|
} while (true);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (_fromIndex == length - 1)
|
2019-11-23 20:27:39 +08:00
|
|
|
to = from;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
_betweenDuration = to->getFrameIndex() - from->getFrameIndex();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
} while (0);
|
|
|
|
|
|
|
|
_currentKeyFrame = from;
|
|
|
|
_currentKeyFrame->onEnter(to, frameIndex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_TIMELINE_END
|