// NeL - MMORPG Framework
// 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 .
#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 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=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 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;iselectTexture (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;
}
}