mirror of
https://github.com/nillerusr/source-engine.git
synced 2025-01-11 23:57:59 +00:00
686 lines
18 KiB
C++
686 lines
18 KiB
C++
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose: Renders a cone for spotlight entities. Only renders when the parent
|
||
|
// entity is selected.
|
||
|
//
|
||
|
//=============================================================================//
|
||
|
|
||
|
#include "stdafx.h"
|
||
|
#include "Box3D.h"
|
||
|
#include "fgdlib/HelperInfo.h"
|
||
|
#include "MapDefs.h" // dvs: For COORD_NOTINIT
|
||
|
#include "MapEntity.h"
|
||
|
#include "MapLightCone.h"
|
||
|
#include "Render3D.h"
|
||
|
#include "Material.h"
|
||
|
#include "materialsystem/imaterialsystem.h"
|
||
|
#include "TextureSystem.h"
|
||
|
#include "hammer.h"
|
||
|
#include "Options.h"
|
||
|
|
||
|
|
||
|
// memdbgon must be the last include file in a .cpp file!!!
|
||
|
#include <tier0/memdbgon.h>
|
||
|
|
||
|
#define NUM_LIGHTCONE_ZONES 5
|
||
|
|
||
|
|
||
|
IMPLEMENT_MAPCLASS(CMapLightCone)
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Factory function. Used for creating a CMapLightCone helper from a
|
||
|
// set of string parameters from the FGD file.
|
||
|
// Input : *pInfo - Pointer to helper info class which gives us information
|
||
|
// about how to create the helper.
|
||
|
// Output : Returns a pointer to the helper, NULL if an error occurs.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapClass *CMapLightCone::Create(CHelperInfo *pHelperInfo, CMapEntity *pParent)
|
||
|
{
|
||
|
CMapLightCone *new1=new CMapLightCone;
|
||
|
if( new1 != NULL )
|
||
|
{
|
||
|
//
|
||
|
// The first parameter should be the inner fov key name. If it isn't
|
||
|
// there we assume "_inner_cone".
|
||
|
//
|
||
|
const char *pszKeyName = pHelperInfo->GetParameter(0);
|
||
|
if (pszKeyName != NULL)
|
||
|
{
|
||
|
strcpy(new1->m_szInnerConeKeyName, pszKeyName);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
strcpy(new1->m_szInnerConeKeyName, "_inner_cone");
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// The second parameter should be the outer fov key name. If it isn't
|
||
|
// there we assume "_cone".
|
||
|
//
|
||
|
pszKeyName = pHelperInfo->GetParameter(1);
|
||
|
if (pszKeyName != NULL)
|
||
|
{
|
||
|
strcpy(new1->m_szOuterConeKeyName, pszKeyName);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
strcpy(new1->m_szOuterConeKeyName, "_cone");
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// The third parameter should be the color of the light. If it isn't
|
||
|
// there we assume "_light".
|
||
|
//
|
||
|
pszKeyName = pHelperInfo->GetParameter(2);
|
||
|
if (pszKeyName != NULL)
|
||
|
{
|
||
|
strcpy(new1->m_szColorKeyName, pszKeyName);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
strcpy(new1->m_szColorKeyName, "_light");
|
||
|
}
|
||
|
|
||
|
pszKeyName = pHelperInfo->GetParameter(3);
|
||
|
if (pszKeyName != NULL)
|
||
|
{
|
||
|
new1->m_flPitchScale = Q_atof( pszKeyName );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
new1->m_flPitchScale = 1.0f;
|
||
|
}
|
||
|
}
|
||
|
return new1;
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapLightCone::CMapLightCone(void)
|
||
|
{
|
||
|
m_fQuadraticAttn = 1;
|
||
|
m_fLinearAttn = 0;
|
||
|
m_fConstantAttn = 0;
|
||
|
m_bPitchSet = false;
|
||
|
m_fPitch = 0;
|
||
|
m_fFocus = 1;
|
||
|
m_flPitchScale = 1;
|
||
|
|
||
|
m_fBrightness = 100;
|
||
|
m_fInnerConeAngle = 0;
|
||
|
m_fOuterConeAngle = 45;
|
||
|
|
||
|
m_fFiftyPercentDistance = -1; // disabled - use attenuation
|
||
|
m_Angles.Init();
|
||
|
SignalUpdate( EVTYPE_LIGHTING_CHANGED );
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Destructor. Deletes faces allocated by BuildCone.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapLightCone::~CMapLightCone(void)
|
||
|
{
|
||
|
for (int i = 0; i < m_Faces.Count(); i++)
|
||
|
{
|
||
|
CMapFace *pFace = m_Faces.Element(i);
|
||
|
delete pFace;
|
||
|
}
|
||
|
SignalUpdate( EVTYPE_LIGHTING_CHANGED );
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Builds the light cone faces in local space. Does NOT call CalcBounds,
|
||
|
// because that CalcBounds updates the parent, which causes problems
|
||
|
// in the undo system.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::BuildCone(void)
|
||
|
{
|
||
|
//
|
||
|
// Delete the current face list.
|
||
|
//
|
||
|
for (int i = 0; i < m_Faces.Count(); i++)
|
||
|
{
|
||
|
CMapFace *pFace = m_Faces.Element(i);
|
||
|
delete pFace;
|
||
|
}
|
||
|
m_Faces.RemoveAll();
|
||
|
|
||
|
//
|
||
|
// Make sure at least one of the lighting coefficients is nonzero.
|
||
|
//
|
||
|
if ((m_fQuadraticAttn == 0) && (m_fLinearAttn == 0) && (m_fConstantAttn == 0))
|
||
|
{
|
||
|
m_fConstantAttn = 1;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Solve for the lighting scale factor by which the brightness will be multiplied.
|
||
|
//
|
||
|
float fScaleFactor = m_fQuadraticAttn * 10000 + m_fLinearAttn * 100 + m_fConstantAttn;
|
||
|
if (fScaleFactor == 0)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Calculate the distances from the light origin to the various zones.
|
||
|
//
|
||
|
float fOffsetDist = 0;
|
||
|
// Constant attenuation factor doesn't actually offset the cone yet. If it does, uncomment this:
|
||
|
//SolveQuadratic(fOffsetDist, 0, m_fQuadraticAttn, m_fLinearAttn, -m_fConstantAttn);
|
||
|
|
||
|
float fZoneDist[NUM_LIGHTCONE_ZONES];
|
||
|
memset( fZoneDist, 0, sizeof( fZoneDist ) );
|
||
|
fZoneDist[0] = 0;
|
||
|
SolveQuadratic(fZoneDist[1], 0.25 * fScaleFactor, m_fQuadraticAttn, m_fLinearAttn, m_fConstantAttn);
|
||
|
SolveQuadratic(fZoneDist[2], fScaleFactor, m_fQuadraticAttn, m_fLinearAttn, m_fConstantAttn);
|
||
|
SolveQuadratic(fZoneDist[3], 4 * fScaleFactor, m_fQuadraticAttn, m_fLinearAttn, m_fConstantAttn);
|
||
|
SolveQuadratic(fZoneDist[4], Options.view3d.fLightConeLength * fScaleFactor, m_fQuadraticAttn, m_fLinearAttn, m_fConstantAttn);
|
||
|
|
||
|
//
|
||
|
// there's no cone if it's greater then 90 degrees
|
||
|
//
|
||
|
if (m_fOuterConeAngle < 90)
|
||
|
{
|
||
|
|
||
|
//
|
||
|
// Calculate the cone radius at each zone.
|
||
|
//
|
||
|
float fZoneRadius[NUM_LIGHTCONE_ZONES];
|
||
|
for (int i = 0; i < NUM_LIGHTCONE_ZONES; i++)
|
||
|
{
|
||
|
fZoneRadius[i] = (fOffsetDist + fZoneDist[i]) * tan(DEG2RAD(m_fOuterConeAngle));
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Build the new face list using the new parameters.
|
||
|
//
|
||
|
float fStepSize = 360.0 / 15.0;
|
||
|
for (int nZone = 0; nZone < NUM_LIGHTCONE_ZONES - 1; nZone++)
|
||
|
{
|
||
|
float fSin0 = 0;
|
||
|
float fCos0 = 1;
|
||
|
|
||
|
float fTopDist = fZoneDist[nZone];
|
||
|
float fBottomDist = fZoneDist[nZone + 1];
|
||
|
|
||
|
float fTopRadius = fZoneRadius[nZone];
|
||
|
float fBottomRadius = fZoneRadius[nZone + 1];
|
||
|
|
||
|
for (int fAngle = fStepSize; fAngle <= 361; fAngle += fStepSize)
|
||
|
{
|
||
|
float fSin1 = sin(DEG2RAD(fAngle));
|
||
|
float fCos1 = cos(DEG2RAD(fAngle));
|
||
|
|
||
|
Vector Points[4];
|
||
|
|
||
|
Points[0][2] = fBottomRadius * fCos1;
|
||
|
Points[0][1] = fBottomRadius * fSin1;
|
||
|
Points[0][0] = fBottomDist;
|
||
|
|
||
|
Points[1][2] = fBottomRadius * fCos0;
|
||
|
Points[1][1] = fBottomRadius * fSin0;
|
||
|
Points[1][0] = fBottomDist;
|
||
|
|
||
|
Points[2][2] = fTopRadius * fCos0;
|
||
|
Points[2][1] = fTopRadius * fSin0;
|
||
|
Points[2][0] = fTopDist;
|
||
|
|
||
|
int nPoints = 3;
|
||
|
if (fTopRadius != 0)
|
||
|
{
|
||
|
Points[3][2] = fTopRadius * fCos1;
|
||
|
Points[3][1] = fTopRadius * fSin1;
|
||
|
Points[3][0] = fTopDist;
|
||
|
nPoints = 4;
|
||
|
}
|
||
|
|
||
|
CMapFace *pFace = new CMapFace;
|
||
|
pFace->SetRenderColor(r * (1 - nZone / (float)NUM_LIGHTCONE_ZONES), g * (1 - nZone / (float)NUM_LIGHTCONE_ZONES), b * (1 - nZone / (float)NUM_LIGHTCONE_ZONES));
|
||
|
pFace->SetRenderAlpha(180);
|
||
|
pFace->CreateFace(Points, nPoints);
|
||
|
pFace->RenderUnlit(true);
|
||
|
m_Faces.AddToTail(pFace);
|
||
|
|
||
|
fSin0 = fSin1;
|
||
|
fCos0 = fCos1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Lobe's aren't defined for > 90
|
||
|
//
|
||
|
if (m_fOuterConeAngle > 90)
|
||
|
return;
|
||
|
|
||
|
//
|
||
|
// Build the a face list that shows light-angle falloff
|
||
|
//
|
||
|
float fStepSize = 360.0 / 15.0;
|
||
|
float fPitchStepSize = 90.0 / 15.0;
|
||
|
float fFocusRadius0 = 0;
|
||
|
float fFocusDist0 = fZoneDist[1];
|
||
|
float fInnerDot = cos(DEG2RAD(m_fInnerConeAngle));
|
||
|
float fOuterDot = cos(DEG2RAD(m_fOuterConeAngle));
|
||
|
|
||
|
for (float fPitch = fPitchStepSize; fPitch < m_fOuterConeAngle + fPitchStepSize; fPitch += fPitchStepSize)
|
||
|
{
|
||
|
float fSin0 = 0;
|
||
|
float fCos0 = 1;
|
||
|
|
||
|
// clamp to edge of cone
|
||
|
if (fPitch > m_fOuterConeAngle)
|
||
|
fPitch = m_fOuterConeAngle;
|
||
|
|
||
|
float fIllumination = 0;
|
||
|
if (fPitch <= m_fInnerConeAngle)
|
||
|
{
|
||
|
fIllumination = 1.0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
float fPitchDot = cos(DEG2RAD(fPitch));
|
||
|
|
||
|
fIllumination = (fPitchDot - fOuterDot) / (fInnerDot - fOuterDot);
|
||
|
|
||
|
if ((m_fFocus != 1) && (m_fFocus != 0))
|
||
|
{
|
||
|
fIllumination = pow( fIllumination, m_fFocus );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// cosine falloff ^ exponent
|
||
|
|
||
|
// draw as lobe
|
||
|
float fFocusDist1 = cos(DEG2RAD(fPitch)) * fIllumination * fZoneDist[1];
|
||
|
float fFocusRadius1 = sin(DEG2RAD(fPitch)) * fIllumination * fZoneDist[1];
|
||
|
|
||
|
// draw as disk
|
||
|
// float fFocusDist1 = fZoneDist[1];
|
||
|
// float fFocusRadius1 = sin(DEG2RAD(fPitch)) * fZoneRadius[1] / sin(DEG_RAD * m_fConeAngle);
|
||
|
|
||
|
for (int fAngle = fStepSize; fAngle <= 361; fAngle += fStepSize)
|
||
|
{
|
||
|
float fSin1 = sin(DEG2RAD(fAngle));
|
||
|
float fCos1 = cos(DEG2RAD(fAngle));
|
||
|
|
||
|
Vector Points[4];
|
||
|
|
||
|
Points[0][2] = fFocusRadius1 * fCos0;
|
||
|
Points[0][1] = fFocusRadius1 * fSin0;
|
||
|
Points[0][0] = fFocusDist1;
|
||
|
|
||
|
Points[1][2] = fFocusRadius1 * fCos1;
|
||
|
|
||
|
Points[1][1] = fFocusRadius1 * fSin1;
|
||
|
Points[1][0] = fFocusDist1;
|
||
|
|
||
|
Points[2][2] = fFocusRadius0 * fCos1;
|
||
|
Points[2][1] = fFocusRadius0 * fSin1;
|
||
|
Points[2][0] = fFocusDist0;
|
||
|
|
||
|
int nPoints = 3;
|
||
|
if (fFocusRadius0 != 0)
|
||
|
{
|
||
|
Points[3][2] = fFocusRadius0 * fCos0;
|
||
|
Points[3][1] = fFocusRadius0 * fSin0;
|
||
|
Points[3][0] = fFocusDist0;
|
||
|
nPoints = 4;
|
||
|
}
|
||
|
|
||
|
CMapFace *pFace = new CMapFace;
|
||
|
pFace->SetRenderColor(r * fIllumination, g * fIllumination, b * fIllumination);
|
||
|
pFace->SetRenderAlpha(180);
|
||
|
pFace->CreateFace(Points, nPoints);
|
||
|
pFace->RenderUnlit(true);
|
||
|
m_Faces.AddToTail(pFace);
|
||
|
|
||
|
fSin0 = fSin1;
|
||
|
fCos0 = fCos1;
|
||
|
}
|
||
|
fFocusRadius0 = fFocusRadius1;
|
||
|
fFocusDist0 = fFocusDist1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : bFullUpdate -
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::CalcBounds(BOOL bFullUpdate)
|
||
|
{
|
||
|
CMapClass::CalcBounds(bFullUpdate);
|
||
|
|
||
|
//
|
||
|
// HACK: Update our origin to stick to our parent.
|
||
|
//
|
||
|
if (m_pParent != NULL)
|
||
|
{
|
||
|
GetParent()->GetOrigin(m_Origin);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Pretend to be very small for the 2D view. Won't be necessary when 2D
|
||
|
// rendering is done in the map classes.
|
||
|
//
|
||
|
m_Render2DBox.ResetBounds();
|
||
|
m_Render2DBox.UpdateBounds(m_Origin);
|
||
|
|
||
|
SetCullBoxFromFaceList( &m_Faces );
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : Angles -
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::GetAngles(QAngle &Angles)
|
||
|
{
|
||
|
Angles = m_Angles;
|
||
|
|
||
|
if (m_bPitchSet)
|
||
|
{
|
||
|
Angles[PITCH] = m_fPitch;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Output : CMapClass
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapClass *CMapLightCone::Copy(bool bUpdateDependencies)
|
||
|
{
|
||
|
CMapLightCone *pCopy = new CMapLightCone;
|
||
|
|
||
|
if (pCopy != NULL)
|
||
|
{
|
||
|
pCopy->CopyFrom(this, bUpdateDependencies);
|
||
|
}
|
||
|
|
||
|
return(pCopy);
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : pObject -
|
||
|
// Output : CMapClass
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapClass *CMapLightCone::CopyFrom(CMapClass *pObject, bool bUpdateDependencies)
|
||
|
{
|
||
|
Assert(pObject->IsMapClass(MAPCLASS_TYPE(CMapLightCone)));
|
||
|
CMapLightCone *pFrom = (CMapLightCone *)pObject;
|
||
|
|
||
|
CMapClass::CopyFrom(pObject, bUpdateDependencies);
|
||
|
|
||
|
m_fBrightness = pFrom->m_fBrightness;
|
||
|
|
||
|
m_fQuadraticAttn = pFrom->m_fQuadraticAttn;
|
||
|
m_fLinearAttn = pFrom->m_fLinearAttn;
|
||
|
m_fConstantAttn = pFrom->m_fConstantAttn;
|
||
|
m_flPitchScale = pFrom->m_flPitchScale;
|
||
|
|
||
|
m_fInnerConeAngle = pFrom->m_fInnerConeAngle;
|
||
|
m_fOuterConeAngle = pFrom->m_fOuterConeAngle;
|
||
|
|
||
|
m_Angles = pFrom->m_Angles;
|
||
|
|
||
|
m_bPitchSet = pFrom->m_bPitchSet;
|
||
|
m_fPitch = pFrom->m_fPitch;
|
||
|
|
||
|
m_fFocus = pFrom->m_fFocus;
|
||
|
|
||
|
m_fFiftyPercentDistance = pFrom->m_fFiftyPercentDistance;
|
||
|
m_fZeroPercentDistance = pFrom->m_fZeroPercentDistance;
|
||
|
m_LightColor = pFrom->m_LightColor;
|
||
|
|
||
|
Q_strncpy( m_szColorKeyName, pFrom->m_szColorKeyName, sizeof( m_szColorKeyName ) );
|
||
|
Q_strncpy( m_szInnerConeKeyName, pFrom->m_szInnerConeKeyName, sizeof( m_szInnerConeKeyName ) );
|
||
|
Q_strncpy( m_szOuterConeKeyName, pFrom->m_szOuterConeKeyName, sizeof( m_szOuterConeKeyName ) );
|
||
|
|
||
|
BuildCone();
|
||
|
|
||
|
SignalUpdate( EVTYPE_LIGHTING_CHANGED );
|
||
|
return(this);
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Notifies that this object's parent entity has had a key value change.
|
||
|
// Input : szKey - The key that changed.
|
||
|
// szValue - The new value of the key.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::OnParentKeyChanged(const char *szKey, const char *szValue)
|
||
|
{
|
||
|
bool bRebuild = true;
|
||
|
|
||
|
if (!stricmp(szKey, "angles"))
|
||
|
{
|
||
|
sscanf(szValue, "%f %f %f", &m_Angles[PITCH], &m_Angles[YAW], &m_Angles[ROLL]);
|
||
|
}
|
||
|
else if (!stricmp(szKey, m_szColorKeyName))
|
||
|
{
|
||
|
int nRed;
|
||
|
int nGreen;
|
||
|
int nBlue;
|
||
|
int nBrightness;
|
||
|
sscanf(szValue, "%d %d %d %d", &nRed, &nGreen, &nBlue, &nBrightness);
|
||
|
|
||
|
r = m_LightColor.x = nRed;
|
||
|
g = m_LightColor.y = nGreen;
|
||
|
b = m_LightColor.z = nBlue;
|
||
|
m_fBrightness = nBrightness;
|
||
|
}
|
||
|
else if (!stricmp(szKey, "pitch"))
|
||
|
{
|
||
|
// Pitch
|
||
|
m_bPitchSet = true;
|
||
|
m_fPitch = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_constant_attn"))
|
||
|
{
|
||
|
// Constant attenuation
|
||
|
m_fConstantAttn = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_linear_attn"))
|
||
|
{
|
||
|
// Linear attenuation
|
||
|
m_fLinearAttn = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_quadratic_attn"))
|
||
|
{
|
||
|
// Quadratic attenuation
|
||
|
m_fQuadraticAttn = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_exponent"))
|
||
|
{
|
||
|
// Focus
|
||
|
m_fFocus = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_fifty_percent_distance"))
|
||
|
{
|
||
|
// Focus
|
||
|
m_fFiftyPercentDistance = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, "_zero_percent_distance"))
|
||
|
{
|
||
|
// Focus
|
||
|
m_fZeroPercentDistance = atof(szValue);
|
||
|
}
|
||
|
else if (!stricmp(szKey, m_szInnerConeKeyName) || !stricmp(szKey, m_szOuterConeKeyName))
|
||
|
{
|
||
|
// check both of these together since they might be the same key.
|
||
|
if( !stricmp(szKey, m_szInnerConeKeyName ))
|
||
|
{
|
||
|
// Inner Cone angle
|
||
|
m_fInnerConeAngle = atof(szValue);
|
||
|
}
|
||
|
if( !stricmp(szKey, m_szOuterConeKeyName ))
|
||
|
{
|
||
|
// Outer Cone angle
|
||
|
m_fOuterConeAngle = atof(szValue);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
bRebuild = false;
|
||
|
}
|
||
|
|
||
|
if (bRebuild)
|
||
|
{
|
||
|
SignalUpdate( EVTYPE_LIGHTING_CHANGED );
|
||
|
BuildCone();
|
||
|
PostUpdate(Notify_Changed);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Called after the entire map has been loaded. This allows the object
|
||
|
// to perform any linking with other map objects or to do other operations
|
||
|
// that require all world objects to be present.
|
||
|
// Input : pWorld - The world that we are in.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::PostloadWorld(CMapWorld *pWorld)
|
||
|
{
|
||
|
CMapClass::PostloadWorld(pWorld);
|
||
|
|
||
|
BuildCone();
|
||
|
SignalUpdate( EVTYPE_LIGHTING_CHANGED );
|
||
|
CalcBounds();
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : pRender -
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CMapLightCone::Render3D(CRender3D *pRender)
|
||
|
{
|
||
|
if (m_pParent->IsSelected())
|
||
|
{
|
||
|
CMatRenderContextPtr pRenderContext( MaterialSystemInterface() );
|
||
|
pRenderContext->MatrixMode(MATERIAL_MODEL);
|
||
|
pRenderContext->PushMatrix();
|
||
|
|
||
|
pRenderContext->Translate(m_Origin[0], m_Origin[1], m_Origin[2]);
|
||
|
|
||
|
QAngle Angles;
|
||
|
GetAngles(Angles);
|
||
|
|
||
|
pRenderContext->Rotate(Angles[YAW], 0, 0, 1);
|
||
|
pRenderContext->Rotate(m_flPitchScale * Angles[PITCH], 0, -1, 0);
|
||
|
pRenderContext->Rotate(Angles[ROLL], 1, 0, 0);
|
||
|
|
||
|
if (
|
||
|
(pRender->GetCurrentRenderMode() != RENDER_MODE_LIGHT_PREVIEW2) &&
|
||
|
(pRender->GetCurrentRenderMode() != RENDER_MODE_LIGHT_PREVIEW_RAYTRACED) &&
|
||
|
(GetSelectionState() != SELECT_MODIFY )
|
||
|
)
|
||
|
{
|
||
|
// Render the cone faces flatshaded.
|
||
|
pRender->PushRenderMode( RENDER_MODE_TRANSLUCENT_FLAT );
|
||
|
|
||
|
for (int i = 0; i < m_Faces.Count(); i++)
|
||
|
{
|
||
|
CMapFace *pFace = m_Faces.Element(i);
|
||
|
pFace->Render3D(pRender);
|
||
|
}
|
||
|
|
||
|
pRender->PopRenderMode();
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Render the cone faces in yellow wireframe (on top)
|
||
|
//
|
||
|
pRender->PushRenderMode( RENDER_MODE_WIREFRAME );
|
||
|
|
||
|
for (int i = 0; i < m_Faces.Count(); i++)
|
||
|
{
|
||
|
CMapFace *pFace = m_Faces.Element(i);
|
||
|
pFace->Render3D(pRender);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Restore the default rendering mode.
|
||
|
//
|
||
|
pRender->PopRenderMode();
|
||
|
|
||
|
pRenderContext->PopMatrix();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : File -
|
||
|
// bRMF -
|
||
|
// Output : int
|
||
|
//-----------------------------------------------------------------------------
|
||
|
int CMapLightCone::SerializeRMF(std::fstream &File, BOOL bRMF)
|
||
|
{
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
// Input : File -
|
||
|
// bRMF -
|
||
|
// Output : int
|
||
|
//-----------------------------------------------------------------------------
|
||
|
int CMapLightCone::SerializeMAP(std::fstream &File, BOOL bRMF)
|
||
|
{
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Solves a quadratic equation with the given coefficients.
|
||
|
// Input : x - Receives solution.
|
||
|
// y - Root to solve for.
|
||
|
// A, B, C - Quadratic, linear, and constant coefficients.
|
||
|
// Output : Returns true if a real solution was found, false if not.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
bool CMapLightCone::SolveQuadratic(float &x, float y, float A, float B, float C)
|
||
|
{
|
||
|
C -= y;
|
||
|
|
||
|
if (A == 0)
|
||
|
{
|
||
|
if (B != 0)
|
||
|
{
|
||
|
x = -C / B;
|
||
|
return(true);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
float fDeterminant = B * B - 4 * A * C;
|
||
|
if (fDeterminant > 0)
|
||
|
{
|
||
|
x = (-B + sqrt(fDeterminant)) / (2 * A);
|
||
|
return(true);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Never select anything because of this helper.
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CMapClass *CMapLightCone::PrepareSelection(SelectMode_t eSelectMode)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|