metaforce/Runtime/Character/CAnimTreeTweenBase.cpp

129 lines
4.4 KiB
C++
Raw Normal View History

2016-09-04 02:27:35 +00:00
#include "CAnimTreeTweenBase.hpp"
#include "CSegStatementSet.hpp"
#include "CSegIdList.hpp"
2016-09-04 02:27:35 +00:00
2018-12-08 05:30:43 +00:00
namespace urde {
2016-09-04 02:27:35 +00:00
2016-09-11 18:40:33 +00:00
s32 CAnimTreeTweenBase::sAdvancementDepth = 0;
2016-09-10 04:50:00 +00:00
CAnimTreeTweenBase::CAnimTreeTweenBase(bool b1, const std::weak_ptr<CAnimTreeNode>& a,
2017-11-13 06:19:18 +00:00
const std::weak_ptr<CAnimTreeNode>& b, int flags, std::string_view name)
2018-12-08 05:30:43 +00:00
: CAnimTreeDoubleChild(a, b, name), x1c_flags(flags) {
x20_24_b1 = b1;
x20_25_cullSelector = 0;
2016-09-04 02:27:35 +00:00
}
void CAnimTreeTweenBase::VGetWeightedReaders(
2018-12-08 05:30:43 +00:00
rstl::reserved_vector<std::pair<float, std::weak_ptr<IAnimReader>>, 16>& out, float w) const {
float weight = GetBlendingWeight();
x14_a->VGetWeightedReaders(out, (1.f - weight) * w);
x18_b->VGetWeightedReaders(out, weight * w);
}
2016-09-11 01:25:59 +00:00
2018-12-08 05:30:43 +00:00
void CAnimTreeTweenBase::VGetSegStatementSet(const CSegIdList& list, CSegStatementSet& setOut) const {
float w = GetBlendingWeight();
static int sStack = 0;
++sStack;
if (w >= 1.f) {
x18_b->VGetSegStatementSet(list, setOut);
} else if (sStack > 3) {
auto& n = w > 0.5f ? x18_b : x14_a;
auto ptr = n->GetBestUnblendedChild();
if (!ptr)
ptr = n;
ptr->VGetSegStatementSet(list, setOut);
} else {
CSegStatementSet setA, setB;
x14_a->VGetSegStatementSet(list, setA);
x18_b->VGetSegStatementSet(list, setB);
for (CSegId id : list.GetList()) {
if (w < 0.0001f) {
setOut[id].x0_rotation = setA[id].x0_rotation;
if (setA[id].x1c_hasOffset) {
setOut[id].x10_offset = setA[id].x10_offset;
setOut[id].x1c_hasOffset = true;
}
2018-12-08 05:30:43 +00:00
} else {
setOut[id].x0_rotation = zeus::CQuaternion::slerpShort(setA[id].x0_rotation, setB[id].x0_rotation, w);
if (setA[id].x1c_hasOffset && setB[id].x1c_hasOffset) {
setOut[id].x10_offset = zeus::CVector3f::lerp(setA[id].x10_offset, setB[id].x10_offset, w);
setOut[id].x1c_hasOffset = true;
}
}
}
2018-12-08 05:30:43 +00:00
}
--sStack;
}
2016-09-04 02:27:35 +00:00
void CAnimTreeTweenBase::VGetSegStatementSet(const CSegIdList& list, CSegStatementSet& setOut,
2018-12-08 05:30:43 +00:00
const CCharAnimTime& time) const {
float w = GetBlendingWeight();
static int sStack = 0;
++sStack;
if (w >= 1.f) {
x18_b->VGetSegStatementSet(list, setOut, time);
} else if (sStack > 3) {
auto& n = w > 0.5f ? x18_b : x14_a;
n->GetBestUnblendedChild()->VGetSegStatementSet(list, setOut, time);
} else {
CSegStatementSet setA, setB;
x14_a->VGetSegStatementSet(list, setA, time);
x18_b->VGetSegStatementSet(list, setB, time);
for (CSegId id : list.GetList()) {
setOut[id].x0_rotation = zeus::CQuaternion::slerpShort(setA[id].x0_rotation, setB[id].x0_rotation, w);
if (setA[id].x1c_hasOffset && setB[id].x1c_hasOffset) {
setOut[id].x10_offset = zeus::CVector3f::lerp(setA[id].x10_offset, setB[id].x10_offset, w);
setOut[id].x1c_hasOffset = true;
}
}
2018-12-08 05:30:43 +00:00
}
--sStack;
2016-09-04 02:27:35 +00:00
}
2018-12-08 05:30:43 +00:00
bool CAnimTreeTweenBase::VHasOffset(const CSegId& seg) const {
return (x14_a->VHasOffset(seg) && x18_b->VHasOffset(seg));
2016-09-11 01:25:59 +00:00
}
2016-09-04 02:27:35 +00:00
2018-12-08 05:30:43 +00:00
zeus::CVector3f CAnimTreeTweenBase::VGetOffset(const CSegId& seg) const {
const float weight = GetBlendingWeight();
if (weight >= 1.0f)
return x18_b->VGetOffset(seg);
2016-09-14 05:45:46 +00:00
2018-12-08 05:30:43 +00:00
const zeus::CVector3f oA = x14_a->VGetOffset(seg);
const zeus::CVector3f oB = x18_b->VGetOffset(seg);
return zeus::CVector3f::lerp(oA, oB, weight);
2016-09-11 01:25:59 +00:00
}
2016-09-04 02:27:35 +00:00
2018-12-08 05:30:43 +00:00
zeus::CQuaternion CAnimTreeTweenBase::VGetRotation(const CSegId& seg) const {
const float weight = GetBlendingWeight();
if (weight >= 1.0f)
return x18_b->VGetRotation(seg);
2016-09-04 02:27:35 +00:00
2018-12-08 05:30:43 +00:00
const zeus::CQuaternion qA = x14_a->VGetRotation(seg);
const zeus::CQuaternion qB = x18_b->VGetRotation(seg);
return zeus::CQuaternion::slerp(qA, qB, weight);
2016-09-11 01:25:59 +00:00
}
2016-09-14 05:45:46 +00:00
2018-12-08 05:30:43 +00:00
std::experimental::optional<std::unique_ptr<IAnimReader>> CAnimTreeTweenBase::VSimplified() {
if (x20_25_cullSelector == 0) {
auto simpA = x14_a->Simplified();
auto simpB = x18_b->Simplified();
if (!simpA && !simpB)
return {};
auto clone = Clone();
if (simpA)
static_cast<CAnimTreeTweenBase&>(*clone).x14_a = CAnimTreeNode::Cast(std::move(*simpA));
if (simpB)
static_cast<CAnimTreeTweenBase&>(*clone).x18_b = CAnimTreeNode::Cast(std::move(*simpB));
return {std::move(clone)};
} else {
auto tmp = (x20_25_cullSelector == 1) ? x18_b : x14_a;
auto tmpUnblended = tmp->GetBestUnblendedChild();
if (!tmpUnblended)
return {tmp->Clone()};
else
2018-12-08 05:30:43 +00:00
return {tmpUnblended->Clone()};
}
2016-09-04 02:27:35 +00:00
}
2018-12-08 05:30:43 +00:00
} // namespace urde