8#ifndef PXR_BASE_TS_SPLINE_DATA_H
9#define PXR_BASE_TS_SPLINE_DATA_H
12#include "pxr/base/ts/api.h"
13#include "pxr/base/ts/knotData.h"
14#include "pxr/base/ts/types.h"
15#include "pxr/base/ts/typeHelpers.h"
18#include "pxr/base/tf/type.h"
22#include <unordered_map>
28PXR_NAMESPACE_OPEN_SCOPE
47 static Ts_SplineData* Create(
49 const Ts_SplineData *overallParamSource =
nullptr);
51 virtual ~Ts_SplineData();
56 virtual TfType GetValueType()
const = 0;
57 virtual size_t GetKnotStructSize()
const = 0;
58 virtual Ts_SplineData* Clone()
const = 0;
60 virtual bool operator==(
const Ts_SplineData &other)
const = 0;
62 virtual void ReserveForKnotCount(
size_t count) = 0;
63 virtual void PushKnot(
64 const Ts_KnotData *knotData,
74 virtual size_t SetKnot(
75 const Ts_KnotData *knotData,
80 virtual size_t SetKnotFromDouble(
81 const Ts_TypedKnotData<double>* knotData,
84 virtual Ts_KnotData* CloneKnotAtIndex(
size_t index)
const = 0;
85 virtual Ts_KnotData* CloneKnotAtTime(TsTime time)
const = 0;
86 virtual Ts_KnotData* GetKnotPtrAtIndex(
size_t index) = 0;
87 virtual const Ts_KnotData* GetKnotPtrAtIndex(
size_t index)
const = 0;
88 virtual Ts_TypedKnotData<double>
89 GetKnotDataAsDouble(
size_t index)
const = 0;
90 virtual double GetKnotValueAsDouble(
size_t index)
const = 0;
91 virtual double GetKnotPreValueAsDouble(
size_t index)
const = 0;
93 virtual void ClearKnots() = 0;
94 virtual void RemoveKnotAtTime(TsTime time) = 0;
96 virtual void ApplyOffsetAndScale(
100 virtual bool HasValueBlocks()
const = 0;
101 virtual bool HasValueBlockAtTime(TsTime time)
const = 0;
103 virtual bool UpdateKnotTangentsAtIndex(
size_t index) = 0;
110 size_t *firstProtoIndexOut =
nullptr)
const;
133 TsCurveType curveType : 2;
144 std::vector<TsTime> times;
147 std::unordered_map<TsTime, VtDictionary> customData;
154struct Ts_TypedSplineData final :
158 TfType GetValueType()
const override;
159 size_t GetKnotStructSize()
const override;
160 Ts_SplineData* Clone()
const override;
162 bool operator==(
const Ts_SplineData &other)
const override;
164 void ReserveForKnotCount(
size_t count)
override;
166 const Ts_KnotData *knotData,
174 const Ts_KnotData *knotData,
179 size_t SetKnotFromDouble(
180 const Ts_TypedKnotData<double>* knotData,
183 Ts_KnotData* CloneKnotAtIndex(
size_t index)
const override;
184 Ts_KnotData* CloneKnotAtTime(TsTime time)
const override;
185 Ts_KnotData* GetKnotPtrAtIndex(
size_t index)
override;
186 const Ts_KnotData* GetKnotPtrAtIndex(
size_t index)
const override;
187 Ts_TypedKnotData<double>
188 GetKnotDataAsDouble(
size_t index)
const override;
189 double GetKnotValueAsDouble(
size_t index)
const override;
190 double GetKnotPreValueAsDouble(
size_t index)
const override;
192 void ClearKnots()
override;
193 void RemoveKnotAtTime(TsTime time)
override;
202 void ApplyOffsetAndScale(
204 double scale)
override;
206 bool HasValueBlocks()
const override;
207 bool HasValueBlockAtTime(TsTime time)
const override;
209 bool UpdateKnotTangentsAtIndex(
size_t index)
override;
213 std::vector<Ts_TypedKnotData<T>> knots;
224Ts_GetSplineData(
const TsSpline &spline);
227Ts_TypedSplineData<T>*
228Ts_GetTypedSplineData(
TsSpline &spline);
231const Ts_TypedSplineData<T>*
232Ts_GetTypedSplineData(
const TsSpline &spline);
239TfType Ts_TypedSplineData<T>::GetValueType()
const
246 return Ts_GetType<T>();
250size_t Ts_TypedSplineData<T>::GetKnotStructSize()
const
252 return sizeof(Ts_TypedKnotData<T>);
257Ts_TypedSplineData<T>::Clone()
const
259 return new Ts_TypedSplineData<T>(*
this);
263bool Ts_TypedSplineData<T>::operator==(
264 const Ts_SplineData &other)
const
267 if (isTyped != other.isTyped
268 || timeValued != other.timeValued
269 || curveType != other.curveType
270 || preExtrapolation != other.preExtrapolation
271 || postExtrapolation != other.postExtrapolation
272 || loopParams != other.loopParams
273 || customData != other.customData)
280 const Ts_TypedSplineData<T>*
const typedOther =
281 dynamic_cast<const Ts_TypedSplineData<T>*
>(&other);
288 return knots == typedOther->knots;
292void Ts_TypedSplineData<T>::ReserveForKnotCount(
295 times.reserve(count);
296 knots.reserve(count);
300void Ts_TypedSplineData<T>::PushKnot(
301 const Ts_KnotData*
const knotData,
304 const Ts_TypedKnotData<T>*
const typedKnotData =
305 static_cast<const Ts_TypedKnotData<T>*
>(knotData);
307 times.push_back(knotData->time);
308 knots.push_back(*typedKnotData);
310 if (!customDataIn.
empty())
312 customData[knotData->time] = customDataIn;
356size_t Ts_TypedSplineData<T>::SetKnot(
357 const Ts_KnotData*
const knotData,
360 const Ts_TypedKnotData<T>*
const typedKnotData =
361 static_cast<const Ts_TypedKnotData<T>*
>(knotData);
365 std::lower_bound(times.begin(), times.end(), knotData->time);
368 const bool overwrite =
369 (it != times.end() && *it == knotData->time);
374 times[idx] = knotData->time;
375 knots[idx] = *typedKnotData;
379 times.insert(it, knotData->time);
380 knots.insert(knots.begin() + idx, *typedKnotData);
384 if (!customDataIn.
empty())
386 customData[knotData->time] = customDataIn;
393size_t Ts_TypedSplineData<T>::SetKnotFromDouble(
394 const Ts_TypedKnotData<double>* knotData,
398 if constexpr(std::is_same_v<T, double>) {
399 return SetKnot(knotData, customDataIn);
402 Ts_TypedKnotData<T> typedData;
406 static_cast<Ts_KnotData&
>(typedData) =
407 static_cast<const Ts_KnotData&
>(*knotData);
415 return std::min(T(v), std::numeric_limits<T>::max());
417 return std::max(T(v), std::numeric_limits<T>::lowest());
422 typedData.value = _Clamp_cast(knotData->value);
423 typedData.preValue = _Clamp_cast(knotData->preValue);
429 auto _ConvertTangent =
430 [&_Clamp_cast](
double slope,
double* width) -> T
432 T typedSlope = T(slope);
435 if (Ts_IsFinite(typedSlope)) {
441 double height = *width * slope;
444 typedSlope = _Clamp_cast(slope);
447 *width = height / typedSlope;
452 typedData.preTanSlope = _ConvertTangent(knotData->preTanSlope,
453 &typedData.preTanWidth);
454 typedData.postTanSlope = _ConvertTangent(knotData->postTanSlope,
455 &typedData.postTanWidth);
457 return SetKnot(&typedData, customDataIn);
462Ts_TypedSplineData<T>::CloneKnotAtIndex(
463 const size_t index)
const
465 return new Ts_TypedKnotData<T>(knots[index]);
470Ts_TypedSplineData<T>::CloneKnotAtTime(
471 const TsTime time)
const
473 const auto it = std::lower_bound(times.begin(), times.end(), time);
474 if (it == times.end() || *it != time)
479 const auto knotIt = knots.begin() + (it - times.begin());
480 return new Ts_TypedKnotData<T>(*knotIt);
485Ts_TypedSplineData<T>::GetKnotPtrAtIndex(
488 return &(knots[index]);
493Ts_TypedSplineData<T>::GetKnotPtrAtIndex(
494 const size_t index)
const
496 return &(knots[index]);
501Ts_TypedKnotData<double>
502Ts_TypedSplineData<T>::GetKnotDataAsDouble(
503 const size_t index)
const
505 const Ts_TypedKnotData<T> &in = knots[index];
506 Ts_TypedKnotData<double> out;
510 static_cast<Ts_KnotData&
>(out) =
static_cast<const Ts_KnotData&
>(in);
513 out.value = in.value;
514 out.preValue = in.preValue;
515 out.preTanSlope = in.preTanSlope;
516 out.postTanSlope = in.postTanSlope;
524Ts_TypedSplineData<T>::GetKnotValueAsDouble(
525 const size_t index)
const
527 const Ts_TypedKnotData<T> &typedData = knots[index];
528 return typedData.value;
534Ts_TypedSplineData<T>::GetKnotPreValueAsDouble(
535 const size_t index)
const
537 const Ts_TypedKnotData<T> &typedData = knots[index];
538 return typedData.GetPreValue();
542void Ts_TypedSplineData<T>::ClearKnots()
550void Ts_TypedSplineData<T>::RemoveKnotAtTime(
553 const auto it = std::lower_bound(times.begin(), times.end(), time);
554 if (it == times.end() || *it != time)
560 const size_t idx = it - times.begin();
562 customData.erase(time);
563 knots.erase(knots.begin() + idx);
567 UpdateKnotTangentsAtIndex(idx - 1);
569 if (idx < times.size()) {
570 UpdateKnotTangentsAtIndex(idx);
575static void _ApplyOffsetAndScaleToKnot(
576 Ts_TypedKnotData<T>*
const knotData,
584 knotData->time = knotData->time * scale + offset;
587 knotData->preTanWidth *= scale;
588 knotData->postTanWidth *= scale;
591 knotData->preTanSlope /= scale;
592 knotData->postTanSlope /= scale;
596void Ts_TypedSplineData<T>::ApplyOffsetAndScale(
603 "collapsing/reversing time and spline representation "
615 if (preExtrapolation.mode == TsExtrapSloped)
617 preExtrapolation.slope /= scale;
619 if (postExtrapolation.mode == TsExtrapSloped)
621 postExtrapolation.slope /= scale;
625 if (loopParams.protoEnd > loopParams.protoStart)
628 loopParams.protoStart = loopParams.protoStart * scale + offset;
629 loopParams.protoEnd = loopParams.protoEnd * scale + offset;
633 for (TsTime &time : times) {
634 time = time * scale + offset;
642 for (Ts_TypedKnotData<T> &knotData : knots)
644 _ApplyOffsetAndScaleToKnot(&knotData, offset, scale);
648 static_cast<T
>(knotData.value * scale + offset);
650 static_cast<T
>(knotData.preValue * scale + offset);
655 for (Ts_TypedKnotData<T> &knotData : knots) {
656 _ApplyOffsetAndScaleToKnot(&knotData, offset, scale);
661 if (!customData.empty())
663 std::unordered_map<TsTime, VtDictionary> newCustomData;
664 for (
const auto &mapPair : customData) {
665 newCustomData[mapPair.first * scale + offset] = mapPair.second;
667 customData.swap(newCustomData);
672bool Ts_TypedSplineData<T>::HasValueBlocks()
const
679 if (preExtrapolation.mode == TsExtrapValueBlock
680 || postExtrapolation.mode == TsExtrapValueBlock)
685 for (
const Ts_TypedKnotData<T> &knotData : knots)
687 if (knotData.nextInterp == TsInterpValueBlock)
697bool Ts_TypedSplineData<T>::HasValueBlockAtTime(
698 const TsTime time)
const
708 std::lower_bound(times.begin(), times.end(), time);
712 if (lbIt == times.end())
714 return postExtrapolation.mode == TsExtrapValueBlock;
721 const auto knotIt = knots.begin() + (lbIt - times.begin());
722 return knotIt->nextInterp == TsInterpValueBlock;
727 if (lbIt == times.begin())
729 return preExtrapolation.mode == TsExtrapValueBlock;
734 const auto knotIt = knots.begin() + (lbIt - times.begin());
735 return (knotIt - 1)->nextInterp == TsInterpValueBlock;
739bool Ts_TypedSplineData<T>::UpdateKnotTangentsAtIndex(
size_t index)
743 "Knot index (%zd) out of range [0 .. %zd)",
744 index, knots.size()))
749 Ts_TypedKnotData<T>* prevKnot = (index > 0 ? &knots[index - 1] :
nullptr);
750 Ts_TypedKnotData<T>* knot = &knots[index];
751 Ts_TypedKnotData<T>* nextKnot = (index < knots.size() - 1
755 return knot->UpdateTangents(prevKnot, nextKnot, curveType);
759Ts_TypedSplineData<T>*
760Ts_GetTypedSplineData(
TsSpline &spline)
762 return static_cast<Ts_TypedSplineData<T>*
>(
763 Ts_GetSplineData(spline));
767const Ts_TypedSplineData<T>*
768Ts_GetTypedSplineData(
const TsSpline &spline)
770 return static_cast<Ts_TypedSplineData<T>*
>(
771 Ts_GetSplineData(spline));
775PXR_NAMESPACE_CLOSE_SCOPE
Low-level utilities for informing users of various internal and external diagnostic conditions.
TfType represents a dynamic runtime type.
A mathematical description of a curved function from time to value.
A map with string keys and VtValue values.
VT_API bool empty() const
true if the VtDictionary's size is 0.
#define TF_CODING_ERROR(fmt, args)
Issue an internal programming error, but continue execution.
#define TF_VERIFY(cond, format,...)
Checks a condition and reports an error if it evaluates false.