From 35de34c01927dc74d13ea9e9bdcd34e6929677a9 Mon Sep 17 00:00:00 2001 From: Capostrophic Date: Fri, 7 Aug 2020 10:07:19 +0300 Subject: [PATCH 1/2] Don't clamp GeomMorpherController recovered weight value Seems that Morrowind doesn't do it. --- components/nifosg/controller.cpp | 1 - 1 file changed, 1 deletion(-) diff --git a/components/nifosg/controller.cpp b/components/nifosg/controller.cpp index a4db2cba3e..58f1c17a74 100644 --- a/components/nifosg/controller.cpp +++ b/components/nifosg/controller.cpp @@ -197,7 +197,6 @@ void GeomMorpherController::update(osg::NodeVisitor *nv, osg::Drawable *drawable float val = 0; if (!(*it).empty()) val = it->interpKey(input); - val = std::max(0.f, std::min(1.f, val)); SceneUtil::MorphGeometry::MorphTarget& target = morphGeom->getMorphTarget(i); if (target.getWeight() != val) From aa131262ea7efadf0cde9258b913d4c2ae95bc20 Mon Sep 17 00:00:00 2001 From: Capostrophic Date: Fri, 7 Aug 2020 10:22:54 +0300 Subject: [PATCH 2/2] Implement quadratic interpolation for scalars and vectors --- components/nif/nifkey.hpp | 10 +++++----- components/nifosg/controller.hpp | 19 +++++++++++++++++++ 2 files changed, 24 insertions(+), 5 deletions(-) diff --git a/components/nif/nifkey.hpp b/components/nif/nifkey.hpp index 333d8a7cfa..4c10327e13 100644 --- a/components/nif/nifkey.hpp +++ b/components/nif/nifkey.hpp @@ -26,11 +26,11 @@ enum InterpolationType template struct KeyT { T mValue; + T mInTan; // Only for Quadratic interpolation, and never for QuaternionKeyList + T mOutTan; // Only for Quadratic interpolation, and never for QuaternionKeyList - // FIXME: Implement Quadratic and TBC interpolation + // FIXME: Implement TBC interpolation /* - T mForwardValue; // Only for Quadratic interpolation, and never for QuaternionKeyList - T mBackwardValue; // Only for Quadratic interpolation, and never for QuaternionKeyList float mTension; // Only for TBC interpolation float mBias; // Only for TBC interpolation float mContinuity; // Only for TBC interpolation @@ -136,8 +136,8 @@ private: static void readQuadratic(NIFStream &nif, KeyT &key) { readValue(nif, key); - /*key.mForwardValue = */(nif.*getValue)(); - /*key.mBackwardValue = */(nif.*getValue)(); + key.mInTan = (nif.*getValue)(); + key.mOutTan = (nif.*getValue)(); } static void readQuadratic(NIFStream &nif, KeyT &key) diff --git a/components/nifosg/controller.hpp b/components/nifosg/controller.hpp index df1086f56f..be1292359f 100644 --- a/components/nifosg/controller.hpp +++ b/components/nifosg/controller.hpp @@ -110,6 +110,24 @@ namespace NifOsg { case Nif::InterpolationType_Constant: return fraction > 0.5f ? b.mValue : a.mValue; + case Nif::InterpolationType_Quadratic: + { + // Using a cubic Hermite spline. + // b1(t) = 2t^3 - 3t^2 + 1 + // b2(t) = -2t^3 + 3t^2 + // b3(t) = t^3 - 2t^2 + t + // b4(t) = t^3 - t^2 + // f(t) = a.mValue * b1(t) + b.mValue * b2(t) + a.mOutTan * b3(t) + b.mInTan * b4(t) + const float t = fraction; + const float t2 = t * t; + const float t3 = t2 * t; + const float b1 = 2.f * t3 - 3.f * t2 + 1; + const float b2 = -2.f * t3 + 3.f * t2; + const float b3 = t3 - 2.f * t2 + t; + const float b4 = t3 - t2; + return a.mValue * b1 + b.mValue * b2 + a.mOutTan * b3 + b.mInTan * b4; + } + // TODO: Implement TBC interpolation default: return a.mValue + ((b.mValue - a.mValue) * fraction); } @@ -120,6 +138,7 @@ namespace NifOsg { case Nif::InterpolationType_Constant: return fraction > 0.5f ? b.mValue : a.mValue; + // TODO: Implement Quadratic and TBC interpolation default: { osg::Quat result;