// 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/texture_emboss.h"
using namespace NLMISC;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NL3D {
// ***********************************************************************************************************
CTextureEmboss::CTextureEmboss() : _Ambient(CRGBA::Black),
_Diffuse(CRGBA::White),
_DisableSharing(false),
_SlopeFactor(1.f)
{
_LightDir.set(1.f, 1.f, 1.f);
_LightDir.normalize();
}
// ***********************************************************************************************************
void CTextureEmboss::setHeightMap(ITexture *heightMap)
{
if (heightMap != _HeightMap)
{
_HeightMap = heightMap;
touch();
}
}
// ***********************************************************************************************************
void CTextureEmboss::serial(NLMISC::IStream &f)
{
f.serialVersion(0);
ITexture::serial(f);
ITexture *tex = NULL;
if (f.isReading())
{
f.serialPolyPtr(tex);
_HeightMap = tex;
touch();
}
else
{
tex = _HeightMap;
f.serialPolyPtr(tex);
}
f.serial(_Ambient);
f.serial(_Diffuse);
f.serial(_LightDir);
f.serial(_DisableSharing);
}
// ***********************************************************************************************************
bool CTextureEmboss::supportSharing() const
{
return !_DisableSharing && _HeightMap && _HeightMap->supportSharing();
}
// ***********************************************************************************************************
std::string CTextureEmboss::getShareName() const
{
nlassert(supportSharing());
return "Emboss:" + _HeightMap->getShareName();
}
// ***********************************************************************************************************
void CTextureEmboss::release()
{
ITexture::release();
if (_HeightMap != NULL)
{
if (_HeightMap->getReleasable())
{
_HeightMap->release();
}
}
}
// ***********************************************************************************************************
void CTextureEmboss::doGenerate(bool /* async */)
{
if (!_HeightMap)
{
makeDummy();
return;
}
// generate the height map
_HeightMap->generate();
if (!_HeightMap->convertToType(CBitmap::RGBA))
{
makeDummy();
return;
}
CBitmap::resize(_HeightMap->getWidth(), _HeightMap->getHeight(), CBitmap::RGBA);
releaseMipMaps();
uint width = _HeightMap->getWidth();
uint height = _HeightMap->getHeight();
const CRGBA *src = (CRGBA *) &(_HeightMap->getPixels(0)[0]);
CRGBA *dest = (CRGBA *) &(getPixels(0)[0]);
CVector normal;
for (uint y = 0; y < height; ++y)
{
for (uint x = 0; x < width; ++x)
{
// get position of each adajacent pixel (sure, it can be optimized)
const CRGBA *pixelRight = (x != width - 1) ? (src + 1) : (src + 1 - width );
const CRGBA *pixelLeft = (x != 0) ? (src - 1) : (src + width - 1);
const CRGBA *pixelTop = (y != 0) ? (src - width) : (src + width * (height - 1));
const CRGBA *pixelBottom = (y != (height - 1)) ? (src + width) : (src - width * (height - 1));
normal.x = ((sint16) pixelRight->R - (sint16) pixelLeft->R) / 256.f;
normal.y = ((sint16) pixelTop->R - (sint16) pixelBottom->R) / 256.f;
normal.x *= _SlopeFactor;
NLMISC::clamp(normal.x, -1.f, 1.f);
normal.y *= _SlopeFactor;
NLMISC::clamp(normal.y, -1.f, 1.f);
normal.z = 1.f;
normal.normalize();
float colorValue = _LightDir * normal;
if (colorValue <= 0.f) colorValue = 0.f;
CRGBA diffuse;
*dest = _Ambient;
diffuse.modulateFromui(_Diffuse, (uint) (255.f * colorValue));
dest->add(*dest, diffuse);
dest->A = 255;
++ src;
++ dest;
}
}
if (_HeightMap->getReleasable())
{
_HeightMap->release();
}
}
}