|
|
|
/*==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 "max.h"
|
|
|
|
#include "resource.h"
|
|
|
|
#include "plComponent.h"
|
|
|
|
#include "plComponentReg.h"
|
|
|
|
|
|
|
|
#include "MaxMain/plMaxNode.h"
|
|
|
|
#include "MaxMain/plPlasmaRefMsgs.h"
|
|
|
|
|
|
|
|
#include "MaxConvert/hsConverterUtils.h"
|
|
|
|
#include "MaxConvert/hsControlConverter.h"
|
|
|
|
#include "plInterp/plController.h"
|
|
|
|
#include "MaxMain/plMaxNode.h"
|
|
|
|
#include "pnKeyedObject/plKey.h"
|
|
|
|
|
|
|
|
#include "plgDispatch.h"
|
|
|
|
#include "MaxMain/plPluginResManager.h"
|
|
|
|
|
|
|
|
#include "pnSceneObject/plSceneObject.h"
|
|
|
|
#include "pnSceneObject/plCoordinateInterface.h"
|
|
|
|
|
|
|
|
// Swivel related
|
|
|
|
#include "pfAnimation/plViewFaceModifier.h" // ViewFace Comp
|
|
|
|
|
|
|
|
// Line Follow related
|
|
|
|
#include "plInterp/plAnimPath.h"
|
|
|
|
#include "pfAnimation/plLineFollowMod.h"
|
|
|
|
#include "pfAnimation/plFollowMod.h"
|
|
|
|
|
|
|
|
#include "pnMessage/plRefMsg.h"
|
|
|
|
|
|
|
|
// Stereizer
|
|
|
|
#include "pfAnimation/plStereizer.h"
|
|
|
|
|
|
|
|
const Class_ID STEREIZE_COMP_CID(0x15066ec7, 0x64ea7381);
|
|
|
|
const Class_ID LINEFOLLOW_COMP_CID(0x64ec57f6, 0x292d47f6);
|
|
|
|
const Class_ID SWIVEL_COMP_CID(0x106a466b, 0x1c1700f7);
|
|
|
|
|
|
|
|
void DummyCodeIncludeFuncLineFollow()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// LineFollow Component
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
kFollowModeRadio,
|
|
|
|
kPathObjectSel,
|
|
|
|
kFollowObjectSel,
|
|
|
|
kOffsetActive,
|
|
|
|
kOffsetDegrees,
|
|
|
|
kOffsetClampActive,
|
|
|
|
kOffsetClamp,
|
|
|
|
kForceToLine,
|
|
|
|
kSpeedClampActive,
|
|
|
|
kSpeedClamp
|
|
|
|
};
|
|
|
|
|
|
|
|
// When one of our parameters that is a ref changes, send out the component ref
|
|
|
|
// changed message. Normally, messages from component refs are ignored since
|
|
|
|
// they pass along all the messages of the ref, which generates a lot of false
|
|
|
|
// converts.
|
|
|
|
class plLineObjAccessor : public PBAccessor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void Set(PB2Value& v, ReferenceMaker* owner, ParamID id, int tabIndex, TimeValue t)
|
|
|
|
{
|
|
|
|
if( (id == kFollowObjectSel) || (id == kPathObjectSel) )
|
|
|
|
{
|
|
|
|
plComponentBase* comp = (plComponentBase*)owner;
|
|
|
|
comp->NotifyDependents(FOREVER, PART_ALL, REFMSG_USER_COMP_REF_CHANGED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
plLineObjAccessor gLineObjAccessor;
|
|
|
|
|
|
|
|
|
|
|
|
class plLineFollowComponentProc : public ParamMap2UserDlgProc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
BOOL DlgProc(TimeValue t, IParamMap2* map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
switch (msg)
|
|
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
|
|
|
{
|
|
|
|
IParamBlock2* pb = map->GetParamBlock();
|
|
|
|
map->SetTooltip(kPathObjectSel, TRUE, "Press the button, & select the path source object in one of the Viewports" );
|
|
|
|
map->SetTooltip(kFollowObjectSel, TRUE, "Press the button, & select the object to follow in one of the Viewports" );
|
|
|
|
map->SetTooltip(kOffsetDegrees, TRUE, "Positive angle to right, negative to left." );
|
|
|
|
if( pb->GetInt(kFollowModeRadio) == plLineFollowMod::kFollowObject )
|
|
|
|
map->Enable(kFollowObjectSel, TRUE);
|
|
|
|
else
|
|
|
|
map->Enable(kFollowObjectSel, FALSE);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
//////////////////
|
|
|
|
case WM_COMMAND:
|
|
|
|
{
|
|
|
|
if( (LOWORD(wParam) == IDC_RADIO_LISTENER)
|
|
|
|
|| (LOWORD(wParam) == IDC_RADIO_CAMERA)
|
|
|
|
|| (LOWORD(wParam) == IDC_RADIO_OBJECT) )
|
|
|
|
{
|
|
|
|
IParamBlock2* pb = map->GetParamBlock();
|
|
|
|
if( pb->GetInt(kFollowModeRadio) == plLineFollowMod::kFollowObject )
|
|
|
|
map->Enable(kFollowObjectSel, TRUE);
|
|
|
|
else
|
|
|
|
map->Enable(kFollowObjectSel, FALSE);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
void DeleteThis() {}
|
|
|
|
};
|
|
|
|
static plLineFollowComponentProc gLineFollowProc;
|
|
|
|
|
|
|
|
|
|
|
|
//Class that accesses the paramblock below.
|
|
|
|
class plLineFollowComponent : public plComponent
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
hsBool fValid;
|
|
|
|
|
|
|
|
plLineFollowMod* fLineMod;
|
|
|
|
|
|
|
|
hsBool IMakeLineMod(plMaxNode* pNode, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
public:
|
|
|
|
plLineFollowComponent();
|
|
|
|
|
|
|
|
hsBool SetupProperties(plMaxNode* pNode, plErrorMsg* pErrMsg);
|
|
|
|
hsBool PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg);
|
|
|
|
hsBool Convert(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
plLineFollowMod* GetLineMod(plErrorMsg* pErrMsg);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CLASS_DESC(plLineFollowComponent, gLineFollowDesc, "LineFollow", "LineFollow", COMP_TYPE_GRAPHICS, LINEFOLLOW_COMP_CID)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ParamBlockDesc2 gLineFollowBk
|
|
|
|
(
|
|
|
|
plComponent::kBlkComp, _T("LineFollow"), 0, &gLineFollowDesc, P_AUTO_CONSTRUCT+P_AUTO_UI, plComponent::kRefComp,
|
|
|
|
|
|
|
|
IDD_COMP_LINEFOLLOW, IDS_COMP_LINEFOLLOWS, 0, 0, &gLineFollowProc,
|
|
|
|
|
|
|
|
kFollowModeRadio, _T("FollowMode"), TYPE_INT, 0, 0,
|
|
|
|
p_ui, TYPE_RADIO, 3, IDC_RADIO_LISTENER, IDC_RADIO_CAMERA, IDC_RADIO_OBJECT,
|
|
|
|
p_vals, plLineFollowMod::kFollowListener, plLineFollowMod::kFollowCamera, plLineFollowMod::kFollowObject,
|
|
|
|
p_default, plLineFollowMod::kFollowListener,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kPathObjectSel, _T("PathObjectChoice"), TYPE_INODE, 0, 0,
|
|
|
|
p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_LINE_CHOOSE_PATH,
|
|
|
|
p_prompt, IDS_COMP_LINE_CHOSE_PATH,
|
|
|
|
p_accessor, &gLineObjAccessor,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kFollowObjectSel, _T("FollowObjectChoice"), TYPE_INODE, 0, 0,
|
|
|
|
p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_LINE_CHOOSE_OBJECT,
|
|
|
|
p_prompt, IDS_COMP_LINE_CHOSE_OBJECT,
|
|
|
|
p_accessor, &gLineObjAccessor,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kOffsetActive, _T("OffsetActive"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_LINE_OFFSETACTIVE,
|
|
|
|
p_enable_ctrls, 4, kOffsetDegrees, kOffsetClampActive, kOffsetClamp, kForceToLine,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kOffsetDegrees, _T("OffsetDegrees"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 0.0,
|
|
|
|
p_range, -85.0, 85.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_FLOAT,
|
|
|
|
IDC_COMP_LINE_OFFSETDEGREES, IDC_COMP_LINE_OFFSETDEGREES_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kOffsetClampActive, _T("OffsetClampActive"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_LINE_OFFSETCLAMPACTIVE,
|
|
|
|
p_enable_ctrls, 1, kOffsetClamp,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kOffsetClamp, _T("OffsetClamp"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 0.0,
|
|
|
|
p_range, 0.0, 1000.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_LINE_OFFSETCLAMP, IDC_COMP_LINE_OFFSETCLAMP_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kForceToLine, _T("ForceToLine"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_LINE_FORCETOLINE,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kSpeedClampActive, _T("SpeedClampActive"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_LINE_SPEEDCLAMPACTIVE,
|
|
|
|
p_enable_ctrls, 1, kSpeedClamp,
|
|
|
|
end,
|
|
|
|
|
|
|
|
kSpeedClamp, _T("SpeedClamp"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 30.0,
|
|
|
|
p_range, 3.0, 1000.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_LINE_SPEEDCLAMP, IDC_COMP_LINE_SPEEDCLAMP_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
plLineFollowComponent::plLineFollowComponent()
|
|
|
|
: fValid(false),
|
|
|
|
fLineMod(nil)
|
|
|
|
{
|
|
|
|
fClassDesc = &gLineFollowDesc;
|
|
|
|
fClassDesc->MakeAutoParamBlocks(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plLineFollowComponent::IMakeLineMod(plMaxNode* pNode, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
plLineFollowMod::FollowMode mode = plLineFollowMod::FollowMode(fCompPB->GetInt(kFollowModeRadio));
|
|
|
|
|
|
|
|
plLineFollowMod* lineMod = TRACKED_NEW plLineFollowMod;
|
|
|
|
hsgResMgr::ResMgr()->NewKey(IGetUniqueName(pNode), lineMod, pNode->GetLocation());
|
|
|
|
|
|
|
|
if( plLineFollowMod::kFollowObject == mode )
|
|
|
|
{
|
|
|
|
if(fCompPB->GetINode(kFollowObjectSel) != NULL)
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
plMaxNode* targNode = (plMaxNode*)fCompPB->GetINode(kFollowObjectSel);
|
|
|
|
//plMaxNodeData* pMD = targNode->GetMaxNodeData();
|
|
|
|
if( targNode->CanConvert() )
|
|
|
|
{
|
|
|
|
plSceneObject* targObj = targNode->GetSceneObject();
|
|
|
|
if( targObj )
|
|
|
|
{
|
|
|
|
plGenRefMsg* refMsg = TRACKED_NEW plGenRefMsg(lineMod->GetKey(), plRefMsg::kOnCreate, 0, plLineFollowMod::kRefObject);
|
|
|
|
hsgResMgr::ResMgr()->AddViaNotify(targObj->GetKey(), refMsg, plRefFlags::kPassiveRef);
|
|
|
|
|
|
|
|
lineMod->SetFollowMode(plLineFollowMod::kFollowObject);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lineMod->SetFollowMode(mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kPathObjectSel);
|
|
|
|
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 = hsControlConverter::Instance().MakeTransformController(maxTm, pathNode);
|
|
|
|
|
|
|
|
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 = TRACKED_NEW plAnimPath;
|
|
|
|
animPath->SetController(tmc);
|
|
|
|
animPath->InitParts(initParts);
|
|
|
|
|
|
|
|
lineMod->SetPath(animPath);
|
|
|
|
|
|
|
|
if( !pathNode->GetParentNode()->IsRootNode() )
|
|
|
|
{
|
|
|
|
plMaxNode* parNode = (plMaxNode*)pathNode->GetParentNode();
|
|
|
|
plSceneObject* parObj = parNode->GetSceneObject();
|
|
|
|
if( parObj )
|
|
|
|
{
|
|
|
|
plGenRefMsg* refMsg = TRACKED_NEW plGenRefMsg(lineMod->GetKey(), plRefMsg::kOnCreate, 0, plLineFollowMod::kRefParent);
|
|
|
|
hsgResMgr::ResMgr()->AddViaNotify(parObj->GetKey(), refMsg, plRefFlags::kPassiveRef);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fCompPB->GetInt(kOffsetActive) )
|
|
|
|
{
|
|
|
|
lineMod->SetOffsetDegrees(fCompPB->GetFloat(kOffsetDegrees));
|
|
|
|
|
|
|
|
if( fCompPB->GetInt(kOffsetClampActive) )
|
|
|
|
{
|
|
|
|
lineMod->SetOffsetClamp(fCompPB->GetFloat(kOffsetClamp));
|
|
|
|
}
|
|
|
|
|
|
|
|
lineMod->SetForceToLine(fCompPB->GetInt(kForceToLine));
|
|
|
|
}
|
|
|
|
|
|
|
|
if( fCompPB->GetInt(kSpeedClampActive) )
|
|
|
|
{
|
|
|
|
lineMod->SetSpeedClamp(fCompPB->GetFloat(kSpeedClamp));
|
|
|
|
}
|
|
|
|
|
|
|
|
fLineMod = lineMod;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
plLineFollowMod* plLineFollowComponent::GetLineMod(plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
if( !fValid )
|
|
|
|
return nil;
|
|
|
|
if( !fLineMod )
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < NumTargets(); i++ )
|
|
|
|
{
|
|
|
|
plMaxNode* targ = (plMaxNode*)GetTarget(i);
|
|
|
|
IMakeLineMod(targ, pErrMsg);
|
|
|
|
if( fLineMod )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return fLineMod;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plLineFollowComponent::Convert(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
if( !fValid )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if( !fLineMod )
|
|
|
|
{
|
|
|
|
if( !IMakeLineMod(node, pErrMsg) )
|
|
|
|
{
|
|
|
|
fValid = false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
node->AddModifier(fLineMod, IGetUniqueName(node));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plLineFollowComponent::SetupProperties(plMaxNode* pNode, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
fValid = false;
|
|
|
|
fLineMod = nil;
|
|
|
|
|
|
|
|
if( !fCompPB->GetINode(kPathObjectSel) )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kPathObjectSel);
|
|
|
|
if( !pathNode )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if( !pathNode->IsTMAnimated() )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
pathNode->SetCanConvert(false);
|
|
|
|
|
|
|
|
if( plLineFollowMod::kFollowObject == fCompPB->GetInt(kFollowModeRadio) )
|
|
|
|
{
|
|
|
|
if( !fCompPB->GetINode(kFollowObjectSel) )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fValid = true;
|
|
|
|
pNode->SetForceLocal(true);
|
|
|
|
pNode->SetMovable(true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plLineFollowComponent::PreConvert(plMaxNode* pNode, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
if( !fValid )
|
|
|
|
return true;
|
|
|
|
fValid = false;
|
|
|
|
|
|
|
|
if( plLineFollowMod::kFollowObject == fCompPB->GetInt(kFollowModeRadio) )
|
|
|
|
{
|
|
|
|
plMaxNode* followNode = (plMaxNode*)fCompPB->GetINode(kFollowObjectSel);
|
|
|
|
if( !followNode->CanConvert() )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
plMaxNode* pathNode = (plMaxNode*)fCompPB->GetINode(kPathObjectSel);
|
|
|
|
if( !pathNode->GetParentNode()->IsRootNode() )
|
|
|
|
{
|
|
|
|
if( !((plMaxNode*)pathNode->GetParentNode())->CanConvert() )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fValid = true;
|
|
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// The stereizer component is sort of related to LineFollow. Really.
|
|
|
|
|
|
|
|
class plStereizeComp : public plComponent
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
kLeft,
|
|
|
|
kAmbientDist,
|
|
|
|
kTransition,
|
|
|
|
kSepAngle,
|
|
|
|
kMaxDist,
|
|
|
|
kMinDist
|
|
|
|
};
|
|
|
|
|
|
|
|
plLineFollowMod* ISetMaster(plStereizer* stereo, plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
public:
|
|
|
|
plStereizeComp();
|
|
|
|
void DeleteThis() { delete this; }
|
|
|
|
|
|
|
|
// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
|
|
|
|
// of properties on the MaxNode, as it's still indeterminant.
|
|
|
|
virtual hsBool SetupProperties(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
virtual hsBool PreConvert(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
virtual hsBool Convert(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
hsBool Bail(plMaxNode* node, const char* msg, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CLASS_DESC(plStereizeComp, gStereizeDesc, "Stereo-ize", "Stereize", COMP_TYPE_AUDIO, STEREIZE_COMP_CID)
|
|
|
|
|
|
|
|
ParamBlockDesc2 gStereizeBk
|
|
|
|
(
|
|
|
|
plComponent::kBlkComp, _T("Stereize"), 0, &gStereizeDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp,
|
|
|
|
|
|
|
|
IDD_COMP_STEREIZE, IDS_COMP_STEREIZE, 0, 0, NULL,
|
|
|
|
|
|
|
|
plStereizeComp::kLeft, _T("Left"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_STEREIZE_LEFT,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plStereizeComp::kAmbientDist, _T("AmbientDist"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 50.0,
|
|
|
|
p_range, 0.0, 1000.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_STEREIZE_AMB, IDC_COMP_STEREIZE_AMB_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plStereizeComp::kTransition, _T("Transition"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 25.0,
|
|
|
|
p_range, 1.0, 500.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_STEREIZE_TRANS, IDC_COMP_STEREIZE_TRANS_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plStereizeComp::kSepAngle, _T("SepAngle"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 30.0,
|
|
|
|
p_range, 1.0, 80.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_STEREIZE_ANG, IDC_COMP_STEREIZE_ANG_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plStereizeComp::kMaxDist, _T("MaxDist"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 100.0,
|
|
|
|
p_range, 1.0, 1000.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_STEREIZE_MAXDIST, IDC_COMP_STEREIZE_MAXDIST_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plStereizeComp::kMinDist, _T("MinDist"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 5.0,
|
|
|
|
p_range, 1.0, 50.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_STEREIZE_MINDIST, IDC_COMP_STEREIZE_MINDIST_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
end
|
|
|
|
);
|
|
|
|
|
|
|
|
plStereizeComp::plStereizeComp()
|
|
|
|
{
|
|
|
|
fClassDesc = &gStereizeDesc;
|
|
|
|
fClassDesc->MakeAutoParamBlocks(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plStereizeComp::SetupProperties(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
node->SetForceLocal(true);
|
|
|
|
if( !node->GetParentNode()->IsRootNode() )
|
|
|
|
((plMaxNode*)node->GetParentNode())->SetForceLocal(true);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plStereizeComp::PreConvert(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plStereizeComp::Bail(plMaxNode* node, const char* msg, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
pErrMsg->Set(true, node->GetName(), msg).CheckAndAsk();
|
|
|
|
pErrMsg->Set(false);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plStereizeComp::Convert(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
plStereizer* stereo = TRACKED_NEW plStereizer;
|
|
|
|
|
|
|
|
stereo->SetAmbientDist(fCompPB->GetFloat(kAmbientDist));
|
|
|
|
stereo->SetTransition(fCompPB->GetFloat(kTransition));
|
|
|
|
|
|
|
|
float ang = fCompPB->GetFloat(kSepAngle);
|
|
|
|
if( ang > 80.f )
|
|
|
|
ang = 80.f;
|
|
|
|
stereo->SetSepAngle(hsDegreesToRadians(ang));
|
|
|
|
|
|
|
|
stereo->SetMaxSepDist(fCompPB->GetFloat(kMaxDist));
|
|
|
|
stereo->SetMinSepDist(fCompPB->GetFloat(kMinDist));
|
|
|
|
|
|
|
|
stereo->SetParentInitPos(node->GetLocalToParent44().GetTranslate());
|
|
|
|
|
|
|
|
stereo->SetAsLeftChannel(fCompPB->GetInt(kLeft));
|
|
|
|
|
|
|
|
node->AddModifier(stereo, IGetUniqueName(node));
|
|
|
|
|
|
|
|
// Do this after AddModifier, cuz that's when stereo gets a plKey.
|
|
|
|
ISetMaster(stereo, node, pErrMsg);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
plLineFollowMod* plStereizeComp::ISetMaster(plStereizer* stereo, plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
int numComp = node->NumAttachedComponents(false);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
for( i = 0; i < numComp; i++ )
|
|
|
|
{
|
|
|
|
plComponentBase *comp = node->GetAttachedComponent(i);
|
|
|
|
if( comp && (comp->ClassID() == LINEFOLLOW_COMP_CID) )
|
|
|
|
{
|
|
|
|
plLineFollowComponent* lineComp = (plLineFollowComponent*)comp;
|
|
|
|
plLineFollowMod* lineMod = lineComp->GetLineMod(pErrMsg);
|
|
|
|
if( lineMod )
|
|
|
|
{
|
|
|
|
lineMod->AddStereizer(stereo->GetKey());
|
|
|
|
return lineMod;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// The swivel component is also sort of related to LineFollow. And stereizer. Really.
|
|
|
|
|
|
|
|
class plSwivelComp : public plComponent
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
kFaceTypeRadio,
|
|
|
|
kFaceObjectSel,
|
|
|
|
kPivotY,
|
|
|
|
kOffsetActive,
|
|
|
|
kOffsetX,
|
|
|
|
kOffsetY,
|
|
|
|
kOffsetZ,
|
|
|
|
kOffsetLocal
|
|
|
|
};
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
kFaceCamera,
|
|
|
|
kFaceListener,
|
|
|
|
kFacePlayer,
|
|
|
|
kFaceObject
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
plSwivelComp();
|
|
|
|
void DeleteThis() { delete this; }
|
|
|
|
|
|
|
|
// SetupProperties - Internal setup and write-only set properties on the MaxNode. No reading
|
|
|
|
// of properties on the MaxNode, as it's still indeterminant.
|
|
|
|
virtual hsBool SetupProperties(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
virtual hsBool PreConvert(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
virtual hsBool Convert(plMaxNode* node, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
hsBool Bail(plMaxNode* node, const char* msg, plErrorMsg* pErrMsg);
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class plSwivelComponentProc : public ParamMap2UserDlgProc
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
BOOL DlgProc(TimeValue t, IParamMap2* map, HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
switch (msg)
|
|
|
|
{
|
|
|
|
case WM_INITDIALOG:
|
|
|
|
{
|
|
|
|
IParamBlock2* pb = map->GetParamBlock();
|
|
|
|
if( pb->GetInt(plSwivelComp::kFaceTypeRadio) == plSwivelComp::kFaceObject )
|
|
|
|
map->Enable(plSwivelComp::kFaceObjectSel, TRUE);
|
|
|
|
else
|
|
|
|
map->Enable(plSwivelComp::kFaceObjectSel, FALSE);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
|
|
|
|
//////////////////
|
|
|
|
case WM_COMMAND:
|
|
|
|
{
|
|
|
|
if( (LOWORD(wParam) == IDC_RADIO_LISTENER)
|
|
|
|
|| (LOWORD(wParam) == IDC_RADIO_PLAYER)
|
|
|
|
|| (LOWORD(wParam) == IDC_RADIO_CAMERA)
|
|
|
|
|| (LOWORD(wParam) == IDC_RADIO_OBJECT) )
|
|
|
|
{
|
|
|
|
IParamBlock2* pb = map->GetParamBlock();
|
|
|
|
if( pb->GetInt(plSwivelComp::kFaceTypeRadio) == plSwivelComp::kFaceObject )
|
|
|
|
map->Enable(plSwivelComp::kFaceObjectSel, TRUE);
|
|
|
|
else
|
|
|
|
map->Enable(plSwivelComp::kFaceObjectSel, FALSE);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
void DeleteThis() {}
|
|
|
|
};
|
|
|
|
static plSwivelComponentProc gSwivelProc;
|
|
|
|
|
|
|
|
|
|
|
|
CLASS_DESC(plSwivelComp, gSwivelDesc, "Swivel", "Swivel", COMP_TYPE_GRAPHICS, SWIVEL_COMP_CID)
|
|
|
|
|
|
|
|
ParamBlockDesc2 gSwivelBk
|
|
|
|
(
|
|
|
|
plComponent::kBlkComp, _T("Swivel"), 0, &gSwivelDesc, P_AUTO_CONSTRUCT + P_AUTO_UI, plComponent::kRefComp,
|
|
|
|
|
|
|
|
IDD_COMP_SWIVEL, IDS_COMP_SWIVEL, 0, 0, &gSwivelProc,
|
|
|
|
|
|
|
|
plSwivelComp::kFaceTypeRadio, _T("FaceType"), TYPE_INT, 0, 0,
|
|
|
|
p_ui, TYPE_RADIO, 4, IDC_RADIO_CAMERA, IDC_RADIO_LISTENER, IDC_RADIO_PLAYER, IDC_RADIO_OBJECT,
|
|
|
|
p_vals, plSwivelComp::kFaceCamera, plSwivelComp::kFaceListener, plSwivelComp::kFacePlayer, plSwivelComp::kFaceObject,
|
|
|
|
p_default, plSwivelComp::kFacePlayer,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kFaceObjectSel, _T("FaceObjectChoice"), TYPE_INODE, 0, 0,
|
|
|
|
p_ui, TYPE_PICKNODEBUTTON, IDC_COMP_CHOOSE_OBJECT,
|
|
|
|
p_prompt, IDS_COMP_CHOOSE_OBJECT,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kPivotY, _T("PivotY"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, TRUE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_PIVOTY,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kOffsetActive, _T("OffsetActive"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, FALSE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_OFFSETACTIVE,
|
|
|
|
p_enable_ctrls, 4, plSwivelComp::kOffsetX, plSwivelComp::kOffsetY, plSwivelComp::kOffsetZ, plSwivelComp::kOffsetLocal,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kOffsetX, _T("X"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 0.0,
|
|
|
|
p_range, 0.0, 100.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_OFFSETX, IDC_COMP_OFFSETX_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kOffsetY, _T("Y"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 0.0,
|
|
|
|
p_range, 0.0, 100.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_OFFSETY, IDC_COMP_OFFSETY_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kOffsetZ, _T("Z"), TYPE_FLOAT, 0, 0,
|
|
|
|
p_default, 0.0,
|
|
|
|
p_range, 0.0, 100.0,
|
|
|
|
p_ui, TYPE_SPINNER, EDITTYPE_POS_FLOAT,
|
|
|
|
IDC_COMP_OFFSETZ, IDC_COMP_OFFSETZ_SPIN, 1.0,
|
|
|
|
end,
|
|
|
|
|
|
|
|
plSwivelComp::kOffsetLocal, _T("OffsetLocal"), TYPE_BOOL, 0, 0,
|
|
|
|
p_default, TRUE,
|
|
|
|
p_ui, TYPE_SINGLECHEKBOX, IDC_COMP_OFFSETLOCAL,
|
|
|
|
end,
|
|
|
|
|
|
|
|
end
|
|
|
|
);
|
|
|
|
|
|
|
|
plSwivelComp::plSwivelComp()
|
|
|
|
{
|
|
|
|
fClassDesc = &gSwivelDesc;
|
|
|
|
fClassDesc->MakeAutoParamBlocks(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plSwivelComp::SetupProperties(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
node->SetForceLocal(true);
|
|
|
|
node->SetMovable(true);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plSwivelComp::PreConvert(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plSwivelComp::Bail(plMaxNode* node, const char* msg, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
pErrMsg->Set(true, node->GetName(), msg).CheckAndAsk();
|
|
|
|
pErrMsg->Set(false);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
hsBool plSwivelComp::Convert(plMaxNode* node, plErrorMsg* pErrMsg)
|
|
|
|
{
|
|
|
|
plViewFaceModifier* pMod = TRACKED_NEW plViewFaceModifier;
|
|
|
|
pMod->SetOrigTransform(node->GetLocalToParent44(), node->GetParentToLocal44());
|
|
|
|
node->AddModifier(pMod, IGetUniqueName(node));
|
|
|
|
|
|
|
|
if( fCompPB->GetInt(kPivotY) )
|
|
|
|
pMod->SetFlag(plViewFaceModifier::kPivotY);
|
|
|
|
else
|
|
|
|
pMod->SetFlag(plViewFaceModifier::kPivotFace);
|
|
|
|
|
|
|
|
switch( fCompPB->GetInt(kFaceTypeRadio) )
|
|
|
|
{
|
|
|
|
case kFaceCamera:
|
|
|
|
pMod->SetFollowMode(plViewFaceModifier::kFollowCamera);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kFaceListener:
|
|
|
|
pMod->SetFollowMode(plViewFaceModifier::kFollowListener);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kFacePlayer:
|
|
|
|
pMod->SetFollowMode(plViewFaceModifier::kFollowPlayer);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kFaceObject:
|
|
|
|
{
|
|
|
|
plMaxNode* targNode = (plMaxNode*)fCompPB->GetINode(kFaceObjectSel);
|
|
|
|
if( targNode && targNode->CanConvert() )
|
|
|
|
{
|
|
|
|
pMod->SetFollowMode(plViewFaceModifier::kFollowObject, targNode->GetKey());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pErrMsg->Set(true, node->GetName(), "Swivel to look at component %s has no Plasma object to look at.", GetINode()->GetName()).Show();
|
|
|
|
pErrMsg->Set(false);
|
|
|
|
pMod->SetFollowMode(plViewFaceModifier::kFollowCamera);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pMod->SetOffsetActive(fCompPB->GetInt(kOffsetActive));
|
|
|
|
if( fCompPB->GetInt(kOffsetActive) )
|
|
|
|
{
|
|
|
|
hsVector3 off(fCompPB->GetFloat(kOffsetX), fCompPB->GetFloat(kOffsetY), fCompPB->GetFloat(kOffsetZ));
|
|
|
|
pMod->SetOffset(off, fCompPB->GetInt(kOffsetLocal));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|