mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-07 09:43:40 +08:00
769 lines
26 KiB
C++
769 lines
26 KiB
C++
|
//========= Copyright <20> 1996-2005, Valve Corporation, All rights reserved. ============//
|
|||
|
//
|
|||
|
// Purpose: Namespace for functions having to do with WC Edit mode
|
|||
|
//
|
|||
|
// $Workfile: $
|
|||
|
// $Date: $
|
|||
|
//
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// $Log: $
|
|||
|
//
|
|||
|
// $NoKeywords: $
|
|||
|
//=============================================================================//
|
|||
|
|
|||
|
#include "cbase.h"
|
|||
|
#include "mathlib.h"
|
|||
|
#include "player.h"
|
|||
|
#include "wcedit.h"
|
|||
|
#include "ai_network.h"
|
|||
|
#include "ai_initutils.h"
|
|||
|
#include "ai_hull.h"
|
|||
|
#include "ai_link.h"
|
|||
|
#include "ai_node.h"
|
|||
|
#include "ai_dynamiclink.h"
|
|||
|
#include "ai_networkmanager.h"
|
|||
|
#include "ndebugoverlay.h"
|
|||
|
#include "editor_sendcommand.h"
|
|||
|
#include "movevars_shared.h"
|
|||
|
#include "model_types.h"
|
|||
|
// UNDONE: Reduce some dependency here!
|
|||
|
#include "physics_prop_ragdoll.h"
|
|||
|
#include "items.h"
|
|||
|
|
|||
|
// memdbgon must be the last include file in a .cpp file!!!
|
|||
|
#include "tier0/memdbgon.h"
|
|||
|
|
|||
|
extern CAI_Node* FindPickerAINode( CBasePlayer* pPlayer, NodeType_e nNodeType );
|
|||
|
extern CAI_Link* FindPickerAILink( CBasePlayer* pPlayer );
|
|||
|
extern float GetFloorZ(const Vector &origin);
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Make sure the version of the map in WC is the same as the map
|
|||
|
// that's being edited
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
bool NWCEdit::IsWCVersionValid(void)
|
|||
|
{
|
|||
|
int status = Editor_CheckVersion(STRING(gpGlobals->mapname), gpGlobals->mapversion, false);
|
|||
|
if (!status)
|
|||
|
{
|
|||
|
return true;
|
|||
|
}
|
|||
|
else if (status == Editor_NotRunning)
|
|||
|
{
|
|||
|
Msg("\nAborting map_edit\nWorldcraft not running...\n\n");
|
|||
|
UTIL_CenterPrintAll( "Worldcraft not running..." );
|
|||
|
engine->ServerCommand("disconnect\n");
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
Msg("\nAborting map_edit\nWC/Engine map versions different...\n\n");
|
|||
|
UTIL_CenterPrintAll( "WC/Engine map versions different..." );
|
|||
|
engine->ServerCommand("disconnect\n");
|
|||
|
}
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose : Figure out placement position of air nodes form where player is
|
|||
|
// looking. Keep distance from player constant but adjust height
|
|||
|
// based on viewing angle
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
Vector NWCEdit::AirNodePlacementPosition( void )
|
|||
|
{
|
|||
|
CBasePlayer* pPlayer = UTIL_PlayerByIndex(CBaseEntity::m_nDebugPlayer);
|
|||
|
|
|||
|
if (!pPlayer)
|
|||
|
{
|
|||
|
return vec3_origin;
|
|||
|
}
|
|||
|
|
|||
|
Vector pForward;
|
|||
|
pPlayer->EyeVectors( &pForward );
|
|||
|
|
|||
|
Vector floorVec = pForward;
|
|||
|
floorVec.z = 0;
|
|||
|
VectorNormalize( floorVec );
|
|||
|
VectorNormalize( pForward );
|
|||
|
|
|||
|
float cosAngle = DotProduct(floorVec,pForward);
|
|||
|
|
|||
|
float lookDist = g_pAINetworkManager->GetEditOps()->m_flAirEditDistance/cosAngle;
|
|||
|
Vector lookPos = pPlayer->EyePosition()+pForward * lookDist;
|
|||
|
|
|||
|
return lookPos;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: For create nodes in wc edit mode
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::CreateAINode( CBasePlayer *pPlayer )
|
|||
|
{
|
|||
|
// -------------------------------------------------------------
|
|||
|
// Check that WC is running with the right map version
|
|||
|
// -------------------------------------------------------------
|
|||
|
if ( !IsWCVersionValid() || !pPlayer )
|
|||
|
return;
|
|||
|
|
|||
|
pPlayer->AddSolidFlags( FSOLID_NOT_SOLID );
|
|||
|
|
|||
|
int hullType = g_pAINetworkManager->GetEditOps()->m_iHullDrawNum;
|
|||
|
|
|||
|
// -----------------------------------
|
|||
|
// Get position of node to create
|
|||
|
// -----------------------------------
|
|||
|
Vector vNewNodePos = vec3_origin;
|
|||
|
bool bPositionValid = false;
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
vNewNodePos = NWCEdit::AirNodePlacementPosition();
|
|||
|
|
|||
|
// Make sure we can see the node
|
|||
|
trace_t tr;
|
|||
|
UTIL_TraceLine(pPlayer->EyePosition(), vNewNodePos, MASK_NPCSOLID_BRUSHONLY, pPlayer, COLLISION_GROUP_NONE, &tr);
|
|||
|
if (tr.fraction == 1.0)
|
|||
|
{
|
|||
|
bPositionValid = true;
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// Place node by where the player is looking
|
|||
|
Vector forward;
|
|||
|
pPlayer->EyeVectors( &forward );
|
|||
|
Vector startTrace = pPlayer->EyePosition();
|
|||
|
Vector endTrace = pPlayer->EyePosition() + forward * MAX_TRACE_LENGTH;
|
|||
|
trace_t tr;
|
|||
|
UTIL_TraceLine(startTrace,endTrace,MASK_NPCSOLID, pPlayer, COLLISION_GROUP_NONE, &tr );
|
|||
|
if ( tr.fraction != 1.0)
|
|||
|
{
|
|||
|
// Raise the end position up off the floor, place the node and drop him down
|
|||
|
tr.endpos.z += 48;
|
|||
|
vNewNodePos = tr.endpos;
|
|||
|
bPositionValid = true;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// -------------------------------------------------------------------------------
|
|||
|
// Now check that this is a valid location for the new node bu using test hull
|
|||
|
// -------------------------------------------------------------------------------
|
|||
|
if (bPositionValid)
|
|||
|
{
|
|||
|
CBaseEntity *testHull = (CBaseEntity*)CAI_TestHull::GetTestHull();
|
|||
|
|
|||
|
// Set the size of the test hull
|
|||
|
UTIL_SetSize(testHull, NAI_Hull::Mins(hullType), NAI_Hull::Maxs(hullType));
|
|||
|
|
|||
|
// Set origin of test hull
|
|||
|
testHull->SetLocalOrigin( vNewNodePos );
|
|||
|
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
// If a ground node, drop to floor and make sure can stand at test postion
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
if (!g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
UTIL_DropToFloor( testHull, MASK_NPCSOLID );
|
|||
|
vNewNodePos = testHull->GetAbsOrigin();
|
|||
|
CTraceFilterSimple traceFilter( testHull, COLLISION_GROUP_NONE );
|
|||
|
if (!UTIL_CheckBottom(testHull, &traceFilter, sv_stepsize.GetFloat()))
|
|||
|
{
|
|||
|
CAI_TestHull::ReturnTestHull();
|
|||
|
bPositionValid = false;
|
|||
|
goto DoneCreate;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
// Make sure hull fits at location by seeing if it can move up a fraction
|
|||
|
// -----------------------------------------------------------------------
|
|||
|
Vector vUpBit = testHull->GetAbsOrigin();
|
|||
|
vUpBit.z += 1;
|
|||
|
trace_t tr;
|
|||
|
UTIL_TraceHull( testHull->GetAbsOrigin(), vUpBit, NAI_Hull::Mins(hullType),
|
|||
|
NAI_Hull::Maxs(hullType), MASK_NPCSOLID, testHull, COLLISION_GROUP_NONE, &tr );
|
|||
|
if (tr.startsolid || tr.fraction != 1.0)
|
|||
|
{
|
|||
|
CAI_TestHull::ReturnTestHull();
|
|||
|
bPositionValid = false;
|
|||
|
goto DoneCreate;
|
|||
|
}
|
|||
|
|
|||
|
// <<TEMP>> Round position till DS fixed WC bug
|
|||
|
testHull->SetLocalOrigin( Vector( floor(testHull->GetAbsOrigin().x),
|
|||
|
floor(testHull->GetAbsOrigin().y ), floor(testHull->GetAbsOrigin().z) ) );
|
|||
|
|
|||
|
// ---------------------------------------
|
|||
|
// Send new node to WC
|
|||
|
// ---------------------------------------
|
|||
|
int status;
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
status = Editor_CreateNode("info_node_air", g_pAINetworkManager->GetEditOps()->m_nNextWCIndex, testHull->GetLocalOrigin().x, testHull->GetLocalOrigin().y, testHull->GetLocalOrigin().z, false);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// Create slightly higher in WC so it can be dropped when its loaded again
|
|||
|
Vector origin = testHull->GetLocalOrigin();
|
|||
|
origin.z += 24.0;
|
|||
|
testHull->SetLocalOrigin( origin );
|
|||
|
status = Editor_CreateNode("info_node", g_pAINetworkManager->GetEditOps()->m_nNextWCIndex, testHull->GetLocalOrigin().x, testHull->GetLocalOrigin().y, testHull->GetLocalOrigin().z, false);
|
|||
|
}
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on creation...\n" );
|
|||
|
CAI_TestHull::ReturnTestHull();
|
|||
|
}
|
|||
|
else if (status == Editor_OK)
|
|||
|
{
|
|||
|
// -----------------------
|
|||
|
// Create a new ai node
|
|||
|
// -----------------------
|
|||
|
CNodeEnt *pNodeEnt;
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
pNodeEnt = (CNodeEnt*)CreateEntityByName("info_node_air");
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
pNodeEnt = (CNodeEnt*)CreateEntityByName("info_node");
|
|||
|
}
|
|||
|
|
|||
|
// Note this is a new entity being created as part of wc editing
|
|||
|
pNodeEnt->SetLocalOrigin( testHull->GetLocalOrigin() );
|
|||
|
CAI_TestHull::ReturnTestHull();
|
|||
|
|
|||
|
pNodeEnt->m_NodeData.nWCNodeID = g_pAINetworkManager->GetEditOps()->m_nNextWCIndex;
|
|||
|
|
|||
|
pNodeEnt->m_debugOverlays |= OVERLAY_WC_CHANGE_ENTITY;
|
|||
|
pNodeEnt->Spawn();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
DoneCreate:
|
|||
|
// ----------------------------------------------------------
|
|||
|
// Flash a red box as a warning that the hull won't fit here
|
|||
|
// ----------------------------------------------------------
|
|||
|
if (!bPositionValid)
|
|||
|
{
|
|||
|
NDebugOverlay::Box(vNewNodePos, NAI_Hull::Mins(hullType), NAI_Hull::Maxs(hullType), 255,0,0,0,0.1);
|
|||
|
}
|
|||
|
|
|||
|
// Restore player collidability
|
|||
|
pPlayer->SetSolid( SOLID_BBOX );
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose:
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::UndoDestroyAINode(void)
|
|||
|
{
|
|||
|
// -------------------------------------------------------------
|
|||
|
// Check that WC is running with the right map version
|
|||
|
// -------------------------------------------------------------
|
|||
|
if (!IsWCVersionValid())
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode)
|
|||
|
{
|
|||
|
Vector nodePos = g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode->GetOrigin();
|
|||
|
|
|||
|
int status;
|
|||
|
int nOldWCID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode->GetId()];
|
|||
|
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
status = Editor_CreateNode("info_node_air", nOldWCID, nodePos.x, nodePos.y, nodePos.z, false);
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
status = Editor_CreateNode("info_node", nOldWCID, nodePos.x, nodePos.y, nodePos.z, false);
|
|||
|
}
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on creation...\n" );
|
|||
|
}
|
|||
|
else if (status == Editor_OK)
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode->SetType( NODE_GROUND );
|
|||
|
//@ tofo g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode->m_pNetwork->BuildNetworkGraph();
|
|||
|
g_pAINetworkManager->BuildNetworkGraph();
|
|||
|
g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode = NULL;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: For destroying nodes in wc edit mode
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::DestroyAINode( CBasePlayer *pPlayer )
|
|||
|
{
|
|||
|
// -------------------------------------------------------------
|
|||
|
// Check that WC is running with the right map version
|
|||
|
// -------------------------------------------------------------
|
|||
|
if (!IsWCVersionValid())
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
if (!pPlayer)
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
NodeType_e nNodeType = NODE_GROUND;
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
nNodeType = NODE_AIR;
|
|||
|
}
|
|||
|
|
|||
|
CAI_Node* pAINode = FindPickerAINode(pPlayer, nNodeType);
|
|||
|
if (pAINode)
|
|||
|
{
|
|||
|
int status = Editor_DeleteNode(g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[pAINode->GetId()], false);
|
|||
|
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on deletion...\n" );
|
|||
|
}
|
|||
|
else if (status == Editor_OK)
|
|||
|
{
|
|||
|
// Mark this node as deleted and changed
|
|||
|
pAINode->SetType( NODE_DELETED );
|
|||
|
pAINode->m_eNodeInfo |= bits_NODE_WC_CHANGED;
|
|||
|
|
|||
|
// Note that network needs to be rebuild
|
|||
|
g_pAINetworkManager->GetEditOps()->SetRebuildFlags();
|
|||
|
g_pAINetworkManager->GetEditOps()->m_pLastDeletedNode = pAINode;
|
|||
|
|
|||
|
// Now go through at delete any dynamic links that were attached to this node
|
|||
|
for (int link = 0; link < pAINode->NumLinks(); link++)
|
|||
|
{
|
|||
|
int nSrcID = pAINode->GetLinkByIndex(link)->m_iSrcID;
|
|||
|
int nDstID = pAINode->GetLinkByIndex(link)->m_iDestID;
|
|||
|
if (CAI_DynamicLink::GetDynamicLink(nSrcID, nDstID))
|
|||
|
{
|
|||
|
int nWCSrcID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[nSrcID];
|
|||
|
int nWCDstID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[nDstID];
|
|||
|
int status = Editor_DeleteNodeLink(nWCSrcID, nWCDstID);
|
|||
|
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on node link deletion...\n" );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: For restroring links in WC edit mode. This actually means
|
|||
|
// destroying links in WC that have been marked as
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::CreateAILink( CBasePlayer* pPlayer )
|
|||
|
{
|
|||
|
// -------------------------------------------------------------
|
|||
|
// Check that WC is running with the right map version
|
|||
|
// -------------------------------------------------------------
|
|||
|
if (!IsWCVersionValid())
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
CAI_Link* pAILink = FindPickerAILink(pPlayer);
|
|||
|
if (pAILink && (pAILink->m_LinkInfo & bits_LINK_OFF))
|
|||
|
{
|
|||
|
int nWCSrcID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[pAILink->m_iSrcID];
|
|||
|
int nWCDstID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[pAILink->m_iDestID];
|
|||
|
int status = Editor_DeleteNodeLink(nWCSrcID, nWCDstID, false);
|
|||
|
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on node link creation...\n" );
|
|||
|
}
|
|||
|
else if (status == Editor_OK)
|
|||
|
{
|
|||
|
// Don't actually destroy the dynamic link while editing. Just mark the link
|
|||
|
pAILink->m_LinkInfo &= ~bits_LINK_OFF;
|
|||
|
|
|||
|
CAI_DynamicLink* pDynamicLink = CAI_DynamicLink::GetDynamicLink(pAILink->m_iSrcID, pAILink->m_iDestID);
|
|||
|
UTIL_Remove(pDynamicLink);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: For destroying links in wc edit mode. Actually have to create
|
|||
|
// a link in WC that is marked as off
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::DestroyAILink( CBasePlayer *pPlayer )
|
|||
|
{
|
|||
|
// -------------------------------------------------------------
|
|||
|
// Check that WC is running with the right map version
|
|||
|
// -------------------------------------------------------------
|
|||
|
if (!IsWCVersionValid())
|
|||
|
{
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
CAI_Link* pAILink = FindPickerAILink(pPlayer);
|
|||
|
if (pAILink)
|
|||
|
{
|
|||
|
int nWCSrcID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[pAILink->m_iSrcID];
|
|||
|
int nWCDstID = g_pAINetworkManager->GetEditOps()->m_pNodeIndexTable[pAILink->m_iDestID];
|
|||
|
int status = Editor_CreateNodeLink(nWCSrcID, nWCDstID, false);
|
|||
|
|
|||
|
if (status == Editor_BadCommand)
|
|||
|
{
|
|||
|
Msg( "Worldcraft failed on node link creation...\n" );
|
|||
|
}
|
|||
|
else if (status == Editor_OK)
|
|||
|
{
|
|||
|
// Create dynamic link and mark the link
|
|||
|
CAI_DynamicLink* pNewLink = (CAI_DynamicLink*)CreateEntityByName("info_node_link" );;
|
|||
|
pNewLink->m_nSrcID = pAILink->m_iSrcID;
|
|||
|
pNewLink->m_nDestID = pAILink->m_iDestID;
|
|||
|
pNewLink->m_nLinkState = LINK_OFF;
|
|||
|
pAILink->m_LinkInfo |= bits_LINK_OFF;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
Vector *g_EntityPositions = NULL;
|
|||
|
QAngle *g_EntityOrientations = NULL;
|
|||
|
string_t *g_EntityClassnames = NULL;
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Saves the entity's position for future communication with Hammer
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::RememberEntityPosition( CBaseEntity *pEntity )
|
|||
|
{
|
|||
|
if ( !(pEntity->ObjectCaps() & FCAP_WCEDIT_POSITION) )
|
|||
|
return;
|
|||
|
|
|||
|
if ( !g_EntityPositions )
|
|||
|
{
|
|||
|
g_EntityPositions = new Vector[NUM_ENT_ENTRIES];
|
|||
|
g_EntityOrientations = new QAngle[NUM_ENT_ENTRIES];
|
|||
|
// have to save these too because some entities change the classname on spawn (e.g. prop_physics_override, physics_prop)
|
|||
|
g_EntityClassnames = new string_t[NUM_ENT_ENTRIES];
|
|||
|
}
|
|||
|
int index = pEntity->entindex();
|
|||
|
g_EntityPositions[index] = pEntity->GetAbsOrigin();
|
|||
|
g_EntityOrientations[index] = pEntity->GetAbsAngles();
|
|||
|
g_EntityClassnames[index] = pEntity->m_iClassname;
|
|||
|
}
|
|||
|
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
// Purpose: Sends Hammer an update to the current position
|
|||
|
//-----------------------------------------------------------------------------
|
|||
|
void NWCEdit::UpdateEntityPosition( CBaseEntity *pEntity )
|
|||
|
{
|
|||
|
const Vector &newPos = pEntity->GetAbsOrigin();
|
|||
|
const QAngle &newAng = pEntity->GetAbsAngles();
|
|||
|
|
|||
|
DevMsg( 1, "%s\n origin %f %f %f\n angles %f %f %f\n", pEntity->GetClassname(), newPos.x, newPos.y, newPos.z, newAng.x, newAng.y, newAng.z );
|
|||
|
if ( Ragdoll_IsPropRagdoll(pEntity) )
|
|||
|
{
|
|||
|
char tmp[2048];
|
|||
|
Ragdoll_GetAngleOverrideString( tmp, sizeof(tmp), pEntity );
|
|||
|
DevMsg( 1, "pose: %s\n", tmp );
|
|||
|
}
|
|||
|
|
|||
|
if ( !(pEntity->ObjectCaps() & FCAP_WCEDIT_POSITION) )
|
|||
|
return;
|
|||
|
|
|||
|
// can't do this unless in edit mode
|
|||
|
if ( !engine->IsInEditMode() )
|
|||
|
return;
|
|||
|
|
|||
|
int entIndex = pEntity->entindex();
|
|||
|
Vector pos = g_EntityPositions[entIndex];
|
|||
|
EditorSendResult_t result = Editor_BadCommand;
|
|||
|
const char *pClassname = STRING(g_EntityClassnames[entIndex]);
|
|||
|
|
|||
|
if ( pEntity->GetModel() && modelinfo->GetModelType(pEntity->GetModel()) == mod_brush )
|
|||
|
{
|
|||
|
QAngle xformAngles;
|
|||
|
RotationDelta( g_EntityOrientations[entIndex], newAng, &xformAngles );
|
|||
|
if ( xformAngles.Length() > 1e-4 )
|
|||
|
{
|
|||
|
result = Editor_RotateEntity( pClassname, pos.x, pos.y, pos.z, xformAngles );
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// don't call through for an identity rotation, may just increase error
|
|||
|
result = Editor_OK;
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if ( Ragdoll_IsPropRagdoll(pEntity) )
|
|||
|
{
|
|||
|
char tmp[2048];
|
|||
|
Ragdoll_GetAngleOverrideString( tmp, sizeof(tmp), pEntity );
|
|||
|
result = Editor_SetKeyValue( pClassname, pos.x, pos.y, pos.z, "angleOverride", tmp );
|
|||
|
if ( result != Editor_OK )
|
|||
|
goto error;
|
|||
|
}
|
|||
|
result = Editor_SetKeyValue( pClassname, pos.x, pos.y, pos.z, "angles", CFmtStr("%f %f %f", newAng.x, newAng.y, newAng.z) );
|
|||
|
}
|
|||
|
if ( result != Editor_OK )
|
|||
|
goto error;
|
|||
|
|
|||
|
result = Editor_SetKeyValue( pClassname, pos.x, pos.y, pos.z, "origin", CFmtStr("%f %f %f", newPos.x, newPos.y, newPos.z) );
|
|||
|
if ( result != Editor_OK )
|
|||
|
goto error;
|
|||
|
|
|||
|
NDebugOverlay::EntityBounds(pEntity, 0, 255, 0, 0 ,5);
|
|||
|
// save the update
|
|||
|
RememberEntityPosition( pEntity );
|
|||
|
return;
|
|||
|
|
|||
|
error:
|
|||
|
NDebugOverlay::EntityBounds(pEntity, 255, 0, 0, 0 ,5);
|
|||
|
}
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_Create( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode())
|
|||
|
{
|
|||
|
CBaseEntity::m_nDebugPlayer = UTIL_GetCommandClientIndex();
|
|||
|
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bLinkEditMode)
|
|||
|
{
|
|||
|
NWCEdit::CreateAILink(UTIL_GetCommandClient());
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
NWCEdit::CreateAINode(UTIL_GetCommandClient());
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_create("wc_create", CC_WC_Create, "When in WC edit mode, creates a node where the player is looking if a node is allowed at that location for the currently selected hull size (see ai_next_hull)", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_Destroy( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode())
|
|||
|
{
|
|||
|
CBaseEntity::m_nDebugPlayer = UTIL_GetCommandClientIndex();
|
|||
|
|
|||
|
// UNDONE: For now just deal with info_nodes
|
|||
|
//CBaseEntity* pEntity = FindEntity( pEdict, ""); - use when generalize this to any class
|
|||
|
//int status = Editor_DeleteEntity("info_node", pEdict->origin.x, pEdict->origin.y, pEdict->origin.z, false);
|
|||
|
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bLinkEditMode)
|
|||
|
{
|
|||
|
NWCEdit::DestroyAILink(UTIL_GetCommandClient());
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
NWCEdit::DestroyAINode(UTIL_GetCommandClient());
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_destroy("wc_destroy", CC_WC_Destroy, "When in WC edit mode, destroys the node that the player is nearest to looking at. (The node will be highlighted by a red box).", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_DestroyUndo( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode())
|
|||
|
{
|
|||
|
CBaseEntity::m_nDebugPlayer = UTIL_GetCommandClientIndex();
|
|||
|
|
|||
|
NWCEdit::UndoDestroyAINode();
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_destroy_undo("wc_destroy_undo", CC_WC_DestroyUndo, "When in WC edit mode restores the last deleted node", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_AirNodeEdit( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode())
|
|||
|
{
|
|||
|
// Toggle air edit mode state
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_bAirEditMode = false;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_bAirEditMode = true;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_air_node_edit("wc_air_node_edit", CC_WC_AirNodeEdit, "When in WC edit mode, toggles laying down or air nodes instead of ground nodes", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_AirNodeEditFurther( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode() && g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_flAirEditDistance += 10.0;
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_air_edit_further("wc_air_edit_further", CC_WC_AirNodeEditFurther, "When in WC edit mode and editing air nodes, moves position of air node crosshair and placement location further away from player", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_AirNodeEditNearer( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode() && g_pAINetworkManager->GetEditOps()->m_bAirEditMode)
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_flAirEditDistance -= 10.0;
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_air_edit_nearer("wc_air_edit_nearer", CC_WC_AirNodeEditNearer, "When in WC edit mode and editing air nodes, moves position of air node crosshair and placement location nearer to from player", FCVAR_CHEAT);
|
|||
|
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
// Purpose :
|
|||
|
// Input :
|
|||
|
// Output :
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
void CC_WC_LinkEdit( void )
|
|||
|
{
|
|||
|
// Only allowed in wc_edit_mode
|
|||
|
if (engine->IsInEditMode())
|
|||
|
{
|
|||
|
// Toggle air edit mode state
|
|||
|
if (g_pAINetworkManager->GetEditOps()->m_bLinkEditMode)
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_bLinkEditMode = false;
|
|||
|
}
|
|||
|
// Don't allow link mode if graph outdated
|
|||
|
else if (!(g_pAINetworkManager->GetEditOps()->m_debugNetOverlays & bits_debugNeedRebuild))
|
|||
|
{
|
|||
|
g_pAINetworkManager->GetEditOps()->m_bLinkEditMode = true;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
static ConCommand wc_link_edit("wc_link_edit", CC_WC_LinkEdit, 0, FCVAR_CHEAT);
|
|||
|
|
|||
|
CON_COMMAND( wc_update_entity, "Updates the entity's position/angles when in edit mode" )
|
|||
|
{
|
|||
|
if ( engine->Cmd_Argc() < 2 )
|
|||
|
{
|
|||
|
CBasePlayer *pPlayer = UTIL_GetCommandClient();
|
|||
|
trace_t tr;
|
|||
|
Vector forward;
|
|||
|
pPlayer->EyeVectors( &forward );
|
|||
|
UTIL_TraceLine(pPlayer->EyePosition(), pPlayer->EyePosition() + forward * MAX_COORD_RANGE,
|
|||
|
MASK_SHOT_HULL|CONTENTS_GRATE|CONTENTS_DEBRIS, pPlayer, COLLISION_GROUP_NONE, &tr );
|
|||
|
|
|||
|
if ( tr.DidHit() && !tr.DidHitWorld() )
|
|||
|
{
|
|||
|
NWCEdit::UpdateEntityPosition( tr.m_pEnt );
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
CBaseEntity *pEnt = NULL;
|
|||
|
while ((pEnt = gEntList.FindEntityGeneric(pEnt, engine->Cmd_Argv(1))) != NULL)
|
|||
|
{
|
|||
|
NWCEdit::UpdateEntityPosition( pEnt );
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
CON_COMMAND( wc_update_safe_entities, "Updates entities in the map that can safely be updated (don't have parents or are affected by constraints)." )
|
|||
|
{
|
|||
|
int iCount = 0;
|
|||
|
CBaseEntity *pEnt = NULL;
|
|||
|
|
|||
|
Msg("\n====================================================\nPerforming Safe Entity Update\n" );
|
|||
|
|
|||
|
for ( pEnt = gEntList.FirstEnt(); pEnt != NULL; pEnt = gEntList.NextEnt(pEnt) )
|
|||
|
{
|
|||
|
if ( !(pEnt->ObjectCaps() & FCAP_WCEDIT_POSITION) )
|
|||
|
continue;
|
|||
|
|
|||
|
// If we have a parent, or any children, we're not safe to update
|
|||
|
if ( pEnt->GetMoveParent() || pEnt->FirstMoveChild() )
|
|||
|
continue;
|
|||
|
|
|||
|
IPhysicsObject *pPhysics = pEnt->VPhysicsGetObject();
|
|||
|
if ( !pPhysics )
|
|||
|
continue;
|
|||
|
// If we are affected by any constraints, we're not safe to update
|
|||
|
if ( pPhysics->IsAttachedToConstraint(false) )
|
|||
|
continue;
|
|||
|
// Motion disabled?
|
|||
|
if ( !pPhysics->IsMoveable() )
|
|||
|
continue;
|
|||
|
|
|||
|
NWCEdit::UpdateEntityPosition( pEnt );
|
|||
|
iCount++;
|
|||
|
}
|
|||
|
|
|||
|
while ( (pEnt = gEntList.FindEntityByClassname( pEnt, "prop_ragdoll" )) != NULL )
|
|||
|
{
|
|||
|
// If we have a parent, or any children, we're not safe to update
|
|||
|
if ( pEnt->GetMoveParent() || pEnt->FirstMoveChild() )
|
|||
|
continue;
|
|||
|
|
|||
|
IPhysicsObject *pPhysics = pEnt->VPhysicsGetObject();
|
|||
|
if ( !pPhysics )
|
|||
|
continue;
|
|||
|
// If we are affected by any constraints, we're not safe to update
|
|||
|
if ( pPhysics->IsAttachedToConstraint(true) )
|
|||
|
{
|
|||
|
Msg("Skipping constrained ragdoll %s\n", pEnt->GetDebugName() );
|
|||
|
continue;
|
|||
|
}
|
|||
|
// Motion disabled?
|
|||
|
if ( !pPhysics->IsMoveable() )
|
|||
|
continue;
|
|||
|
|
|||
|
NWCEdit::UpdateEntityPosition( pEnt );
|
|||
|
iCount++;
|
|||
|
}
|
|||
|
|
|||
|
Msg("Updated %d entities.\n", iCount);
|
|||
|
}
|