/****************************************************************************** * Spine Runtimes License Agreement * Last updated April 5, 2025. Replaces all prior versions. * * Copyright (c) 2013-2025, Esoteric Software LLC * * Integration of the Spine Runtimes into software or otherwise creating * derivative works of the Spine Runtimes is permitted under the terms and * conditions of Section 2 of the Spine Editor License Agreement: * http://esotericsoftware.com/spine-editor-license * * Otherwise, it is permitted to integrate the Spine Runtimes into software * or otherwise create derivative works of the Spine Runtimes (collectively, * "Products"), provided that each user of the Products must obtain their own * Spine Editor license and redistribution of the Products in any form must * include this license and copyright notice. * * THE SPINE RUNTIMES ARE PROVIDED BY ESOTERIC SOFTWARE LLC "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL ESOTERIC SOFTWARE LLC BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, * BUSINESS INTERRUPTION, OR LOSS OF USE, DATA, OR PROFITS) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THE SPINE RUNTIMES, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *****************************************************************************/ #include #include #include #include #include #include #include #include #include using namespace spine; Animation::Animation(const String &name, Array &timelines, float duration) : _timelines(), _timelineIds(), _bones(), _duration(duration), _name(name) { setTimelines(timelines); } bool Animation::hasTimeline(Array &ids) { for (size_t i = 0; i < ids.size(); i++) { if (_timelineIds.containsKey(ids[i])) return true; } return false; } Animation::~Animation() { ArrayUtils::deleteElements(_timelines); } void Animation::apply(Skeleton &skeleton, float lastTime, float time, bool loop, Array *events, float alpha, bool fromSetup, bool add, bool out, bool appliedPose) { if (loop && _duration != 0) { time = MathUtil::fmod(time, _duration); if (lastTime > 0) { lastTime = MathUtil::fmod(lastTime, _duration); } } for (size_t i = 0, n = _timelines.size(); i < n; ++i) { _timelines[i]->apply(skeleton, lastTime, time, events, alpha, fromSetup, add, out, appliedPose); } } const String &Animation::getName() { return _name; } const Array &Animation::getBones() { return _bones; } Array &Animation::getTimelines() { return _timelines; } float Animation::getDuration() { return _duration; } void Animation::setDuration(float inValue) { _duration = inValue; } int Animation::search(Array &frames, float target) { size_t n = (int) frames.size(); for (size_t i = 1; i < n; i++) { if (frames[i] > target) return (int) (i - 1); } return (int) (n - 1); } int Animation::search(Array &frames, float target, int step) { size_t n = frames.size(); for (size_t i = step; i < n; i += step) if (frames[i] > target) return (int) (i - step); return (int) (n - step); } void Animation::setTimelines(Array &timelines) { _timelines = timelines; size_t n = timelines.size(); _timelineIds.clear(); _bones.clear(); HashMap boneSet; for (size_t i = 0; i < n; i++) { Timeline *timeline = timelines[i]; Array &propertyIds = timeline->getPropertyIds(); for (size_t ii = 0; ii < propertyIds.size(); ii++) { _timelineIds.put(propertyIds[ii], true); } int boneIndex = -1; if (timeline->getRTTI().instanceOf(BoneTimeline1::rtti)) { boneIndex = static_cast(timeline)->getBoneIndex(); } else if (timeline->getRTTI().instanceOf(BoneTimeline2::rtti)) { boneIndex = static_cast(timeline)->getBoneIndex(); } if (boneIndex >= 0 && !boneSet.containsKey(boneIndex)) { boneSet.put(boneIndex, true); _bones.add(boneIndex); } } }