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;
92 virtual void ClearKnots() = 0;
93 virtual void RemoveKnotAtTime(TsTime time) = 0;
95 virtual void ApplyOffsetAndScale(
99 virtual bool HasValueBlocks()
const = 0;
100 virtual bool HasValueBlockAtTime(TsTime time)
const = 0;
102 virtual bool UpdateKnotTangentsAtIndex(
size_t index) = 0;
109 size_t *firstProtoIndexOut =
nullptr)
const;
132 TsCurveType curveType : 2;
143 std::vector<TsTime> times;
146 std::unordered_map<TsTime, VtDictionary> customData;
153struct Ts_TypedSplineData final :
157 TfType GetValueType()
const override;
158 size_t GetKnotStructSize()
const override;
159 Ts_SplineData* Clone()
const override;
161 bool operator==(
const Ts_SplineData &other)
const override;
163 void ReserveForKnotCount(
size_t count)
override;
165 const Ts_KnotData *knotData,
173 const Ts_KnotData *knotData,
178 size_t SetKnotFromDouble(
179 const Ts_TypedKnotData<double>* knotData,
182 Ts_KnotData* CloneKnotAtIndex(
size_t index)
const override;
183 Ts_KnotData* CloneKnotAtTime(TsTime time)
const override;
184 Ts_KnotData* GetKnotPtrAtIndex(
size_t index)
override;
185 const Ts_KnotData* GetKnotPtrAtIndex(
size_t index)
const override;
186 Ts_TypedKnotData<double>
187 GetKnotDataAsDouble(
size_t index)
const override;
188 double GetKnotValueAsDouble(
size_t index)
const override;
190 void ClearKnots()
override;
191 void RemoveKnotAtTime(TsTime time)
override;
200 void ApplyOffsetAndScale(
202 double scale)
override;
204 bool HasValueBlocks()
const override;
205 bool HasValueBlockAtTime(TsTime time)
const override;
207 bool UpdateKnotTangentsAtIndex(
size_t index)
override;
211 std::vector<Ts_TypedKnotData<T>> knots;
222Ts_GetSplineData(
const TsSpline &spline);
225Ts_TypedSplineData<T>*
226Ts_GetTypedSplineData(
TsSpline &spline);
229const Ts_TypedSplineData<T>*
230Ts_GetTypedSplineData(
const TsSpline &spline);
237TfType Ts_TypedSplineData<T>::GetValueType()
const
244 return Ts_GetType<T>();
248size_t Ts_TypedSplineData<T>::GetKnotStructSize()
const
250 return sizeof(Ts_TypedKnotData<T>);
255Ts_TypedSplineData<T>::Clone()
const
257 return new Ts_TypedSplineData<T>(*
this);
261bool Ts_TypedSplineData<T>::operator==(
262 const Ts_SplineData &other)
const
265 if (isTyped != other.isTyped
266 || timeValued != other.timeValued
267 || curveType != other.curveType
268 || preExtrapolation != other.preExtrapolation
269 || postExtrapolation != other.postExtrapolation
270 || loopParams != other.loopParams
271 || customData != other.customData)
278 const Ts_TypedSplineData<T>*
const typedOther =
279 dynamic_cast<const Ts_TypedSplineData<T>*
>(&other);
286 return knots == typedOther->knots;
290void Ts_TypedSplineData<T>::ReserveForKnotCount(
293 times.reserve(count);
294 knots.reserve(count);
298void Ts_TypedSplineData<T>::PushKnot(
299 const Ts_KnotData*
const knotData,
302 const Ts_TypedKnotData<T>*
const typedKnotData =
303 static_cast<const Ts_TypedKnotData<T>*
>(knotData);
305 times.push_back(knotData->time);
306 knots.push_back(*typedKnotData);
308 if (!customDataIn.
empty())
310 customData[knotData->time] = customDataIn;
354size_t Ts_TypedSplineData<T>::SetKnot(
355 const Ts_KnotData*
const knotData,
358 const Ts_TypedKnotData<T>*
const typedKnotData =
359 static_cast<const Ts_TypedKnotData<T>*
>(knotData);
363 std::lower_bound(times.begin(), times.end(), knotData->time);
366 const bool overwrite =
367 (it != times.end() && *it == knotData->time);
372 times[idx] = knotData->time;
373 knots[idx] = *typedKnotData;
377 times.insert(it, knotData->time);
378 knots.insert(knots.begin() + idx, *typedKnotData);
382 if (!customDataIn.
empty())
384 customData[knotData->time] = customDataIn;
391size_t Ts_TypedSplineData<T>::SetKnotFromDouble(
392 const Ts_TypedKnotData<double>* knotData,
396 if constexpr(std::is_same_v<T, double>) {
397 return SetKnot(knotData, customDataIn);
400 Ts_TypedKnotData<T> typedData;
404 static_cast<Ts_KnotData&
>(typedData) =
405 static_cast<const Ts_KnotData&
>(*knotData);
413 return std::min(T(v), std::numeric_limits<T>::max());
415 return std::max(T(v), std::numeric_limits<T>::lowest());
420 typedData.value = _Clamp_cast(knotData->value);
421 typedData.preValue = _Clamp_cast(knotData->preValue);
427 auto _ConvertTangent =
428 [&_Clamp_cast](
double slope,
double* width) -> T
430 T typedSlope = T(slope);
433 if (Ts_IsFinite(typedSlope)) {
439 double height = *width * slope;
442 typedSlope = _Clamp_cast(slope);
445 *width = height / typedSlope;
450 typedData.preTanSlope = _ConvertTangent(knotData->preTanSlope,
451 &typedData.preTanWidth);
452 typedData.postTanSlope = _ConvertTangent(knotData->postTanSlope,
453 &typedData.postTanWidth);
455 return SetKnot(&typedData, customDataIn);
460Ts_TypedSplineData<T>::CloneKnotAtIndex(
461 const size_t index)
const
463 return new Ts_TypedKnotData<T>(knots[index]);
468Ts_TypedSplineData<T>::CloneKnotAtTime(
469 const TsTime time)
const
471 const auto it = std::lower_bound(times.begin(), times.end(), time);
472 if (it == times.end() || *it != time)
477 const auto knotIt = knots.begin() + (it - times.begin());
478 return new Ts_TypedKnotData<T>(*knotIt);
483Ts_TypedSplineData<T>::GetKnotPtrAtIndex(
486 return &(knots[index]);
491Ts_TypedSplineData<T>::GetKnotPtrAtIndex(
492 const size_t index)
const
494 return &(knots[index]);
499Ts_TypedKnotData<double>
500Ts_TypedSplineData<T>::GetKnotDataAsDouble(
501 const size_t index)
const
503 const Ts_TypedKnotData<T> &in = knots[index];
504 Ts_TypedKnotData<double> out;
508 static_cast<Ts_KnotData&
>(out) =
static_cast<const Ts_KnotData&
>(in);
511 out.value = in.value;
512 out.preValue = in.preValue;
513 out.preTanSlope = in.preTanSlope;
514 out.postTanSlope = in.postTanSlope;
522Ts_TypedSplineData<T>::GetKnotValueAsDouble(
523 const size_t index)
const
525 const Ts_TypedKnotData<T> &typedData = knots[index];
526 return typedData.value;
530void Ts_TypedSplineData<T>::ClearKnots()
538void Ts_TypedSplineData<T>::RemoveKnotAtTime(
541 const auto it = std::lower_bound(times.begin(), times.end(), time);
542 if (it == times.end() || *it != time)
548 const size_t idx = it - times.begin();
550 customData.erase(time);
551 knots.erase(knots.begin() + idx);
555 UpdateKnotTangentsAtIndex(idx - 1);
557 if (idx < times.size()) {
558 UpdateKnotTangentsAtIndex(idx);
563static void _ApplyOffsetAndScaleToKnot(
564 Ts_TypedKnotData<T>*
const knotData,
572 knotData->time = knotData->time * scale + offset;
575 knotData->preTanWidth *= scale;
576 knotData->postTanWidth *= scale;
579 knotData->preTanSlope /= scale;
580 knotData->postTanSlope /= scale;
584void Ts_TypedSplineData<T>::ApplyOffsetAndScale(
591 "collapsing/reversing time and spline representation "
603 if (preExtrapolation.mode == TsExtrapSloped)
605 preExtrapolation.slope /= scale;
607 if (postExtrapolation.mode == TsExtrapSloped)
609 postExtrapolation.slope /= scale;
613 if (loopParams.protoEnd > loopParams.protoStart)
616 loopParams.protoStart = loopParams.protoStart * scale + offset;
617 loopParams.protoEnd = loopParams.protoEnd * scale + offset;
621 for (TsTime &time : times) {
622 time = time * scale + offset;
630 for (Ts_TypedKnotData<T> &knotData : knots)
632 _ApplyOffsetAndScaleToKnot(&knotData, offset, scale);
636 static_cast<T
>(knotData.value * scale + offset);
638 static_cast<T
>(knotData.preValue * scale + offset);
643 for (Ts_TypedKnotData<T> &knotData : knots) {
644 _ApplyOffsetAndScaleToKnot(&knotData, offset, scale);
649 if (!customData.empty())
651 std::unordered_map<TsTime, VtDictionary> newCustomData;
652 for (
const auto &mapPair : customData) {
653 newCustomData[mapPair.first * scale + offset] = mapPair.second;
655 customData.swap(newCustomData);
660bool Ts_TypedSplineData<T>::HasValueBlocks()
const
667 if (preExtrapolation.mode == TsExtrapValueBlock
668 || postExtrapolation.mode == TsExtrapValueBlock)
673 for (
const Ts_TypedKnotData<T> &knotData : knots)
675 if (knotData.nextInterp == TsInterpValueBlock)
685bool Ts_TypedSplineData<T>::HasValueBlockAtTime(
686 const TsTime time)
const
696 std::lower_bound(times.begin(), times.end(), time);
700 if (lbIt == times.end())
702 return postExtrapolation.mode == TsExtrapValueBlock;
709 const auto knotIt = knots.begin() + (lbIt - times.begin());
710 return knotIt->nextInterp == TsInterpValueBlock;
715 if (lbIt == times.begin())
717 return preExtrapolation.mode == TsExtrapValueBlock;
722 const auto knotIt = knots.begin() + (lbIt - times.begin());
723 return (knotIt - 1)->nextInterp == TsInterpValueBlock;
727bool Ts_TypedSplineData<T>::UpdateKnotTangentsAtIndex(
size_t index)
731 "Knot index (%zd) out of range [0 .. %zd)",
732 index, knots.size()))
737 Ts_TypedKnotData<T>* prevKnot = (index > 0 ? &knots[index - 1] :
nullptr);
738 Ts_TypedKnotData<T>* knot = &knots[index];
739 Ts_TypedKnotData<T>* nextKnot = (index < knots.size() - 1
743 return knot->UpdateTangents(prevKnot, nextKnot, curveType);
747Ts_TypedSplineData<T>*
748Ts_GetTypedSplineData(
TsSpline &spline)
750 return static_cast<Ts_TypedSplineData<T>*
>(
751 Ts_GetSplineData(spline));
755const Ts_TypedSplineData<T>*
756Ts_GetTypedSplineData(
const TsSpline &spline)
758 return static_cast<Ts_TypedSplineData<T>*
>(
759 Ts_GetSplineData(spline));
763PXR_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.