/*==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==*/ #include "HeadSpin.h" #include "plgDispatch.h" #include "hsResMgr.h" #include "plComponentReg.h" #include "plAnimComponent.h" #include "plCameraComponents.h" #include "plMiscComponents.h" #include "plPhysicalComponents.h" #include "MaxMain/plMaxNode.h" #include "MaxMain/plMaxNodeData.h" #include <iparamm2.h> #include <string> #include <vector> #include "resource.h" #pragma hdrstop #include "pnSceneObject/plSceneObject.h" // Ibid #include "pnSceneObject/plCoordinateInterface.h" #include "plScene/plSceneNode.h" // Ibid #include "MaxConvert/plConvert.h" #include "MaxConvert/hsConverterUtils.h" //Conversion Dependencies #include "MaxConvert/hsControlConverter.h" // Ibid #include "plPhysical/plSimDefs.h" #include "pnMessage/plObjRefMsg.h" // Ibid #include "pnMessage/plIntRefMsg.h" // Ibid #include "pnMessage/plNodeRefMsg.h" // Ibid #include "pnMessage/plCameraMsg.h" // Ibid #include "MaxMain/plPlasmaRefMsgs.h" // Ibid #include "pfAnimation/plLineFollowMod.h" #include "plPhysical/plCollisionDetector.h" // MM #include "pfCamera/plCameraBrain.h" #include "pfCamera/plCameraModifier.h" #include "MaxMain/plPhysicalProps.h" // Line Follow related #include "plInterp/plAnimPath.h" #include "plInterp/plController.h" #include "pfAnimation/plLineFollowMod.h" #include "pfAnimation/plFollowMod.h" // // DummyCodeIncludeFuncPhys Function. // Necessary to keep the compiler from throwing away this file. // No functions within are inherently called otherwise.... // // void DummyCodeIncludeFuncCameras() { } // struct for storing transition information temporarily: // used instead of camtrans because we don't have keys yet, // but need to set up at pre-convert struct PreTrans { // used when creating default track transitions at runtime PreTrans(plSceneObject* to) { fTransTo = to; fAccel = 60.0f; fDecel = 60.0f; fVelocity = 60.0f; fPOADecel = 60.0f; fPOAAccel = 60.0f; fPOAVelocity = 60.0f; fCutPos = false; fCutPOA = false; fIgnore = false; } plSceneObject* fTransTo; bool fCutPos; bool fCutPOA; bool fIgnore; float fAccel; float fDecel; float fVelocity; float fPOAAccel; float fPOADecel; float fPOAVelocity; }; /// /// /// /// /// /// /// limit pan component class /// /// /// /// /// /// //Max desc stuff necessary. CLASS_DESC(plLimitPanComponent, gLimitPaneraDesc, "Allow Camera Panning", "Allow Camera Panning", COMP_TYPE_CAMERA, LIMITPAN_CID) enum { kLimitPanX, kLimitPanZ, kPanXDeg, kPanZDeg, kMouseSensitivity, // obsolete }; //Max paramblock2 stuff below. ParamBlockDesc2 gLimitPaneraBk ( 1, _T("camera"), 0, &gLimitPaneraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERAPAN, IDS_COMP_ALLOW_PAN, 0, 0, NULL, kLimitPanX, _T("Limit X"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_X_AXIS, end, kLimitPanZ, _T("Limit Z"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_Z_AXIS, end, kPanXDeg, _T("X degrees"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 180.0f, p_default, 90.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kPanZDeg, _T("Z degrees"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 180.0f, p_default, 90.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, end ); plLimitPanComponent::plLimitPanComponent() { fClassDesc = &gLimitPaneraDesc; fClassDesc->MakeAutoParamBlocks(this); } // SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool plLimitPanComponent::SetupProperties(plMaxNode* pNode, plErrorMsg *pErrMsg) { return true; } bool plLimitPanComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { return true; } bool plLimitPanComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } /// /// /// /// /// /// /// Camera Zoom component class /// /// /// /// /// /// //Max desc stuff necessary. CLASS_DESC(plCameraZoomComponent, gCameraZoomeraDesc, "Allow Camera Zoom", "Allow Camera Zoom", COMP_TYPE_CAMERA, CAMERAZOOM_CID) enum { kZoomMaxDeg, kZoomMinDeg, kZoomRate, }; //Max paramblock2 stuff below. ParamBlockDesc2 gCameraZoomeraBk ( 1, _T("camera"), 0, &gCameraZoomeraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERAZOOM, IDS_COMP_CAMERAZOOM, 0, 0, NULL, kZoomMaxDeg, _T("max degrees"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 180.0f, p_default, 120.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kZoomMinDeg, _T("min degrees"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 180.0f, p_default, 35.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kZoomRate, _T("degrees per sec"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 180.0f, p_default, 90.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY2, IDC_CAMERACMD_SPIN_OFFSETY2, SPIN_AUTOSCALE, end, end ); plCameraZoomComponent::plCameraZoomComponent() { fClassDesc = &gCameraZoomeraDesc; fClassDesc->MakeAutoParamBlocks(this); } // SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool plCameraZoomComponent::SetupProperties(plMaxNode* pNode, plErrorMsg *pErrMsg) { return true; } bool plCameraZoomComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { return true; } bool plCameraZoomComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } // // // // // // // // // // // Override transition component // // // // //Max desc stuff necessary. CLASS_DESC(plTransOverrideComponent, gTransOverrideeraDesc, "Override Transition", "Override Transition", COMP_TYPE_CAMERA, TRANSCAM_CID) enum { kTransitionTo, kCutTrans, kTrackTrans, kTransSpeed, kTransAccel, kCutPOA, kTrackPOA, kTransPOASpeed, kTransPOAAccel, kTransDecel, kTransPOADecel, kTransVelocity, kTrans, kIgnoreTrans, kTransPOAVelocity, }; //Max paramblock2 stuff below. ParamBlockDesc2 gTransOverrideeraBk ( 1, _T("camera"), 0, &gTransOverrideeraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERA_TRANS, IDS_COMP_CAMERATRANS, 0, 0, NULL, kTransitionTo, _T("transitionto"), TYPE_INODE, 0, 0, p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_CAMERARGN_PICKSTATE_BASE, p_sclassID, CAMERA_CLASS_ID, p_prompt, IDS_COMP_PHYS_CHOSEN_BASE, end, kCutTrans, _T("cut"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK1, end, kTrackTrans, _T("track"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK2, p_enable_ctrls, 3, kTransVelocity, kTransAccel, kTransDecel, end, kCutPOA, _T("cutPOA"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK3, end, kTrackPOA, _T("trackPOA"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK4, p_enable_ctrls, 3, kTransPOASpeed, kTransPOAAccel, kTransPOADecel, end, kTransVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL, IDC_CAMERA_VEL_SPIN, SPIN_AUTOSCALE, end, kTransAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL, IDC_CAMERA_ACCEL_SPIN, SPIN_AUTOSCALE, end, kTransDecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL, IDC_CAMERA_DECEL_SPIN, SPIN_AUTOSCALE, end, kTransPOAVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL2, IDC_CAMERA_VEL_SPIN2, SPIN_AUTOSCALE, end, kTransPOAAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL2, IDC_CAMERA_ACCEL_SPIN2, SPIN_AUTOSCALE, end, kTransPOADecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -200.0f, 200.0f, p_default, 100.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL2, IDC_CAMERA_DECEL_SPIN2, SPIN_AUTOSCALE, end, kIgnoreTrans, _T("ignore"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK_IGNORE, end, end ); plTransOverrideComponent::plTransOverrideComponent() { fClassDesc = &gTransOverrideeraDesc; fClassDesc->MakeAutoParamBlocks(this); } // SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool plTransOverrideComponent::SetupProperties(plMaxNode* pNode, plErrorMsg *pErrMsg) { fTransKeys.clear(); return true; } bool plTransOverrideComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { // see if there is a camera specified INode* pCamNode = fCompPB->GetINode(kTransitionTo); PreTrans* pTrans = nil; if (pCamNode) pTrans = new PreTrans(((plMaxNode*)pCamNode)->GetSceneObject()); else pTrans = new PreTrans(nil); if (fCompPB->GetInt(kIgnoreTrans)) { pTrans->fIgnore = true; } else if (fCompPB->GetInt(kTrackTrans)) { pTrans->fCutPos = false; pTrans->fAccel = fCompPB->GetFloat(kTransAccel); pTrans->fDecel = fCompPB->GetFloat(kTransDecel); pTrans->fVelocity = fCompPB->GetFloat(kTransVelocity); } else { pTrans->fCutPos = true; } if (fCompPB->GetInt(kTrackPOA)) { pTrans->fCutPOA = false; pTrans->fPOAAccel = fCompPB->GetFloat(kTransPOAAccel); pTrans->fPOADecel = fCompPB->GetFloat(kTransPOADecel); pTrans->fPOAVelocity = fCompPB->GetFloat(kTransPOAVelocity); } else { pTrans->fCutPOA = true; } fTransKeys[pNode] = pTrans; return true; } bool plTransOverrideComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } bool plTransOverrideComponent::DeInit(plMaxNode *node, plErrorMsg *pErrMsg) { TransitionKeys::iterator i = fTransKeys.begin(); for( ; i != fTransKeys.end(); i++ ) { delete (*i).second; } fTransKeys.clear(); return plComponent::DeInit( node, pErrMsg ); } ///////////////////////////////////////////////////////////////////////////////////////////////// // // POA components // // // // // // // // // // // // // avatar POA component... //Max desc stuff necessary. CLASS_DESC(plPOAAvatarComponent, gPOAAvatareraDesc, "Avatar POA", "Avatar POA", COMP_TYPE_CAMERA, AVATAR_POA_CID) enum { kAvPOAOffX, kAvPOAOffY, kAvPOAOffZ, kAvPOAWorldspace, }; //Max paramblock2 stuff below. ParamBlockDesc2 gPOAAvatareraBk ( 1, _T("camera"), 0, &gPOAAvatareraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_AVATAR_POA, IDS_AVATARPOA, 0, 0, NULL, kAvPOAOffX, _T("PX Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX5, IDC_CAMERACMD_SPIN_OFFSETX5, SPIN_AUTOSCALE, end, kAvPOAOffY, _T("PY Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY4, IDC_CAMERACMD_SPIN_OFFSETY4, SPIN_AUTOSCALE, end, kAvPOAOffZ, _T("PZ Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ2, IDC_CAMERACMD_SPIN_OFFSETZ2, SPIN_AUTOSCALE, end, kAvPOAWorldspace, _T("POAworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_POA_WORLDSPACE, end, end ); plPOAAvatarComponent::plPOAAvatarComponent() { fClassDesc = &gPOAAvatareraDesc; fClassDesc->MakeAutoParamBlocks(this); } // // // // // // // Object POA component // // //Max desc stuff necessary. CLASS_DESC(plPOAObjectComponent, gPOAObjecteraDesc, "Object POA", "Object POA", COMP_TYPE_CAMERA, OBJECT_POA_CID) enum { kPOAObject, kPOAObjOffsetX, kPOAObjOffsetY, kPOAObjOffsetZ, kPOAObjWorldspace, }; //Max paramblock2 stuff below. ParamBlockDesc2 gPOAObjecteraBk ( 1, _T("camera"), 0, &gPOAObjecteraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_OBJECT_POA, IDS_COMP_OBJECTPOA, 0, 0, NULL, kPOAObject, _T("objectPOA"), TYPE_INODE, 0, 0, p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_CLICK_PROXY, p_sclassID, GEOMOBJECT_CLASS_ID, p_prompt, IDS_COMP_PHYS_CHOSEN_BASE, end, kPOAObjOffsetX, _T("X Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kPOAObjOffsetY, _T("Y Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kPOAObjOffsetZ, _T("Z Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ, IDC_CAMERACMD_SPIN_OFFSETZ, SPIN_AUTOSCALE, end, kPOAObjWorldspace, _T("POAworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_OBJECTPOA_WORLDSPACE, end, end ); plPOAObjectComponent::plPOAObjectComponent() { fClassDesc = &gPOAObjecteraDesc; fClassDesc->MakeAutoParamBlocks(this); } // SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool plPOAObjectComponent::SetupProperties(plMaxNode* pNode, plErrorMsg *pErrMsg) { if (fCompPB->GetINode(kPOAObject)) { plMaxNode *pPOA = ((plMaxNode*)fCompPB->GetINode(kPOAObject)); if (pPOA) { pPOA->SetForceLocal(true); return true; } } return false; } bool plPOAObjectComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { return true; } plKey plPOAObjectComponent::GetObjectKey() { if (fCompPB->GetINode(kPOAObject)) return ((plMaxNode*)(fCompPB->GetINode(kPOAObject)))->GetSceneObject()->GetKey(); return nil; } // // // // // // // // Make Default component... // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plMakeDefaultCamComponent, gMakeDefaultCameraDesc, "Make Default Camera", "Make Default Camera", COMP_TYPE_CAMERA, DEFAULTCAM_CID) enum { kMakeDefaultCam, }; //Max paramblock2 stuff below. ParamBlockDesc2 gMakeDefaultCameraBk ( 1, _T("camera"), 0, &gMakeDefaultCameraDesc, P_AUTO_CONSTRUCT, plComponent::kRefComp, end ); plMakeDefaultCamComponent::plMakeDefaultCamComponent() { fClassDesc = &gMakeDefaultCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } // // // // // // // // // // // // base class for all camera components which make a plCameraModifier // // // // bool plCameraBaseComponent::IsValidNodeType(plMaxNode *pNode) { Object *obj = pNode->EvalWorldState(hsConverterUtils::Instance().GetTime(pNode->GetInterface())).obj; TimeValue Now = hsConverterUtils::Instance().GetTime(pNode->GetInterface()); if(obj->ConvertToType(Now, Class_ID(LOOKAT_CAM_CLASS_ID, 0))) return true; else return false; } bool plCameraBaseComponent::SetupProperties(plMaxNode* pNode, plErrorMsg* pErrMsg) { fModKeys.clear(); bool ValidNode = IsValidNodeType(pNode); if(!ValidNode){ if(pErrMsg->Set(true, "Invalid Camera Object", "The camera %s is not a 'Max Target Camera type'. This camera will be disabled..\nKill the export?",((INode*)pNode)->GetName()).Ask()) pErrMsg->Set(true, "", ""); else pErrMsg->Set(false); // Don't want to abort return false; } plMaxNode* node = ((plMaxNode*)pNode->GetTarget()); if (node) { node->SetDrawable(false); node->SetMovable(true); node->SetForceLocal(true); } return true; } bool plCameraBaseComponent::PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg) { return true; } bool plCameraBaseComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { // check for overriden transitions and special animation commands int count = node->NumAttachedComponents(); for (uint32_t x = 0; x < count; x++) { plComponentBase *comp = node->GetAttachedComponent(x); if (comp->ClassID() == TRANSCAM_CID) { plTransOverrideComponent* pTrans = (plTransOverrideComponent*)comp; int idx = 0; plTransOverrideComponent::TransitionKeys::const_iterator it; for (it = pTrans->fTransKeys.begin(); it != pTrans->fTransKeys.end(); it++) { if (it->first != node) continue; PreTrans *trans = it->second; // convert the pre transition to a camera transition (get the camera modifier key that we are going to) plSceneObject* pObj = trans->fTransTo; const plCameraModifier1* pCamMod = nil; if (pObj) { for (int i = 0; i < pObj->GetNumModifiers(); i++) { pCamMod = plCameraModifier1::ConvertNoRef(pObj->GetModifier(i)); if (pCamMod) break; } } CamTrans* camTrans = nil; if (!pCamMod) camTrans = new CamTrans(nil); else camTrans = new CamTrans(pCamMod->GetKey()); camTrans->fAccel = trans->fAccel; camTrans->fDecel = trans->fDecel; camTrans->fVelocity = trans->fVelocity; camTrans->fPOAAccel = trans->fPOAAccel; camTrans->fPOADecel = trans->fPOADecel; camTrans->fPOAVelocity = trans->fPOAVelocity; camTrans->fCutPOA = trans->fCutPOA; camTrans->fCutPos = trans->fCutPos; camTrans->fIgnore = trans->fIgnore; fModKeys[node]->AddTrans(camTrans); idx++; } } } return true; } plCameraModifier1* plCameraBaseComponent::ICreateCameraModifier(plMaxNode* pNode, plErrorMsg* pErrMsg) { GenCamera* theCam = nil; Object* obj = pNode->EvalWorldState(hsConverterUtils::Instance().GetTime(pNode->GetInterface())).obj; TimeValue Now = hsConverterUtils::Instance().GetTime(pNode->GetInterface()); if (obj->ConvertToType(Now, Class_ID(LOOKAT_CAM_CLASS_ID, 0))) theCam = (GenCamera *) obj->ConvertToType(Now, Class_ID(LOOKAT_CAM_CLASS_ID, 0)); else { return nil; } pNode->SetDrawable(false); pNode->SetForceLocal(true); pNode->SetItinerant(true); // create run-time objects plCameraModifier1* pMod = new plCameraModifier1; plKey modifierKey = hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pMod, pNode->GetLocation()); hsgResMgr::ResMgr()->AddViaNotify(modifierKey, new plObjRefMsg(pNode->GetSceneObject()->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kModifier), plRefFlags::kActiveRef); obj = pNode->EvalWorldState(hsConverterUtils::Instance().GetTime(pNode->GetInterface())).obj; hsVector3 pt; float FOVvalue= 0.0; //Currently in Radians // radians FOVvalue = theCam->GetFOV(Now); // convert FOVvalue = FOVvalue*(180/3.141592); int FOVType = theCam->GetFOVType(); float wDeg, hDeg; switch(FOVType) { case 0: // FOV_W { wDeg = FOVvalue; hDeg = (wDeg*3)/4; } break; case 1: // FOV_H { hDeg = FOVvalue; wDeg = (hDeg*4)/3; } break; } pMod->SetFOVw(wDeg); pMod->SetFOVh(hDeg); pNode->AddModifier(pMod, IGetUniqueName(pNode)); return pMod; } void plCameraBaseComponent::ISetLimitPan(plMaxNode* pNode, plCameraBrain1* pBrain) { plComponentBase* LimitPanComp = 0; for (uint32_t x = 0; x < pNode->NumAttachedComponents(); x++) { plComponentBase *comp = pNode->GetAttachedComponent(x); if (comp->ClassID() == LIMITPAN_CID) LimitPanComp = comp; } if (LimitPanComp) { // set this camera to limit panning x degrees IParamBlock2* pBlk = LimitPanComp->GetParamBlock(plComponentBase::kRefComp); if ( pBlk && pBlk->GetInt(kLimitPanX) ) { float deg = pBlk->GetFloat(kPanZDeg); float rad = hsDegreesToRadians(deg); pBrain->SetXPanLimit( rad * 0.5f ); } if ( pBlk && pBlk->GetInt(kLimitPanZ) ) { float deg = pBlk->GetFloat(kPanXDeg); float rad = hsDegreesToRadians(deg); pBrain->SetZPanLimit( rad * 0.5f ); } } } void plCameraBaseComponent::ISetLimitZoom(plMaxNode* pNode, plCameraBrain1* pBrain) { plComponentBase* LimitZoomComp = 0; for (uint32_t x = 0; x < pNode->NumAttachedComponents(); x++) { plComponentBase *comp = pNode->GetAttachedComponent(x); if (comp->ClassID() == CAMERAZOOM_CID) LimitZoomComp = comp; } if (LimitZoomComp) { // set this camera to limit panning x degrees IParamBlock2* pBlk = LimitZoomComp->GetParamBlock(plComponentBase::kRefComp); float max = pBlk->GetFloat(kZoomMaxDeg); float min = pBlk->GetFloat(kZoomMinDeg); float rate = pBlk->GetFloat(kZoomRate); pBrain->SetZoomParams(max / 1.33333333, min / 1.33333333, rate); } } void plCameraBaseComponent::ISetIgnoreSubworld(plMaxNode* pNode, plCameraBrain1* pBrain) { plComponentBase* subComp = 0; for (uint32_t x = 0; x < pNode->NumAttachedComponents(); x++) { plComponentBase *comp = pNode->GetAttachedComponent(x); if (comp->ClassID() == CAM_IGNORE_SUB_CID) subComp = comp; } if (subComp) { // set this camera to ignore subworld movement pBrain->SetFlags(plCameraBrain1::kIgnoreSubworldMovement); } } plCameraModifier1* plCameraBaseComponent::ICreateFocalPointObject(plMaxNode* pNode, plErrorMsg* pErrMsg) { plMaxNode* node = ((plMaxNode*)pNode->GetTarget()); plCameraModifier1* pPOAMod = new plCameraModifier1; plKey poaModifierKey = hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pPOAMod, node->GetLocation()); hsgResMgr::ResMgr()->AddViaNotify(poaModifierKey, new plObjRefMsg(node->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kModifier), plRefFlags::kActiveRef); plCameraBrain1* pPOABrain = new plCameraBrain1(pPOAMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pPOABrain, pNode->GetLocation()); plGenRefMsg* pGRMsg = new plGenRefMsg(pPOAMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pGRMsg->SetRef( (hsKeyedObject*)pPOABrain ); plConvert::Instance().AddMessageToQueue(pGRMsg); return (pPOAMod); } bool plCameraBaseComponent::ISetPOA(plMaxNode* pNode, plCameraBrain1* pBrain, plErrorMsg* pErrMsg) { // do we want a special POA for this brain bool bResult = false; bool bAvPOA = false; plComponentBase* POAComp = 0; bool bPOAObject = false; plComponentBase* objPOAComp = 0; for (uint32_t x = 0; x < pNode->NumAttachedComponents(); x++) { plComponentBase *comp = pNode->GetAttachedComponent(x); if (comp->ClassID() == OBJECT_POA_CID) { if (objPOAComp != 0 || POAComp != 0) { pErrMsg->Set(true, "Export Error - Cameras", "Object %s : Cameras must have one and only one POA component!\n", pNode->GetName()).Show(); pErrMsg->Set(false); return false; } objPOAComp = comp; bPOAObject = true; } if (comp->ClassID() == AVATAR_POA_CID) { if (objPOAComp != 0 || POAComp != 0) { pErrMsg->Set(true, "Export Error - Cameras", "Object %s : Cameras must have one and only one POA component!\n", pNode->GetName()).Show(); pErrMsg->Set(false); return false; } bAvPOA = true; POAComp = comp; } } if (bPOAObject) { IParamBlock2* pBlk = objPOAComp->GetParamBlock(plComponentBase::kRefComp); if (!pBlk) return false; plKey fCamKey = ((plPOAObjectComponent*)objPOAComp)->GetObjectKey(); if (fCamKey) { pBrain->SetSubject(plSceneObject::ConvertNoRef(fCamKey->GetObjectPtr())); if (!plCameraBrain1_Avatar::ConvertNoRef(pBrain)) { pBrain->SetFlags(plCameraBrain1::kCutPOA); pBrain->SetFlags(plCameraBrain1::kCutPos); } hsVector3 pt; pt.fX = pBlk->GetFloat(kPOAObjOffsetX); pt.fY = pBlk->GetFloat(kPOAObjOffsetY); pt.fZ = pBlk->GetFloat(kPOAObjOffsetZ); pBrain->SetPOAOffset(pt); if (pBlk->GetInt(kPOAObjWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePOA); bResult = true; } else { pErrMsg->Set(true, "Export Error - Cameras", "Object POA component of camera %s has no object specified!\n", pNode->GetName()).Show(); pErrMsg->Set(false); return false; } } if (bAvPOA) { pBrain->SetFlags(plCameraBrain1::kFollowLocalAvatar); IParamBlock2* pBlk = POAComp->GetParamBlock(plComponentBase::kRefComp); if (!pBlk) return false; hsVector3 pt; pt.fX = pBlk->GetFloat(kAvPOAOffX); pt.fY = pBlk->GetFloat(kAvPOAOffY); pt.fZ = pBlk->GetFloat(kAvPOAOffZ); if (pBlk->GetInt(kAvPOAWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePOA); pBrain->SetPOAOffset(pt); bResult = true; } return bResult; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // Camera1Component // // the fixed camera component for the new camera code.. // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plCamera1Component, gCamera1Desc, "FixedCamera", "Fixed Camera", COMP_TYPE_CAMERA, FIXEDCAM_CID) //Max paramblock2 stuff below. ParamBlockDesc2 gCamera1Bk ( 1, _T("camera"), 0, &gCamera1Desc, P_AUTO_CONSTRUCT, plComponent::kRefComp, end ); plCamera1Component::plCamera1Component() { fClassDesc = &gCamera1Desc; fClassDesc->MakeAutoParamBlocks(this); } bool plCamera1Component::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; //this is a fixed camera using the built-in target plCameraBrain1_Fixed* pBrain = new plCameraBrain1_Fixed(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); plGenRefMsg* pMsg = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg); pBrain->SetFlags(plCameraBrain1::kCutPOA); pBrain->SetFlags(plCameraBrain1::kCutPos); if (!ISetPOA(pNode, pBrain, pErrMsg)) pBrain->SetTargetPoint(ICreateFocalPointObject(pNode, pErrMsg)); ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); fModKeys[pNode] = pMod; return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // IgnoreSubworldComponent // // tells the camera to ignore all subworld matrix movement // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plCameraIgnoreSub, gCameraIgnoreSubDesc, "IgnoreSubworldMovement", "Ignore Subworld Movement", COMP_TYPE_CAMERA, CAM_IGNORE_SUB_CID) //Max paramblock2 stuff below. ParamBlockDesc2 gCameraIgnoreSubBk ( 1, _T("camera"), 0, &gCameraIgnoreSubDesc, P_AUTO_CONSTRUCT, plComponent::kRefComp, end ); plCameraIgnoreSub::plCameraIgnoreSub() { fClassDesc = &gCameraIgnoreSubDesc; fClassDesc->MakeAutoParamBlocks(this); } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // AutoCamComponent // // the 3rd person camera component for the new camera code.. // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plAutoCamComponent, gAutoCameraDesc, "AutoCamera", "Auto Camera", COMP_TYPE_CAMERA, AUTOCAM_CID) enum { kAutoCamOffX, kAutoCamOffY, kAutoCamOffZ, kAutoCamWorldspace, kAutoCamLOS, kAutoCamSpeed, kAutoCamAccel, kAutoPOAOffX, kAutoPOAOffY, kAutoPOAOffZ, kAutoCamCut, kAutoCamPOAWorldspace, kAutoCamPosWorldspace, kAutoCamDecel, kAutoCamPOAAccel, kAutoCamPOADecel, kAutoCamPOASpeed, kAutoCamCutPOA, kAutoCamVelocity, kAutoCamVerticalInFall, kAutoCamFastRun, }; //Max paramblock2 stuff below. ParamBlockDesc2 gAutoCameraBk ( 1, _T("camera"), 0, &gAutoCameraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_AUTOCAM, IDS_COMP_AUTOCAM, 0, 0, NULL, kAutoCamOffX, _T("X Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kAutoCamOffY, _T("Y Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 10.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kAutoCamOffZ, _T("Z Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ, IDC_CAMERACMD_SPIN_OFFSETZ, SPIN_AUTOSCALE, end, kAutoPOAOffX, _T("PX Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX5, IDC_CAMERACMD_SPIN_OFFSETX5, SPIN_AUTOSCALE, end, kAutoPOAOffY, _T("PY Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY4, IDC_CAMERACMD_SPIN_OFFSETY4, SPIN_AUTOSCALE, end, kAutoPOAOffZ, _T("PZ Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ2, IDC_CAMERACMD_SPIN_OFFSETZ2, SPIN_AUTOSCALE, end, kAutoCamLOS, _T("maintainLOS"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS, end, kAutoCamCut, _T("cutPos"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS2, end, kAutoCamCutPOA, _T("cutPOA"), TYPE_BOOL, 0, 0, p_default, 1, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS3, end, kAutoCamPosWorldspace, _T("PosWorldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_POS_WORLDSPACE, end, kAutoCamPOAWorldspace, _T("POAworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_POA_WORLDSPACE, end, kAutoCamVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL, IDC_CAMERA_VEL_SPIN, SPIN_AUTOSCALE, end, kAutoCamAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL, IDC_CAMERA_ACCEL_SPIN, SPIN_AUTOSCALE, end, kAutoCamDecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL, IDC_CAMERA_DECEL_SPIN, SPIN_AUTOSCALE, end, kAutoCamPOASpeed, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL2, IDC_CAMERA_VEL_SPIN2, SPIN_AUTOSCALE, end, kAutoCamPOAAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL2, IDC_CAMERA_ACCEL_SPIN2, SPIN_AUTOSCALE, end, kAutoCamPOADecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL2, IDC_CAMERA_DECEL_SPIN2, SPIN_AUTOSCALE, end, kAutoCamVerticalInFall, _T("vertical"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_VERTICAL, end, kAutoCamFastRun, _T("runspeedup"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_RUN, end, end ); plAutoCamComponent::plAutoCamComponent() { fClassDesc = &gAutoCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plAutoCamComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; plCameraBrain1_Avatar* pBrain = new plCameraBrain1_Avatar(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); plGenRefMsg* pMsg = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg); hsVector3 pt; pt.Set(fCompPB->GetFloat(kAutoCamOffX),fCompPB->GetFloat(kAutoCamOffY),fCompPB->GetFloat(kAutoCamOffZ)); pBrain->SetOffset(pt); pt.Set(fCompPB->GetFloat(kAutoPOAOffX),fCompPB->GetFloat(kAutoPOAOffY),fCompPB->GetFloat(kAutoPOAOffZ)); pBrain->SetPOAOffset(pt); if (fCompPB->GetInt(kAutoCamPosWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePos); if (fCompPB->GetInt(kAutoCamPOAWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePOA); if (fCompPB->GetInt(kAutoCamLOS)) pBrain->SetFlags(plCameraBrain1::kMaintainLOS); if (fCompPB->GetInt(kAutoCamCut)) pBrain->SetFlags(plCameraBrain1::kCutPos); if (fCompPB->GetInt(kAutoCamCutPOA)) pBrain->SetFlags(plCameraBrain1::kCutPOA); if (fCompPB->GetInt(kAutoCamVerticalInFall)) pBrain->SetFlags(plCameraBrain1::kVerticalWhenFalling); if (fCompPB->GetInt(kAutoCamFastRun)) pBrain->SetFlags(plCameraBrain1::kSpeedUpWhenRunning); // We want a local Avatar POA for this brain pBrain->SetFlags(plCameraBrain1::kFollowLocalAvatar); fModKeys[pNode] = pMod; // set brain parameters ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); pBrain->SetAccel(fCompPB->GetFloat(kAutoCamAccel)); pBrain->SetDecel(fCompPB->GetFloat(kAutoCamDecel)); pBrain->SetVelocity(fCompPB->GetFloat(kAutoCamVelocity)); pBrain->SetPOAAccel(fCompPB->GetFloat(kAutoCamPOAAccel)); pBrain->SetPOADecel(fCompPB->GetFloat(kAutoCamPOADecel)); pBrain->SetPOAVelocity(fCompPB->GetFloat(kAutoCamPOASpeed)); return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // FPCamComponent // // the first-person camera component for the new camera code.. // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plFPCamComponent, gFPCameraDesc, "First Person Camera", "First Person Camera", COMP_TYPE_CAMERA, FPCAM_CID) enum { kFPCamOffX, kFPCamOffY, kFPCamOffZ, kFPCamPOAOffX, kFPCamPOAOffY, kFPCamPOAOffZ, kFpCamPOAWorldspace, kFpCamPosWorldspace, }; //Max paramblock2 stuff below. ParamBlockDesc2 gFPCameraBk ( 1, _T("camera"), 0, &gFPCameraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_FIRSTPERSON_CAM, IDS_COMP_FIRST_PERSONCAM, 0, 0, NULL, kFPCamOffX, _T("X Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kFPCamOffY, _T("Y Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 1.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kFPCamOffZ, _T("Z Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 6.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ, IDC_CAMERACMD_SPIN_OFFSETZ, SPIN_AUTOSCALE, end, kFPCamPOAOffX, _T("PX Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX5, IDC_CAMERACMD_SPIN_OFFSETX5, SPIN_AUTOSCALE, end, kFPCamPOAOffY, _T("PY Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY4, IDC_CAMERACMD_SPIN_OFFSETY4, SPIN_AUTOSCALE, end, kFPCamPOAOffZ, _T("PZ Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 6.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ2, IDC_CAMERACMD_SPIN_OFFSETZ2, SPIN_AUTOSCALE, end, kFpCamPOAWorldspace, _T("POAworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_POA_WORLDSPACE, end, kFpCamPosWorldspace, _T("Posworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_POS_WORLDSPACE, end, end ); plFPCamComponent::plFPCamComponent() { fClassDesc = &gFPCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plFPCamComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; plCameraBrain1_FirstPerson* pBrain = new plCameraBrain1_FirstPerson(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); plGenRefMsg* pMsg2 = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg2->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg2); hsVector3 pt; pt.Set(fCompPB->GetFloat(kFPCamOffX),fCompPB->GetFloat(kFPCamOffY),fCompPB->GetFloat(kFPCamOffZ)); pBrain->SetOffset(pt); pt.Set(fCompPB->GetFloat(kFPCamPOAOffX),fCompPB->GetFloat(kFPCamPOAOffY),fCompPB->GetFloat(kFPCamPOAOffZ)); pBrain->SetPOAOffset(pt); // We want a local Avatar POA for this brain pBrain->SetFlags(plCameraBrain1::kFollowLocalAvatar); // and we don't want any lag at all. pBrain->SetFlags(plCameraBrain1::kCutPos); pBrain->SetFlags(plCameraBrain1::kCutPOA); fModKeys[pNode] = pMod; if (fCompPB->GetInt(kFpCamPOAWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePOA); if (fCompPB->GetInt(kFpCamPosWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePos); ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // RailCameraComponent // // the fixed camera component for the new camera code.. // // // // // // // // // // enum { kRailCamObj, kRailFarthest, kRailSpeed, kRailAccel, kRailDecel, kRailPOASpeed, kRailPOAAccel, kRailPOADecel, kRailVelocity, }; CLASS_DESC(plRailCameraComponent, gRailCameraDesc, "Rail Camera", "Rail Camera", COMP_TYPE_CAMERA, RAIL_CAM_CID) ParamBlockDesc2 gRailCameraBk ( plComponent::kBlkComp, _T("RailCamera"), 0, &gRailCameraDesc, P_AUTO_CONSTRUCT+P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERA_RAIL, IDS_COMP_RAILCAMERA, 0, 0, 0, kRailCamObj, _T("PathObjectChoice"), TYPE_INODE, 0, 0, p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_LINE_CHOOSE_PATH, end, kRailFarthest, _T("farthest"), TYPE_BOOL, 0, 0, p_default, FALSE, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK1, end, kRailVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 5.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL, IDC_CAMERA_VEL_SPIN, SPIN_AUTOSCALE, end, kRailAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 5.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL, IDC_CAMERA_ACCEL_SPIN, SPIN_AUTOSCALE, end, kRailDecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 5.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL, IDC_CAMERA_DECEL_SPIN, SPIN_AUTOSCALE, end, kRailPOASpeed, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL2, IDC_CAMERA_VEL_SPIN2, SPIN_AUTOSCALE, end, kRailPOAAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL2, IDC_CAMERA_ACCEL_SPIN2, SPIN_AUTOSCALE, end, kRailPOADecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL2, IDC_CAMERA_DECEL_SPIN2, SPIN_AUTOSCALE, end, end ); plRailCameraComponent::plRailCameraComponent() : fValid(false), fLineMod(nil) { fClassDesc = &gRailCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plRailCameraComponent::IMakeLineMod(plMaxNode* pNode, plErrorMsg* pErrMsg) { plRailCameraMod* lineMod = new plRailCameraMod; hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), lineMod, pNode->GetLocation()); lineMod->SetFollowMode(plLineFollowMod::kFollowLocalAvatar); plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kRailCamObj); if(!pathNode) { pErrMsg->Set(true, "Path Node Failure", "Path Node %s was set to be Ignored or empty. Path Component ignored.", ((INode*)pathNode)->GetName()).Show(); pErrMsg->Set(false); return false; } //hsAssert(pathNode, "If valid is true, this must be set"); Control* maxTm = pathNode->GetTMController(); plCompoundController* tmc = plCompoundController::ConvertNoRef(hsControlConverter::Instance().MakeTransformController(maxTm, pathNode)); if( !(tmc && tmc->GetPosController() && !plCompoundController::ConvertNoRef(tmc->GetPosController())) ) { delete tmc; pErrMsg->Set(true, pNode->GetName(), "Rail Camera Path Node %s has no suitable animation. Rail Camera ignored", pathNode->GetName()).Show(); pErrMsg->Set(false); return false; } Matrix3 w2p(true); Matrix3 l2w = pathNode->GetNodeTM(TimeValue(0)); if( !pathNode->GetParentNode()->IsRootNode() ) w2p = Inverse(pathNode->GetParentNode()->GetNodeTM(TimeValue(0))); hsMatrix44 loc2Par = pathNode->Matrix3ToMatrix44(l2w * w2p); gemAffineParts ap; decomp_affine(loc2Par.fMap, &ap); hsAffineParts initParts; AP_SET(initParts, ap); plAnimPath* animPath = new plAnimPath; animPath->SetController(tmc); animPath->InitParts(initParts); animPath->SetFarthest(fCompPB->GetInt(kRailFarthest)); lineMod->SetPath(animPath); if( !pathNode->GetParentNode()->IsRootNode() ) { plMaxNode* parNode = (plMaxNode*)pathNode->GetParentNode(); plSceneObject* parObj = parNode->GetSceneObject(); if( parObj ) { plGenRefMsg* refMsg = new plGenRefMsg(lineMod->GetKey(), plRefMsg::kOnCreate, 0, plLineFollowMod::kRefParent); hsgResMgr::ResMgr()->AddViaNotify(parObj->GetKey(), refMsg, plRefFlags::kPassiveRef); } } plLeafController *controller = plLeafController::ConvertNoRef(tmc->GetPosController()); hsPoint3 start, end; controller->Interp(0, &start); controller->Interp(controller->GetLength(), &end); animPath->SetWrap(start == end); lineMod->SetForceToLine(true); fLineMod = lineMod; return true; } bool plRailCameraComponent::SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg) { if (!plCameraBaseComponent::SetupProperties(pNode, pErrMsg)) return false; fValid = false; fLineMod = nil; if( !fCompPB->GetINode(kRailCamObj) ) { return true; } plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kRailCamObj); if( !pathNode ) { return true; } if( !pathNode->IsTMAnimated() ) { return true; } pathNode->SetCanConvert(false); fValid = true; pNode->SetForceLocal(true); pNode->SetMovable(true); return true; } bool plRailCameraComponent::PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; //this is a fixed camera using the built-in target plCameraBrain1_Fixed* pBrain = new plCameraBrain1_Fixed(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); // make the rail cam have pretty slow acceleration pBrain->SetAccel(5.0f); pBrain->SetDecel(5.0f); plGenRefMsg* pMsg = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg); pBrain->SetVelocity(fCompPB->GetFloat(kRailVelocity)); pBrain->SetAccel(fCompPB->GetFloat(kRailAccel)); pBrain->SetDecel(fCompPB->GetFloat(kRailDecel)); // need to cap these in legacy datasets if (pBrain->GetAccel() > 10.0f) pBrain->SetAccel(10.0f); if (pBrain->GetDecel() > 10.0f) pBrain->SetDecel(10.0f); if (pBrain->GetVelocity() > 10.0f) pBrain->SetVelocity(10.0f); pBrain->SetPOAVelocity(fCompPB->GetFloat(kRailPOASpeed)); pBrain->SetPOAAccel(fCompPB->GetFloat(kRailPOAAccel)); pBrain->SetPOADecel(fCompPB->GetFloat(kRailPOADecel)); if (!ISetPOA(pNode, pBrain, pErrMsg)) pBrain->SetTargetPoint(ICreateFocalPointObject(pNode, pErrMsg)); ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); fModKeys[pNode] = pMod; // rail camera part plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kRailCamObj); if( !pathNode ) { pErrMsg->Set(true, "Invald Rail Camera", "Rail Camera component on %s. has no path object selected. This component will not be exported.\n", ((INode*)pNode)->GetName()).Show(); pErrMsg->Set(false); return false; } if( !fLineMod ) { if( !IMakeLineMod(pNode, pErrMsg) ) { fValid = false; return true; } } pNode->AddModifier(fLineMod, IGetUniqueName(pNode)); pBrain->SetRail(fLineMod); return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // Circle camera Component // // a circle camera component for the new camera code.. // // // // // // // // // // enum { kCircleFarthest, kCircleSpeed, kCirclePOASpeed, kCirclePOAAccel, kCirclePOADecel, kCirclePOAVelocity, }; CLASS_DESC(plCircleCameraComponent, gCircleCameraDesc, "Circle Camera", "Circle Camera", COMP_TYPE_CAMERA, CIRCLE_CAM_CID) ParamBlockDesc2 gCircleCameraBk ( plComponent::kBlkComp, _T("CircleCamera"), 0, &gCircleCameraDesc, P_AUTO_CONSTRUCT+P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERA_CIRCLE, IDS_COMP_CIRCLECAMERA, 0, 0, 0, kCircleFarthest, _T("farthest"), TYPE_BOOL, 0, 0, p_default, FALSE, p_ui, TYPE_SINGLECHEKBOX, IDC_CHECK1, end, kCircleSpeed, _T("Lag Scale"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.1f, 1.0f, p_default, 0.1f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX3, IDC_CAMERACMD_SPIN_OFFSETX3, SPIN_AUTOSCALE, end, kCirclePOAVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL, IDC_CAMERA_VEL_SPIN, SPIN_AUTOSCALE, end, kCirclePOAAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL, IDC_CAMERA_ACCEL_SPIN, SPIN_AUTOSCALE, end, kCirclePOADecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL, IDC_CAMERA_DECEL_SPIN, SPIN_AUTOSCALE, end, end ); plCircleCameraComponent::plCircleCameraComponent() : fValid(false) { fClassDesc = &gCircleCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plCircleCameraComponent::PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; //this is a circle camera using the built-in target plCameraBrain1_Circle* pBrain = new plCameraBrain1_Circle(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); // make the circle cam have pretty slow acceleration pBrain->SetAccel(10.0f); pBrain->SetDecel(10.0f); pBrain->SetVelocity(15.0f); // Stay away from / close to the local avatar pBrain->SetFlags(plCameraBrain1::kFollowLocalAvatar); if (fCompPB->GetInt(kCircleFarthest)) pBrain->SetFarCircleCam(true); plGenRefMsg* pMsg = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg); pBrain->SetPOAVelocity(fCompPB->GetFloat(kCirclePOAVelocity)); pBrain->SetPOAAccel(fCompPB->GetFloat(kCirclePOAAccel)); pBrain->SetPOADecel(fCompPB->GetFloat(kCirclePOADecel)); if (fCompPB->GetFloat(kCircleSpeed)) pBrain->SetCircumferencePerSec(fCompPB->GetFloat(kCircleSpeed)); if (!ISetPOA(pNode, pBrain, pErrMsg)) pBrain->SetTargetPoint(ICreateFocalPointObject(pNode, pErrMsg)); else pBrain->SetCircleFlags(plCameraBrain1_Circle::kCircleLocalAvatar); // set radius and center point hsPoint3 point, point2; TimeValue Now = hsConverterUtils::Instance().GetTime(pNode->GetInterface()); Matrix3 ReturnMatrix = pNode->GetTarget()->GetNodeTM(Now); Point3 ReturnVal = ReturnMatrix.GetRow(3); point.fX = ReturnVal.x; point.fY = ReturnVal.y; point.fZ = ReturnVal.z; ReturnMatrix = pNode->GetLocalToWorld(Now); ReturnVal = ReturnMatrix.GetRow(3); point2.fX = ReturnVal.x; point2.fY = ReturnVal.y; point2.fZ = ReturnVal.z; hsVector3 vec(point - point2); pBrain->SetRadius(vec.Magnitude()); pBrain->SetCenter(&point); ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); fModKeys[pNode] = pMod; return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // the detector for triggering camera changes /////////////////////////////////////////////////////// class plCameraDetectorComponent : public plPhysicCoreComponent { public: plCameraDetectorComponent(); void DeleteThis() { delete this; } bool SetupProperties(plMaxNode *node, plErrorMsg *pErrMsg); bool Convert(plMaxNode *node, plErrorMsg *pErrMsg); bool PreConvert(plMaxNode *node, plErrorMsg *pErrMsg); virtual void CollectNonDrawables(INodeTab& nonDrawables) { AddTargetsToList(nonDrawables); } }; CLASS_DESC(plCameraDetectorComponent, gCameraDetectorDesc, "Camera Region", "CameraRegion", COMP_TYPE_CAMERA, CAM_REGION_CID) enum { kCameraTarget, }; ParamBlockDesc2 gCameraRegionBlock ( plComponent::kBlkComp, _T("cameraRegion"), 0, &gCameraDetectorDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERARGN, IDS_CAMERARGN, 0, 0, NULL, kCameraTarget, _T("CameraTarget"), TYPE_INODE, 0, 0, p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_CAMERARGN_PICKSTATE_BASE, p_sclassID, CAMERA_CLASS_ID, p_prompt, IDS_COMP_PHYS_CHOSEN_BASE, end, end ); plCameraDetectorComponent::plCameraDetectorComponent() { fClassDesc = &gCameraDetectorDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plCameraDetectorComponent::SetupProperties(plMaxNode *pNode, plErrorMsg *pErrMsg) { pNode->SetForceLocal(true); pNode->SetDrawable(false); plPhysicalProps *physProps = pNode->GetPhysicalProps(); physProps->SetPinned(true, pNode, pErrMsg); // only if movable will it have mass (then it will keep track of movements in PhysX) if ( pNode->IsMovable() || pNode->IsTMAnimatedRecur() ) physProps->SetMass(1.0, pNode, pErrMsg); physProps->SetFriction(0.0, pNode, pErrMsg); physProps->SetRestitution(0.0, pNode, pErrMsg); physProps->SetBoundsType(plSimDefs::kHullBounds, pNode, pErrMsg); physProps->SetGroup(plSimDefs::kGroupDetector, pNode, pErrMsg); physProps->SetReportGroup(1<<plSimDefs::kGroupAvatar, pNode, pErrMsg); /// physProps->SetAllowLOS(true, pNode, pErrMsg); return true; } bool plCameraDetectorComponent::PreConvert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } bool plCameraDetectorComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { plSceneObject *obj = node->GetSceneObject(); plLocation loc = node->GetLocation(); plCameraRegionDetector *detector = new plCameraRegionDetector; // Register the detector plKey detectorKey = hsgResMgr::ResMgr()->NewKey(IGetUniqueName(node), detector, loc); hsgResMgr::ResMgr()->AddViaNotify(detectorKey, new plObjRefMsg(obj->GetKey(), plRefMsg::kOnCreate, -1, plObjRefMsg::kModifier), plRefFlags::kActiveRef); plCameraMsg* pMsg = new plCameraMsg; pMsg->SetBCastFlag(plMessage::kBCastByType); // need to get the key for the camera here... plMaxNode* pCamNode = (plMaxNode*)fCompPB->GetINode(kCameraTarget); if (pCamNode) { if(pCamNode->CanConvert()) { pMsg->SetCmd(plCameraMsg::kRegionPushCamera); pMsg->SetNewCam(((plMaxNode*)pCamNode)->GetSceneObject()->GetKey()); int count = ((plMaxNode*)pCamNode)->NumAttachedComponents(); for (uint32_t x = 0; x < count; x++) { plComponentBase *comp = ((plMaxNode*)pCamNode)->GetAttachedComponent(x); if (comp->ClassID() == DEFAULTCAM_CID) { pMsg->SetCmd(plCameraMsg::kSetAsPrimary); break; } } } else { pErrMsg->Set(true, "Improper Cam Region Selection", "Cam Choice %s was set to be Ignored. No Camera selected.", ((INode*)pCamNode)->GetName()); pErrMsg->Set(false); return false; } } else { pErrMsg->Set(true, "Camera Region", "Camera Region %s has no camera assigned to it.", ((INode*)node)->GetName()).Show(); pErrMsg->Set(false); return false; } detector->AddMessage(pMsg); return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // Object Follow cam Component // // the 3rd person camera component for objects other than the avatar // // // // // // // // // // //Max desc stuff necessary. CLASS_DESC(plFollowCamComponent, gFollowCameraDesc, "FollowCamera", "Object Follow Camera", COMP_TYPE_CAMERA, FOLLOWCAM_CID) enum { kFollowCamOffX, kFollowCamOffY, kFollowCamOffZ, kFollowCamWorldspace, kFollowCamLOS, kFollowCamSpeed, kFollowCamAccel, kFollowPOAOffX, kFollowPOAOffY, kFollowPOAOffZ, kFollowCamCut, kFollowCamPOAWorldspace, kFollowCamPosWorldspace, kFollowCamDecel, kFollowCamPOAAccel, kFollowCamPOADecel, kFollowCamPOASpeed, kFollowCamCutPOA, kFollowCamVelocity, }; //Max paramblock2 stuff below. ParamBlockDesc2 gFollowCameraBk ( 1, _T("camera"), 0, &gFollowCameraDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_OBJECT_FOLLOWCAM, IDS_COMP_OBJECT_FOLLOWCAM, 0, 0, NULL, kFollowCamOffX, _T("X Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kFollowCamOffY, _T("Y Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 10.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kFollowCamOffZ, _T("Z Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ, IDC_CAMERACMD_SPIN_OFFSETZ, SPIN_AUTOSCALE, end, kFollowPOAOffX, _T("PX Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX5, IDC_CAMERACMD_SPIN_OFFSETX5, SPIN_AUTOSCALE, end, kFollowPOAOffY, _T("PY Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY4, IDC_CAMERACMD_SPIN_OFFSETY4, SPIN_AUTOSCALE, end, kFollowPOAOffZ, _T("PZ Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -50.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ2, IDC_CAMERACMD_SPIN_OFFSETZ2, SPIN_AUTOSCALE, end, kFollowCamLOS, _T("maintainLOS"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS, end, kFollowCamCut, _T("cutPos"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS2, end, kFollowCamCutPOA, _T("cutPOA"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_LOS3, end, kFollowCamPosWorldspace, _T("PosWorldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_POS_WORLDSPACE, end, kFollowCamPOAWorldspace, _T("POAworldspace"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_AUTOCAM_POA_WORLDSPACE, end, kFollowCamVelocity, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL, IDC_CAMERA_VEL_SPIN, SPIN_AUTOSCALE, end, kFollowCamAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL, IDC_CAMERA_ACCEL_SPIN, SPIN_AUTOSCALE, end, kFollowCamDecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL, IDC_CAMERA_DECEL_SPIN, SPIN_AUTOSCALE, end, kFollowCamPOASpeed, _T("Velocity"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_VEL2, IDC_CAMERA_VEL_SPIN2, SPIN_AUTOSCALE, end, kFollowCamPOAAccel, _T("Accel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_ACCEL2, IDC_CAMERA_ACCEL_SPIN2, SPIN_AUTOSCALE, end, kFollowCamPOADecel, _T("Decel"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, -100.0f, 100.0f, p_default, 60.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERA_DECEL2, IDC_CAMERA_DECEL_SPIN2, SPIN_AUTOSCALE, end, end ); plFollowCamComponent::plFollowCamComponent() { fClassDesc = &gFollowCameraDesc; fClassDesc->MakeAutoParamBlocks(this); } bool plFollowCamComponent::PreConvert(plMaxNode *pNode, plErrorMsg *pErrMsg) { plCameraModifier1* pMod = ICreateCameraModifier(pNode, pErrMsg); if (!pMod) return false; plCameraBrain1_Avatar* pBrain = new plCameraBrain1_Avatar(pMod); // Give the brain a key hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), pBrain, pNode->GetLocation()); plGenRefMsg* pMsg = new plGenRefMsg(pMod->GetKey(), plRefMsg::kOnCreate, -1, 0); pMsg->SetRef( (hsKeyedObject*)pBrain ); plConvert::Instance().AddMessageToQueue(pMsg); hsVector3 pt; pt.Set(fCompPB->GetFloat(kFollowCamOffX),fCompPB->GetFloat(kFollowCamOffY),fCompPB->GetFloat(kFollowCamOffZ)); pBrain->SetOffset(pt); pt.Set(fCompPB->GetFloat(kFollowPOAOffX),fCompPB->GetFloat(kFollowPOAOffY),fCompPB->GetFloat(kFollowPOAOffZ)); pBrain->SetPOAOffset(pt); if (fCompPB->GetInt(kFollowCamPosWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePos); if (fCompPB->GetInt(kFollowCamPOAWorldspace)) pBrain->SetFlags(plCameraBrain1::kWorldspacePOA); if (fCompPB->GetInt(kFollowCamLOS)) pBrain->SetFlags(plCameraBrain1::kMaintainLOS); if (fCompPB->GetInt(kFollowCamCut)) pBrain->SetFlags(plCameraBrain1::kCutPos); if (fCompPB->GetInt(kFollowCamCutPOA)) pBrain->SetFlags(plCameraBrain1::kCutPOA); fModKeys[pNode] = pMod; if (!ISetPOA(pNode, pBrain, pErrMsg)) { if(pErrMsg->Set(true, "Invalid Object Follow Camera", "The camera %s does NOT have an Object POA to go with its Object Follow Cam Component. This camera will be disabled..\nKill the export?",((INode*)pNode)->GetName()).Ask()) pErrMsg->Set(true, "", ""); else pErrMsg->Set(false); // Don't want to abort } // set brain parameters ISetLimitPan(pNode, pBrain); ISetLimitZoom(pNode, pBrain); ISetIgnoreSubworld(pNode, pBrain); pBrain->SetAccel(fCompPB->GetFloat(kFollowCamAccel)); pBrain->SetDecel(fCompPB->GetFloat(kFollowCamDecel)); pBrain->SetVelocity(fCompPB->GetFloat(kFollowCamVelocity)); pBrain->SetPOAAccel(fCompPB->GetFloat(kFollowCamPOAAccel)); pBrain->SetPOADecel(fCompPB->GetFloat(kFollowCamPOADecel)); pBrain->SetPOAVelocity(fCompPB->GetFloat(kFollowCamPOASpeed)); return true; } /////////////////////////////////////////////////////// /////////////////////////////////////////////////////// // // // animated camera command component - // does nothing if you haven't put an animation component on the camera this is on // // // // // // // // CLASS_DESC(plCameraAnimCmdComponent, gAnimcamCmdDesc, "Animated Camera Commands", "AnimatedCameraCmd", COMP_TYPE_CAMERA, ANIMCAM_CMD_CID) enum { kAnimateOnPush, kStopOnPop, kResetOnPop, kIgnoreFOV }; ParamBlockDesc2 gAnimcamCmdBlock ( plComponent::kBlkComp, _T("animCamCmd"), 0, &gAnimcamCmdDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_ANIMCAM_COMMANDS, IDS_COMP_ANIM_CAM_CMD, 0, 0, NULL, kAnimateOnPush, _T("animOnPush"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_BEGINONPUSH, end, kStopOnPop, _T("stopOnPop"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_STOPONPOP, end, kResetOnPop, _T("resetOnPop"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_RESETONPOP, end, kIgnoreFOV, _T("ignoreFOV"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_IGNOREFOV, end, end ); plCameraAnimCmdComponent::plCameraAnimCmdComponent() { fClassDesc = &gAnimcamCmdDesc; fClassDesc->MakeAutoParamBlocks(this); fIgnoreFOV = false; } bool plCameraAnimCmdComponent::PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg) { fIgnoreFOV = fCompPB->GetInt(kIgnoreFOV); return true; } bool plCameraAnimCmdComponent::Convert(plMaxNode* pNode, plErrorMsg* pErrMsg) { plSceneObject* pObj = pNode->GetSceneObject(); const plCameraModifier1* pCamMod = nil; if (pObj) { for (int i = 0; i < pObj->GetNumModifiers(); i++) { pCamMod = plCameraModifier1::ConvertNoRef(pObj->GetModifier(i)); if (pCamMod) break; } if (!pCamMod) return false; // forgive me oh const-ness gods. it is only the exporter, after all... const_cast<plCameraModifier1*>(pCamMod)->SetAnimCommands( fCompPB->GetInt(kAnimateOnPush), fCompPB->GetInt(kStopOnPop), fCompPB->GetInt(kResetOnPop) ); } return true; } // obsolete camera components: #define CAMERACMD_CID Class_ID(0x6edb72d1, 0xd8a1f43) class plCameraCmdComponent : public plComponent { public: protected: public: plCameraCmdComponent(); // Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool SetupProperties(plMaxNode *node, plErrorMsg *pErrMsg); bool PreConvert(plMaxNode *node, plErrorMsg* pErrMsg); bool Convert(plMaxNode *node, plErrorMsg *pErrMsg); }; OBSOLETE_CLASS_DESC(plCameraCmdComponent, gCameraCmdDesc, "(ex)Camera Command Region", "CameraCmdRegion", COMP_TYPE_MISC, CAMERACMD_CID) enum { kCommand, kOffsetX, kOffsetY, kOffsetZ, kCustomBoundListStuff, kSmooth, }; class plCameraCmdComponentProc : public ParamMap2UserDlgProc { public: BOOL DlgProc(TimeValue t, IParamMap2 *map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam) { return false; } void DeleteThis() {} protected: void IEnableControls(IParamMap2 *map, int type) { } void IAddComboItem(HWND hCombo, const char *name, int id) { } void ISetComboSel(HWND hCombo, int type) { } }; static plCameraCmdComponentProc gCameraCmdComponentProc; enum { kCommandSetOffset, kCommandSetFP, kCommandSetFixedCam, }; ParamBlockDesc2 gCameraCmdBlock ( plComponent::kBlkComp, _T("cameraComp"), 0, &gCameraCmdDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp, IDD_COMP_CAMERACMD, IDS_COMP_CAMERACMD, 0, 0, &gCameraCmdComponentProc, kCommand, _T("Command"), TYPE_INT, 0, 0, p_default, kCommandSetFixedCam, end, kOffsetX, _T("X Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 50.0f, p_default, 0.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETX, IDC_CAMERACMD_SPIN_OFFSETX, SPIN_AUTOSCALE, end, kOffsetY, _T("Y Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 50.0f, p_default, 10.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETY, IDC_CAMERACMD_SPIN_OFFSETY, SPIN_AUTOSCALE, end, kOffsetZ, _T("Z Offset"), TYPE_FLOAT, P_ANIMATABLE, 0, p_range, 0.0f, 50.0f, p_default, 3.0f, p_ui, TYPE_SPINNER, EDITTYPE_FLOAT, IDC_CAMERACMD_OFFSETZ, IDC_CAMERACMD_SPIN_OFFSETZ, SPIN_AUTOSCALE, end, kCustomBoundListStuff, _T("FixedCamera"), TYPE_INODE, 0, 0, p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_CAMERACMD_PICKSTATE_BASE, p_sclassID, CAMERA_CLASS_ID, p_prompt, IDS_COMP_PHYS_CHOSEN_BASE, end, kSmooth, _T("useCut"), TYPE_BOOL, 0, 0, p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_CAMERACMD_CUT, end, end ); plCameraCmdComponent::plCameraCmdComponent() { fClassDesc = &gCameraCmdDesc; fClassDesc->MakeAutoParamBlocks(this); } // Internal setup and write-only set properties on the MaxNode. No reading // of properties on the MaxNode, as it's still indeterminant. bool plCameraCmdComponent::SetupProperties(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } bool plCameraCmdComponent::PreConvert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; } bool plCameraCmdComponent::Convert(plMaxNode *node, plErrorMsg *pErrMsg) { return true; }