2009-05-16 17:58:08 +00:00
|
|
|
TerrainMesh::TerrainMesh(QuadData* qd, float segSize, float startX, float startY,
|
|
|
|
const Ogre::Vector3 &pos,
|
|
|
|
int width, int depth, bool skirts,
|
|
|
|
Ogre::SceneNode *parent)
|
|
|
|
: Ogre::Renderable(),
|
|
|
|
Ogre::MovableObject(),
|
|
|
|
mWidth(width),
|
|
|
|
mUseSkirts(skirts),
|
|
|
|
mBuilt(false),
|
|
|
|
mDepth(depth),
|
|
|
|
mVertexes(0),
|
|
|
|
mIndices(0),
|
|
|
|
mLODMorphFactor(0),
|
|
|
|
mTextureFadeFactor(0),
|
|
|
|
mMin(30000),
|
|
|
|
mMax(-30000),
|
|
|
|
mExtraMorphAmount(0),
|
|
|
|
mHasFadePass(false),
|
|
|
|
mQuadData(qd),
|
|
|
|
mSegmentSize(segSize),
|
|
|
|
mX(startX),
|
|
|
|
mY(startY)
|
|
|
|
{
|
|
|
|
// From QuadSegment()
|
|
|
|
assert(qd);
|
|
|
|
assert(segSize>0&&segSize<=1);
|
|
|
|
assert(mY>=0&&mY<=1);
|
|
|
|
assert(mX>=0&&mY<=1);
|
|
|
|
|
|
|
|
#ifdef QSDEBUG
|
|
|
|
{
|
|
|
|
//check sizes
|
|
|
|
const float qw = mQuadData->getVertexWidth()-1;
|
|
|
|
const float fsw = qw*segSize;
|
|
|
|
const int isw = (int)fsw;
|
|
|
|
assert(fsw==isw);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//precalc offsets, as getVertex/getNormal get called a lot (1000s of times)
|
|
|
|
computeOffsets();
|
|
|
|
|
|
|
|
// From Quad
|
|
|
|
node = parent->createChildSceneNode(pos);
|
|
|
|
|
|
|
|
// From create()
|
|
|
|
createVertexBuffer();
|
|
|
|
calculateVetexValues();
|
|
|
|
calculateIndexValues();
|
|
|
|
setBounds();
|
|
|
|
|
|
|
|
node->attachObject(this);
|
|
|
|
|
|
|
|
createMaterial();
|
|
|
|
|
|
|
|
if ( g_Terrain->isMorhpingEnabled() && mDepth != g_Terrain->getMaxDepth() ) {
|
|
|
|
Ogre::Technique* tech = getMaterial()->getTechnique(0);
|
|
|
|
for ( size_t i = 0; i < tech->getNumPasses(); ++i ) {
|
|
|
|
assert(g_Terrain->isMorhpingEnabled());
|
|
|
|
tech->getPass(i)->setVertexProgram(MORPH_VERTEX_PROGRAM);
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isMorhpingEnabled() )
|
|
|
|
calculateDeltaValues();
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mBuilt = true;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::destroy() {
|
|
|
|
if ( !mBuilt ) return;
|
|
|
|
|
|
|
|
//deleting null values is fine iirc
|
|
|
|
delete mIndices;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
# if ENABLED_CRASHING == 1
|
2009-05-16 17:58:08 +00:00
|
|
|
{
|
|
|
|
delete mVertexes;
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
# else
|
2009-05-16 17:58:08 +00:00
|
|
|
{
|
|
|
|
if ( mDepth != g_Terrain->getMaxDepth() ){
|
|
|
|
delete mVertexes;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
# endif
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mBuilt = false;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::update(Ogre::Real time, Ogre::Real camDist, Ogre::Real usplitDist, Ogre::Real morphDist)
|
|
|
|
{
|
|
|
|
TRACE("TerrainMesh::update");
|
|
|
|
//if ( USE_MORPH ){
|
|
|
|
|
|
|
|
//as aprocesh mUnsplitDistance, lower detail
|
|
|
|
if ( camDist > morphDist && mDepth > 1 ) {
|
|
|
|
mLODMorphFactor = 1 - (usplitDist - camDist)/(usplitDist-morphDist);
|
|
|
|
} else
|
|
|
|
mLODMorphFactor = 0;
|
|
|
|
mTextureFadeFactor = mLODMorphFactor;
|
|
|
|
|
|
|
|
|
|
|
|
//on an split, it sets the extra morph amount to 1, we then ensure this ends up at 0... slowly
|
|
|
|
if ( mExtraMorphAmount > 0 ) {
|
|
|
|
mLODMorphFactor += mExtraMorphAmount;
|
|
|
|
mExtraMorphAmount -= (time/g_Terrain->getMorphSpeed()); //decrease slowly
|
|
|
|
}
|
|
|
|
if ( mExtraFadeAmount > 0 ) {
|
|
|
|
mTextureFadeFactor += mExtraFadeAmount;
|
|
|
|
mExtraFadeAmount -= (time/g_Terrain->getTextureFadeSpeed());
|
|
|
|
}
|
|
|
|
|
|
|
|
//Ensure within valid bounds
|
|
|
|
if ( mLODMorphFactor > 1 )
|
|
|
|
mLODMorphFactor = 1;
|
|
|
|
else if ( mLODMorphFactor < 0 )
|
|
|
|
mLODMorphFactor = 0;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( mTextureFadeFactor > 1 )
|
|
|
|
mTextureFadeFactor = 1;
|
|
|
|
else if ( mTextureFadeFactor < 0 )
|
|
|
|
mTextureFadeFactor = 0;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//remove pass. Keep outside in case terrain fading is removed while it is active
|
|
|
|
if ( mHasFadePass && mTextureFadeFactor == 0 ) {
|
|
|
|
removeFadePass();
|
|
|
|
} else if ( g_Terrain->isTextureFadingEnabled() &&
|
|
|
|
!mHasFadePass &&
|
|
|
|
mTextureFadeFactor > 0 &&
|
|
|
|
hasParentTexture() ) {
|
|
|
|
addFadePass();
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::addFadePass() {
|
|
|
|
assert(mHasFadePass==false);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( mDepth == g_Terrain->getMaxDepth() ) return;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mHasFadePass = true;
|
|
|
|
Ogre::MaterialPtr mat = getMaterial();
|
|
|
|
Ogre::Pass* newPass = mat->getTechnique(0)->createPass();
|
|
|
|
newPass->setSceneBlending(Ogre::SBF_SOURCE_ALPHA, Ogre::SBF_ONE_MINUS_SOURCE_ALPHA);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//set fragment program
|
|
|
|
assert(g_Terrain->isTextureFadingEnabled());
|
|
|
|
newPass->setFragmentProgram(FADE_FRAGMENT_PROGRAM);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isMorhpingEnabled() && mDepth != g_Terrain->getMaxDepth() ) {
|
|
|
|
assert(g_Terrain->isMorhpingEnabled());
|
|
|
|
newPass->setVertexProgram(MORPH_VERTEX_PROGRAM);
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//set texture to be used
|
|
|
|
newPass->createTextureUnitState(getParentTexture(), 1);
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::removeFadePass() {
|
|
|
|
assert(mHasFadePass==true);
|
|
|
|
mHasFadePass = false;
|
|
|
|
Ogre::MaterialPtr mat = getMaterial();
|
|
|
|
Ogre::Technique* tech = mat->getTechnique(0);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
tech->removePass(tech->getNumPasses()-1);
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::justSplit() {
|
|
|
|
mExtraMorphAmount = 1;
|
|
|
|
mLODMorphFactor = 1;
|
|
|
|
mTextureFadeFactor = 1;
|
|
|
|
mExtraFadeAmount = 1;
|
|
|
|
|
|
|
|
if ( g_Terrain->isTextureFadingEnabled() && hasParentTexture() )
|
|
|
|
addFadePass();
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::_updateCustomGpuParameter(
|
|
|
|
const GpuProgramParameters::AutoConstantEntry& constantEntry,
|
|
|
|
GpuProgramParameters* params) const {
|
|
|
|
using namespace Ogre;
|
|
|
|
if (constantEntry.data == MORPH_CUSTOM_PARAM_ID)
|
|
|
|
params->_writeRawConstant(constantEntry.physicalIndex, mLODMorphFactor);
|
|
|
|
else if ( constantEntry.data == FADE_CUSTOM_PARAM_ID )
|
|
|
|
params->_writeRawConstant(constantEntry.physicalIndex, mTextureFadeFactor);
|
|
|
|
else
|
|
|
|
Renderable::_updateCustomGpuParameter(constantEntry, params);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
float TerrainMesh::getVertexHeight(int x, int y) {
|
|
|
|
return getVertex(x,y);
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
Ogre::Vector3 TerrainMesh::getVertexPosition(int x, int y) {
|
|
|
|
return Ogre::Vector3(x*getVertexSeperation(), getVertexHeight(x,y) , y*getVertexSeperation());
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::calculateVetexValues() {
|
|
|
|
using namespace Ogre;
|
|
|
|
|
|
|
|
//get the texture offsets for the higher uv
|
|
|
|
float xUVOffset = 0;
|
|
|
|
float yUVOffset = 0;
|
|
|
|
|
|
|
|
if ( g_Terrain->isTextureFadingEnabled() ) {
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
switch (mInterface->getLocation()) {
|
|
|
|
case Quad::QL_NW :
|
|
|
|
yUVOffset = 32.0f/64.0f;
|
|
|
|
break;
|
|
|
|
case Quad::QL_NE:
|
|
|
|
yUVOffset = 32.0f/64.0f;
|
|
|
|
xUVOffset = 32.0f/64.0f;
|
|
|
|
break;
|
|
|
|
case Quad::QL_SE:
|
|
|
|
xUVOffset = 32.0f/64.0f;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
*/
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
int start = 0;
|
|
|
|
int end = mWidth;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( mUseSkirts ) {
|
|
|
|
--start;
|
|
|
|
++end;
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
float* verts = static_cast<float*>(mMainBuffer->lock(HardwareBuffer::HBL_DISCARD));
|
|
|
|
for ( int y = start; y < end; y++ ) {
|
|
|
|
for ( int x = start; x < end; x++ ) {
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//the skirts
|
|
|
|
if ( y < 0 || y > (mWidth-1) || x < 0 || x > (mWidth-1) ) {
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( x < 0 ) *verts++ = 0;
|
|
|
|
else if ( x > (mWidth-1) ) *verts++ = (mWidth-1)*getVertexSeperation();
|
|
|
|
else *verts++ = x*getVertexSeperation();
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
*verts++ = -4096; //2048 below base sea floor height
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( y < 0 ) *verts++ = 0;
|
|
|
|
else if ( y > (mWidth-1) ) *verts++ = (mWidth-1)*getVertexSeperation();
|
|
|
|
else *verts++ = y*getVertexSeperation();
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
for ( Ogre::uint i = 0; i < 3; i++ )
|
|
|
|
*verts++ = 0;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
float u = (float)(x) / (mWidth-1);
|
|
|
|
float v = (float)(y) / (mWidth-1);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//clamped, so shouldn't matter if > 1
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
*verts++ = u;
|
|
|
|
*verts++ = v;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isTextureFadingEnabled() ) {
|
|
|
|
*verts++ = u;
|
|
|
|
*verts++ = v;
|
|
|
|
}
|
|
|
|
} else {
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
assert(y*mWidth+x>=0&&y*mWidth+x<mWidth*mWidth);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//verts
|
|
|
|
*verts++ = x*getVertexSeperation();
|
|
|
|
*verts++ = getVertexHeight(x,y);
|
|
|
|
*verts++ = y*getVertexSeperation();
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mMax = std::max(mMax, getVertexHeight(x,y));
|
|
|
|
mMin = std::min(mMin, getVertexHeight(x,y));
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//normals
|
|
|
|
for ( Ogre::uint i = 0; i < 3; i++ )
|
|
|
|
*verts++ = getNormal(x,y,i);
|
|
|
|
//*verts++ = mInterface->getNormal((y*mWidth+x)*3+i);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
const float u = (float)(x) / (mWidth-1);
|
|
|
|
const float v = (float)(y) / (mWidth-1);
|
|
|
|
assert(u>=0&&v>=0);
|
|
|
|
assert(u<=1&&v<=1);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
*verts++ = u;
|
|
|
|
*verts++ = v;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isTextureFadingEnabled() ) {
|
|
|
|
*verts++ = u/2.0f + xUVOffset;
|
|
|
|
*verts++ = v/2.0f + yUVOffset;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
}
|
|
|
|
mMainBuffer->unlock();
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::setBounds() {
|
|
|
|
mBounds.setExtents(0,mMin,0,
|
|
|
|
(mWidth - 1) * getVertexSeperation(),
|
|
|
|
mMax,
|
|
|
|
(mWidth - 1) * getVertexSeperation());
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mCenter = Ogre::Vector3( ( (mWidth - 1) * getVertexSeperation() ) / 2,
|
|
|
|
( mMin + mMax ) / 2,
|
|
|
|
( (mWidth - 1) * getVertexSeperation() ) / 2);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mBoundingRadius = (mBounds.getMaximum() - mBounds.getMinimum()).length() / 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::createVertexBuffer() {
|
|
|
|
using namespace Ogre;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
size_t vw = mWidth;
|
|
|
|
if ( mUseSkirts ) vw += 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mVertexes = new VertexData();
|
|
|
|
mVertexes->vertexStart = 0;
|
|
|
|
mVertexes->vertexCount = vw*vw;// VERTEX_WIDTH;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
VertexDeclaration* vertexDecl = mVertexes->vertexDeclaration;
|
|
|
|
size_t currOffset = 0;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
vertexDecl->addElement(MAIN_BINDING, currOffset, VET_FLOAT3, VES_POSITION);
|
|
|
|
currOffset += VertexElement::getTypeSize(VET_FLOAT3);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
vertexDecl->addElement(MAIN_BINDING, currOffset, VET_FLOAT3, VES_NORMAL);
|
|
|
|
currOffset += VertexElement::getTypeSize(VET_FLOAT3);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
vertexDecl->addElement(MAIN_BINDING, currOffset, VET_FLOAT2, VES_TEXTURE_COORDINATES, 0);
|
|
|
|
currOffset += VertexElement::getTypeSize(VET_FLOAT2);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isTextureFadingEnabled() ) {
|
|
|
|
vertexDecl->addElement(MAIN_BINDING, currOffset, VET_FLOAT2, Ogre::VES_TEXTURE_COORDINATES, 1);
|
|
|
|
currOffset += VertexElement::getTypeSize(VET_FLOAT2);
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mMainBuffer = HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
vertexDecl->getVertexSize(0), // size of one whole vertex
|
|
|
|
mVertexes->vertexCount, // number of vertices
|
|
|
|
HardwareBuffer::HBU_STATIC_WRITE_ONLY, // usage
|
|
|
|
false); // no shadow buffer
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mVertexes->vertexBufferBinding->setBinding(MAIN_BINDING, mMainBuffer); //bind the data
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( g_Terrain->isMorhpingEnabled() )
|
|
|
|
vertexDecl->addElement(DELTA_BINDING, 0, VET_FLOAT1, VES_BLEND_WEIGHTS);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
Ogre::HardwareVertexBufferSharedPtr TerrainMesh::createDeltaBuffer( ) {
|
|
|
|
size_t vw = mWidth;
|
|
|
|
if ( mUseSkirts ) vw += 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
const size_t totalVerts = (vw * vw);
|
|
|
|
return Ogre::HardwareBufferManager::getSingleton().createVertexBuffer(
|
|
|
|
Ogre::VertexElement::getTypeSize(Ogre::VET_FLOAT1),
|
|
|
|
totalVerts,
|
|
|
|
Ogre::HardwareBuffer::HBU_STATIC_WRITE_ONLY,
|
|
|
|
false); //no shadow buff
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//----------------------------------------------
|
|
|
|
#define SET_DELTA_AT(x, y, v) \
|
|
|
|
if ( mUseSkirts ) pDeltas[( y + 1)*vw+ x + 1] = v; \
|
|
|
|
else pDeltas[( y)*vw+ x] = v;
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::calculateDeltaValues() {
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
using namespace Ogre;
|
|
|
|
size_t vw = mWidth;
|
|
|
|
if ( mUseSkirts ) vw += 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//must be using morphing to use this function
|
|
|
|
assert(g_Terrain->isMorhpingEnabled());
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
const size_t step = 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
// Create a set of delta values
|
|
|
|
mDeltaBuffer = createDeltaBuffer();
|
|
|
|
float* pDeltas = static_cast<float*>(mDeltaBuffer->lock(HardwareBuffer::HBL_DISCARD));
|
|
|
|
memset(pDeltas, 0, (vw)*(vw) * sizeof(float));
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
return;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
bool flag=false;
|
|
|
|
for ( size_t y = 0; y < mWidth-step; y += step ) {
|
|
|
|
for ( size_t x = 0; x < mWidth-step; x += step ) {
|
|
|
|
//create the diffrence between the full vertex if the vertex wasn't their
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
float bottom_left = getVertexHeight(x,y);
|
|
|
|
float bottom_right = getVertexHeight(x+step,y);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
float top_left = getVertexHeight(x,y+step);
|
|
|
|
float top_right = getVertexHeight(x+step,y+step);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//const int vw = mWidth+2;
|
|
|
|
SET_DELTA_AT(x, y+1, (bottom_left+top_left)/2 - getVertexHeight(x, y+1)) //left
|
|
|
|
SET_DELTA_AT(x+2, y+1, (bottom_right+top_right)/2 - getVertexHeight(x+2, y+1)) //right
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
SET_DELTA_AT(x+1, y+2, (top_right+top_left)/2 - getVertexHeight(x+1, y+2)) //top
|
|
|
|
SET_DELTA_AT(x+1, y, (bottom_right+bottom_left)/2 - getVertexHeight(x+1, y)) //bottom
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//this might not be correct
|
|
|
|
if ( !flag )
|
|
|
|
SET_DELTA_AT(x+1, y+1, (bottom_left+top_right)/2 - getVertexHeight(x+1, y+1)) //center
|
|
|
|
else
|
|
|
|
SET_DELTA_AT(x+1, y+1, (bottom_right+top_left)/2 - getVertexHeight(x+1, y+1)) //center
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
flag = !flag;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
flag = !flag; //flip tries for next row
|
|
|
|
}
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mDeltaBuffer->unlock();
|
|
|
|
mVertexes->vertexBufferBinding->setBinding(DELTA_BINDING,mDeltaBuffer);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
#undef SET_DELTA_AT
|
|
|
|
|
|
|
|
//----------------------------------------------
|
2009-05-16 17:58:08 +00:00
|
|
|
void TerrainMesh::calculateIndexValues() {
|
|
|
|
using namespace Ogre;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
size_t vw = mWidth-1;
|
|
|
|
if ( mUseSkirts ) vw += 2;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
const size_t indexCount = (vw)*(vw)*6;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
//need to manage allocation if not null
|
|
|
|
assert(mIndices==0);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
mIndices = new IndexData();
|
|
|
|
mIndices->indexCount = indexCount;
|
|
|
|
mIndices->indexBuffer = HardwareBufferManager::getSingleton().createIndexBuffer(
|
|
|
|
HardwareIndexBuffer::IT_16BIT,
|
|
|
|
indexCount, HardwareBuffer::HBU_STATIC_WRITE_ONLY, false);
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
unsigned short* indices = static_cast<unsigned short*>(mIndices->indexBuffer->lock(0,
|
|
|
|
mIndices->indexBuffer->getSizeInBytes(),
|
|
|
|
HardwareBuffer::HBL_DISCARD));
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
bool flag = false;
|
|
|
|
Ogre::uint indNum = 0;
|
|
|
|
for ( Ogre::uint y = 0; y < (vw); y+=1 ) {
|
|
|
|
for ( Ogre::uint x = 0; x < (vw); x+=1 ) {
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
const int line1 = y * (vw+1) + x;
|
|
|
|
const int line2 = (y + 1) * (vw+1) + x;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
if ( flag ) {
|
|
|
|
*indices++ = line1;
|
|
|
|
*indices++ = line2;
|
|
|
|
*indices++ = line1 + 1;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
*indices++ = line1 + 1;
|
|
|
|
*indices++ = line2;
|
|
|
|
*indices++ = line2 + 1;
|
|
|
|
} else {
|
|
|
|
*indices++ = line1;
|
|
|
|
*indices++ = line2;
|
|
|
|
*indices++ = line2 + 1;
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
*indices++ = line1;
|
|
|
|
*indices++ = line2 + 1;
|
|
|
|
*indices++ = line1 + 1;
|
|
|
|
}
|
|
|
|
flag = !flag; //flip tris for next time
|
2009-05-15 07:04:50 +00:00
|
|
|
|
2009-05-16 17:58:08 +00:00
|
|
|
indNum+=6;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|
2009-05-16 17:58:08 +00:00
|
|
|
flag = !flag; //flip tries for next row
|
|
|
|
}
|
|
|
|
assert(indNum==indexCount);
|
|
|
|
mIndices->indexBuffer->unlock();
|
|
|
|
//return mIndices;
|
2009-05-15 07:04:50 +00:00
|
|
|
}
|