You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
632 lines
16 KiB
C++
632 lines
16 KiB
C++
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
|
// Copyright (C) 2010 Winch Gate Property Limited
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
// License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#include "std3d.h"
|
|
|
|
#include "nel/3d/material.h"
|
|
#include "nel/3d/texture.h"
|
|
#include "nel/3d/driver.h"
|
|
#include "nel/misc/stream.h"
|
|
|
|
using namespace std;
|
|
using namespace NLMISC;
|
|
|
|
#ifdef DEBUG_NEW
|
|
#define new DEBUG_NEW
|
|
#endif
|
|
|
|
namespace NL3D
|
|
{
|
|
|
|
// ***************************************************************************
|
|
CMaterial::CMaterial()
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
_Touched= 0;
|
|
_Flags= IDRV_MAT_ZWRITE;
|
|
// Must init All the flags by default.
|
|
_ShaderType= Normal;
|
|
_SrcBlend= srcalpha;
|
|
_DstBlend= invsrcalpha;
|
|
_ZFunction= lessequal;
|
|
_ZBias= 0;
|
|
_Color.set(255,255,255,255);
|
|
_StainedGlassWindow = false;
|
|
_AlphaTestThreshold= 0.5f;
|
|
_TexCoordGenMode= 0;
|
|
_LightMapsMulx2= false;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::initUnlit()
|
|
{
|
|
setShader(Normal);
|
|
setLighting(false);
|
|
setColor(CRGBA(255,255,255,255));
|
|
for(uint32 i=0;i<IDRV_MAT_MAXTEXTURES;i++)
|
|
setTexture((uint8)i ,NULL);
|
|
setZBias(0);
|
|
setZFunc(lessequal);
|
|
setZWrite(true);
|
|
setBlend(false);
|
|
setAlphaTestThreshold(0.5f);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
void CMaterial::initLighted()
|
|
{
|
|
initUnlit();
|
|
setLighting(true);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
CMaterial &CMaterial::operator=(const CMaterial &mat)
|
|
{
|
|
_ShaderType= mat._ShaderType;
|
|
_Flags= mat._Flags;
|
|
_SrcBlend= mat._SrcBlend;
|
|
_DstBlend= mat._DstBlend;
|
|
_ZFunction= mat._ZFunction;
|
|
_ZBias= mat._ZBias;
|
|
_Color= mat._Color;
|
|
_Emissive= mat._Emissive;
|
|
_Ambient= mat._Ambient;
|
|
_Diffuse= mat._Diffuse;
|
|
_Specular= mat._Specular;
|
|
_Shininess= mat._Shininess;
|
|
_AlphaTestThreshold= mat._AlphaTestThreshold;
|
|
_TexCoordGenMode= mat._TexCoordGenMode;
|
|
|
|
for(uint32 i=0;i<IDRV_MAT_MAXTEXTURES;i++)
|
|
{
|
|
_Textures[i]= mat._Textures[i];
|
|
_TexEnvs[i]= mat._TexEnvs[i];
|
|
_TexAddrMode[i] = mat._TexAddrMode[i];
|
|
}
|
|
|
|
// copy lightmaps.
|
|
_LightMaps= mat._LightMaps;
|
|
_LightMapsMulx2= mat._LightMapsMulx2;
|
|
|
|
// copy texture matrix if there.
|
|
if (mat._TexUserMat.get())
|
|
{
|
|
CUniquePtr<CUserTexMat> texMatClone(new CUserTexMat(*(mat._TexUserMat))); // make cpy
|
|
//std::swap(texMatClone, _TexUserMat); // swap with old
|
|
_TexUserMat = CUniquePtrMove(texMatClone);
|
|
}
|
|
else
|
|
{
|
|
_TexUserMat.reset();
|
|
}
|
|
|
|
// Must do not copy drv info.
|
|
|
|
// All states of material is modified.
|
|
_Touched= IDRV_TOUCHED_ALL;
|
|
|
|
return *this;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
CMaterial::~CMaterial()
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
// Must kill the drv mirror of this material.
|
|
_MatDrvInfo.kill();
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::serial(NLMISC::IStream &f)
|
|
{
|
|
/* ***********************************************
|
|
* WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance
|
|
* It can be loaded/called through CAsyncFileManager for instance
|
|
* ***********************************************/
|
|
|
|
/*
|
|
Version 9:
|
|
- Added support for third operand (for Mad operator)
|
|
Version 8:
|
|
- Serial _TexCoordGenMode
|
|
Version 7:
|
|
- Lightmap color and Mulx2
|
|
Version 6:
|
|
- Texture matrix animation
|
|
Version 5:
|
|
- AlphaTest threshold
|
|
Version 4:
|
|
- Texture Addressing modes
|
|
Version 3:
|
|
- LightMaps.
|
|
Version 2:
|
|
- Shininess.
|
|
Version 1:
|
|
- texture environement.
|
|
Version 0:
|
|
- base version.
|
|
*/
|
|
|
|
sint ver= f.serialVersion(9);
|
|
// For the version <=1:
|
|
nlassert(IDRV_MAT_MAXTEXTURES==4);
|
|
|
|
f.serialEnum(_ShaderType);
|
|
f.serial(_Flags);
|
|
f.serialEnum(_SrcBlend);
|
|
f.serialEnum(_DstBlend);
|
|
f.serialEnum(_ZFunction);
|
|
f.serial(_ZBias);
|
|
f.serial(_Color);
|
|
f.serial(_Emissive, _Ambient, _Diffuse, _Specular);
|
|
if(ver>=2)
|
|
{
|
|
f.serial(_Shininess);
|
|
}
|
|
if(ver>=5)
|
|
{
|
|
f.serial(_AlphaTestThreshold);
|
|
}
|
|
if(ver>=8)
|
|
{
|
|
f.serial(_TexCoordGenMode);
|
|
}
|
|
else
|
|
_TexCoordGenMode = 0;
|
|
|
|
|
|
for(uint32 i=0;i<IDRV_MAT_MAXTEXTURES;i++)
|
|
{
|
|
// Serial texture descriptor.
|
|
_Textures[i].serialPolyPtr(f);
|
|
|
|
// Read texture environnement, or setup them.
|
|
if(ver>=1)
|
|
{
|
|
_TexEnvs[i].serial(f, ver >= 9 ? 1 : 0);
|
|
}
|
|
else
|
|
{
|
|
// Else setup as default behavior, like before...
|
|
if(f.isReading())
|
|
_TexEnvs[i].setDefault();
|
|
}
|
|
}
|
|
|
|
if(ver>=3)
|
|
{
|
|
if(ver>=7)
|
|
{
|
|
uint32 n;
|
|
if (f.isReading())
|
|
{
|
|
f.serial(n);
|
|
_LightMaps.resize(n);
|
|
}
|
|
else
|
|
{
|
|
n = (uint32)_LightMaps.size();
|
|
f.serial(n);
|
|
}
|
|
for (uint32 i = 0; i < n; ++i)
|
|
_LightMaps[i].serial2(f);
|
|
f.serial(_LightMapsMulx2);
|
|
}
|
|
else
|
|
{
|
|
f.serialCont(_LightMaps);
|
|
}
|
|
}
|
|
|
|
if (ver >= 4)
|
|
{
|
|
if (_Flags & IDRV_MAT_TEX_ADDR)
|
|
{
|
|
for(uint32 i=0;i<IDRV_MAT_MAXTEXTURES;i++)
|
|
{
|
|
f.serial(_TexAddrMode[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(f.isReading())
|
|
{
|
|
// Converte Deprecated DEFMAT to std Mat.
|
|
if(_Flags & IDRV_MAT_DEFMAT)
|
|
{
|
|
setEmissive(CRGBA::Black);
|
|
setAmbient(CRGBA::White);
|
|
setDiffuse(CRGBA::White);
|
|
setSpecular(CRGBA::Black);
|
|
}
|
|
|
|
// All states of material are modified.
|
|
_Touched= IDRV_TOUCHED_ALL;
|
|
|
|
if ((_Flags & IDRV_MAT_USER_TEX_MAT_ALL)) // are there user textrue coordinates matrix ?
|
|
{
|
|
CUniquePtr<CUserTexMat> newPtr(new CUserTexMat); // create new
|
|
//std::swap(_TexUserMat, newPtr); // replace old
|
|
_TexUserMat = CUniquePtrMove(newPtr);
|
|
}
|
|
}
|
|
|
|
if (ver >= 6)
|
|
{
|
|
for(uint i=0; i < IDRV_MAT_MAXTEXTURES; ++i)
|
|
{
|
|
if (isUserTexMatEnabled(i))
|
|
{
|
|
f.serial(_TexUserMat->TexMat[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setShader(TShader val)
|
|
{
|
|
// First, reset all textures.
|
|
uint nTexts= IDRV_MAT_MAXTEXTURES;
|
|
// If user color or lightmap, set only the 1st.
|
|
if(_ShaderType==LightMap || _ShaderType==UserColor)
|
|
nTexts=1;
|
|
// reset all needed
|
|
for(uint i=0;i<nTexts;i++)
|
|
setTexture(i ,NULL);
|
|
|
|
// If userColor, use TexEnv caps (we got it, so use it :) ).
|
|
if(val== CMaterial::UserColor)
|
|
{
|
|
// force normal, to setup TexEnvMode correclty.
|
|
_ShaderType=CMaterial::Normal;
|
|
|
|
// First stage, interpolate Constant and texture with Alpha of texture.
|
|
texEnvOpRGB(0, InterpolateTexture);
|
|
texEnvArg0RGB(0, Texture, SrcColor);
|
|
texEnvArg1RGB(0, Constant, SrcColor);
|
|
// And just use Alpha Diffuse.
|
|
texEnvOpAlpha(0, Replace);
|
|
texEnvArg0Alpha(0, Previous, SrcAlpha);
|
|
|
|
// Second stage, modulate result with diffuse color.
|
|
texEnvOpRGB(1, Modulate);
|
|
texEnvArg0RGB(1, Previous, SrcColor);
|
|
texEnvArg1RGB(1, Diffuse, SrcColor);
|
|
// And just use Alpha Diffuse.
|
|
texEnvOpAlpha(1, Replace);
|
|
texEnvArg0Alpha(1, Previous, SrcAlpha);
|
|
}
|
|
|
|
_ShaderType= val;
|
|
_Touched|=IDRV_TOUCHED_SHADER;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setTexture(uint8 n, ITexture* ptex)
|
|
{
|
|
nlassert(n<IDRV_MAT_MAXTEXTURES);
|
|
|
|
// User Color material?
|
|
if( _ShaderType== CMaterial::UserColor)
|
|
{
|
|
// user color. Only texture 0 can be set.
|
|
nlassert( n==0 );
|
|
|
|
// Affect the 2 first textures.
|
|
_Textures[0]=ptex;
|
|
_Textures[1]=ptex;
|
|
_Touched|=IDRV_TOUCHED_TEX[0];
|
|
_Touched|=IDRV_TOUCHED_TEX[1];
|
|
}
|
|
else if( _ShaderType== CMaterial::LightMap)
|
|
{
|
|
// Only texture 0 can be set.
|
|
nlassert( n==0 );
|
|
_Textures[n]=ptex;
|
|
_Touched|=IDRV_TOUCHED_TEX[n];
|
|
}
|
|
// Normal material?
|
|
else
|
|
{
|
|
_Textures[n]=ptex;
|
|
_Touched|=IDRV_TOUCHED_TEX[n];
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::flushTextures (IDriver &driver, uint selectedTexture)
|
|
{
|
|
// For each textures
|
|
for (uint tex=0; tex<IDRV_MAT_MAXTEXTURES; tex++)
|
|
{
|
|
// Texture exist ?
|
|
if (_Textures[tex])
|
|
{
|
|
// Select the good texture
|
|
_Textures[tex]->selectTexture (selectedTexture);
|
|
|
|
// Force setup texture
|
|
driver.setupTexture (*_Textures[tex]);
|
|
}
|
|
}
|
|
|
|
// If Lightmap material
|
|
if(_ShaderType==LightMap)
|
|
{
|
|
// For each lightmap
|
|
for (uint lmap=0; lmap<_LightMaps.size(); lmap++)
|
|
{
|
|
// Texture exist?
|
|
if(_LightMaps[lmap].Texture)
|
|
{
|
|
// Force setup texture
|
|
driver.setupTexture (*_LightMaps[lmap].Texture);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setLightMap(uint lmapId, ITexture *lmap)
|
|
{
|
|
nlassert(_ShaderType==CMaterial::LightMap);
|
|
if(lmapId>=_LightMaps.size())
|
|
_LightMaps.resize(lmapId+1);
|
|
_LightMaps[lmapId].Texture= lmap;
|
|
|
|
_Touched|=IDRV_TOUCHED_LIGHTMAP;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
ITexture *CMaterial::getLightMap(uint lmapId) const
|
|
{
|
|
nlassert(_ShaderType==CMaterial::LightMap);
|
|
if(lmapId<_LightMaps.size())
|
|
return _LightMaps[lmapId].Texture;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setLightMapFactor(uint lmapId, CRGBA factor)
|
|
{
|
|
if (_ShaderType==CMaterial::LightMap)
|
|
{
|
|
if(lmapId>=_LightMaps.size())
|
|
_LightMaps.resize(lmapId+1);
|
|
_LightMaps[lmapId].Factor= factor;
|
|
|
|
_Touched|=IDRV_TOUCHED_LIGHTMAP;
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setLMCColors(uint lmapId, CRGBA ambColor, CRGBA diffColor)
|
|
{
|
|
if (_ShaderType==CMaterial::LightMap)
|
|
{
|
|
if(lmapId>=_LightMaps.size())
|
|
_LightMaps.resize(lmapId+1);
|
|
_LightMaps[lmapId].LMCAmbient= ambColor;
|
|
_LightMaps[lmapId].LMCDiffuse= diffColor;
|
|
|
|
_Touched|=IDRV_TOUCHED_LIGHTMAP;
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// DEPRECATED VERSION
|
|
void CMaterial::CLightMap::serial(NLMISC::IStream &f)
|
|
{
|
|
f.serial(Factor);
|
|
// Serial texture descriptor.
|
|
Texture.serialPolyPtr(f);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::CLightMap::serial2(NLMISC::IStream &f)
|
|
{
|
|
sint ver= f.serialVersion(1);
|
|
|
|
f.serial(Factor);
|
|
f.serial(LMCDiffuse);
|
|
if(ver>=1)
|
|
f.serial(LMCAmbient);
|
|
// Serial texture descriptor.
|
|
Texture.serialPolyPtr(f);
|
|
}
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::enableTexAddrMode(bool enable /*= true*/)
|
|
{
|
|
if (enable)
|
|
{
|
|
if (!(_Flags & IDRV_MAT_TEX_ADDR))
|
|
{
|
|
_Flags |= IDRV_MAT_TEX_ADDR;
|
|
for (uint32 k = 0; k < IDRV_MAT_MAXTEXTURES; ++k)
|
|
{
|
|
_TexAddrMode[k] = (uint8) TextureOff;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_Flags &= ~IDRV_MAT_TEX_ADDR;
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CMaterial::texAddrEnabled() const
|
|
{
|
|
return( _Flags & IDRV_MAT_TEX_ADDR) != 0;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::setTexAddressingMode(uint8 stage, TTexAddressingMode mode)
|
|
{
|
|
nlassert(_Flags & IDRV_MAT_TEX_ADDR);
|
|
nlassert(stage < IDRV_MAT_MAXTEXTURES);
|
|
nlassert(mode < TexAddrCount);
|
|
_TexAddrMode[stage] = (uint8) mode;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
CMaterial::TTexAddressingMode CMaterial::getTexAddressingMode(uint8 stage)
|
|
{
|
|
nlassert(_Flags & IDRV_MAT_TEX_ADDR);
|
|
nlassert(stage < IDRV_MAT_MAXTEXTURES);
|
|
return (TTexAddressingMode) _TexAddrMode[stage];
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::decompUserTexMat(uint stage, float &uTrans, float &vTrans, float &wRot, float &uScale, float &vScale)
|
|
{
|
|
nlassert(stage < IDRV_MAT_MAXTEXTURES);
|
|
nlassert(isUserTexMatEnabled(stage)); // must activate animated texture matrix for this stage
|
|
CMatrix convMat; // exported v are already inverted (todo: optim this...)
|
|
convMat.setRot(CVector::I, -CVector::J, CVector::K);
|
|
convMat.setPos(CVector::J);
|
|
|
|
const NLMISC::CMatrix texMat = convMat * _TexUserMat->TexMat[stage] * convMat;
|
|
/// find the rotation around w
|
|
NLMISC::CVector i = texMat.getI();
|
|
NLMISC::CVector j = texMat.getJ();
|
|
uScale = sqrtf(i.x * i.x + j.x * j.x);
|
|
vScale = sqrtf(i.y * i.y + j.y * j.y);
|
|
//
|
|
i.normalize();
|
|
//
|
|
float angle = acosf(i.x / i.norm());
|
|
if (i.y < 0)
|
|
{
|
|
angle = 2.f * (float) NLMISC::Pi - angle;
|
|
}
|
|
wRot = angle;
|
|
|
|
// compute position
|
|
CMatrix InvSR;
|
|
InvSR.setRot(texMat.getI(), texMat.getJ(), texMat.getK());
|
|
InvSR.invert();
|
|
CVector half(0.5f, 0.5f, 0.f);
|
|
CVector offset = half + InvSR * (texMat.getPos() -half);
|
|
uTrans = - offset.x;
|
|
vTrans = - offset.y;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CMaterial::selectTextureSet(uint index)
|
|
{
|
|
for (uint k = 0; k < IDRV_MAT_MAXTEXTURES; ++k)
|
|
{
|
|
if (_Textures[k] != NULL) _Textures[k]->selectTexture(index);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
IMaterialDrvInfos::~IMaterialDrvInfos()
|
|
{
|
|
_Driver->removeMatDrvInfoPtr(_DriverIterator);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
uint CMaterial::getNumUsedTextureStages() const
|
|
{
|
|
for(uint k = 0; k < IDRV_MAT_MAXTEXTURES; ++k)
|
|
{
|
|
if (!_Textures[k]) return k;
|
|
}
|
|
return IDRV_MAT_MAXTEXTURES;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
bool CMaterial::isSupportedByDriver(IDriver &drv, bool forceBaseCaps) const
|
|
{
|
|
uint numTexStages = drv.getNbTextureStages();
|
|
// special case for radeon : though 3 stages are supported, do as if there were only 2, because of the texEnvColor feature
|
|
// not managed in Direct3D : emulation is provided, but for no more than 2 constants (and if diffuse is not used)
|
|
if (numTexStages == 3) numTexStages = 2;
|
|
if (forceBaseCaps) numTexStages = std::min(numTexStages, (uint) 2);
|
|
switch(getShader())
|
|
{
|
|
case Normal:
|
|
{
|
|
if (getNumUsedTextureStages() > numTexStages) return false;
|
|
// see if each tex env is supported
|
|
for(uint k = 0; k < IDRV_MAT_MAXTEXTURES; ++k)
|
|
{
|
|
if (getTexture(k))
|
|
{
|
|
switch(getTexEnvOpRGB(k))
|
|
{
|
|
case InterpolateConstant: if (!drv.supportBlendConstantColor()) return false;
|
|
case EMBM: if (forceBaseCaps || !drv.supportEMBM() || !drv.isEMBMSupportedAtStage(k)) return false;
|
|
case Mad: if (!drv.supportMADOperator()) return false;
|
|
default: break;
|
|
}
|
|
switch(getTexEnvOpAlpha(k))
|
|
{
|
|
case InterpolateConstant: if (!drv.supportBlendConstantColor()) return false;
|
|
case EMBM: if (forceBaseCaps || !drv.supportEMBM() || !drv.isEMBMSupportedAtStage(k)) return false;
|
|
case Mad: if (!drv.supportMADOperator()) return false;
|
|
default: break;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
break;
|
|
case Bump: return false; // not impl.
|
|
case UserColor: return true;
|
|
case LightMap: return true;
|
|
case Specular: return true;
|
|
case Caustics: return false;
|
|
case PerPixelLighting: return drv.supportPerPixelLighting(true);
|
|
case PerPixelLightingNoSpec: return drv.supportPerPixelLighting(false);
|
|
case Cloud: return true;
|
|
case Water: return true;
|
|
default:
|
|
nlassert(0); // unknown shader, must complete
|
|
}
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|