//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============// // // Purpose: // // $NoKeywords: $ // //=============================================================================// // nav_edit.cpp // Implementation of Navigation Mesh edit mode // Author: Michael Booth, 2003-2004 #include "cbase.h" #include "nav_mesh.h" #include "nav_pathfind.h" #include "nav_node.h" #include "nav_colors.h" #include "Color.h" #include "tier0/vprof.h" #include "collisionutils.h" ConVar nav_show_area_info( "nav_show_area_info", "0.5", FCVAR_GAMEDLL, "Duration in seconds to show nav area ID and attributes while editing" ); ConVar nav_snap_to_grid( "nav_snap_to_grid", "0", FCVAR_GAMEDLL, "Snap to the nav generation grid when creating new nav areas" ); ConVar nav_create_place_on_ground( "nav_create_place_on_ground", "0", FCVAR_GAMEDLL, "If true, nav areas will be placed flush with the ground when created by hand." ); #if DEBUG_NAV_NODES extern ConVar nav_show_nodes; #endif // DEBUG_NAV_NODES //-------------------------------------------------------------------------------------------------------------- void EditNav_Precache(void *pUser) { CBaseEntity::PrecacheScriptSound( "Bot.EditSwitchOn" ); CBaseEntity::PrecacheScriptSound( "EDIT_TOGGLE_PLACE_MODE" ); CBaseEntity::PrecacheScriptSound( "Bot.EditSwitchOff" ); CBaseEntity::PrecacheScriptSound( "EDIT_PLACE_PICK" ); CBaseEntity::PrecacheScriptSound( "EDIT_DELETE" ); CBaseEntity::PrecacheScriptSound( "EDIT.ToggleAttribute" ); CBaseEntity::PrecacheScriptSound( "EDIT_SPLIT.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_SPLIT.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_MERGE.Enable" ); CBaseEntity::PrecacheScriptSound( "EDIT_MERGE.Disable" ); CBaseEntity::PrecacheScriptSound( "EDIT_MARK.Enable" ); CBaseEntity::PrecacheScriptSound( "EDIT_MARK.Disable" ); CBaseEntity::PrecacheScriptSound( "EDIT_MARK_UNNAMED.Enable" ); CBaseEntity::PrecacheScriptSound( "EDIT_MARK_UNNAMED.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_MARK_UNNAMED.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_CONNECT.AllDirections" ); CBaseEntity::PrecacheScriptSound( "EDIT_CONNECT.Added" ); CBaseEntity::PrecacheScriptSound( "EDIT_DISCONNECT.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_DISCONNECT.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_SPLICE.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_SPLICE.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_SELECT_CORNER.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_SELECT_CORNER.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_MOVE_CORNER.MarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_MOVE_CORNER.NoMarkedArea" ); CBaseEntity::PrecacheScriptSound( "EDIT_BEGIN_AREA.Creating" ); CBaseEntity::PrecacheScriptSound( "EDIT_BEGIN_AREA.NotCreating" ); CBaseEntity::PrecacheScriptSound( "EDIT_END_AREA.Creating" ); CBaseEntity::PrecacheScriptSound( "EDIT_END_AREA.NotCreating" ); CBaseEntity::PrecacheScriptSound( "EDIT_WARP_TO_MARK" ); } #ifdef CSTRIKE_DLL PRECACHE_REGISTER_FN( EditNav_Precache ); #endif //-------------------------------------------------------------------------------------------------------------- inline float round( float val, float unit ) { val = val + ((val < 0.0f) ? -unit*0.5f : unit*0.5f); return (float)( unit * ( ((int)val) / (int)unit ) ); } //-------------------------------------------------------------------------------------------------------------- int GetGridSize( bool forceGrid = false ) { if ( TheNavMesh->IsGenerating() ) { return (int)GenerationStepSize; } int snapVal = nav_snap_to_grid.GetInt(); if ( forceGrid && !snapVal ) { snapVal = 1; } if ( snapVal == 0 ) { return 0; } int scale = (int)GenerationStepSize; switch ( snapVal ) { case 3: scale = 1; break; case 2: scale = 5; break; case 1: default: break; } return scale; } //-------------------------------------------------------------------------------------------------------------- Vector CNavMesh::SnapToGrid( const Vector& in, bool snapX, bool snapY, bool forceGrid ) const { int scale = GetGridSize( forceGrid ); if ( !scale ) { return in; } Vector out( in ); if ( snapX ) { out.x = round( in.x, scale ); } if ( snapY ) { out.y = round( in.y, scale ); } return out; } //-------------------------------------------------------------------------------------------------------------- float CNavMesh::SnapToGrid( float x, bool forceGrid ) const { int scale = GetGridSize( forceGrid ); if ( !scale ) { return x; } x = round( x, scale ); return x; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::GetEditVectors( Vector *pos, Vector *forward ) { if ( !pos || !forward ) { return; } CBasePlayer *player = UTIL_GetListenServerHost(); if ( !player ) { return; } Vector dir; AngleVectors( player->EyeAngles() + player->GetPunchAngle(), forward ); *pos = player->EyePosition(); } //-------------------------------------------------------------------------------------------------------------- /** * Convenience function to find the nav area a player is looking at, for editing commands */ bool CNavMesh::GetActiveNavArea( void ) { VPROF( "CNavMesh::GetActiveNavArea" ); m_splitAlongX = false; m_splitEdge = 0.0f; m_selectedArea = NULL; m_climbableSurface = false; m_selectedLadder = NULL; CBasePlayer *player = UTIL_GetListenServerHost(); if ( player == NULL ) return false; const float maxRange = 1000.0f; // 500 Vector from, dir; GetEditVectors( &from, &dir ); Vector to = from + maxRange * dir; trace_t result; UTIL_TraceLine( from, to, MASK_PLAYERSOLID_BRUSHONLY, player, COLLISION_GROUP_NONE, &result ); if (result.fraction != 1.0f) { if ( m_navEditMode != NAV_EDIT_CREATE ) { m_climbableSurface = physprops->GetSurfaceData( result.surface.surfaceProps )->game.climbable != 0; if ( !m_climbableSurface ) { m_climbableSurface = (result.contents & CONTENTS_LADDER) != 0; } m_surfaceNormal = result.plane.normal; if ( m_climbableSurface ) { // check if we're on the same plane as the original point when we're building a ladder if ( m_isCreatingLadder ) { if ( m_surfaceNormal != m_ladderNormal ) { m_climbableSurface = false; } } if ( m_surfaceNormal.z > 0.9f ) { m_climbableSurface = false; // don't try to build ladders on flat ground } } } if ( ( m_climbableSurface && !m_isCreatingLadder ) || m_navEditMode != NAV_EDIT_CREATE ) { float closestDistSqr = 200.0f * 200.0f; FOR_EACH_LL( m_ladderList, it ) { CNavLadder *ladder = m_ladderList[ it ]; Vector absMin = ladder->m_bottom; Vector absMax = ladder->m_top; Vector left( 0, 0, 0), right(0, 0, 0), up( 0, 0, 0); VectorVectors( ladder->GetNormal(), right, up ); right *= ladder->m_width * 0.5f; left = -right; absMin.x += MIN( left.x, right.x ); absMin.y += MIN( left.y, right.y ); absMax.x += MAX( left.x, right.x ); absMax.y += MAX( left.y, right.y ); Extent e; e.lo = absMin + Vector( -5, -5, -5 ); e.hi = absMax + Vector( 5, 5, 5 ); if ( e.Contains( m_editCursorPos ) ) { m_selectedLadder = ladder; break; } if ( !m_climbableSurface ) continue; Vector p1 = (ladder->m_bottom + ladder->m_top)/2; Vector p2 = m_editCursorPos; float distSqr = p1.DistToSqr( p2 ); if ( distSqr < closestDistSqr ) { m_selectedLadder = ladder; closestDistSqr = distSqr; } } } m_editCursorPos = result.endpos; // find the area the player is pointing at if ( !m_climbableSurface && !m_selectedLadder ) { // Try to clip our trace to nav areas if ( m_grid ) { Vector start = result.startpos; Vector end = result.endpos + 10.0f * dir; // extend a few units into the ground Ray_t ray; ray.Init( start, end, vec3_origin, vec3_origin ); float bestDist = 1.0f; Extent extent; extent.lo = extent.hi = start; extent.Encompass( end ); int loX = WorldToGridX( extent.lo.x ); int loY = WorldToGridY( extent.lo.y ); int hiX = WorldToGridX( extent.hi.x ); int hiY = WorldToGridY( extent.hi.y ); //int navAreasTouched = 0; //int gridBucketsTouched = 0; for( int y = loY; y <= hiY; ++y ) { for( int x = loX; x <= hiX; ++x ) { NavAreaList &areaGrid = m_grid[ x + y*m_gridSizeX ]; //++gridBucketsTouched; FOR_EACH_LL( areaGrid, it ) { //++navAreasTouched; CNavArea *area = areaGrid[ it ]; Vector nw = area->m_extent.lo; Vector se = area->m_extent.hi; Vector ne, sw; ne.x = se.x; ne.y = nw.y; ne.z = area->m_neZ; sw.x = nw.x; sw.y = se.y; sw.z = area->m_swZ; float dist = IntersectRayWithTriangle( ray, nw, ne, se, false ); if ( dist > 0 && dist < bestDist ) { m_selectedArea = area; bestDist = dist; } dist = IntersectRayWithTriangle( ray, se, sw, nw, false ); if ( dist > 0 && dist < bestDist ) { m_selectedArea = area; bestDist = dist; } } } } } //engine->Con_NPrintf( 20, "%d areas queried in %d grid buckets, instead of %d areas", navAreasTouched, gridBucketsTouched, TheNavAreaList.Count() ); // Failing that, get the closest area to the endpoint if ( !m_selectedArea ) { m_selectedArea = TheNavMesh->GetNearestNavArea( result.endpos, false, 500.0f ); } } if ( m_selectedArea ) { float yaw = player->EyeAngles().y; while( yaw > 360.0f ) yaw -= 360.0f; while( yaw < 0.0f ) yaw += 360.0f; if ((yaw < 45.0f || yaw > 315.0f) || (yaw > 135.0f && yaw < 225.0f)) { m_splitEdge = SnapToGrid( result.endpos.y, true ); m_splitAlongX = true; } else { m_splitEdge = SnapToGrid( result.endpos.x, true ); m_splitAlongX = false; } } if ( !m_climbableSurface && !m_isCreatingLadder ) { m_editCursorPos = SnapToGrid( m_editCursorPos ); } return true; } return false; } //-------------------------------------------------------------------------------------------------------------- bool CheckForClimbableSurface( const Vector &start, const Vector &end ) { trace_t result; UTIL_TraceLine( start, end, MASK_PLAYERSOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, &result ); bool climbableSurface = false; if (result.fraction != 1.0f) { climbableSurface = physprops->GetSurfaceData( result.surface.surfaceProps )->game.climbable != 0; if ( !climbableSurface ) { climbableSurface = (result.contents & CONTENTS_LADDER) != 0; } } return climbableSurface; } //-------------------------------------------------------------------------------------------------------------- void StepAlongClimbableSurface( Vector &pos, const Vector &increment, const Vector &probe ) { while ( CheckForClimbableSurface( pos + increment - probe, pos + increment + probe ) ) { pos += increment; } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavBuildLadder( void ) { if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || !m_climbableSurface ) { return; } // We've got a ladder at m_editCursorPos, with a normal of m_surfaceNormal Vector right, up; VectorVectors( -m_surfaceNormal, right, up ); Vector startPos = m_editCursorPos; Vector leftEdge = startPos; Vector rightEdge = startPos; Vector topEdge = startPos; Vector bottomEdge = startPos; // trace to the sides to find the width Vector probe = m_surfaceNormal * -HalfHumanWidth; const float StepSize = 1.0f; StepAlongClimbableSurface( leftEdge, right * -StepSize, probe ); StepAlongClimbableSurface( rightEdge, right * StepSize, probe ); StepAlongClimbableSurface( topEdge, up * StepSize, probe ); StepAlongClimbableSurface( bottomEdge, up * -StepSize, probe ); Vector2D ladderDir = m_surfaceNormal.AsVector2D(); trace_t result; CNavLadder *ladder = new CNavLadder; // compute top & bottom of ladder ladder->m_top = topEdge; ladder->m_bottom = bottomEdge; ladder->m_width = leftEdge.DistTo( rightEdge ); if ( fabs( ladderDir.x ) > fabs( ladderDir.y ) ) { if ( ladderDir.x > 0.0f ) { ladder->SetDir( EAST ); } else { ladder->SetDir( WEST ); } } else { if ( ladderDir.y > 0.0f ) { ladder->SetDir( SOUTH ); } else { ladder->SetDir( NORTH ); } } // adjust top and bottom of ladder to make sure they are reachable // (cs_office has a crate right in front of the base of a ladder) Vector along = ladder->m_top - ladder->m_bottom; float length = along.NormalizeInPlace(); Vector on, out; const float minLadderClearance = 32.0f; // adjust bottom to bypass blockages const float inc = 10.0f; float t; for( t = 0.0f; t <= length; t += inc ) { on = ladder->m_bottom + t * along; out = on + ladder->GetNormal() * minLadderClearance; UTIL_TraceLine( on, out, MASK_PLAYERSOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, &result ); if (result.fraction == 1.0f && !result.startsolid) { // found viable ladder bottom ladder->m_bottom = on; break; } } // adjust top to bypass blockages for( t = 0.0f; t <= length; t += inc ) { on = ladder->m_top - t * along; out = on + ladder->GetNormal() * minLadderClearance; UTIL_TraceLine( on, out, MASK_PLAYERSOLID_BRUSHONLY, NULL, COLLISION_GROUP_NONE, &result ); if (result.fraction == 1.0f && !result.startsolid) { // found viable ladder top ladder->m_top = on; break; } } ladder->m_length = (ladder->m_top - ladder->m_bottom).Length(); ladder->SetDir( ladder->GetDir() ); // now that we've adjusted the top and bottom, re-check the normal ladder->m_bottomArea = NULL; ladder->m_topForwardArea = NULL; ladder->m_topLeftArea = NULL; ladder->m_topRightArea = NULL; ladder->m_topBehindArea = NULL; ladder->ConnectGeneratedLadder(); // add ladder to global list m_ladderList.AddToTail( ladder ); } //-------------------------------------------------------------------------------------------------------------- /** * Flood fills all areas with current place */ class PlaceFloodFillFunctor { public: PlaceFloodFillFunctor( CNavArea *area ) { m_initialPlace = area->GetPlace(); } bool operator() ( CNavArea *area ) { if (area->GetPlace() != m_initialPlace) return false; area->SetPlace( TheNavMesh->GetNavPlace() ); return true; } private: unsigned int m_initialPlace; }; //-------------------------------------------------------------------------------------------------------------- /** * Called when edit mode has just been enabled */ void CNavMesh::OnEditModeStart( void ) { /* CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; const Vector &eye = player->EyePosition(); Vector forward, right, up; AngleVectors( player->EyeAngles(), &forward, &right, &up ); // darken the world so the edit lines show up clearly const float screenHalfSize = 1000.0f; const float screenRange = 50.0f; Vector upLeft = eye + screenRange * forward + screenHalfSize * (up - right); Vector upRight = eye + screenRange * forward + screenHalfSize * (up + right); Vector downLeft = eye + screenRange * forward - screenHalfSize * (up + right); Vector downRight = eye + screenRange * forward + screenHalfSize * (right - up); const int alpha = 200; NDebugOverlay::Triangle( downLeft, upLeft, upRight, 0, 0, 0, alpha, true, 999999.9f ); NDebugOverlay::Triangle( downLeft, upRight, downRight, 0, 0, 0, alpha, true, 999999.9f ); */ } //-------------------------------------------------------------------------------------------------------------- /** * Called when edit mode has just been disabled */ void CNavMesh::OnEditModeEnd( void ) { } //-------------------------------------------------------------------------------------------------------------- /** * Draw navigation areas and edit them * @todo Clean the whole edit system up - its structure is legacy from peculiarities in GoldSrc. */ void CNavMesh::DrawEditMode( void ) { VPROF( "CNavMesh::DrawEditMode" ); CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; const float maxRange = 1000.0f; // 500 #if DEBUG_NAV_NODES if ( nav_show_nodes.GetBool() ) { for ( CNavNode *node = CNavNode::GetFirst(); node != NULL; node = node->GetNext() ) { if ( m_editCursorPos.DistToSqr( *node->GetPosition() ) < 150*150 ) { node->Draw(); } } } #endif // DEBUG_NAV_NODES // draw approach points for marked area if (nav_show_approach_points.GetBool() && GetMarkedArea()) { Vector ap; float halfWidth; for( int i=0; iGetApproachInfoCount(); ++i ) { const CNavArea::ApproachInfo *info = GetMarkedArea()->GetApproachInfo( i ); // compute approach point if (info->hereToNextHow <= GO_WEST) { info->here.area->ComputePortal( info->next.area, (NavDirType)info->hereToNextHow, &ap, &halfWidth ); ap.z = info->next.area->GetZ( ap ); } else { // use the area's center as an approach point ap = info->here.area->GetCenter(); } NavDrawLine( ap + Vector( 0, 0, 50 ), ap + Vector( 10, 0, 0 ), NavApproachPointColor ); NavDrawLine( ap + Vector( 0, 0, 50 ), ap + Vector( -10, 0, 0 ), NavApproachPointColor ); NavDrawLine( ap + Vector( 0, 0, 50 ), ap + Vector( 0, 10, 0 ), NavApproachPointColor ); NavDrawLine( ap + Vector( 0, 0, 50 ), ap + Vector( 0, -10, 0 ), NavApproachPointColor ); } } Vector from, dir; GetEditVectors( &from, &dir ); Vector to = from + maxRange * dir; /* IN_PROGRESS: if ( m_navEditMode != NAV_EDIT_PLACE && nav_snap_to_grid.GetBool() ) { Vector center = SnapToGrid( m_editCursorPos ); const int GridCount = 3; const int GridArraySize = GridCount * 2 + 1; const int GridSize = GetGridSize(); // fill in an array of heights for the grid Vector pos[GridArraySize][GridArraySize]; int x, y; for ( x=0; xGetID() ); NDebugOverlay::ScreenText( 0.5, 0.53, buffer, 255, 255, 0, 128, nav_show_area_info.GetBool() ? 0.1 : 0.5 ); } // draw the ladder we are pointing at and all connected areas m_selectedLadder->DrawLadder(); m_selectedLadder->DrawConnectedAreas(); } if ( m_markedLadder && m_navEditMode != NAV_EDIT_PLACE ) { // draw the "marked" ladder m_markedLadder->DrawLadder(); } if ( m_markedArea && m_navEditMode != NAV_EDIT_PLACE ) { // draw the "marked" area m_markedArea->Draw(); } // find the area the player is pointing at if (m_selectedArea) { m_lastSelectedLadder = NULL; // if area changed, print its ID if ( m_selectedArea != m_lastSelectedArea ) { m_showAreaInfoTimer.Start( nav_show_area_info.GetFloat() ); m_lastSelectedArea = m_selectedArea; } if (m_showAreaInfoTimer.HasStarted() && !m_showAreaInfoTimer.IsElapsed() ) { char buffer[80]; char attrib[80]; char locName[80]; if (m_selectedArea->GetPlace()) { const char *name = TheNavMesh->PlaceToName( m_selectedArea->GetPlace() ); if (name) strcpy( locName, name ); else strcpy( locName, "ERROR" ); } else { locName[0] = '\000'; } if (m_navEditMode == NAV_EDIT_PLACE) { attrib[0] = '\000'; } else { attrib[0] = 0; int attributes = m_selectedArea->GetAttributes(); if ( attributes & NAV_MESH_CROUCH ) Q_strncat( attrib, "CROUCH ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_JUMP ) Q_strncat( attrib, "JUMP ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_PRECISE ) Q_strncat( attrib, "PRECISE ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_NO_JUMP ) Q_strncat( attrib, "NO_JUMP ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_STOP ) Q_strncat( attrib, "STOP ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_RUN ) Q_strncat( attrib, "RUN ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_WALK ) Q_strncat( attrib, "WALK ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_AVOID ) Q_strncat( attrib, "AVOID ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_TRANSIENT ) Q_strncat( attrib, "TRANSIENT ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_DONT_HIDE ) Q_strncat( attrib, "DONT_HIDE ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_STAND ) Q_strncat( attrib, "STAND ", sizeof( attrib ), -1 ); if ( attributes & NAV_MESH_NO_HOSTAGES )Q_strncat( attrib, "NO HOSTAGES ", sizeof( attrib ), -1 ); if ( m_selectedArea->IsBlocked() ) Q_strncat( attrib, "BLOCKED ", sizeof( attrib ), -1 ); if ( m_selectedArea->IsUnderwater() ) Q_strncat( attrib, "UNDERWATER ", sizeof( attrib ), -1 ); } Q_snprintf( buffer, sizeof( buffer ), "Area #%d %s %s\n", m_selectedArea->GetID(), locName, attrib ); NDebugOverlay::ScreenText( 0.5, 0.53, buffer, 255, 255, 0, 128, 0.1 ); // do "place painting" if (m_isPlacePainting) { if (m_selectedArea->GetPlace() != TheNavMesh->GetNavPlace()) { m_selectedArea->SetPlace( TheNavMesh->GetNavPlace() ); player->EmitSound( "Bot.EditSwitchOn" ); } } } if (m_navEditMode == NAV_EDIT_PLACE) { m_selectedArea->DrawConnectedAreas(); } else // normal editing mode { // draw split line const Extent &extent = m_selectedArea->GetExtent(); float yaw = player->EyeAngles().y; while( yaw > 360.0f ) yaw -= 360.0f; while( yaw < 0.0f ) yaw += 360.0f; if (m_splitAlongX) { from.x = extent.lo.x; from.y = m_splitEdge; from.z = m_selectedArea->GetZ( from ); to.x = extent.hi.x; to.y = m_splitEdge; to.z = m_selectedArea->GetZ( to ); } else { from.x = m_splitEdge; from.y = extent.lo.y; from.z = m_selectedArea->GetZ( from ); to.x = m_splitEdge; to.y = extent.hi.y; to.z = m_selectedArea->GetZ( to ); } NavDrawLine( from, to, NavSplitLineColor ); // draw the area we are pointing at and all connected areas m_selectedArea->DrawConnectedAreas(); } } } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::SetEditMode( bool isPlaceMode ) { if ( isPlaceMode ) { m_markedLadder = NULL; m_markedArea = NULL; m_markedCorner = NUM_CORNERS; } m_navEditMode = (isPlaceMode)?NAV_EDIT_PLACE:NAV_EDIT_NORMAL; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::SetPlacePaintingMode( bool painting ) { if ( m_navEditMode == NAV_EDIT_PLACE ) { m_markedLadder = NULL; m_markedArea = NULL; m_markedCorner = NUM_CORNERS; m_isPlacePainting = painting; } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::SetMarkedLadder( CNavLadder *ladder ) { m_markedLadder = ladder; m_markedArea = NULL; m_markedCorner = NUM_CORNERS; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::SetMarkedArea( CNavArea *area ) { m_markedLadder = NULL; m_markedArea = area; m_markedCorner = NUM_CORNERS; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavDelete( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { player->EmitSound( "EDIT_DELETE" ); TheNavAreaList.FindAndRemove( m_selectedArea ); delete m_selectedArea; } else if ( m_selectedLadder ) { player->EmitSound( "EDIT_DELETE" ); m_ladderList.FindAndRemove( m_selectedLadder ); delete m_selectedLadder; } StripNavigationAreas(); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavDeleteMarked( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; CNavArea *markedArea = GetMarkedArea(); if( markedArea ) { player->EmitSound( "EDIT_DELETE" ); TheNavAreaList.FindAndRemove( markedArea ); delete markedArea; } StripNavigationAreas(); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavSplit( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (m_selectedArea->SplitEdit( m_splitAlongX, m_splitEdge )) player->EmitSound( "EDIT_SPLIT.MarkedArea" ); else player->EmitSound( "EDIT_SPLIT.NoMarkedArea" ); } StripNavigationAreas(); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- bool MakeSniperSpots( CNavArea *area ) { if ( !area ) return false; bool splitAlongX; float splitEdge; const float minSplitSize = 2.0f; // ensure the first split is larger than this float sizeX = area->GetSizeX(); float sizeY = area->GetSizeY(); if ( sizeX > GenerationStepSize && sizeX > sizeY ) { splitEdge = round( area->GetExtent().lo.x, GenerationStepSize ); if ( splitEdge < area->GetExtent().lo.x + minSplitSize ) splitEdge += GenerationStepSize; splitAlongX = false; } else if ( sizeY > GenerationStepSize && sizeY > sizeX ) { splitEdge = round( area->GetExtent().lo.y, GenerationStepSize ); if ( splitEdge < area->GetExtent().lo.y + minSplitSize ) splitEdge += GenerationStepSize; splitAlongX = true; } else { return false; } CNavArea *first, *second; if ( !area->SplitEdit( splitAlongX, splitEdge, &first, &second ) ) { return false; } first->Disconnect( second ); second->Disconnect( first ); MakeSniperSpots( first ); MakeSniperSpots( second ); return true; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavMakeSniperSpots( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { // recursively split the area if ( MakeSniperSpots( m_selectedArea ) ) { player->EmitSound( "EDIT_SPLIT.MarkedArea" ); } else { player->EmitSound( "EDIT_SPLIT.NoMarkedArea" ); } } else { player->EmitSound( "EDIT_SPLIT.NoMarkedArea" ); } StripNavigationAreas(); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavMerge( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if ( m_markedArea && m_markedArea != m_selectedArea ) { if ( m_selectedArea->MergeEdit( GetMarkedArea() ) ) player->EmitSound( "EDIT_MERGE.Enable" ); else player->EmitSound( "EDIT_MERGE.Disable" ); } else { Msg( "To merge, mark an area, highlight a second area, then invoke the merge command" ); player->EmitSound( "EDIT_MERGE.Disable" ); } } StripNavigationAreas(); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavMark( const CCommand &args ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_markedArea || m_markedLadder ) { // Unmark area or ladder player->EmitSound( "EDIT_MARK.Enable" ); Msg("Area unmarked.\n"); SetMarkedArea( NULL ); } else if ( args.ArgC() > 1 ) { const char *areaIDNameToMark = args[1]; if( areaIDNameToMark != NULL ) { unsigned int areaIDToMark = atoi(areaIDNameToMark); if( areaIDToMark != 0 ) { CNavArea *areaToMark = NULL; FOR_EACH_LL( TheNavAreaList, nit ) { if( TheNavAreaList[nit]->GetID() == areaIDToMark ) { areaToMark = TheNavAreaList[nit]; break; } } if( areaToMark ) { SetMarkedArea( areaToMark ); int connected = 0; connected += GetMarkedArea()->GetAdjacentCount( NORTH ); connected += GetMarkedArea()->GetAdjacentCount( SOUTH ); connected += GetMarkedArea()->GetAdjacentCount( EAST ); connected += GetMarkedArea()->GetAdjacentCount( WEST ); Msg( "Marked Area is connected to %d other Areas\n", connected ); } } } } else if ( m_selectedArea ) { // Mark an area player->EmitSound( "EDIT_MARK.Disable" ); SetMarkedArea( m_selectedArea ); int connected = 0; connected += GetMarkedArea()->GetAdjacentCount( NORTH ); connected += GetMarkedArea()->GetAdjacentCount( SOUTH ); connected += GetMarkedArea()->GetAdjacentCount( EAST ); connected += GetMarkedArea()->GetAdjacentCount( WEST ); Msg( "Marked Area is connected to %d other Areas\n", connected ); } else if ( m_selectedLadder ) { // Mark a ladder player->EmitSound( "EDIT_MARK.Disable" ); SetMarkedLadder( m_selectedLadder ); int connected = 0; connected += m_markedLadder->m_topForwardArea != NULL; connected += m_markedLadder->m_topLeftArea != NULL; connected += m_markedLadder->m_topRightArea != NULL; connected += m_markedLadder->m_topBehindArea != NULL; connected += m_markedLadder->m_bottomArea != NULL; Msg( "Marked Ladder is connected to %d Areas\n", connected ); } m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavUnmark( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; player->EmitSound( "EDIT_MARK.Enable" ); SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavBeginArea( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_PLACE ) { player->EmitSound( "EDIT_END_AREA.NotCreating" ); return; } GetActiveNavArea(); if (m_navEditMode == NAV_EDIT_CREATE) { m_navEditMode = NAV_EDIT_NORMAL; player->EmitSound( "EDIT_BEGIN_AREA.Creating" ); } else if ( m_isCreatingLadder ) { m_isCreatingLadder = false; player->EmitSound( "EDIT_BEGIN_AREA.Creating" ); } else if ( m_climbableSurface ) { player->EmitSound( "EDIT_BEGIN_AREA.NotCreating" ); m_isCreatingLadder = true; // m_ladderAnchor starting corner m_ladderAnchor = m_editCursorPos; m_ladderNormal = m_surfaceNormal; } else { player->EmitSound( "EDIT_BEGIN_AREA.NotCreating" ); m_navEditMode = NAV_EDIT_CREATE; // m_anchor starting corner m_anchor = m_editCursorPos; } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavEndArea( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_PLACE ) { player->EmitSound( "EDIT_END_AREA.NotCreating" ); return; } if ( m_navEditMode == NAV_EDIT_CREATE ) { // create the new nav area Vector endPos = m_editCursorPos; endPos.z = m_anchor.z; CNavArea *newArea = new CNavArea( m_anchor, endPos ); TheNavAreaList.AddToTail( newArea ); TheNavMesh->AddNavArea( newArea ); player->EmitSound( "EDIT_END_AREA.Creating" ); if ( nav_create_place_on_ground.GetBool() ) { newArea->PlaceOnGround( NUM_CORNERS ); } // if we have a marked area, inter-connect the two if (GetMarkedArea()) { const Extent &extent = GetMarkedArea()->GetExtent(); if (m_anchor.x > extent.hi.x && m_editCursorPos.x > extent.hi.x) { GetMarkedArea()->ConnectTo( newArea, EAST ); newArea->ConnectTo( GetMarkedArea(), WEST ); } else if (m_anchor.x < extent.lo.x && m_editCursorPos.x < extent.lo.x) { GetMarkedArea()->ConnectTo( newArea, WEST ); newArea->ConnectTo( GetMarkedArea(), EAST ); } else if (m_anchor.y > extent.hi.y && m_editCursorPos.y > extent.hi.y) { GetMarkedArea()->ConnectTo( newArea, SOUTH ); newArea->ConnectTo( GetMarkedArea(), NORTH ); } else if (m_anchor.y < extent.lo.y && m_editCursorPos.y < extent.lo.y) { GetMarkedArea()->ConnectTo( newArea, NORTH ); newArea->ConnectTo( GetMarkedArea(), SOUTH ); } // propogate marked area to new area SetMarkedArea( newArea ); } m_navEditMode = NAV_EDIT_NORMAL; } else if ( m_isCreatingLadder ) { player->EmitSound( "EDIT_END_AREA.Creating" ); // the two points defining the ladder are m_ladderAnchor and m_editCursorPos. The normal is m_ladderNormal. Vector mins, maxs; mins.x = MIN( m_ladderAnchor.x, m_editCursorPos.x ); mins.y = MIN( m_ladderAnchor.y, m_editCursorPos.y ); mins.z = MIN( m_ladderAnchor.z, m_editCursorPos.z ); maxs.x = MAX( m_ladderAnchor.x, m_editCursorPos.x ); maxs.y = MAX( m_ladderAnchor.y, m_editCursorPos.y ); maxs.z = MAX( m_ladderAnchor.z, m_editCursorPos.z ); Vector2D ladderDir = m_ladderNormal.AsVector2D(); CreateLadder( mins, maxs, &ladderDir ); m_isCreatingLadder = false; } else { player->EmitSound( "EDIT_END_AREA.NotCreating" ); } m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavConnect( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if ( m_markedLadder ) { m_markedLadder->ConnectTo( m_selectedArea ); player->EmitSound( "EDIT_CONNECT.Added" ); } else if ( m_markedArea ) { NavDirType dir = GetMarkedArea()->ComputeDirection( &m_editCursorPos ); if (dir == NUM_DIRECTIONS) { player->EmitSound( "EDIT_CONNECT.AllDirections" ); } else { m_markedArea->ConnectTo( m_selectedArea, dir ); player->EmitSound( "EDIT_CONNECT.Added" ); } } else { Msg( "To connect areas, mark an area, highlight a second area, then invoke the connect command. Make sure the cursor is directly north, south, east, or west of the marked area." ); player->EmitSound( "EDIT_CONNECT.AllDirections" ); } } else if ( m_selectedLadder ) { if ( m_markedArea ) { m_markedArea->ConnectTo( m_selectedLadder ); player->EmitSound( "EDIT_CONNECT.Added" ); } else { Msg( "To connect areas, mark an area, highlight a second area, then invoke the connect command. Make sure the cursor is directly north, south, east, or west of the marked area." ); player->EmitSound( "EDIT_CONNECT.AllDirections" ); } } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavDisconnect( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if ( m_markedArea ) { m_markedArea->Disconnect( m_selectedArea ); m_selectedArea->Disconnect( m_markedArea ); player->EmitSound( "EDIT_DISCONNECT.MarkedArea" ); } else { if ( m_markedLadder ) { m_markedLadder->Disconnect( m_selectedArea ); m_selectedArea->Disconnect( m_markedLadder ); player->EmitSound( "EDIT_DISCONNECT.MarkedArea" ); } else { Msg( "To disconnect areas, mark an area, highlight a second area, then invoke the disconnect command. This will remove all connections between the two areas." ); player->EmitSound( "EDIT_DISCONNECT.NoMarkedArea" ); } } } else if ( m_selectedLadder ) { if ( m_markedArea ) { m_markedArea->Disconnect( m_selectedLadder ); m_selectedLadder->Disconnect( m_markedArea ); player->EmitSound( "EDIT_DISCONNECT.MarkedArea" ); } else { Msg( "To disconnect areas, mark an area, highlight a second area, then invoke the disconnect command. This will remove all connections between the two areas." ); player->EmitSound( "EDIT_DISCONNECT.NoMarkedArea" ); } } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavSplice( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (GetMarkedArea()) { if (m_selectedArea->SpliceEdit( GetMarkedArea() )) player->EmitSound( "EDIT_SPLICE.MarkedArea" ); else player->EmitSound( "EDIT_SPLICE.NoMarkedArea" ); } else { Msg( "To splice, mark an area, highlight a second area, then invoke the splice command to create an area between them" ); player->EmitSound( "EDIT_SPLICE.NoMarkedArea" ); } } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavToggleAttribute( NavAttributeType attribute ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { player->EmitSound( "EDIT.ToggleAttribute" ); m_selectedArea->SetAttributes( m_selectedArea->GetAttributes() ^ attribute ); if ( attribute == NAV_MESH_TRANSIENT ) { if ( m_selectedArea->GetAttributes() & NAV_MESH_TRANSIENT ) { m_transientAreas.AddToTail( m_selectedArea ); } else { m_transientAreas.FindAndRemove( m_selectedArea ); } } } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavTogglePlaceMode( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder ) return; player->EmitSound( "EDIT_TOGGLE_PLACE_MODE" ); m_navEditMode = (m_navEditMode == NAV_EDIT_PLACE) ? NAV_EDIT_NORMAL : NAV_EDIT_PLACE; SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavPlaceFloodFill( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode != NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { PlaceFloodFillFunctor pff( m_selectedArea ); SearchSurroundingAreas( m_selectedArea, m_selectedArea->GetCenter(), pff ); } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavPlacePick( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode != NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { player->EmitSound( "EDIT_PLACE_PICK" ); TheNavMesh->SetNavPlace( m_selectedArea->GetPlace() ); } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavTogglePlacePainting( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode != NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (m_isPlacePainting) { m_isPlacePainting = false; player->EmitSound( "Bot.EditSwitchOff" ); } else { m_isPlacePainting = true; player->EmitSound( "Bot.EditSwitchOn" ); // paint the initial area m_selectedArea->SetPlace( TheNavMesh->GetNavPlace() ); } } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavMarkUnnamed( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (GetMarkedArea()) { player->EmitSound( "EDIT_MARK_UNNAMED.Enable" ); SetMarkedArea( NULL ); } else { SetMarkedArea( NULL ); FOR_EACH_LL( TheNavAreaList, it ) { CNavArea *area = TheNavAreaList[ it ]; if ( area->GetPlace() == 0 ) { SetMarkedArea( area ); break; } } if ( !GetMarkedArea() ) { player->EmitSound( "EDIT_MARK_UNNAMED.NoMarkedArea" ); } else { player->EmitSound( "EDIT_MARK_UNNAMED.MarkedArea" ); int connected = 0; connected += GetMarkedArea()->GetAdjacentCount( NORTH ); connected += GetMarkedArea()->GetAdjacentCount( SOUTH ); connected += GetMarkedArea()->GetAdjacentCount( EAST ); connected += GetMarkedArea()->GetAdjacentCount( WEST ); int totalUnnamedAreas = 0; FOR_EACH_LL( TheNavAreaList, it ) { CNavArea *area = TheNavAreaList[ it ]; if ( area->GetPlace() == 0 ) { ++totalUnnamedAreas; } } Msg( "Marked Area is connected to %d other Areas - there are %d total unnamed areas\n", connected, totalUnnamedAreas ); } } } m_markedCorner = NUM_CORNERS; // clear the corner selection } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavPickArea( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; if ( m_markedLadder ) { // Unmark ladder player->EmitSound( "EDIT_MARK.Enable" ); Msg("Ladder unmarked.\n"); SetMarkedArea( NULL ); m_markedCorner = NUM_CORNERS; // clear the corner selection return; } else if ( m_selectedArea ) { NavCornerType bestCorner = m_selectedArea->GetCornerUnderCursor(); if ( m_markedCorner == bestCorner && m_selectedArea == m_markedArea ) { // Unmark area player->EmitSound( "EDIT_MARK.Enable" ); Msg("Area unmarked.\n"); SetMarkedArea( NULL ); m_markedCorner = NUM_CORNERS; // clear the corner selection return; } // Mark an area SetMarkedArea( m_selectedArea ); m_markedCorner = bestCorner; int connected = 0; connected += GetMarkedArea()->GetAdjacentCount( NORTH ); connected += GetMarkedArea()->GetAdjacentCount( SOUTH ); connected += GetMarkedArea()->GetAdjacentCount( EAST ); connected += GetMarkedArea()->GetAdjacentCount( WEST ); player->EmitSound( "EDIT_MARK.Disable" ); Msg( "Marked Area is connected to %d other Areas\n", connected ); } else if ( m_selectedLadder ) { // Mark a ladder player->EmitSound( "EDIT_MARK.Disable" ); SetMarkedLadder( m_selectedLadder ); int connected = 0; connected += m_markedLadder->m_topForwardArea != NULL; connected += m_markedLadder->m_topLeftArea != NULL; connected += m_markedLadder->m_topRightArea != NULL; connected += m_markedLadder->m_topBehindArea != NULL; connected += m_markedLadder->m_bottomArea != NULL; Msg( "Marked Ladder is connected to %d Areas\n", connected ); m_markedCorner = NUM_CORNERS; // clear the corner selection } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavResizeHorizontal( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( !m_markedArea ) { CommandNavPickArea(); // try to grab an area first } if ( !m_markedArea ) { player->EmitSound( "EDIT_END_AREA.Creating" ); return; } if ( m_navEditMode == NAV_EDIT_RESIZE_HORIZONTAL ) { m_navEditMode = NAV_EDIT_NORMAL; player->EmitSound( "EDIT_BEGIN_AREA.Creating" ); return; } if ( m_navEditMode != NAV_EDIT_NORMAL ) { return; } m_navEditMode = NAV_EDIT_RESIZE_HORIZONTAL; player->EmitSound( "EDIT_BEGIN_AREA.NotCreating" ); m_anchor = m_editCursorPos; if ( nav_snap_to_grid.GetBool() ) { m_anchor = SnapToGrid( m_anchor ); } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavResizeVertical( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( !m_markedArea ) { player->EmitSound( "EDIT_END_AREA.Creating" ); return; } if ( m_navEditMode == NAV_EDIT_RESIZE_VERTICAL ) { m_navEditMode = NAV_EDIT_NORMAL; player->EmitSound( "EDIT_BEGIN_AREA.Creating" ); return; } if ( m_navEditMode != NAV_EDIT_NORMAL ) { return; } m_navEditMode = NAV_EDIT_RESIZE_VERTICAL; player->EmitSound( "EDIT_BEGIN_AREA.NotCreating" ); m_anchor = m_editCursorPos; if ( nav_snap_to_grid.GetBool() ) { m_anchor = SnapToGrid( m_anchor ); } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavResizeEnd( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode != NAV_EDIT_RESIZE_HORIZONTAL && m_navEditMode != NAV_EDIT_RESIZE_VERTICAL ) { player->EmitSound( "EDIT_END_AREA.Creating" ); return; } if ( !m_markedArea ) { player->EmitSound( "EDIT_END_AREA.Creating" ); m_navEditMode = NAV_EDIT_NORMAL; return; } Vector cornerPos = ( m_markedCorner == NUM_CORNERS ) ? m_markedArea->GetCenter() : m_markedArea->GetCorner( m_markedCorner ); if ( m_navEditMode == NAV_EDIT_RESIZE_HORIZONTAL ) { // find the new pos Vector from, dir; GetEditVectors( &from, &dir ); const float dist = 10000.0f; float t = IntersectRayWithAAPlane( from, from + dir * dist, 2, 1.0f, cornerPos.z ); if ( t > 0 && t < 1 ) { Vector newPos = from + dir * dist * t; if ( nav_snap_to_grid.GetBool() ) { newPos = SnapToGrid( newPos ); } m_markedArea->SetCorner( m_markedCorner, newPos ); } } player->EmitSound( "EDIT_END_AREA.NotCreating" ); m_navEditMode = NAV_EDIT_NORMAL; } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavCornerSelect( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (GetMarkedArea()) { int corner = (m_markedCorner + 1) % (NUM_CORNERS + 1); m_markedCorner = (NavCornerType)corner; player->EmitSound( "EDIT_SELECT_CORNER.MarkedArea" ); } else { player->EmitSound( "EDIT_SELECT_CORNER.NoMarkedArea" ); } } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavCornerRaise( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (GetMarkedArea()) { GetMarkedArea()->RaiseCorner( m_markedCorner, 1 ); player->EmitSound( "EDIT_MOVE_CORNER.MarkedArea" ); } else { player->EmitSound( "EDIT_MOVE_CORNER.NoMarkedArea" ); } } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavCornerLower( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { if (GetMarkedArea()) { GetMarkedArea()->RaiseCorner( m_markedCorner, -1 ); player->EmitSound( "EDIT_MOVE_CORNER.MarkedArea" ); } else { player->EmitSound( "EDIT_MOVE_CORNER.NoMarkedArea" ); } } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavCornerPlaceOnGround( const CCommand &args ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedArea ) { float inset = 0.0f; if ( args.ArgC() == 2 ) inset = atof( args[1] ); if ( m_markedArea ) { m_markedArea->PlaceOnGround( m_markedCorner, inset ); } else { m_selectedArea->PlaceOnGround( NUM_CORNERS, inset ); } player->EmitSound( "EDIT_MOVE_CORNER.MarkedArea" ); } else { player->EmitSound( "EDIT_MOVE_CORNER.NoMarkedArea" ); } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavWarpToMark( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; if (GetMarkedArea()) { if ( ( player->IsDead() || player->IsObserver() ) && player->GetObserverMode() == OBS_MODE_ROAMING ) { Vector origin = GetMarkedArea()->GetCenter() + Vector( 0, 0, 0.75f * HumanHeight ); UTIL_SetOrigin( player, origin ); } else { player->EmitSound( "EDIT_WARP_TO_MARK" ); } } else if (GetMarkedLadder()) { CNavLadder *ladder = GetMarkedLadder(); if ( ( player->IsDead() || player->IsObserver() ) && player->GetObserverMode() == OBS_MODE_ROAMING ) { Vector origin = (ladder->m_top + ladder->m_bottom)/2; origin.x += ladder->GetNormal().x * GenerationStepSize; origin.y += ladder->GetNormal().y * GenerationStepSize; UTIL_SetOrigin( player, origin ); } else { player->EmitSound( "EDIT_WARP_TO_MARK" ); } } else { player->EmitSound( "EDIT_WARP_TO_MARK" ); } } //-------------------------------------------------------------------------------------------------------------- void CNavMesh::CommandNavLadderFlip( void ) { CBasePlayer *player = UTIL_GetListenServerHost(); if (player == NULL) return; if ( m_navEditMode == NAV_EDIT_CREATE || m_isCreatingLadder || m_navEditMode == NAV_EDIT_PLACE ) return; GetActiveNavArea(); if ( m_selectedLadder ) { CNavArea *area; // flip direction player->EmitSound( "EDIT_MOVE_CORNER.MarkedArea" ); m_selectedLadder->SetDir( OppositeDirection( m_selectedLadder->GetDir() ) ); // and reverse ladder's area pointers area = m_selectedLadder->m_topBehindArea; m_selectedLadder->m_topBehindArea = m_selectedLadder->m_topForwardArea; m_selectedLadder->m_topForwardArea = area; area = m_selectedLadder->m_topRightArea; m_selectedLadder->m_topRightArea = m_selectedLadder->m_topLeftArea; m_selectedLadder->m_topLeftArea = area; } SetMarkedArea( NULL ); // unmark the mark area m_markedCorner = NUM_CORNERS; // clear the corner selection } //--------------------------------------------------------------------------------------------------------------