#include "Runtime/Character/CAnimTreeDoubleChild.hpp" namespace urde { CAnimTreeDoubleChild::CAnimTreeDoubleChild(const std::weak_ptr<CAnimTreeNode>& a, const std::weak_ptr<CAnimTreeNode>& b, std::string_view name) : CAnimTreeNode(name), x14_a(a.lock()), x18_b(b.lock()) {} CAnimTreeDoubleChild::CDoubleChildAdvancementResult CAnimTreeDoubleChild::AdvanceViewBothChildren(const CCharAnimTime& time, bool runLeft, bool loopLeft) { CCharAnimTime lRemTime = time; CCharAnimTime totalTime; if (!runLeft) totalTime = CCharAnimTime(); else if (loopLeft) totalTime = CCharAnimTime::Infinity(); else totalTime = x14_a->VGetTimeRemaining(); SAdvancementDeltas leftDeltas, rightDeltas; CCharAnimTime rRemTime = time; if (time.GreaterThanZero()) { while (lRemTime.GreaterThanZero() && !lRemTime.EpsilonZero() && totalTime.GreaterThanZero() && (loopLeft || !totalTime.EpsilonZero())) { SAdvancementResults res = x14_a->VAdvanceView(lRemTime); auto simp = x14_a->Simplified(); if (simp) x14_a = CAnimTreeNode::Cast(std::move(*simp)); leftDeltas.x0_posDelta += res.x8_deltas.x0_posDelta; leftDeltas.xc_rotDelta = leftDeltas.xc_rotDelta * res.x8_deltas.xc_rotDelta; if (!loopLeft) totalTime = x14_a->VGetTimeRemaining(); lRemTime = res.x0_remTime; } while (rRemTime.GreaterThanZero() && !rRemTime.EpsilonZero()) { SAdvancementResults res = x18_b->VAdvanceView(rRemTime); auto simp = x18_b->Simplified(); if (simp) x18_b = CAnimTreeNode::Cast(std::move(*simp)); rightDeltas.x0_posDelta += res.x8_deltas.x0_posDelta; rightDeltas.xc_rotDelta = rightDeltas.xc_rotDelta * res.x8_deltas.xc_rotDelta; rRemTime = res.x0_remTime; } } return {time, leftDeltas, rightDeltas}; } SAdvancementResults CAnimTreeDoubleChild::VAdvanceView(const CCharAnimTime& a) { SAdvancementResults resA = x14_a->VAdvanceView(a); SAdvancementResults resB = x14_a->VAdvanceView(a); return (resA.x0_remTime > resB.x0_remTime) ? resA : resB; } size_t CAnimTreeDoubleChild::VGetBoolPOIList(const CCharAnimTime& time, CBoolPOINode* listOut, size_t capacity, size_t iterator, u32 unk) const { size_t newCapacity = x14_a->GetBoolPOIList(time, listOut, capacity, iterator, unk); newCapacity += x18_b->GetBoolPOIList(time, listOut, capacity, newCapacity + iterator, unk); if (newCapacity > capacity) { newCapacity = capacity; } std::sort(listOut, listOut + newCapacity); return newCapacity; } size_t CAnimTreeDoubleChild::VGetInt32POIList(const CCharAnimTime& time, CInt32POINode* listOut, size_t capacity, size_t iterator, u32 unk) const { size_t newCapacity = x14_a->GetInt32POIList(time, listOut, capacity, iterator, unk); newCapacity += x18_b->GetInt32POIList(time, listOut, capacity, newCapacity + iterator, unk); if (newCapacity > capacity) { newCapacity = capacity; } std::sort(listOut, listOut + newCapacity); return newCapacity; } size_t CAnimTreeDoubleChild::VGetParticlePOIList(const CCharAnimTime& time, CParticlePOINode* listOut, size_t capacity, size_t iterator, u32 unk) const { size_t newCapacity = x14_a->GetParticlePOIList(time, listOut, capacity, iterator, unk); newCapacity += x18_b->GetParticlePOIList(time, listOut, capacity, newCapacity + iterator, unk); if (newCapacity > capacity) { newCapacity = capacity; } std::sort(listOut, listOut + newCapacity); return newCapacity; } size_t CAnimTreeDoubleChild::VGetSoundPOIList(const CCharAnimTime& time, CSoundPOINode* listOut, size_t capacity, size_t iterator, u32 unk) const { size_t newCapacity = x14_a->GetSoundPOIList(time, listOut, capacity, iterator, unk); newCapacity += x18_b->GetSoundPOIList(time, listOut, capacity, newCapacity + iterator, unk); if (newCapacity > capacity) { newCapacity = capacity; } std::sort(listOut, listOut + newCapacity); return newCapacity; } bool CAnimTreeDoubleChild::VGetBoolPOIState(std::string_view name) const { return x18_b->VGetBoolPOIState(name); } s32 CAnimTreeDoubleChild::VGetInt32POIState(std::string_view name) const { return x18_b->VGetInt32POIState(name); } CParticleData::EParentedMode CAnimTreeDoubleChild::VGetParticlePOIState(std::string_view name) const { return x18_b->VGetParticlePOIState(name); } void CAnimTreeDoubleChild::VSetPhase(float phase) { x14_a->VSetPhase(phase); x18_b->VSetPhase(phase); } SAdvancementResults CAnimTreeDoubleChild::VGetAdvancementResults(const CCharAnimTime& a, const CCharAnimTime& b) const { SAdvancementResults resA = x14_a->VGetAdvancementResults(a, b); SAdvancementResults resB = x18_b->VGetAdvancementResults(a, b); return (resA.x0_remTime > resB.x0_remTime) ? resA : resB; } u32 CAnimTreeDoubleChild::Depth() const { return std::max(x14_a->Depth(), x18_b->Depth()) + 1; } CAnimTreeEffectiveContribution CAnimTreeDoubleChild::VGetContributionOfHighestInfluence() const { CAnimTreeEffectiveContribution cA = x14_a->GetContributionOfHighestInfluence(); CAnimTreeEffectiveContribution cB = x18_b->GetContributionOfHighestInfluence(); float leftWeight = (1.f - VGetRightChildWeight()) * cA.GetContributionWeight(); float rightWeight = VGetRightChildWeight() * cB.GetContributionWeight(); if (leftWeight > rightWeight) { return {leftWeight, cA.GetPrimitiveName(), cA.GetSteadyStateAnimInfo(), cA.GetTimeRemaining(), cA.GetAnimDatabaseIndex()}; } else { return {rightWeight, cB.GetPrimitiveName(), cB.GetSteadyStateAnimInfo(), cB.GetTimeRemaining(), cB.GetAnimDatabaseIndex()}; } } u32 CAnimTreeDoubleChild::VGetNumChildren() const { return x14_a->VGetNumChildren() + x18_b->VGetNumChildren() + 2; } std::shared_ptr<IAnimReader> CAnimTreeDoubleChild::VGetBestUnblendedChild() const { std::shared_ptr<CAnimTreeNode> bestChild = (VGetRightChildWeight() > 0.5f) ? x18_b : x14_a; if (!bestChild) return {}; return bestChild->GetBestUnblendedChild(); } void CAnimTreeDoubleChild::VGetWeightedReaders( rstl::reserved_vector<std::pair<float, std::weak_ptr<IAnimReader>>, 16>& out, float w) const { x14_a->VGetWeightedReaders(out, w); x18_b->VGetWeightedReaders(out, w); } } // namespace urde