// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This source file has been modified by the following contributors: // Copyright (C) 2012 Jan BOON (Kaetemi) // // 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/point_light.h" #include "nel/3d/light.h" #include "nel/3d/transform.h" #include "nel/misc/common.h" using namespace NLMISC; using namespace std; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** //NLMISC::CBlockMemory CPointLight::_LightedModelListMemory(NL3D_LIGHTED_MODEL_ALLOC_BLOCKSIZE); // *************************************************************************** CPointLight::CPointLight() : _LightedModels(/*&_LightedModelListMemory*/) { _Position= CVector::Null; _Ambient= CRGBA::Black; _Diffuse= _Specular= CRGBA::White; // Default setup. this is arbitrary _Type= PointLight; _AttenuationBegin= 10; _AttenuationEnd= 30; // Default spot setup. this is arbitrary _SpotDirection.set(0,1,0); _SpotAngleBegin= float(NLMISC::Pi/4); _SpotAngleEnd= float(NLMISC::Pi/2); // compute AttenuationFactors only one time. static bool done= false; static float cAtt, lAtt, qAtt; static float spotCOOD, spotCAE, spotEXP; if(!done) { done= true; computeAttenuationFactors(); computeSpotAttenuationFactors(); // bkup setup. cAtt= _ConstantAttenuation; lAtt= _LinearAttenuation; qAtt= _QuadraticAttenuation; spotCAE= _CosSpotAngleEnd; spotCOOD= _OOCosSpotAngleDelta; spotEXP= _SpotExponent; } else { // just copy bkuped setup. _ConstantAttenuation= cAtt; _LinearAttenuation= lAtt; _QuadraticAttenuation= qAtt; _CosSpotAngleEnd= spotCAE; _OOCosSpotAngleDelta= spotCOOD; _SpotExponent= spotEXP; } _AddAmbientWithSun= false; } // *************************************************************************** CPointLight::CPointLight(const CPointLight &o) : _LightedModels(/*&_LightedModelListMemory*/) { // copy (no need to init) operator=(o); } // *************************************************************************** CPointLight &CPointLight::operator=(const CPointLight &o) { /// copy all but _LightedModels !! _Type= o._Type; _Position= o._Position; _Ambient= o._Ambient; _Diffuse= o._Diffuse; _Specular= o._Specular; _AttenuationBegin= o._AttenuationBegin; _AttenuationEnd= o._AttenuationEnd; _OODeltaAttenuation= o._OODeltaAttenuation; _ConstantAttenuation= o._ConstantAttenuation; _LinearAttenuation= o._LinearAttenuation; _QuadraticAttenuation= o._QuadraticAttenuation; _SpotDirection= o._SpotDirection; _SpotAngleBegin= o._SpotAngleBegin; _SpotAngleEnd= o._SpotAngleEnd; _CosSpotAngleEnd= o._CosSpotAngleEnd; _OOCosSpotAngleDelta= o._OOCosSpotAngleDelta; _SpotExponent= o._SpotExponent; _AddAmbientWithSun= o._AddAmbientWithSun; return *this; } // *************************************************************************** CPointLight::~CPointLight() { resetLightedModels(); } // *************************************************************************** void CPointLight::setType(TType type) { _Type= type; } CPointLight::TType CPointLight::getType() const { return _Type; } // *************************************************************************** void CPointLight::setupAttenuation(float attenuationBegin, float attenuationEnd) { // set values. attenuationBegin= max(attenuationBegin, 0.f); attenuationEnd= max(attenuationEnd, attenuationBegin); _AttenuationBegin= attenuationBegin; _AttenuationEnd= attenuationEnd; // update factors. computeAttenuationFactors(); } // *************************************************************************** void CPointLight::setupSpotAngle(float spotAngleBegin, float spotAngleEnd) { clamp(spotAngleBegin, 0.f, float(Pi)); clamp(spotAngleEnd, spotAngleBegin, float(Pi)); _SpotAngleBegin= spotAngleBegin; _SpotAngleEnd= spotAngleEnd; // update factors. computeSpotAttenuationFactors(); } // *************************************************************************** void CPointLight::setupSpotDirection(const CVector &dir) { _SpotDirection= dir; _SpotDirection.normalize(); } // *************************************************************************** void CPointLight::computeAttenuationFactors() { // disable attenuation? if(_AttenuationBegin==0 && _AttenuationEnd==0) { // setup for attenuation disabled. _ConstantAttenuation= 1; _LinearAttenuation= 0; _QuadraticAttenuation= 0; } else { // precompute attenuation values, with help of CLight formula!! CLight dummyLight; dummyLight.setupAttenuation(_AttenuationBegin, _AttenuationEnd); _ConstantAttenuation= dummyLight.getConstantAttenuation(); _LinearAttenuation= dummyLight.getLinearAttenuation(); _QuadraticAttenuation= dummyLight.getQuadraticAttenuation(); // setup _OODeltaAttenuation _OODeltaAttenuation= _AttenuationEnd - _AttenuationBegin; if(_OODeltaAttenuation <=0 ) _OODeltaAttenuation= 0; else _OODeltaAttenuation= 1.0f / _OODeltaAttenuation; } } // *************************************************************************** void CPointLight::computeSpotAttenuationFactors() { // Factors for linear Attenuation. float cosSpotAngleBegin= (float)cosf(_SpotAngleBegin); _CosSpotAngleEnd= (float)cos(_SpotAngleEnd); if(cosSpotAngleBegin - _CosSpotAngleEnd > 0) _OOCosSpotAngleDelta= 1.0f / (cosSpotAngleBegin - _CosSpotAngleEnd); else _OOCosSpotAngleDelta= 1e10f; // compute an exponent such that at middleAngle, att is 0.5f. float caMiddle= (cosSpotAngleBegin + _CosSpotAngleEnd) /2; float divid=(float)log (caMiddle); if (divid==0.f) divid=0.0001f; _SpotExponent= (float)(log (0.5)/divid); } // *************************************************************************** void CPointLight::serial(NLMISC::IStream &f) { sint ver= f.serialVersion(2); if(ver>=2) f.serial(_AddAmbientWithSun); else _AddAmbientWithSun= false; if(ver>=1) { f.serialEnum(_Type); f.serial(_SpotDirection); f.serial(_SpotAngleBegin); f.serial(_SpotAngleEnd); } else if(f.isReading()) { _Type= PointLight; _SpotDirection.set(0,1,0); _SpotAngleBegin= float(NLMISC::Pi/4); _SpotAngleEnd= float(NLMISC::Pi/2); } f.serial(_Position); f.serial(_Ambient); f.serial(_Diffuse); f.serial(_Specular); f.serial(_AttenuationBegin); f.serial(_AttenuationEnd); // precompute. if(f.isReading()) { computeAttenuationFactors(); computeSpotAttenuationFactors(); } } // *************************************************************************** float CPointLight::computeLinearAttenuation(const CVector &pos) const { return computeLinearAttenuation(pos, (pos - _Position).norm() ); } // *************************************************************************** float CPointLight::computeLinearAttenuation(const CVector &pos, float dist, float modelRadius) const { float gAtt; // Attenuation Distance if(_AttenuationEnd==0) gAtt= 1; else { float distMinusRadius= dist - modelRadius; if(distMinusRadius<_AttenuationBegin) gAtt= 1; else if(distMinusRadius<_AttenuationEnd) { gAtt= (_AttenuationEnd - distMinusRadius) * _OODeltaAttenuation; } else gAtt= 0; } // Spot Attenuation if(_Type== SpotLight) { float spotAtt; // Compute unnormalized direction CVector dir= pos - _Position; // get cosAngle(dir, SpotDirection): float cosAngleDirSpot= (dir*_SpotDirection) / dist; // Modify with modelRadius. NB: made Only for big models. if(modelRadius>0) { // If the pointLight is in the model, consider no spotAtt if(modelRadius > dist) spotAtt= 1; else { // compute the angle of the cone made by the model sphere and the pointLightCenter. float cosAngleSphere= modelRadius / sqrtf( sqr(dist) + sqr(modelRadius) ); /* If this one is smaller than cosAngleDirSpot, it's mean that the angle of this cone is greater than the angleDirSpot, hence a part of the sphere "ps" exist such that _SportDirection*(ps-_Position).normed() == 1 => no spotAttenuation */ if(cosAngleSphere < cosAngleDirSpot) spotAtt= 1; else { // Must compute cos( AngleDirSpot-AngleSphere ) float sinAngleSphere= sqrtf(1 - sqr(cosAngleSphere)); float sinAngleDirSpot= sqrtf(1 - sqr(cosAngleDirSpot)); float cosDelta= cosAngleSphere * cosAngleDirSpot + sinAngleSphere * sinAngleDirSpot; // spot attenuation on the exterior of the sphere spotAtt= (cosDelta - _CosSpotAngleEnd) * _OOCosSpotAngleDelta; } } } else { // spot attenuation spotAtt= (cosAngleDirSpot - _CosSpotAngleEnd) * _OOCosSpotAngleDelta; } // modulate clamp(spotAtt, 0.f, 1.f); gAtt*= spotAtt; } return gAtt; } // *************************************************************************** void CPointLight::setupDriverLight(CLight &light, uint8 factor) { // expand 0..255 to 0..256, to avoid loss of precision. uint ufactor= factor + (factor>>7); // add 0 or 1. // modulate with factor CRGBA ambient, diffuse, specular; ambient.modulateFromuiRGBOnly(_Ambient, ufactor); diffuse.modulateFromuiRGBOnly(_Diffuse, ufactor); specular.modulateFromuiRGBOnly(_Specular, ufactor); // setup the pointLight if(_Type == SpotLight ) { light.setupSpotLight(ambient, diffuse, specular, _Position, _SpotDirection, _SpotExponent, float(NLMISC::Pi/2) , _ConstantAttenuation, _LinearAttenuation, _QuadraticAttenuation); } // PointLight or AmbientLight else { light.setupPointLight(ambient, diffuse, specular, _Position, CVector::Null, _ConstantAttenuation, _LinearAttenuation, _QuadraticAttenuation); } } // *************************************************************************** void CPointLight::setupDriverLightUserAttenuation(CLight &light, uint8 factor) { // expand 0..255 to 0..256, to avoid loss of precision. uint ufactor= factor + (factor>>7); // add 0 or 1. // modulate with factor CRGBA ambient, diffuse, specular; ambient.modulateFromuiRGBOnly(_Ambient, ufactor); diffuse.modulateFromuiRGBOnly(_Diffuse, ufactor); specular.modulateFromuiRGBOnly(_Specular, ufactor); // setup the pointLight, disabling attenuation. // NB: setup a pointLight even if it is a SpotLight because already attenuated light.setupPointLight(ambient, diffuse, specular, _Position, CVector::Null, 1, 0, 0); } // *************************************************************************** void CPointLight::resetLightedModels() { // For each transform, resetLighting him. while(_LightedModels.begin() != _LightedModels.end() ) { CTransform *model= *_LightedModels.begin(); // reset lighting model->resetLighting(); // NB: the transform must erase him from this list. nlassert( _LightedModels.begin() == _LightedModels.end() || *_LightedModels.begin() != model ); } } // *************************************************************************** CPointLight::ItTransformList CPointLight::appendLightedModel(CTransform *model) { // append the entry in the list _LightedModels.push_back(model); ItTransformList it= _LightedModels.end(); it--; return it; } // *************************************************************************** void CPointLight::removeLightedModel(ItTransformList it) { // delete the entry in the list. _LightedModels.erase(it); } // *************************************************************************** void CPointLight::purge () { //_LightedModelListMemory.purge(); } // *************************************************************************** void CPointLight::setAddAmbientWithSun(bool state) { _AddAmbientWithSun= state; } } // NL3D