// 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/misc/aabbox.h" #include "nel/misc/matrix.h" #include "nel/3d/ps_util.h" #include "nel/3d/particle_system.h" #include "nel/3d/driver.h" #include "nel/3d/vertex_buffer.h" #include "nel/3d/index_buffer.h" #include "nel/3d/material.h" #include "nel/3d/nelu.h" #include "nel/3d/font_generator.h" #include "nel/3d/font_manager.h" #include "nel/3d/computed_string.h" #include "nel/3d/dru.h" #include "nel/3d/ps_located.h" #include "nel/3d/ps_sound.h" #include "nel/3d/ps_light.h" #include "nel/3d/particle_system_shape.h" namespace NL3D { using NLMISC::CVector; //#ifdef NL_DEBUG bool CPSUtil::_CosTableInitialized = false; bool CPSUtil::_PerlinNoiseTableInitialized = false; //#endif float CPSUtil::_CosTable[256]; float CPSUtil::_SinTable[256]; float CPSUtil::_PerlinNoiseTab[1024]; //========================================================================== void CPSUtil::initPerlinNoiseTable(void) { NL_PS_FUNC(CPSUtil_initPerlinNoiseTable) for (uint32 k = 0; k < 1024; ++k) { _PerlinNoiseTab[k] = (rand() % 30000) / 30000.f; } //#ifdef NL_DEBUG _PerlinNoiseTableInitialized = true; //#endif } //========================================================================== void CPSUtil::initFastCosNSinTable(void) { NL_PS_FUNC(CPSUtil_initFastCosNSinTable) for (uint32 k = 0; k < 256; k++) { const float angle = k / 256.0f * 2.0f * float(NLMISC::Pi); _CosTable[k] = (float) cos( angle ); _SinTable[k] = (float) sin( angle ); } //#ifdef NL_DEBUG _CosTableInitialized = true; //#endif } //========================================================================== void CPSUtil::registerSerialParticleSystem(void) { NL_PS_FUNC(CPSUtil_registerSerialParticleSystem) NLMISC_REGISTER_CLASS(CPSLocated); NLMISC_REGISTER_CLASS(CParticleSystemShape); NLMISC_REGISTER_CLASS(CPSSound); NLMISC_REGISTER_CLASS(CPSLight); registerParticles(); registerForces(); registerEmitters(); registerZones(); registerAttribs(); // while we are here, we perform some important inits initFastCosNSinTable(); // init fast cosine lookup table initPerlinNoiseTable(); // init perlin noise table } //========================================================================== void CPSUtil::displayBBox(NL3D::IDriver *driver, const NLMISC::CAABBox &box, NLMISC::CRGBA col /* = NLMISC::CRGBA::White */) { NL_PS_FUNC(CPSUtil_displayBBox) CVector max = box.getMax() ,min = box.getMin(); CVertexBuffer vb; vb.setVertexFormat(CVertexBuffer::PositionFlag); vb.setNumVertices(8); { CVertexBufferReadWrite vba; vb.lock (vba); vba.setVertexCoord(0, min); vba.setVertexCoord(1, CVector(max.x, min.y, min.z)); vba.setVertexCoord(2, CVector(min.x, max.y, min.z)); vba.setVertexCoord(3, CVector(max.x, max.y, min.z)); vba.setVertexCoord(4, CVector(min.x, min.y, max.z)); vba.setVertexCoord(5, CVector(max.x, min.y, max.z)); vba.setVertexCoord(6, CVector(min.x, max.y, max.z)); vba.setVertexCoord(7, max); } CMaterial material; material.setColor(col); material.setLighting(false); material.setBlendFunc(CMaterial::one, CMaterial::one); material.setZWrite(false); material.setBlend(true); CIndexBuffer pb; pb.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT); pb.setNumIndexes(2*12); { CIndexBufferReadWrite ibaWrite; pb.lock (ibaWrite); ibaWrite.setLine(0, 0, 1); ibaWrite.setLine(2, 1, 5); ibaWrite.setLine(4, 5, 4); ibaWrite.setLine(6, 4, 0); ibaWrite.setLine(8, 0, 2); ibaWrite.setLine(10, 1, 3); ibaWrite.setLine(12, 4, 6); ibaWrite.setLine(14, 5, 7); ibaWrite.setLine(16, 6, 7); ibaWrite.setLine(18, 7, 3); ibaWrite.setLine(20, 3, 2); ibaWrite.setLine(22, 2, 6); } driver->activeVertexBuffer(vb); driver->activeIndexBuffer(pb); driver->renderLines(material, 0, pb.getNumIndexes()/2); } //========================================================================== void CPSUtil::displayArrow(IDriver *driver, const CVector &start, const CVector &v, float size, CRGBA col1, CRGBA col2) { NL_PS_FUNC(CPSUtil_displayArrow) const float coneSize = size * 0.1f; static CIndexBuffer vTab; static const TIndexType vTabIndexes[] = { 1, 2, 4, 4, 2, 3, 1, 2, 0, 2, 3, 0, 3, 4, 0, 4, 1, 0 }; vTab.setFormat(NL_DEFAULT_INDEX_BUFFER_FORMAT); if (vTab.getNumIndexes()==0) { vTab.setNumIndexes (sizeofarray(vTabIndexes)); CIndexBufferReadWrite iba; vTab.lock(iba); memcpy (iba.getPtr(), vTabIndexes, sizeof(vTabIndexes)); } CVector end = start + size * v; CDRU::drawLine(start, end, col1, *driver); CMatrix m; buildSchmidtBasis(v, m); CVertexBuffer vb; vb.setVertexFormat(CVertexBuffer::PositionFlag); vb.setNumVertices(5); { CVertexBufferReadWrite vba; vb.lock (vba); vba.setVertexCoord(0, end + m * CVector(0, 0, 3.0f * coneSize) ); vba.setVertexCoord(1, end + m * CVector(-coneSize, -coneSize, 0) ); vba.setVertexCoord(2, end + m * CVector(coneSize, -coneSize, 0) ); vba.setVertexCoord(3, end + m * CVector(coneSize, coneSize, 0) ); vba.setVertexCoord(4, end + m * CVector(-coneSize, coneSize, 0) ); } CMaterial material; material.setColor(col2); material.setLighting(false); material.setBlendFunc(CMaterial::one, CMaterial::one); material.setZWrite(false); material.setBlend(true); material.setDoubleSided(true); driver->activeVertexBuffer(vb); driver->activeIndexBuffer(vTab); driver->renderTriangles(material, 0, 6); } //========================================================================== void CPSUtil::displayBasis(IDriver *driver, const CMatrix &modelMat, const NLMISC::CMatrix &m, float size, CFontGenerator &fg, CFontManager &fm) { NL_PS_FUNC(CPSUtil_displayBasis) CMaterial material; driver->setupModelMatrix(modelMat); displayArrow(driver, m.getPos(), m.getI(), size, CRGBA(127, 127, 127), CRGBA(0, 0, 80)); displayArrow(driver, m.getPos(), m.getJ(), size, CRGBA(127, 127, 127), CRGBA(0, 0, 80)); displayArrow(driver, m.getPos(), m.getK(), size, CRGBA(127, 127, 127), CRGBA(200, 0, 80)); // draw the letters CPSUtil::print(driver, std::string("x"), fg, fm, modelMat * m * CVector(1.4f * size, 0, 0), 15.0f * size); CPSUtil::print(driver, std::string("y"), fg, fm, modelMat * m * CVector(0, 1.4f * size, 0), 15.0f * size); CPSUtil::print(driver, std::string("z"), fg, fm, modelMat * m * CVector(0, 0, 1.4f * size), 15.0f * size); }; //========================================================================== void CPSUtil::print(IDriver *driver, const std::string &text, CFontGenerator &fg, CFontManager &fm, const CVector &pos, float size, NLMISC::CRGBA col /*= NLMISC::CRGBA::White*/) { NL_PS_FUNC(CPSUtil_print) nlassert((&fg) && (&fm)); CComputedString cptedString; fm.computeString ( text, &fg, col, 16, false, false, driver, cptedString); CMatrix mat = driver->getViewMatrix(); mat.setPos(CVector::Null); mat.scale(CVector(size, size, size)); mat.transpose(); mat.setPos(pos); cptedString.render3D(*driver, mat); } //========================================================================== void CPSUtil::buildSchmidtBasis(const CVector &k_, NLMISC::CMatrix &result) { NL_PS_FUNC(CPSUtil_buildSchmidtBasis) const float epsilon = 10E-4f; CVector k = k_; k.normalize(); CVector i; if ((1.0f - fabsf(k * CVector::I)) > epsilon) { i = k ^ CVector::I; } else if ((1.0f - fabs(k * CVector::J)) > epsilon) { i = k ^ CVector::J; } else { i = k ^ CVector::K; } i = i - (k * i) * k; i.normalize(); result.setRot(i, k ^ i, k, true); } //========================================================================== void CPSUtil::displaySphere(IDriver &driver, float radius, const CVector ¢er, uint nbSubdiv, CRGBA color) { NL_PS_FUNC(CPSUtil_displaySphere) uint x, y, k; CVector p, p1, p2; static const CVector lK[] = { CVector::I, -CVector::I ,CVector::J, -CVector::J ,CVector::K, -CVector::K }; /* static const CVector lI = { CVector::J, -CVector::J ,CVector::K, -CVector::K ,CVector::I, -CVector::I };*/ for (k = 0; k < 6; ++k) { const CVector &I = lK[(k + 2) % 6]; const CVector &K = lK[k]; const CVector J = K ^ I; for (x = 0; x < nbSubdiv; ++x) { for (y = 0; y < nbSubdiv; ++y) { p = ((2.f * x / float(nbSubdiv) ) - 1.f) * I + ((2.f * y / float(nbSubdiv) ) - 1.f) * J + K; p1 = p + 2.f / float(nbSubdiv) * I; p2 = p + 2.f / float(nbSubdiv) * J; p.normalize(); p1.normalize(); p2.normalize(); p = center + radius * p; p1 = center + radius * p1; p2 = center + radius * p2; CDRU::drawLine(p, p1, color, driver); CDRU::drawLine(p, p2, color, driver); } } } } //========================================================================== void CPSUtil::displayDisc(IDriver &driver, float radius, const CVector ¢er, const CMatrix &mat, uint nbSubdiv, CRGBA color) { NL_PS_FUNC(CPSUtil_displayDisc) // not optimized, but for edition only float thetaDelta = (float) NLMISC::Pi * 2.f / nbSubdiv; float theta = 0.f; const CVector &I = mat.getI(); const CVector &J = mat.getJ(); for (uint k = 0; k < nbSubdiv; ++k) { CDRU::drawLine(center + radius * ((float) cos(theta) * I + (float) sin(theta) * J) , center + radius * ((float) cos(theta + thetaDelta) * I + (float) sin(theta + thetaDelta) * J) , color, driver); theta += thetaDelta; } } //========================================================================== void CPSUtil::displayCylinder(IDriver &driver, const CVector ¢er, const CMatrix &mat, const CVector &dim, uint nbSubdiv, CRGBA color) { NL_PS_FUNC(CPSUtil_displayCylinder) // not optimized, but for edition only float thetaDelta = (float) NLMISC::Pi * 2.f / nbSubdiv; float theta = 0.f; const CVector &I = mat.getI(); const CVector &J = mat.getJ(); const CVector &K = mat.getK(); for (uint k = 0; k < nbSubdiv; ++k) { CDRU::drawLine(center + dim.z * K + dim.x * cosf(theta) * I + dim.y * sinf(theta) * J , center + dim.z * K + dim.x * cosf(theta + thetaDelta) * I + dim.y * sinf(theta + thetaDelta) * J , color, driver); CDRU::drawLine(center - dim.z * K + dim.x * cosf(theta) * I + dim.y * sinf(theta) * J , center - dim.z * K + dim.x * cosf(theta + thetaDelta) * I + dim.y * sinf(theta + thetaDelta) * J , color, driver); CDRU::drawLine(center + dim.z * K + dim.x * cosf(theta) * I + dim.y * sinf(theta) * J , center - dim.z * K + dim.x * cosf(theta) * I + dim.y * sinf(theta) * J , color, driver); theta += thetaDelta; } } //========================================================================== void CPSUtil::display3DQuad(IDriver &driver, const CVector &c1, const CVector &c2 ,const CVector &c3, const CVector &c4, CRGBA color) { NL_PS_FUNC(CPSUtil_display3DQuad) CDRU::drawLine(c1, c2, color, driver); CDRU::drawLine(c2, c3, color, driver); CDRU::drawLine(c3, c4, color, driver); CDRU::drawLine(c4, c1, color, driver); } } // NL3D