/****************************************** * * * RenderWare(TM) Graphics Library * * * ******************************************/ /* * This file is a product of Criterion Software Ltd. * * This file is provided as is with no warranties of any kind and is * provided without any obligation on Criterion Software Ltd. * or Canon Inc. to assist in its use or modification. * * Criterion Software Ltd. and Canon Inc. will not, under any * circumstances, be liable for any lost revenue or other damages * arising from the use of this file. * * Copyright (c) 1998. Criterion Software Ltd. * All Rights Reserved. */ /*************************************************************************** * * * Module : rpanim.h * * * * Purpose : Hierarchical animation * * * **************************************************************************/ #ifndef RPHANIM_H #define RPHANIM_H /** * Hierarchal animation plugin */ /* Doxygen plugin groups. */ /** * \defgroup rphanim RpHAnim * \ingroup objectframehanim * * Hierarchical Animation Plugin for RenderWare Graphics. */ /** * \defgroup rphanimchanges RpHAnim Changes * \ingroup rphanim * */ /**************************************************************************** Includes */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <rwcore.h> #include <rpworld.h> #include <rpcriter.h> /* Note: each vendor can choose their own method for * allocation of unique ID's. This file defines * the ID's used by Criterion. */ #include <rphanim.rpe> /* automatically generated header file */ #include <rtquat.h> #include <rtanim.h> #define rpHANIMSTREAMCURRENTVERSION 0x100 #if (!defined(DOXYGEN)) typedef struct RpHAnimAtomicGlobalVars RpHAnimAtomicGlobalVars; struct RpHAnimAtomicGlobalVars { RwInt32 engineOffset; RwFreeList *HAnimFreeList; }; #endif /* (!defined(DOXYGEN)) */ #define rpHANIMSTDKEYFRAMESIZE sizeof(RpHAnimKeyFrame) #define rpHANIMSTDKEYFRAMETYPEID 0x1 #define RpV3dInterpolate(o,a,s,b) \ MACRO_START \ { \ (o)->x = (((a)->x) + ((s)) * (((b)->x) - ((a)->x))); \ (o)->y = (((a)->y) + ((s)) * (((b)->y) - ((a)->y))); \ (o)->z = (((a)->z) + ((s)) * (((b)->z) - ((a)->z))); \ } \ MACRO_STOP /** * \ingroup rphanim * \ref RpHAnimKeyFrame * typedef for struct RpHAnimKeyFrame. Based on \ref RtAnimKeyFrameHeader. */ typedef struct RpHAnimKeyFrame RpHAnimKeyFrame; /** * \ingroup rphanim * \struct RpHAnimKeyFrame * A structure representing the standard keyframe data. Sequences of * such keyframes in an \ref RtAnimAnimation defines the animation of each * node in a hierarchy. */ struct RpHAnimKeyFrame { RpHAnimKeyFrame *prevFrame; /**< Pointer to the previous keyframe */ RwReal time; /**< Time at keyframe */ RtQuat q; /**< Quaternion rotation at keyframe */ RwV3d t; /**< Translation at keyframe */ }; /** * \ingroup rphanim * \ref RpHAnimInterpFrame * typedef for struct RpHAnimInterpFrame. Based on \ref RtAnimInterpFrameHeader. */ typedef struct RpHAnimInterpFrame RpHAnimInterpFrame; /** * \ingroup rphanim * \struct RpHAnimInterpFrame * A structure representing an interpolated keyframe. The initial part of the * structure matches \ref RtAnimInterpFrameHeader. */ struct RpHAnimInterpFrame { RpHAnimKeyFrame *keyFrame1; /**< Pointer to 1st keyframe of current interpolation pair */ RpHAnimKeyFrame *keyFrame2; /**< Pointer to 2nd keyframe of current interpolation pair */ RtQuat q; /**< Quaternion rotation */ RwV3d t; /**< Translation */ }; /** * \ingroup rphanim * \ref RpHAnimHierarchy typedef for struct RpHAnimHierarchy */ typedef struct RpHAnimHierarchy RpHAnimHierarchy; /* Flags for FrameInfos */ #define rpHANIMPOPPARENTMATRIX 0x01 #define rpHANIMPUSHPARENTMATRIX 0x02 /** * \ingroup rphanim * \ref RpHAnimNodeInfo * typedef for struct RpHAnimNodeInfo */ typedef struct RpHAnimNodeInfo RpHAnimNodeInfo; /** * \ingroup rphanim * \struct RpHAnimNodeInfo * * Used to describe a hierarchy toplogy. * It holds flags representing its position in the * hierarchy as well as a pointer to the matching \ref RwFrame if the * hierarchy has been attached to a \ref RwFrame hierarchy. * */ struct RpHAnimNodeInfo { RwInt32 nodeID; /**< User defined ID for this node */ RwInt32 nodeIndex; /**< Array index of node */ RwInt32 flags; /**< Matrix push/pop flags */ RwFrame * pFrame; /**< Pointer to an attached RwFrame (see \ref RpHAnimHierarchyAttach) */ }; /** * \ingroup rphanim * \ref RpHAnimHierarchyFlag defines type and update modes in HAnimHierarchies */ enum RpHAnimHierarchyFlag { /* creation flags */ rpHANIMHIERARCHYSUBHIERARCHY = 0x01, /**< This hierarchy is a sub-hierarchy */ rpHANIMHIERARCHYNOMATRICES = 0x02, /**< This hierarchy has no local matrices */ /* update flags */ rpHANIMHIERARCHYUPDATEMODELLINGMATRICES = 0x1000, /**< This hierarchy updates modeling matrices */ rpHANIMHIERARCHYUPDATELTMS = 0x2000, /**< This hierarchy updates LTMs */ rpHANIMHIERARCHYLOCALSPACEMATRICES = 0x4000, /**< This hierarchy calculates matrices in a space relative to its root */ rpHANIMHIERARCHYFLAGFORCEENUMSIZEINT = RWFORCEENUMSIZEINT }; /* * These flags are used to control the creation and * update status of the hierarchy */ typedef enum RpHAnimHierarchyFlag RpHAnimHierarchyFlag; /** * \ingroup rphanim * \struct RpHAnimHierarchy * An RpHAnimHierarchy is used to "play back" an animation. * * The structure of a hierarchy is defined by an array * of \ref RpHAnimNodeInfo structures. * * The hierarchy is defined by running through the node array in order, * pushing the parent-node's matrix whenever a child is reached that has * more than one sibling, and popping the parent matrix when a "leaf" * node is encountered. * */ struct RpHAnimHierarchy { RwInt32 flags; /**< Flags for the hierarchy */ RwInt32 numNodes; /**< Number of nodes in the hierarchy */ RwMatrix *pMatrixArray; /**< Pointer to node matrices*/ void *pMatrixArrayUnaligned; /**< Pointer to memory used for node matrices * from which the aligned pMatrixArray is allocated */ RpHAnimNodeInfo *pNodeInfo; /**< Array of node information (push/pop flags etc) */ RwFrame *parentFrame; /**< Pointer to the Root RwFrame of the hierarchy this * RpHAnimHierarchy represents */ RpHAnimHierarchy *parentHierarchy; /**< Internal use */ RwInt32 rootParentOffset; /**< Internal use */ RtAnimInterpolator *currentAnim; /**< Internal use */ }; /** * \ingroup rphanim * \ref RpHAnimFrameExtension typedef for struct RpHAnimFrameExtension */ typedef struct RpHAnimFrameExtension RpHAnimFrameExtension; /** * \ingroup rphanim * \struct RpHAnimFrameExtension * * Used to extend \ref RwFrame objects, and thus * allow the mapping between animation hierarchy node ID and \ref RwFrame. * */ struct RpHAnimFrameExtension { RwInt32 id; /**< ID given to this RwFrame (default of -1) */ RpHAnimHierarchy *hierarchy; /**< Pointer to Animation hierarchy attached to this RwFrame */ }; /*--- Plugin API Functions ---*/ #define RpHAnimHierarchySetFlagsMacro(hierarchy,_flags) \ MACRO_START \ { \ (hierarchy)->flags = _flags; \ } \ MACRO_STOP #define RpHAnimHierarchyGetFlagsMacro(hierarchy) \ ((hierarchy)->flags) #define RpHAnimKeyFrameToMatrixMacro(_matrix,_voidIFrame) \ MACRO_START \ { \ RpHAnimInterpFrame * iFrame = (RpHAnimInterpFrame *)(_voidIFrame); \ \ /* \ * RpHAnim uses the same types of quaternion as RtQuat \ * hence no conjugate call as in RpSkin \ */ \ \ RtQuatUnitConvertToMatrix(&iFrame->q,(_matrix)); \ \ (_matrix)->pos.x = iFrame->t.x; \ (_matrix)->pos.y = iFrame->t.y; \ (_matrix)->pos.z = iFrame->t.z; \ } \ MACRO_STOP #if (! defined(RWDEBUG)) #define RpHAnimHierarchySetFlags(hierarchy,_flags) \ RpHAnimHierarchySetFlagsMacro(hierarchy,_flags) #define RpHAnimHierarchyGetFlags(hierarchy) \ (RpHAnimHierarchyFlag)RpHAnimHierarchyGetFlagsMacro(hierarchy) #endif /* (! defined(RWDEBUG)) */ #ifdef __cplusplus extern "C" { #endif /* __cplusplus */ extern RpHAnimAtomicGlobalVars RpHAnimAtomicGlobals; #if (defined(RWDEBUG)) extern RpHAnimHierarchy * RpHAnimHierarchySetFlags(RpHAnimHierarchy *hierarchy, RpHAnimHierarchyFlag flags); extern RpHAnimHierarchyFlag RpHAnimHierarchyGetFlags(RpHAnimHierarchy *hierarchy); #endif /* (defined(RWDEBUG)) */ /* Animation hierarchy creation */ extern void RpHAnimHierarchySetFreeListCreateParams(RwInt32 blockSize,RwInt32 numBlocksToPrealloc); extern RpHAnimHierarchy * RpHAnimHierarchyCreate(RwInt32 numNodes, RwUInt32 *nodeFlags, RwInt32 *nodeIDs, RpHAnimHierarchyFlag flags, RwInt32 maxKeyFrameSize); extern RpHAnimHierarchy * RpHAnimHierarchyCreateFromHierarchy(RpHAnimHierarchy *hierarchy, RpHAnimHierarchyFlag flags, RwInt32 maxKeyFrameSize); extern RpHAnimHierarchy * RpHAnimHierarchyDestroy(RpHAnimHierarchy *hierarchy); extern RpHAnimHierarchy * RpHAnimHierarchyCreateSubHierarchy(RpHAnimHierarchy *parentHierarchy, RwInt32 startNode, RpHAnimHierarchyFlag flags, RwInt32 maxKeyFrameSize); extern RpHAnimHierarchy * RpHAnimHierarchyAttach(RpHAnimHierarchy *hierarchy); extern RpHAnimHierarchy * RpHAnimHierarchyDetach(RpHAnimHierarchy *hierarchy); extern RpHAnimHierarchy * RpHAnimHierarchyAttachFrameIndex(RpHAnimHierarchy *hierarchy, RwInt32 nodeIndex); extern RpHAnimHierarchy * RpHAnimHierarchyDetachFrameIndex(RpHAnimHierarchy *hierarchy, RwInt32 nodeIndex); extern RwBool RpHAnimFrameSetHierarchy(RwFrame *frame, RpHAnimHierarchy *hierarchy); extern RpHAnimHierarchy * RpHAnimFrameGetHierarchy(RwFrame *frame); /* Macros for legacy support of old function names */ #define RpHAnimSetHierarchy(frame,hierarchy) \ RpHAnimFrameSetHierarchy(frame,hierarchy) #define RpHAnimGetHierarchy(frame) RpHAnimFrameGetHierarchy(frame) extern RwMatrix * RpHAnimHierarchyGetMatrixArray(RpHAnimHierarchy *hierarchy); extern RwBool RpHAnimHierarchyUpdateMatrices(RpHAnimHierarchy *hierarchy); /* Macro for legacy support of old function name */ #define RpHAnimUpdateHierarchyMatrices RpHAnimHierarchyUpdateMatrices extern RwInt32 RpHAnimIDGetIndex(RpHAnimHierarchy *hierarchy, RwInt32 ID); /* Plugin support */ extern RwBool RpHAnimPluginAttach(void); /* Hanim keyframe functions */ extern void RpHAnimKeyFrameApply(void *matrix, void *voidIFrame); extern void RpHAnimKeyFrameBlend(void *voidOut, void *voidIn1, void *voidIn2, RwReal alpha); extern void RpHAnimKeyFrameInterpolate(void *voidOut, void *voidIn1, void *voidIn2, RwReal time); extern void RpHAnimKeyFrameAdd(void *voidOut, void *voidIn1, void *voidIn2); extern void RpHAnimKeyFrameMulRecip(void *voidFrame, void *voidStart); extern RtAnimAnimation * RpHAnimKeyFrameStreamRead(RwStream *stream, RtAnimAnimation *animation); extern RwBool RpHAnimKeyFrameStreamWrite(RtAnimAnimation *animation, RwStream *stream); extern RwInt32 RpHAnimKeyFrameStreamGetSize(RtAnimAnimation *animation); /* Access to RwFrame ID's */ extern RwBool RpHAnimFrameSetID(RwFrame *frame, RwInt32 id); extern RwInt32 RpHAnimFrameGetID(RwFrame *frame); /* * Utility Functions */ #define RpHAnimHierarchySetCurrentAnimMacro(hierarchy,anim)\ RtAnimInterpolatorSetCurrentAnim((hierarchy)->currentAnim,anim) #define RpHAnimHierarchyGetCurrentAnimMacro(hierarchy)\ RtAnimInterpolatorGetCurrentAnim((hierarchy)->currentAnim) #define RpHAnimHierarchySetCurrentAnimTimeMacro(hierarchy,time)\ RtAnimInterpolatorSetCurrentTime((hierarchy)->currentAnim,time) #define RpHAnimHierarchyAddAnimTimeMacro(hierarchy,time)\ RtAnimInterpolatorAddAnimTime((hierarchy)->currentAnim,time) #define RpHAnimHierarchySubAnimTimeMacro(hierarchy,time)\ RtAnimInterpolatorSubAnimTime((hierarchy)->currentAnim,time) #define RpHAnimHierarchySetKeyFrameCallBacksMacro(hierarchy,keyFrameTypeID) \ RtAnimInterpolatorSetKeyFrameCallBacks((hierarchy)->currentAnim,\ keyFrameTypeID) #define RpHAnimHierarchyBlendMacro(outHierarchy,inHierarchy1,inHierarchy2,alpha)\ RtAnimInterpolatorBlend((outHierarchy)->currentAnim,\ (inHierarchy1)->currentAnim,\ (inHierarchy2)->currentAnim,\ alpha) #define RpHAnimHierarchyAddTogetherMacro(outHierarchy,inHierarchy1,inHierarchy2)\ RtAnimInterpolatorAddTogether((outHierarchy)->currentAnim,\ (inHierarchy1)->currentAnim,\ (inHierarchy2)->currentAnim) #define RpHAnimHierarchySetAnimCallBackMacro(hierarchy,callBack,time,data)\ RtAnimInterpolatorSetAnimCallBack((hierarchy)->currentAnim,callBack,time,data) #define RpHAnimHierarchySetAnimLoopCallBackMacro(hierarchy,callBack,data)\ RtAnimInterpolatorSetAnimLoopCallBack((hierarchy)->currentAnim,callBack,data) #define RpHAnimHierarchyBlendSubHierarchyMacro(outHierarchy,inHierarchy1,inHierarchy2,alpha)\ RtAnimInterpolatorBlendSubInterpolator((outHierarchy)->currentAnim,(inHierarchy1)->currentAnim,(inHierarchy2)->currentAnim,alpha) #define RpHAnimHierarchyAddSubHierarchyMacro(outHierarchy,mainHierarchy,subHierarchy)\ RtAnimInterpolatorAddSubInterpolator((outHierarchy)->currentAnim,(mainHierarchy)->currentAnim,(subHierarchy)->currentAnim) #define RpHAnimHierarchyCopyMacro(outHierarchy,inHierarchy)\ RtAnimInterpolatorCopy((outHierarchy)->currentAnim,(inHierarchy)->currentAnim) #ifdef RWDEBUG extern RwBool RpHAnimHierarchySetCurrentAnim(RpHAnimHierarchy *hierarchy, RtAnimAnimation *anim); extern RtAnimAnimation * RpHAnimHierarchyGetCurrentAnim(RpHAnimHierarchy *hierarchy); extern RwBool RpHAnimHierarchySetCurrentAnimTime(RpHAnimHierarchy *hierarchy, RwReal time); extern RwBool RpHAnimHierarchyAddAnimTime(RpHAnimHierarchy *hierarchy, RwReal time); extern RwBool RpHAnimHierarchySubAnimTime(RpHAnimHierarchy *hierarchy, RwReal time); extern RwBool RpHAnimHierarchySetKeyFrameCallBacks(RpHAnimHierarchy *hierarchy, RwInt32 keyFrameTypeID); extern void RpHAnimHierarchySetAnimCallBack(RpHAnimHierarchy *hierarchy, RtAnimCallBack callBack, RwReal time, void *data); extern RwBool RpHAnimHierarchyBlend(RpHAnimHierarchy *outHierarchy, RpHAnimHierarchy *inHierarchy1, RpHAnimHierarchy *inHierarchy2, RwReal alpha); extern RwBool RpHAnimHierarchyAddTogether(RpHAnimHierarchy *outHierarchy, RpHAnimHierarchy *inHierarchy1, RpHAnimHierarchy *inHierarchy2); extern void RpHAnimHierarchySetAnimLoopCallBack(RpHAnimHierarchy *hierarchy, RtAnimCallBack callBack, void *data); extern RwBool RpHAnimHierarchyBlendSubHierarchy(RpHAnimHierarchy *outHierarchy, RpHAnimHierarchy *inHierarchy1, RpHAnimHierarchy *inHierarchy2, RwReal alpha); extern RwBool RpHAnimHierarchyAddSubHierarchy(RpHAnimHierarchy *outHierarchy, RpHAnimHierarchy *mainHierarchy1, RpHAnimHierarchy *subHierarchy2); extern RwBool RpHAnimHierarchyCopy(RpHAnimHierarchy *outHierarchy, RpHAnimHierarchy *inHierarchy); #else #define RpHAnimHierarchySetCurrentAnim(hierarchy,anim) \ RpHAnimHierarchySetCurrentAnimMacro((hierarchy),(anim)) #define RpHAnimHierarchyGetCurrentAnim(hierarchy) \ RpHAnimHierarchyGetCurrentAnimMacro((hierarchy)) #define RpHAnimHierarchySetCurrentAnimTime(hierarchy,time) \ RpHAnimHierarchySetCurrentAnimTimeMacro((hierarchy),(time)) #define RpHAnimHierarchyAddAnimTime(hierarchy,time) \ RpHAnimHierarchyAddAnimTimeMacro((hierarchy),(time)) #define RpHAnimHierarchySubAnimTime(hierarchy,time) \ RpHAnimHierarchySubAnimTimeMacro((hierarchy),(time)) #define RpHAnimHierarchySetKeyFrameCallBacks(hierarchy,keyFrameTypeID) \ RpHAnimHierarchySetKeyFrameCallBacksMacro((hierarchy),(keyFrameTypeID)) #define RpHAnimHierarchyBlend(outHierarchy,inHierarchy1,inHierarchy2,alpha) \ RpHAnimHierarchyBlendMacro((outHierarchy),(inHierarchy1),(inHierarchy2),(alpha)) #define RpHAnimHierarchyAddTogether(outHierarchy,inHierarchy1,inHierarchy2) \ RpHAnimHierarchyAddTogetherMacro((outHierarchy),(inHierarchy1),(inHierarchy2)) #define RpHAnimHierarchySetAnimCallBack(hierarchy,callBack,time,data)\ RpHAnimHierarchySetAnimCallBackMacro((hierarchy),(callBack),(time),(data)) #define RpHAnimHierarchySetAnimLoopCallBack(hierarchy,callBack,data)\ RpHAnimHierarchySetAnimLoopCallBackMacro((hierarchy),(callBack),(data)) #define RpHAnimHierarchyBlendSubHierarchy(outHierarchy,inHierarchy1,inHierarchy2,alpha)\ RpHAnimHierarchyBlendSubHierarchyMacro((outHierarchy),(inHierarchy1),(inHierarchy2),(alpha)) #define RpHAnimHierarchyAddSubHierarchy(outHierarchy,mainHierarchy,subHierarchy)\ RpHAnimHierarchyAddSubHierarchyMacro((outHierarchy),(mainHierarchy),(subHierarchy)) #define RpHAnimHierarchyCopy(outHierarchy,inHierarchy)\ RpHAnimHierarchyCopyMacro((outHierarchy),(inHierarchy)) #endif /* RWDEBUG */ #ifdef __cplusplus } #endif /* __cplusplus */ /* Legacy TypeDef */ typedef RtAnimAnimation RpHAnimAnimation; typedef RpHAnimKeyFrame RpHAnimStdKeyFrame; /* Legacy Macros */ /* Animations */ #define RpHAnimAnimationCreate(typeID,numFrames,flags,duration)\ RtAnimAnimationCreate((typeID),(numFrames),(flags),(duration)) #define RpHAnimAnimationDestroy(animation)\ RtAnimAnimationDestroy((animation)) #define RpHAnimAnimationGetTypeID(animation)\ RtAnimAnimationGetTypeID((animation)) #define RpHAnimAnimationRead(filename)\ RtAnimAnimationRead((filename)) #define RpHAnimAnimationWrite(animation,filename)\ RtAnimAnimationWrite((animation),(filename)) #define RpHAnimAnimationStreamRead(stream)\ RtAnimAnimationStreamRead((stream)) #define RpHAnimAnimationStreamWrite(animation,stream)\ RtAnimAnimationStreamWrite((animation),(stream)) #define RpHAnimAnimationStreamGetSize(animation)\ RtAnimAnimationStreamGetSize((animation)) #define RpHAnimAnimationMakeDelta(animation,numNodes,time)\ RtAnimAnimationMakeDelta((animation),(numNodes),(time)) /* Animation Interpolator */ #define RpHAnimHierarchyStdKeyFrameAddAnimTime(hierarchy,time)\ RpHAnimHierarchyHAnimKeyFrameAddAnimTime((hierarchy),(time)) #define RpHAnimHierarchyHAnimKeyFrameAddAnimTime(hierarchy,time)\ RpHAnimHierarchyAddAnimTime((hierarchy),(time)) #endif /* RPHANIM_H */