/*==LICENSE==* CyanWorlds.com Engine - MMOG client, server and tools Copyright (C) 2011 Cyan Worlds, Inc. This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. Additional permissions under GNU GPL version 3 section 7 If you modify this Program, or any covered work, by linking or combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK (or a modified version of those libraries), containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the licensors of this Program grant you additional permission to convey the resulting work. Corresponding Source for a non-source form of such a combination shall include the source code for the parts of OpenSSL and IJG JPEG Library used as well as that of the covered work. You can contact Cyan Worlds, Inc. by email legal@cyan.com or by snail mail at: Cyan Worlds, Inc. 14617 N Newport Hwy Mead, WA 99021 *==LICENSE==*/ ///////////////////////////////////////////////////////////////////////////////////////// // // INCLUDES // ///////////////////////////////////////////////////////////////////////////////////////// #include <cmath> // singular #include "plAvTaskSeek.h" // local #include "plAvBrainHuman.h" #include "plAnimation/plAGAnim.h" #include "plArmatureMod.h" #include "plAvatarMgr.h" #include "plPhysicalControllerCore.h" // other #include "plMessage/plAvatarMsg.h" #include "pnMessage/plCameraMsg.h" #include "pnInputCore/plControlEventCodes.h" #include "plPipeline/plDebugText.h" #include "plStatusLog/plStatusLog.h" #include "pnSceneObject/plCoordinateInterface.h" #include "hsTimer.h" #include "plgDispatch.h" ///////////////////////////////////////////////////////////////////////////////////////// // // PROTOTYPES // ///////////////////////////////////////////////////////////////////////////////////////// float QuatAngleDiff(const hsQuat &a, const hsQuat &b); void MakeMatrixUpright(hsMatrix44 &mat); ///////////////////////////////////////////////////////////////////////////////////////// // // DEFINES // ///////////////////////////////////////////////////////////////////////////////////////// #define kSeekTimeout 5.0f #define kRotSpeed 1.0f // normal rotation speed is 1.0 radians per second #define kFloatSpeed 3.0f #define kMaxRadiansPerSecond 1.5 #define kDefaultShuffleRange 0.5f #define kDefaultMaxSidleRange 4.0f #define kDefaultMaxSidleAngle 0.2f bool plAvTaskSeek::fLogProcess = false; ///////////////////////////////////////////////////////////////////////////////////////// // // IMPLEMENTATION // ///////////////////////////////////////////////////////////////////////////////////////// void plAvTaskSeek::IInitDefaults() { fSeekObject = nullptr; fMovingTarget = false; fAlign = kAlignHandle; fAnimName = ""; fPosGoalHit = false; fRotGoalHit = false; fStillPositioning = true; fStillRotating = true; fShuffleRange = kDefaultShuffleRange; fMaxSidleRange = kDefaultMaxSidleRange; fMaxSidleAngle = kDefaultMaxSidleAngle; fFlags = kSeekFlagForce3rdPersonOnStart; fState = kSeekRunNormal; fNotifyFinishedKey = nullptr; fFinishMsg = nullptr; } // plAvTaskSeek ------------ // ------------- plAvTaskSeek::plAvTaskSeek() {} plAvTaskSeek::plAvTaskSeek(plAvSeekMsg *msg) { IInitDefaults(); fAlign = msg->fAlignType; fAnimName = msg->fAnimName; plKey &target = msg->fSeekPoint; if (target) SetTarget(target); else SetTarget(msg->fTargetPos, msg->fTargetLookAt); if (msg->UnForce3rdPersonOnFinish()) fFlags |= kSeekFlagUnForce3rdPersonOnFinish; else fFlags &= ~kSeekFlagUnForce3rdPersonOnFinish; if (msg->Force3rdPersonOnStart()) fFlags |= kSeekFlagForce3rdPersonOnStart; else fFlags &= ~kSeekFlagForce3rdPersonOnStart; if (msg->NoWarpOnTimeout()) fFlags |= kSeekFlagNoWarpOnTimeout; else fFlags &= ~kSeekFlagNoWarpOnTimeout; if (msg->RotationOnly()) { fFlags |= kSeekFlagRotationOnly; fStillPositioning = false; fPosGoalHit = true; } else fFlags &= ~kSeekFlagRotationOnly; fNotifyFinishedKey = msg->fFinishKey; fFinishMsg = msg->fFinishMsg; } // plAvTaskSeek ------------------------ // ------------- plAvTaskSeek::plAvTaskSeek(plKey target) { IInitDefaults(); SetTarget(target); } // plAvTaskSeek ------------------------------------------------------------------------------------------- // ------------- plAvTaskSeek::plAvTaskSeek(plKey target, plAvAlignment align, const plString& animName, bool moving) { IInitDefaults(); fMovingTarget = moving; fAlign = align; fAnimName = animName; SetTarget(target); } void plAvTaskSeek::SetTarget(plKey target) { hsAssert(target, "Bad key to seek task"); if(target) { fSeekObject = plSceneObject::ConvertNoRef(target->ObjectIsLoaded()); } else { fSeekObject = nil; } } void plAvTaskSeek::SetTarget(hsPoint3 &pos, hsPoint3 &lookAt) { fSeekPos = pos; hsVector3 up(0.f, 0.f, 1.f); float angle = atan2(lookAt.fY - pos.fY, lookAt.fX - pos.fX) + M_PI / 2; fSeekRot.SetAngleAxis(angle, up); } // Start ----------------------------------------------------------------------------------------- // ------ bool plAvTaskSeek::Start(plArmatureMod *avatar, plArmatureBrain *brain, double time, float elapsed) { plAvBrainHuman *huBrain = plAvBrainHuman::ConvertNoRef(brain); hsAssert(huBrain, "Seek task only works on human brains"); plAvatarMgr::GetInstance()->GetLog()->AddLine("Starting SMART SEEK"); //controller needs to know we are seeking. prevents controller from interacting with exclusion regions if (avatar->GetController() ) avatar->GetController()->SetSeek(true); fStartTime = time; if(huBrain) { avatar->SuspendInput(); // stop accepting input from the user, but queue any messages // ...and save our current input state. ILimitPlayersInput(avatar); if (plAvOneShotTask::fForce3rdPerson && avatar->IsLocalAvatar() && (fFlags & plAvSeekMsg::kSeekFlagForce3rdPersonOnStart)) { // create message plCameraMsg* pMsg = new plCameraMsg; pMsg->SetBCastFlag(plMessage::kBCastByExactType); pMsg->SetBCastFlag(plMessage::kNetPropagate, false); pMsg->SetCmd(plCameraMsg::kResponderSetThirdPerson); plgDispatch::MsgSend( pMsg ); // whoosh... off it goes } huBrain->IdleOnly(); // Makes sure to kill jumps too. Just calling ClearInputFlags isn't enough IUpdateObjective(avatar); return true; } else { return false; } } // Process ------------------------------------------------------------------------------------------- // -------- bool plAvTaskSeek::Process(plArmatureMod *avatar, plArmatureBrain *brain, double time, float elapsed) { if (fState == kSeekAbort) return false; plAvBrainHuman *uBrain = plAvBrainHuman::ConvertNoRef(brain); if (uBrain) { if (fMovingTarget) { IUpdateObjective(avatar); } IAnalyze(avatar); bool result = IMoveTowardsGoal(avatar, uBrain, time, elapsed); if (fLogProcess) DumpToAvatarLog(avatar); return result; } return false; } // Finish --------------------------------------------------------------------------------------- // ------- void plAvTaskSeek::Finish(plArmatureMod *avatar, plArmatureBrain *brain, double time, float elapsed) { plAvBrainHuman *huBrain = plAvBrainHuman::ConvertNoRef(brain); if(huBrain) { // this will process any queued input messages so if the user pressed or released a key while we were busy, we'll note it now. avatar->ResumeInput(); IUndoLimitPlayersInput(avatar); if (plAvOneShotTask::fForce3rdPerson && avatar->IsLocalAvatar() && (fFlags & plAvSeekMsg::kSeekFlagUnForce3rdPersonOnFinish)) { // create message plCameraMsg* pMsg = new plCameraMsg; pMsg->SetBCastFlag(plMessage::kBCastByExactType); pMsg->SetBCastFlag(plMessage::kNetPropagate, false); pMsg->SetCmd(plCameraMsg::kResponderUndoThirdPerson); plgDispatch::MsgSend( pMsg ); // whoosh... off it goes } avatar->SynchIfLocal(hsTimer::GetSysSeconds(), false); } if (fNotifyFinishedKey) { plAvTaskSeekDoneMsg *msg = new plAvTaskSeekDoneMsg(avatar->GetKey(), fNotifyFinishedKey); msg->fAborted = (fState == kSeekAbort); msg->Send(); } plAvatarMgr::GetInstance()->GetLog()->AddLine("Finished SMART SEEK"); //inform controller we are done seeking if (avatar->GetController()) avatar->GetController()->SetSeek(false); if (fFinishMsg) fFinishMsg->Send(); } void plAvTaskSeek::LeaveAge(plArmatureMod *avatar) { fSeekObject = nil; fState = kSeekAbort; } // IAnalyze ---------------------------------------- // --------- bool plAvTaskSeek::IAnalyze(plArmatureMod *avatar) { avatar->GetPositionAndRotationSim(&fPosition, &fRotation); hsScalarTriple tmp(fSeekPos - fPosition); fGoalVec.Set(&tmp); hsVector3 normalizedGoalVec(fGoalVec); normalizedGoalVec.Normalize(); fDistance = sqrt(fGoalVec.fX * fGoalVec.fX + fGoalVec.fY * fGoalVec.fY); if(fDistance < 3.0f) { // we're in "near target" mode fMinFwdAngle = .5f; // walk forward if target's in 90' cone straight ahead fMaxBackAngle = -.2f; // walk backward if target's in a 144' cone behind } else { // we're in "far target" mode fMinFwdAngle = .2f; // walk forward if target's in a 144' cone ahead fMaxBackAngle = -2.0; // disable backing up if goal is far out (-1 is the minimum usable value here) } hsQuat invRot = fRotation.Conjugate(); hsPoint3 globFwd = invRot.Rotate(&kAvatarForward); hsPoint3 globRight = invRot.Rotate(&kAvatarRight); hsPoint3 locGoalVec = fRotation.Rotate(&fGoalVec); fDistForward = -(locGoalVec.fY); fDistRight = -(locGoalVec.fX); fAngForward = globFwd.InnerProduct(normalizedGoalVec); fAngRight = globRight.InnerProduct(normalizedGoalVec); return true; } // IMoveTowardsGoal -------------------------------------------------------------- // ----------------- bool plAvTaskSeek::IMoveTowardsGoal(plArmatureMod *avatar, plAvBrainHuman *brain, double time, float elapsed) { bool stillRunning = true; avatar->ClearInputFlags(false, false); double duration = time - fStartTime; if(duration > kSeekTimeout) { if (fFlags & kSeekFlagNoWarpOnTimeout) { fState = kSeekAbort; return false; } fSeekRot.Normalize(); avatar->SetPositionAndRotationSim(&fSeekPos, &fSeekRot); IAnalyze(avatar); // Recalcs fPosition, fDistance, etc. hsStatusMessage("Timing out on smart seek - jumping to target."); stillRunning = false; // We just set the pos/rot, so we know these are hit. fPosGoalHit = true; fRotGoalHit = true; } if (!(fDistance > fShuffleRange)) fPosGoalHit = true; if (!fPosGoalHit) { bool right = fAngRight > 0.0f; bool inSidleRange = fDistance < fMaxSidleRange; bool sidling = fabs(fDistRight) > fabs(fDistForward) && inSidleRange; if(sidling) { if(right) avatar->SetStrafeRightKeyDown(); else avatar->SetStrafeLeftKeyDown(); } else { if(fAngForward < fMaxBackAngle) avatar->SetBackwardKeyDown(); else { if(fAngForward > fMinFwdAngle) avatar->SetForwardKeyDown(); if(right) avatar->SetTurnRightKeyDown(); else avatar->SetTurnLeftKeyDown(); } } } else { if (!(QuatAngleDiff(fRotation, fSeekRot) > .1)) fRotGoalHit = true; if (!fRotGoalHit) { hsQuat invRot = fSeekRot.Conjugate(); hsPoint3 globFwd = invRot.Rotate(&kAvatarForward); globFwd = fRotation.Rotate(&globFwd); if (globFwd.fX < 0) avatar->SetTurnRightKeyDown(); else avatar->SetTurnLeftKeyDown(); } } if (fPosGoalHit && fRotGoalHit) stillRunning = ITryFinish(avatar, brain, time, elapsed); return stillRunning; } // ITRYFINISH bool plAvTaskSeek::ITryFinish(plArmatureMod *avatar, plAvBrainHuman *brain, double time, float elapsed) { bool animsDone = brain->IsMovementZeroBlend(); hsPoint3 newPosition = fPosition; hsQuat newRotation = fRotation; if (!(fFlags & kSeekFlagRotationOnly) && (fStillPositioning || !animsDone)) fStillPositioning = IFinishPosition(newPosition, avatar, brain, time, elapsed); if (fStillRotating || !animsDone) fStillRotating = IFinishRotation(newRotation, avatar, brain, time, elapsed); newRotation.Normalize(); if (hsCheckBits(fFlags, kSeekFlagRotationOnly)) avatar->SetPositionAndRotationSim(nil, &newRotation); else avatar->SetPositionAndRotationSim(&newPosition, &newRotation); return fStillPositioning || fStillRotating || !animsDone; } bool plAvTaskSeek::IFinishPosition(hsPoint3 &newPosition, plArmatureMod *avatar, plAvBrainHuman *brain, double time, float elapsed) { // While warping, we might be hovering just above the ground. Don't want that to // trigger any falling behavior. if(brain&&brain->fWalkingStrategy) { brain->fWalkingStrategy->ResetAirTime(); } // how far will we translate this frame? float thisDist = kFloatSpeed * elapsed; // what percentage of the remaining distance will we cover? float thisPct = (fDistance ? thisDist / fDistance : 1.f); if(thisPct > 0.9f) { // we're pretty much done; just hop the rest of the way newPosition = fSeekPos; return false; // we're done } else { // move incrementally toward the target position hsVector3 thisMove = fGoalVec * thisPct; newPosition = fPosition + thisMove; return true; // we're still processing } return true; } // IFinishRotation -------------------------------------- // ---------------- bool plAvTaskSeek::IFinishRotation(hsQuat &newRotation, plArmatureMod *avatar, plAvBrainHuman *brain, double time, float elapsed) { // we're pretty much done; just hop the rest of the way newRotation = fSeekRot; return false; } // IUpdateObjective ---------------------------------------- // ----------------- bool plAvTaskSeek::IUpdateObjective(plArmatureMod *avatar) { // This is an entirely valid case. It just means our goal is fixed. if (fSeekObject == nil) return true; // goal here is to express the target matrix in the avatar's PHYSICAL space hsMatrix44 targL2W = fSeekObject->GetLocalToWorld(); const plCoordinateInterface* subworldCI = nil; if (avatar->GetController()) subworldCI = avatar->GetController()->GetSubworldCI(); if (subworldCI) targL2W = subworldCI->GetWorldToLocal() * targL2W; MakeMatrixUpright(targL2W); switch(fAlign) { // match our handle to the target matrix at the end of the given animation // This case isn't currently used but will be important someday. The idea // is that you have a target point and an animation, and you want to seek // the avatar to a point where he can start playing the animation and wind // up, after the animation completes, at the target location. // Hence "AlignHandleAnimEnd" = "align the avatar so the animation will // end on the target." case kAlignHandleAnimEnd: { hsMatrix44 adjustment; plAGAnim *anim = avatar->FindCustomAnim(fAnimName); // don't need to do this every frame; the animation doesn't change. // *** cache the adjustment; GetStartToEndTransform(anim, nil, &adjustment, "Handle"); // actually getting end-to-start // ... but we do still need to multiply by the (potentially changed) target targL2W = targL2W * adjustment; } break; case kAlignHandle: // targetMat is already correct default: break; }; GetPositionAndRotation(targL2W, &fSeekPos, &fSeekRot); return true; } // DumpDebug ----------------------------------------------------------------------------------------------------- // ---------- void plAvTaskSeek::DumpDebug(const char *name, int &x, int&y, int lineHeight, plDebugText &debugTxt) { debugTxt.DrawString(x, y, plFormat("duration: {.2f} pos: ({.3f}, {.3f}, {.3f}) goalPos: ({.3f}, {.3f}, {.3f}) ", hsTimer::GetSysSeconds() - fStartTime, fPosition.fX, fPosition.fY, fPosition.fZ, fSeekPos.fX, fSeekPos.fY, fSeekPos.fZ)); y += lineHeight; debugTxt.DrawString(x, y, plFormat("positioning: {} rotating {} goalVec: ({.3f}, {.3f}, {.3f}) dist: {.3f} angFwd: {.3f} angRt: {.3f}", fStillPositioning, fStillRotating, fGoalVec.fX, fGoalVec.fY, fGoalVec.fZ, fDistance, fAngForward, fAngRight)); y += lineHeight; debugTxt.DrawString(x, y, plFormat(" distFwd: {.3f} distRt: {.3f} shufRange: {.3f} sidAngle: {.3f} sidRange: {.3f}, fMinWalk: {.3f}", fDistForward, fDistRight, fShuffleRange, fMaxSidleAngle, fMaxSidleRange, fMinFwdAngle)); y += lineHeight; } void plAvTaskSeek::DumpToAvatarLog(plArmatureMod *avatar) { plStatusLog *log = plAvatarMgr::GetInstance()->GetLog(); log->AddLine(avatar->GetMoveKeyString().c_str()); log->AddLine(plFormat(" duration: {.2f} pos: ({.3f}, {.3f}, {.3f}) goalPos: ({.3f}, {.3f}, {.3f}) ", hsTimer::GetSysSeconds() - fStartTime, fPosition.fX, fPosition.fY, fPosition.fZ, fSeekPos.fX, fSeekPos.fY, fSeekPos.fZ).c_str()); log->AddLine(plFormat(" positioning: {} rotating {} goalVec: ({.3f}, {.3f}, {.3f}) dist: {.3f} angFwd: {.3f} angRt: {.3f}", fStillPositioning, fStillRotating, fGoalVec.fX, fGoalVec.fY, fGoalVec.fZ, fDistance, fAngForward, fAngRight).c_str()); log->AddLine(plFormat(" distFwd: {.3f} distRt: {.3f} shufRange: {.3f} sidAngle: {.3f} sidRange: {.3f}, fMinWalk: {.3f}", fDistForward, fDistRight, fShuffleRange, fMaxSidleAngle, fMaxSidleRange, fMinFwdAngle).c_str()); } ///////////////////////////////////////////////////////////////////////////////////////// // // LOCALS // ///////////////////////////////////////////////////////////////////////////////////////// // QuatAngleDiff ------------------------------------ // -------------- float QuatAngleDiff(const hsQuat &a, const hsQuat &b) { float theta; /* angle between A and B */ float cos_t; /* sine, cosine of theta */ /* cosine theta = dot product of A and B */ cos_t = a.Dot(b); /* if B is on opposite hemisphere from A, use -B instead */ if (cos_t < 0.0) { cos_t = -cos_t; } // Calling acos on 1.0 is returning an undefined value. Need to check for it. float epsilon = 0.00001; if (fabs(cos_t - 1.f) < epsilon) return 0; theta = acos(cos_t); return theta; } // MakeMatrixUpright ------------------------------------------- // ------------------ // ensure that the z axis of the given matrix points at the sky. // does not orthonormalize // man, I could have sworn I did this somewhere else... void MakeMatrixUpright(hsMatrix44 &mat) { mat.fMap[0][2] = 0.0f; // eliminate any z in the x axis mat.fMap[1][2] = 0.0f; // eliminate any z in the y axis mat.fMap[2][0] = 0.0f; mat.fMap[2][1] = 0.0f; mat.fMap[2][2] = 1.0f; // z axis = pure sky }