mirror of
https://github.com/alliedmodders/hl2sdk.git
synced 2025-01-12 03:32:11 +08:00
4698 lines
150 KiB
C++
4698 lines
150 KiB
C++
//===== Copyright © 1996-2005, Valve Corporation, All rights reserved. ======//
|
|
//
|
|
// Purpose: Responsible for drawing the scene
|
|
//
|
|
// $NoKeywords: $
|
|
//===========================================================================//
|
|
|
|
|
|
#include "cbase.h"
|
|
#include "view.h"
|
|
#include "iviewrender.h"
|
|
#include "view_shared.h"
|
|
#include "ivieweffects.h"
|
|
#include "iinput.h"
|
|
#include "model_types.h"
|
|
#include "clientsideeffects.h"
|
|
#include "particlemgr.h"
|
|
#include "viewrender.h"
|
|
#include "iclientmode.h"
|
|
#include "voice_status.h"
|
|
#include "glow_overlay.h"
|
|
#include "materialsystem/imesh.h"
|
|
#include "materialsystem/ITexture.h"
|
|
#include "materialsystem/IMaterial.h"
|
|
#include "materialsystem/IMaterialVar.h"
|
|
#include "materialsystem/IColorCorrection.h"
|
|
#include "DetailObjectSystem.h"
|
|
#include "tier0/vprof.h"
|
|
#include "datacache/imdlcache.h"
|
|
#include "engine/IEngineTrace.h"
|
|
#include "engine/ivmodelinfo.h"
|
|
#include "vstdlib/icommandline.h"
|
|
|
|
#include "view_scene.h"
|
|
#include "particles_ez.h"
|
|
#include "engine/IStaticPropMgr.h"
|
|
#include "engine/ivdebugoverlay.h"
|
|
#include "smoke_fog_overlay.h"
|
|
#include "c_pixel_visibility.h"
|
|
#include "ClientEffectPrecacheSystem.h"
|
|
#include "c_rope.h"
|
|
#include "c_effects.h"
|
|
#include "smoke_fog_overlay.h"
|
|
#include "materialsystem/imaterialsystemhardwareconfig.h"
|
|
#include "vgui_int.h"
|
|
#include "ienginevgui.h"
|
|
#include "datacache/imdlcache.h"
|
|
#include "ScreenSpaceEffects.h"
|
|
|
|
#if defined( HL2_CLIENT_DLL ) || defined( CSTRIKE_DLL )
|
|
#define USE_MONITORS
|
|
#endif
|
|
|
|
// GR
|
|
#include "rendertexture.h"
|
|
#ifdef _XBOX
|
|
#include "tier0/mem.h"
|
|
#endif
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
extern void ComputeCameraVariables( const Vector &vecOrigin, const QAngle &vecAngles,
|
|
Vector *pVecForward, Vector *pVecRight, Vector *pVecUp, VMatrix *pMatCamInverse );
|
|
|
|
static ConVar cl_drawmonitors( "cl_drawmonitors", "1" );
|
|
static ConVar cl_overdraw_test( "cl_overdraw_test", "0", FCVAR_CHEAT | FCVAR_NEVER_AS_STRING );
|
|
static ConVar r_eyewaterepsilon( "r_eyewaterepsilon", "7.0f", FCVAR_CHEAT );
|
|
|
|
static void SetClearColorToFogColor( unsigned char ucFogColor[3] )
|
|
{
|
|
materials->GetFogColor( ucFogColor );
|
|
if( g_pMaterialSystemHardwareConfig->GetHDRType() == HDR_TYPE_INTEGER )
|
|
{
|
|
float scale = LinearToGammaFullRange( materials->GetToneMappingScaleLinear().x );
|
|
ucFogColor[0] *= scale;
|
|
ucFogColor[1] *= scale;
|
|
ucFogColor[2] *= scale;
|
|
}
|
|
materials->ClearColor4ub( ucFogColor[0], ucFogColor[1], ucFogColor[2], 255 );
|
|
}
|
|
|
|
// Used to verify frame syncing.
|
|
void GenerateOverdrawForTesting()
|
|
{
|
|
if ( !cl_overdraw_test.GetInt() )
|
|
return;
|
|
|
|
for ( int i=0; i < 40; i++ )
|
|
{
|
|
g_SmokeFogOverlayAlpha = 20 / 255.0;
|
|
DrawSmokeFogOverlay();
|
|
}
|
|
g_SmokeFogOverlayAlpha = 0;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Convars related to controlling rendering
|
|
//-----------------------------------------------------------------------------
|
|
static ConVar cl_maxrenderable_dist("cl_maxrenderable_dist", "3000", FCVAR_CHEAT, "Max distance from the camera at which things will be rendered" );
|
|
|
|
ConVar r_updaterefracttexture( "r_updaterefracttexture", "1" );
|
|
|
|
ConVar r_entityclips( "r_entityclips", "1" ); //FIXME: Nvidia drivers before 81.94 on cards that support user clip planes will have problems with this, require driver update? Detect and disable?
|
|
|
|
// Matches the version in the engine
|
|
static ConVar r_drawopaqueworld( "r_drawopaqueworld", "1", FCVAR_CHEAT );
|
|
static ConVar r_drawtranslucentworld( "r_drawtranslucentworld", "1", FCVAR_CHEAT );
|
|
static ConVar r_3dsky( "r_3dsky","1", 0, "Enable the rendering of 3d sky boxes" );
|
|
static ConVar r_skybox( "r_skybox","1", FCVAR_CHEAT, "Enable the rendering of sky boxes" );
|
|
ConVar r_drawviewmodel( "r_drawviewmodel","1", FCVAR_CHEAT );
|
|
static ConVar r_drawtranslucentrenderables( "r_drawtranslucentrenderables", "1", FCVAR_CHEAT );
|
|
static ConVar r_drawopaquerenderables( "r_drawopaquerenderables", "1", FCVAR_CHEAT );
|
|
|
|
static ConVar r_flashlightdrawdepth( "r_flashlightdrawdepth", "0" );
|
|
|
|
ConVar r_DrawDetailProps( "r_DrawDetailProps", "1", FCVAR_CHEAT, "0=Off, 1=Normal, 2=Wireframe" );
|
|
|
|
ConVar mat_bloomscale( "mat_bloomscale", "1" );
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Convars related to fog color
|
|
//-----------------------------------------------------------------------------
|
|
static ConVar fog_override( "fog_override", "0", FCVAR_CHEAT );
|
|
// set any of these to use the maps fog
|
|
static ConVar fog_start( "fog_start", "-1" );
|
|
static ConVar fog_end( "fog_end", "-1" );
|
|
static ConVar fog_color( "fog_color", "-1 -1 -1" );
|
|
static ConVar fog_enable( "fog_enable", "1" );
|
|
static ConVar fog_startskybox( "fog_startskybox", "-1" );
|
|
static ConVar fog_endskybox( "fog_endskybox", "-1" );
|
|
static ConVar fog_colorskybox( "fog_colorskybox", "-1 -1 -1" );
|
|
static ConVar fog_enableskybox( "fog_enableskybox", "1" );
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Water-related convars
|
|
//-----------------------------------------------------------------------------
|
|
static ConVar r_debugcheapwater( "r_debugcheapwater", "0", FCVAR_CHEAT );
|
|
static ConVar r_waterforceexpensive( "r_waterforceexpensive", "0" );
|
|
static ConVar r_waterforcereflectentities( "r_waterforcereflectentities", "0" );
|
|
static ConVar r_WaterDrawRefraction( "r_WaterDrawRefraction", "1", 0, "Enable water refraction" );
|
|
static ConVar r_WaterDrawReflection( "r_WaterDrawReflection", "1", 0, "Enable water reflection" );
|
|
static ConVar r_ForceWaterLeaf( "r_ForceWaterLeaf", "1", 0, "Enable for optimization to water - considers view in leaf under water for purposes of culling" );
|
|
static ConVar mat_drawwater( "mat_drawwater", "1", FCVAR_CHEAT );
|
|
static ConVar mat_clipz( "mat_clipz", "1" );
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// debugging
|
|
//-----------------------------------------------------------------------------
|
|
static ConVar r_visocclusion( "r_visocclusion", "0", FCVAR_CHEAT );
|
|
// (the engine owns this cvar).
|
|
ConVar mat_wireframe( "mat_wireframe", "0", FCVAR_CHEAT );
|
|
|
|
// sv_cheats is replicated and lives in the engine. re-declaring it here without FCVAR_REPLICATED
|
|
// breaks the linkage, and declaring it here with FCVAR_REPLICATED wants it to be declared in the server also.
|
|
const ConVar *sv_cheats = NULL;
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// debugging overlays
|
|
//-----------------------------------------------------------------------------
|
|
static ConVar cl_drawmaterial( "cl_drawmaterial", "", FCVAR_CHEAT, "Draw a particular material over the frame" );
|
|
static ConVar cl_drawshadowtexture( "cl_drawshadowtexture", "0", FCVAR_CHEAT );
|
|
static ConVar mat_showwatertextures( "mat_showwatertextures", "0", FCVAR_CHEAT );
|
|
static ConVar mat_wateroverlaysize( "mat_wateroverlaysize", "128" );
|
|
static ConVar mat_showframebuffertexture( "mat_showframebuffertexture", "0", FCVAR_CHEAT );
|
|
static ConVar mat_framebuffercopyoverlaysize( "mat_framebuffercopyoverlaysize", "128" );
|
|
static ConVar mat_showcamerarendertarget( "mat_showcamerarendertarget", "0", FCVAR_CHEAT );
|
|
static ConVar mat_camerarendertargetoverlaysize( "mat_camerarendertargetoverlaysize", "128", FCVAR_CHEAT );
|
|
static ConVar mat_hsv( "mat_hsv", "0" );
|
|
static ConVar mat_yuv( "mat_yuv", "0" );
|
|
|
|
#ifdef _XBOX
|
|
static ConVar mat_viewTextureEnable("mat_viewTextureEnable", "", 0, "Enable view texture");
|
|
static ConVar mat_viewTextureName("mat_viewTextureName", "", 0, "Set the view texture name");
|
|
static ConVar mat_viewTextureScale("mat_viewTextureScale", "1", 0, "Set the view texture scale");
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Other convars
|
|
//-----------------------------------------------------------------------------
|
|
|
|
ConVar r_DoCovertTransitions("r_DoCovertTransitions", "1", FCVAR_NEVER_AS_STRING ); // internally used by game code and engine to choose when to allow LOD transitions.
|
|
ConVar r_TransitionSensitivity("r_TransitionSensitivity", "6", 0, "Controls when LODs are changed. Lower numbers cause more overt LOD transitions.");
|
|
|
|
static ConVar r_screenfademinsize( "r_screenfademinsize", "0" );
|
|
static ConVar r_screenfademaxsize( "r_screenfademaxsize", "0" );
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Globals
|
|
//-----------------------------------------------------------------------------
|
|
static Vector g_vecCurrentRenderOrigin(0,0,0);
|
|
static QAngle g_vecCurrentRenderAngles(0,0,0);
|
|
static Vector g_vecCurrentVForward(0,0,0), g_vecCurrentVRight(0,0,0), g_vecCurrentVUp(0,0,0);
|
|
static VMatrix g_matCurrentCamInverse;
|
|
static bool s_bCanAccessCurrentView = false;
|
|
IntroData_t *g_pIntroData = NULL;
|
|
static bool g_bRenderingView = false; // For debugging...
|
|
static int g_CurrentViewID = VIEW_NONE;
|
|
bool g_bRenderingScreenshot = false;
|
|
int g_viewscene_refractUpdateFrame = 0;
|
|
|
|
// mapmaker controlled autoexposure
|
|
bool g_bUseCustomAutoExposureMin = false;
|
|
bool g_bUseCustomAutoExposureMax = false;
|
|
bool g_bUseCustomBloomScale = false;
|
|
float g_flCustomAutoExposureMin = 0;
|
|
float g_flCustomAutoExposureMax = 0;
|
|
float g_flCustomBloomScale = 0.0f;
|
|
float g_flCustomBloomScaleMinimum = 0.0f;
|
|
//-----------------------------------------------------------------------------
|
|
// Precache of necessary materials
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#ifdef HL2_CLIENT_DLL
|
|
CLIENTEFFECT_REGISTER_BEGIN( PrecacheViewRender )
|
|
CLIENTEFFECT_MATERIAL( "scripted/intro_screenspaceeffect" )
|
|
CLIENTEFFECT_REGISTER_END()
|
|
#endif
|
|
|
|
#ifndef _XBOX
|
|
CLIENTEFFECT_REGISTER_BEGIN( PrecachePostProcessingEffects )
|
|
CLIENTEFFECT_MATERIAL( "dev/downsample" )
|
|
CLIENTEFFECT_MATERIAL( "dev/downsample_non_hdr" )
|
|
CLIENTEFFECT_MATERIAL( "dev/blurfiltery_and_add_nohdr" )
|
|
CLIENTEFFECT_MATERIAL( "dev/no_pixel_write" )
|
|
CLIENTEFFECT_MATERIAL( "dev/lumcompare" )
|
|
CLIENTEFFECT_MATERIAL( "dev/blurfilterx" )
|
|
CLIENTEFFECT_MATERIAL( "dev/blurfilterx_nohdr" )
|
|
CLIENTEFFECT_MATERIAL( "dev/floattoscreen_combine" )
|
|
CLIENTEFFECT_MATERIAL( "dev/copyfullframefb_vanilla" )
|
|
CLIENTEFFECT_MATERIAL( "dev/copyfullframefb" )
|
|
CLIENTEFFECT_MATERIAL( "dev/blurfiltery" )
|
|
CLIENTEFFECT_REGISTER_END_CONDITIONAL( engine->GetDXSupportLevel() >= 90)
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Accessors to return the current view being rendered
|
|
//-----------------------------------------------------------------------------
|
|
const Vector &CurrentViewOrigin()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_vecCurrentRenderOrigin;
|
|
}
|
|
|
|
const QAngle &CurrentViewAngles()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_vecCurrentRenderAngles;
|
|
}
|
|
|
|
const Vector &CurrentViewForward()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_vecCurrentVForward;
|
|
}
|
|
|
|
const Vector &CurrentViewRight()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_vecCurrentVRight;
|
|
}
|
|
|
|
const Vector &CurrentViewUp()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_vecCurrentVUp;
|
|
}
|
|
|
|
const VMatrix &CurrentWorldToViewMatrix()
|
|
{
|
|
Assert( s_bCanAccessCurrentView );
|
|
return g_matCurrentCamInverse;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Methods to set the current view/guard access to view parameters
|
|
//-----------------------------------------------------------------------------
|
|
void AllowCurrentViewAccess( bool allow )
|
|
{
|
|
s_bCanAccessCurrentView = allow;
|
|
}
|
|
|
|
bool IsCurrentViewAccessAllowed()
|
|
{
|
|
return s_bCanAccessCurrentView;
|
|
}
|
|
|
|
void SetupCurrentView( const Vector &vecOrigin, const QAngle &angles, view_id_t viewID )
|
|
{
|
|
// Store off view origin and angles
|
|
g_vecCurrentRenderOrigin = vecOrigin;
|
|
g_vecCurrentRenderAngles = angles;
|
|
|
|
// Compute the world->main camera transform
|
|
ComputeCameraVariables( vecOrigin, angles,
|
|
&g_vecCurrentVForward, &g_vecCurrentVRight, &g_vecCurrentVUp, &g_matCurrentCamInverse );
|
|
|
|
g_CurrentViewID = viewID;
|
|
s_bCanAccessCurrentView = true;
|
|
|
|
// Cache off fade distances
|
|
float flScreenFadeMinSize, flScreenFadeMaxSize;
|
|
view->GetScreenFadeDistances( &flScreenFadeMinSize, &flScreenFadeMaxSize );
|
|
modelinfo->SetViewScreenFadeRange( flScreenFadeMinSize, flScreenFadeMaxSize );
|
|
}
|
|
|
|
view_id_t CurrentViewID()
|
|
{
|
|
return ( view_id_t )g_CurrentViewID;
|
|
}
|
|
|
|
void FinishCurrentView()
|
|
{
|
|
s_bCanAccessCurrentView = false;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Constructor
|
|
//-----------------------------------------------------------------------------
|
|
CViewRender::CViewRender()
|
|
{
|
|
m_AnglesHistoryCounter = 0;
|
|
memset(m_AnglesHistory, 0, sizeof(m_AnglesHistory));
|
|
m_flCheapWaterStartDistance = 0.0f;
|
|
m_flCheapWaterEndDistance = 0.1f;
|
|
m_BaseDrawFlags = m_DrawFlags = 0;
|
|
m_bOverrideVisData = false;
|
|
m_iForceViewLeaf = -1;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
inline bool CViewRender::ShouldDrawEntities( void )
|
|
{
|
|
return ( !m_pDrawEntities || (m_pDrawEntities->GetInt() != 0) );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Check all conditions which would prevent drawing the view model
|
|
// Input : drawViewmodel -
|
|
// *viewmodel -
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CViewRender::ShouldDrawViewModel( bool bDrawViewmodel )
|
|
{
|
|
if ( !bDrawViewmodel )
|
|
return false;
|
|
|
|
if ( !r_drawviewmodel.GetBool() )
|
|
return false;
|
|
|
|
if ( C_BasePlayer::ShouldDrawLocalPlayer() )
|
|
return false;
|
|
|
|
if ( !ShouldDrawEntities() )
|
|
return false;
|
|
|
|
if ( render->GetViewEntity() > gpGlobals->maxClients )
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void CViewRender::DrawRenderablesInList( CUtlVector< IClientRenderable * > &list )
|
|
{
|
|
int nCount = list.Count();
|
|
for( int i=0; i < nCount; ++i )
|
|
{
|
|
IClientUnknown *pUnk = list[i]->GetIClientUnknown();
|
|
Assert( pUnk );
|
|
C_BaseEntity *ent = pUnk->GetBaseEntity();
|
|
Assert( ent );
|
|
|
|
// Non-view models wanting to render in view model list...
|
|
if ( ent->ShouldDraw() )
|
|
{
|
|
ent->DrawModel( STUDIO_RENDER );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Actually draw the view model
|
|
// Input : drawViewModel -
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawViewModels( const CViewSetup &view, bool drawViewmodel )
|
|
{
|
|
VPROF( "CViewRender::DrawViewModel" );
|
|
|
|
if ( !ShouldDrawViewModel( drawViewmodel ) )
|
|
return;
|
|
|
|
// Restore the matrices
|
|
materials->MatrixMode( MATERIAL_PROJECTION );
|
|
materials->PushMatrix();
|
|
|
|
// Set up for drawing the view model
|
|
render->SetProjectionMatrix( view.fovViewmodel, view.zNearViewmodel, view.zFarViewmodel );
|
|
|
|
const bool bUseDepthHack = true;
|
|
|
|
// FIXME: Add code to read the current depth range
|
|
float depthmin = 0.0f;
|
|
float depthmax = 1.0f;
|
|
|
|
// HACK HACK: Munge the depth range to prevent view model from poking into walls, etc.
|
|
// Force clipped down range
|
|
if( bUseDepthHack )
|
|
materials->DepthRange( 0.0f, 0.1f );
|
|
|
|
CUtlVector< IClientRenderable * > opaqueViewModelList( 32 );
|
|
CUtlVector< IClientRenderable * > translucentViewModelList( 32 );
|
|
|
|
ClientLeafSystem()->CollateViewModelRenderables( opaqueViewModelList, translucentViewModelList );
|
|
DrawRenderablesInList( opaqueViewModelList );
|
|
DrawRenderablesInList( translucentViewModelList );
|
|
|
|
// Reset the depth range to the original values
|
|
if( bUseDepthHack )
|
|
materials->DepthRange( depthmin, depthmax );
|
|
|
|
// Restore the matrices
|
|
materials->MatrixMode( MATERIAL_PROJECTION );
|
|
materials->PopMatrix();
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : *pEnt -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
VPlane* CViewRender::GetFrustum()
|
|
{
|
|
// The frustum is only valid while in a RenderView call.
|
|
Assert(g_bRenderingView || g_bRenderingCameraView || g_bRenderingScreenshot);
|
|
return m_Frustum;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Output : Returns true on success, false on failure.
|
|
//-----------------------------------------------------------------------------
|
|
bool CViewRender::ShouldDrawBrushModels( void )
|
|
{
|
|
if ( m_pDrawBrushModels && !m_pDrawBrushModels->GetInt() )
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Sort entities in a back-to-front ordering
|
|
//-----------------------------------------------------------------------------
|
|
void SortEntities( CRenderList::CEntry *pEntities, int nEntities )
|
|
{
|
|
// Don't sort if we only have 1 entity
|
|
if ( nEntities <= 1 )
|
|
return;
|
|
|
|
float dists[CRenderList::MAX_GROUP_ENTITIES];
|
|
|
|
const Vector &vecRenderOrigin = CurrentViewOrigin();
|
|
const Vector &vecRenderForward = CurrentViewForward();
|
|
|
|
// First get a distance for each entity.
|
|
int i;
|
|
for( i=0; i < nEntities; i++ )
|
|
{
|
|
IClientRenderable *pRenderable = pEntities[i].m_pRenderable;
|
|
|
|
// Compute the center of the object (needed for translucent brush models)
|
|
Vector boxcenter;
|
|
Vector mins,maxs;
|
|
pRenderable->GetRenderBounds( mins, maxs );
|
|
VectorAdd( mins, maxs, boxcenter );
|
|
VectorMA( pRenderable->GetRenderOrigin(), 0.5f, boxcenter, boxcenter );
|
|
|
|
// Compute distance...
|
|
Vector delta;
|
|
VectorSubtract( boxcenter, vecRenderOrigin, delta );
|
|
dists[i] = DotProduct( delta, vecRenderForward );
|
|
}
|
|
|
|
// H-sort.
|
|
int stepSize = 4;
|
|
while( stepSize )
|
|
{
|
|
int end = nEntities - stepSize;
|
|
for( i=0; i < end; i += stepSize )
|
|
{
|
|
if( dists[i] > dists[i+stepSize] )
|
|
{
|
|
swap( pEntities[i], pEntities[i+stepSize] );
|
|
swap( dists[i], dists[i+stepSize] );
|
|
|
|
if( i == 0 )
|
|
{
|
|
i = -stepSize;
|
|
}
|
|
else
|
|
{
|
|
i -= stepSize << 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
stepSize >>= 1;
|
|
}
|
|
}
|
|
|
|
|
|
void CViewRender::SetupRenderList( const CViewSetup *pView, ClientWorldListInfo_t& info, CRenderList &renderList )
|
|
{
|
|
VPROF( "CViewRender::SetupRenderList" );
|
|
|
|
// Clear the list.
|
|
int i;
|
|
for( i=0; i < RENDER_GROUP_COUNT; i++ )
|
|
{
|
|
renderList.m_RenderGroupCounts[i] = 0;
|
|
}
|
|
|
|
// Precache information used commonly in CollateRenderables
|
|
SetupRenderInfo_t setupInfo;
|
|
setupInfo.m_nRenderFrame = m_BuildRenderableListsNumber;
|
|
setupInfo.m_nDetailBuildFrame = m_BuildWorldListsNumber;
|
|
setupInfo.m_pRenderList = &renderList;
|
|
setupInfo.m_bDrawDetailObjects = g_pClientMode->ShouldDrawDetailObjects() && r_DrawDetailProps.GetInt();
|
|
|
|
if (pView)
|
|
{
|
|
setupInfo.m_vecRenderOrigin = pView->origin;
|
|
setupInfo.m_flRenderDistSq = cl_maxrenderable_dist.GetFloat();
|
|
setupInfo.m_flRenderDistSq *= setupInfo.m_flRenderDistSq;
|
|
}
|
|
else
|
|
{
|
|
setupInfo.m_flRenderDistSq = 0.0f;
|
|
}
|
|
|
|
// Now collate the entities in the leaves.
|
|
if( ShouldDrawEntities() )
|
|
{
|
|
IClientLeafSystem *pClientLeafSystem = ClientLeafSystem();
|
|
for( i=0; i < info.m_LeafCount; i++ )
|
|
{
|
|
int nTranslucent = renderList.m_RenderGroupCounts[RENDER_GROUP_TRANSLUCENT_ENTITY];
|
|
|
|
// Add renderables from this leaf...
|
|
pClientLeafSystem->CollateRenderablesInLeaf( info.m_pLeafList[i], i, setupInfo );
|
|
|
|
int nNewTranslucent = renderList.m_RenderGroupCounts[RENDER_GROUP_TRANSLUCENT_ENTITY] - nTranslucent;
|
|
if( nNewTranslucent )
|
|
{
|
|
// Sort the new translucent entities.
|
|
SortEntities( &renderList.m_RenderGroups[RENDER_GROUP_TRANSLUCENT_ENTITY][nTranslucent], nNewTranslucent );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void OverlayWaterTexture( IMaterial *pMaterial, int xOffset, int yOffset, bool bFlip )
|
|
{
|
|
#ifndef _XBOX
|
|
float xBaseOffset = 0;
|
|
float yBaseOffset = 0;
|
|
#else
|
|
// screen safe
|
|
float xBaseOffset = 32;
|
|
float yBaseOffset = 32;
|
|
#endif
|
|
float offsetS = ( 0.5f / 256.0f );
|
|
float offsetT = ( 0.5f / 256.0f );
|
|
float fFlip0 = bFlip ? 1.0f : 0.0f;
|
|
float fFlip1 = bFlip ? 0.0f : 1.0f;
|
|
if( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
materials->Bind( pMaterial );
|
|
IMesh* pMesh = materials->GetDynamicMesh( true );
|
|
|
|
float w = mat_wateroverlaysize.GetFloat();
|
|
float h = mat_wateroverlaysize.GetFloat();
|
|
|
|
CMeshBuilder meshBuilder;
|
|
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
|
|
|
|
meshBuilder.Position3f( xBaseOffset + xOffset * w, yBaseOffset + yOffset * h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, fFlip1 + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( xBaseOffset + ( xOffset + 1 ) * w, yBaseOffset + yOffset * h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, fFlip1 + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( xBaseOffset + ( xOffset + 1 ) * w, yBaseOffset + ( yOffset + 1 ) * h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, fFlip0 + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( xBaseOffset + xOffset * w, yBaseOffset + ( yOffset + 1 ) * h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, fFlip0 + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.End();
|
|
pMesh->Draw();
|
|
}
|
|
}
|
|
|
|
static void OverlayWaterTextures( void )
|
|
{
|
|
OverlayWaterTexture( materials->FindMaterial( "debug/debugreflect", NULL ), 0, 0, false );
|
|
OverlayWaterTexture( materials->FindMaterial( "debug/debugrefract", NULL ), 0, 1, true );
|
|
}
|
|
|
|
void OverlayCameraRenderTarget( const char *pszMaterialName, float flX, float flY, float w, float h )
|
|
{
|
|
float offsetS = ( 0.5f / 256.0f );
|
|
float offsetT = ( 0.5f / 256.0f );
|
|
IMaterial *pMaterial;
|
|
pMaterial = materials->FindMaterial( pszMaterialName, TEXTURE_GROUP_OTHER, true );
|
|
if( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
materials->Bind( pMaterial );
|
|
IMesh* pMesh = materials->GetDynamicMesh( true );
|
|
|
|
CMeshBuilder meshBuilder;
|
|
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
|
|
|
|
meshBuilder.Position3f( flX, flY, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, 0.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( flX+w, flY, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, 0.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( flX+w, flY+h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, 1.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( flX, flY+h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, 1.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.End();
|
|
pMesh->Draw();
|
|
}
|
|
}
|
|
|
|
|
|
static void OverlayFrameBufferTexture( int nFrameBufferIndex )
|
|
{
|
|
float offsetS = ( 0.5f / 256.0f );
|
|
float offsetT = ( 0.5f / 256.0f );
|
|
IMaterial *pMaterial;
|
|
char buf[MAX_PATH];
|
|
Q_snprintf( buf, MAX_PATH, "debug/debugfbtexture%d", nFrameBufferIndex );
|
|
pMaterial = materials->FindMaterial( buf, TEXTURE_GROUP_OTHER, true );
|
|
if( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
materials->Bind( pMaterial );
|
|
IMesh* pMesh = materials->GetDynamicMesh( true );
|
|
|
|
float w = mat_framebuffercopyoverlaysize.GetFloat();
|
|
float h = mat_framebuffercopyoverlaysize.GetFloat();
|
|
|
|
CMeshBuilder meshBuilder;
|
|
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
|
|
|
|
meshBuilder.Position3f( w * nFrameBufferIndex, 0.0f, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, 0.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( w * ( nFrameBufferIndex + 1 ), 0.0f, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, 0.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( w * ( nFrameBufferIndex + 1 ), h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f + offsetS, 1.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( w * nFrameBufferIndex, h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f + offsetS, 1.0f + offsetT );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.End();
|
|
pMesh->Draw();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void UpdateClientRenderableInPVSStatus()
|
|
{
|
|
// Vis for this view should already be setup at this point.
|
|
|
|
// For each client-only entity, notify it if it's newly coming into the PVS.
|
|
CUtlLinkedList<CClientEntityList::CPVSNotifyInfo,unsigned short> &theList = ClientEntityList().GetPVSNotifiers();
|
|
FOR_EACH_LL( theList, i )
|
|
{
|
|
CClientEntityList::CPVSNotifyInfo *pInfo = &theList[i];
|
|
|
|
if ( pInfo->m_InPVSStatus & INPVS_YES )
|
|
{
|
|
// Ok, this entity already thinks it's in the PVS. No need to notify it.
|
|
// We need to set the INPVS_YES_THISFRAME flag if it's in this frame at all, so we
|
|
// don't tell the entity it's not in the PVS anymore at the end of the frame.
|
|
if ( !( pInfo->m_InPVSStatus & INPVS_THISFRAME ) )
|
|
{
|
|
if ( g_pClientLeafSystem->IsRenderableInPVS( pInfo->m_pRenderable ) )
|
|
{
|
|
pInfo->m_InPVSStatus |= INPVS_THISFRAME;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This entity doesn't think it's in the PVS yet. If it is now in the PVS, let it know.
|
|
if ( g_pClientLeafSystem->IsRenderableInPVS( pInfo->m_pRenderable ) )
|
|
{
|
|
pInfo->m_pNotify->OnPVSStatusChanged( true );
|
|
pInfo->m_InPVSStatus |= INPVS_YES;
|
|
pInfo->m_InPVSStatus |= INPVS_THISFRAME;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Debugging aid to display a texture
|
|
//-----------------------------------------------------------------------------
|
|
#ifdef _XBOX
|
|
static void OverlayShowTexture( const char* textureName, float scale )
|
|
{
|
|
bool foundVar;
|
|
IMaterial *pMaterial;
|
|
IMaterialVar *BaseTextureVar;
|
|
ITexture *pTex;
|
|
float x, y, w, h;
|
|
|
|
// screen safe
|
|
x = 32;
|
|
y = 32;
|
|
|
|
pMaterial = materials->FindMaterial( "___debug", TEXTURE_GROUP_OTHER, true );
|
|
BaseTextureVar = pMaterial->FindVar( "$basetexture", &foundVar, false );
|
|
if (!foundVar)
|
|
return;
|
|
|
|
if ( textureName && textureName[0] )
|
|
{
|
|
pTex = materials->FindTexture( textureName, TEXTURE_GROUP_OTHER, false );
|
|
BaseTextureVar->SetTextureValue( pTex );
|
|
|
|
w = pTex->GetActualWidth() * scale;
|
|
h = pTex->GetActualHeight() * scale;
|
|
}
|
|
else
|
|
{
|
|
w = h = 64.0f * scale;
|
|
}
|
|
|
|
materials->Bind( pMaterial );
|
|
IMesh* pMesh = materials->GetDynamicMesh( true );
|
|
|
|
CMeshBuilder meshBuilder;
|
|
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
|
|
meshBuilder.Position3f( x, y, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f, 0.0f );
|
|
meshBuilder.AdvanceVertex();
|
|
meshBuilder.Position3f( x+w, y, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f, 0.0f );
|
|
meshBuilder.AdvanceVertex();
|
|
meshBuilder.Position3f( x+w, y+h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f, 1.0f );
|
|
meshBuilder.AdvanceVertex();
|
|
meshBuilder.Position3f( x, x+h, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f, 1.0f );
|
|
meshBuilder.AdvanceVertex();
|
|
meshBuilder.End();
|
|
pMesh->Draw();
|
|
}
|
|
#endif
|
|
|
|
extern ConVar cl_leveloverview;
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Builds lists of things to render in the world, called once per view
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::BuildWorldRenderLists( const CViewSetup *pView,
|
|
ClientWorldListInfo_t& info, bool bUpdateLightmaps, bool bDrawEntities, int iForceViewLeaf )
|
|
{
|
|
VPROF_BUDGET( "BuildWorldRenderLists", VPROF_BUDGETGROUP_WORLD_RENDERING );
|
|
|
|
// Server entities already know which ones are in the PVS, but client-only entities don't.
|
|
// We need to know which client-only entities are in the PVS at this point so the shadow
|
|
// manager can project/occlude their shadows correctly.
|
|
UpdateClientRenderableInPVSStatus();
|
|
|
|
++m_BuildWorldListsNumber;
|
|
// Override vis data if specified this render, otherwise use default behavior with NULL
|
|
VisOverrideData_t* pVisData = (m_bOverrideVisData)?(&m_VisData):(NULL);
|
|
render->BuildWorldLists_VisOverride( &info, bUpdateLightmaps, iForceViewLeaf, pVisData );
|
|
|
|
// Return to default area portal behavior by using the CurrentView as the starting area
|
|
m_bOverrideVisData = false;
|
|
m_iForceViewLeaf = -1;
|
|
|
|
if ( bDrawEntities )
|
|
{
|
|
// Now that we have the list of all leaves, regenerate shadows cast
|
|
g_pClientShadowMgr->ComputeShadowTextures( pView, info.m_LeafCount, info.m_pLeafList );
|
|
|
|
// Compute the prop opacity based on the view position and fov zoom scale
|
|
float flFactor = 1.0f;
|
|
C_BasePlayer *pLocal = C_BasePlayer::GetLocalPlayer();
|
|
if ( pLocal )
|
|
{
|
|
flFactor = pLocal->GetFOVDistanceAdjustFactor();
|
|
}
|
|
|
|
if ( cl_leveloverview.GetFloat() > 0 )
|
|
{
|
|
// disable prop fading
|
|
flFactor = -1;
|
|
}
|
|
|
|
// When zoomed in, tweak the opacity to stay visible from further away
|
|
staticpropmgr->ComputePropOpacity( CurrentViewOrigin(), flFactor );
|
|
|
|
// Build a list of detail props to render
|
|
DetailObjectSystem()->BuildDetailObjectRenderLists();
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Computes the actual world list info based on the render flags
|
|
//-----------------------------------------------------------------------------
|
|
ClientWorldListInfo_t *CViewRender::ComputeActualWorldListInfo( const ClientWorldListInfo_t& info, int nDrawFlags, ClientWorldListInfo_t& tmpInfo )
|
|
{
|
|
// Drawing everything? Just return the world list info as-is
|
|
int nWaterDrawFlags = nDrawFlags & (DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER);
|
|
if ( nWaterDrawFlags == (DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER) )
|
|
return const_cast<ClientWorldListInfo_t*>(&info);
|
|
|
|
tmpInfo.m_ViewFogVolume = info.m_ViewFogVolume;
|
|
tmpInfo.m_LeafCount = 0;
|
|
|
|
// Not drawing anything? Then don't bother with renderable lists
|
|
if ( nWaterDrawFlags == 0 )
|
|
return &tmpInfo;
|
|
|
|
// Create a sub-list based on the actual leaves being rendered
|
|
bool bRenderingUnderwater = (nWaterDrawFlags & DF_RENDER_UNDERWATER) != 0;
|
|
for ( int i = 0; i < info.m_LeafCount; ++i )
|
|
{
|
|
bool bLeafIsUnderwater = ( info.m_pLeafFogVolume[i] != -1 );
|
|
if ( bRenderingUnderwater == bLeafIsUnderwater )
|
|
{
|
|
tmpInfo.m_pLeafList[ tmpInfo.m_LeafCount ] = info.m_pLeafList[ i ];
|
|
tmpInfo.m_pLeafFogVolume[ tmpInfo.m_LeafCount ] = info.m_pLeafFogVolume[ i ];
|
|
tmpInfo.m_pActualLeafIndex[ tmpInfo.m_LeafCount ] = i;
|
|
++tmpInfo.m_LeafCount;
|
|
}
|
|
}
|
|
|
|
return &tmpInfo;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Builds render lists for renderables. Called once for refraction, once for over water
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::BuildRenderableRenderLists( const CViewSetup *pView, ClientWorldListInfo_t& info, CRenderList &renderList )
|
|
{
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
++m_BuildRenderableListsNumber;
|
|
|
|
// For better sorting, find out the leaf *nearest* to the camera
|
|
// and render translucent objects as if they are in that leaf.
|
|
if( ShouldDrawEntities() )
|
|
{
|
|
ClientLeafSystem()->ComputeTranslucentRenderLeaf(
|
|
info.m_LeafCount, info.m_pLeafList, info.m_pLeafFogVolume, m_BuildRenderableListsNumber );
|
|
}
|
|
|
|
SetupRenderList( pView, info, renderList );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Computes draw flags for the engine to build its world surface lists
|
|
//-----------------------------------------------------------------------------
|
|
static inline unsigned long BuildDrawFlags( bool bDrawSkybox, bool bDrawUnderWater, bool bDrawAboveWater, bool bDrawWaterSurface, bool bClipSkybox )
|
|
{
|
|
unsigned long nEngineFlags = 0;
|
|
|
|
if ( bDrawSkybox )
|
|
{
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_SKYBOX;
|
|
}
|
|
|
|
if ( bDrawAboveWater )
|
|
{
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_STRICTLYABOVEWATER;
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_INTERSECTSWATER;
|
|
}
|
|
|
|
if ( bDrawUnderWater )
|
|
{
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_STRICTLYUNDERWATER;
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_INTERSECTSWATER;
|
|
}
|
|
|
|
if ( bDrawWaterSurface )
|
|
{
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_WATERSURFACE;
|
|
}
|
|
|
|
if( bClipSkybox )
|
|
{
|
|
nEngineFlags |= DRAWWORLDLISTS_DRAW_CLIPSKYBOX;
|
|
}
|
|
|
|
return nEngineFlags;
|
|
}
|
|
|
|
void CViewRender::DrawWorld( ClientWorldListInfo_t& info, CRenderList &renderList, int flags, float waterZAdjust )
|
|
{
|
|
VPROF_INCREMENT_COUNTER( "RenderWorld", 1 );
|
|
VPROF_BUDGET( "DrawWorld", VPROF_BUDGETGROUP_WORLD_RENDERING );
|
|
if( !r_drawopaqueworld.GetBool() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
bool drawSkybox = (flags & DF_DRAWSKYBOX) != 0;
|
|
bool drawUnderWater = (flags & DF_RENDER_UNDERWATER) != 0;
|
|
bool drawAboveWater = (flags & DF_RENDER_ABOVEWATER) != 0;
|
|
bool drawWaterSurface = (flags & DF_RENDER_WATER) != 0;
|
|
bool bClipSkybox = (flags & DF_CLIP_SKYBOX ) != 0;
|
|
|
|
unsigned long engineFlags = BuildDrawFlags( drawSkybox, drawUnderWater, drawAboveWater, drawWaterSurface, bClipSkybox );
|
|
if ( flags & DF_MAINTAINWORLDLISTS )
|
|
{
|
|
engineFlags |= DRAWWORLDLISTS_MAINTAIN_RENDER_LISTS;
|
|
}
|
|
|
|
int oldDrawFlags = m_DrawFlags;
|
|
m_DrawFlags |= flags;
|
|
|
|
render->DrawWorldLists( engineFlags, waterZAdjust );
|
|
|
|
m_DrawFlags = oldDrawFlags;
|
|
}
|
|
|
|
static inline void DrawOpaqueRenderable( IClientRenderable *pEnt, bool twoPass )
|
|
{
|
|
float color[3];
|
|
|
|
pEnt->GetColorModulation( color );
|
|
render->SetColorModulation( color );
|
|
|
|
|
|
int flags = STUDIO_RENDER;
|
|
if (twoPass)
|
|
{
|
|
flags |= STUDIO_TWOPASS;
|
|
}
|
|
|
|
float *pRenderClipPlane = NULL;
|
|
|
|
if( !materials->UsingFastClipping() && //do NOT change the fast clip plane mid-scene, depth problems result. Regular user clip planes are fine though
|
|
r_entityclips.GetBool() ) //checking here reduces the r_entityclips.GetBool() check from 2 times to 1
|
|
pRenderClipPlane = pEnt->GetRenderClipPlane();
|
|
|
|
if( pRenderClipPlane )
|
|
materials->PushCustomClipPlane( pRenderClipPlane );
|
|
|
|
pEnt->DrawModel( flags );
|
|
|
|
if( pRenderClipPlane )
|
|
materials->PopCustomClipPlane();
|
|
}
|
|
|
|
|
|
static inline void DrawTranslucentRenderable( IClientRenderable *pEnt, bool twoPass )
|
|
{
|
|
// Determine blending amount and tell engine
|
|
float blend = (float)( pEnt->GetFxBlend() / 255.0f );
|
|
|
|
// Totally gone
|
|
if ( blend <= 0.0f )
|
|
return;
|
|
|
|
// Tell engine
|
|
render->SetBlend( blend );
|
|
|
|
float color[3];
|
|
pEnt->GetColorModulation( color );
|
|
render->SetColorModulation( color );
|
|
|
|
int flags = STUDIO_RENDER | STUDIO_TRANSPARENCY;
|
|
if (twoPass)
|
|
flags |= STUDIO_TWOPASS;
|
|
|
|
#if 0
|
|
Vector mins, maxs;
|
|
pEnt->GetRenderBounds( mins, maxs );
|
|
debugoverlay->AddBoxOverlay( pEnt->GetRenderOrigin(), mins, maxs, pEnt->GetRenderAngles(), 255, 255, 255, 64, 0.01 );
|
|
if ( pEnt->GetModel() )
|
|
{
|
|
const char *pName = modelinfo->GetModelName( pEnt->GetModel() );
|
|
if ( Q_stricmp( pName, "models/props_c17/tv_monitor01_screen.mdl" ) )
|
|
{
|
|
debugoverlay->AddTextOverlay( pEnt->GetRenderOrigin(), 0.01, pName );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
float *pRenderClipPlane = NULL;
|
|
|
|
if( !materials->UsingFastClipping() && //do NOT change the fast clip plane mid-scene, depth problems result. Regular user clip planes are fine though
|
|
r_entityclips.GetBool() ) //checking here reduces the r_entityclips.GetBool() check from 2 times to 1
|
|
pRenderClipPlane = pEnt->GetRenderClipPlane();
|
|
|
|
if( pRenderClipPlane )
|
|
materials->PushCustomClipPlane( pRenderClipPlane );
|
|
|
|
pEnt->DrawModel( flags );
|
|
|
|
if( pRenderClipPlane )
|
|
materials->PopCustomClipPlane();
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws all opaque renderables in leaves that were rendered
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawOpaqueRenderables( ClientWorldListInfo_t& info, CRenderList &renderList )
|
|
{
|
|
VPROF("CViewRender::DrawOpaqueRenderables");
|
|
|
|
if( !r_drawopaquerenderables.GetBool() )
|
|
return;
|
|
|
|
if( !ShouldDrawEntities() )
|
|
return;
|
|
|
|
render->SetBlend( 1 );
|
|
|
|
// Iterate over all leaves that were visible, and draw opaque things in them.
|
|
RopeManager()->ResetRenderCache();
|
|
|
|
// Iterate over all leaves that were visible, and draw opaque things in them.
|
|
int i;
|
|
|
|
// first do the brush models
|
|
CRenderList::CEntry *pEntities = renderList.m_RenderGroups[RENDER_GROUP_OPAQUE_BRUSH];
|
|
int nOpaque = renderList.m_RenderGroupCounts[RENDER_GROUP_OPAQUE_BRUSH];
|
|
for( i=0; i < nOpaque; ++i )
|
|
{
|
|
Assert(pEntities[i].m_TwoPass==0);
|
|
DrawOpaqueRenderable( pEntities[i].m_pRenderable, false );
|
|
}
|
|
|
|
// now the static props
|
|
pEntities = renderList.m_RenderGroups[RENDER_GROUP_OPAQUE_STATIC];
|
|
nOpaque = renderList.m_RenderGroupCounts[RENDER_GROUP_OPAQUE_STATIC];
|
|
if ( nOpaque )
|
|
{
|
|
nOpaque = min( nOpaque, 512 );
|
|
|
|
IClientRenderable *pStatics[512];
|
|
for( i=0; i < nOpaque; ++i )
|
|
{
|
|
pStatics[i] = pEntities[i].m_pRenderable;
|
|
}
|
|
staticpropmgr->DrawStaticProps( pStatics, nOpaque );
|
|
}
|
|
|
|
|
|
// now the other opaque entities
|
|
pEntities = renderList.m_RenderGroups[RENDER_GROUP_OPAQUE_ENTITY];
|
|
nOpaque = renderList.m_RenderGroupCounts[RENDER_GROUP_OPAQUE_ENTITY];
|
|
|
|
for( i=0; i < nOpaque; ++i )
|
|
{
|
|
DrawOpaqueRenderable( pEntities[i].m_pRenderable, (pEntities[i].m_TwoPass != 0) );
|
|
}
|
|
|
|
RopeManager()->DrawRenderCache();
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Renders all translucent world + detail objects in a particular set of leaves
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawTranslucentWorldInLeaves( int iCurLeafIndex, int iFinalLeafIndex, ClientWorldListInfo_t &info, int nDrawFlags )
|
|
{
|
|
VPROF_BUDGET( "CViewRender::DrawTranslucentWorldInLeaves", VPROF_BUDGETGROUP_WORLD_RENDERING );
|
|
for( ; iCurLeafIndex >= iFinalLeafIndex; iCurLeafIndex-- )
|
|
{
|
|
int nActualLeafIndex = info.m_pActualLeafIndex ? info.m_pActualLeafIndex[ iCurLeafIndex ] : iCurLeafIndex;
|
|
Assert( nActualLeafIndex != INVALID_LEAF_INDEX );
|
|
if ( render->LeafContainsTranslucentSurfaces( nActualLeafIndex, nDrawFlags ) )
|
|
{
|
|
// Now draw the surfaces in this leaf
|
|
render->DrawTranslucentSurfaces( nActualLeafIndex, nDrawFlags );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Renders all translucent world + detail objects in a particular set of leaves
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawTranslucentWorldAndDetailPropsInLeaves( int iCurLeafIndex, int iFinalLeafIndex,
|
|
ClientWorldListInfo_t &info, int nDrawFlags, int &nDetailLeafCount, LeafIndex_t* pDetailLeafList )
|
|
{
|
|
VPROF_BUDGET( "CViewRender::DrawTranslucentWorldAndDetailPropsInLeaves", VPROF_BUDGETGROUP_WORLD_RENDERING );
|
|
for( ; iCurLeafIndex >= iFinalLeafIndex; iCurLeafIndex-- )
|
|
{
|
|
int nActualLeafIndex = info.m_pActualLeafIndex ? info.m_pActualLeafIndex[ iCurLeafIndex ] : iCurLeafIndex;
|
|
Assert( nActualLeafIndex != INVALID_LEAF_INDEX );
|
|
if ( render->LeafContainsTranslucentSurfaces( nActualLeafIndex, nDrawFlags ) )
|
|
{
|
|
// First draw any queued-up detail props from previously visited leaves
|
|
DetailObjectSystem()->RenderTranslucentDetailObjects( CurrentViewOrigin(), CurrentViewForward(), nDetailLeafCount, pDetailLeafList );
|
|
nDetailLeafCount = 0;
|
|
|
|
// Now draw the surfaces in this leaf
|
|
render->DrawTranslucentSurfaces( nActualLeafIndex, nDrawFlags );
|
|
}
|
|
|
|
// Queue up detail props that existed in this leaf
|
|
if ( ClientLeafSystem()->ShouldDrawDetailObjectsInLeaf( info.m_pLeafList[iCurLeafIndex], m_BuildWorldListsNumber ) )
|
|
{
|
|
pDetailLeafList[nDetailLeafCount] = info.m_pLeafList[iCurLeafIndex];
|
|
++nDetailLeafCount;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Renders all translucent entities in the render list
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawTranslucentRenderablesNoWorld( CRenderList &renderList, bool bInSkybox )
|
|
{
|
|
VPROF( "CViewRender::DrawTranslucentRenderablesNoWorld" );
|
|
|
|
if ( !ShouldDrawEntities() || !r_drawtranslucentrenderables.GetBool() )
|
|
return;
|
|
|
|
// Draw the particle singletons.
|
|
DrawParticleSingletons( bInSkybox );
|
|
|
|
CRenderList::CEntry *pEntities = renderList.m_RenderGroups[RENDER_GROUP_TRANSLUCENT_ENTITY];
|
|
int iCurTranslucentEntity = renderList.m_RenderGroupCounts[RENDER_GROUP_TRANSLUCENT_ENTITY] - 1;
|
|
|
|
while( iCurTranslucentEntity >= 0 )
|
|
{
|
|
IClientRenderable *pRenderable = pEntities[iCurTranslucentEntity].m_pRenderable;
|
|
if ( pRenderable->UsesFrameBufferTexture() )
|
|
{
|
|
UpdateRefractTexture();
|
|
}
|
|
DrawTranslucentRenderable( pRenderable, (pEntities[iCurTranslucentEntity].m_TwoPass != 0) );
|
|
--iCurTranslucentEntity;
|
|
}
|
|
|
|
// Reset the blend state.
|
|
render->SetBlend( 1 );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Renders all translucent world, entities, and detail objects in a particular set of leaves
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawTranslucentRenderables( ClientWorldListInfo_t& info, CRenderList &renderList,
|
|
int nFlags, bool bInSkybox )
|
|
{
|
|
if ( !r_drawtranslucentworld.GetBool() )
|
|
{
|
|
DrawTranslucentRenderablesNoWorld( renderList, bInSkybox );
|
|
return;
|
|
}
|
|
|
|
VPROF( "CViewRender::DrawTranslucentRenderables" );
|
|
int iPrevLeaf = info.m_LeafCount - 1;
|
|
int nDetailLeafCount = 0;
|
|
LeafIndex_t *pDetailLeafList = (LeafIndex_t*)stackalloc( info.m_LeafCount * sizeof(LeafIndex_t) );
|
|
|
|
bool bDrawUnderWater = (nFlags & DF_RENDER_UNDERWATER) != 0;
|
|
bool bDrawAboveWater = (nFlags & DF_RENDER_ABOVEWATER) != 0;
|
|
bool bDrawWater = (nFlags & DF_RENDER_WATER) != 0;
|
|
bool bClipSkybox = (nFlags & DF_CLIP_SKYBOX ) != 0;
|
|
unsigned long nDrawFlags = BuildDrawFlags( false, bDrawUnderWater, bDrawAboveWater, bDrawWater, bClipSkybox );
|
|
|
|
DetailObjectSystem()->BeginTranslucentDetailRendering();
|
|
|
|
if ( ShouldDrawEntities() && r_drawtranslucentrenderables.GetBool() )
|
|
{
|
|
// Draw the particle singletons.
|
|
DrawParticleSingletons( bInSkybox );
|
|
|
|
CRenderList::CEntry *pEntities = renderList.m_RenderGroups[RENDER_GROUP_TRANSLUCENT_ENTITY];
|
|
int iCurTranslucentEntity = renderList.m_RenderGroupCounts[RENDER_GROUP_TRANSLUCENT_ENTITY] - 1;
|
|
|
|
bool bRenderingWaterRenderTargets = nFlags & ( DF_RENDER_REFRACTION | DF_RENDER_REFLECTION );
|
|
|
|
while( iCurTranslucentEntity >= 0 )
|
|
{
|
|
// Seek the current leaf up to our current translucent-entity leaf.
|
|
int iThisLeaf = pEntities[iCurTranslucentEntity].m_iWorldListInfoLeaf;
|
|
|
|
// First draw the translucent parts of the world up to and including those in this leaf
|
|
DrawTranslucentWorldAndDetailPropsInLeaves( iPrevLeaf, iThisLeaf, info, nDrawFlags, nDetailLeafCount, pDetailLeafList );
|
|
|
|
// We're traversing the leaf list backwards to get the appropriate sort ordering (back to front)
|
|
iPrevLeaf = iThisLeaf - 1;
|
|
|
|
// Draw all the translucent entities with this leaf.
|
|
int nLeaf = info.m_pLeafList[iThisLeaf];
|
|
|
|
bool bDrawDetailProps = ClientLeafSystem()->ShouldDrawDetailObjectsInLeaf( nLeaf, m_BuildWorldListsNumber );
|
|
if ( bDrawDetailProps )
|
|
{
|
|
// Draw detail props up to but not including this leaf
|
|
Assert( nDetailLeafCount > 0 );
|
|
--nDetailLeafCount;
|
|
Assert( pDetailLeafList[nDetailLeafCount] == nLeaf );
|
|
DetailObjectSystem()->RenderTranslucentDetailObjects( CurrentViewOrigin(), CurrentViewForward(), nDetailLeafCount, pDetailLeafList );
|
|
|
|
// Draw translucent renderables in the leaf interspersed with detail props
|
|
for( ;pEntities[iCurTranslucentEntity].m_iWorldListInfoLeaf == iThisLeaf && iCurTranslucentEntity >= 0; --iCurTranslucentEntity )
|
|
{
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
IClientRenderable *pRenderable = pEntities[iCurTranslucentEntity].m_pRenderable;
|
|
|
|
// Draw any detail props in this leaf that's farther than the entity
|
|
const Vector &vecRenderOrigin = pRenderable->GetRenderOrigin();
|
|
DetailObjectSystem()->RenderTranslucentDetailObjectsInLeaf(
|
|
CurrentViewOrigin(), CurrentViewForward(), nLeaf, &vecRenderOrigin );
|
|
|
|
if ( pRenderable->UsesFrameBufferTexture() )
|
|
{
|
|
if( bRenderingWaterRenderTargets )
|
|
{
|
|
continue;
|
|
}
|
|
UpdateRefractTexture();
|
|
}
|
|
|
|
// Then draw the translucent renderable
|
|
DrawTranslucentRenderable( pRenderable, (pEntities[iCurTranslucentEntity].m_TwoPass != 0) );
|
|
}
|
|
|
|
// Draw all remaining props in this leaf
|
|
DetailObjectSystem()->RenderTranslucentDetailObjectsInLeaf( CurrentViewOrigin(), CurrentViewForward(), nLeaf, NULL );
|
|
}
|
|
else
|
|
{
|
|
// Draw queued up detail props (we know that the list of detail leaves won't include this leaf, since ShouldDrawDetailObjectsInLeaf is false)
|
|
// Therefore no fixup on nDetailLeafCount is required as in the above section
|
|
DetailObjectSystem()->RenderTranslucentDetailObjects( CurrentViewOrigin(), CurrentViewForward(), nDetailLeafCount, pDetailLeafList );
|
|
|
|
for( ;pEntities[iCurTranslucentEntity].m_iWorldListInfoLeaf == iThisLeaf && iCurTranslucentEntity >= 0; --iCurTranslucentEntity )
|
|
{
|
|
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
IClientRenderable *pRenderable = pEntities[iCurTranslucentEntity].m_pRenderable;
|
|
if ( pRenderable->UsesFrameBufferTexture() )
|
|
{
|
|
if( bRenderingWaterRenderTargets )
|
|
{
|
|
continue;
|
|
}
|
|
UpdateRefractTexture();
|
|
}
|
|
DrawTranslucentRenderable( pRenderable, (pEntities[iCurTranslucentEntity].m_TwoPass != 0) );
|
|
|
|
}
|
|
}
|
|
|
|
nDetailLeafCount = 0;
|
|
}
|
|
}
|
|
|
|
// Draw the rest of the surfaces in world leaves
|
|
DrawTranslucentWorldAndDetailPropsInLeaves( iPrevLeaf, 0, info, nDrawFlags, nDetailLeafCount, pDetailLeafList );
|
|
|
|
// Draw any queued-up detail props from previously visited leaves
|
|
DetailObjectSystem()->RenderTranslucentDetailObjects( CurrentViewOrigin(), CurrentViewForward(), nDetailLeafCount, pDetailLeafList );
|
|
|
|
// Reset the blend state.
|
|
render->SetBlend( 1 );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Renders the shadow texture to screen...
|
|
//-----------------------------------------------------------------------------
|
|
static void RenderMaterial( const char *pMaterialName )
|
|
{
|
|
// So it's not in the very top left
|
|
float x = 100.0f, y = 100.0f;
|
|
// float x = 0.0f, y = 0.0f;
|
|
|
|
IMaterial *pMaterial = materials->FindMaterial( pMaterialName, TEXTURE_GROUP_OTHER, false );
|
|
if ( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
materials->Bind( pMaterial );
|
|
IMesh* pMesh = materials->GetDynamicMesh( true );
|
|
|
|
CMeshBuilder meshBuilder;
|
|
meshBuilder.Begin( pMesh, MATERIAL_QUADS, 1 );
|
|
|
|
meshBuilder.Position3f( x, y, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f, 0.0f );
|
|
meshBuilder.Color4ub( 255, 255, 255, 255 );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( x + pMaterial->GetMappingWidth(), y, 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f, 0.0f );
|
|
meshBuilder.Color4ub( 255, 255, 255, 255 );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( x + pMaterial->GetMappingWidth(), y + pMaterial->GetMappingHeight(), 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 1.0f, 1.0f );
|
|
meshBuilder.Color4ub( 255, 255, 255, 255 );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.Position3f( x, y + pMaterial->GetMappingHeight(), 0.0f );
|
|
meshBuilder.TexCoord2f( 0, 0.0f, 1.0f );
|
|
meshBuilder.Color4ub( 255, 255, 255, 255 );
|
|
meshBuilder.AdvanceVertex();
|
|
|
|
meshBuilder.End();
|
|
pMesh->Draw();
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the screen effect
|
|
//-----------------------------------------------------------------------------
|
|
static void DrawScreenEffectMaterial( IMaterial *pMaterial, int x, int y, int w, int h )
|
|
{
|
|
Rect_t actualRect;
|
|
UpdateScreenEffectTexture( 0, x, y, w, h, false, &actualRect );
|
|
ITexture *pTexture = GetFullFrameFrameBufferTexture( 0 );
|
|
|
|
materials->DrawScreenSpaceRectangle( pMaterial, x, y, w, h,
|
|
actualRect.x, actualRect.y, actualRect.x+actualRect.width-1, actualRect.y+actualRect.height-1,
|
|
pTexture->GetActualWidth(), pTexture->GetActualHeight() );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Performs screen space effects, if any
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::PerformScreenSpaceEffects( int x, int y, int w, int h )
|
|
{
|
|
VPROF("CViewRender::PerformScreenSpaceEffects()");
|
|
|
|
// FIXME: Screen-space effects are busted in the editor
|
|
if ( engine->IsHammerRunning() )
|
|
return;
|
|
|
|
g_pScreenSpaceEffects->RenderEffects( x, y, w, h );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Sets the screen space effect material (can't be done during rendering)
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::SetScreenOverlayMaterial( IMaterial *pMaterial )
|
|
{
|
|
m_ScreenOverlayMaterial.Init( pMaterial );
|
|
}
|
|
|
|
IMaterial *CViewRender::GetScreenOverlayMaterial( )
|
|
{
|
|
return m_ScreenOverlayMaterial;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Performs screen space effects, if any
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::PerformScreenOverlay( int x, int y, int w, int h )
|
|
{
|
|
VPROF("CViewRender::PerformScreenOverlay()");
|
|
|
|
if (m_ScreenOverlayMaterial)
|
|
{
|
|
if ( m_ScreenOverlayMaterial->NeedsFullFrameBufferTexture() )
|
|
{
|
|
DrawScreenEffectMaterial( m_ScreenOverlayMaterial, x, y, w, h );
|
|
}
|
|
else if ( m_ScreenOverlayMaterial->NeedsPowerOfTwoFrameBufferTexture() )
|
|
{
|
|
// First copy the FB off to the offscreen texture
|
|
UpdateRefractTexture( x, y, w, h, true );
|
|
|
|
// Now draw the entire screen using the material...
|
|
ITexture *pTexture = GetPowerOfTwoFrameBufferTexture( );
|
|
int sw = pTexture->GetActualWidth();
|
|
int sh = pTexture->GetActualHeight();
|
|
materials->DrawScreenSpaceRectangle( m_ScreenOverlayMaterial, x, y, w, h,
|
|
0, 0, sw-1, sh-1, sw, sh );
|
|
}
|
|
else
|
|
{
|
|
byte color[4] = { 255, 255, 255, 255 };
|
|
render->ViewDrawFade( color, m_ScreenOverlayMaterial );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders world and all entities, etc.
|
|
//-----------------------------------------------------------------------------
|
|
#include "tier0/memdbgoff.h"
|
|
|
|
void CViewRender::DrawWorldAndEntities( bool bDrawSkybox, const CViewSetup &view, int nClearFlags )
|
|
{
|
|
VPROF("CViewRender::DrawWorldAndEntities");
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
|
|
ClientWorldListInfo_t info;
|
|
#ifndef _XBOX
|
|
CRenderList renderList;
|
|
#else
|
|
void *ptr = MemAllocScratch(sizeof(CRenderList));
|
|
CRenderList *renderList = new (ptr) CRenderList;
|
|
#endif
|
|
|
|
EnableWorldFog();
|
|
|
|
int nFlags = DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER | DF_RENDER_WATER;
|
|
if (bDrawSkybox)
|
|
{
|
|
nFlags |= DF_DRAWSKYBOX;
|
|
}
|
|
|
|
render->Push3DView( view, nClearFlags, false, NULL, m_Frustum );
|
|
|
|
BuildWorldRenderLists( &view, info, true, true, m_iForceViewLeaf );
|
|
#ifndef _XBOX
|
|
BuildRenderableRenderLists( &view, info, renderList );
|
|
#else
|
|
BuildRenderableRenderLists( &view, info, *renderList );
|
|
#endif
|
|
|
|
// Make sure sound doesn't stutter
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
m_DrawFlags = m_BaseDrawFlags;
|
|
#ifndef _XBOX
|
|
DrawWorld( info, renderList, nFlags, 0.0f );
|
|
DrawOpaqueRenderables( info, renderList );
|
|
DrawTranslucentRenderables( info, renderList, nFlags, false );
|
|
#else
|
|
DrawWorld( info, *renderList, nFlags, 0.0f );
|
|
DrawOpaqueRenderables( info, *renderList );
|
|
DrawTranslucentRenderables( info, *renderList, nFlags, false );
|
|
MemFreeScratch();
|
|
#endif
|
|
|
|
m_DrawFlags = 0;
|
|
|
|
ParticleMgr()->DrawBeforeViewModelEffects();
|
|
|
|
render->PopView( m_Frustum );
|
|
}
|
|
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Computes us some geometry to render the frustum planes
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::ComputeFrustumRenderGeometry( Vector pRenderPoint[8] )
|
|
{
|
|
Vector viewPoint = CurrentViewOrigin();
|
|
|
|
// Find lines along each of the plane intersections.
|
|
// We know these lines are perpendicular to both plane normals,
|
|
// so we can take the cross product to find them.
|
|
static int edgeIdx[4][2] =
|
|
{
|
|
{ 0, 2 }, { 0, 3 }, { 1, 3 }, { 1, 2 }
|
|
};
|
|
|
|
int i;
|
|
Vector edges[4];
|
|
for ( i = 0; i < 4; ++i)
|
|
{
|
|
CrossProduct( GetFrustum()[edgeIdx[i][0]].m_Normal,
|
|
GetFrustum()[edgeIdx[i][1]].m_Normal, edges[i] );
|
|
VectorNormalize( edges[i] );
|
|
}
|
|
|
|
// Figure out four near points by intersection lines with the near plane
|
|
// Figure out four far points by intersection with lines against far plane
|
|
for (i = 0; i < 4; ++i)
|
|
{
|
|
float t = (GetFrustum()[4].m_Dist - DotProduct(GetFrustum()[4].m_Normal, viewPoint)) /
|
|
DotProduct(GetFrustum()[4].m_Normal, edges[i]);
|
|
VectorMA( viewPoint, t, edges[i], pRenderPoint[i] );
|
|
|
|
/*
|
|
t = (m_FrustumPlanes[5][3] - DotProduct(GetFrustum()[5], viewPoint)) /
|
|
DotProduct(GetFrustum()[5], edges[i]);
|
|
VectorMA( viewPoint, t, edges[i], pRenderPoint[i + 4] );
|
|
*/
|
|
if (t < 0)
|
|
{
|
|
edges[i] *= -1;
|
|
}
|
|
|
|
VectorMA( pRenderPoint[i], 200.0, edges[i], pRenderPoint[i + 4] );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// renders the frustum
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::RenderFrustum( )
|
|
{
|
|
static int indices[] =
|
|
{
|
|
0, 1, 1, 2, 2, 3, 3, 0, // near square
|
|
4, 5, 5, 6, 6, 7, 7, 4, // far square
|
|
0, 4, 1, 5, 2, 6, 3, 7 // connections between them
|
|
};
|
|
|
|
int numIndices = sizeof(indices) / sizeof(int);
|
|
|
|
Vector vecFrustumRenderPoint[8];
|
|
ComputeFrustumRenderGeometry( vecFrustumRenderPoint );
|
|
|
|
int i;
|
|
for ( i = 0; i < numIndices; i += 2 )
|
|
{
|
|
debugoverlay->AddLineOverlay( vecFrustumRenderPoint[indices[i]],
|
|
vecFrustumRenderPoint[indices[i+1]], 0, 0, 255, 255, 1.0f );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws all the debugging info
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::Draw3DDebuggingInfo( const CViewSetup &view )
|
|
{
|
|
VPROF("CViewRender::Draw3DDebuggingInfo");
|
|
|
|
// Draw 3d overlays
|
|
render->Draw3DDebugOverlays();
|
|
|
|
// Draw the line file used for debugging leaks
|
|
render->DrawLineFile();
|
|
|
|
// Draw client side effects
|
|
// NOTE: These are not sorted against the rest of the frame
|
|
clienteffects->DrawEffects( gpGlobals->frametime );
|
|
|
|
// Mark the frame as locked down for client fx additions
|
|
SetFXCreationAllowed( false );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws all the debugging info
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::Draw2DDebuggingInfo( const CViewSetup &view )
|
|
{
|
|
if ( IsXbox() && IsRetail() )
|
|
return;
|
|
|
|
// HDRFIXME: Assert NULL rendertarget
|
|
if ( mat_yuv.GetInt() && (engine->GetDXSupportLevel() >= 80) )
|
|
{
|
|
IMaterial *pMaterial;
|
|
pMaterial = materials->FindMaterial( "debug/yuv", TEXTURE_GROUP_OTHER, true );
|
|
if( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
DrawScreenEffectMaterial( pMaterial, view.x, view.y, view.width, view.height );
|
|
}
|
|
}
|
|
|
|
#ifndef _XBOX
|
|
if ( mat_hsv.GetInt() && (engine->GetDXSupportLevel() >= 90) )
|
|
{
|
|
IMaterial *pMaterial;
|
|
pMaterial = materials->FindMaterial( "debug/hsv", TEXTURE_GROUP_OTHER, true );
|
|
if( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
DrawScreenEffectMaterial( pMaterial, view.x, view.y, view.width, view.height );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Draw debugging lightmaps
|
|
render->DrawLightmaps();
|
|
|
|
if ( cl_drawshadowtexture.GetInt() )
|
|
{
|
|
g_pClientShadowMgr->RenderShadowTexture( 256, 256 );
|
|
}
|
|
|
|
const char *pDrawMaterial = cl_drawmaterial.GetString();
|
|
if ( pDrawMaterial && pDrawMaterial[0] )
|
|
{
|
|
RenderMaterial( pDrawMaterial );
|
|
}
|
|
|
|
if ( mat_showwatertextures.GetBool() )
|
|
{
|
|
OverlayWaterTextures();
|
|
}
|
|
|
|
if ( mat_showcamerarendertarget.GetBool() )
|
|
{
|
|
float w = mat_wateroverlaysize.GetFloat();
|
|
float h = mat_wateroverlaysize.GetFloat();
|
|
OverlayCameraRenderTarget( "debug/debugcamerarendertarget", 0, 0, w, h );
|
|
}
|
|
|
|
if ( mat_showframebuffertexture.GetBool() )
|
|
{
|
|
// HDRFIXME: Get rid of these rendertarget sets assuming that the assert at the top of this function is true.
|
|
materials->PushRenderTargetAndViewport( NULL );
|
|
OverlayFrameBufferTexture( 0 );
|
|
OverlayFrameBufferTexture( 1 );
|
|
materials->PopRenderTargetAndViewport( );
|
|
}
|
|
|
|
#ifdef _XBOX
|
|
if ( mat_viewTextureEnable.GetBool() )
|
|
{
|
|
OverlayShowTexture( mat_viewTextureName.GetString(), mat_viewTextureScale.GetFloat() );
|
|
}
|
|
#endif
|
|
|
|
if( r_flashlightdrawdepth.GetBool() )
|
|
{
|
|
shadowmgr->DrawFlashlightDepthTexture( );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Returns the min/max fade distances
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::GetScreenFadeDistances( float *min, float *max )
|
|
{
|
|
if ( min )
|
|
{
|
|
*min = r_screenfademinsize.GetFloat();
|
|
}
|
|
|
|
if ( max )
|
|
{
|
|
*max = r_screenfademaxsize.GetFloat();
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders world and all entities, etc.
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::ViewDrawScene( bool bDrew3dSkybox, bool bSkyboxVisible, const CViewSetup &view,
|
|
int nClearFlags, view_id_t viewID, bool bDrawViewModel, int baseDrawFlags )
|
|
{
|
|
VPROF( "CViewRender::ViewDrawScene" );
|
|
|
|
m_BaseDrawFlags = baseDrawFlags;
|
|
m_DrawFlags = 0;
|
|
|
|
SetupCurrentView( view.origin, view.angles, viewID );
|
|
|
|
// Invoke pre-render methods
|
|
IGameSystem::PreRenderAllSystems();
|
|
|
|
// Start view
|
|
unsigned int visFlags;
|
|
SetupVis( view, visFlags );
|
|
|
|
if ( !bDrew3dSkybox &&
|
|
!bSkyboxVisible && ( visFlags & IVRenderView::VIEW_SETUP_VIS_EX_RETURN_FLAGS_USES_RADIAL_VIS ) )
|
|
{
|
|
// This covers the case where we don't see a 3dskybox, yet radial vis is clipping
|
|
// the far plane. Need to clear to fog color in this case.
|
|
nClearFlags |= VIEW_CLEAR_COLOR;
|
|
unsigned char ucFogColor[3];
|
|
SetClearColorToFogColor( ucFogColor );
|
|
}
|
|
|
|
bool drawSkybox = r_skybox.GetBool();
|
|
if ( bDrew3dSkybox || !bSkyboxVisible )
|
|
{
|
|
drawSkybox = false;
|
|
}
|
|
|
|
ParticleMgr()->IncrementFrameCode();
|
|
|
|
WaterDrawWorldAndEntities( drawSkybox, view, nClearFlags );
|
|
|
|
// Disable fog for the rest of the stuff
|
|
DisableFog();
|
|
|
|
// UNDONE: Don't do this with masked brush models, they should probably be in a separate list
|
|
// render->DrawMaskEntities()
|
|
|
|
// Here are the overlays...
|
|
|
|
// This is an overlay that goes over everything else
|
|
|
|
CGlowOverlay::DrawOverlays( view.m_bCacheFullSceneState );
|
|
|
|
// issue the pixel visibility tests
|
|
PixelVisibility_EndCurrentView();
|
|
|
|
// Draw rain..
|
|
DrawPrecipitation();
|
|
|
|
// Make sure sound doesn't stutter
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
// Debugging info goes over the top
|
|
Draw3DDebuggingInfo( view );
|
|
|
|
// Invoke post-render methods
|
|
IGameSystem::PostRenderAllSystems();
|
|
|
|
FinishCurrentView();
|
|
|
|
m_DrawFlags = 0;
|
|
}
|
|
|
|
void CheckAndTransitionColor( float flPercent, float *pColor, float *pLerpToColor )
|
|
{
|
|
if ( pLerpToColor[0] != pColor[0] || pLerpToColor[1] != pColor[1] || pLerpToColor[2] != pColor[2] )
|
|
{
|
|
float flDestColor[3];
|
|
|
|
flDestColor[0] = pLerpToColor[0];
|
|
flDestColor[1] = pLerpToColor[1];
|
|
flDestColor[2] = pLerpToColor[2];
|
|
|
|
pColor[0] = FLerp( pColor[0], flDestColor[0], flPercent );
|
|
pColor[1] = FLerp( pColor[1], flDestColor[1], flPercent );
|
|
pColor[2] = FLerp( pColor[2], flDestColor[2], flPercent );
|
|
}
|
|
else
|
|
{
|
|
pColor[0] = pLerpToColor[0];
|
|
pColor[1] = pLerpToColor[1];
|
|
pColor[2] = pLerpToColor[2];
|
|
}
|
|
}
|
|
|
|
static void GetFogColorTransition( CPlayerLocalData *local, float *pColorPrimary, float *pColorSecondary )
|
|
{
|
|
if ( local == NULL )
|
|
return;
|
|
|
|
if ( local->m_fog.lerptime >= gpGlobals->curtime )
|
|
{
|
|
float flPercent = 1.0f - (( local->m_fog.lerptime - gpGlobals->curtime ) / local->m_fog.duration );
|
|
|
|
float flPrimaryColorLerp[3] = { local->m_fog.colorPrimaryLerpTo.GetR(), local->m_fog.colorPrimaryLerpTo.GetG(), local->m_fog.colorPrimaryLerpTo.GetB() };
|
|
float flSecondaryColorLerp[3] = { local->m_fog.colorSecondaryLerpTo.GetR(), local->m_fog.colorSecondaryLerpTo.GetG(), local->m_fog.colorSecondaryLerpTo.GetB() };
|
|
|
|
CheckAndTransitionColor( flPercent, pColorPrimary, flPrimaryColorLerp );
|
|
CheckAndTransitionColor( flPercent, pColorSecondary, flSecondaryColorLerp );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Returns the fog color to use in rendering the current frame.
|
|
//-----------------------------------------------------------------------------
|
|
static void GetFogColor( float *pColor )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
const char *fogColorString = fog_color.GetString();
|
|
if( fog_override.GetInt() && fogColorString )
|
|
{
|
|
sscanf( fogColorString, "%f%f%f", pColor, pColor+1, pColor+2 );
|
|
}
|
|
else
|
|
{
|
|
float flPrimaryColor[3] = { local->m_fog.colorPrimary.GetR(), local->m_fog.colorPrimary.GetG(), local->m_fog.colorPrimary.GetB() };
|
|
float flSecondaryColor[3] = { local->m_fog.colorSecondary.GetR(), local->m_fog.colorSecondary.GetG(), local->m_fog.colorSecondary.GetB() };
|
|
|
|
GetFogColorTransition( local, flPrimaryColor, flSecondaryColor );
|
|
|
|
if( local->m_fog.blend )
|
|
{
|
|
//
|
|
// Blend between two fog colors based on viewing angle.
|
|
// The secondary fog color is at 180 degrees to the primary fog color.
|
|
//
|
|
Vector forward;
|
|
AngleVectors(pbp->GetAbsAngles(), &forward);
|
|
|
|
Vector vNormalized = local->m_fog.dirPrimary;
|
|
VectorNormalize( vNormalized );
|
|
local->m_fog.dirPrimary = vNormalized;
|
|
|
|
float flBlendFactor = 0.5 * forward.Dot( local->m_fog.dirPrimary ) + 0.5;
|
|
|
|
// FIXME: convert to linear colorspace
|
|
pColor[0] = flPrimaryColor[0] * flBlendFactor + flSecondaryColor[0] * ( 1 - flBlendFactor );
|
|
pColor[1] = flPrimaryColor[1] * flBlendFactor + flSecondaryColor[1] * ( 1 - flBlendFactor );
|
|
pColor[2] = flPrimaryColor[2] * flBlendFactor + flSecondaryColor[2] * ( 1 - flBlendFactor );
|
|
}
|
|
else
|
|
{
|
|
pColor[0] = flPrimaryColor[0];
|
|
pColor[1] = flPrimaryColor[1];
|
|
pColor[2] = flPrimaryColor[2];
|
|
}
|
|
}
|
|
|
|
VectorScale( pColor, 1.0f / 255.0f, pColor );
|
|
}
|
|
|
|
|
|
static float GetFogStart( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return 0.0f;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_start.GetFloat() == -1.0f )
|
|
{
|
|
return local->m_fog.start;
|
|
}
|
|
else
|
|
{
|
|
return fog_start.GetFloat();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( local->m_fog.lerptime > gpGlobals->curtime )
|
|
{
|
|
if ( local->m_fog.start != local->m_fog.startLerpTo )
|
|
{
|
|
if ( local->m_fog.lerptime > gpGlobals->curtime )
|
|
{
|
|
float flPercent = 1.0f - (( local->m_fog.lerptime - gpGlobals->curtime ) / local->m_fog.duration );
|
|
|
|
return FLerp( local->m_fog.start, local->m_fog.startLerpTo, flPercent );
|
|
}
|
|
else
|
|
{
|
|
if ( local->m_fog.start != local->m_fog.startLerpTo )
|
|
{
|
|
local->m_fog.start = local->m_fog.startLerpTo;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return local->m_fog.start;
|
|
}
|
|
}
|
|
|
|
static float GetFogEnd( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return 0.0f;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_end.GetFloat() == -1.0f )
|
|
{
|
|
return local->m_fog.end;
|
|
}
|
|
else
|
|
{
|
|
return fog_end.GetFloat();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( local->m_fog.lerptime > gpGlobals->curtime )
|
|
{
|
|
if ( local->m_fog.end != local->m_fog.endLerpTo )
|
|
{
|
|
if ( local->m_fog.lerptime > gpGlobals->curtime )
|
|
{
|
|
float flPercent = 1.0f - (( local->m_fog.lerptime - gpGlobals->curtime ) / local->m_fog.duration );
|
|
|
|
return FLerp( local->m_fog.end, local->m_fog.endLerpTo, flPercent );
|
|
}
|
|
else
|
|
{
|
|
if ( local->m_fog.end != local->m_fog.endLerpTo )
|
|
{
|
|
local->m_fog.end = local->m_fog.endLerpTo;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return local->m_fog.end;
|
|
}
|
|
}
|
|
|
|
static bool GetFogEnable( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return false;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if ( cl_leveloverview.GetFloat() > 0 )
|
|
return false;
|
|
|
|
// Ask the clientmode
|
|
if ( g_pClientMode->ShouldDrawFog() == false )
|
|
return false;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_enable.GetInt() )
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return local->m_fog.enable != false;
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Returns the skybox fog color to use in rendering the current frame.
|
|
//-----------------------------------------------------------------------------
|
|
static void GetSkyboxFogColor( float *pColor )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
const char *fogColorString = fog_colorskybox.GetString();
|
|
if( fog_override.GetInt() && fogColorString )
|
|
{
|
|
sscanf( fogColorString, "%f%f%f", pColor, pColor+1, pColor+2 );
|
|
}
|
|
else
|
|
{
|
|
if( local->m_skybox3d.fog.blend )
|
|
{
|
|
//
|
|
// Blend between two fog colors based on viewing angle.
|
|
// The secondary fog color is at 180 degrees to the primary fog color.
|
|
//
|
|
Vector forward;
|
|
AngleVectors( pbp->GetAbsAngles(), &forward );
|
|
|
|
Vector vNormalized = local->m_fog.dirPrimary;
|
|
VectorNormalize( vNormalized );
|
|
local->m_fog.dirPrimary = vNormalized;
|
|
|
|
float flBlendFactor = 0.5 * forward.Dot( local->m_fog.dirPrimary ) + 0.5;
|
|
|
|
// FIXME: convert to linear colorspace
|
|
pColor[0] = local->m_skybox3d.fog.colorPrimary.GetR() * flBlendFactor + local->m_skybox3d.fog.colorSecondary.GetR() * ( 1 - flBlendFactor );
|
|
pColor[1] = local->m_skybox3d.fog.colorPrimary.GetG() * flBlendFactor + local->m_skybox3d.fog.colorSecondary.GetG() * ( 1 - flBlendFactor );
|
|
pColor[2] = local->m_skybox3d.fog.colorPrimary.GetB() * flBlendFactor + local->m_skybox3d.fog.colorSecondary.GetB() * ( 1 - flBlendFactor );
|
|
}
|
|
else
|
|
{
|
|
pColor[0] = local->m_skybox3d.fog.colorPrimary.GetR();
|
|
pColor[1] = local->m_skybox3d.fog.colorPrimary.GetG();
|
|
pColor[2] = local->m_skybox3d.fog.colorPrimary.GetB();
|
|
}
|
|
}
|
|
|
|
VectorScale( pColor, 1.0f / 255.0f, pColor );
|
|
}
|
|
|
|
|
|
static float GetSkyboxFogStart( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return 0.0f;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_startskybox.GetFloat() == -1.0f )
|
|
{
|
|
return local->m_skybox3d.fog.start;
|
|
}
|
|
else
|
|
{
|
|
return fog_startskybox.GetFloat();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return local->m_skybox3d.fog.start;
|
|
}
|
|
}
|
|
|
|
static float GetSkyboxFogEnd( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return 0.0f;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_endskybox.GetFloat() == -1.0f )
|
|
{
|
|
return local->m_skybox3d.fog.end;
|
|
}
|
|
else
|
|
{
|
|
return fog_endskybox.GetFloat();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return local->m_skybox3d.fog.end;
|
|
}
|
|
}
|
|
|
|
static bool GetSkyboxFogEnable( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return false;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( fog_override.GetInt() )
|
|
{
|
|
if( fog_enableskybox.GetInt() )
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return !!local->m_skybox3d.fog.enable;
|
|
}
|
|
}
|
|
|
|
void CViewRender::EnableWorldFog( void )
|
|
{
|
|
VPROF("CViewRender::EnableWorldFog");
|
|
if( GetFogEnable() )
|
|
{
|
|
float fogColor[3];
|
|
GetFogColor( fogColor );
|
|
materials->FogMode( MATERIAL_FOG_LINEAR );
|
|
materials->FogColor3fv( fogColor );
|
|
materials->FogStart( GetFogStart() );
|
|
materials->FogEnd( GetFogEnd() );
|
|
}
|
|
else
|
|
{
|
|
materials->FogMode( MATERIAL_FOG_NONE );
|
|
}
|
|
}
|
|
|
|
void CViewRender::Enable3dSkyboxFog( void )
|
|
{
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
if( !pbp )
|
|
{
|
|
return;
|
|
}
|
|
CPlayerLocalData *local = &pbp->m_Local;
|
|
|
|
if( GetSkyboxFogEnable() )
|
|
{
|
|
float fogColor[3];
|
|
GetSkyboxFogColor( fogColor );
|
|
float scale = 1.0f;
|
|
if ( local->m_skybox3d.scale > 0.0f )
|
|
{
|
|
scale = 1.0f / local->m_skybox3d.scale;
|
|
}
|
|
materials->FogMode( MATERIAL_FOG_LINEAR );
|
|
materials->FogColor3fv( fogColor );
|
|
materials->FogStart( GetSkyboxFogStart() * scale );
|
|
materials->FogEnd( GetSkyboxFogEnd() * scale );
|
|
}
|
|
else
|
|
{
|
|
materials->FogMode( MATERIAL_FOG_NONE );
|
|
}
|
|
}
|
|
|
|
void CViewRender::DisableFog( void )
|
|
{
|
|
VPROF("CViewRander::DisableFog()");
|
|
|
|
materials->FogMode( MATERIAL_FOG_NONE );
|
|
}
|
|
|
|
#include "tier0/memdbgoff.h"
|
|
|
|
void CViewRender::Draw3dSkyboxworld( const CViewSetup &view, int &nClearFlags,
|
|
bool &bDrew3dSkybox, bool &bSkyboxVisible )
|
|
{
|
|
VPROF_BUDGET( "CViewRender::Draw3dSkyboxworld", "3D Skybox" );
|
|
|
|
bDrew3dSkybox = false;
|
|
|
|
// The skybox might not be visible from here
|
|
bSkyboxVisible = engine->IsSkyboxVisibleFromPoint( view.origin );
|
|
|
|
if ( !bSkyboxVisible && r_3dsky.GetInt() != 2 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
// render the 3D skybox
|
|
if ( !r_3dsky.GetInt() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
C_BasePlayer *pbp = C_BasePlayer::GetLocalPlayer();
|
|
|
|
// No local player object yet...
|
|
if ( !pbp )
|
|
return;
|
|
|
|
CPlayerLocalData* local = &pbp->m_Local;
|
|
if ( local->m_skybox3d.area == 255 )
|
|
return;
|
|
|
|
unsigned char **areabits = render->GetAreaBits();
|
|
unsigned char *savebits;
|
|
unsigned char tmpbits[ 32 ];
|
|
savebits = *areabits;
|
|
memset( tmpbits, 0, sizeof(tmpbits) );
|
|
|
|
// set the sky area bit
|
|
tmpbits[local->m_skybox3d.area>>3] |= 1 << (local->m_skybox3d.area&7);
|
|
|
|
*areabits = tmpbits;
|
|
CViewSetup skyView = view;
|
|
skyView.zNear = 0.5;
|
|
skyView.zFar = 18000;
|
|
|
|
// scale origin by sky scale
|
|
if ( local->m_skybox3d.scale > 0 )
|
|
{
|
|
float scale = 1.0f / local->m_skybox3d.scale;
|
|
VectorScale( skyView.origin, scale, skyView.origin );
|
|
}
|
|
Enable3dSkyboxFog();
|
|
VectorAdd( skyView.origin, local->m_skybox3d.origin, skyView.origin );
|
|
|
|
skyView.m_vUnreflectedOrigin = skyView.origin;
|
|
|
|
// BUGBUG: Fix this!!! We shouldn't need to call setup vis for the sky if we're connecting
|
|
// the areas. We'd have to mark all the clusters in the skybox area in the PVS of any
|
|
// cluster with sky. Then we could just connect the areas to do our vis.
|
|
//m_bOverrideVisOrigin could hose us here, so call direct
|
|
render->ViewSetupVis( false, 1, &local->m_skybox3d.origin.Get() );
|
|
render->Push3DView( skyView, nClearFlags, false, NULL, m_Frustum );
|
|
|
|
// At this point, we've cleared everything we need to clear
|
|
// The next path will need to clear depth, though.
|
|
nClearFlags &= ~(VIEW_CLEAR_COLOR | VIEW_CLEAR_DEPTH | VIEW_CLEAR_FULL_TARGET);
|
|
nClearFlags |= VIEW_CLEAR_DEPTH;
|
|
|
|
// Store off view origin and angles
|
|
SetupCurrentView( skyView.origin, skyView.angles, VIEW_3DSKY );
|
|
|
|
// Invoke pre-render methods
|
|
IGameSystem::PreRenderAllSystems();
|
|
|
|
ClientWorldListInfo_t info;
|
|
#ifndef _XBOX
|
|
CRenderList renderList;
|
|
CRenderList *pRenderList = &renderList;
|
|
#else
|
|
void *ptr = MemAllocScratch(sizeof(CRenderList));
|
|
CRenderList *pRenderList = new (ptr) CRenderList;
|
|
#endif
|
|
|
|
BuildWorldRenderLists( NULL, info, true, true, m_iForceViewLeaf );
|
|
BuildRenderableRenderLists( NULL, info, *pRenderList );
|
|
|
|
int flags = DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER | DF_RENDER_WATER;
|
|
if( r_skybox.GetBool() )
|
|
{
|
|
flags |= DF_DRAWSKYBOX;
|
|
}
|
|
|
|
int oldDrawFlags = m_DrawFlags;
|
|
m_DrawFlags |= flags;
|
|
DrawWorld( info, *pRenderList, flags, 0.0f );
|
|
|
|
// Iterate over all leaves and render objects in those leaves
|
|
DrawOpaqueRenderables( info, *pRenderList );
|
|
|
|
// Iterate over all leaves and render objects in those leaves
|
|
DrawTranslucentRenderables( info, *pRenderList, flags, true );
|
|
|
|
DisableFog();
|
|
|
|
CGlowOverlay::UpdateSkyOverlays( skyView.zFar, view.m_bCacheFullSceneState );
|
|
|
|
PixelVisibility_EndCurrentView();
|
|
|
|
// restore old area bits
|
|
*areabits = savebits;
|
|
|
|
// Invoke post-render methods
|
|
IGameSystem::PostRenderAllSystems();
|
|
|
|
FinishCurrentView();
|
|
|
|
m_DrawFlags = oldDrawFlags;
|
|
|
|
#ifdef _XBOX
|
|
MemFreeScratch();
|
|
#endif
|
|
|
|
render->PopView( m_Frustum );
|
|
bDrew3dSkybox = true;
|
|
}
|
|
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::SetupVis( const CViewSetup& view, unsigned int &visFlags )
|
|
{
|
|
VPROF( "CViewRender::SetupVis" );
|
|
|
|
if ( m_bOverrideVisOrigin )
|
|
{
|
|
// Pass array or vis origins to merge
|
|
render->ViewSetupVisEx( m_bForceNoVis, m_nNumVisOrigins, m_rgVisOrigins, visFlags );
|
|
}
|
|
else
|
|
{
|
|
// Use render origin as vis origin by default
|
|
render->ViewSetupVisEx( m_bForceNoVis, 1, &view.origin, visFlags );
|
|
}
|
|
}
|
|
|
|
#ifndef _XBOX
|
|
|
|
// hdr parameters
|
|
ConVar mat_hdr_level( "mat_hdr_level", "2" );
|
|
ConVar mat_bloomamount_rate( "mat_bloomamount_rate", "0.05f" );
|
|
static ConVar debug_postproc( "mat_debug_postprocessing_effects", "0" );
|
|
static ConVar split_postproc( "mat_debug_process_halfscreen", "0" );
|
|
static ConVar mat_dynamic_tonemapping( "mat_dynamic_tonemapping", "0" );
|
|
static ConVar mat_show_ab_hdr( "mat_show_ab_hdr", "0" );
|
|
static ConVar mat_tonemapping_occlusion_use_stencil( "mat_tonemapping_occlusion_use_stencil", "1" );
|
|
ConVar mat_debug_autoexposure("mat_debug_autoexposure","0");
|
|
static ConVar mat_autoexposure_max( "mat_autoexposure_max", "2" );
|
|
static ConVar mat_autoexposure_min( "mat_autoexposure_min", "0.5" );
|
|
static ConVar mat_show_histogram( "mat_show_histogram", "0" );
|
|
ConVar mat_hdr_tonemapscale( "mat_hdr_tonemapscale", "1.0", FCVAR_CHEAT );
|
|
ConVar mat_force_bloom("mat_force_bloom","0");
|
|
ConVar mat_disable_bloom("mat_disable_bloom","0");
|
|
ConVar mat_debug_bloom("mat_debug_bloom","0");
|
|
|
|
// fudge factor to make non-hdr bloom more closely match hdr bloom. Because of auto-exposure, high
|
|
// bloomscales don't blow out as much in hdr. this factor was derived by comparing images in a
|
|
// reference scene.
|
|
ConVar mat_non_hdr_bloom_scalefactor("mat_non_hdr_bloom_scalefactor",".3");
|
|
|
|
|
|
enum PostProcessingCondition {
|
|
PPP_ALWAYS,
|
|
PPP_IF_COND_VAR,
|
|
PPP_IF_NOT_COND_VAR
|
|
};
|
|
|
|
struct PostProcessingPass {
|
|
PostProcessingCondition ppp_test;
|
|
ConVar const *cvar_to_test;
|
|
char const *material_name; // terminate list with null
|
|
char const *dest_rendering_target;
|
|
char const *src_rendering_target; // can be null. needed for source scaling
|
|
int xdest_scale,ydest_scale; // allows scaling down
|
|
int xsrc_scale,ysrc_scale; // allows scaling down
|
|
CMaterialReference m_mat_ref; // so we don't have to keep searching
|
|
};
|
|
|
|
#define PPP_PROCESS_PARTIAL_SRC(srcmatname,dest_rt_name,src_tname,scale) \
|
|
{PPP_ALWAYS,0,srcmatname,dest_rt_name,src_tname,1,1,scale,scale}
|
|
#define PPP_PROCESS_PARTIAL_DEST(srcmatname,dest_rt_name,src_tname,scale) \
|
|
{PPP_ALWAYS,0,srcmatname,dest_rt_name,src_tname,scale,scale,1,1}
|
|
#define PPP_PROCESS_PARTIAL_SRC_PARTIAL_DEST(srcmatname,dest_rt_name,src_tname,srcscale,destscale) \
|
|
{PPP_ALWAYS,0,srcmatname,dest_rt_name,src_tname,destscale,destscale,srcscale,srcscale}
|
|
#define PPP_END {PPP_ALWAYS,0,NULL,NULL,0,0,0,0,0}
|
|
#define PPP_PROCESS(srcmatname,dest_rt_name) {PPP_ALWAYS,0,srcmatname,dest_rt_name,0,1,1,1,1}
|
|
#define PPP_PROCESS_IF_CVAR(cvarptr,srcmatname,dest_rt_name) \
|
|
{PPP_IF_COND_VAR,cvarptr,srcmatname,dest_rt_name,0,1,1,1,1}
|
|
#define PPP_PROCESS_IF_NOT_CVAR(cvarptr,srcmatname,dest_rt_name) \
|
|
{PPP_IF_NOT_COND_VAR,cvarptr,srcmatname,dest_rt_name,0,1,1,1,1}
|
|
#define PPP_PROCESS_IF_NOT_CVAR_SRCTEXTURE(cvarptr,srcmatname,src_tname,dest_rt_name) \
|
|
{PPP_IF_NOT_COND_VAR,cvarptr,srcmatname,dest_rt_name,src_tname,1,1,1,1}
|
|
#define PPP_PROCESS_IF_CVAR_SRCTEXTURE(cvarptr,srcmatname,src_txtrname,dest_rt_name) \
|
|
{PPP_IF_COND_VAR,cvarptr,srcmatname,dest_rt_name,src_txtrname,1,1,1,1}
|
|
#define PPP_PROCESS_SRCTEXTURE(srcmatname,src_tname,dest_rt_name) \
|
|
{PPP_ALWAYS,0,srcmatname,dest_rt_name,src_tname,1,1,1,1}
|
|
|
|
struct ClipBox
|
|
{
|
|
int m_minx, m_miny;
|
|
int m_maxx, m_maxy;
|
|
};
|
|
|
|
static void DrawClippedScreenSpaceRectangle(
|
|
IMaterial *pMaterial,
|
|
int destx, int desty,
|
|
int width, int height,
|
|
float src_texture_x0, float src_texture_y0, // which texel you want to appear at
|
|
// destx/y
|
|
float src_texture_x1, float src_texture_y1, // which texel you want to appear at
|
|
// destx+width-1, desty+height-1
|
|
int src_texture_width, int src_texture_height, // needed for fixup
|
|
ClipBox const *clipbox
|
|
)
|
|
{
|
|
if (clipbox)
|
|
{
|
|
if ((destx>clipbox->m_maxx) || (desty>clipbox->m_maxy))
|
|
return;
|
|
if ((destx+width-1<clipbox->m_minx) || (desty+height-1<clipbox->m_miny))
|
|
return;
|
|
// left clip
|
|
if (destx<clipbox->m_minx)
|
|
{
|
|
src_texture_x0=FLerp(src_texture_x0,src_texture_x1,destx,destx+width-1,clipbox->m_minx);
|
|
width-=(clipbox->m_minx-destx);
|
|
destx=clipbox->m_minx;
|
|
}
|
|
// top clip
|
|
if (desty<clipbox->m_miny)
|
|
{
|
|
src_texture_y0=FLerp(src_texture_y0,src_texture_y1,desty,desty+height-1,clipbox->m_miny);
|
|
height-=(clipbox->m_miny-desty);
|
|
desty=clipbox->m_miny;
|
|
}
|
|
// right clip
|
|
if (destx+width-1>clipbox->m_maxx)
|
|
{
|
|
src_texture_x1=FLerp(src_texture_x0,src_texture_x1,destx,destx+width-1,clipbox->m_maxx);
|
|
width=clipbox->m_maxx-destx;
|
|
}
|
|
// bottom clip
|
|
if (desty+height-1>clipbox->m_maxy)
|
|
{
|
|
src_texture_y1=FLerp(src_texture_y0,src_texture_y1,desty,desty+height-1,clipbox->m_maxy);
|
|
height=clipbox->m_maxy-desty;
|
|
}
|
|
}
|
|
materials->DrawScreenSpaceRectangle(pMaterial,destx,desty,width,height,src_texture_x0,
|
|
src_texture_y0,src_texture_x1,src_texture_y1,
|
|
src_texture_width,src_texture_height);
|
|
}
|
|
|
|
void ApplyPostProcessingPasses(PostProcessingPass *pass_list, // table of effects to apply
|
|
ClipBox const *clipbox=0, // clipping box for these effects
|
|
ClipBox *dest_coords_out=0) // receives dest coords of last blit
|
|
{
|
|
ITexture *pSaveRenderTarget = materials->GetRenderTarget();
|
|
int pcount=0;
|
|
if ( debug_postproc.GetInt() )
|
|
{
|
|
materials->SetRenderTarget(NULL);
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
materials->ClearColor3ub(255,0,0);
|
|
// materials->ClearBuffers(true,true);
|
|
}
|
|
while(pass_list->material_name)
|
|
{
|
|
bool do_it=true;
|
|
|
|
switch(pass_list->ppp_test)
|
|
{
|
|
case PPP_IF_COND_VAR:
|
|
do_it=(pass_list->cvar_to_test)->GetBool();
|
|
break;
|
|
case PPP_IF_NOT_COND_VAR:
|
|
do_it=! ((pass_list->cvar_to_test)->GetBool());
|
|
break;
|
|
}
|
|
if ((pass_list->dest_rendering_target==0) && (debug_postproc.GetInt()))
|
|
do_it=0;
|
|
if (do_it)
|
|
{
|
|
ClipBox const *cb=0;
|
|
if (pass_list->dest_rendering_target==0)
|
|
{
|
|
cb=clipbox;
|
|
}
|
|
|
|
IMaterial *src_mat=pass_list->m_mat_ref;
|
|
if (! src_mat)
|
|
{
|
|
src_mat=materials->FindMaterial(pass_list->material_name,
|
|
TEXTURE_GROUP_OTHER,true);
|
|
if (src_mat)
|
|
{
|
|
pass_list->m_mat_ref.Init(src_mat);
|
|
}
|
|
}
|
|
if (pass_list->dest_rendering_target)
|
|
{
|
|
ITexture *dest_rt=materials->FindTexture(pass_list->dest_rendering_target,
|
|
TEXTURE_GROUP_RENDER_TARGET);
|
|
materials->SetRenderTarget( dest_rt);
|
|
}
|
|
else
|
|
{
|
|
materials->SetRenderTarget( NULL );
|
|
}
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
dest_width/=pass_list->xdest_scale;
|
|
dest_height/=pass_list->ydest_scale;
|
|
|
|
if (pass_list->src_rendering_target)
|
|
{
|
|
ITexture *src_rt=materials->FindTexture(pass_list->src_rendering_target,
|
|
TEXTURE_GROUP_RENDER_TARGET);
|
|
int src_width=src_rt->GetActualWidth();
|
|
int src_height=src_rt->GetActualHeight();
|
|
int ssrc_width=(src_width-1)/pass_list->xsrc_scale;
|
|
int ssrc_height=(src_height-1)/pass_list->ysrc_scale;
|
|
DrawClippedScreenSpaceRectangle(
|
|
src_mat,0,0,dest_width,dest_height,
|
|
0,0,ssrc_width,ssrc_height,src_width,src_height,cb);
|
|
if ((pass_list->dest_rendering_target) && debug_postproc.GetInt())
|
|
{
|
|
materials->SetRenderTarget(NULL);
|
|
int row=pcount/2;
|
|
int col=pcount %2;
|
|
int vdest_width,vdest_height;
|
|
materials->GetRenderTargetDimensions( vdest_width, vdest_height );
|
|
materials->Viewport( 0, 0, vdest_width, vdest_height );
|
|
materials->DrawScreenSpaceRectangle(
|
|
src_mat,col*400,200+row*300,dest_width,dest_height,
|
|
0,0,ssrc_width,ssrc_height,src_width,src_height);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// just draw the whole source
|
|
if ((pass_list->dest_rendering_target==0) && split_postproc.GetInt())
|
|
{
|
|
DrawClippedScreenSpaceRectangle(src_mat,0,0,dest_width/2,dest_height,
|
|
0,0,.5,1,1,1,cb);
|
|
}
|
|
else
|
|
{
|
|
DrawClippedScreenSpaceRectangle(src_mat,0,0,dest_width,dest_height,
|
|
0,0,1,1,1,1,cb);
|
|
}
|
|
if ((pass_list->dest_rendering_target) && debug_postproc.GetInt())
|
|
{
|
|
materials->SetRenderTarget(NULL);
|
|
int row=pcount/4;
|
|
int col=pcount %4;
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
DrawClippedScreenSpaceRectangle(src_mat,10+col*220,10+row*220,
|
|
200,200,
|
|
0,0,1,1,1,1,cb);
|
|
}
|
|
}
|
|
if (dest_coords_out)
|
|
{
|
|
dest_coords_out->m_minx=0;
|
|
dest_coords_out->m_maxx=dest_width-1;
|
|
dest_coords_out->m_miny=0;
|
|
dest_coords_out->m_maxy=dest_height-1;
|
|
}
|
|
}
|
|
pass_list++;
|
|
pcount++;
|
|
}
|
|
materials->SetRenderTarget(pSaveRenderTarget);
|
|
}
|
|
|
|
PostProcessingPass HDRFinal_Float[] =
|
|
{
|
|
PPP_PROCESS_SRCTEXTURE("dev/downsample","_rt_FullFrameFB", "_rt_SmallFB0"),
|
|
PPP_PROCESS("dev/blurfilterx","_rt_SmallFB1"),
|
|
PPP_PROCESS("dev/blurfiltery","_rt_SmallFB0"),
|
|
PPP_PROCESS("dev/floattoscreen_combine",NULL),
|
|
PPP_END
|
|
};
|
|
|
|
PostProcessingPass HDRFinal_Float_NoBloom[] =
|
|
{
|
|
PPP_PROCESS_SRCTEXTURE("dev/copyfullframefb", "_rt_FullFrameFB",NULL),
|
|
PPP_END
|
|
};
|
|
|
|
PostProcessingPass HDRSimulate_NonHDR[]={
|
|
PPP_PROCESS("dev/copyfullframefb_vanilla",NULL),
|
|
PPP_END
|
|
};
|
|
|
|
static bool IsSplitScreenHDR(void)
|
|
{
|
|
return ( mat_show_ab_hdr.GetInt() &&
|
|
(g_pMaterialSystemHardwareConfig->GetHDRType() != HDR_TYPE_NONE) );
|
|
}
|
|
|
|
static void SetRenderTargetAndViewPort(ITexture *rt)
|
|
{
|
|
materials->SetRenderTarget(rt);
|
|
materials->Viewport(0,0,rt->GetActualWidth(),rt->GetActualHeight());
|
|
}
|
|
|
|
#define FILTER_KERNEL_SLOP 20
|
|
|
|
// Note carefully about the downsampling: the first downsampling samples from the full rendertarget
|
|
// down to a temp. When doing this sampling, the texture source clamping will take care of the out
|
|
// of bounds sampling done because of the filter kernels's width. However, on any of the subsequent
|
|
// sampling operations, we will be sampling from a partially filled render target. So, texture
|
|
// coordinate clamping cannot help us here. So, we need to always render a few more pixels to the
|
|
// destination than we actually intend to, so as to replicate the border pixels so that garbage
|
|
// pixels do not get sucked into the sampling. To deal with this, we always add FILTER_KERNEL_SLOP
|
|
// to our widths/heights if there is room for them in the destination.
|
|
static void DrawScreenSpaceRectangleWithSlop(
|
|
ITexture *dest_rt,
|
|
IMaterial *pMaterial,
|
|
int destx, int desty,
|
|
int width, int height,
|
|
float src_texture_x0, float src_texture_y0, // which texel you want to appear at
|
|
// destx/y
|
|
float src_texture_x1, float src_texture_y1, // which texel you want to appear at
|
|
// destx+width-1, desty+height-1
|
|
int src_texture_width, int src_texture_height // needed for fixup
|
|
)
|
|
{
|
|
// add slop
|
|
int slopwidth=width+FILTER_KERNEL_SLOP; //min(dest_rt->GetActualWidth()-destx,width+FILTER_KERNEL_SLOP);
|
|
int slopheight=height+FILTER_KERNEL_SLOP; //min(dest_rt->GetActualHeight()-desty,height+FILTER_KERNEL_SLOP);
|
|
// adjust coordinates for slop
|
|
src_texture_x1=FLerp(src_texture_x0,src_texture_x1,destx,destx+width-1,destx+slopwidth-1);
|
|
src_texture_y1=FLerp(src_texture_y0,src_texture_y1,desty,desty+height-1,desty+slopheight-1);
|
|
width=slopwidth;
|
|
height=slopheight;
|
|
materials->DrawScreenSpaceRectangle(pMaterial,destx,desty,width,height,
|
|
src_texture_x0,src_texture_y0,
|
|
src_texture_x1,src_texture_y1,
|
|
src_texture_width,src_texture_height);
|
|
|
|
}
|
|
|
|
|
|
|
|
enum Histogram_entry_state_t
|
|
{
|
|
HESTATE_INITIAL=0,
|
|
HESTATE_FIRST_QUERY_IN_FLIGHT,
|
|
HESTATE_QUERY_IN_FLIGHT,
|
|
HESTATE_QUERY_DONE,
|
|
};
|
|
|
|
|
|
ConVar mat_exposure_center_region_x( "mat_exposure_center_region_x","0.75", FCVAR_CHEAT );
|
|
ConVar mat_exposure_center_region_y( "mat_exposure_center_region_y","0.80", FCVAR_CHEAT );
|
|
ConVar mat_exposure_center_region_x_flashlight( "mat_exposure_center_region_x_flashlight","0.33", FCVAR_CHEAT );
|
|
ConVar mat_exposure_center_region_y_flashlight( "mat_exposure_center_region_y_flashlight","0.33", FCVAR_CHEAT );
|
|
|
|
#define N_LUMINANCE_RANGES 31
|
|
#define MAX_QUERIES_PER_FRAME 1
|
|
|
|
class CHistogram_entry_t
|
|
{
|
|
public:
|
|
Histogram_entry_state_t m_state;
|
|
OcclusionQueryObjectHandle_t m_occ_handle; // the occlusion query handle
|
|
int m_frame_queued; // when this query was last queued
|
|
int m_npixels; // # of pixels this histogram represents
|
|
int m_npixels_in_range;
|
|
float m_min_lum, m_max_lum; // the luminance range this entry was queried with
|
|
float m_minx,m_miny,m_maxx,m_maxy; // range is 0..1 in fractions of the screen
|
|
|
|
bool ContainsValidData( void )
|
|
{
|
|
return ( m_state==HESTATE_QUERY_DONE ) || ( m_state==HESTATE_QUERY_IN_FLIGHT );
|
|
}
|
|
|
|
void IssueQuery( int frm_num )
|
|
{
|
|
if ( ! m_occ_handle)
|
|
m_occ_handle=materials->CreateOcclusionQueryObject();
|
|
int xl,yl,dest_width,dest_height;
|
|
materials->GetViewport( xl,yl,dest_width,dest_height);
|
|
// first, set stencil bits where the colors match
|
|
IMaterial *test_mat=materials->FindMaterial(
|
|
"dev/lumcompare",
|
|
TEXTURE_GROUP_OTHER,true);
|
|
IMaterialVar *pMinVar = test_mat->FindVar( "$C0_X", NULL );
|
|
pMinVar->SetFloatValue( m_min_lum);
|
|
IMaterialVar *pMaxVar = test_mat->FindVar( "$C0_Y", NULL );
|
|
if (m_max_lum==1.0)
|
|
pMaxVar->SetFloatValue( 10000.0); // count all pixels >1.0 as 1.0
|
|
else
|
|
pMaxVar->SetFloatValue( m_max_lum);
|
|
int scrx_min=FLerp(xl,xl+dest_width-1,0,1,m_minx);
|
|
int scrx_max=FLerp(xl,xl+dest_width-1,0,1,m_maxx);
|
|
int scry_min=FLerp(yl,yl+dest_height-1,0,1,m_miny);
|
|
int scry_max=FLerp(yl,yl+dest_height-1,0,1,m_maxy);
|
|
|
|
float exposure_width_scale, exposure_height_scale;
|
|
|
|
// now, shrink region of interest if the flashlight is on
|
|
C_BasePlayer *pLocal = C_BasePlayer::GetLocalPlayer();
|
|
if ( pLocal->IsEffectActive( EF_DIMLIGHT ) ) // check flashlight
|
|
{
|
|
exposure_width_scale=
|
|
(0.5*(1.0-mat_exposure_center_region_x_flashlight.GetFloat()));
|
|
exposure_height_scale=
|
|
(0.5*(1.0-mat_exposure_center_region_y_flashlight.GetFloat()));
|
|
}
|
|
else
|
|
{
|
|
exposure_width_scale=
|
|
(0.5*(1.0-mat_exposure_center_region_x.GetFloat()));
|
|
exposure_height_scale=
|
|
(0.5*(1.0-mat_exposure_center_region_y.GetFloat()));
|
|
}
|
|
int skip_edgex=(1+scrx_max-scrx_min)*exposure_width_scale;
|
|
int skip_edgey=(1+scry_max-scry_min)*exposure_height_scale;
|
|
|
|
// now, do luminance compare
|
|
float tscale=1.0;
|
|
if (g_pMaterialSystemHardwareConfig->GetHDRType()==HDR_TYPE_FLOAT)
|
|
{
|
|
tscale=materials->GetToneMappingScaleLinear().x;
|
|
}
|
|
IMaterialVar *use_t_scale = test_mat->FindVar( "$C0_Z", NULL );
|
|
use_t_scale->SetFloatValue( tscale);
|
|
|
|
|
|
m_npixels=(1+scrx_max-scrx_min)*(1+scry_max-scry_min);
|
|
if (mat_tonemapping_occlusion_use_stencil.GetInt())
|
|
{
|
|
materials->SetStencilWriteMask(1);
|
|
materials->ClearStencilBufferRectangle(scrx_min,scry_min,scrx_max,scry_max,0);
|
|
materials->SetStencilEnable(true);
|
|
materials->SetStencilPassOperation(STENCILOPERATION_REPLACE);
|
|
materials->SetStencilCompareFunction(STENCILCOMPARISONFUNCTION_ALWAYS);
|
|
materials->SetStencilFailOperation(STENCILOPERATION_KEEP);
|
|
materials->SetStencilZFailOperation(STENCILOPERATION_KEEP);
|
|
materials->SetStencilReferenceValue(1);
|
|
}
|
|
else
|
|
materials->BeginOcclusionQueryDrawing(m_occ_handle);
|
|
scrx_min+=skip_edgex;
|
|
scry_min+=skip_edgey;
|
|
scrx_max-=skip_edgex;
|
|
scry_max-=skip_edgey;
|
|
materials->DrawScreenSpaceRectangle(test_mat,
|
|
scrx_min,scry_min,
|
|
1+scrx_max-scrx_min,
|
|
1+scry_max-scry_min,
|
|
scrx_min,scry_min,
|
|
scrx_max, scry_max,
|
|
dest_width,dest_height);
|
|
|
|
if (mat_tonemapping_occlusion_use_stencil.GetInt())
|
|
{
|
|
// now, start counting how many pixels had their stencil bit set via an occlusion query
|
|
materials->BeginOcclusionQueryDrawing(m_occ_handle);
|
|
// now, issue an occlusion query using stencil as the mask
|
|
materials->SetStencilEnable(true);
|
|
materials->SetStencilTestMask(1);
|
|
materials->SetStencilPassOperation(STENCILOPERATION_KEEP);
|
|
materials->SetStencilCompareFunction(STENCILCOMPARISONFUNCTION_EQUAL);
|
|
materials->SetStencilFailOperation(STENCILOPERATION_KEEP);
|
|
materials->SetStencilZFailOperation(STENCILOPERATION_KEEP);
|
|
materials->SetStencilReferenceValue(1);
|
|
IMaterial *stest_mat=materials->FindMaterial(
|
|
"dev/no_pixel_write",
|
|
TEXTURE_GROUP_OTHER,true);
|
|
materials->DrawScreenSpaceRectangle(stest_mat,
|
|
scrx_min,scry_min,
|
|
1+scrx_max-scrx_min,
|
|
1+scry_max-scry_min,
|
|
scrx_min,scry_min,
|
|
scrx_max, scry_max,
|
|
dest_width,dest_height);
|
|
materials->SetStencilEnable(false);
|
|
}
|
|
materials->EndOcclusionQueryDrawing(m_occ_handle);
|
|
if (m_state==HESTATE_INITIAL)
|
|
m_state=HESTATE_FIRST_QUERY_IN_FLIGHT;
|
|
else
|
|
m_state=HESTATE_QUERY_IN_FLIGHT;
|
|
m_frame_queued=frm_num;
|
|
}
|
|
|
|
};
|
|
|
|
|
|
#define HISTOGRAM_BAR_SIZE 200
|
|
|
|
class CLuminanceHistogramSystem
|
|
{
|
|
CHistogram_entry_t CurHistogram[N_LUMINANCE_RANGES];
|
|
int cur_query_frame;
|
|
public:
|
|
float GetAverageLuminance( void )
|
|
{
|
|
float total=0;
|
|
int total_pixels=0;
|
|
float scale_value=1.0;
|
|
if (CurHistogram[N_LUMINANCE_RANGES-1].ContainsValidData())
|
|
{
|
|
scale_value=CurHistogram[N_LUMINANCE_RANGES-1].m_npixels*
|
|
(1.0/CurHistogram[N_LUMINANCE_RANGES-1].m_npixels_in_range);
|
|
if (mat_debug_autoexposure.GetInt())
|
|
Warning("scale value=%f\n",scale_value);
|
|
}
|
|
else
|
|
return 0.5;
|
|
|
|
for(int i=0;i<NELEMS( CurHistogram )-1;i++)
|
|
{
|
|
if ( CurHistogram[i].ContainsValidData() )
|
|
{
|
|
total+=scale_value*CurHistogram[i].m_npixels_in_range*
|
|
AVG( CurHistogram[i].m_min_lum,CurHistogram[i].m_max_lum );
|
|
total_pixels+=CurHistogram[i].m_npixels;
|
|
}
|
|
else
|
|
return 0.5; // always return 0.5 until we've queried a whole frame
|
|
}
|
|
if (total_pixels > 0)
|
|
return total*(1.0/total_pixels);
|
|
else
|
|
return 0.5;
|
|
}
|
|
|
|
void Update(void)
|
|
{
|
|
// find which histogram entries should have something done this frame
|
|
int n_queries_issued_this_frame=0;
|
|
cur_query_frame++;
|
|
|
|
for(int i=0;i<NELEMS( CurHistogram );i++)
|
|
{
|
|
switch( CurHistogram[i].m_state )
|
|
{
|
|
case HESTATE_INITIAL:
|
|
if ( n_queries_issued_this_frame<MAX_QUERIES_PER_FRAME )
|
|
{
|
|
CurHistogram[i].IssueQuery(cur_query_frame);
|
|
n_queries_issued_this_frame++;
|
|
}
|
|
break;
|
|
|
|
case HESTATE_FIRST_QUERY_IN_FLIGHT:
|
|
case HESTATE_QUERY_IN_FLIGHT:
|
|
if ( cur_query_frame>CurHistogram[i].m_frame_queued+2 )
|
|
{
|
|
int np= materials->OcclusionQuery_GetNumPixelsRendered(
|
|
CurHistogram[i].m_occ_handle);
|
|
if (np!=-1) // -1=query not finished. wait until
|
|
// next time
|
|
{
|
|
CurHistogram[i].m_npixels_in_range=np;
|
|
// if (mat_debug_autoexposure.GetInt())
|
|
// Warning("min=%f max=%f np = %d\n",CurHistogram[i].m_min_lum,CurHistogram[i].m_max_lum,np);
|
|
CurHistogram[i].m_state=HESTATE_QUERY_DONE;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
// now, issue queries for the oldest finished queries we have
|
|
while( n_queries_issued_this_frame<MAX_QUERIES_PER_FRAME )
|
|
{
|
|
int oldest_so_far=-1;
|
|
for(int i=0;i<NELEMS( CurHistogram );i++)
|
|
if ( (CurHistogram[i].m_state==HESTATE_QUERY_DONE ) &&
|
|
( (oldest_so_far==-1) ||
|
|
(CurHistogram[i].m_frame_queued<
|
|
CurHistogram[oldest_so_far].m_frame_queued) ) )
|
|
oldest_so_far=i;
|
|
if ( oldest_so_far==-1 ) // nothing to do
|
|
break;
|
|
CurHistogram[oldest_so_far].IssueQuery(cur_query_frame);
|
|
n_queries_issued_this_frame++;
|
|
}
|
|
}
|
|
|
|
void DisplayHistogram(void)
|
|
{
|
|
int xp=10;
|
|
for(int l=0;l<N_LUMINANCE_RANGES-1;l++)
|
|
{
|
|
int np=0;
|
|
CHistogram_entry_t &e=CurHistogram[l];
|
|
if (e.ContainsValidData())
|
|
np+=e.m_npixels_in_range;
|
|
int width=500*(e.m_max_lum-e.m_min_lum);
|
|
if (np)
|
|
{
|
|
int height=max(1,min(HISTOGRAM_BAR_SIZE,np/6000));
|
|
materials->Viewport(xp,HISTOGRAM_BAR_SIZE-height,width,height);
|
|
materials->ClearColor3ub(255,0,0);
|
|
materials->ClearBuffers(true,true);
|
|
}
|
|
xp+=width+2;
|
|
}
|
|
}
|
|
|
|
CLuminanceHistogramSystem(void)
|
|
{
|
|
cur_query_frame=0;
|
|
for(int bucket=0;bucket<N_LUMINANCE_RANGES;bucket++)
|
|
{
|
|
int idx=bucket;
|
|
CHistogram_entry_t &e=CurHistogram[idx];
|
|
e.m_state=HESTATE_INITIAL;
|
|
e.m_minx=0;
|
|
e.m_maxx=1;
|
|
e.m_miny=0;
|
|
e.m_maxy=1;
|
|
if (bucket!=N_LUMINANCE_RANGES-1) // last bucket is special
|
|
{
|
|
// use a logarithmic ramp for high range in the low range
|
|
e.m_min_lum=-.01+exp(FLerp(log(.01),log(.01+1),0,N_LUMINANCE_RANGES-1,bucket));
|
|
e.m_max_lum=-.01+exp(FLerp(log(.01),log(.01+1),0,N_LUMINANCE_RANGES-1,bucket+1));
|
|
}
|
|
else
|
|
{
|
|
// the last bucket is used as a test to determine the return range for occlusion
|
|
// queries to use as a scale factor. some boards (nvidia) have their occlusion
|
|
// query return values larger when using AA.
|
|
e.m_min_lum=0;
|
|
e.m_max_lum=100000.0;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
static CLuminanceHistogramSystem g_HDR_HistogramSystem;
|
|
|
|
static float s_MovingAverageToneMapScale[10];
|
|
static int s_nInAverage;
|
|
|
|
void ResetToneMapping(float value)
|
|
{
|
|
s_nInAverage=0;
|
|
materials->ResetToneMappingScale(value);
|
|
}
|
|
|
|
static void SetToneMapScale(float newvalue, float minvalue, float maxvalue)
|
|
{
|
|
mat_hdr_tonemapscale.SetValue(newvalue);
|
|
#if 1
|
|
if (s_nInAverage<NELEMS(s_MovingAverageToneMapScale))
|
|
s_MovingAverageToneMapScale[s_nInAverage++]=newvalue;
|
|
else
|
|
{
|
|
// scroll, losing oldest
|
|
for(int i=0;i<NELEMS(s_MovingAverageToneMapScale)-1;i++)
|
|
s_MovingAverageToneMapScale[i]=s_MovingAverageToneMapScale[i+1];
|
|
s_MovingAverageToneMapScale[NELEMS(s_MovingAverageToneMapScale)-1]=newvalue;
|
|
}
|
|
// now, use the average of the last tonemap calculations as our goal scale
|
|
if (s_nInAverage==NELEMS(s_MovingAverageToneMapScale)) // got full buffer yet?
|
|
{
|
|
float avg=0.;
|
|
float sumweights=0;
|
|
int sample_pt=NELEMS(s_MovingAverageToneMapScale)/2;
|
|
for(int i=0;i<NELEMS(s_MovingAverageToneMapScale);i++)
|
|
{
|
|
float weight=abs(i-sample_pt)*(1.0/(NELEMS(s_MovingAverageToneMapScale)/2));
|
|
sumweights+=weight;
|
|
avg+=weight*s_MovingAverageToneMapScale[i];
|
|
}
|
|
avg*=(1.0/sumweights);
|
|
avg=min(maxvalue,max(minvalue,avg));
|
|
mat_hdr_tonemapscale.SetValue(avg);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void DoPostProcessingEffects( int x, int y, int w, int h )
|
|
{
|
|
bool bloom_enabled = (mat_hdr_level.GetInt() >= 1);
|
|
if ( !engine->MapHasHDRLighting() )
|
|
bloom_enabled = false;
|
|
if ( mat_force_bloom.GetInt() )
|
|
bloom_enabled = true;
|
|
if ( mat_disable_bloom.GetInt() )
|
|
bloom_enabled = false;
|
|
if ( building_cubemaps.GetBool() )
|
|
bloom_enabled = false;
|
|
|
|
HDRType_t hdrType = g_pMaterialSystemHardwareConfig->GetHDRType();
|
|
|
|
float bloom_coeff=0.0;
|
|
if (bloom_enabled)
|
|
{
|
|
static float currentBloomAmount = 1.0f;
|
|
float rate = mat_bloomamount_rate.GetFloat();
|
|
|
|
// Use the appropriate bloom scale settings. Mapmakers's overrides the convar settings.
|
|
float flCurrentBloomScale = 1.0f;
|
|
if ( g_bUseCustomBloomScale )
|
|
{
|
|
flCurrentBloomScale = g_flCustomBloomScale;
|
|
}
|
|
else
|
|
{
|
|
flCurrentBloomScale = mat_bloomscale.GetFloat();
|
|
}
|
|
|
|
currentBloomAmount = flCurrentBloomScale * rate + ( 1.0f - rate ) * currentBloomAmount;
|
|
bloom_coeff=currentBloomAmount;
|
|
}
|
|
|
|
if ( hdrType == HDR_TYPE_NONE )
|
|
bloom_coeff *= mat_non_hdr_bloom_scalefactor.GetFloat();
|
|
|
|
// Use the appropriate autoexposure min / max settings.
|
|
// Mapmaker's overrides the convar settings.
|
|
float flAutoExposureMin;
|
|
float flAutoExposureMax;
|
|
if ( g_bUseCustomAutoExposureMin )
|
|
{
|
|
flAutoExposureMin = g_flCustomAutoExposureMin;
|
|
}
|
|
else
|
|
{
|
|
flAutoExposureMin = mat_autoexposure_min.GetFloat();
|
|
}
|
|
if ( g_bUseCustomAutoExposureMax )
|
|
{
|
|
flAutoExposureMax = g_flCustomAutoExposureMax;
|
|
}
|
|
else
|
|
{
|
|
flAutoExposureMax = mat_autoexposure_max.GetFloat();
|
|
}
|
|
|
|
switch( hdrType )
|
|
{
|
|
case HDR_TYPE_NONE:
|
|
case HDR_TYPE_INTEGER:
|
|
{
|
|
if (mat_debug_bloom.GetInt()==1)
|
|
{
|
|
static int wx=0;
|
|
wx=(wx+1) & 63;
|
|
materials->SetRenderTarget(NULL);
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
materials->ClearColor3ub(0,0,0);
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( wx+100, 100, 20,20 );
|
|
materials->ClearColor3ub(255,255,255);
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( 20,20,20,20 );
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( 950,20,20,20 );
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( 950,950,20,20 );
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( 20,950,20,20 );
|
|
materials->ClearBuffers(true,true);
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
}
|
|
bool did_update=false;
|
|
if ( ( hdrType != HDR_TYPE_NONE && mat_dynamic_tonemapping.GetInt() ) || mat_show_histogram.GetInt())
|
|
{
|
|
did_update=true;
|
|
// if (mat_debug_autoexposure.GetInt())
|
|
// {
|
|
// materials->ClearColor3ub(128,0,0);
|
|
// materials->ClearBuffers(true,true);
|
|
// }
|
|
Rect_t actualRect;
|
|
UpdateScreenEffectTexture( 0, x, y, w, h, false, &actualRect );
|
|
// UpdateScreenEffectTexture( 0, x, y, w, h, true );
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
g_HDR_HistogramSystem.Update();
|
|
float avg_lum=g_HDR_HistogramSystem.GetAverageLuminance();
|
|
float last_scale=materials->GetToneMappingScaleLinear().x;
|
|
float actual_unscaled_luminance=avg_lum*(1.0/last_scale);
|
|
actual_unscaled_luminance=max(0.000000001,avg_lum);
|
|
float target_scale=0.005/actual_unscaled_luminance;
|
|
float scalevalue=max(flAutoExposureMin,
|
|
min(flAutoExposureMax,target_scale));
|
|
if (mat_debug_autoexposure.GetInt())
|
|
Warning("avg_lum=%f ra=%f target=%f adj target=%f\n",
|
|
avg_lum,actual_unscaled_luminance,target_scale,scalevalue);
|
|
if (mat_dynamic_tonemapping.GetInt())
|
|
SetToneMapScale(scalevalue,flAutoExposureMin,flAutoExposureMax);
|
|
}
|
|
// bloom
|
|
if ( bloom_enabled && ( bloom_coeff > 0.0 ) && ( engine->GetDXSupportLevel() >= 80 ) )
|
|
{
|
|
materials->SetRenderTarget(NULL);
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
if ( !did_update )
|
|
{
|
|
UpdateScreenEffectTexture( 0, x, y, w, h, true );
|
|
did_update=true;
|
|
}
|
|
|
|
IMaterial *downsample_mat = materials->FindMaterial( "dev/downsample_non_hdr", TEXTURE_GROUP_OTHER, true);
|
|
IMaterial *xblur_mat = materials->FindMaterial( "dev/blurfilterx_nohdr", TEXTURE_GROUP_OTHER, true );
|
|
IMaterial *yblur_mat = materials->FindMaterial( "dev/blurfiltery_and_add_nohdr", TEXTURE_GROUP_OTHER, true );
|
|
ITexture *dest_rt0 = materials->FindTexture( "_rt_SmallFB0", TEXTURE_GROUP_RENDER_TARGET );
|
|
ITexture *dest_rt1 = materials->FindTexture( "_rt_SmallFB1", TEXTURE_GROUP_RENDER_TARGET );
|
|
|
|
// downsample fb to rt0
|
|
SetRenderTargetAndViewPort( dest_rt0 );
|
|
// note the -2's below. Thats because we are downsampling on each axis and the shader
|
|
// accesses pixels on both sides of the source coord
|
|
materials->DrawScreenSpaceRectangle( downsample_mat, 0, 0, dest_width/4, dest_height/4,
|
|
0, 0, dest_width-2, dest_height-2,
|
|
dest_width, dest_height );
|
|
// guassian blur x rt0 to rt1
|
|
SetRenderTargetAndViewPort( dest_rt1 );
|
|
materials->DrawScreenSpaceRectangle( xblur_mat, 0, 0, dest_width/4, dest_height/4,
|
|
0, 0, dest_width/4-1, dest_height/4-1,
|
|
dest_width/4, dest_height/4 );
|
|
// gaussian blur y and add rt1 to frame buffer
|
|
materials->Viewport( 0, 0, dest_width, dest_height );
|
|
materials->SetRenderTarget( NULL );
|
|
IMaterialVar *pBloomAmountVar = yblur_mat->FindVar( "$bloomamount", NULL );
|
|
pBloomAmountVar->SetFloatValue( bloom_coeff );
|
|
int bloomadd_x=0;
|
|
int bloomadd_y=0;
|
|
int bloomadd_width=dest_width;
|
|
int bloomadd_height=dest_height;
|
|
float bloomsrc_x=0;
|
|
float bloomsrc_y=-.5; // compensate for phase shift due to mismatched input/output sizes
|
|
float bloomsrc_maxx=dest_width/4-1;
|
|
float bloomsrc_maxy=dest_height/4-1;
|
|
if ( IsSplitScreenHDR() )
|
|
{
|
|
bloomadd_x+=dest_width/2;
|
|
bloomadd_width-=dest_width/2;
|
|
bloomsrc_x+=dest_width/8;
|
|
}
|
|
materials->DrawScreenSpaceRectangle(yblur_mat,bloomadd_x,bloomadd_y,
|
|
bloomadd_width,bloomadd_height,
|
|
bloomsrc_x,bloomsrc_y,
|
|
bloomsrc_maxx,bloomsrc_maxy,
|
|
dest_rt1->GetActualWidth(),
|
|
dest_rt1->GetActualHeight());
|
|
}
|
|
|
|
if ( mat_show_histogram.GetInt() && ( engine->GetDXSupportLevel() >= 90 ) )
|
|
{
|
|
g_HDR_HistogramSystem.DisplayHistogram();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case HDR_TYPE_FLOAT:
|
|
{
|
|
int dest_width,dest_height;
|
|
materials->GetRenderTargetDimensions( dest_width, dest_height );
|
|
if (mat_dynamic_tonemapping.GetInt() || mat_show_histogram.GetInt())
|
|
{
|
|
g_HDR_HistogramSystem.Update();
|
|
// Warning("avg_lum=%f\n",g_HDR_HistogramSystem.GetAverageLuminance());
|
|
if (mat_dynamic_tonemapping.GetInt())
|
|
{
|
|
float avg_lum=max(0.0001,g_HDR_HistogramSystem.GetAverageLuminance());
|
|
float scalevalue=max(flAutoExposureMin,
|
|
min(flAutoExposureMax,0.18/avg_lum));
|
|
mat_hdr_tonemapscale.SetValue(scalevalue);
|
|
}
|
|
}
|
|
IMaterial *pBloomMaterial;
|
|
pBloomMaterial = materials->FindMaterial( "dev/floattoscreen_combine", "" );
|
|
IMaterialVar *pBloomAmountVar = pBloomMaterial->FindVar( "$bloomamount", NULL );
|
|
pBloomAmountVar->SetFloatValue( bloom_coeff);
|
|
if ( IsSplitScreenHDR() )
|
|
{
|
|
// do split screen
|
|
ClipBox mycb;
|
|
mycb.m_minx=mycb.m_miny=0;
|
|
mycb.m_maxx=dest_width/2;
|
|
mycb.m_maxy=dest_height-1;
|
|
ApplyPostProcessingPasses(HDRSimulate_NonHDR,&mycb);
|
|
mycb.m_minx=mycb.m_maxx;
|
|
mycb.m_maxx=dest_width-1;
|
|
ApplyPostProcessingPasses(HDRFinal_Float,&mycb);
|
|
}
|
|
else
|
|
{
|
|
if( bloom_enabled)
|
|
ApplyPostProcessingPasses(HDRFinal_Float);
|
|
else
|
|
ApplyPostProcessingPasses(HDRFinal_Float_NoBloom);
|
|
}
|
|
materials->SetRenderTarget(NULL);
|
|
if ( mat_show_histogram.GetInt() && (engine->GetDXSupportLevel()>=90))
|
|
g_HDR_HistogramSystem.DisplayHistogram();
|
|
if (mat_dynamic_tonemapping.GetInt())
|
|
{
|
|
float avg_lum=max(0.0001,g_HDR_HistogramSystem.GetAverageLuminance());
|
|
float scalevalue=max(flAutoExposureMin,
|
|
min(flAutoExposureMax,0.023/avg_lum));
|
|
SetToneMapScale(scalevalue,flAutoExposureMin,flAutoExposureMax);
|
|
}
|
|
materials->SetRenderTarget( NULL );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#else
|
|
void ResetToneMapping(float value)
|
|
{
|
|
}
|
|
#endif // !_XBOX
|
|
|
|
#ifdef _XBOX
|
|
|
|
#define FULLSCREEN_WIDTH 640
|
|
#define FULLSCREEN_HEIGHT 480
|
|
#define DOWNSAMPLE_WIDTH (FULLSCREEN_WIDTH>>1)
|
|
#define DOWNSAMPLE_HEIGHT (FULLSCREEN_HEIGHT>>1)
|
|
|
|
void CViewRender::DoScreenSpaceBloom()
|
|
{
|
|
int rt_width, rt_height;
|
|
|
|
if ( !mat_bloomscale.GetFloat() || !m_BloomDownsample.IsValid() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
// rt_waterReflection = DownSample( backBuffer )
|
|
ITexture *dest_rt = GetWaterReflectionTexture();
|
|
materials->PushRenderTargetAndViewport( dest_rt );
|
|
materials->GetRenderTargetDimensions( rt_width, rt_height );
|
|
Assert( rt_width >= DOWNSAMPLE_WIDTH && rt_height >= DOWNSAMPLE_HEIGHT );
|
|
materials->DrawScreenSpaceRectangle(
|
|
m_BloomDownsample, 0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT,
|
|
0, 0, FULLSCREEN_WIDTH, FULLSCREEN_HEIGHT, 1, 1 );
|
|
|
|
// rt_waterRefraction = BlurX( rt_waterReflection )
|
|
dest_rt = GetWaterRefractionTexture();
|
|
materials->SetRenderTarget( dest_rt );
|
|
materials->DrawScreenSpaceRectangle(
|
|
m_BloomBlurX, 0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT,
|
|
0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT, rt_width, rt_height );
|
|
|
|
// rt_waterReflection = BlurY( rt_waterRefraction )
|
|
dest_rt = GetWaterReflectionTexture();
|
|
materials->SetRenderTarget( dest_rt );
|
|
materials->DrawScreenSpaceRectangle(
|
|
m_BloomBlurY, 0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT,
|
|
0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT, rt_width, rt_height );
|
|
|
|
// restore screen
|
|
materials->PopRenderTargetAndViewport();
|
|
|
|
// FrameBuffer += rt_waterReflection
|
|
materials->DrawScreenSpaceRectangle(
|
|
m_BloomAdd, 0, 0, FULLSCREEN_WIDTH, FULLSCREEN_HEIGHT,
|
|
0, 0, DOWNSAMPLE_WIDTH, DOWNSAMPLE_HEIGHT, rt_width, rt_height );
|
|
}
|
|
#endif
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders voice feedback and other sprites attached to players
|
|
// Input : none
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::RenderPlayerSprites()
|
|
{
|
|
#ifndef _XBOX
|
|
GetClientVoiceMgr()->DrawHeadLabels();
|
|
#endif
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Sets up, cleans up the main 3D view
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::SetupMain3DView( const CViewSetup &view, int &nClearFlags )
|
|
{
|
|
// FIXME: I really want these fields removed from CViewSetup
|
|
// and passed in as independent flags
|
|
// Clear the color here if requested.
|
|
nClearFlags &= ~VIEW_CLEAR_DEPTH;
|
|
if ( nClearFlags & VIEW_CLEAR_COLOR )
|
|
{
|
|
nClearFlags |= VIEW_CLEAR_DEPTH;
|
|
}
|
|
|
|
// If we are using HDR, we render to the HDR full frame buffer texture
|
|
// instead of whatever was previously the render target
|
|
if( g_pMaterialSystemHardwareConfig->GetHDRType() == HDR_TYPE_FLOAT )
|
|
{
|
|
render->Push3DView( view, nClearFlags, true, GetFullFrameFrameBufferTexture( 0 ), m_Frustum );
|
|
}
|
|
else
|
|
{
|
|
render->Push3DView( view, nClearFlags, false, NULL, m_Frustum );
|
|
}
|
|
|
|
// If we didn't clear the depth here, we'll need to clear it later
|
|
nClearFlags ^= VIEW_CLEAR_DEPTH;
|
|
if ( nClearFlags & VIEW_CLEAR_COLOR )
|
|
{
|
|
// If we cleared the color here, we don't need to clear it later
|
|
nClearFlags &= ~( VIEW_CLEAR_COLOR | VIEW_CLEAR_FULL_TARGET );
|
|
}
|
|
}
|
|
|
|
void CViewRender::CleanupMain3DView( const CViewSetup &view )
|
|
{
|
|
render->PopView( m_Frustum );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Queues up an overlay rendering
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::QueueOverlayRenderView( const CViewSetup &view, int nClearFlags, int whatToDraw )
|
|
{
|
|
#ifndef _XBOX
|
|
// Can't have 2 in a single scene
|
|
Assert( !m_bDrawOverlay );
|
|
m_bDrawOverlay = true;
|
|
m_OverlayViewSetup = view;
|
|
m_OverlayClearFlags = nClearFlags;
|
|
m_OverlayDrawFlags = whatToDraw;
|
|
#endif
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: This renders the entire 3D view and the in-game hud/viewmodel
|
|
// Input : &view -
|
|
// whatToDraw -
|
|
//-----------------------------------------------------------------------------
|
|
// This renders the entire 3D view.
|
|
void CViewRender::RenderViewEx( const CViewSetup &view, int nClearFlags, int whatToDraw )
|
|
{
|
|
m_CurrentView = view;
|
|
|
|
VPROF( "CViewRender::RenderViewEx" );
|
|
|
|
ITexture *saveRenderTarget = materials->GetRenderTarget();
|
|
|
|
g_pClientShadowMgr->AdvanceFrame();
|
|
|
|
#ifdef USE_MONITORS
|
|
if ( cl_drawmonitors.GetBool() && g_pMaterialSystemHardwareConfig->GetDXSupportLevel() >= 70 )
|
|
{
|
|
DrawMonitors( view );
|
|
}
|
|
#endif
|
|
|
|
g_bRenderingView = true;
|
|
|
|
// Must be first
|
|
render->SceneBegin();
|
|
materials->TurnOnToneMapping();
|
|
|
|
// clear happens here probably
|
|
SetupMain3DView( view, nClearFlags );
|
|
|
|
bool bDrew3dSkybox = false;
|
|
bool bSkyboxVisible = false;
|
|
|
|
// if the 3d skybox world is drawn, then don't draw the normal skybox
|
|
Draw3dSkyboxworld( view, nClearFlags, bDrew3dSkybox, bSkyboxVisible );
|
|
|
|
// Force it to clear the framebuffer if they're in solid space.
|
|
if ( ( nClearFlags & VIEW_CLEAR_COLOR ) == 0 )
|
|
{
|
|
if ( enginetrace->GetPointContents( view.origin ) == CONTENTS_SOLID )
|
|
{
|
|
nClearFlags |= VIEW_CLEAR_COLOR;
|
|
}
|
|
}
|
|
|
|
// Render world and all entities, particles, etc.
|
|
if( !g_pIntroData )
|
|
{
|
|
ViewDrawScene( bDrew3dSkybox, bSkyboxVisible, view, nClearFlags, VIEW_MAIN, whatToDraw & RENDERVIEW_DRAWVIEWMODEL );
|
|
}
|
|
else
|
|
{
|
|
ViewDrawScene_Intro( view, nClearFlags, *g_pIntroData );
|
|
}
|
|
|
|
// We can still use the 'current view' stuff set up in ViewDrawScene
|
|
s_bCanAccessCurrentView = true;
|
|
|
|
if ( !IsXbox() )
|
|
{
|
|
engine->DrawPortals();
|
|
}
|
|
|
|
DisableFog();
|
|
|
|
// Finish scene
|
|
render->SceneEnd();
|
|
|
|
// Draw lightsources if enabled
|
|
render->DrawLights();
|
|
|
|
RenderPlayerSprites();
|
|
|
|
// Now actually draw the viewmodel
|
|
DrawViewModels( view, whatToDraw & RENDERVIEW_DRAWVIEWMODEL );
|
|
|
|
PixelVisibility_EndScene();
|
|
|
|
// Draw fade over entire screen if needed
|
|
byte color[4];
|
|
bool blend;
|
|
vieweffects->GetFadeParams( view.context, &color[0], &color[1], &color[2], &color[3], &blend );
|
|
|
|
// Draw an overlay to make it even harder to see inside smoke particle systems.
|
|
DrawSmokeFogOverlay();
|
|
|
|
// Overlay screen fade on entire screen
|
|
IMaterial* pMaterial = blend ? m_ModulateSingleColor : m_TranslucentSingleColor;
|
|
render->ViewDrawFade( color, pMaterial );
|
|
PerformScreenOverlay( view.x, view.y, view.width, view.height );
|
|
|
|
// Prevent sound stutter if going slow
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
#ifndef _XBOX
|
|
if ( !building_cubemaps.GetBool() && view.m_bDoBloomAndToneMapping )
|
|
{
|
|
DoPostProcessingEffects( view.x, view.y, view.width, view.height );
|
|
}
|
|
#else
|
|
DoScreenSpaceBloom();
|
|
#endif
|
|
|
|
// And here are the screen-space effects
|
|
|
|
// Grab the pre-color corrected frame for editing purposes
|
|
engine->GrabPreColorCorrectedFrame( view.x, view.y, view.width, view.height );
|
|
|
|
PerformScreenSpaceEffects( view.x, view.y, view.width, view.height );
|
|
if ( !IsXbox() && g_pMaterialSystemHardwareConfig->GetHDRType() == HDR_TYPE_INTEGER )
|
|
{
|
|
materials->SetToneMappingScaleLinear(Vector(1,1,1));
|
|
}
|
|
|
|
CleanupMain3DView( view );
|
|
|
|
materials->SetRenderTarget( saveRenderTarget );
|
|
|
|
#ifndef _XBOX
|
|
// Draw the overlay
|
|
if ( m_bDrawOverlay )
|
|
{
|
|
// This allows us to be ok if there are nested overlay views
|
|
CViewSetup currentView = m_CurrentView;
|
|
CViewSetup tempView = m_OverlayViewSetup;
|
|
tempView.fov = ScaleFOVByWidthRatio( tempView.fov, tempView.m_flAspectRatio / ( 4.0f / 3.0f ) );
|
|
tempView.m_bDoBloomAndToneMapping = false; // FIXME: Hack to get Mark up and running
|
|
m_bDrawOverlay = false;
|
|
RenderViewEx( tempView, m_OverlayClearFlags, m_OverlayDrawFlags );
|
|
m_CurrentView = currentView;
|
|
}
|
|
#endif
|
|
|
|
// Draw the 2D graphics
|
|
render->Push2DView( view, 0, false, NULL, m_Frustum );
|
|
|
|
Render2DEffectsPreHUD( view );
|
|
|
|
if ( whatToDraw & RENDERVIEW_DRAWHUD )
|
|
{
|
|
VPROF_BUDGET( "VGui_DrawHud", VPROF_BUDGETGROUP_OTHER_VGUI );
|
|
// paint the vgui screen
|
|
VGui_PreRender();
|
|
|
|
// Make sure the client .dll root panel is at the proper point before doing the "SolveTraverse" calls
|
|
vgui::VPANEL root = enginevgui->GetPanel( PANEL_CLIENTDLL );
|
|
if ( root != 0 )
|
|
{
|
|
vgui::ipanel()->SetPos( root, view.x, view.y );
|
|
vgui::ipanel()->SetSize( root, view.width, view.height );
|
|
}
|
|
// Same for client .dll tools
|
|
root = enginevgui->GetPanel( PANEL_CLIENTDLL_TOOLS );
|
|
if ( root != 0 )
|
|
{
|
|
vgui::ipanel()->SetPos( root, view.x, view.y );
|
|
vgui::ipanel()->SetSize( root, view.width, view.height );
|
|
}
|
|
|
|
// The crosshair, etc. needs to get at the current setup stuff
|
|
AllowCurrentViewAccess( true );
|
|
|
|
// Draw the in-game stuff based on the actual viewport being used
|
|
render->VGui_Paint( PAINT_INGAMEPANELS );
|
|
|
|
AllowCurrentViewAccess( false );
|
|
|
|
VGui_PostRender();
|
|
|
|
g_pClientMode->PostRenderVGui();
|
|
materials->Flush();
|
|
}
|
|
|
|
Draw2DDebuggingInfo( view );
|
|
|
|
m_AnglesHistory[m_AnglesHistoryCounter] = view.angles;
|
|
m_AnglesHistoryCounter = (m_AnglesHistoryCounter+1) & ANGLESHISTORY_MASK;
|
|
|
|
// If the angles are moving fast enough, allow LOD transitions.
|
|
float angleMovementDelta = 0;
|
|
for(int i=0; i < ANGLESHISTORY_SIZE; i++)
|
|
{
|
|
angleMovementDelta += (m_AnglesHistory[(i+1) & ANGLESHISTORY_MASK] - m_AnglesHistory[i]).Length();
|
|
}
|
|
|
|
angleMovementDelta /= ANGLESHISTORY_SIZE;
|
|
if(angleMovementDelta > r_TransitionSensitivity.GetFloat())
|
|
{
|
|
r_DoCovertTransitions.SetValue(1);
|
|
}
|
|
else
|
|
{
|
|
r_DoCovertTransitions.SetValue(0);
|
|
}
|
|
|
|
Render2DEffectsPostHUD( view );
|
|
|
|
g_bRenderingView = false;
|
|
|
|
// We can no longer use the 'current view' stuff set up in ViewDrawScene
|
|
s_bCanAccessCurrentView = false;
|
|
|
|
// Next frame!
|
|
++m_FrameNumber;
|
|
|
|
if ( !IsXbox() )
|
|
{
|
|
GenerateOverdrawForTesting();
|
|
}
|
|
|
|
render->PopView( m_Frustum );
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders extra 2D effects in derived classes while the 2D view is on the stack
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::Render2DEffectsPreHUD( const CViewSetup &view )
|
|
{
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders extra 2D effects in derived classes while the 2D view is on the stack
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::Render2DEffectsPostHUD( const CViewSetup &view )
|
|
{
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: Renders entire view
|
|
// Input : &view -
|
|
// drawViewModel -
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::RenderView( const CViewSetup &view, int nClearFlags, bool bDrawViewModel )
|
|
{
|
|
int flags = RENDERVIEW_DRAWHUD;
|
|
if ( bDrawViewModel )
|
|
{
|
|
flags |= RENDERVIEW_DRAWVIEWMODEL;
|
|
}
|
|
RenderViewEx( view, nClearFlags, flags );
|
|
}
|
|
|
|
|
|
int CViewRender::GetDrawFlags()
|
|
{
|
|
return m_DrawFlags;
|
|
}
|
|
|
|
|
|
void ViewTransform( const Vector &worldSpace, Vector &viewSpace )
|
|
{
|
|
const VMatrix &viewMatrix = engine->WorldToViewMatrix();
|
|
Vector3DMultiplyPosition( viewMatrix, worldSpace, viewSpace );
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose: UNDONE: Clean this up some, handle off-screen vertices
|
|
// Input : *point -
|
|
// *screen -
|
|
// Output : int
|
|
//-----------------------------------------------------------------------------
|
|
int ScreenTransform( const Vector& point, Vector& screen )
|
|
{
|
|
// UNDONE: Clean this up some, handle off-screen vertices
|
|
float w;
|
|
const VMatrix &worldToScreen = engine->WorldToScreenMatrix();
|
|
|
|
screen.x = worldToScreen[0][0] * point[0] + worldToScreen[0][1] * point[1] + worldToScreen[0][2] * point[2] + worldToScreen[0][3];
|
|
screen.y = worldToScreen[1][0] * point[0] + worldToScreen[1][1] * point[1] + worldToScreen[1][2] * point[2] + worldToScreen[1][3];
|
|
// z = worldToScreen[2][0] * point[0] + worldToScreen[2][1] * point[1] + worldToScreen[2][2] * point[2] + worldToScreen[2][3];
|
|
w = worldToScreen[3][0] * point[0] + worldToScreen[3][1] * point[1] + worldToScreen[3][2] * point[2] + worldToScreen[3][3];
|
|
|
|
// Just so we have something valid here
|
|
screen.z = 0.0f;
|
|
|
|
bool behind;
|
|
if( w < 0.001f )
|
|
{
|
|
behind = true;
|
|
screen.x *= 100000;
|
|
screen.y *= 100000;
|
|
}
|
|
else
|
|
{
|
|
behind = false;
|
|
float invw = 1.0f / w;
|
|
screen.x *= invw;
|
|
screen.y *= invw;
|
|
}
|
|
|
|
return behind;
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
// NOTE: Below here is all of the stuff that needs to be done for water rendering
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Determines what kind of water we're going to use
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DetermineWaterRenderInfo( const VisibleFogVolumeInfo_t &fogVolumeInfo, CViewRender::WaterRenderInfo_t &info )
|
|
{
|
|
// By default, assume cheap water (even if there's no water in the scene!)
|
|
info.m_bCheapWater = true;
|
|
info.m_bRefract = false;
|
|
info.m_bReflect = false;
|
|
info.m_bReflectEntities = false;
|
|
info.m_bDrawWaterSurface = false;
|
|
info.m_bOpaqueWater = true;
|
|
|
|
IMaterial *pWaterMaterial = fogVolumeInfo.m_pFogVolumeMaterial;
|
|
if (( fogVolumeInfo.m_nVisibleFogVolume == -1 ) || !pWaterMaterial )
|
|
return;
|
|
|
|
// Use cheap water if mat_drawwater is set
|
|
info.m_bDrawWaterSurface = mat_drawwater.GetBool();
|
|
if ( !info.m_bDrawWaterSurface )
|
|
{
|
|
info.m_bOpaqueWater = false;
|
|
return;
|
|
}
|
|
|
|
// Determine if the water surface is opaque or not
|
|
info.m_bOpaqueWater = !pWaterMaterial->IsTranslucent();
|
|
|
|
// DX level 70 can't handle anything but cheap water
|
|
if (engine->GetDXSupportLevel() < 80)
|
|
return;
|
|
|
|
bool bForceCheap = false;
|
|
bool bForceExpensive = r_waterforceexpensive.GetBool();
|
|
|
|
// The material can override the default settings though
|
|
IMaterialVar *pForceCheapVar = pWaterMaterial->FindVar( "$forcecheap", NULL, false );
|
|
IMaterialVar *pForceExpensiveVar = pWaterMaterial->FindVar( "$forceexpensive", NULL, false );
|
|
if ( pForceCheapVar && pForceCheapVar->IsDefined() )
|
|
{
|
|
bForceCheap = ( pForceCheapVar->GetIntValue() != 0 );
|
|
if ( bForceCheap )
|
|
{
|
|
bForceExpensive = false;
|
|
}
|
|
}
|
|
if ( !bForceCheap && pForceExpensiveVar && pForceExpensiveVar->IsDefined() )
|
|
{
|
|
bForceExpensive = bForceExpensive || ( pForceExpensiveVar->GetIntValue() != 0 );
|
|
}
|
|
|
|
bool bDebugCheapWater = r_debugcheapwater.GetBool();
|
|
if( bDebugCheapWater )
|
|
{
|
|
Msg( "Water material: %s dist to water: %f\nforcecheap: %s forceexpensive: %s\n",
|
|
pWaterMaterial->GetName(), fogVolumeInfo.m_flDistanceToWater,
|
|
bForceCheap ? "true" : "false", bForceExpensive ? "true" : "false" );
|
|
}
|
|
|
|
// Unless expensive water is active, reflections are off.
|
|
bool bLocalReflection;
|
|
if( !bForceExpensive || !r_WaterDrawReflection.GetBool() )
|
|
{
|
|
bLocalReflection = false;
|
|
}
|
|
else
|
|
{
|
|
IMaterialVar *pReflectTextureVar = pWaterMaterial->FindVar( "$reflecttexture", NULL, false );
|
|
bLocalReflection = pReflectTextureVar && (pReflectTextureVar->GetType() == MATERIAL_VAR_TYPE_TEXTURE);
|
|
}
|
|
|
|
// FIXME: I disabled cheap water LOD when local specular is specified.
|
|
// There are very few places that appear to actually
|
|
// take advantage of it (places where water is in the PVS, but outside of LOD range).
|
|
// It was 2 hours before code lock, and I had the choice of either doubling fill-rate everywhere
|
|
// by making cheap water lod actually work (the water LOD wasn't actually rendering!!!)
|
|
// or to just always render the reflection + refraction if there's a local specular specified.
|
|
// Note that water LOD *does* work with refract-only water
|
|
|
|
// Check if the water is out of the cheap water LOD range; if so, use cheap water
|
|
if ( ( (fogVolumeInfo.m_flDistanceToWater >= m_flCheapWaterEndDistance) && !bLocalReflection ) || bForceCheap )
|
|
return;
|
|
|
|
// Get the material that is for the water surface that is visible and check to see
|
|
// what render targets need to be rendered, if any.
|
|
if ( !r_WaterDrawRefraction.GetBool() )
|
|
{
|
|
info.m_bRefract = false;
|
|
}
|
|
else
|
|
{
|
|
IMaterialVar *pRefractTextureVar = pWaterMaterial->FindVar( "$refracttexture", NULL, false );
|
|
info.m_bRefract = pRefractTextureVar && (pRefractTextureVar->GetType() == MATERIAL_VAR_TYPE_TEXTURE);
|
|
|
|
// Refractive water can be seen through
|
|
if ( info.m_bRefract )
|
|
{
|
|
info.m_bOpaqueWater = false;
|
|
}
|
|
}
|
|
|
|
#ifndef _XBOX
|
|
info.m_bReflect = bLocalReflection;
|
|
if ( info.m_bReflect )
|
|
{
|
|
if( r_waterforcereflectentities.GetBool() )
|
|
{
|
|
info.m_bReflectEntities = true;
|
|
}
|
|
else
|
|
{
|
|
IMaterialVar *pReflectEntitiesVar = pWaterMaterial->FindVar( "$reflectentities", NULL, false );
|
|
info.m_bReflectEntities = pReflectEntitiesVar && (pReflectEntitiesVar->GetIntValue() != 0);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
info.m_bCheapWater = !info.m_bReflect && !info.m_bRefract;
|
|
|
|
if( bDebugCheapWater )
|
|
{
|
|
Warning( "refract: %s reflect: %s\n", info.m_bRefract ? "true" : "false", info.m_bReflect ? "true" : "false" );
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the world and all entities
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::WaterDrawWorldAndEntities( bool bDrawSkybox, const CViewSetup &viewIn, int nClearFlags )
|
|
{
|
|
MDLCACHE_CRITICAL_SECTION();
|
|
|
|
VisibleFogVolumeInfo_t fogVolumeInfo;
|
|
render->GetVisibleFogVolume( viewIn.origin, &fogVolumeInfo );
|
|
|
|
WaterRenderInfo_t info;
|
|
DetermineWaterRenderInfo( fogVolumeInfo, info );
|
|
|
|
if ( info.m_bCheapWater )
|
|
{
|
|
ViewDrawScene_NoWater( bDrawSkybox, viewIn, nClearFlags, fogVolumeInfo, info );
|
|
return;
|
|
}
|
|
|
|
// Blat out the visible fog leaf if we're not going to use it
|
|
if ( !r_ForceWaterLeaf.GetBool() )
|
|
{
|
|
fogVolumeInfo.m_nVisibleFogVolumeLeaf = -1;
|
|
}
|
|
|
|
// We can see water of some sort
|
|
if ( !fogVolumeInfo.m_bEyeInFogVolume )
|
|
{
|
|
ViewDrawScene_EyeAboveWater( bDrawSkybox, viewIn, nClearFlags, fogVolumeInfo, info );
|
|
}
|
|
else
|
|
{
|
|
ViewDrawScene_EyeUnderWater( bDrawSkybox, viewIn, nClearFlags, fogVolumeInfo, info );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Pushes a water render target
|
|
//-----------------------------------------------------------------------------
|
|
static Vector SavedLinearLightMapScale(-1,-1,-1); // x<0 = no saved scale
|
|
|
|
static void SetLightmapScaleForWater(void)
|
|
{
|
|
if (g_pMaterialSystemHardwareConfig->GetHDRType()==HDR_TYPE_INTEGER)
|
|
{
|
|
SavedLinearLightMapScale=materials->GetToneMappingScaleLinear();
|
|
Vector t25=SavedLinearLightMapScale;
|
|
t25*=0.25;
|
|
materials->SetToneMappingScaleLinear(t25);
|
|
}
|
|
}
|
|
void CViewRender::PushWaterRenderTarget( CViewSetup &view, int nClearFlags, float waterHeight, int flags )
|
|
{
|
|
float spread = 2.0f;
|
|
float origWaterHeight = waterHeight;
|
|
if( flags & DF_FUDGE_UP )
|
|
{
|
|
waterHeight += spread;
|
|
}
|
|
else
|
|
{
|
|
waterHeight -= spread;
|
|
}
|
|
|
|
MaterialHeightClipMode_t clipMode = MATERIAL_HEIGHTCLIPMODE_DISABLE;
|
|
if ( ( flags & DF_CLIP_Z ) && mat_clipz.GetBool() )
|
|
{
|
|
if( flags & DF_CLIP_BELOW )
|
|
{
|
|
clipMode = MATERIAL_HEIGHTCLIPMODE_RENDER_ABOVE_HEIGHT;
|
|
}
|
|
else
|
|
{
|
|
clipMode = MATERIAL_HEIGHTCLIPMODE_RENDER_BELOW_HEIGHT;
|
|
}
|
|
}
|
|
|
|
if( flags & DF_RENDER_REFRACTION )
|
|
{
|
|
ITexture *pTexture = GetWaterRefractionTexture();
|
|
|
|
// Use the aspect ratio of the main view! So, don't recompute it here
|
|
view.x = view.y = 0;
|
|
view.width = pTexture->GetActualWidth();
|
|
view.height = pTexture->GetActualHeight();
|
|
|
|
materials->SetFogZ( waterHeight );
|
|
materials->SetHeightClipZ( waterHeight );
|
|
materials->SetHeightClipMode( clipMode );
|
|
|
|
// Have to re-set up the view since we reset the size
|
|
render->Push3DView( view, nClearFlags, true, pTexture, m_Frustum );
|
|
return;
|
|
}
|
|
|
|
if( flags & DF_RENDER_REFLECTION )
|
|
{
|
|
ITexture *pTexture = GetWaterReflectionTexture();
|
|
|
|
materials->SetFogZ( waterHeight );
|
|
|
|
// Use the aspect ratio of the main view! So, don't recompute it here
|
|
view.x = view.y = 0;
|
|
view.width = pTexture->GetActualWidth();
|
|
view.height = pTexture->GetActualHeight();
|
|
view.angles[0] = -view.angles[0];
|
|
view.angles[2] = -view.angles[2];
|
|
view.origin[2] -= 2.0f * ( view.origin[2] - (origWaterHeight));
|
|
bool bSoftwareUserClipPlane = g_pMaterialSystemHardwareConfig->UseFastClipping();
|
|
if( bSoftwareUserClipPlane && ( view.origin[2] > waterHeight - r_eyewaterepsilon.GetFloat() ) )
|
|
{
|
|
waterHeight = view.origin[2] + r_eyewaterepsilon.GetFloat();
|
|
}
|
|
|
|
materials->SetHeightClipZ( waterHeight );
|
|
materials->SetHeightClipMode( clipMode );
|
|
|
|
render->Push3DView( view, nClearFlags, true, pTexture, m_Frustum );
|
|
SetLightmapScaleForWater();
|
|
return;
|
|
}
|
|
|
|
if ( nClearFlags & (VIEW_CLEAR_DEPTH | VIEW_CLEAR_COLOR) )
|
|
{
|
|
materials->ClearBuffers( nClearFlags & VIEW_CLEAR_COLOR, nClearFlags & VIEW_CLEAR_DEPTH );
|
|
}
|
|
|
|
materials->SetHeightClipMode( clipMode );
|
|
if ( clipMode != MATERIAL_HEIGHTCLIPMODE_DISABLE )
|
|
{
|
|
materials->SetHeightClipZ( waterHeight );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Pops a water render target
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::PopWaterRenderTarget( int nFlags )
|
|
{
|
|
materials->SetHeightClipMode( MATERIAL_HEIGHTCLIPMODE_DISABLE );
|
|
if( nFlags & (DF_RENDER_REFRACTION | DF_RENDER_REFLECTION) )
|
|
{
|
|
render->PopView( m_Frustum );
|
|
if (SavedLinearLightMapScale.x>=0)
|
|
{
|
|
materials->SetToneMappingScaleLinear(SavedLinearLightMapScale);
|
|
SavedLinearLightMapScale.x=-1;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the world + entities
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::WaterDrawHelper( const CViewSetup &view, ClientWorldListInfo_t &info, CRenderList &renderList,
|
|
float waterHeight, int flags, view_id_t viewID, float waterZAdjust, int iForceViewLeaf )
|
|
{
|
|
int savedViewID = g_CurrentViewID;
|
|
g_CurrentViewID = viewID;
|
|
|
|
// Need a copy of the view since pushing a water render target
|
|
// make change the aspect ratio of the projection space transform
|
|
CViewSetup actualView = view;
|
|
|
|
int nClearFlags = 0;
|
|
if ( flags & DF_CLEARDEPTH )
|
|
{
|
|
nClearFlags |= VIEW_CLEAR_DEPTH;
|
|
}
|
|
if ( flags & DF_CLEARCOLOR )
|
|
{
|
|
nClearFlags |= VIEW_CLEAR_COLOR;
|
|
}
|
|
|
|
PushWaterRenderTarget( actualView, nClearFlags, waterHeight, flags );
|
|
|
|
if( flags & DF_BUILDWORLDLISTS )
|
|
{
|
|
bool bDrawEntities = ( flags & DF_DRAW_ENTITITES ) != 0;
|
|
bool bUpdateLightmaps = ( flags & DF_UPDATELIGHTMAPS ) != 0;
|
|
BuildWorldRenderLists( &actualView, info, bUpdateLightmaps, bDrawEntities, iForceViewLeaf );
|
|
}
|
|
|
|
// Make sure sound doesn't stutter
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
// Update our render view flags.
|
|
m_DrawFlags = flags | m_BaseDrawFlags;
|
|
|
|
ITexture *pSaveFrameBufferCopyTexture = materials->GetFrameBufferCopyTexture( 0 );
|
|
if ( engine->GetDXSupportLevel() >= 80 )
|
|
{
|
|
materials->SetFrameBufferCopyTexture( GetPowerOfTwoFrameBufferTexture() );
|
|
}
|
|
|
|
DrawWorld( info, renderList, m_DrawFlags, waterZAdjust );
|
|
|
|
// FIXME: This may have to be static, not sure if we'll get a stack overflow
|
|
ClientWorldListInfo_t tmpInfo;
|
|
tmpInfo.m_pLeafList = (LeafIndex_t*)stackalloc( info.m_LeafCount * sizeof(LeafIndex_t) );
|
|
tmpInfo.m_pLeafFogVolume = (LeafFogVolume_t*)stackalloc( info.m_LeafCount * sizeof(LeafFogVolume_t) );
|
|
tmpInfo.m_pActualLeafIndex = (LeafIndex_t*)stackalloc( info.m_LeafCount * sizeof(LeafIndex_t) );
|
|
ClientWorldListInfo_t *pInfo = ComputeActualWorldListInfo( info, m_DrawFlags, tmpInfo );
|
|
|
|
|
|
if ( flags & DF_DRAW_ENTITITES )
|
|
{
|
|
BuildRenderableRenderLists( &actualView, *pInfo, renderList );
|
|
DrawOpaqueRenderables( *pInfo, renderList );
|
|
DrawTranslucentRenderables( *pInfo, renderList, m_DrawFlags, false );
|
|
}
|
|
else
|
|
{
|
|
// Draw translucent world brushes only, no entities
|
|
DrawTranslucentWorldInLeaves( pInfo->m_LeafCount - 1, 0, *pInfo, m_DrawFlags );
|
|
}
|
|
|
|
// issue the pixel visibility tests
|
|
if ( CurrentViewID() != VIEW_MAIN )
|
|
{
|
|
PixelVisibility_EndCurrentView();
|
|
}
|
|
|
|
materials->SetFrameBufferCopyTexture( pSaveFrameBufferCopyTexture );
|
|
PopWaterRenderTarget( m_DrawFlags );
|
|
|
|
m_DrawFlags = m_BaseDrawFlags;
|
|
g_CurrentViewID = savedViewID;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Returns true if the view plane intersects the water
|
|
//-----------------------------------------------------------------------------
|
|
bool DoesViewPlaneIntersectWater( float waterZ, int leafWaterDataID )
|
|
{
|
|
if ( leafWaterDataID == -1 )
|
|
return false;
|
|
|
|
VMatrix viewMatrix, projectionMatrix, viewProjectionMatrix, inverseViewProjectionMatrix;
|
|
materials->GetMatrix( MATERIAL_VIEW, &viewMatrix );
|
|
materials->GetMatrix( MATERIAL_PROJECTION, &projectionMatrix );
|
|
MatrixMultiply( projectionMatrix, viewMatrix, viewProjectionMatrix );
|
|
MatrixInverseGeneral( viewProjectionMatrix, inverseViewProjectionMatrix );
|
|
|
|
Vector mins, maxs;
|
|
ClearBounds( mins, maxs );
|
|
Vector testPoint[4];
|
|
testPoint[0].Init( -1.0f, -1.0f, 0.0f );
|
|
testPoint[1].Init( -1.0f, 1.0f, 0.0f );
|
|
testPoint[2].Init( 1.0f, -1.0f, 0.0f );
|
|
testPoint[3].Init( 1.0f, 1.0f, 0.0f );
|
|
int i;
|
|
bool bAbove = false;
|
|
bool bBelow = false;
|
|
float fudge = 7.0f;
|
|
for( i = 0; i < 4; i++ )
|
|
{
|
|
Vector worldPos;
|
|
Vector3DMultiplyPositionProjective( inverseViewProjectionMatrix, testPoint[i], worldPos );
|
|
AddPointToBounds( worldPos, mins, maxs );
|
|
// Warning( "viewplanez: %f waterZ: %f\n", worldPos.z, waterZ );
|
|
if( worldPos.z + fudge > waterZ )
|
|
{
|
|
bAbove = true;
|
|
}
|
|
if( worldPos.z - fudge < waterZ )
|
|
{
|
|
bBelow = true;
|
|
}
|
|
}
|
|
|
|
// early out if the near plane doesn't cross the z plane of the water.
|
|
if( !( bAbove && bBelow ) )
|
|
return false;
|
|
|
|
Vector vecFudge( fudge, fudge, fudge );
|
|
mins -= vecFudge;
|
|
maxs += vecFudge;
|
|
|
|
// the near plane does cross the z value for the visible water volume. Call into
|
|
// the engine to find out if the near plane intersects the water volume.
|
|
return render->DoesBoxIntersectWaterVolume( mins, maxs, leafWaterDataID );
|
|
}
|
|
|
|
static void CalcWaterEyeAdjustments( const CViewSetup &view, const VisibleFogVolumeInfo_t &fogInfo,
|
|
float &newWaterHeight, float &waterZAdjust, bool bSoftwareUserClipPlane )
|
|
{
|
|
if( !bSoftwareUserClipPlane )
|
|
{
|
|
newWaterHeight = fogInfo.m_flWaterHeight;
|
|
waterZAdjust = 0.0f;
|
|
return;
|
|
}
|
|
|
|
newWaterHeight = fogInfo.m_flWaterHeight;
|
|
float eyeToWaterZDelta = view.origin[2] - fogInfo.m_flWaterHeight;
|
|
float epsilon = r_eyewaterepsilon.GetFloat();
|
|
waterZAdjust = 0.0f;
|
|
if( fabs( eyeToWaterZDelta ) < epsilon )
|
|
{
|
|
if( eyeToWaterZDelta > 0 )
|
|
{
|
|
newWaterHeight = view.origin[2] - epsilon;
|
|
}
|
|
else
|
|
{
|
|
newWaterHeight = view.origin[2] + epsilon;
|
|
}
|
|
waterZAdjust = newWaterHeight - fogInfo.m_flWaterHeight;
|
|
}
|
|
|
|
// Warning( "view.origin[2]: %f newWaterHeight: %f fogInfo.m_flWaterHeight: %f waterZAdjust: %f\n",
|
|
// ( float )view.origin[2], newWaterHeight, fogInfo.m_flWaterHeight, waterZAdjust );
|
|
}
|
|
|
|
|
|
#ifdef _XBOX
|
|
//-----------------------------------------------------------------------------
|
|
// Draws a perspective-correct dudv map into the reflection texture
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::DrawScreenSpaceWaterDuDv( const CViewSetup &view, float waterZAdjust )
|
|
{
|
|
ITexture *pTexture = GetWaterReflectionTexture();
|
|
|
|
// Use the aspect ratio of the main view! So, don't recompute it here
|
|
CViewSetup actualView = view;
|
|
actualView.width = pTexture->GetActualWidth();
|
|
actualView.height = pTexture->GetActualHeight();
|
|
|
|
// these DU/DV constants need to be 64, not 128 on xbox because the texture is signed and we only use the positive half of that space
|
|
materials->ClearColor4ub( 255, 64, 64, 0 );
|
|
render->Push3DView( actualView, VIEW_CLEAR_DEPTH | VIEW_CLEAR_COLOR, true, pTexture, m_Frustum );
|
|
materials->SetHeightClipMode( MATERIAL_HEIGHTCLIPMODE_DISABLE );
|
|
|
|
// Make sure sound doesn't stutter
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
render->DrawWaterDuDv( waterZAdjust );
|
|
|
|
render->PopView( m_Frustum );
|
|
}
|
|
#endif // _XBOX
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the scene when the view point is above the level of the water
|
|
//-----------------------------------------------------------------------------
|
|
#include "tier0/memdbgoff.h"
|
|
|
|
void CViewRender::ViewDrawScene_EyeAboveWater( bool bDrawSkybox, const CViewSetup &view,
|
|
int nClearFlags, const VisibleFogVolumeInfo_t &fogInfo, const WaterRenderInfo_t& waterInfo )
|
|
{
|
|
VPROF( "CViewRender::ViewDrawScene_EyeAboveWater" );
|
|
|
|
bool bSoftwareUserClipPlane = g_pMaterialSystemHardwareConfig->UseFastClipping();
|
|
|
|
ClientWorldListInfo_t info;
|
|
#ifndef _XBOX
|
|
CRenderList renderList;
|
|
CRenderList *pRenderList = &renderList;
|
|
#else
|
|
void *ptr = MemAllocScratch(sizeof(CRenderList));
|
|
CRenderList *pRenderList = new (ptr) CRenderList;
|
|
#endif
|
|
|
|
float newWaterHeight, waterZAdjust;
|
|
CalcWaterEyeAdjustments( view, fogInfo, newWaterHeight, waterZAdjust, bSoftwareUserClipPlane );
|
|
|
|
// eye is outside of water
|
|
|
|
// render the reflection
|
|
if( waterInfo.m_bReflect )
|
|
{
|
|
// NOTE: Clearing the color is unnecessary since we're drawing the skybox
|
|
// and dest-alpha is never used in the reflection
|
|
int nFlags = DF_RENDER_REFLECTION | DF_CLIP_Z | DF_CLIP_BELOW |
|
|
DF_RENDER_ABOVEWATER | DF_CLEARDEPTH | DF_UPDATELIGHTMAPS | DF_BUILDWORLDLISTS;
|
|
|
|
// NOTE: This will cause us to draw the 2d skybox in the reflection
|
|
// (which we want to do instead of drawing the 3d skybox)
|
|
nFlags |= DF_DRAWSKYBOX;
|
|
|
|
if( waterInfo.m_bReflectEntities )
|
|
{
|
|
nFlags |= DF_DRAW_ENTITITES;
|
|
}
|
|
|
|
// Disable occlusion visualization in reflection
|
|
bool bVisOcclusion = r_visocclusion.GetInt();
|
|
r_visocclusion.SetValue( 0 );
|
|
|
|
EnableWorldFog();
|
|
WaterDrawHelper( view, info, *pRenderList, fogInfo.m_flWaterHeight, nFlags, VIEW_REFLECTION,
|
|
0.0f, fogInfo.m_nVisibleFogVolumeLeaf );
|
|
|
|
r_visocclusion.SetValue( bVisOcclusion );
|
|
|
|
materials->Flush();
|
|
}
|
|
|
|
unsigned char ucFogColor[3];
|
|
|
|
int nFlags;
|
|
bool bViewIntersectsWater = false;
|
|
// render refraction
|
|
int nFirstPassFlags = DF_BUILDWORLDLISTS | DF_UPDATELIGHTMAPS | DF_CLEARCOLOR;
|
|
if ( waterInfo.m_bRefract )
|
|
{
|
|
nFlags = nFirstPassFlags | DF_RENDER_REFRACTION | DF_CLIP_Z |
|
|
DF_RENDER_UNDERWATER | DF_FUDGE_UP |
|
|
DF_DRAW_ENTITITES | DF_MAINTAINWORLDLISTS | DF_CLEARDEPTH;
|
|
nFirstPassFlags = 0;
|
|
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, true );
|
|
|
|
SetClearColorToFogColor( ucFogColor );
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, VIEW_REFRACTION,
|
|
waterZAdjust, -1 );
|
|
if( !bSoftwareUserClipPlane )
|
|
{
|
|
bViewIntersectsWater = DoesViewPlaneIntersectWater( fogInfo.m_flWaterHeight, fogInfo.m_nVisibleFogVolume );
|
|
}
|
|
|
|
#ifdef _XBOX
|
|
// This is a workaround for the lack of perspective correction on the XBox for dudv maps
|
|
EnableWorldFog();
|
|
DrawScreenSpaceWaterDuDv( view, waterZAdjust );
|
|
#endif
|
|
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
}
|
|
|
|
materials->Flush();
|
|
|
|
EnableWorldFog();
|
|
|
|
// render the world
|
|
nFlags = nFirstPassFlags | DF_RENDER_ABOVEWATER | DF_CLEARDEPTH | DF_DRAW_ENTITITES | DF_MAINTAINWORLDLISTS;
|
|
if( bViewIntersectsWater && !bSoftwareUserClipPlane )
|
|
{
|
|
// This is necessary to keep the non-water fogged world from drawing underwater in
|
|
// the case where we want to partially see into the water.
|
|
nFlags |= DF_CLIP_Z | DF_CLIP_BELOW;
|
|
}
|
|
|
|
if ( bDrawSkybox )
|
|
{
|
|
nFlags |= DF_DRAWSKYBOX;
|
|
nFlags &= ~DF_CLEARCOLOR;
|
|
}
|
|
if ( waterInfo.m_bDrawWaterSurface )
|
|
{
|
|
nFlags |= DF_RENDER_WATER;
|
|
}
|
|
if ( !waterInfo.m_bRefract && !waterInfo.m_bOpaqueWater )
|
|
{
|
|
nFlags |= DF_RENDER_UNDERWATER;
|
|
}
|
|
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, CurrentViewID(), waterZAdjust, -1 );
|
|
|
|
if( waterZAdjust != 0.0f && bSoftwareUserClipPlane && waterInfo.m_bRefract )
|
|
{
|
|
nFlags = DF_RENDER_UNDERWATER;
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, CurrentViewID(), waterZAdjust,-1 );
|
|
}
|
|
else if( bViewIntersectsWater )
|
|
{
|
|
materials->ClearColor4ub( ucFogColor[0], ucFogColor[1], ucFogColor[2], 255 );
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, true );
|
|
nFlags = DF_RENDER_UNDERWATER | DF_CLIP_Z | DF_DRAW_ENTITITES;
|
|
WaterDrawHelper( view, info, *pRenderList, fogInfo.m_flWaterHeight, nFlags, VIEW_NONE, 0.0f, -1 );
|
|
}
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
|
|
#ifdef _XBOX
|
|
MemFreeScratch();
|
|
#endif
|
|
}
|
|
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the scene when the view point is under the level of the water
|
|
//-----------------------------------------------------------------------------
|
|
#include "tier0/memdbgoff.h"
|
|
|
|
void CViewRender::ViewDrawScene_EyeUnderWater( bool bDrawSkybox, const CViewSetup &view,
|
|
int nClearFlags, const VisibleFogVolumeInfo_t &fogInfo, const WaterRenderInfo_t& waterInfo )
|
|
{
|
|
// FIXME: The 3d skybox shouldn't be drawn when the eye is under water
|
|
|
|
VPROF( "CViewRender::ViewDrawScene_EyeUnderWater" );
|
|
|
|
bool bSoftwareUserClipPlane = g_pMaterialSystemHardwareConfig->UseFastClipping();
|
|
|
|
float newWaterHeight, waterZAdjust;
|
|
CalcWaterEyeAdjustments( view, fogInfo, newWaterHeight, waterZAdjust, bSoftwareUserClipPlane );
|
|
|
|
ClientWorldListInfo_t info;
|
|
|
|
#ifndef _XBOX
|
|
CRenderList renderList;
|
|
CRenderList *pRenderList = &renderList;
|
|
#else
|
|
void *ptr = MemAllocScratch(sizeof(CRenderList));
|
|
CRenderList *pRenderList = new (ptr) CRenderList;
|
|
#endif
|
|
|
|
int nFirstPassFlags = DF_BUILDWORLDLISTS | DF_UPDATELIGHTMAPS;
|
|
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, true );
|
|
unsigned char ucFogColor[3];
|
|
materials->GetFogColor( ucFogColor );
|
|
materials->ClearColor4ub( ucFogColor[0], ucFogColor[1], ucFogColor[2], 255 );
|
|
|
|
// render refraction (out of water)
|
|
if ( waterInfo.m_bRefract )
|
|
{
|
|
// NOTE: Refraction renders into the back buffer, over the top of the 3D skybox
|
|
// It is then blitted out into the refraction target. This is so that
|
|
// we only have to set up 3d sky vis once, and only render it once also!
|
|
int nFlags = nFirstPassFlags | DF_CLIP_Z |
|
|
DF_CLIP_BELOW | DF_RENDER_ABOVEWATER |
|
|
DF_DRAW_ENTITITES | DF_MAINTAINWORLDLISTS | DF_CLEARDEPTH;
|
|
|
|
if ( bDrawSkybox )
|
|
{
|
|
nFlags |= DF_DRAWSKYBOX | DF_CLIP_SKYBOX | DF_CLEARCOLOR;
|
|
}
|
|
nFirstPassFlags = 0;
|
|
|
|
EnableWorldFog();
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, VIEW_REFRACTION, waterZAdjust, -1 );
|
|
|
|
Rect_t srcRect;
|
|
srcRect.x = view.x;
|
|
srcRect.y = view.y;
|
|
srcRect.width = view.width;
|
|
srcRect.height = view.height;
|
|
|
|
ITexture *pTexture = GetWaterRefractionTexture();
|
|
materials->CopyRenderTargetToTextureEx( pTexture, 0, &srcRect, NULL );
|
|
|
|
#ifdef _XBOX
|
|
// This is a workaround for the lack of perspective correction on the XBox for dudv maps
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, false );
|
|
DrawScreenSpaceWaterDuDv( view, waterZAdjust );
|
|
materials->ClearColor4ub( ucFogColor[0], ucFogColor[1], ucFogColor[2], 255 );
|
|
#endif
|
|
}
|
|
|
|
// NOTE: We're not drawing the 2d skybox under water since it's assumed to not be visible.
|
|
|
|
// render the world underwater
|
|
// Clear the color to get the appropriate underwater fog color
|
|
int nFlags = nFirstPassFlags | DF_MAINTAINWORLDLISTS | DF_FUDGE_UP |
|
|
DF_RENDER_UNDERWATER | DF_CLEARDEPTH | DF_DRAW_ENTITITES;
|
|
|
|
if( !bSoftwareUserClipPlane )
|
|
{
|
|
nFlags |= DF_CLIP_Z;
|
|
}
|
|
if ( waterInfo.m_bDrawWaterSurface )
|
|
{
|
|
nFlags |= DF_RENDER_WATER;
|
|
}
|
|
if ( !waterInfo.m_bRefract && !waterInfo.m_bOpaqueWater )
|
|
{
|
|
nFlags |= DF_RENDER_ABOVEWATER;
|
|
}
|
|
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, false );
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, CurrentViewID(), waterZAdjust, -1 );
|
|
|
|
if( waterZAdjust != 0.0f && bSoftwareUserClipPlane && waterInfo.m_bRefract )
|
|
{
|
|
nFlags = DF_RENDER_ABOVEWATER;
|
|
WaterDrawHelper( view, info, *pRenderList, newWaterHeight, nFlags, CurrentViewID(), waterZAdjust, -1 );
|
|
}
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
|
|
#ifdef _XBOX
|
|
MemFreeScratch();
|
|
#endif
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Draws the scene when there's no water or cheap water
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::ViewDrawScene_NoWater( bool bDrawSkybox, const CViewSetup &view, int nClearFlags,
|
|
const VisibleFogVolumeInfo_t &fogInfo, const WaterRenderInfo_t& waterInfo )
|
|
{
|
|
VPROF( "CViewRender::ViewDrawScene_NoWater" );
|
|
|
|
ClientWorldListInfo_t info;
|
|
#ifndef _XBOX
|
|
CRenderList renderList;
|
|
CRenderList *pRenderList = &renderList;
|
|
#else
|
|
void *ptr = MemAllocScratch(sizeof(CRenderList));
|
|
CRenderList *pRenderList = new (ptr) CRenderList;
|
|
#endif
|
|
|
|
int nFlags = DF_BUILDWORLDLISTS | DF_DRAW_ENTITITES | DF_UPDATELIGHTMAPS;
|
|
if ( nClearFlags & VIEW_CLEAR_COLOR )
|
|
{
|
|
nFlags |= DF_CLEARCOLOR;
|
|
}
|
|
if ( nClearFlags & VIEW_CLEAR_DEPTH )
|
|
{
|
|
nFlags |= DF_CLEARDEPTH;
|
|
}
|
|
|
|
if ( !waterInfo.m_bOpaqueWater )
|
|
{
|
|
nFlags |= DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER;
|
|
}
|
|
else
|
|
{
|
|
bool bViewIntersectsWater = DoesViewPlaneIntersectWater( fogInfo.m_flWaterHeight, fogInfo.m_nVisibleFogVolume );
|
|
if( bViewIntersectsWater )
|
|
{
|
|
// have to draw both sides if we can see both.
|
|
nFlags |= DF_RENDER_UNDERWATER | DF_RENDER_ABOVEWATER;
|
|
}
|
|
else if ( fogInfo.m_bEyeInFogVolume )
|
|
{
|
|
nFlags |= DF_RENDER_UNDERWATER;
|
|
}
|
|
else
|
|
{
|
|
nFlags |= DF_RENDER_ABOVEWATER;
|
|
}
|
|
}
|
|
if ( waterInfo.m_bDrawWaterSurface )
|
|
{
|
|
nFlags |= DF_RENDER_WATER;
|
|
}
|
|
|
|
if ( !fogInfo.m_bEyeInFogVolume )
|
|
{
|
|
if ( bDrawSkybox )
|
|
{
|
|
nFlags |= DF_DRAWSKYBOX;
|
|
}
|
|
EnableWorldFog();
|
|
}
|
|
else
|
|
{
|
|
nFlags |= DF_CLEARCOLOR;
|
|
|
|
render->SetFogVolumeState( fogInfo.m_nVisibleFogVolume, false );
|
|
|
|
unsigned char ucFogColor[3];
|
|
materials->GetFogColor( ucFogColor );
|
|
materials->ClearColor4ub( ucFogColor[0], ucFogColor[1], ucFogColor[2], 255 );
|
|
}
|
|
|
|
WaterDrawHelper( view, info, *pRenderList, 0.0f, nFlags, CurrentViewID(), 0.0f, m_iForceViewLeaf );
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
|
|
#ifdef _XBOX
|
|
MemFreeScratch();
|
|
#endif
|
|
}
|
|
|
|
#include "tier0/memdbgon.h"
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Methods related to controlling the cheap water distance
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::SetCheapWaterStartDistance( float flCheapWaterStartDistance )
|
|
{
|
|
m_flCheapWaterStartDistance = flCheapWaterStartDistance;
|
|
}
|
|
|
|
void CViewRender::SetCheapWaterEndDistance( float flCheapWaterEndDistance )
|
|
{
|
|
m_flCheapWaterEndDistance = flCheapWaterEndDistance;
|
|
}
|
|
|
|
void CViewRender::GetWaterLODParams( float &flCheapWaterStartDistance, float &flCheapWaterEndDistance )
|
|
{
|
|
flCheapWaterStartDistance = m_flCheapWaterStartDistance;
|
|
flCheapWaterEndDistance = m_flCheapWaterEndDistance;
|
|
}
|
|
|
|
static void CheapWaterStart_f( void )
|
|
{
|
|
if( engine->Cmd_Argc() == 2 )
|
|
{
|
|
float dist = atof( engine->Cmd_Argv( 1 ) );
|
|
view->SetCheapWaterStartDistance( dist );
|
|
}
|
|
else
|
|
{
|
|
float start, end;
|
|
view->GetWaterLODParams( start, end );
|
|
Warning( "r_cheapwaterstart: %f\n", start );
|
|
}
|
|
}
|
|
|
|
static void CheapWaterEnd_f( void )
|
|
{
|
|
if( engine->Cmd_Argc() == 2 )
|
|
{
|
|
float dist = atof( engine->Cmd_Argv( 1 ) );
|
|
view->SetCheapWaterEndDistance( dist );
|
|
}
|
|
else
|
|
{
|
|
float start, end;
|
|
view->GetWaterLODParams( start, end );
|
|
Warning( "r_cheapwaterend: %f\n", end );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// A console command allowing you to draw a material as an overlay
|
|
//-----------------------------------------------------------------------------
|
|
static void ScreenOverlay_f( void )
|
|
{
|
|
if( engine->Cmd_Argc() == 2 )
|
|
{
|
|
if ( !Q_stricmp( "off", engine->Cmd_Argv(1) ) )
|
|
{
|
|
view->SetScreenOverlayMaterial( NULL );
|
|
}
|
|
else
|
|
{
|
|
IMaterial *pMaterial = materials->FindMaterial( engine->Cmd_Argv(1), TEXTURE_GROUP_OTHER, false );
|
|
if ( !IsErrorMaterial( pMaterial ) )
|
|
{
|
|
view->SetScreenOverlayMaterial( pMaterial );
|
|
}
|
|
else
|
|
{
|
|
view->SetScreenOverlayMaterial( NULL );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
IMaterial *pMaterial = view->GetScreenOverlayMaterial();
|
|
Warning( "r_screenoverlay: %s\n", pMaterial ? pMaterial->GetName() : "off" );
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Purpose:
|
|
// Input : &view -
|
|
// &introData -
|
|
//-----------------------------------------------------------------------------
|
|
void CViewRender::ViewDrawScene_Intro( const CViewSetup &view, int nClearFlags, const IntroData_t &introData )
|
|
{
|
|
VPROF( "CViewRender::ViewDrawScene" );
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Set the clear color to black since we are going to be adding up things
|
|
// in the frame buffer.
|
|
// -----------------------------------------------------------------------
|
|
// Clear alpha to 255 so that masking with the vortigaunts (0) works properly.
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Draw the primary scene and copy it to the first framebuffer texture
|
|
// -----------------------------------------------------------------------
|
|
CViewSetup playerView = view;
|
|
playerView.origin = introData.m_vecCameraView;
|
|
playerView.m_vUnreflectedOrigin = introData.m_vecCameraView;
|
|
playerView.angles = introData.m_vecCameraViewAngles;
|
|
if ( introData.m_playerViewFOV )
|
|
{
|
|
playerView.fov = ScaleFOVByWidthRatio( introData.m_playerViewFOV, engine->GetScreenAspectRatio() / ( 4.0f / 3.0f ) );
|
|
}
|
|
SetupCurrentView( playerView.origin, playerView.angles, VIEW_INTRO_PLAYER );
|
|
|
|
// Invoke pre-render methods
|
|
IGameSystem::PreRenderAllSystems();
|
|
|
|
// Start view, clear frame/z buffer if necessary
|
|
unsigned int visFlags;
|
|
SetupVis( playerView, visFlags );
|
|
|
|
if( introData.m_bDrawSecondary )
|
|
{
|
|
// NOTE: We only increment this once since time doesn't move forward.
|
|
ParticleMgr()->IncrementFrameCode();
|
|
}
|
|
|
|
if( introData.m_bDrawPrimary )
|
|
{
|
|
DrawWorldAndEntities( true /* drawSkybox */, playerView, VIEW_CLEAR_COLOR | VIEW_CLEAR_DEPTH );
|
|
}
|
|
else
|
|
{
|
|
materials->ClearBuffers( true, true );
|
|
}
|
|
|
|
Rect_t actualRect;
|
|
UpdateScreenEffectTexture( 0, view.x, view.y, view.width, view.height, false, &actualRect );
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Draw the secondary scene and copy it to the second framebuffer texture
|
|
// -----------------------------------------------------------------------
|
|
CViewSetup cameraView = view;
|
|
SetupCurrentView( cameraView.origin, cameraView.angles, VIEW_INTRO_CAMERA );
|
|
|
|
// Invoke pre-render methods
|
|
IGameSystem::PreRenderAllSystems();
|
|
|
|
// Start view, clear frame/z buffer if necessary
|
|
SetupVis( cameraView, visFlags );
|
|
|
|
// Clear alpha to 255 so that masking with the vortigaunts (0) works properly.
|
|
materials->ClearColor4ub( 0, 0, 0, 255 );
|
|
|
|
if( introData.m_bDrawSecondary )
|
|
{
|
|
DrawWorldAndEntities( true /* drawSkybox */, cameraView, VIEW_CLEAR_COLOR | VIEW_CLEAR_DEPTH );
|
|
}
|
|
else
|
|
{
|
|
materials->ClearBuffers( true, true );
|
|
}
|
|
UpdateScreenEffectTexture( 1, view.x, view.y, view.width, view.height );
|
|
|
|
// -----------------------------------------------------------------------
|
|
// Draw quads on the screen for each screenspace pass.
|
|
// -----------------------------------------------------------------------
|
|
// Find the material that we use to render the overlays
|
|
IMaterial *pOverlayMaterial = materials->FindMaterial( "scripted/intro_screenspaceeffect", TEXTURE_GROUP_OTHER );
|
|
IMaterialVar *pModeVar = pOverlayMaterial->FindVar( "$mode", NULL );
|
|
IMaterialVar *pAlphaVar = pOverlayMaterial->FindVar( "$alpha", NULL );
|
|
|
|
materials->ClearBuffers( true, true );
|
|
|
|
materials->MatrixMode( MATERIAL_VIEW );
|
|
materials->PushMatrix();
|
|
materials->LoadIdentity();
|
|
|
|
materials->MatrixMode( MATERIAL_PROJECTION );
|
|
materials->PushMatrix();
|
|
materials->LoadIdentity();
|
|
|
|
int passID;
|
|
for( passID = 0; passID < introData.m_Passes.Count(); passID++ )
|
|
{
|
|
const IntroDataBlendPass_t& pass = introData.m_Passes[passID];
|
|
if ( pass.m_Alpha == 0 )
|
|
continue;
|
|
|
|
// Pick one of the blend modes for the material.
|
|
if( pass.m_BlendMode >= 0 && pass.m_BlendMode < 9 )
|
|
{
|
|
pModeVar->SetIntValue( pass.m_BlendMode );
|
|
}
|
|
else
|
|
{
|
|
Assert(0);
|
|
}
|
|
// Set the alpha value for the material.
|
|
pAlphaVar->SetFloatValue( pass.m_Alpha );
|
|
|
|
// Draw a quad for this pass.
|
|
ITexture *pTexture = GetFullFrameFrameBufferTexture( 0 );
|
|
materials->DrawScreenSpaceRectangle( pOverlayMaterial, view.x, view.y, view.width, view.height,
|
|
actualRect.x, actualRect.y, actualRect.x+actualRect.width-1, actualRect.y+actualRect.height-1,
|
|
pTexture->GetActualWidth(), pTexture->GetActualHeight() );
|
|
}
|
|
|
|
materials->MatrixMode( MATERIAL_VIEW );
|
|
materials->PopMatrix();
|
|
|
|
materials->MatrixMode( MATERIAL_PROJECTION );
|
|
materials->PopMatrix();
|
|
|
|
// Draw the starfield
|
|
// FIXME
|
|
// blur?
|
|
|
|
// Disable fog for the rest of the stuff
|
|
DisableFog();
|
|
|
|
// Here are the overlays...
|
|
CGlowOverlay::DrawOverlays( view.m_bCacheFullSceneState );
|
|
|
|
// issue the pixel visibility tests
|
|
PixelVisibility_EndCurrentView();
|
|
|
|
// And here are the screen-space effects
|
|
PerformScreenSpaceEffects( view.x, view.y, view.width, view.height );
|
|
|
|
// Make sure sound doesn't stutter
|
|
engine->Sound_ExtraUpdate();
|
|
|
|
// Debugging info goes over the top
|
|
Draw3DDebuggingInfo( view );
|
|
|
|
// Let the particle manager simulate things that haven't been simulated.
|
|
ParticleMgr()->PostRender();
|
|
|
|
FinishCurrentView();
|
|
}
|
|
|
|
#ifdef _XBOX
|
|
void ViewTexture_f()
|
|
{
|
|
int argc;
|
|
char* enable;
|
|
|
|
argc = engine->Cmd_Argc();
|
|
if (argc == 1)
|
|
{
|
|
// toggle
|
|
enable = mat_viewTextureEnable.GetInt() ? "0" : "1";
|
|
}
|
|
else
|
|
{
|
|
// force on
|
|
enable = "1";
|
|
}
|
|
|
|
mat_viewTextureEnable.SetValue(enable);
|
|
if (argc >= 2)
|
|
mat_viewTextureName.SetValue(engine->Cmd_Argv(1));
|
|
if (argc >= 3)
|
|
mat_viewTextureScale.SetValue(engine->Cmd_Argv(2));
|
|
}
|
|
static ConCommand mat_viewTexture("mat_viewTexture", ViewTexture_f, "Show/Hide texture [name] [scale]");
|
|
#endif
|
|
|
|
|
|
static ConCommand r_cheapwaterstart( "r_cheapwaterstart", CheapWaterStart_f );
|
|
static ConCommand r_cheapwaterend( "r_cheapwaterend", CheapWaterEnd_f );
|
|
static ConCommand r_screenspacematerial( "r_screenoverlay", ScreenOverlay_f );
|
|
|
|
|