mirror of
https://github.com/nillerusr/source-engine.git
synced 2025-01-11 23:57:59 +00:00
547 lines
15 KiB
C++
547 lines
15 KiB
C++
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose:
|
||
|
//
|
||
|
// $NoKeywords: $
|
||
|
//=============================================================================//
|
||
|
|
||
|
#include <stdafx.h>
|
||
|
#include "MapWorld.h"
|
||
|
#include "GlobalFunctions.h"
|
||
|
#include "MainFrm.h"
|
||
|
#include "ToolOverlay.h"
|
||
|
#include "MapDoc.h"
|
||
|
#include "History.h"
|
||
|
#include "CollisionUtils.h"
|
||
|
#include "cmodel.h"
|
||
|
#include "MapView3D.h"
|
||
|
#include "MapView2D.h"
|
||
|
#include "MapSolid.h"
|
||
|
#include "Camera.h"
|
||
|
#include "ObjectProperties.h" // FIXME: For ObjectProperties::RefreshData
|
||
|
#include "Selection.h"
|
||
|
|
||
|
// memdbgon must be the last include file in a .cpp file!!!
|
||
|
#include <tier0/memdbgon.h>
|
||
|
|
||
|
#define OVERLAY_TOOL_SNAP_DISTANCE 35.0f
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Constructor
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CToolOverlay::CToolOverlay()
|
||
|
{
|
||
|
m_bDragging = false;
|
||
|
m_pActiveOverlay = NULL;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Destructor
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CToolOverlay::~CToolOverlay()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::OnActivate()
|
||
|
{
|
||
|
m_bDragging = false;
|
||
|
m_pActiveOverlay = NULL;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::OnDeactivate()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::OnLMouseUp3D( CMapView3D *pView, UINT nFlags, const Vector2D &vPoint )
|
||
|
{
|
||
|
// Post drag events.
|
||
|
PostDrag();
|
||
|
|
||
|
// Update the entity properties dialog.
|
||
|
GetMainWnd()->pObjectProperties->MarkDataDirty();
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::OnLMouseDown3D( CMapView3D *pView, UINT nFlags, const Vector2D &vPoint )
|
||
|
{
|
||
|
// Handle the overlay "handle" selection.
|
||
|
if ( HandleSelection( pView, vPoint ) )
|
||
|
{
|
||
|
PreDrag();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Handle adding and removing overlay entities from the selection list.
|
||
|
OverlaySelection( pView, nFlags, vPoint );
|
||
|
|
||
|
// Handle the overlay creation and placement (if we hit a solid).
|
||
|
ULONG ulFace;
|
||
|
CMapClass *pObject = NULL;
|
||
|
if ( ( pObject = pView->NearestObjectAt( vPoint, ulFace ) ) != NULL )
|
||
|
{
|
||
|
CMapSolid *pSolid = dynamic_cast<CMapSolid*>( pObject );
|
||
|
if ( pSolid )
|
||
|
{
|
||
|
return CreateOverlay( pSolid, ulFace, pView, vPoint );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::OnMouseMove3D( CMapView3D *pView, UINT nFlags, const Vector2D &vPoint )
|
||
|
{
|
||
|
if ( m_bDragging )
|
||
|
{
|
||
|
bool bShift = ( ( GetKeyState( VK_SHIFT ) & 0x8000 ) != 0 );
|
||
|
|
||
|
// Build the ray and drag the overlay handle to the impact point.
|
||
|
const CCamera *pCamera = pView->GetCamera();
|
||
|
if ( pCamera )
|
||
|
{
|
||
|
Vector vecStart, vecEnd;
|
||
|
pView->BuildRay( vPoint, vecStart, vecEnd );
|
||
|
OnDrag( vecStart, vecEnd, bShift );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::CreateOverlay( CMapSolid *pSolid, ULONG iFace, CMapView3D *pView, Vector2D point )
|
||
|
{
|
||
|
// Build a ray to trace against the face that they clicked on to
|
||
|
// find the point of intersection.
|
||
|
Vector vecStart, vecEnd;
|
||
|
pView->BuildRay( point, vecStart, vecEnd );
|
||
|
|
||
|
Vector vecHitPos, vecHitNormal;
|
||
|
CMapFace *pFace = pSolid->GetFace( iFace );
|
||
|
if( pFace->TraceLine( vecHitPos, vecHitNormal, vecStart, vecEnd ) )
|
||
|
{
|
||
|
// Create and initialize the "entity" --> "overlay."
|
||
|
CMapEntity *pEntity = new CMapEntity;
|
||
|
pEntity->SetKeyValue( "material", GetDefaultTextureName() );
|
||
|
pEntity->SetPlaceholder( TRUE );
|
||
|
pEntity->SetOrigin( vecHitPos );
|
||
|
pEntity->SetClass( "info_overlay" );
|
||
|
|
||
|
// Add the entity to the world.
|
||
|
m_pDocument->AddObjectToWorld( pEntity );
|
||
|
|
||
|
// Setup "history."
|
||
|
GetHistory()->MarkUndoPosition( NULL, "Create Overlay" );
|
||
|
GetHistory()->KeepNew( pEntity );
|
||
|
|
||
|
// Initialize the overlay.
|
||
|
InitOverlay( pEntity, pFace );
|
||
|
|
||
|
pEntity->CalcBounds( TRUE );
|
||
|
|
||
|
// Add to selection list.
|
||
|
m_pDocument->SelectObject( pEntity, scSelect );
|
||
|
m_bEmpty = false;
|
||
|
|
||
|
// Set modified and update views.
|
||
|
m_pDocument->SetModifiedFlag();
|
||
|
|
||
|
m_pShoreline = NULL;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::InitOverlay( CMapEntity *pEntity, CMapFace *pFace )
|
||
|
{
|
||
|
// Valid face?
|
||
|
if ( !pFace )
|
||
|
return;
|
||
|
|
||
|
const CMapObjectList *pChildren = pEntity->GetChildren();
|
||
|
|
||
|
FOR_EACH_OBJ( *pChildren, pos )
|
||
|
{
|
||
|
CMapOverlay *pOverlay = dynamic_cast<CMapOverlay*>( pChildren->Element(pos) );
|
||
|
if ( pOverlay )
|
||
|
{
|
||
|
pOverlay->Basis_Init( pFace );
|
||
|
pOverlay->Handles_Init( pFace );
|
||
|
pOverlay->SideList_Init( pFace );
|
||
|
pOverlay->SetOverlayType( OVERLAY_TYPE_GENERIC );
|
||
|
pOverlay->SetLoaded( true );
|
||
|
pOverlay->CalcBounds( true );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::OverlaySelection( CMapView3D *pView, UINT nFlags, const Vector2D &vPoint )
|
||
|
{
|
||
|
CMapObjectList aSelectionList;
|
||
|
m_pDocument->GetSelection()->ClearHitList();
|
||
|
|
||
|
// Find out how many (and what) map objects are under the point clicked on.
|
||
|
HitInfo_t Objects[MAX_PICK_HITS];
|
||
|
int nHits = pView->ObjectsAt( vPoint, Objects, sizeof( Objects ) / sizeof( Objects[0] ) );
|
||
|
if ( nHits != 0 )
|
||
|
{
|
||
|
// We now have an array of pointers to CMapAtoms. Any that can be upcast to CMapClass objects?
|
||
|
for ( int iHit = 0; iHit < nHits; ++iHit )
|
||
|
{
|
||
|
CMapClass *pMapClass = dynamic_cast<CMapClass*>( Objects[iHit].pObject );
|
||
|
if ( pMapClass )
|
||
|
{
|
||
|
aSelectionList.AddToTail( pMapClass );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Did we hit anything?
|
||
|
if ( !aSelectionList.Count() )
|
||
|
{
|
||
|
m_pDocument->SelectFace( NULL, 0, scClear );
|
||
|
m_pDocument->SelectObject( NULL, scClear|scSaveChanges );
|
||
|
SetEmpty();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Find overlays.
|
||
|
bool bUpdateViews = false;
|
||
|
|
||
|
SelectMode_t eSelectMode = m_pDocument->GetSelection()->GetMode();
|
||
|
|
||
|
FOR_EACH_OBJ( aSelectionList, pos )
|
||
|
{
|
||
|
CMapClass *pObject = aSelectionList.Element( pos );
|
||
|
CMapClass *pHitObject = pObject->PrepareSelection( eSelectMode );
|
||
|
if ( pHitObject )
|
||
|
{
|
||
|
if ( pHitObject->IsMapClass( MAPCLASS_TYPE( CMapEntity ) ) )
|
||
|
{
|
||
|
const CMapObjectList *pChildren = pHitObject->GetChildren();
|
||
|
FOR_EACH_OBJ( *pChildren, pos2 )
|
||
|
{
|
||
|
CMapOverlay *pOverlay = dynamic_cast<CMapOverlay*>( pChildren->Element(pos2) );
|
||
|
if ( pOverlay )
|
||
|
{
|
||
|
m_pDocument->GetSelection()->AddHit( pHitObject );
|
||
|
m_bEmpty = false;
|
||
|
|
||
|
UINT cmd = scClear | scSelect | scSaveChanges;
|
||
|
if (nFlags & MK_CONTROL)
|
||
|
{
|
||
|
cmd = scToggle;
|
||
|
}
|
||
|
m_pDocument->SelectObject( pHitObject, cmd );
|
||
|
bUpdateViews = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Update the views.
|
||
|
if ( bUpdateViews )
|
||
|
{
|
||
|
m_pDocument->UpdateAllViews( MAPVIEW_UPDATE_OBJECTS );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::OnContextMenu2D( CMapView2D *pView, UINT nFlags, const Vector2D &vPoint )
|
||
|
{
|
||
|
static CMenu menu, menuOverlay;
|
||
|
static bool bInit = false;
|
||
|
|
||
|
if ( !bInit )
|
||
|
{
|
||
|
// Create the menu.
|
||
|
menu.LoadMenu( IDR_POPUPS );
|
||
|
menuOverlay.Attach( ::GetSubMenu( menu.m_hMenu, 6 ) );
|
||
|
bInit = true;
|
||
|
}
|
||
|
|
||
|
if ( !pView->PointInClientRect(vPoint) )
|
||
|
return false;
|
||
|
|
||
|
if (!IsEmpty())
|
||
|
{
|
||
|
if ( HitTest( pView, vPoint, false ) )
|
||
|
{
|
||
|
CPoint ptScreen( vPoint.x,vPoint.y);
|
||
|
pView->ClientToScreen(&ptScreen);
|
||
|
menuOverlay.TrackPopupMenu( TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_LEFTALIGN, ptScreen.x, ptScreen.y, pView );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::UpdateTranslation( const Vector &vUpdate, UINT nFlags)
|
||
|
{
|
||
|
// if( m_bBoxSelecting )
|
||
|
// return Box3D::UpdateTranslation( pt, nFlags );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::HandlesReset( void )
|
||
|
{
|
||
|
// Go through selection list and reset overlay handles.
|
||
|
const CMapObjectList *pSelection = m_pDocument->GetSelection()->GetList();
|
||
|
for( int iSelection = 0; iSelection < pSelection->Count(); ++iSelection )
|
||
|
{
|
||
|
CMapClass *pMapClass = pSelection->Element( iSelection );
|
||
|
if ( pMapClass && pMapClass->IsMapClass( MAPCLASS_TYPE( CMapEntity ) ) )
|
||
|
{
|
||
|
const CMapObjectList *pChildren = pMapClass->GetChildren();
|
||
|
FOR_EACH_OBJ( *pChildren, pos )
|
||
|
{
|
||
|
CMapOverlay *pOverlay = dynamic_cast<CMapOverlay*>( pChildren->Element(pos) );
|
||
|
if ( pOverlay )
|
||
|
{
|
||
|
pOverlay->HandlesReset();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::HandleSelection( CMapView *pView, const Vector2D &vPoint )
|
||
|
{
|
||
|
// Reset the hit overlay.
|
||
|
m_pActiveOverlay = NULL;
|
||
|
|
||
|
// Go through selection list and test all overlay's handles and set the
|
||
|
// "hit" overlay current.
|
||
|
const CMapObjectList *pSelection = m_pDocument->GetSelection()->GetList();
|
||
|
for ( int iSelection = 0; iSelection < pSelection->Count(); ++iSelection )
|
||
|
{
|
||
|
CMapClass *pMapClass = pSelection->Element( iSelection );
|
||
|
if ( pMapClass && pMapClass->IsMapClass( MAPCLASS_TYPE( CMapEntity ) ) )
|
||
|
{
|
||
|
const CMapObjectList *pChildren = pMapClass->GetChildren();
|
||
|
FOR_EACH_OBJ( *pChildren, pos )
|
||
|
{
|
||
|
CMapOverlay *pOverlay = dynamic_cast<CMapOverlay*>( pChildren->Element(pos) );
|
||
|
if ( pOverlay && pOverlay->IsSelected() )
|
||
|
{
|
||
|
if ( pOverlay->HandlesHitTest( pView, vPoint ) )
|
||
|
{
|
||
|
m_pActiveOverlay = pOverlay;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( !m_pActiveOverlay )
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::HandleSnap( CMapOverlay *pOverlay, Vector &vecHandlePt )
|
||
|
{
|
||
|
Vector vecTmp;
|
||
|
for ( int i = 0; i < OVERLAY_HANDLES_COUNT; i++ )
|
||
|
{
|
||
|
pOverlay->GetHandlePos( i, vecTmp );
|
||
|
vecTmp -= vecHandlePt;
|
||
|
float flDist = vecTmp.Length();
|
||
|
|
||
|
if ( flDist < OVERLAY_TOOL_SNAP_DISTANCE )
|
||
|
{
|
||
|
// Snap!
|
||
|
pOverlay->GetHandlePos( i, vecHandlePt );
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CToolOverlay::HandleInBBox( CMapOverlay *pOverlay, Vector const &vecHandlePt )
|
||
|
{
|
||
|
Vector vecMin, vecMax;
|
||
|
pOverlay->GetCullBox( vecMin, vecMax );
|
||
|
|
||
|
for ( int iAxis = 0; iAxis < 3; iAxis++ )
|
||
|
{
|
||
|
vecMin[iAxis] -= OVERLAY_TOOL_SNAP_DISTANCE;
|
||
|
vecMax[iAxis] += OVERLAY_TOOL_SNAP_DISTANCE;
|
||
|
|
||
|
if( ( vecHandlePt[iAxis] < vecMin[iAxis] ) || ( vecHandlePt[iAxis] > vecMax[iAxis] ) )
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::SnapHandle( Vector &vecHandlePt )
|
||
|
{
|
||
|
CMapWorld *pWorld = GetActiveWorld();
|
||
|
if ( !pWorld )
|
||
|
return;
|
||
|
|
||
|
EnumChildrenPos_t posWorld;
|
||
|
CMapClass *pChild = pWorld->GetFirstDescendent( posWorld );
|
||
|
while ( pChild )
|
||
|
{
|
||
|
CMapEntity *pEntity = dynamic_cast<CMapEntity*>( pChild );
|
||
|
if ( pEntity )
|
||
|
{
|
||
|
const CMapObjectList *pChildren = pEntity->GetChildren();
|
||
|
FOR_EACH_OBJ( *pChildren, pos )
|
||
|
{
|
||
|
CMapOverlay *pOverlay = dynamic_cast<CMapOverlay*>( pChildren->Element(pos) );
|
||
|
if ( pOverlay && pOverlay != m_pActiveOverlay && pOverlay->IsSelected() )
|
||
|
{
|
||
|
// Intersection test and attempt to snap
|
||
|
if ( HandleInBBox( pOverlay, vecHandlePt ) )
|
||
|
{
|
||
|
if ( HandleSnap( pOverlay, vecHandlePt ) )
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
pChild = pWorld->GetNextDescendent( posWorld );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::OnDrag( Vector const &vecRayStart, Vector const &vecRayEnd, bool bShift )
|
||
|
{
|
||
|
// Get the current overlay.
|
||
|
CMapOverlay *pOverlay = m_pActiveOverlay;
|
||
|
if ( !pOverlay )
|
||
|
return;
|
||
|
|
||
|
// Get a list of faces and test for "impact."
|
||
|
Vector vecImpact( 0.0f, 0.0f, 0.0f );
|
||
|
Vector vecImpactNormal( 0.0f, 0.0f, 0.0f );
|
||
|
CMapFace *pFace = NULL;
|
||
|
|
||
|
int nFaceCount = pOverlay->GetFaceCount();
|
||
|
int iFace;
|
||
|
for ( iFace = 0; iFace < nFaceCount; iFace++ )
|
||
|
{
|
||
|
pFace = pOverlay->GetFace( iFace );
|
||
|
if ( pFace )
|
||
|
{
|
||
|
if ( pFace->TraceLineInside( vecImpact, vecImpactNormal, vecRayStart, vecRayEnd ) )
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Test for impact (face index = count mean no impact).
|
||
|
if ( iFace == nFaceCount )
|
||
|
return;
|
||
|
|
||
|
if ( bShift )
|
||
|
{
|
||
|
SnapHandle( vecImpact );
|
||
|
}
|
||
|
|
||
|
// Pass the new handle position to the overlay.
|
||
|
pOverlay->HandlesDragTo( vecImpact, pFace );
|
||
|
|
||
|
m_pDocument->UpdateAllViews( MAPVIEW_UPDATE_ONLY_3D );
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::PreDrag( void )
|
||
|
{
|
||
|
m_bDragging = true;
|
||
|
SetupHandleDragUndo();
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Renders the cordon tool in the 3D view.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::RenderTool3D(CRender3D *pRender)
|
||
|
{
|
||
|
// TODO render tool handles here and not in overlay rendering code
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::PostDrag( void )
|
||
|
{
|
||
|
if ( !m_bDragging )
|
||
|
return;
|
||
|
|
||
|
m_bDragging = false;
|
||
|
|
||
|
// Get the current overlay.
|
||
|
CMapOverlay *pOverlay = m_pActiveOverlay;
|
||
|
if ( pOverlay )
|
||
|
{
|
||
|
pOverlay->DoClip();
|
||
|
pOverlay->CenterEntity();
|
||
|
pOverlay->PostUpdate( Notify_Changed );
|
||
|
m_pDocument->UpdateAllViews( MAPVIEW_UPDATE_OBJECTS );
|
||
|
}
|
||
|
|
||
|
// Reset the overlay handles.
|
||
|
HandlesReset();
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CToolOverlay::SetupHandleDragUndo( void )
|
||
|
{
|
||
|
// Get the current overlay.
|
||
|
CMapOverlay *pOverlay = m_pActiveOverlay;
|
||
|
if ( pOverlay )
|
||
|
{
|
||
|
CMapEntity *pEntity = ( CMapEntity* )pOverlay->GetParent();
|
||
|
if ( pEntity )
|
||
|
{
|
||
|
// Setup for drag undo.
|
||
|
GetHistory()->MarkUndoPosition( m_pDocument->GetSelection()->GetList(), "Drag Overlay Handle" );
|
||
|
GetHistory()->Keep( ( CMapClass* )pEntity );
|
||
|
}
|
||
|
}
|
||
|
}
|