//========= Copyright © 1996-2005, Valve Corporation, All rights reserved. ============// // // Purpose: // // $NoKeywords: $ //=============================================================================// #ifndef VIEW_SCENE_H #define VIEW_SCENE_H #ifdef _WIN32 #pragma once #endif #include "convar.h" #include "iviewrender.h" #include "view_shared.h" #include "rendertexture.h" #include "materialsystem/itexture.h" extern ConVar mat_wireframe; extern ConVar building_cubemaps; // Transform into view space (translate and rotate the camera into the origin). void ViewTransform( const Vector &worldSpace, Vector &viewSpace ); // Transform a world point into normalized screen space (X and Y from -1 to 1). // Returns 0 if the point is behind the viewer. int ScreenTransform( const Vector& point, Vector& screen ); extern ConVar r_updaterefracttexture; extern int g_viewscene_refractUpdateFrame; extern int g_nCurrentPortalRender; extern int g_nRefractUpdatePortalRender; extern bool g_bAllowMultipleRefractUpdatesPerScenePerFrame; bool DrawingShadowDepthView( void ); inline void UpdateRefractTexture( int x, int y, int w, int h, bool bForceUpdate = false ) { Assert( !DrawingShadowDepthView() ); if ( !r_updaterefracttexture.GetBool() ) return; CMatRenderContextPtr pRenderContext( materials ); ITexture *pTexture = GetPowerOfTwoFrameBufferTexture(); #ifdef PORTAL2 if ( IsPC() || bForceUpdate || g_bAllowMultipleRefractUpdatesPerScenePerFrame || ( gpGlobals->framecount != g_viewscene_refractUpdateFrame ) || ( g_nRefractUpdatePortalRender != g_nCurrentPortalRender ) ) #else if ( IsPC() || bForceUpdate || g_bAllowMultipleRefractUpdatesPerScenePerFrame || ( gpGlobals->framecount != g_viewscene_refractUpdateFrame ) ) #endif { // forced or only once per frame Rect_t rect; rect.x = x; rect.y = y; rect.width = w; rect.height = h; pRenderContext->CopyRenderTargetToTextureEx( pTexture, 0, &rect, IsPC() ? NULL : &rect ); #ifdef PORTAL2 g_nRefractUpdatePortalRender = g_nCurrentPortalRender; #endif g_viewscene_refractUpdateFrame = gpGlobals->framecount; } pRenderContext->SetFrameBufferCopyTexture( pTexture ); } inline void UpdateRefractTexture( bool bForceUpdate = false ) { Assert( !DrawingShadowDepthView() ); CMatRenderContextPtr pRenderContext( materials ); int x,y,w,h; pRenderContext->GetViewport( x, y, w, h ); UpdateRefractTexture( x, y, w, h, bForceUpdate ); } inline void UpdateScreenEffectTexture( int textureIndex, int x, int y, int w, int h, bool bDestFullScreen = false, Rect_t *pActualRect = NULL ) { Rect_t srcRect; srcRect.x = x; srcRect.y = y; srcRect.width = w; srcRect.height = h; CMatRenderContextPtr pRenderContext( materials ); ITexture *pTexture = GetFullFrameFrameBufferTexture( textureIndex ); int nSrcWidth, nSrcHeight; pRenderContext->GetRenderTargetDimensions( nSrcWidth, nSrcHeight ); int nDestWidth = pTexture->GetActualWidth(); int nDestHeight = pTexture->GetActualHeight(); Rect_t destRect = srcRect; if( !bDestFullScreen && ( nSrcWidth > nDestWidth || nSrcHeight > nDestHeight ) ) { // the source and target sizes aren't necessarily the same (specifically in dx7 where // nonpow2 rendertargets aren't supported), so lets figure it out here. float scaleX = ( float )nDestWidth / ( float )nSrcWidth; float scaleY = ( float )nDestHeight / ( float )nSrcHeight; destRect.x = srcRect.x * scaleX; destRect.y = srcRect.y * scaleY; destRect.width = srcRect.width * scaleX; destRect.height = srcRect.height * scaleY; destRect.x = clamp( destRect.x, 0, nDestWidth ); destRect.y = clamp( destRect.y, 0, nDestHeight ); destRect.width = clamp( destRect.width, 0, nDestWidth - destRect.x ); destRect.height = clamp( destRect.height, 0, nDestHeight - destRect.y ); } pRenderContext->CopyRenderTargetToTextureEx( pTexture, 0, &srcRect, bDestFullScreen ? NULL : &destRect ); pRenderContext->SetFrameBufferCopyTexture( pTexture, textureIndex ); if ( pActualRect ) { pActualRect->x = destRect.x; pActualRect->y = destRect.y; pActualRect->width = destRect.width; pActualRect->height = destRect.height; } } //----------------------------------------------------------------------------- // Draws the screen effect //----------------------------------------------------------------------------- inline 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 ); CMatRenderContextPtr pRenderContext( materials ); pRenderContext->DrawScreenSpaceRectangle( pMaterial, x, y, w, h, actualRect.x, actualRect.y, actualRect.x+actualRect.width-1, actualRect.y+actualRect.height-1, pTexture->GetActualWidth(), pTexture->GetActualHeight() ); } //intended for use by dynamic meshes to naively update front buffer textures needed by a material inline void UpdateFrontBufferTexturesForMaterial( IMaterial *pMaterial, bool bForce = false ) { Assert( !DrawingShadowDepthView() ); if( pMaterial->NeedsPowerOfTwoFrameBufferTexture() ) { UpdateRefractTexture( bForce ); } else if( pMaterial->NeedsFullFrameBufferTexture() ) { const CViewSetup *pView = view->GetViewSetup(); UpdateScreenEffectTexture( 0, pView->x, pView->y, pView->width, pView->height ); } } inline void UpdateScreenEffectTexture( void ) { Assert( !DrawingShadowDepthView() ); const CViewSetup *pViewSetup = view->GetViewSetup(); UpdateScreenEffectTexture( 0, pViewSetup->x, pViewSetup->y, pViewSetup->width, pViewSetup->height); } // reset the tonem apping to a constant value, and clear the filter bank void ResetToneMapping(float value); void UpdateFullScreenDepthTexture( void ); #endif // VIEW_SCENE_H