#ifndef OPENMW_COMPONENTS_NIF_CONTROLLER_HPP #define OPENMW_COMPONENTS_NIF_CONTROLLER_HPP #include "base.hpp" #include "niftypes.hpp" #include "property.hpp" namespace Nif { struct ControlledBlock { std::string mTargetName; NiInterpolatorPtr mInterpolator; NiTimeControllerPtr mController; NiBlendInterpolatorPtr mBlendInterpolator; uint16_t mBlendIndex; uint8_t mPriority; NiStringPalettePtr mStringPalette; uint32_t mNodeNameOffset; uint32_t mPropertyTypeOffset; uint32_t mControllerTypeOffset; uint32_t mControllerIdOffset; uint32_t mInterpolatorIdOffset; std::string mNodeName; std::string mPropertyType; std::string mControllerType; std::string mControllerId; std::string mInterpolatorId; void read(NIFStream* nif); void post(Reader& nif); }; // Gamebryo KF root node record type (pre-10.0) struct NiSequence : public Record { std::string mName; std::string mAccumRootName; ExtraPtr mTextKeys; uint32_t mArrayGrowBy; std::vector mControlledBlocks; void read(NIFStream* nif) override; void post(Reader& nif) override; }; // Gamebryo KF root node record type (10.0+) struct NiControllerSequence : public NiSequence { float mWeight{ 1.f }; NiTimeController::ExtrapolationMode mExtrapolationMode{ NiTimeController::ExtrapolationMode::Constant }; float mFrequency{ 1.f }; float mPhase{ 1.f }; float mStartTime, mStopTime; bool mPlayBackwards{ false }; NiControllerManagerPtr mManager; NiStringPalettePtr mStringPalette; void read(NIFStream* nif) override; void post(Reader& nif) override; }; // Base class for controllers that use NiInterpolators to animate objects. struct NiInterpController : public NiTimeController { // Usually one of the flags. bool mManagerControlled{ false }; void read(NIFStream* nif) override; }; // Base class for controllers that use one NiInterpolator. struct NiSingleInterpController : public NiInterpController { NiInterpolatorPtr mInterpolator; void read(NIFStream* nif) override; void post(Reader& nif) override; }; // Base class for controllers that use a NiFloatInterpolator to animate their target. struct NiFloatInterpController : public NiSingleInterpController { }; // Ditto for NiBoolInterpolator. struct NiBoolInterpController : public NiSingleInterpController { }; // Ditto for NiPoint3Interpolator. struct NiPoint3InterpController : public NiSingleInterpController { }; struct NiParticleInfo { osg::Vec3f mVelocity; osg::Vec3f mRotationAxis; float mAge; float mLifespan; float mLastUpdate; uint16_t mSpawnGeneration; uint16_t mCode; void read(NIFStream* nif); }; struct NiParticleSystemController : public NiTimeController { enum BSPArrayController { BSPArrayController_AtNode = 0x8, BSPArrayController_AtVertex = 0x10 }; enum EmitFlags { EmitFlag_NoAutoAdjust = 0x1 // If this flag is set, we use the emitRate value. Otherwise, // we calculate an emit rate so that the maximum number of particles // in the system (numParticles) is never exceeded. }; float mSpeed; float mSpeedVariation; float mDeclination; float mDeclinationVariation; float mPlanarAngle; float mPlanarAngleVariation; osg::Vec3f mInitialNormal; osg::Vec4f mInitialColor; float mInitialSize; float mEmitStartTime; float mEmitStopTime; bool mResetParticleSystem{ false }; float mBirthRate; float mLifetime; float mLifetimeVariation; uint16_t mEmitFlags{ 0 }; osg::Vec3f mEmitterDimensions; NiAVObjectPtr mEmitter; uint16_t mNumSpawnGenerations; float mPercentageSpawned; uint16_t mSpawnMultiplier; float mSpawnSpeedChaos; float mSpawnDirChaos; uint16_t mNumParticles; uint16_t mNumValid; std::vector mParticles; NiParticleModifierPtr mModifier; NiParticleModifierPtr mCollider; uint8_t mStaticTargetBound; void read(NIFStream* nif) override; void post(Reader& nif) override; bool noAutoAdjust() const { return mEmitFlags & EmitFlag_NoAutoAdjust; } bool emitAtVertex() const { return mFlags & BSPArrayController_AtVertex; } }; using NiBSPArrayController = NiParticleSystemController; struct NiMaterialColorController : public NiPoint3InterpController { enum class TargetColor { Ambient = 0, Diffuse = 1, Specular = 2, Emissive = 3, }; NiPosDataPtr mData; TargetColor mTargetColor; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiPathController : public NiTimeController { enum Flags { Flag_CVDataNeedsUpdate = 0x01, Flag_OpenCurve = 0x02, Flag_AllowFlip = 0x04, Flag_Bank = 0x08, Flag_ConstVelocity = 0x10, Flag_Follow = 0x20, Flag_FlipFollowAxis = 0x40, }; uint16_t mPathFlags; int32_t mBankDirection; float mMaxBankAngle; float mSmoothing; uint16_t mFollowAxis; NiPosDataPtr mPathData; NiFloatDataPtr mPercentData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiLookAtController : public NiTimeController { enum Flags { Flag_Flip = 0x1, Flag_LookYAxis = 0x2, Flag_LookZAxis = 0x4, }; uint16_t mLookAtFlags{ 0 }; NiAVObjectPtr mLookAt; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiUVController : public NiTimeController { NiUVDataPtr mData; uint16_t mUvSet; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiKeyframeController : public NiSingleInterpController { NiKeyframeDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiMultiTargetTransformController : public NiInterpController { NiAVObjectList mExtraTargets; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiAlphaController : public NiFloatInterpController { NiFloatDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiRollController : public NiSingleInterpController { NiFloatDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiGeomMorpherController : public NiInterpController { bool mUpdateNormals{ false }; bool mAlwaysActive{ false }; NiMorphDataPtr mData; NiInterpolatorList mInterpolators; std::vector mWeights; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiVisController : public NiBoolInterpController { NiVisDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiFlipController : public NiFloatInterpController { NiTexturingProperty::TextureType mTexSlot; float mDelta; // Time between two flips. delta = (start_time - stop_time) / num_sources NiSourceTextureList mSources; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiTextureTransformController : public NiFloatInterpController { bool mShaderMap; NiTexturingProperty::TextureType mTexSlot; uint32_t mTransformMember; NiFloatDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiBoneLODController : NiTimeController { struct SkinInfo { NiTriBasedGeomPtr mShape; NiSkinInstancePtr mSkin; }; uint32_t mLOD; uint32_t mNumNodeGroups; std::vector mNodeGroups; std::vector> mSkinnedShapeGroups; NiTriBasedGeomList mShapeGroups; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct bhkBlendController : public NiTimeController { void read(NIFStream* nif) override; }; struct BSEffectShaderPropertyFloatController : public NiFloatInterpController { uint32_t mControlledVariable; void read(NIFStream* nif) override; }; struct BSEffectShaderPropertyColorController : public NiPoint3InterpController { uint32_t mControlledColor; void read(NIFStream* nif) override; }; struct BSKeyframeController : NiKeyframeController { NiKeyframeDataPtr mData2; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct BSLagBoneController : NiTimeController { float mLinearVelocity; float mLinearRotation; float mMaximumDistance; void read(NIFStream* nif) override; }; struct BSProceduralLightningController : NiTimeController { NiInterpolatorPtr mGenerationInterp; NiInterpolatorPtr mMutationInterp; NiInterpolatorPtr mSubdivisionInterp; NiInterpolatorPtr mNumBranchesInterp; NiInterpolatorPtr mNumBranchesVarInterp; NiInterpolatorPtr mLengthInterp; NiInterpolatorPtr mLengthVarInterp; NiInterpolatorPtr mWidthInterp; NiInterpolatorPtr mArcOffsetInterp; uint16_t mSubdivisions; uint16_t mNumBranches; uint16_t mNumBranchesVar; float mLength; float mLengthVar; float mWidth; float mChildWidthMult; float mArcOffset; bool mFadeMainBolt; bool mFadeChildBolts; bool mAnimateArcOffset; BSShaderPropertyPtr mShaderProperty; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiControllerManager : public NiTimeController { bool mCumulative; NiControllerSequenceList mSequences; NiDefaultAVObjectPalettePtr mObjectPalette; void read(NIFStream* nif) override; void post(Reader& nif) override; }; // Abstract struct NiExtraDataController : NiSingleInterpController { std::string mExtraDataName; void read(NIFStream* nif) override; }; struct NiFloatExtraDataController : NiExtraDataController { NiFloatDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiFloatsExtraDataController : NiExtraDataController { int32_t mFloatsExtraDataIndex; NiFloatDataPtr mData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiFloatsExtraDataPoint3Controller : NiExtraDataController { int32_t mFloatsExtraDataIndex; void read(NIFStream* nif) override; }; // Abstract struct NiInterpolator : public Record { }; template struct TypedNiInterpolator : public NiInterpolator { T mDefaultValue; DataPtr mData; void read(NIFStream* nif) override { nif->read(mDefaultValue); mData.read(nif); } void post(Reader& nif) override { mData.post(nif); } }; using NiPoint3Interpolator = TypedNiInterpolator; using NiBoolInterpolator = TypedNiInterpolator; using NiFloatInterpolator = TypedNiInterpolator; using NiTransformInterpolator = TypedNiInterpolator; using NiColorInterpolator = TypedNiInterpolator; struct NiPathInterpolator : public NiInterpolator { // Uses the same flags as NiPathController uint16_t mFlags; int32_t mBankDirection; float mMaxBankAngle; float mSmoothing; uint16_t mFollowAxis; NiPosDataPtr mPathData; NiFloatDataPtr mPercentData; void read(NIFStream* nif) override; void post(Reader& nif) override; }; struct NiLookAtInterpolator : NiInterpolator { // Uses the same flags as NiLookAtController uint16_t mLookAtFlags{ 0 }; NiAVObjectPtr mLookAt; std::string mLookAtName; NiQuatTransform mTransform; NiInterpolatorPtr mTranslation; NiInterpolatorPtr mRoll; NiInterpolatorPtr mScale; void read(NIFStream* nif) override; void post(Reader& nif) override; }; // Abstract struct NiBlendInterpolator : public NiInterpolator { enum Flags { Flag_ManagerControlled = 0x1, Flag_OnlyUseHighestWeight = 0x2, }; struct Item { NiInterpolatorPtr mInterpolator; float mWeight, mNormalizedWeight; int32_t mPriority; float mEaseSpinner; void read(NIFStream* nif); void post(Reader& nif); }; uint8_t mFlags{ 0 }; uint16_t mArrayGrowBy{ 0 }; float mWeightThreshold; uint16_t mInterpCount; uint16_t mSingleIndex; int32_t mHighPriority, mNextHighPriority; float mSingleTime; float mHighWeightsSum, mNextHighWeightsSum; float mHighEaseSpinner; std::vector mItems; NiInterpolatorPtr mSingleInterpolator; void read(NIFStream* nif) override; void post(Reader& nif) override; }; template struct TypedNiBlendInterpolator : public NiBlendInterpolator { T mValue; void read(NIFStream* nif) override { NiBlendInterpolator::read(nif); nif->read(mValue); } }; template <> struct TypedNiBlendInterpolator : public NiBlendInterpolator { NiQuatTransform mValue; void read(NIFStream* nif) override { NiBlendInterpolator::read(nif); if (nif->getVersion() <= NIFStream::generateVersion(10, 1, 0, 109)) nif->read(mValue); } }; using NiBlendBoolInterpolator = TypedNiBlendInterpolator; using NiBlendFloatInterpolator = TypedNiBlendInterpolator; using NiBlendPoint3Interpolator = TypedNiBlendInterpolator; using NiBlendTransformInterpolator = TypedNiBlendInterpolator; } #endif