// 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_grouped.h" #include "nel/3d/texture_file.h" #include "nel/misc/common.h" #include "nel/misc/path.h" #include #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { std::map CTextureGrouped::_NameToSize; ///===================================================================================================== /// This is used to get the size of a texture, with an optimisation in the case of texture files static inline void GetTextureSize(ITexture *tex, uint &width, uint &height) { if (tex->getClassName() == "CTextureFile") { CTextureFile *tf = static_cast(tex); uint32 srcWidth, srcHeight; if (!tf->getFileName().empty()) { try { CBitmap::loadSize(NLMISC::CPath::lookup(tf->getFileName()), srcWidth, srcHeight); if (srcWidth == 0 || srcHeight == 0) { nlinfo("Unable to get size of texture : %s", tf->getFileName().c_str()); width = height = 0; return; } width = srcWidth; height = srcHeight; } catch (const NLMISC::EPathNotFound &e) { nlinfo("%s", e.what()); width = height = 0; } catch (const NLMISC::EStream &e) { nlinfo("unable to load size from a bitmap ! name = %s", tf->getFileName().c_str()); nlinfo("reason = %s", e.what()); width = height = 0; } } else { width = height = 0; } } else // we must generate the texture to get its size { tex->generate(); width = tex->getWidth(); height = tex->getHeight(); if (tex->getReleasable()) { tex->release(); } } } ///===================================================================================================== CTextureGrouped::CTextureGrouped() : _NbTex(0) { setFilterMode(Linear, LinearMipMapOff); } ///===================================================================================================== CTextureGrouped::CTextureGrouped(const CTextureGrouped &src) : ITexture(src) { // copy the part associated with us; duplicate(src); } ///===================================================================================================== void CTextureGrouped::duplicate(const CTextureGrouped &src) { _NbTex = src._NbTex; TTexList texCopy(src._Textures.begin(), src._Textures.end()); _Textures.swap(texCopy); TFourUVList uvCopy(_TexUVs.begin(), _TexUVs.end()); _TexUVs.swap(uvCopy); } ///===================================================================================================== CTextureGrouped &CTextureGrouped::operator=(const CTextureGrouped &src) { ITexture::operator=(src); // copy parent part duplicate(src); return *this; } ///===================================================================================================== bool CTextureGrouped::areValid(CSmartPtr *textureTab, uint nbTex) { bool result = true; uint k; for(k = 0; k < nbTex; ++k) { textureTab[k]->generate(); if (textureTab[k]->getWidth() != textureTab[0]->getWidth() || textureTab[k]->getHeight() != textureTab[0]->getHeight() || textureTab[k]->getPixelFormat() != textureTab[0]->getPixelFormat() ) { result = false; break; } } for (k = 0; k < nbTex; ++k) { if (textureTab[k]->getReleasable()) textureTab[k]->release(); } return result; } ///===================================================================================================== void CTextureGrouped::setTextures(CSmartPtr *textureTab, uint nbTex, bool checkValid) { nlassert(nbTex > 0); if (checkValid) { if (!areValid(textureTab, nbTex)) { displayIncompatibleTextureWarning(textureTab, nbTex); makeDummies(textureTab, nbTex); } } _Textures.resize(nbTex); std::copy(textureTab, textureTab + nbTex, _Textures.begin()); _TexUVs.resize(nbTex); for(uint k = 0; k < nbTex; ++k) { // real uvs are generated during doGenerate _TexUVs[k].uv0.set(0.f, 0.f); _TexUVs[k].uv1.set(0.f, 0.f); _TexUVs[k].uv2.set(0.f, 0.f); _TexUVs[k].uv3.set(0.f, 0.f); } _DeltaUV.set(0.f, 0.f); _NbTex = nbTex; touch(); // the texture need regeneration nlassert(_NbTex == _Textures.size()); } ///===================================================================================================== void CTextureGrouped::doGenerate(bool async) { nlassert(_NbTex == _Textures.size()); if (_NbTex == 0) { makeDummy(); } else { // Generate the first texture to get the size _Textures[0]->generate(); const uint width = _Textures[0]->getWidth(), height = _Textures[0]->getHeight(); const uint totalHeight = height * _NbTex; const uint realHeight = NLMISC::raiseToNextPowerOf2(totalHeight); resize(width, realHeight, _Textures[0]->getPixelFormat()); uint k; sint32 currY = 0; for(k = 0; k < _NbTex; ++k) { _Textures[k]->generate(); if (_Textures[k]->getWidth() != width || _Textures[k]->getHeight() != height || _Textures[k]->getPixelFormat() != _Textures[0]->getPixelFormat()) { makeDummy(); break; } this->blit(_Textures[k], 0, currY); currY += height; } for(k = 0; k < _NbTex; ++k) { if ( _Textures[k]->getReleasable()) { _Textures[k]->release(); } } // save sub bitmap size so that bitmaps won't be reloaded to get their size the next time _NameToSize[getShareName()] = height; // compute the uvs genUVs(height); } } ///===================================================================================================== void CTextureGrouped::forceGenUVs() { // retrieve size of sub-bitmap std::string shareName = getShareName(); std::map::const_iterator it = _NameToSize.find(shareName); if (it == _NameToSize.end()) { doGenerate(); // this will retrieve size of sub-bitmaps // TODO : only load the size... not a problem for now because textures can be cached at startup release(); it = _NameToSize.find(getShareName()); if (it == _NameToSize.end()) { _TexUVs.resize(_NbTex); // generate default uvs for(uint k = 0; k < _NbTex; ++k) { TFourUV &uvs = _TexUVs[k]; uvs.uv0.set(0.f, 0.f); uvs.uv1.set(1.f, 0.f); uvs.uv2.set(1.f, 1.f); uvs.uv3.set(0.f, 1.f); } return; } } genUVs(it->second); } ///===================================================================================================== void CTextureGrouped::genUVs(uint subBitmapHeight) { _TexUVs.resize(_NbTex); // const uint totalHeight = subBitmapHeight * _NbTex; // *it contains the height of bitmap const uint realHeight = NLMISC::raiseToNextPowerOf2(totalHeight); // TODO : better ordering of sub-bitmaps const float deltaV = realHeight ? (float(totalHeight) / float(realHeight)) * (1.0f / _NbTex) : 0.f; _DeltaUV = CUV(1, deltaV); CUV currentUV(0, 0); for(uint k = 0; k < _NbTex; ++k) { TFourUV &uvs = _TexUVs[k]; uvs.uv0 = currentUV; uvs.uv1 = currentUV + CUV(1, 0); uvs.uv2 = currentUV + _DeltaUV; uvs.uv3 = currentUV + CUV(0, deltaV); currentUV.V += deltaV; } } ///===================================================================================================== void CTextureGrouped::getTextures(CSmartPtr *textureTab) const { CSmartPtr *dest = textureTab; for(TTexList::const_iterator it = _Textures.begin(); it != _Textures.end(); ++it, ++dest) { *dest = *it; } } ///===================================================================================================== bool CTextureGrouped::supportSharing() const { // all textures in this group must support sharing for this one to support it for(TTexList::const_iterator it = _Textures.begin(); it != _Textures.end(); ++it) { if (!(*it)->supportSharing()) return false; } return true; } ///===================================================================================================== std::string CTextureGrouped::getShareName() const { nlassert(supportSharing()); std::string shareName("groupedTex:"); for(TTexList::const_iterator it = _Textures.begin(); it != _Textures.end(); ++it) { shareName += (*it)->getShareName() + std::string("|"); } return shareName; } ///===================================================================================================== void CTextureGrouped::serial(NLMISC::IStream &f) { f.serialVersion(1); if (f.isReading()) { TTexList texList; /// read the number of textures uint32 nbTex; f.serial(nbTex); /// cerate a vector of textures ITexture *ptTex = NULL; texList.reserve(nbTex); for (uint k = 0; k < nbTex; ++k) { f.serialPolyPtr(ptTex); texList.push_back(ptTex); } // setup the textures setTextures(&texList[0], nbTex, false); forceGenUVs(); touch(); } else { f.serial(_NbTex); ITexture *ptTex; for (TTexList::iterator it = _Textures.begin(); it != _Textures.end(); ++it) { ptTex = *it; f.serialPolyPtr(ptTex); } } } ///===================================================================================================== void CTextureGrouped::release() { ITexture::release(); for(uint k = 0; k < _NbTex; ++k) { if ( _Textures[k]->getReleasable()) { _Textures[k]->release(); } } } ///===================================================================================================== void CTextureGrouped::makeDummies(CSmartPtr *textureTab,uint nbTex) { for(uint k = 0; k < nbTex; ++k) { textureTab[k] = new CTextureFile("DummyTex"); // well this shouldn't exist.. } } ///===================================================================================================== void CTextureGrouped::displayIncompatibleTextureWarning(CSmartPtr *textureTab,uint nbTex) { nlwarning("======================================================="); nlwarning("CTextureGrouped : found incompatible textures, that differs by size or format :" ); for(uint k = 0; k < nbTex; ++k) { if (textureTab[k]) { nlwarning((textureTab[k]->getShareName()).c_str()); } else { nlwarning("NULL Texture found"); } } nlwarning("======================================================="); } } // NL3D