RpAnimBlend and frame updates

This commit is contained in:
aap 2020-05-10 23:47:53 +02:00
parent 9e842f1628
commit 198b80f560
8 changed files with 120 additions and 54 deletions

View file

@ -7,7 +7,7 @@
#include "AnimBlendAssociation.h" #include "AnimBlendAssociation.h"
#include "RwHelper.h" #include "RwHelper.h"
//--MIAMI: file done except for one TODO //--MIAMI: file done
CAnimBlendAssociation::CAnimBlendAssociation(void) CAnimBlendAssociation::CAnimBlendAssociation(void)
{ {
@ -160,6 +160,13 @@ CAnimBlendAssociation::Start(float time)
SetCurrentTime(time); SetCurrentTime(time);
} }
void
CAnimBlendAssociation::UpdateTimeStep(float timeDelta, float relSpeed)
{
if(IsRunning())
timeStep = (flags & ASSOC_MOVEMENT ? relSpeed*hierarchy->totalLength : speed) * timeDelta;
}
bool bool
CAnimBlendAssociation::UpdateTime(float timeDelta, float relSpeed) CAnimBlendAssociation::UpdateTime(float timeDelta, float relSpeed)
{ {
@ -170,10 +177,6 @@ CAnimBlendAssociation::UpdateTime(float timeDelta, float relSpeed)
return true; return true;
} }
// TODO(MIAMI): we still need this for some reason
#ifndef NOT_YET
timeStep = (flags & ASSOC_MOVEMENT ? relSpeed*hierarchy->totalLength : speed) * timeDelta;
#endif
currentTime += timeStep; currentTime += timeStep;
if(currentTime >= hierarchy->totalLength){ if(currentTime >= hierarchy->totalLength){

View file

@ -12,12 +12,12 @@ enum {
ASSOC_PARTIAL = 0x10, ASSOC_PARTIAL = 0x10,
ASSOC_MOVEMENT = 0x20, // ??? ASSOC_MOVEMENT = 0x20, // ???
ASSOC_HAS_TRANSLATION = 0x40, ASSOC_HAS_TRANSLATION = 0x40,
ASSOC_WALK = 0x80, // for CPed::PlayFootSteps(void) ASSOC_HAS_X_TRANSLATION = 0x80, // for 2d velocity extraction
ASSOC_FLAG_XPRESS = 0x100, // only used by xpress scratch, see CPed::Chat(void) ASSOC_WALK = 0x100, // for CPed::PlayFootSteps(void)
ASSOC_NOWALK = 0x200, // see CPed::PlayFootSteps(void) ASSOC_FLAG_XPRESS = 0x200, // only used by xpress scratch, see CPed::Chat(void)
ASSOC_BLOCK = 0x400, // unused in assoc description, blocks other anims from being played ASSOC_NOWALK = 0x400, // see CPed::PlayFootSteps(void)
ASSOC_FRONTAL = 0x800, // anims that we fall to front ASSOC_BLOCK = 0x800, // unused in assoc description, blocks other anims from being played
ASSOC_HAS_X_TRANSLATION = 0x1000, // for 2d velocity extraction ASSOC_FRONTAL = 0x1000, // anims that we fall to front
}; };
// Anim hierarchy associated with a clump // Anim hierarchy associated with a clump
@ -74,6 +74,7 @@ public:
void SetCurrentTime(float time); void SetCurrentTime(float time);
void SyncAnimation(CAnimBlendAssociation *other); void SyncAnimation(CAnimBlendAssociation *other);
void Start(float time); void Start(float time);
void UpdateTimeStep(float timeDelta, float relSpeed);
bool UpdateTime(float timeDelta, float relSpeed); bool UpdateTime(float timeDelta, float relSpeed);
bool UpdateBlend(float timeDelta); bool UpdateBlend(float timeDelta);

View file

@ -11,6 +11,7 @@ struct AnimBlendFrameData
IGNORE_TRANSLATION = 4, IGNORE_TRANSLATION = 4,
VELOCITY_EXTRACTION = 8, VELOCITY_EXTRACTION = 8,
VELOCITY_EXTRACTION_3D = 0x10, VELOCITY_EXTRACTION_3D = 0x10,
UPDATE_KEYFRAMES = 0x20,
}; };
uint8 flag; uint8 flag;

View file

@ -6,6 +6,8 @@
#include "AnimBlendAssociation.h" #include "AnimBlendAssociation.h"
#include "RpAnimBlend.h" #include "RpAnimBlend.h"
//--MIAMI: file done
CAnimBlendClumpData *gpAnimBlendClump; CAnimBlendClumpData *gpAnimBlendClump;
// PS2 names without "NonSkinned" // PS2 names without "NonSkinned"
@ -17,7 +19,6 @@ void FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg);
void FrameUpdateCallBackWithVelocityExtractionSkinned(AnimBlendFrameData *frame, void *arg); void FrameUpdateCallBackWithVelocityExtractionSkinned(AnimBlendFrameData *frame, void *arg);
void FrameUpdateCallBackWith3dVelocityExtractionSkinned(AnimBlendFrameData *frame, void *arg); void FrameUpdateCallBackWith3dVelocityExtractionSkinned(AnimBlendFrameData *frame, void *arg);
void void
FrameUpdateCallBackNonSkinned(AnimBlendFrameData *frame, void *arg) FrameUpdateCallBackNonSkinned(AnimBlendFrameData *frame, void *arg)
{ {
@ -47,6 +48,11 @@ FrameUpdateCallBackNonSkinned(AnimBlendFrameData *frame, void *arg)
(*node)->Update(vec, q, 1.0f-totalBlendAmount); (*node)->Update(vec, q, 1.0f-totalBlendAmount);
if((*node)->sequence->HasTranslation()) if((*node)->sequence->HasTranslation())
pos += vec; pos += vec;
#ifdef FIX_BUGS
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
#endif
rot += q; rot += q;
} }
++*node; ++*node;
@ -101,6 +107,11 @@ FrameUpdateCallBackWithVelocityExtractionNonSkinned(AnimBlendFrameData *frame, v
for(node = updateData->nodes; *node; node++){ for(node = updateData->nodes; *node; node++){
if((*node)->sequence){ if((*node)->sequence){
bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount); bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount);
#ifdef FIX_BUGS
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
#endif
rot += q; rot += q;
if((*node)->sequence->HasTranslation()){ if((*node)->sequence->HasTranslation()){
pos += vec; pos += vec;
@ -179,6 +190,11 @@ FrameUpdateCallBackWith3dVelocityExtractionNonSkinned(AnimBlendFrameData *frame,
for(node = updateData->nodes; *node; node++){ for(node = updateData->nodes; *node; node++){
if((*node)->sequence){ if((*node)->sequence){
bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount); bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount);
#ifdef FIX_BUGS
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
#endif
rot += q; rot += q;
if((*node)->sequence->HasTranslation()){ if((*node)->sequence->HasTranslation()){
pos += vec; pos += vec;
@ -212,8 +228,6 @@ FrameUpdateCallBackWith3dVelocityExtractionNonSkinned(AnimBlendFrameData *frame,
RwMatrixUpdate(mat); RwMatrixUpdate(mat);
} }
#ifdef PED_SKIN
void void
FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg) FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg)
{ {
@ -243,6 +257,9 @@ FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg)
(*node)->Update(vec, q, 1.0f-totalBlendAmount); (*node)->Update(vec, q, 1.0f-totalBlendAmount);
if((*node)->sequence->HasTranslation()) if((*node)->sequence->HasTranslation())
pos += vec; pos += vec;
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
rot += q; rot += q;
} }
++*node; ++*node;
@ -298,6 +315,9 @@ FrameUpdateCallBackWithVelocityExtractionSkinned(AnimBlendFrameData *frame, void
for(node = updateData->nodes; *node; node++){ for(node = updateData->nodes; *node; node++){
if((*node)->sequence){ if((*node)->sequence){
bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount); bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount);
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
rot += q; rot += q;
if((*node)->sequence->HasTranslation()){ if((*node)->sequence->HasTranslation()){
pos += vec; pos += vec;
@ -376,6 +396,11 @@ FrameUpdateCallBackWith3dVelocityExtractionSkinned(AnimBlendFrameData *frame, vo
for(node = updateData->nodes; *node; node++){ for(node = updateData->nodes; *node; node++){
if((*node)->sequence){ if((*node)->sequence){
bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount); bool nodelooped = (*node)->Update(vec, q, 1.0f-totalBlendAmount);
#ifdef FIX_BUGS
if(DotProduct(rot, q) < 0.0f)
rot -= q;
else
#endif
rot += q; rot += q;
if((*node)->sequence->HasTranslation()){ if((*node)->sequence->HasTranslation()){
pos += vec; pos += vec;
@ -410,4 +435,9 @@ FrameUpdateCallBackWith3dVelocityExtractionSkinned(AnimBlendFrameData *frame, vo
} }
} }
#endif void
FrameUpdateCallBackOffscreen(AnimBlendFrameData *frame, void *arg)
{
if(frame->flag & AnimBlendFrameData::VELOCITY_EXTRACTION && gpAnimBlendClump->velocity)
FrameUpdateCallBackWithVelocityExtractionSkinned(frame, arg);
}

View file

@ -9,9 +9,9 @@
#include "AnimBlendHierarchy.h" #include "AnimBlendHierarchy.h"
#include "AnimBlendAssociation.h" #include "AnimBlendAssociation.h"
#include "RpAnimBlend.h" #include "RpAnimBlend.h"
#ifdef PED_SKIN
#include "PedModelInfo.h" #include "PedModelInfo.h"
#endif
//--MIAMI: file done
RwInt32 ClumpOffset; RwInt32 ClumpOffset;
@ -141,7 +141,6 @@ FrameInitCBskin(AnimBlendFrameData *frameData, void*)
frameData->flag = 0; frameData->flag = 0;
} }
#ifdef PED_SKIN
void void
RpAnimBlendClumpInitSkinned(RpClump *clump) RpAnimBlendClumpInitSkinned(RpClump *clump)
{ {
@ -155,7 +154,7 @@ RpAnimBlendClumpInitSkinned(RpClump *clump)
RpAnimBlendAllocateData(clump); RpAnimBlendAllocateData(clump);
clumpData = *RPANIMBLENDCLUMPDATA(clump); clumpData = *RPANIMBLENDCLUMPDATA(clump);
atomic = IsClumpSkinned(clump); atomic = GetFirstAtomic(clump);
assert(atomic); assert(atomic);
skin = RpSkinGeometryGetSkin(RpAtomicGetGeometry(atomic)); skin = RpSkinGeometryGetSkin(RpAtomicGetGeometry(atomic));
assert(skin); assert(skin);
@ -175,7 +174,6 @@ RpAnimBlendClumpInitSkinned(RpClump *clump)
clumpData->ForAllFrames(FrameInitCBskin, nil); clumpData->ForAllFrames(FrameInitCBskin, nil);
clumpData->frames[0].flag |= AnimBlendFrameData::VELOCITY_EXTRACTION; clumpData->frames[0].flag |= AnimBlendFrameData::VELOCITY_EXTRACTION;
} }
#endif
void void
RpAnimBlendClumpInitNotSkinned(RpClump *clump) RpAnimBlendClumpInitNotSkinned(RpClump *clump)
@ -199,11 +197,9 @@ RpAnimBlendClumpInitNotSkinned(RpClump *clump)
void void
RpAnimBlendClumpInit(RpClump *clump) RpAnimBlendClumpInit(RpClump *clump)
{ {
#ifdef PED_SKIN
if(IsClumpSkinned(clump)) if(IsClumpSkinned(clump))
RpAnimBlendClumpInitSkinned(clump); RpAnimBlendClumpInitSkinned(clump);
else else
#endif
RpAnimBlendClumpInitNotSkinned(clump); RpAnimBlendClumpInitNotSkinned(clump);
} }
@ -348,9 +344,11 @@ CAnimBlendAssociation*
RpAnimBlendClumpGetFirstAssociation(RpClump *clump) RpAnimBlendClumpGetFirstAssociation(RpClump *clump)
{ {
CAnimBlendClumpData *clumpData = *RPANIMBLENDCLUMPDATA(clump); CAnimBlendClumpData *clumpData = *RPANIMBLENDCLUMPDATA(clump);
if(clumpData == nil) return nil; if(!RpAnimBlendClumpIsInitialized(clump))
if(clumpData->link.next == nil) return nil; return nil;
if(clumpData->link.next)
return CAnimBlendAssociation::FromLink(clumpData->link.next); return CAnimBlendAssociation::FromLink(clumpData->link.next);
return nil;
} }
// FillFrameArrayCallBack on PS2 // FillFrameArrayCallBack on PS2
@ -361,7 +359,6 @@ FillFrameArrayCBnonskin(AnimBlendFrameData *frame, void *arg)
frames[CVisibilityPlugins::GetFrameHierarchyId(frame->frame)] = frame; frames[CVisibilityPlugins::GetFrameHierarchyId(frame->frame)] = frame;
} }
#ifdef PED_SKIN
void void
RpAnimBlendClumpFillFrameArraySkin(RpClump *clump, AnimBlendFrameData **frames) RpAnimBlendClumpFillFrameArraySkin(RpClump *clump, AnimBlendFrameData **frames)
{ {
@ -371,16 +368,13 @@ RpAnimBlendClumpFillFrameArraySkin(RpClump *clump, AnimBlendFrameData **frames)
for(i = PED_MID; i < PED_NODE_MAX; i++) for(i = PED_MID; i < PED_NODE_MAX; i++)
frames[i] = &clumpData->frames[RpHAnimIDGetIndex(hier, ConvertPedNode2BoneTag(i))]; frames[i] = &clumpData->frames[RpHAnimIDGetIndex(hier, ConvertPedNode2BoneTag(i))];
} }
#endif
void void
RpAnimBlendClumpFillFrameArray(RpClump *clump, AnimBlendFrameData **frames) RpAnimBlendClumpFillFrameArray(RpClump *clump, AnimBlendFrameData **frames)
{ {
#ifdef PED_SKIN
if(IsClumpSkinned(clump)) if(IsClumpSkinned(clump))
RpAnimBlendClumpFillFrameArraySkin(clump, frames); RpAnimBlendClumpFillFrameArraySkin(clump, frames);
else else
#endif
(*RPANIMBLENDCLUMPDATA(clump))->ForAllFrames(FillFrameArrayCBnonskin, frames); (*RPANIMBLENDCLUMPDATA(clump))->ForAllFrames(FillFrameArrayCBnonskin, frames);
} }
@ -429,9 +423,27 @@ RpAnimBlendClumpFindBone(RpClump *clump, uint32 boneTag)
} }
void void
RpAnimBlendClumpUpdateAnimations(RpClump *clump, float timeDelta) RpAnimBlendNodeUpdateKeyframes(AnimBlendFrameData *frames, AnimBlendFrameUpdateData *updateData, int32 numNodes)
{
CAnimBlendNode **node;
int i;
for(node = updateData->nodes; *node; node++){
CAnimBlendAssociation *a = (*node)->association;
for(i = 0; i < numNodes; i++)
if((frames[i].flag & AnimBlendFrameData::VELOCITY_EXTRACTION) == 0 ||
gpAnimBlendClump->velocity == nil){
if((*node)[i].sequence)
(*node)[i].FindKeyFrame(a->currentTime - a->timeStep);
}
}
}
void
RpAnimBlendClumpUpdateAnimations(RpClump *clump, float timeDelta, bool doRender)
{ {
int i; int i;
CAnimBlendAssociation *assoc;
AnimBlendFrameUpdateData updateData; AnimBlendFrameUpdateData updateData;
float totalLength = 0.0f; float totalLength = 0.0f;
float totalBlend = 0.0f; float totalBlend = 0.0f;
@ -447,30 +459,45 @@ RpAnimBlendClumpUpdateAnimations(RpClump *clump, float timeDelta)
updateData.foobar = 0; updateData.foobar = 0;
for(link = clumpData->link.next; link; link = next){ for(link = clumpData->link.next; link; link = next){
next = link->next; next = link->next;
CAnimBlendAssociation *assoc = CAnimBlendAssociation::FromLink(link); assoc = CAnimBlendAssociation::FromLink(link);
if(assoc->UpdateBlend(timeDelta)){ if(assoc->UpdateBlend(timeDelta)){
if(assoc->hierarchy->sequences){
//CAnimManager::UncompressAnimation(v6->hierarchy) //CAnimManager::UncompressAnimation(v6->hierarchy)
if(i < 11)
updateData.nodes[i++] = assoc->GetNode(0); updateData.nodes[i++] = assoc->GetNode(0);
if(assoc->flags & ASSOC_MOVEMENT){ if(assoc->flags & ASSOC_MOVEMENT){
totalLength += assoc->hierarchy->totalLength/assoc->speed * assoc->blendAmount; totalLength += assoc->hierarchy->totalLength/assoc->speed * assoc->blendAmount;
totalBlend += assoc->blendAmount; totalBlend += assoc->blendAmount;
}else }else
updateData.foobar = 1; updateData.foobar = 1;
}else
debug("anim %s is not loaded\n", assoc->hierarchy->name);
} }
} }
for(link = clumpData->link.next; link; link = link->next){
assoc = CAnimBlendAssociation::FromLink(link);
assoc->UpdateTimeStep(timeDelta, totalLength == 0.0f ? 1.0f : totalBlend/totalLength);
}
updateData.nodes[i] = nil; updateData.nodes[i] = nil;
#ifdef PED_SKIN if(doRender){
if(clumpData->frames[0].flag & AnimBlendFrameData::UPDATE_KEYFRAMES)
RpAnimBlendNodeUpdateKeyframes(clumpData->frames, &updateData, clumpData->numFrames);
if(IsClumpSkinned(clump)) if(IsClumpSkinned(clump))
clumpData->ForAllFrames(FrameUpdateCallBackSkinned, &updateData); clumpData->ForAllFrames(FrameUpdateCallBackSkinned, &updateData);
else else
#endif
clumpData->ForAllFrames(FrameUpdateCallBackNonSkinned, &updateData); clumpData->ForAllFrames(FrameUpdateCallBackNonSkinned, &updateData);
clumpData->frames[0].flag &= ~AnimBlendFrameData::UPDATE_KEYFRAMES;
}else{
clumpData->ForAllFrames(FrameUpdateCallBackOffscreen, &updateData);
clumpData->frames[0].flag |= AnimBlendFrameData::UPDATE_KEYFRAMES;
}
for(link = clumpData->link.next; link; link = link->next){ for(link = clumpData->link.next; link; link = link->next){
CAnimBlendAssociation *assoc = CAnimBlendAssociation::FromLink(link); assoc = CAnimBlendAssociation::FromLink(link);
float relSpeed = totalLength == 0.0f ? 1.0f : totalBlend/totalLength; assoc->UpdateTime(timeDelta, totalLength == 0.0f ? 1.0f : totalBlend/totalLength);
assoc->UpdateTime(timeDelta, relSpeed);
} }
RwFrameUpdateObjects(RpClumpGetFrame(clump)); RwFrameUpdateObjects(RpClumpGetFrame(clump));
} }

View file

@ -35,9 +35,11 @@ CAnimBlendAssociation *RpAnimBlendClumpGetMainAssociation_N(RpClump *clump, int
CAnimBlendAssociation *RpAnimBlendClumpGetMainPartialAssociation_N(RpClump *clump, int n); CAnimBlendAssociation *RpAnimBlendClumpGetMainPartialAssociation_N(RpClump *clump, int n);
CAnimBlendAssociation *RpAnimBlendClumpGetFirstAssociation(RpClump *clump, uint32 mask); CAnimBlendAssociation *RpAnimBlendClumpGetFirstAssociation(RpClump *clump, uint32 mask);
CAnimBlendAssociation *RpAnimBlendClumpGetFirstAssociation(RpClump *clump); CAnimBlendAssociation *RpAnimBlendClumpGetFirstAssociation(RpClump *clump);
void RpAnimBlendClumpUpdateAnimations(RpClump* clump, float timeDelta); void RpAnimBlendNodeUpdateKeyframes(AnimBlendFrameData *frames, AnimBlendFrameUpdateData *updateData, int32 numNodes);
void RpAnimBlendClumpUpdateAnimations(RpClump* clump, float timeDelta, bool doRender = true);
extern CAnimBlendClumpData *gpAnimBlendClump; extern CAnimBlendClumpData *gpAnimBlendClump;
void FrameUpdateCallBackNonSkinned(AnimBlendFrameData *frame, void *arg); void FrameUpdateCallBackNonSkinned(AnimBlendFrameData *frame, void *arg);
void FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg); void FrameUpdateCallBackSkinned(AnimBlendFrameData *frame, void *arg);
void FrameUpdateCallBackOffscreen(AnimBlendFrameData *frame, void *arg);

View file

@ -29,7 +29,7 @@ bool PrintDebugCode = false;
int16 DebugCamMode; int16 DebugCamMode;
#ifdef FREE_CAM #ifdef FREE_CAM
bool CCamera::bFreeCam = false; bool CCamera::bFreeCam = true;
int nPreviousMode = -1; int nPreviousMode = -1;
#endif #endif

View file

@ -1926,6 +1926,7 @@ CWorld::Process(void)
if(csObj && csObj->m_entryInfoList.first) { if(csObj && csObj->m_entryInfoList.first) {
if(csObj->m_rwObject && RwObjectGetType(csObj->m_rwObject) == rpCLUMP && if(csObj->m_rwObject && RwObjectGetType(csObj->m_rwObject) == rpCLUMP &&
RpAnimBlendClumpGetFirstAssociation(csObj->GetClump())) { RpAnimBlendClumpGetFirstAssociation(csObj->GetClump())) {
// TODO(MIAMI): doRender argument
RpAnimBlendClumpUpdateAnimations(csObj->GetClump(), RpAnimBlendClumpUpdateAnimations(csObj->GetClump(),
0.02f * (csObj->IsObject() 0.02f * (csObj->IsObject()
? CTimer::GetTimeStepNonClipped() ? CTimer::GetTimeStepNonClipped()
@ -1944,6 +1945,7 @@ CWorld::Process(void)
CEntity *movingEnt = (CEntity *)node->item; CEntity *movingEnt = (CEntity *)node->item;
if(movingEnt->m_rwObject && RwObjectGetType(movingEnt->m_rwObject) == rpCLUMP && if(movingEnt->m_rwObject && RwObjectGetType(movingEnt->m_rwObject) == rpCLUMP &&
RpAnimBlendClumpGetFirstAssociation(movingEnt->GetClump())) { RpAnimBlendClumpGetFirstAssociation(movingEnt->GetClump())) {
// TODO(MIAMI): doRender argument
RpAnimBlendClumpUpdateAnimations(movingEnt->GetClump(), RpAnimBlendClumpUpdateAnimations(movingEnt->GetClump(),
0.02f * (movingEnt->IsObject() 0.02f * (movingEnt->IsObject()
? CTimer::GetTimeStepNonClipped() ? CTimer::GetTimeStepNonClipped()