| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#include "platform/platform.h"#include "renderInstance/renderGlowMgr.h"#include "renderInstance/renderParticleMgr.h"#include "scene/sceneManager.h"#include "scene/sceneRenderState.h"#include "materials/sceneData.h"#include "materials/matInstance.h"#include "materials/materialFeatureTypes.h"#include "materials/processedMaterial.h"#include "postFx/postEffect.h"#include "gfx/gfxTransformSaver.h"#include "gfx/gfxDebugEvent.h"#include "math/util/matrixSet.h"IMPLEMENT_CONOBJECT( RenderGlowMgr );ConsoleDocClass( RenderGlowMgr,    "@brief A render bin for the glow pass.\n\n"   "When the glow buffer PostEffect is enabled this bin gathers mesh render "   "instances with glow materials and renders them to the #glowbuffer offscreen "   "render target.\n\n"   "This render target is then used by the 'GlowPostFx' PostEffect to blur and "   "render the glowing portions of the screen.\n\n"   "@ingroup RenderBin\n" );const MatInstanceHookType RenderGlowMgr::GlowMaterialHook::Type( "Glow" );RenderGlowMgr::GlowMaterialHook::GlowMaterialHook( BaseMatInstance *matInst )   : mGlowMatInst( NULL ){   mGlowMatInst = (MatInstance*)matInst->getMaterial()->createMatInstance();   mGlowMatInst->getFeaturesDelegate().bind( &GlowMaterialHook::_overrideFeatures );   mGlowMatInst->setUserObject(matInst->getUserObject());   mGlowMatInst->init(  matInst->getRequestedFeatures(),                         matInst->getVertexFormat() );}RenderGlowMgr::GlowMaterialHook::~GlowMaterialHook(){   SAFE_DELETE( mGlowMatInst );}void RenderGlowMgr::GlowMaterialHook::_overrideFeatures( ProcessedMaterial *mat,                                                         U32 stageNum,                                                         MaterialFeatureData &fd,                                                          const FeatureSet &features ){   // If this isn't a glow pass... then add the glow mask feature.   if (  mat->getMaterial() &&          !mat->getMaterial()->mGlow[stageNum] )      fd.features.addFeature( MFT_GlowMask );   // Don't allow fog or HDR encoding on    // the glow materials.   fd.features.removeFeature( MFT_Fog );   fd.features.removeFeature( MFT_HDROut );}RenderGlowMgr::RenderGlowMgr()   : RenderTexTargetBinManager(  RenderPassManager::RIT_Mesh,                                  1.0f,                                  1.0f,                                 GFXFormatR8G8B8A8,                                 Point2I( 512, 512 ) ){   notifyType( RenderPassManager::RIT_Decal );   notifyType( RenderPassManager::RIT_Translucent );   notifyType( RenderPassManager::RIT_Particle );   mParticleRenderMgr = NULL;   mNamedTarget.registerWithName( "glowbuffer" );   mTargetSizeType = WindowSize;}RenderGlowMgr::~RenderGlowMgr(){}PostEffect* RenderGlowMgr::getGlowEffect(){   if ( !mGlowEffect )      mGlowEffect = dynamic_cast<PostEffect*>( Sim::findObject( "GlowPostFx" ) );      return mGlowEffect;}bool RenderGlowMgr::isGlowEnabled(){   return getGlowEffect() && getGlowEffect()->isEnabled();}void RenderGlowMgr::addElement( RenderInst *inst ){   // Skip out if we don't have the glow post    // effect enabled at this time.   if ( !isGlowEnabled() )      return;   // TODO: We need to get the scene state here in a more reliable   // manner so we can skip glow in a non-diffuse render pass.   //if ( !mParentManager->getSceneManager()->getSceneState()->isDiffusePass() )      //return RenderBinManager::arSkipped;   ParticleRenderInst *particleInst = NULL;   if(inst->type == RenderPassManager::RIT_Particle)      particleInst = static_cast<ParticleRenderInst*>(inst);   if(particleInst && particleInst->glow)   {      internalAddElement(inst);      return;   }   // Skip it if we don't have a glowing material.   BaseMatInstance *matInst = getMaterial( inst );   if ( !matInst || !matInst->hasGlow() )         return;   internalAddElement(inst);}void RenderGlowMgr::render( SceneRenderState *state ){   PROFILE_SCOPE( RenderGlowMgr_Render );      if ( !isGlowEnabled() )      return;   const U32 binSize = mElementList.size();   // If this is a non-diffuse pass or we have no objects to   // render then tell the effect to skip rendering.   if ( !state->isDiffusePass() || binSize == 0 )   {      getGlowEffect()->setSkip( true );      return;   }   GFXDEBUGEVENT_SCOPE( RenderGlowMgr_Render, ColorI::GREEN );   GFXTransformSaver saver;   // Respect the current viewport   mNamedTarget.setViewport(GFX->getViewport());   // Tell the superclass we're about to render, preserve contents   const bool isRenderingToTarget = _onPreRender( state, true );   // Clear all the buffers to black.   GFX->clear( GFXClearTarget, ColorI::BLACK, 1.0f, 0);   // Restore transforms   MatrixSet &matrixSet = getRenderPass()->getMatrixSet();   matrixSet.restoreSceneViewProjection();   // init loop data   SceneData sgData;   sgData.init( state, SceneData::GlowBin );   for( U32 j=0; j<binSize; )   {      RenderInst *_ri = mElementList[j].inst;      if(_ri->type == RenderPassManager::RIT_Particle)      {         // Find the particle render manager (if we don't have it)         if(mParticleRenderMgr == NULL)         {            RenderPassManager *rpm = state->getRenderPass();            for( U32 i = 0; i < rpm->getManagerCount(); i++ )            {               RenderBinManager *bin = rpm->getManager(i);               if( bin->getRenderInstType() == RenderParticleMgr::RIT_Particles )               {                  mParticleRenderMgr = reinterpret_cast<RenderParticleMgr *>(bin);                  break;               }            }         }         ParticleRenderInst *ri = static_cast<ParticleRenderInst*>(_ri);         mParticleRenderMgr->renderParticle(ri, state);         j++;         continue;      }      MeshRenderInst *ri = static_cast<MeshRenderInst*>(_ri);      setupSGData( ri, sgData );      BaseMatInstance *mat = ri->matInst;      GlowMaterialHook *hook = mat->getHook<GlowMaterialHook>();      if ( !hook )      {         hook = new GlowMaterialHook( ri->matInst );         ri->matInst->addHook( hook );      }      BaseMatInstance *glowMat = hook->getMatInstance();      U32 matListEnd = j;      while( glowMat && glowMat->setupPass( state, sgData ) )      {         U32 a;         for( a=j; a<binSize; a++ )         {            if (mElementList[a].inst->type == RenderPassManager::RIT_Particle)               break;            MeshRenderInst *passRI = static_cast<MeshRenderInst*>(mElementList[a].inst);            if ( newPassNeeded( ri, passRI ) )               break;            matrixSet.setWorld(*passRI->objectToWorld);            matrixSet.setView(*passRI->worldToCamera);            matrixSet.setProjection(*passRI->projection);            glowMat->setTransforms(matrixSet, state);            glowMat->setSceneInfo(state, sgData);            glowMat->setBuffers(passRI->vertBuff, passRI->primBuff);            if ( passRI->prim )               GFX->drawPrimitive( *passRI->prim );            else               GFX->drawPrimitive( passRI->primBuffIndex );         }         matListEnd = a;         setupSGData( ri, sgData );      }      // force increment if none happened, otherwise go to end of batch      j = ( j == matListEnd ) ? j+1 : matListEnd;   }   // Finish up.   if ( isRenderingToTarget )      _onPostRender();   // Make sure the effect is gonna render.   getGlowEffect()->setSkip( false );}
 |