Torque3D/Engine/source/scene/reflectionMatHook.cpp
Azaezel 8c5810adad The final step (barring any overlooked missing bits, requested refactors, and of course, rolling in dependencies already submitted as PRs) consists of:
renderPrePassMgr.cpp related:
A) shifting .addFeature( MFT_XYZ); calls from ProcessedShaderMaterial::_determineFeatures to ProcessedPrePassMaterial::_determineFeatures
B) mimicking the "// set the XXX if different" entries from RenderMeshMgr::render in RenderPrePassMgr::render
C) fleshing out ProcessedPrePassMaterial::getNumStages() so that it shares a 1:1 correlation with ProcessedShaderMaterial::getNumStages()
D) causing inline void Swizzle<T, mapLength>::ToBuffer( void *destination, const void *source, const dsize_t size )  to silently fail rather than fatally assert if a source or destination buffer is not yet ready to be filled. (support for #customTarget scripted render targets)

Reflections:
A) removing reflectRenderState.disableAdvancedLightingBins(true); entries. this would otherwise early out from prepass and provide no color data whatsoever.
B) removing the fd.features.addFeature( MFT_ForwardShading ); entry forcing all materials to be forward lit when reflected.
C) 2 things best described bluntly as working hacks:
C1) when reflected, a scattersky is rotated PI along it's z then x axis in order to draw properly.
C2) along similar lines, in terraincellmaterial, we shut off culling if it's a prepass material.

Skies: scattersky is given a pair of rotations for reflection purposes, all sky objects are given a z value for depth testing.
2016-02-16 02:50:49 -06:00

107 lines
3.6 KiB
C++

//-----------------------------------------------------------------------------
// 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 "scene/reflectionMatHook.h"
#include "materials/materialManager.h"
#include "materials/customMaterialDefinition.h"
#include "materials/materialFeatureTypes.h"
#include "materials/materialFeatureData.h"
#include "shaderGen/featureType.h"
#include "shaderGen/featureMgr.h"
#include "scene/sceneRenderState.h"
const MatInstanceHookType ReflectionMaterialHook::Type( "Reflection" );
ReflectionMaterialHook::ReflectionMaterialHook() :
mReflectMat(NULL)
{
}
ReflectionMaterialHook::~ReflectionMaterialHook()
{
SAFE_DELETE(mReflectMat);
}
void ReflectionMaterialHook::init( BaseMatInstance *inMat )
{
if( !inMat->isValid() )
return;
Material *reflectMat = (Material*)inMat->getMaterial();
if ( inMat->isCustomMaterial() )
{
// This is a custom material... who knows what it really does...do something
// smart here later.
}
// We may want to disable some states that the material might enable for us.
GFXStateBlockDesc refractState = inMat->getUserStateBlock();
// Always z-read, and z-write if the material isn't translucent
refractState.setZReadWrite( true, reflectMat->isTranslucent() ? false : true );
// Create reflection material instance.
BaseMatInstance *newMat = new ReflectionMatInstance( reflectMat );
newMat->setUserObject( inMat->getUserObject() );
newMat->getFeaturesDelegate().bind( &ReflectionMaterialHook::_overrideFeatures );
newMat->addStateBlockDesc( refractState );
if( !newMat->init( inMat->getFeatures(), inMat->getVertexFormat() ) )
{
SAFE_DELETE( newMat );
newMat = MATMGR->createWarningMatInstance();
}
mReflectMat = newMat;
}
void ReflectionMaterialHook::_overrideFeatures( ProcessedMaterial *mat,
U32 stageNum,
MaterialFeatureData &fd,
const FeatureSet &features )
{
// First stage only in reflections
if( stageNum != 0 )
{
fd.features.clear();
return;
}
fd.features.addFeature( MFT_Fog );
}
//------------------------------------------------------------------------------
ReflectionMatInstance::ReflectionMatInstance( Material *mat )
: MatInstance( *mat )
{
}
bool ReflectionMatInstance::setupPass( SceneRenderState *state, const SceneData &sgData )
{
return Parent::setupPass(state, sgData);
}