// 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/tessellation.h"
#include "nel/3d/patch.h"
#include "nel/3d/zone.h"
#include "nel/misc/common.h"
#include "nel/3d/landscape_profile.h"
#include "nel/3d/landscape.h"
#include "nel/3d/patchdlm_context.h"
using namespace NLMISC;
using namespace std;
namespace NL3D
{
// ***************************************************************************
#define NL3D_TESS_USE_QUADRANT_THRESHOLD 0.1f
// ***************************************************************************
// The normal Uvs format.
const uint8 TileUvFmtNormal1= 0;
const uint8 TileUvFmtNormal2= 1;
const uint8 TileUvFmtNormal3= 2;
const uint8 TileUvFmtNormal4= 3;
const uint8 TileUvFmtNormal5= 4;
// ***************************************************************************
const float TileSize= 128;
// ***************************************************************************
// ***************************************************************************
// CTileMaterial
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
CTileMaterial::CTileMaterial()
{
// By default, all pass are NULL.
for(uint i=0; iappendRdrPatchTile(i, &Pass[i], patchNumRenderableFaces);
}
}
}
// ***************************************************************************
// ***************************************************************************
// CTessVertex
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTessVertex::computeGeomPos()
{
// Compute Basic ErrorMetric.
float sqrDist= (StartPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float pgeom= MaxFaceSize * CLandscapeGlobals::OORefineThreshold / sqrDist;
// Compute ErrorMetric modified by TileNear transition, only if TileNear transition.
if( sqrDist< CLandscapeGlobals::TileDistFarSqr )
{
// Soft optim: do it only if necessary, ie result of max(errorMetric, errorMetricModified) is foreseeable here.
if(pgeom < MaxNearLimit)
{
float f= (CLandscapeGlobals::TileDistFarSqr - sqrDist) * CLandscapeGlobals::OOTileDistDeltaSqr;
clamp(f, 0, 1);
// ^4 gives better smooth result
f= sqr(f);
f= sqr(f);
// interpolate the errorMetric
pgeom= MaxNearLimit*f + pgeom*(1-f);
}
}
// Interpolate StartPos to EndPos, between 1 and 2.
if(pgeom<=1.0f)
Pos= StartPos;
else if(pgeom>=2.0f)
Pos= EndPos;
else
{
float f= pgeom - 1.0f;
Pos= f * (EndPos-StartPos) + StartPos;
}
}
// ***************************************************************************
// ***************************************************************************
// CTessFace
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
CTessFace CTessFace::CantMergeFace;
CTessFace CTessFace::MultipleBindFace;
// ***************************************************************************
CTessFace::CTessFace()
{
// Don't modify any of it!!
// Patch, SonLeft and SonRight nullity are very useful for MultiplePatch faces, and CantMergeFace.
Patch= NULL;
VBase=VLeft=VRight= NULL;
FBase=FLeft=FRight= NULL;
Father=SonLeft=SonRight= NULL;
Level=0;
ErrorMetricDate= 0;
// Size, Center, paramcoord undetermined.
TileMaterial= NULL;
// Very important (for split reasons). Init Tilefaces to NULL.
for(sint i=0;igetLandscape()->deleteTessFace(SonLeft);
Patch->getLandscape()->deleteTessFace(SonRight);
// update neighbors.
if(FBase) FBase->changeNeighbor(this, NULL);
if(FLeft) FLeft->changeNeighbor(this, NULL);
if(FRight) FRight->changeNeighbor(this, NULL);
FBase=FLeft=FRight= NULL;
*/
NL3D_PROFILE_LAND_ADD(ProfNTessFace, -1);
// Ensure correclty removed from landscape ShadowMap.
nlassert(ShadowMapTriId== -1);
}
// ***************************************************************************
float CTessFace::computeNearLimit()
{
// General formula for Level, function of Size, treshold etc...:
// WantedLevel= log2(BaseSize / sqrdist / RefineThreshold);
// <=> WantedLevel= log2( CurSize*2^Level / sqrdist / RefineThreshold).
// <=> WantedLevel= log2( ProjectedSize* 2^Level / RefineThreshold).
// <=> 2^WantedLevel= ProjectedSize* 2^Level / RefineThreshold.
// <=> ProjectedSize= (2^WantedLevel) * RefineThreshold / (2^Level);
// <=> ProjectedSize= (1<TileLimitLevel) / (1<TileLimitLevel) * (OOBigValue*(BigValue>>Level));
}
// ***************************************************************************
void CTessFace::computeTileErrorMetric()
{
// We must take a more correct errometric here: We must have sons face which have
// lower projectedsize than father. This is not the case if Center of face is taken (but when not in
// tile mode this is nearly the case). So take the min dist from 3 points.
float s0= (VBase->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float s1= (VLeft->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float s2= (VRight->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float sqrdist= minof(s0, s1, s2);
// It is also VERY important to take the min of 3, to ensure the split in TileMode when Far1 vertex begin
// to blend (see Patch::renderFar1() render).
// NB: VertexProgram geomorph take sqrdist= (SplitPoint - RefineCenter).sqrnorm();
// It's OK because geomorph will start "far" after the split.
if(sqrdist< CLandscapeGlobals::TileDistFarSqr)
{
float nearLimit;
nearLimit= CLandscapeGlobals::RefineThreshold * computeNearLimit();
// If we are not so subdivided.
if(ErrorMetric= CLandscapeGlobals::CurrentDate)
return;
CVector viewdir= SplitPoint - CLandscapeGlobals::RefineCenter;
float sqrdist= viewdir.sqrnorm();
// trivial formula.
//-----------------
ErrorMetric= Size/ sqrdist;
// Hoppe97 formula: k^2= a^2 * ("v-e"^2 - ((v-e).n)^2) / "v-e"^4.
//-----------------
// Can't do it because geomorph is made on Graphic card, so the simplier is the better.
// TileMode Impact.
//-----------------
/* TileMode Impact. We must split at least at TileLimitLevel, but only if the triangle
has a chance to enter in the TileDistFar sphere.
*/
if( LevelTileLimitLevel && sqrdist < sqr(CLandscapeGlobals::TileDistFar+MaxDistToSplitPoint) )
{
computeTileErrorMetric();
}
ErrorMetricDate= CLandscapeGlobals::CurrentDate;
}
// ***************************************************************************
inline float CTessFace::computeTileEMForUpdateRefine(float distSplitPoint, float distMinFace, float nearLimit)
{
float ema;
// Normal ErrorMetric simulation.
ema= Size / sqr(distSplitPoint);
// TileErrorMetric simulation.
if(distMinFace < CLandscapeGlobals::TileDistFar)
{
// If we are not so subdivided.
if( emaEndPos + VBase->EndPos)/2;
}
else
{
// If it is a square triangle, it will be splitted on middle of VLeft/VRight.
// So for good geomorph compute per vertex, we must have this SplitPoint on this middle.
SplitPoint= (VLeft->EndPos + VRight->EndPos)/2;
}
// compute MaxDistToSplitPoint
float d0= (VBase->EndPos - SplitPoint).sqrnorm();
float d1= (VLeft->EndPos - SplitPoint).sqrnorm();
float d2= (VRight->EndPos - SplitPoint).sqrnorm();
MaxDistToSplitPoint= max(d0, d1);
MaxDistToSplitPoint= max(MaxDistToSplitPoint, d2);
// Get the distance.
MaxDistToSplitPoint= sqrtf(MaxDistToSplitPoint);
}
// ***************************************************************************
void CTessFace::allocTileUv(TTileUvId id)
{
// TileFaces must have been build.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
// what src??
CTessVertex *vertexSrc;
switch(id)
{
case IdUvBase: vertexSrc= VBase; break;
case IdUvLeft: vertexSrc= VLeft; break;
case IdUvRight: vertexSrc= VRight; break;
default: vertexSrc = 0; nlstop; break;
};
// Do it for all possible pass
for(sint i=0;igetLandscape()->newTessNearVertex();
newNear->Src= vertexSrc;
TileFaces[i]->V[id]= newNear;
Patch->appendNearVertexToRenderList(TileMaterial, newNear);
// May Allocate/Fill VB. Do it after allocTileUv, because UVs are not comuted yet.
}
}
}
// ***************************************************************************
void CTessFace::checkCreateFillTileVB(TTileUvId id)
{
// TileFaces must have been build.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
for(sint i=0;iV[id];
// May Allocate/Fill VB.
Patch->checkCreateVertexVBNear(vertNear);
Patch->checkFillVertexVBNear(vertNear);
}
}
}
// ***************************************************************************
void CTessFace::checkFillTileVB(TTileUvId id)
{
// TileFaces must have been build.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
for(sint i=0;iV[id];
// May Fill VB.
Patch->checkFillVertexVBNear(vertNear);
}
}
}
// ***************************************************************************
void CTessFace::deleteTileUv(TTileUvId id)
{
// TileFaces must still exist.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
for(sint i=0;iV[id];
TileFaces[i]->V[id]=NULL;
// May delete this vertex from VB.
Patch->checkDeleteVertexVBNear(oldNear);
Patch->removeNearVertexFromRenderList(TileMaterial, oldNear);
Patch->getLandscape()->deleteTessNearVertex(oldNear);
}
}
}
// ***************************************************************************
void CTessFace::copyTileUv(TTileUvId dstId, CTessFace *srcFace, TTileUvId srcId)
{
// TileFaces must have been build.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
// Since this a ptr-copy, no need to add/remove the renderlist of near vertices.
for(sint i=0;iTileFaces[i]);
// copy from src.
CTessNearVertex *copyNear;
copyNear= srcFace->TileFaces[i]->V[srcId];
// copy to dst.
TileFaces[i]->V[dstId]= copyNear;
}
}
}
// ***************************************************************************
void CTessFace::heritTileUv(CTessFace *baseFace)
{
// TileFaces must have been build.
nlassert(TileFaces[NL3D_TILE_PASS_RGB0]);
for(sint i=0;iTileFaces[i]);
// VBase should be allocated.
nlassert(TileFaces[i]->V[IdUvBase]);
TileFaces[i]->V[IdUvBase]->initMiddleUv(
*baseFace->TileFaces[i]->V[IdUvLeft], *baseFace->TileFaces[i]->V[IdUvRight]);
}
}
}
// ***************************************************************************
void CTessFace::buildTileFaces()
{
nlassert(TileMaterial);
// Do nothgin for lightmap pass, of course.
for(sint i=0;iPass[i].PatchRdrPass)
{
TileFaces[i]= Patch->getLandscape()->newTileFace();
TileFaces[i]->V[IdUvBase]= NULL;
TileFaces[i]->V[IdUvLeft]= NULL;
TileFaces[i]->V[IdUvRight]= NULL;
}
}
}
// ***************************************************************************
void CTessFace::deleteTileFaces()
{
nlassert(TileMaterial);
// Do nothgin for lightmap pass, of course.
for(sint i=0;iPass[i].PatchRdrPass)
{
nlassert(TileFaces[i]);
Patch->getLandscape()->deleteTileFace(TileFaces[i]);
TileFaces[i]= NULL;
}
else
{
nlassert(TileFaces[i]==NULL);
}
}
}
// ***************************************************************************
bool CTessFace::emptyTileFaces()
{
// Do nothgin for lightmap pass, of course.
for(sint i=0;igetTileUvInfo(TileId, pass, alpha, orient, uvScaleBias, is256, uvOff);
// Orient the UV.
float u= uv.U;
float v= uv.V;
// Speed rotation.
switch(orient)
{
case 0:
uv.U= u;
uv.V= v;
break;
case 1:
uv.U= 1-v;
uv.V= u;
break;
case 2:
uv.U= 1-u;
uv.V= 1-v;
break;
case 3:
uv.U= v;
uv.V= 1-u;
break;
}
// Do the 256x256.
if(is256)
{
uv*= 0.5;
if(uvOff==2 || uvOff==3)
uv.U+= 0.5;
if(uvOff==1 || uvOff==2)
uv.V+= 0.5;
}
// Do the HalfPixel scale bias.
float hBiasXY, hBiasZ;
if(is256)
{
hBiasXY= CLandscapeGlobals::TilePixelBias256;
hBiasZ = CLandscapeGlobals::TilePixelScale256;
}
else
{
hBiasXY= CLandscapeGlobals::TilePixelBias128;
hBiasZ = CLandscapeGlobals::TilePixelScale128;
}
// Scale the UV.
uv.U*= uvScaleBias.z*hBiasZ;
uv.V*= uvScaleBias.z*hBiasZ;
uv.U+= uvScaleBias.x+hBiasXY;
uv.V+= uvScaleBias.y+hBiasXY;
}
// ***************************************************************************
void CTessFace::initTileUvLightmap(CParamCoord pointCoord, CParamCoord middle, CUV &uv)
{
// Get good coordinate according to patch orientation.
uv.U= pointCoord.S<=middle.S? 0.0f: 1.0f;
uv.V= pointCoord.T<=middle.T? 0.0f: 1.0f;
// Get Tile Lightmap Uv info: bias and scale.
CVector uvScaleBias;
Patch->getTileLightMapUvInfo(TileMaterial->TileS, TileMaterial->TileT, uvScaleBias);
// Scale the UV.
uv.U*= uvScaleBias.z;
uv.V*= uvScaleBias.z;
uv.U+= uvScaleBias.x;
uv.V+= uvScaleBias.y;
}
// ***************************************************************************
void CTessFace::initTileUvDLM(CParamCoord pointCoord, CUV &uv)
{
// get the dlm context from the patch.
CPatchDLMContext *dlmCtx= Patch->_DLMContext;
// mmust exist at creation of the tile.
nlassert(dlmCtx);
// get coord in 0..1 range, along the patch.
uv.U= pointCoord.getS();
uv.V= pointCoord.getT();
// ScaleBias according to the context.
uv.U*= dlmCtx->DLMUScale;
uv.V*= dlmCtx->DLMVScale;
uv.U+= dlmCtx->DLMUBias;
uv.V+= dlmCtx->DLMVBias;
}
// ***************************************************************************
void CTessFace::computeTileMaterial()
{
// 0. Compute TileId.
//-------------------
// Square Order Patch assumption: assume that when a CTessFace become a tile, his base can ONLY be diagonal...
/* a Patch:
A ________
|\ /|
| \ / |
C|__\B_/ |
| / \ |
| / \ |
|/______\|
Here, if OrderS*OrderT=2*2, ABC is a new CTessFace of a Tile, and AB is the diagonal of the tile.
Hence the middle of the tile is the middle of AB.
C must be created, but A and B may be created or copied from neighbor.
*/
CParamCoord middle(PVLeft,PVRight);
uint16 ts= ((uint16)middle.S * (uint16)Patch->OrderS) / 0x8000;
uint16 tt= ((uint16)middle.T * (uint16)Patch->OrderT) / 0x8000;
TileId= (uint8)(tt*Patch->OrderS + ts);
// 1. Compute Tile Material.
//--------------------------
// if base neighbor is already at TileLimitLevel just ptr-copy, else create the TileMaterial...
nlassert(!FBase || FBase->Patch!=Patch || FBase->Level<=Patch->TileLimitLevel);
bool copyFromBase;
copyFromBase= (FBase && FBase->Patch==Patch);
copyFromBase= copyFromBase && (FBase->Level==Patch->TileLimitLevel && FBase->TileMaterial!=NULL);
// NB: because of delete/recreateTileUvs(), FBase->TileMaterial may be NULL, even if face is at good TileLimitLevel.
if(copyFromBase)
{
TileMaterial= FBase->TileMaterial;
nlassert(FBase->TileId== TileId);
}
else
{
sint i;
TileMaterial= Patch->getLandscape()->newTileMaterial();
TileMaterial->TileS= uint8(ts);
TileMaterial->TileT= uint8(tt);
// Add this new material to the render list.
Patch->appendTileMaterialToRenderList(TileMaterial);
// First, get a lightmap for this tile. NB: important to do this after appendTileMaterialToRenderList(),
// because use TessBlocks.
Patch->getTileLightMap(ts, tt, TileMaterial->Pass[NL3D_TILE_PASS_LIGHTMAP].PatchRdrPass);
// Fill pass of multi pass material.
for(i=0;iPass[i].PatchRdrPass= Patch->getTileRenderPass(TileId, i);
}
// Fill Pass Info.
for(i=0;iPass[i].TileMaterial= TileMaterial;
}
// Do not append this tile to each RenderPass, done in preRender().
}
// 2. Compute Uvs.
//----------------
// NB: TileMaterial is already setup. Useful for initTileUvLightmap() and initTileUvRGBA().
// First, must create The TileFaces, according to the TileMaterial passes.
buildTileFaces();
// Must allocate the Base, and insert into list.
allocTileUv(IdUvBase);
// Init LightMap UV, in RGB0 pass, UV1..
initTileUvLightmap(PVBase, middle, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvBase]->PUv1);
// Init DLM Uv, in RGB0 pass, UV2.
initTileUvDLM(PVBase, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvBase]->PUv2);
// Init UV RGBA, for all pass (but lightmap).
for(sint i=0;iPass[i].PatchRdrPass)
{
// Face must exist.
nlassert(TileFaces[i]);
// Compute RGB UV in UV0.
initTileUvRGBA(i, false, PVBase, middle, TileFaces[i]->V[IdUvBase]->PUv0);
// If transition tile, compute alpha UV in UV1.
// Do it also for Additive, because may have Transition
if(i== NL3D_TILE_PASS_RGB1 || i==NL3D_TILE_PASS_RGB2 || i==NL3D_TILE_PASS_ADD)
initTileUvRGBA(i, true, PVBase, middle, TileFaces[i]->V[IdUvBase]->PUv1);
}
}
// UVs are computed, may create and fill VB.
checkCreateFillTileVB(IdUvBase);
// if base neighbor is already at TileLimitLevel just ptr-copy, else create the left/right TileUvs...
if(copyFromBase)
{
// Just cross-copy the pointers.
// Make Left near vertices be the Right vertices of FBase
copyTileUv(IdUvLeft, FBase, IdUvRight);
// Make Right near vertices be the Left vertices of FBase
copyTileUv(IdUvRight, FBase, IdUvLeft);
}
else
{
// Must allocate the left/right uv (and insert into list).
allocTileUv(IdUvLeft);
allocTileUv(IdUvRight);
// Init LightMap UV, in UvPass 0, UV1..
initTileUvLightmap(PVLeft, middle, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvLeft]->PUv1);
initTileUvLightmap(PVRight, middle, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvRight]->PUv1);
// Init DLM Uv, in RGB0 pass, UV2.
initTileUvDLM(PVLeft, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvLeft]->PUv2);
initTileUvDLM(PVRight, TileFaces[NL3D_TILE_PASS_RGB0]->V[IdUvRight]->PUv2);
// Init UV RGBA!
for(sint i=0;iPass[i].PatchRdrPass)
{
// Face must exist.
nlassert(TileFaces[i]);
// Compute RGB UV in UV0.
initTileUvRGBA(i, false, PVLeft, middle, TileFaces[i]->V[IdUvLeft]->PUv0);
initTileUvRGBA(i, false, PVRight, middle, TileFaces[i]->V[IdUvRight]->PUv0);
// If transition tile, compute alpha UV in UV1.
// Do it also for Additive, because may have Transition
if(i== NL3D_TILE_PASS_RGB1 || i==NL3D_TILE_PASS_RGB2 || i==NL3D_TILE_PASS_ADD)
{
initTileUvRGBA(i, true, PVLeft, middle, TileFaces[i]->V[IdUvLeft]->PUv1);
initTileUvRGBA(i, true, PVRight, middle, TileFaces[i]->V[IdUvRight]->PUv1);
}
}
}
// UVs are computed, may create and fill VB.
checkCreateFillTileVB(IdUvLeft);
checkCreateFillTileVB(IdUvRight);
}
}
// ***************************************************************************
void CTessFace::releaseTileMaterial()
{
// Hence, must release the tile. TileUvBase is differnet for each of leaves.
deleteTileUv(IdUvBase);
nlassert(!FBase || FBase->Level<=Patch->TileLimitLevel);
if(FBase && FBase->Level==Patch->TileLimitLevel && FBase->TileMaterial!=NULL)
{
// Do not release Uvs, since neighbor need it...
// But release faces.
deleteTileFaces();
// Do not release TileMaterial, since neighbor need it...
TileMaterial= NULL;
}
else
{
// release Uvs.
deleteTileUv(IdUvLeft);
deleteTileUv(IdUvRight);
// After, release Tile faces.
deleteTileFaces();
// Release the tile lightmap part. Do it before removeTileMaterialFromRenderList().
Patch->releaseTileLightMap(TileMaterial->TileS, TileMaterial->TileT);
// Remove this material from the render list. DO it before deletion of course :).
// NB: TileS/TileT still valid.
Patch->removeTileMaterialFromRenderList(TileMaterial);
Patch->getLandscape()->deleteTileMaterial(TileMaterial);
TileMaterial= NULL;
}
}
// ***************************************************************************
void CTessFace::updateNearFarVertices()
{
nlassert(VBase && FVBase);
nlassert(VLeft && FVLeft);
nlassert(VRight && FVRight);
FVBase->Src= VBase;
FVLeft->Src= VLeft;
FVRight->Src= VRight;
FVBase->PCoord= PVBase;
FVLeft->PCoord= PVLeft;
FVRight->PCoord= PVRight;
// Update VB for far vertices (if needed)
Patch->checkFillVertexVBFar(FVBase);
Patch->checkFillVertexVBFar(FVLeft);
Patch->checkFillVertexVBFar(FVRight);
// Near Vertices update (Src only).
for(sint i=0; iV[IdUvBase]->Src= VBase;
TileFaces[i]->V[IdUvLeft]->Src= VLeft;
TileFaces[i]->V[IdUvRight]->Src= VRight;
// Update VB for near vertices (if needed)
Patch->checkFillVertexVBNear(TileFaces[i]->V[IdUvBase]);
Patch->checkFillVertexVBNear(TileFaces[i]->V[IdUvLeft]);
Patch->checkFillVertexVBNear(TileFaces[i]->V[IdUvRight]);
}
}
}
// ***************************************************************************
void CTessFace::splitRectangular(bool propagateSplit)
{
CTessFace *f0= this;
CTessFace *f1= FBase;
// Rectangular case: FBase must exist.
nlassert(f1);
// In rectangular case, we split at the same time this and FBase (f0 and f1).
/*
Tesselation is:
lt rt lt top rt
--------------------- ---------------------
|---- | |\ |\ |
| ---- f1 | | \ f1l | \ f1r |
| ---- | --> | \ | \ |
| f0 ---- | | f0r \ | f0l \ |
| ---| | \| \|
--------------------- ---------------------
lb rb lb bot rb
Why? For symetry and bind/split reasons: FBase->SonLeft->VBase is always the good vertex to take
(see vertex binding).
*/
CParamCoord pclt= f1->PVLeft;
CParamCoord pclb= f0->PVBase;
CParamCoord pcrt= f1->PVBase;
CParamCoord pcrb= f1->PVRight;
CTessVertex *vlt= f1->VLeft;
CTessVertex *vlb= f0->VBase;
CTessVertex *vrt= f1->VBase;
CTessVertex *vrb= f1->VRight;
CTessFarVertex *farvlt= f1->FVLeft;
CTessFarVertex *farvlb= f0->FVBase;
CTessFarVertex *farvrt= f1->FVBase;
CTessFarVertex *farvrb= f1->FVRight;
// 1. create new vertices.
//------------------------
// Create splitted vertices.
CParamCoord pctop(f1->PVBase, f1->PVLeft);
CParamCoord pcbot(f0->PVBase, f0->PVLeft);
CTessVertex *vtop= NULL;
CTessVertex *vbot= NULL;
// Compute top.
if(f1->FLeft==NULL || f1->FLeft->isLeaf())
{
// The base neighbor is a leaf or NULL. So must create the new vertex.
vtop= Patch->getLandscape()->newTessVertex();
// Compute pos.
vtop->StartPos= (f1->VLeft->EndPos + f1->VBase->EndPos)/2;
vtop->EndPos= f1->Patch->computeVertex(pctop.getS(), pctop.getT());
// Init Pos= InitialPos. Important in the case of enforced split.
vtop->Pos= vtop->StartPos;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
vtop->MaxFaceSize= f1->Size;
vtop->MaxNearLimit= f1->computeNearLimit();
}
else
{
// Else, get from neighbor.
// NB: since *FLeft is not a leaf, FBase->SonLeft!=NULL...
// NB: this work with both rectangular and square triangles.
vtop= f1->FLeft->SonLeft->VBase;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
vtop->MaxFaceSize= max( vtop->MaxFaceSize, f1->Size);
vtop->MaxNearLimit= max( vtop->MaxNearLimit, f1->computeNearLimit());
}
// Compute bot.
if(f0->FLeft==NULL || f0->FLeft->isLeaf())
{
// The base neighbor is a leaf or NULL. So must create the new vertex.
vbot= Patch->getLandscape()->newTessVertex();
// Compute pos.
vbot->StartPos= (f0->VLeft->EndPos + f0->VBase->EndPos)/2;
vbot->EndPos= Patch->computeVertex(pcbot.getS(), pcbot.getT());
// Init Pos= InitialPos. Important in the case of enforced split.
vbot->Pos= vbot->StartPos;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
vbot->MaxFaceSize= f0->Size;
vbot->MaxNearLimit= f0->computeNearLimit();
}
else
{
// Else, get from neighbor.
// NB: since *FLeft is not a leaf, FBase->SonLeft!=NULL...
// NB: this work with both rectangular and square triangles.
vbot= f0->FLeft->SonLeft->VBase;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
vbot->MaxFaceSize= max( vbot->MaxFaceSize, f0->Size);
vbot->MaxNearLimit= max( vbot->MaxNearLimit, f0->computeNearLimit());
}
// In all case, must create new FarVertices, since rect split occurs on border!!
CTessFarVertex *farvtop= Patch->getLandscape()->newTessFarVertex();
CTessFarVertex *farvbot= Patch->getLandscape()->newTessFarVertex();
farvtop->Src= vtop;
farvbot->Src= vbot;
farvtop->PCoord= pctop;
farvbot->PCoord= pcbot;
Patch->appendFarVertexToRenderList(farvtop);
Patch->appendFarVertexToRenderList(farvbot);
// May Allocate/Fill VB.
// NB: vtop / vbot are well computed and ready for the fill in VB.
Patch->checkCreateVertexVBFar(farvtop);
Patch->checkFillVertexVBFar(farvtop);
Patch->checkCreateVertexVBFar(farvbot);
Patch->checkFillVertexVBFar(farvbot);
// For VertexProgram only, must refill the Far vertex of neighbor(s),
// because MaxFaceSize, and MaxNearLimit may have change.
if( CLandscapeGlobals::VertexProgramEnabled )
{
// f0
if( ! (f0->FLeft==NULL || f0->FLeft->isLeaf()) )
f0->FLeft->Patch->checkFillVertexVBFar(f0->FLeft->SonLeft->FVBase);
// f1
if( ! (f1->FLeft==NULL || f1->FLeft->isLeaf()) )
f1->FLeft->Patch->checkFillVertexVBFar(f1->FLeft->SonLeft->FVBase);
}
// 2. Create sons, and update links.
//----------------------------------
CTessFace *f0l, *f0r;
CTessFace *f1l, *f1r;
// create and bind Sons.
f0l= f0->SonLeft= Patch->getLandscape()->newTessFace();
f0r= f0->SonRight= Patch->getLandscape()->newTessFace();
f1l= f1->SonLeft= Patch->getLandscape()->newTessFace();
f1r= f1->SonRight= Patch->getLandscape()->newTessFace();
// subdivision left.
f0l->Patch= f0->Patch;
f0l->Father= f0;
f0l->Level= f0->Level+1;
f0l->Size= f0->Size*0.5f;
// subdivision right.
f0r->Patch= f0->Patch;
f0r->Father= f0;
f0r->Level= f0->Level+1;
f0r->Size= f0->Size*0.5f;
// subdivision left.
f1l->Patch= f1->Patch;
f1l->Father= f1;
f1l->Level= f1->Level+1;
f1l->Size= f1->Size*0.5f;
// subdivision right.
f1r->Patch= f1->Patch;
f1r->Father= f1;
f1r->Level= f1->Level+1;
f1r->Size= f1->Size*0.5f;
// Patch coordinates.
f0r->PVRight= pclt;
f0r->PVBase= pclb;
f0r->PVLeft= pcbot;
f1l->PVBase= pctop;
f1l->PVLeft= f0r->PVRight;
f1l->PVRight= f0r->PVLeft;
f0l->PVRight= pctop;
f0l->PVBase= pcbot;
f0l->PVLeft= pcrb;
f1r->PVBase= pcrt;
f1r->PVLeft= f0l->PVRight;
f1r->PVRight= f0l->PVLeft;
// link existing vertex.
f0r->VRight= vlt;
f0r->VBase= vlb;
f0r->VLeft= vbot;
f1l->VBase= vtop;
f1l->VLeft= f0r->VRight;
f1l->VRight= f0r->VLeft;
f0l->VRight= vtop;
f0l->VBase= vbot;
f0l->VLeft= vrb;
f1r->VBase= vrt;
f1r->VLeft= f0l->VRight;
f1r->VRight= f0l->VLeft;
// link Far vertices.
f0r->FVRight= farvlt;
f0r->FVBase= farvlb;
f0r->FVLeft= farvbot;
f1l->FVBase= farvtop;
f1l->FVLeft= f0r->FVRight;
f1l->FVRight= f0r->FVLeft;
f0l->FVRight= farvtop;
f0l->FVBase= farvbot;
f0l->FVLeft= farvrb;
f1r->FVBase= farvrt;
f1r->FVLeft= f0l->FVRight;
f1r->FVRight= f0l->FVLeft;
// link neigbhor faces.
f0r->FBase= f1l;
f1l->FBase= f0r;
f0l->FBase= f1r;
f1r->FBase= f0l;
f1l->FRight= f0l;
f0l->FRight= f1l;
f0r->FRight= f0->FRight;
if(f0->FRight)
f0->FRight->changeNeighbor(f0, f0r);
f1r->FRight= f1->FRight;
if(f1->FRight)
f1->FRight->changeNeighbor(f1, f1r);
// 4 links (all FLeft sons ) are stil invalid here.
f0l->FLeft= NULL;
f0r->FLeft= NULL;
f1l->FLeft= NULL;
f1r->FLeft= NULL;
// Neigbors pointers of undetermined splitted face are not changed. Must Doesn't change this.
// Used and Updated in section 5. ...
// 3. Update Tile infos.
//----------------------
// There is no update tileinfo with rectangular patch, since tiles are always squares. (TileLimitLevel>SquareLimitLevel).
// NB: but must test update of tile info for neighboring, ie 2 faces around the splits.
// For Vertex program only
if( CLandscapeGlobals::VertexProgramEnabled )
{
// if neighbor face splitted, and if 2 different patchs, we must update the Tile vertices
// because MaxFaceSize and MaxNearLimit may have changed.
if( f0->FLeft!=NULL && !f0->FLeft->isLeaf() && f0->FLeft->Patch!=Patch )
{
// If neighbors sons at tile level, must update their Tile vertices.
if( f0->FLeft->SonLeft->Level >= f0->FLeft->Patch->TileLimitLevel )
{
f0->FLeft->SonLeft->checkFillTileVB(IdUvBase);
f0->FLeft->SonRight->checkFillTileVB(IdUvBase);
}
}
// idem for f1.
if( f1->FLeft!=NULL && !f1->FLeft->isLeaf() && f1->FLeft->Patch!=Patch )
{
// If neighbors sons at tile level, must update their Tile vertices.
if( f1->FLeft->SonLeft->Level >= f1->FLeft->Patch->TileLimitLevel )
{
f1->FLeft->SonLeft->checkFillTileVB(IdUvBase);
f1->FLeft->SonRight->checkFillTileVB(IdUvBase);
}
}
}
// 4. Compute centers.
//--------------------
f0r->computeSplitPoint();
f0l->computeSplitPoint();
f1r->computeSplitPoint();
f1l->computeSplitPoint();
// 5. Propagate, or link sons of base.
//------------------------------------
for(sint i=0;i<2;i++)
{
CTessFace *f, *fl, *fr;
// TOP face.
if(i==0)
{
f= f1;
fl= f1l;
fr= f1r;
}
// then BOT face.
else
{
f= f0;
fl= f0l;
fr= f0r;
}
// If current face and FBase has sons, just links.
if(f->FLeft==NULL)
{
// Just update sons neighbors.
fl->FLeft= NULL;
fr->FLeft= NULL;
}
else if(!f->FLeft->isLeaf())
{
CTessFace *toLeft, *toRight;
toLeft= f->FLeft->SonLeft;
toRight= f->FLeft->SonRight;
// Cross connection of sons.
if( !f->FLeft->isRectangular() )
{
// Case neigbhor is square.
fl->FLeft= toLeft;
fr->FLeft= toRight;
toLeft->FRight= fl;
toRight->FLeft= fr;
}
else
{
// Case neigbhor is rectangle.
fl->FLeft= toRight;
fr->FLeft= toLeft;
toLeft->FLeft= fr;
toRight->FLeft= fl;
}
}
else if (propagateSplit)
{
// Warning: at each iteration, the pointer of FLeft may change (because of split() which can change the neighbor
// and so f).
while(f->FLeft->isLeaf())
f->FLeft->split();
// There is a possible bug here (maybe easily patched). Sons may have be propagated splitted.
// And problems may arise because this face hasn't yet good connectivity (especially for rectangles!! :) ).
nlassert(fl->isLeaf() && fr->isLeaf());
}
}
// 6. Must remove father from rdr list, and insert sons.
//------------------------------------------------------
// UGLY REFCOUNT SIDE EFFECT: do the append first.
Patch->appendFaceToRenderList(f0l);
Patch->appendFaceToRenderList(f0r);
Patch->appendFaceToRenderList(f1l);
Patch->appendFaceToRenderList(f1r);
Patch->removeFaceFromRenderList(f0);
Patch->removeFaceFromRenderList(f1);
// 7. Update priority list.
//------------------------------------------------------
// Since we are freshly splitted, unlink from any list, and link to the MergePriorityList, because must look
// now when should merge.
Patch->getLandscape()->_MergePriorityList.insert(0, 0, f0);
Patch->getLandscape()->_MergePriorityList.insert(0, 0, f1);
// Since we are split, no need to test father for merge, because it cannot!
if(f0->Father)
{
// remove father from any priority list.
f0->Father->unlinkInPList();
}
if(f1->Father)
{
// remove father from any priority list.
f1->Father->unlinkInPList();
}
}
// ***************************************************************************
void CTessFace::split(bool propagateSplit)
{
// 0. Some easy ending.
//---------------------
// Already splitted??
if(!isLeaf())
return;
// Don't do this!!!
//if(Level>=LS_MAXLEVEL)
// return;
// since split() may reach LS_MAXLEVEL, but enforce splits which outpass this stage!!
NL3D_PROFILE_LAND_ADD(ProfNSplits, 1);
// Special Rectangular case.
if(isRectangular())
{
splitRectangular(propagateSplit);
return;
}
// 1. Create sons, and update links.
//----------------------------------
// create and bind Sons.
SonLeft= Patch->getLandscape()->newTessFace();
SonRight= Patch->getLandscape()->newTessFace();
// subdivision left.
SonLeft->Patch= Patch;
SonLeft->Father= this;
SonLeft->Level= Level+1;
SonLeft->Size= Size*0.5f;
// subdivision right.
SonRight->Patch= Patch;
SonRight->Father= this;
SonRight->Level= Level+1;
SonRight->Size= Size*0.5f;
// link Left Son.
// link neighbor face.
SonLeft->FBase= FLeft;
if(FLeft) FLeft->changeNeighbor(this, SonLeft);
SonLeft->FLeft= SonRight;
SonLeft->FRight= NULL; // Temporary. updated later.
// link neighbor vertex.
SonLeft->VLeft= VBase;
SonLeft->VRight= VLeft;
// link neighbor Far vertex.
SonLeft->FVLeft= FVBase;
SonLeft->FVRight= FVLeft;
// Patch coordinates.
SonLeft->PVBase= CParamCoord(PVLeft, PVRight);
SonLeft->PVLeft= PVBase;
SonLeft->PVRight= PVLeft;
// linkRight Son.
// link neighbor face.
SonRight->FBase= FRight;
if(FRight) FRight->changeNeighbor(this, SonRight);
SonRight->FLeft= NULL; // Temporary. updated later.
SonRight->FRight= SonLeft;
// link neighbor vertex.
SonRight->VLeft= VRight;
SonRight->VRight= VBase;
// link neighbor Far vertex.
SonRight->FVLeft= FVRight;
SonRight->FVRight= FVBase;
// Patch coordinates.
SonRight->PVBase= CParamCoord(PVLeft, PVRight);
SonRight->PVLeft= PVRight;
SonRight->PVRight= PVBase;
// FBase->FBase==this. Must Doesn't change this. Used and Updated in section 5. ...
// 2. Update/Create Vertex infos.
//-------------------------------
// Must create/link *->VBase.
if(FBase==NULL || FBase->isLeaf())
{
// The base neighbor is a leaf or NULL. So must create the new vertex.
CTessVertex *newVertex= Patch->getLandscape()->newTessVertex();
SonRight->VBase= newVertex;
SonLeft->VBase= newVertex;
// Compute pos.
newVertex->StartPos= (VLeft->EndPos + VRight->EndPos)/2;
newVertex->EndPos= Patch->computeVertex(SonLeft->PVBase.getS(), SonLeft->PVBase.getT());
// Init Pos= InitialPos. Important in the case of enforced split.
newVertex->Pos= newVertex->StartPos;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
newVertex->MaxFaceSize= Size;
newVertex->MaxNearLimit= computeNearLimit();
}
else
{
// Else, get from neighbor.
// NB: since *FBase is not a leaf, FBase->SonLeft!=NULL...
// NB: this work with both rectangular and square triangles (see splitRectangular()).
SonRight->VBase= FBase->SonLeft->VBase;
SonLeft->VBase= FBase->SonLeft->VBase;
// For geomorph (VertexProgram or soft), compute MaxFaceSize and MaxNearLimit.
SonLeft->VBase->MaxFaceSize= max( SonLeft->VBase->MaxFaceSize, Size);
SonLeft->VBase->MaxNearLimit= max( SonLeft->VBase->MaxNearLimit, computeNearLimit());
}
// Must create/link *->FVBase.
// HERE, we must create a FarVertex too if the neighbor is not of the same patch as me.
if(FBase==NULL || FBase->isLeaf() || FBase->Patch!=Patch)
{
// The base neighbor is a leaf or NULL. So must create the new far vertex.
CTessFarVertex *newFar= Patch->getLandscape()->newTessFarVertex();
SonRight->FVBase= newFar;
SonLeft->FVBase= newFar;
// Compute.
newFar->Src= SonLeft->VBase;
newFar->PCoord= SonLeft->PVBase;
// Append.
Patch->appendFarVertexToRenderList(newFar);
// May Allocate/Fill VB.
// NB: SonLeft->VBase->Pos is well computed and ready for the fill in VB.
Patch->checkCreateVertexVBFar(newFar);
Patch->checkFillVertexVBFar(newFar);
// For VertexProgram only, must refill the Far vertex of neighbor,
// because MaxFaceSize, and MaxNearLimit may have change.
if( CLandscapeGlobals::VertexProgramEnabled && ! (FBase==NULL || FBase->isLeaf()) )
FBase->Patch->checkFillVertexVBFar(FBase->SonLeft->FVBase);
}
else
{
// Else, get from neighbor.
// NB: since *FBase is not a leaf, FBase->SonLeft!=NULL...
// NB: this work with both rectangular and square triangles (see splitRectangular()).
SonRight->FVBase= FBase->SonLeft->FVBase;
SonLeft->FVBase= FBase->SonLeft->FVBase;
// NB For VertexProgram only: no need to refill the Far vertex of neighbor, because neighbor is of same Patch
// So MaxNearLimit and MaxFaceSize should be the same.
}
// 3. Update Tile infos.
//----------------------
// NB: must do it before appendFaceToRenderList().
// NB: must do it after compute of SonLeft->VBase->Pos for good filling in VBuffer.
// There is no problem with rectangular patch, since tiles are always squares.
// If new tile ....
if(SonLeft->Level==Patch->TileLimitLevel)
{
SonLeft->computeTileMaterial();
SonRight->computeTileMaterial();
}
// else Tile herit.
else if(SonLeft->Level > Patch->TileLimitLevel)
{
heritTileMaterial();
}
// For Vertex program only
if( CLandscapeGlobals::VertexProgramEnabled )
{
// if neighbor face splitted, and if 2 different patchs, we must update the Tile vertices
// because MaxFaceSize and MaxNearLimit may have changed.
if( FBase!=NULL && !FBase->isLeaf() && FBase->Patch!=Patch )
{
// If neighbors sons at tile level, must update their Tile vertices.
if( FBase->SonLeft->Level >= FBase->Patch->TileLimitLevel )
{
FBase->SonLeft->checkFillTileVB(IdUvBase);
FBase->SonRight->checkFillTileVB(IdUvBase);
}
}
}
// 4. Compute centers.
//--------------------
SonRight->computeSplitPoint();
SonLeft->computeSplitPoint();
// 5. Propagate, or link sons of base.
//------------------------------------
// If current face and FBase has sons, just links.
if(FBase==NULL)
{
// Just update sons neighbors.
SonLeft->FRight= NULL;
SonRight->FLeft= NULL;
}
else if(!FBase->isLeaf())
{
CTessFace *toLeft, *toRight;
CTessFace *fl, *fr;
fl= SonLeft;
fr= SonRight;
toLeft= FBase->SonLeft;
toRight= FBase->SonRight;
// Cross connection of sons.
if(!FBase->isRectangular())
{
// Case neigbhor is square.
fl->FRight= toRight;
fr->FLeft= toLeft;
toLeft->FRight= fr;
toRight->FLeft= fl;
}
else
{
// Case neigbhor is rectangular.
fl->FRight= toLeft;
fr->FLeft= toRight;
toLeft->FLeft= fl;
toRight->FLeft= fr;
}
}
else if (propagateSplit)
{
// Warning: at each iteration, the pointer of FBase may change (because of split() which can change the neighbor
// and so "this").
while(FBase->isLeaf())
FBase->split();
// There is a possible bug here (maybe easily patched). Sons may have be propagated splitted.
// And problems may arise because this face hasn't yet good connectivity.
nlassert(SonLeft->isLeaf() && SonRight->isLeaf());
}
// 6. Must remove father from rdr list, and insert sons.
//------------------------------------------------------
// UGLY REFCOUNT SIDE EFFECT: do the append first.
Patch->appendFaceToRenderList(SonLeft);
Patch->appendFaceToRenderList(SonRight);
Patch->removeFaceFromRenderList(this);
// 7. Update priority list.
//------------------------------------------------------
// Since we are freshly splitted, unlink from any list, and link to the MergePriorityList, because must look
// now when should merge.
Patch->getLandscape()->_MergePriorityList.insert(0, 0, this);
// Since we are split, no need to test father for merge, because it cannot!
if(Father)
{
// remove father from any priority list.
Father->unlinkInPList();
}
}
// ***************************************************************************
bool CTessFace::canMerge(bool testEm)
{
if(this== &CantMergeFace)
return false;
nlassert(!isLeaf());
// Test diamond config (sons must be leaves).
if(!SonLeft->isLeaf())
return false;
if(!SonRight->isLeaf())
return false;
// If Errormetric must be considered for this test.
if(testEm)
{
updateErrorMetric();
float ps2= ErrorMetric;
ps2*= CLandscapeGlobals::OORefineThreshold;
if(ps2>=1.0f)
return false;
}
// Then test neighbors.
RecursMarkCanMerge= true;
bool ok= true;
if(!isRectangular())
{
if(FBase && !FBase->RecursMarkCanMerge)
{
if(!FBase->canMerge(testEm))
ok= false;
}
}
else
{
// Rectangular case. May have a longer propagation...
if(FBase && !FBase->RecursMarkCanMerge)
{
if(!FBase->canMerge(testEm))
ok= false;
}
if(ok && FLeft && !FLeft->RecursMarkCanMerge)
{
if(!FLeft->canMerge(testEm))
ok= false;
}
}
// Must not return false in preceding tests, because must set RecursMarkCanMerge to false.
RecursMarkCanMerge= false;
return ok;
}
// ***************************************************************************
void CTessFace::doMerge()
{
// Assume that canMerge() return true.
// And Assume that !isLeaf().
nlassert(!isLeaf());
if(!isRectangular())
{
// 1. Let's merge vertex.
//-----------------------
// Delete vertex, only if not already done by the neighbor (ie neighbor not already merged to a leaf).
// NB: this work even if neigbor is rectnagular.
if(!FBase || !FBase->isLeaf())
Patch->getLandscape()->deleteTessVertex(SonLeft->VBase);
// Delete Far Vertex. Idem, but test too if != patch...
if(!FBase || !FBase->isLeaf() || FBase->Patch!=Patch)
{
// May delete this vertex from VB.
Patch->checkDeleteVertexVBFar(SonLeft->FVBase);
Patch->removeFarVertexFromRenderList(SonLeft->FVBase);
Patch->getLandscape()->deleteTessFarVertex(SonLeft->FVBase);
}
// 2. Must remove sons from rdr list, and insert father.
//------------------------------------------------------
// Must do it BEFORE the TileFaces are released.
// UGLY REFCOUNT SIDE EFFECT: do the append first.
Patch->appendFaceToRenderList(this);
Patch->removeFaceFromRenderList(SonLeft);
Patch->removeFaceFromRenderList(SonRight);
// 3. Let's merge Uv.
//-------------------
// Delete Uv.
// Must do it for this and FBase separately, since they may not have same tile level (if != patch).
if(SonLeft->Level== Patch->TileLimitLevel)
{
// Square patch assumption: the sons are not of the same TileId/Patch.
nlassert(!sameTile(SonLeft, SonRight));
// release tiles: NearVertices, TileFaces, and TileMaterial.
SonLeft->releaseTileMaterial();
SonRight->releaseTileMaterial();
}
else if(SonLeft->Level > Patch->TileLimitLevel)
{
// Delete Uv, only if not already done by the neighbor (ie neighbor not already merged to a leaf).
// But Always delete if neighbor exist and has not same tile as me.
// NB: this work with rectangular neigbor patch, since sameTile() will return false if different patch.
if(!FBase || !FBase->isLeaf() || !sameTile(this, FBase))
{
SonLeft->deleteTileUv(IdUvBase);
}
// In all case, must delete the tilefaces of those face.
SonLeft->deleteTileFaces();
SonRight->deleteTileFaces();
}
// 4. Let's merge Face.
//-------------------
// Change father 's neighbor pointers.
FLeft= SonLeft->FBase;
if(FLeft) FLeft->changeNeighbor(SonLeft, this);
FRight= SonRight->FBase;
if(FRight) FRight->changeNeighbor(SonRight, this);
// delete sons.
Patch->getLandscape()->deleteTessFace(SonLeft);
Patch->getLandscape()->deleteTessFace(SonRight);
SonLeft=NULL;
SonRight=NULL;
// If not already done, merge the neighbor.
if(FBase!=NULL && !FBase->isLeaf())
{
FBase->doMerge();
}
}
else
{
// Rectangular case.
// Since minimum Order is 2, Sons of rectangular face are NEVER at TileLimitLevel. => no Uv merge to do.
nlassert(SonLeft->Level< Patch->TileLimitLevel);
nlassert(FBase);
// 1. Let's merge vertex.
//-----------------------
// Delete vertex, only if not already done by the neighbor (ie neighbor not already merged to a leaf).
// NB: this work even if neigbor is rectangular (see tesselation rules in splitRectangular()).
if(!FLeft || !FLeft->isLeaf())
Patch->getLandscape()->deleteTessVertex(SonLeft->VBase);
// Delete Far Vertex. Rect patch: neightb must be of a != pathc as me => must delete FarVertex.
nlassert(!FLeft || FLeft->Patch!=Patch);
// May delete this vertex from VB.
Patch->checkDeleteVertexVBFar(SonLeft->FVBase);
Patch->removeFarVertexFromRenderList(SonLeft->FVBase);
Patch->getLandscape()->deleteTessFarVertex(SonLeft->FVBase);
// 2. Must remove sons from rdr list, and insert father.
//------------------------------------------------------
// UGLY REFCOUNT SIDE EFFECT: do the append first.
Patch->appendFaceToRenderList(this);
Patch->removeFaceFromRenderList(SonLeft);
Patch->removeFaceFromRenderList(SonRight);
// 3. Let's merge Face.
//-------------------
// Change father 's neighbor pointers (see splitRectangular()).
FRight= SonRight->FRight;
if(FRight) FRight->changeNeighbor(SonRight, this);
// delete sons.
Patch->getLandscape()->deleteTessFace(SonLeft);
Patch->getLandscape()->deleteTessFace(SonRight);
SonLeft=NULL;
SonRight=NULL;
// First, do it for my rectangular co-worker FBase (if not already done).
if(!FBase->isLeaf())
{
FBase->doMerge();
}
// If not already done, merge the neighbor.
if(FLeft!=NULL && !FLeft->isLeaf())
{
FLeft->doMerge();
}
}
// Update priority list.
//------------------------------------------------------
// Since we are freshly merged, unlink from any list, and link to the SplitPriorityList, because must look
// now when we should split again.
Patch->getLandscape()->_SplitPriorityList.insert(0, 0, this);
// since we are now merged maybe re-insert father in priority list.
if(Father)
{
nlassert(!Father->isLeaf());
// If sons of father are both leaves (ie this, and the other (complexe case if rectangle) )
if( Father->SonLeft->isLeaf() && Father->SonRight->isLeaf() )
{
Patch->getLandscape()->_MergePriorityList.insert(0, 0, Father);
}
}
}
// ***************************************************************************
bool CTessFace::merge()
{
// Must not be a leaf.
nlassert(!isLeaf());
// 0. Verify if merge is posible.
//----------------------------
if(!canMerge(false))
return false;
NL3D_PROFILE_LAND_ADD(ProfNMerges, 1);
// 1. Let's merge the face.
//-----------------------
// Propagation is done in doMerge().
doMerge();
return true;
}
// ***************************************************************************
void CTessFace::refineAll()
{
NL3D_PROFILE_LAND_ADD(ProfNRefineFaces, 1);
NL3D_PROFILE_LAND_ADD(ProfNRefineLeaves, isLeaf()?1:0);
/*
if(psRefineThreshold*2), the face is fully splitted/geomoprhed (tests reported on sons...).
*/
// Test for Split or merge.
//-----------------------
{
NL3D_PROFILE_LAND_ADD(ProfNRefineComputeFaces, 1);
updateErrorMetric();
float ps=ErrorMetric;
ps*= CLandscapeGlobals::OORefineThreshold;
// 1.0f is the point of split().
// 2.0f is the end of geomorph.
// Test split/merge.
//---------------------
// If wanted, not already done, and limit not reached, split().
if(isLeaf())
{
if(ps>1.0f && Level< (Patch->TileLimitLevel + CLandscapeGlobals::TileMaxSubdivision) )
split();
}
else
{
// Else, if splitted, must merge (if not already the case).
if(ps<1.0f)
{
// Merge only if agree, and neighbors agree.
// canMerge() test all the good thing: FBase==CantMergeFace, or this is rectangular etc...
// The test is propagated to neighbors.
if(canMerge(true))
{
merge();
}
}
}
}
// Recurs.
//-----------------------
if(SonLeft)
{
SonLeft->refineAll();
SonRight->refineAll();
}
}
// ***************************************************************************
// Some updateRefine***() Doc:
// Split or merge, and meaning of errorMetric:
/*
if(errorMetricRefineThreshold*2), the face is fully splitted/geomoprhed.
*/
// Compute distNormalSplitMerge: distance from refineCenter to normal split/merge (ie without tile transition):
/*
normal ErrorMetric formula is:
em = Size*OORefineThreshold/ dist^2; with dist == (SplitPoint - CLandscapeGlobals::RefineCenter).norm()
So inverse this function and we have:
dist= sqrt(Size*OORefineThreshold/em).
Split or merge is when em==1, so
distSplitMerge= sqrt(Size*OORefineThreshold)
*/
// Compute distTileTransSplitMerge.
/* When we are sure that CLandscapeGlobals::TileDistNear < distMinFace < CLandscapeGlobals::TileDistFar,
the clamp in the original formula is skipped
So the TileErrorMetric formula is:
{
ema= Sife*OORefineThreshold / distSP^2.
f= (TileDistFar^2 - distMinFace^2) * OOTileDeltaDist^2
f= f ^ 4. // no clamp. see above.
emb= NL*f + ema*(1-f)
emFinal= max(ema, emb).
}
The problem is that the formula is too complex (degree 8 equation).
So search for the result recursively.
*/
// Quadrant Selection
/*
Quadrant/Direction is interesting for updateRefineSplit() only.
In the 2 most simples cases, the action we want is "Know when we enters in a bounding volume"
This fit well for Quadrant notion.
In the case "TileDistNear to TileDistFar", the rule is too complicated and there is also
the notion of "Know when we LEAVE the TileDistFar sphere around the face" which is incompatible
with Quadrant behavior (since can go in any direction to leave the sphere).
updateRefineMerge() are at least all notion of "Know when we LEAVE the SplitSphere around the SplitPoint"
which is incompatible with Quadrant behavior.
This is why updateRefineMerge() don't bother at all quadrant, and so the _MergePriorityList is inited with 0
quadrants.
*/
// ***************************************************************************
void CTessFace::updateRefineSplit()
{
NL3D_PROFILE_LAND_ADD(ProfNRefineFaces, 1);
nlassert(Patch);
// The face must be not splitted, because tested for split.
nlassert(isLeaf());
/*
NB: see above for some updateRefine*** doc.
*/
// Test for Split.
//-----------------------
bool splitted= false;
{
updateErrorMetric();
float ps=ErrorMetric;
ps*= CLandscapeGlobals::OORefineThreshold;
// 1.0f is the point of split().
// 2.0f is the end of geomorph.
// Test split.
//---------------------
// If wanted and limit not reached, split().
if(ps>1.0f && Level< (Patch->TileLimitLevel + CLandscapeGlobals::TileMaxSubdivision) )
{
split();
// if split ok
if(!isLeaf())
{
splitted= true;
}
}
}
// Insert the face in the priority list.
//-----------------------
// If splitted, then insertion in Landscape->MergePriorityList at 0 has been done. so nothing to update.
// Else, must compute when whe should re-test.
if(!splitted)
{
// the face is not splitted here.
nlassert(isLeaf());
float minDeltaDistToUpdate;
// by default insert in the quadrant-less rolling table.
uint quadrantId= 0;
CVector dirToSplitPoint= SplitPoint - CLandscapeGlobals::RefineCenter;
// The distance of SplitPoint to center.
float distSplitPoint= dirToSplitPoint.norm();
// The distance where we should split/merge. see updateRefin() doc.
float distNormalSplitMerge= (float)sqrt(Size*CLandscapeGlobals::OORefineThreshold);
// If the face is at its max subdivision
if(Level>=Patch->TileLimitLevel+CLandscapeGlobals::TileMaxSubdivision)
{
// special case: the face do not need to be tested for splitting, because Max subdivision reached.
// Hence just unlink from any list, and return.
unlinkInPList();
return;
}
else if(Level>=Patch->TileLimitLevel)
{
// Always normal ErrorMetric. Because Faces at Tile level decide to split or merge their sons independently
// of "Tile ErrorMetric".
// The test is "when do we enter in the Split area?", so we can use Quadrant PriorityList
quadrantId= Patch->getLandscape()->_SplitPriorityList.selectQuadrant(dirToSplitPoint);
// compute distance to split as default "No Quadrant"
minDeltaDistToUpdate= distSplitPoint - distNormalSplitMerge;
// If a quadrant is selected, try to use it
if(quadrantId>0)
{
const CVector &quadrantDir= Patch->getLandscape()->_SplitPriorityList.getQuadrantDirection(quadrantId);
// We must not approach the SplitPoint at distNormalSplitMerge
float dMin= quadrantDir*dirToSplitPoint - distNormalSplitMerge;
// If the dist with quadrant is too small then use the std way (without quadrant).
if( dMinEndPos - CLandscapeGlobals::RefineCenter;
CVector dirToV1= VLeft->EndPos - CLandscapeGlobals::RefineCenter;
CVector dirToV2= VRight->EndPos - CLandscapeGlobals::RefineCenter;
float s0= dirToV0.sqrnorm();
float s1= dirToV1.sqrnorm();
float s2= dirToV2.sqrnorm();
float distMinFace= (float)sqrt( minof(s0, s1, s2) );
// compute the delta distance to the normal split point. See above for doc.
float normalEMDeltaDist;
normalEMDeltaDist= distSplitPoint - distNormalSplitMerge;
/*
There is 3 possibles cases, according to level, and the distances minFaceDist:
*/
/// TileDistFar to +oo.
if( distMinFace > CLandscapeGlobals::TileDistFar )
{
// The test is "when do we enter in the Split area OR in TileDistFar area?", so we can use Quadrant PriorityList
quadrantId= Patch->getLandscape()->_SplitPriorityList.selectQuadrant(dirToSplitPoint);
// compute deltaDist as default "Direction less quadrant"
minDeltaDistToUpdate= normalEMDeltaDist;
// We must know when we enter in TileErrorMetric zone, because the computing is different.
minDeltaDistToUpdate= min(minDeltaDistToUpdate, distMinFace - CLandscapeGlobals::TileDistFar);
// try with quadrant if > 0.
if(quadrantId>0)
{
const CVector &quadrantDir= Patch->getLandscape()->_SplitPriorityList.getQuadrantDirection(quadrantId);
// We must not approach the SplitPoint at distNormalSplitMerge
float dMin= quadrantDir*dirToSplitPoint - distNormalSplitMerge;
// and we must not reach one of the 3 sphere (Vi, TileDistFar).
float d0 = quadrantDir*dirToV0 - CLandscapeGlobals::TileDistFar;
float d1 = quadrantDir*dirToV1 - CLandscapeGlobals::TileDistFar;
float d2 = quadrantDir*dirToV2 - CLandscapeGlobals::TileDistFar;
// take min dist
dMin= minof(dMin, d0, d1, d2);
// If the dist with quadrant is too small then use the std way (without quadrant).
if( dMin CLandscapeGlobals::TileDistNear )
{
// NB: can't use quadrant behavior here. Leave quadrantId at 0.
// Profile
NL3D_PROFILE_LAND_ADD(ProfNRefineInTileTransition, 1);
// Compute distance to split/Merge in TileTransition
float distTileTransSplitMerge;
float maxDeltaDist= 8;
float minDeltaDist= 0;
uint nbRecurs= 6;
float nearLimit;
nearLimit= CLandscapeGlobals::RefineThreshold * computeNearLimit();
// search the distance to split recursively.
for(uint i=0; i< nbRecurs; i++)
{
float pivotDeltaDist= (maxDeltaDist-minDeltaDist)/2;
// If the em computed with this distance is still <1 (ie merged), then we can move further.
if ( computeTileEMForUpdateRefine(distSplitPoint-pivotDeltaDist, distMinFace-pivotDeltaDist, nearLimit ) < 1)
minDeltaDist= pivotDeltaDist;
// else we must move not as far
else
maxDeltaDist= pivotDeltaDist;
}
// And so take the minimum resulting delta distance
distTileTransSplitMerge= minDeltaDist;
// take the min with distance of distMinFace to the TileDistNear and TileDistFar sphere, because formula change at
// those limits.
minDeltaDistToUpdate= min(distTileTransSplitMerge, CLandscapeGlobals::TileDistFar - distMinFace );
minDeltaDistToUpdate= min(minDeltaDistToUpdate, distMinFace - CLandscapeGlobals::TileDistNear);
}
/// 0 to TileDistNear.
else
{
// because the face is not a Tile Level (ie LevelTileLimitLevel), it should be splitted,
// and won't merge until reaching at least the TileDistNear sphere.
// if not splited (should not arise), force the split next time.
minDeltaDistToUpdate= 0;
}
}
// Profile.
if(minDeltaDistToUpdate<0.0625)
{
NL3D_PROFILE_LAND_ADD(ProfNRefineWithLowDistance, 1);
}
// insert in the Split priority list.
// Until the RefineCenter move under minDeltaDistToUpdate, we don't need to test face.
Patch->getLandscape()->_SplitPriorityList.insert(quadrantId, minDeltaDistToUpdate, this);
}
}
// ***************************************************************************
void CTessFace::updateRefineMerge()
{
NL3D_PROFILE_LAND_ADD(ProfNRefineFaces, 1);
nlassert(Patch);
// The face must be splitted, because tested for merge.
nlassert(!isLeaf());
/*
NB: see above for some updateRefine*** doc.
*/
// Test for merge.
//-----------------------
bool merged= false;
{
updateErrorMetric();
float ps=ErrorMetric;
ps*= CLandscapeGlobals::OORefineThreshold;
// 1.0f is the point of split().
// 2.0f is the end of geomorph.
// Test merge.
//---------------------
// Else, must merge ??
if(ps<1.0f)
{
// Merge only if agree, and neighbors agree.
// canMerge() test all the good thing: FBase==CantMergeFace, or this is rectangular etc...
// The test is propagated to neighbors.
if(canMerge(true))
{
merge();
// NB: here, merge() is not propagated to fathers (supposed to be not very useful).
if(isLeaf())
{
merged= true;
}
}
}
}
// Insert the face in the priority list.
//-----------------------
// If merged, then insertion in Landscape->SplitPriorityList at 0 has been done. so nothing to update.
// Else, must compute when whe should re-test.
if(!merged)
{
// the face is splitted here.
nlassert(!isLeaf());
float minDeltaDistToUpdate;
// The distance of SplitPoint to center.
float distSplitPoint= (SplitPoint - CLandscapeGlobals::RefineCenter).norm();
// Compute distance from refineCenter to normal split/merge (ie without tile transition).
float distNormalSplitMerge= (float)sqrt(Size*CLandscapeGlobals::OORefineThreshold);
// If the face is at its max subdivision
if(Level>=Patch->TileLimitLevel+CLandscapeGlobals::TileMaxSubdivision)
{
// since the face is splitted, then must test always this face, because we must merge it (as soon as it is possible).
minDeltaDistToUpdate= 0;
}
else if(Level>=Patch->TileLimitLevel)
{
// Always normal ErrorMetric. Because Faces at Tile level decide to split or merge their sons independently
// of "Tile ErrorMetric".
// since splitted, compute distance to merge.
minDeltaDistToUpdate= distNormalSplitMerge - distSplitPoint;
// NB: it is possible that minDeltaDistToUpdate<0. A good example is when we are enforced split.
// Then, distSplitMerge may be < distSplitPoint, meaning we should have not split, but a neigbhor has enforced us.
// So now, must test every frame if we can merge....
minDeltaDistToUpdate= max( 0.f, minDeltaDistToUpdate );
}
else
{
// Compute Distance of the face from RefineCenter. It is the min of the 3 points, as in computeTileErrorMetric().
float s0= (VBase->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float s1= (VLeft->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float s2= (VRight->EndPos - CLandscapeGlobals::RefineCenter).sqrnorm();
float distMinFace= (float)sqrt( minof(s0, s1, s2) );
// compute the delta distance to the normal split point. See above for doc.
float normalEMDeltaDist;
normalEMDeltaDist= distNormalSplitMerge - distSplitPoint;
normalEMDeltaDist= max( 0.f, normalEMDeltaDist );
/*
There is 3 possibles cases, according to level, and the distances minFaceDist:
*/
/// TileDistFar to +oo.
if( distMinFace > CLandscapeGlobals::TileDistFar )
{
// normal geomorph. Any face compute the distance to the SplitPoint, and take min with distance to
// the TileDistFar sphere.
minDeltaDistToUpdate= normalEMDeltaDist;
// We must know when we enter in TileErrorMetric zone, because the computing is different.
minDeltaDistToUpdate= min(minDeltaDistToUpdate, distMinFace - CLandscapeGlobals::TileDistFar);
}
/// TileDistNear to TileDistFar.
else if( distMinFace > CLandscapeGlobals::TileDistNear )
{
// Profile
NL3D_PROFILE_LAND_ADD(ProfNRefineInTileTransition, 1);
// Compute distance to split/Merge in TileTransition
float distTileTransSplitMerge;
float maxDeltaDist= 8;
float minDeltaDist= 0;
uint nbRecurs= 6;
float nearLimit;
nearLimit= CLandscapeGlobals::RefineThreshold * computeNearLimit();
// Since splitted, compute distance to merge.
// search the distance recursively.
for(uint i=0; i< nbRecurs; i++)
{
float pivotDeltaDist= (maxDeltaDist-minDeltaDist)/2;
// If the em computed with this distance is still >1 (ie splitted), then we can move further.
if ( computeTileEMForUpdateRefine(distSplitPoint+pivotDeltaDist, distMinFace+pivotDeltaDist, nearLimit ) > 1)
minDeltaDist= pivotDeltaDist;
// else we must move not as far
else
maxDeltaDist= pivotDeltaDist;
}
// And so take the minimum resulting delta distance
distTileTransSplitMerge= minDeltaDist;
// take the min with distance of distMinFace to the TileDistNear and TileDistFar sphere, because formula change at
// those limits.
minDeltaDistToUpdate= min(distTileTransSplitMerge, CLandscapeGlobals::TileDistFar - distMinFace );
minDeltaDistToUpdate= min(minDeltaDistToUpdate, distMinFace - CLandscapeGlobals::TileDistNear);
}
/// 0 to TileDistNear.
else
{
// because the face is not a Tile Level (ie LevelTileLimitLevel), it should be splitted,
// and won't merge until reaching at least the TileDistNear sphere.
// Since splitted, Must enter in TileErrorMetric area to know when to merge.
minDeltaDistToUpdate= CLandscapeGlobals::TileDistNear - distMinFace;
}
}
// Merge Refine Threshold: because of enforced splits, we have lot of faces whit minDeltaDistToUpdate<0, because
// they alwayas want to merge. To avoid this, add a delta, which delay the test for merge.
// The caveat is that faces which do not need this may merge later. But 2 meters won't add too many faces.
minDeltaDistToUpdate+= NL3D_REFINE_MERGE_THRESHOLD;
// insert in the Merge priority list.
// Until the RefineCenter move under minDeltaDistToUpdate, we don't need to test face.
Patch->getLandscape()->_MergePriorityList.insert(0, minDeltaDistToUpdate, this);
}
}
// ***************************************************************************
void CTessFace::unbind()
{
// NB: since CantMergeFace has a NULL patch ptr, it is unbound too.
// Square case.
//=============
if(!isRectangular())
{
// Change Left/Right neighbors.
if(isLeaf())
{
// FLeft and FRight pointers are only valid in Leaves nodes.
if(FLeft && FLeft->Patch!=Patch)
{
FLeft->changeNeighbor(this, NULL);
FLeft= NULL;
}
if(FRight && FRight->Patch!=Patch)
{
FRight->changeNeighbor(this, NULL);
FRight= NULL;
}
}
// Change Base neighbors.
if(FBase && FBase->Patch!=Patch)
{
CTessFace *oldNeigbhorFace= FBase;
FBase->changeNeighbor(this, NULL);
FBase= NULL;
if(!isLeaf())
{
// Duplicate the VBase of sons, so the unbind is correct and no vertices are shared.
CTessVertex *old= SonLeft->VBase;
SonLeft->VBase= Patch->getLandscape()->newTessVertex();
*(SonLeft->VBase)= *old;
SonRight->VBase= SonLeft->VBase;
// For geomorph (VertexProgram or soft), compute good MaxFaceSize and MaxNearLimit (change since unbinded)
// update us.
SonLeft->VBase->MaxFaceSize= Size;
SonLeft->VBase->MaxNearLimit= computeNearLimit();
// update our neigbhor, only if not a multiple patch face.
if(oldNeigbhorFace->Patch)
{
old->MaxFaceSize= oldNeigbhorFace->Size;
old->MaxNearLimit= oldNeigbhorFace->computeNearLimit();
}
}
}
}
// Rectangular case.
//==================
else
{
// Doens't need to test FBase, since must be same patch.
// In rectangular, FLeft has the behavior of FBase in square case.
if(FLeft && FLeft->Patch!=Patch)
{
CTessFace *oldNeigbhorFace= FLeft;
FLeft->changeNeighbor(this, NULL);
FLeft= NULL;
if(!isLeaf())
{
// Duplicate the VBase of sons, so the unbind is correct and no vertices are shared.
// NB: this code is a bit different from square case.
CTessVertex *old= SonLeft->VBase;
SonLeft->VBase= Patch->getLandscape()->newTessVertex();
*(SonLeft->VBase)= *old;
// This is the difference: (see rectangle tesselation rules).
SonRight->VLeft= SonLeft->VBase;
// Yoyo_patch_himself (12/02/2001): I forgot this one!!
nlassert(FBase && FBase->SonLeft);
FBase->SonLeft->VRight= SonLeft->VBase;
// For geomorph (VertexProgram or soft), compute good MaxFaceSize and MaxNearLimit (change since unbinded)
// update us.
SonLeft->VBase->MaxFaceSize= Size;
SonLeft->VBase->MaxNearLimit= computeNearLimit();
// update our neigbhor, only if not a multiple patch face.
if(oldNeigbhorFace->Patch)
{
old->MaxFaceSize= oldNeigbhorFace->Size;
old->MaxNearLimit= oldNeigbhorFace->computeNearLimit();
}
}
}
// But FRight still valid in leaves nodes only.
if(isLeaf())
{
if(FRight && FRight->Patch!=Patch)
{
FRight->changeNeighbor(this, NULL);
FRight= NULL;
}
}
}
// Propagate unbind.
//==================
if(!isLeaf())
{
// update sons vertex pointers (since they may have been updated by me or my grandfathers).
if(!isRectangular())
{
SonLeft->VLeft= VBase;
SonLeft->VRight= VLeft;
SonRight->VLeft= VRight;
SonRight->VRight= VBase;
}
else
{
// Rectangular case. Update only ptrs which may have changed.
SonLeft->VLeft= VLeft;
SonRight->VBase= VBase;
SonRight->VRight= VRight;
}
// Must re-create good Vertex links for Far and Near Vertices!!!
SonLeft->updateNearFarVertices();
SonRight->updateNearFarVertices();
if(isRectangular())
{
//NB: must do this for Base neighbor (see unbind() rectangular case...).
nlassert(FBase && FBase->SonLeft && FBase->SonRight);
FBase->SonLeft->updateNearFarVertices();
FBase->SonRight->updateNearFarVertices();
}
// unbind the sons.
SonLeft->unbind();
SonRight->unbind();
}
}
// ***************************************************************************
void CTessFace::forceMerge()
{
if(this== &CantMergeFace)
return;
if(!isLeaf())
{
// First, force merge of Sons and neighbor sons, to have a diamond configuration.
SonLeft->forceMerge();
SonRight->forceMerge();
// forceMerge of necessary neighbors.
RecursMarkForceMerge=true;
if(!isRectangular())
{
if(FBase && !FBase->RecursMarkForceMerge)
FBase->forceMerge();
}
else
{
// Rectangular case. May have a longer propagation...
if(FBase && !FBase->RecursMarkForceMerge)
FBase->forceMerge();
if(FLeft && !FLeft->RecursMarkForceMerge)
FLeft->forceMerge();
}
RecursMarkForceMerge=false;
// If still a parent, merge.
if(!isLeaf())
merge();
}
}
// ***************************************************************************
void CTessFace::forceMergeAtTileLevel()
{
if(this== &CantMergeFace)
return;
if(!isLeaf())
{
SonLeft->forceMergeAtTileLevel();
SonRight->forceMergeAtTileLevel();
}
else
{
// If we are at tile subdivision, we must force our sons to merge.
if(Level==Patch->TileLimitLevel)
forceMerge();
}
}
// ***************************************************************************
void CTessFace::averageTesselationVertices()
{
// If we are not splitted, no-op.
if(isLeaf())
return;
CTessFace *neighbor;
// Normal square case.
if(!isRectangular())
{
neighbor= FBase;
}
// Special Rectangular case.
else
{
// NB: here, just need to compute average of myself with FLeft, because my neighbor FBase
// is on same patch (see splitRectangular()), and is average with its FLeft neighbor is done
// on another branch of the recurs call.
neighbor= FLeft;
}
/* Try to average with neighbor.
- if no neighbor, no-op :).
- if neighbor is bind 1/N (CantMergeFace), no-op too, because the vertex is a BaseVertex, so don't modify.
- if my patch is same than my neighbor, then we are on a same patch :), and so no need to average.
*/
if(neighbor!=NULL && neighbor!=&CantMergeFace && Patch!= neighbor->Patch)
{
nlassert(neighbor->Patch);
nlassert(!neighbor->isLeaf());
// must compute average beetween me and my neighbor.
// NB: this work with both rectangular and square triangles (see split*()).
nlassert(SonLeft->VBase == neighbor->SonLeft->VBase);
CVector v0= Patch->computeVertex(SonLeft->PVBase.getS(), SonLeft->PVBase.getT());
CVector v1= neighbor->Patch->computeVertex(neighbor->SonLeft->PVBase.getS(), neighbor->SonLeft->PVBase.getT());
// And so set the average.
SonLeft->VBase->EndPos= (v0+v1)/2;
}
// Do same thing for sons. NB: see above, we are not a leaf.
SonLeft->averageTesselationVertices();
SonRight->averageTesselationVertices();
}
// ***************************************************************************
void CTessFace::refreshTesselationGeometry()
{
// must enlarge the little tessBlock (if any), for clipping.
Patch->extendTessBlockWithEndPos(this);
// If we are not splitted, no-op.
if(isLeaf())
return;
/* NB: rectangular case: just need to take SonLeft->VBase, because my neighbor on FBase will compute his son
on another branch of the recurs call.
*/
// re-compute this position (maybe with new noise geometry in Tile Edition).
SonLeft->VBase->EndPos= Patch->computeVertex(SonLeft->PVBase.getS(), SonLeft->PVBase.getT());
// overwrite cur Pos (NB: specialy hardcoded for Tile edition).
SonLeft->VBase->Pos= SonLeft->VBase->EndPos;
// Do same thing for sons. NB: see above, we are not a leaf.
SonLeft->refreshTesselationGeometry();
SonRight->refreshTesselationGeometry();
}
// ***************************************************************************
bool CTessFace::updateBindEdge(CTessFace *&edgeFace, bool &splitWanted)
{
// Return true, when the bind should be Ok, or if a split has occured.
// Return false only if pointers are updated, without splits.
if(edgeFace==NULL)
return true;
if(edgeFace->isLeaf())
return true;
/*
Look at the callers, and you'll see that "this" is always a leaf.
Therefore, edgeFace is a valid pointer (either if it is FLeft, FRight or FBase).
*/
// MultiPatch face case.
//======================
// If neighbor is a multiple face.
if(edgeFace->Patch==NULL && edgeFace->FBase==this)
{
splitWanted= true;
return true;
}
// neighbor is a "Square face" case.
//==================================
if(!edgeFace->isRectangular())
{
// NB: this code works either if I AM a rectangular face or a square face.
// If the neighbor is splitted on ourself, split...
if(edgeFace->FBase==this)
{
splitWanted= true;
return true;
}
else
{
// Just update pointers...
if(edgeFace->FLeft==this)
{
CTessFace *sonLeft= edgeFace->SonLeft;
sonLeft->FBase= this;
edgeFace= sonLeft;
}
else if(edgeFace->FRight==this)
{
CTessFace *sonRight= edgeFace->SonRight;
sonRight->FBase= this;
edgeFace= sonRight;
}
else
{
// Look at the callers, and you'll see that "this" is always a leaf.
// Therefore, we should never be here.
nlstop;
}
}
}
// neighbor is a "Rectangle face" case.
//=====================================
else
{
// NB: this code works either if I AM a rectangular face or a square face.
// If the neighbor is splitted on ourself, split...
// Test FLeft because of rectangular case... :)
// FBase should be tested too. If edgeFace->FBase==this, I should be myself a rectangular face.
if(edgeFace->FLeft==this || edgeFace->FBase==this)
{
splitWanted= true;
return true;
}
else
{
if(edgeFace->FRight==this)
{
// See rectangular tesselation rules, too know why we do this.
CTessFace *sonRight= edgeFace->SonRight;
sonRight->FRight= this;
edgeFace= sonRight;
}
else
{
// Look at the callers, and you'll see that "this" is always a leaf.
// Therefore, we should never be here.
nlstop;
}
}
}
return false;
}
// ***************************************************************************
void CTessFace::updateBindAndSplit()
{
bool splitWanted= false;
CTessFace *f0= NULL;
CTessFace *f1= NULL;
/*
Look at the callers, and you'll see that "this" is always a leaf.
Therefore, FBase, FLeft and FRight are good pointers, and *FLeft and *FRight should be Ok too.
*/
nlassert(isLeaf());
while(!updateBindEdge(FBase, splitWanted));
// FLeft and FRight pointers are only valid in Leaves nodes.
while(!updateBindEdge(FLeft, splitWanted));
while(!updateBindEdge(FRight, splitWanted));
// In rectangular case, we MUST also update edges of FBase.
// Because splitRectangular() split those two faces at the same time.
if(isRectangular())
{
f0= this;
f1= FBase;
nlassert(FBase);
nlassert(FBase->isLeaf());
// Doesn't need to update FBase->FBase, since it's me!
// FLeft and FRight pointers are only valid in Leaves nodes.
while(!FBase->updateBindEdge(FBase->FLeft, splitWanted));
while(!FBase->updateBindEdge(FBase->FRight, splitWanted));
}
CTessFace *fmult= NULL;
CTessFace *fmult0= NULL;
CTessFace *fmult1= NULL;
// If multipatch face case.
//=========================
if(!isRectangular())
{
// multipatch face case are detected when face->Patch==NULL !!!
if(FBase && FBase->Patch==NULL)
{
fmult= FBase;
// First, trick: FBase is NULL, so during the split. => no ptr problem.
FBase= NULL;
}
}
else
{
// multipatch face case are detected when face->Patch==NULL !!!
if(f0->FLeft && f0->FLeft->Patch==NULL)
{
fmult0= f0->FLeft;
// First, trick: neighbor is NULL, so during the split. => no ptr problem.
f0->FLeft= NULL;
}
// multipatch face case are detected when face->Patch==NULL !!!
if(f1->FLeft && f1->FLeft->Patch==NULL)
{
fmult1= f1->FLeft;
// First, trick: neighbor is NULL, so during the split. => no ptr problem.
f1->FLeft= NULL;
}
}
// Then split, and propagate.
//===========================
split(false);
if(!isRectangular())
{
if(FBase)
{
while(FBase->isLeaf())
FBase->updateBindAndSplit();
}
// There is a possible bug here (maybe easily patched). Sons may have be propagated splitted.
// And problems may arise because this face hasn't yet good connectivity.
nlassert(SonLeft->isLeaf() && SonRight->isLeaf());
}
else
{
if(f0->FLeft)
{
while(f0->FLeft->isLeaf())
f0->FLeft->updateBindAndSplit();
}
if(f1->FLeft)
{
while(f1->FLeft->isLeaf())
f1->FLeft->updateBindAndSplit();
}
// There is a possible bug here (maybe easily patched). Sons may have be propagated splitted.
// And problems may arise because this face hasn't yet good connectivity.
nlassert(f0->SonLeft->isLeaf() && f0->SonRight->isLeaf());
nlassert(f1->SonLeft->isLeaf() && f1->SonRight->isLeaf());
}
// If multipatch face case, update neighbors.
//===========================================
if(!isRectangular() && fmult)
{
// Update good Face neighbors.
//============================
SonLeft->FRight= fmult->SonRight;
fmult->SonRight->changeNeighbor(&CTessFace::MultipleBindFace, SonLeft);
SonRight->FLeft= fmult->SonLeft;
fmult->SonLeft->changeNeighbor(&CTessFace::MultipleBindFace, SonRight);
// NB: this work auto with 1/2 or 1/4. See CPatch::bind(), to understand.
// In 1/4 case, fmult->SonLeft and fmult->SonRight are themselves MultiPatch face. So it will recurse.
// Update good vertex pointer.
//============================
CTessVertex *vert= fmult->VBase;
// Copy the good coordinate: those splitted (because of noise).
vert->Pos= vert->StartPos= vert->EndPos= SonLeft->VBase->EndPos;
// But delete the pointer.
Patch->getLandscape()->deleteTessVertex(SonLeft->VBase);
// And update sons pointers, to good vertex.
SonLeft->VBase= vert;
SonRight->VBase= vert;
// Compute correct centers.
SonRight->computeSplitPoint();
SonLeft->computeSplitPoint();
// Update good Far vertex pointer.
//================================
// Because *->VBase may have been merged to the multiple bind face, Near/FarVertices which pointed on it must
// be setup.
// We do not have to propagate this vertex ptr change since sons are leaves!!
nlassert(SonLeft->isLeaf() && SonRight->isLeaf());
// update pointers on vertex.
SonLeft->updateNearFarVertices();
SonRight->updateNearFarVertices();
// Bind FBase to a false face which indicate a bind 1/N.
// This face prevent for "this" face to be merged...
FBase= &CantMergeFace;
// Therefore, the vertex will be never deleted (since face not merged).
// The only way to do this, is to unbind the patch from all (then the vertex is cloned), then the merge will be Ok.
}
// Else if rectangular.
else if(fmult0 || fmult1)
{
CTessFace *f;
sint i;
// Same reasoning for rectangular patchs, as above.
for(i=0;i<2;i++)
{
if(i==0)
f= f0, fmult= fmult0;
else
f= f1, fmult= fmult1;
if(fmult)
{
// Update good Face neighbors (when I am a rectangle).
//============================
// Consider the fmult face as a square face.
CTessFace *toLeft, *toRight;
CTessFace *fl=f->SonLeft, *fr=f->SonRight;
toLeft= fmult->SonLeft;
toRight= fmult->SonRight;
// Cross connection of sons.
fl->FLeft= toLeft;
fr->FLeft= toRight;
toLeft->changeNeighbor(&CTessFace::MultipleBindFace, fl);
toRight->changeNeighbor(&CTessFace::MultipleBindFace, fr);
// Update good vertex pointer.
//============================
CTessVertex *vert= fmult->VBase;
// Copy the good coordinate: those splitted (because of noise).
// NB: this work too with rectangular patch (see tesselation rules).
vert->Pos= vert->StartPos= vert->EndPos= fl->VBase->EndPos;
// But delete the pointer.
Patch->getLandscape()->deleteTessVertex(fl->VBase);
// And update sons pointers, to good vertex (rectangular case, see tesselation rules).
fl->VBase= vert;
fr->VLeft= vert;
f->FBase->SonLeft->VRight= vert;
// Point to a bind 1/N indicator.
f->FLeft= &CantMergeFace;
}
}
// After all updates done. recompute centers of both sons 's faces, and update far vertices pointers.
for(i=0;i<2;i++)
{
if(i==0)
f= f0;
else
f= f1;
// Compute correct centers.
f->SonRight->computeSplitPoint();
f->SonLeft->computeSplitPoint();
// Update good Far vertex pointer.
//================================
// Because *->VBase may have been merged to the multiple bind face, Near/FarVertices which pointed on it must
// be setup.
// We do not have to propagate this vertex ptr change, since sons are leaves!!
nlassert(f->SonLeft->isLeaf() && f->SonRight->isLeaf());
// update pointers on vertex.
f->SonLeft->updateNearFarVertices();
f->SonRight->updateNearFarVertices();
}
}
}
// ***************************************************************************
void CTessFace::updateBind()
{
/*
Remind that updateBind() is called ONLY on the patch which is binded (not the neighbors).
Since updateBind() is called on the bintree, and that precedent propagated split may have occur, we may not
be a leaf here. So we are not sure that FLeft and FRight are good, and we doesn't need to update them (since we have
sons).
Also, since we are splitted, and correctly linked (this may not be the case in updateBindAndSplit()), FBase IS
correct. His FBase neighbor and him form a diamond. So we don't need to update him.
Same remarks for rectangular patchs.
*/
if(isLeaf())
{
bool splitWanted= false;
while(!updateBindEdge(FBase, splitWanted));
// FLeft and FRight pointers are only valid in Leaves nodes.
while(!updateBindEdge(FLeft, splitWanted));
while(!updateBindEdge(FRight, splitWanted));
if(splitWanted)
updateBindAndSplit();
}
// Recurse to sons.
if(!isLeaf())
{
// Update bind of sons.
SonLeft->updateBind();
SonRight->updateBind();
}
}
// ***************************************************************************
static inline bool matchEdge(const CVector2f &uv0, const CVector2f &uv1, const CVector2f &uva, const CVector2f &uvb)
{
if(uv0==uva && uv1==uvb)
return true;
if(uv0==uvb && uv1==uva)
return true;
return false;
}
// ***************************************************************************
CTessFace *CTessFace::linkTessFaceWithEdge(const CVector2f &uv0, const CVector2f &uv1, CTessFace *linkTo)
{
// Compute 0,1 coords of 3 patch coords.
CVector2f vb( PVBase.getS(), PVBase.getT() );
CVector2f vl( PVLeft.getS(), PVLeft.getT() );
CVector2f vr( PVRight.getS(), PVRight.getT() );
// Search if one of the 3 edges of this triangle match the wanted edge.
// Base Edge
if(matchEdge(uv0, uv1, vl, vr))
{
// If leaf, check if unbound (else ptr is invalid)
nlassert(FBase==NULL || !isLeaf());
FBase= linkTo;
return this;
}
// Left Edge
if(matchEdge(uv0, uv1, vb, vl))
{
// If leaf, check if unbound (else ptr is invalid)
nlassert(FLeft==NULL || !isLeaf());
FLeft= linkTo;
return this;
}
// Right Edge
if(matchEdge(uv0, uv1, vb, vr))
{
// If leaf, check if unbound (else ptr is invalid)
nlassert(FRight==NULL || !isLeaf());
FRight= linkTo;
return this;
}
// If not found here, recurs to children
CTessFace *ret= NULL;
if( !isLeaf() )
{
ret= SonLeft->linkTessFaceWithEdge(uv0, uv1, linkTo);
// if no found in this branch, recusr right branch.
if(!ret)
ret= SonRight->linkTessFaceWithEdge(uv0, uv1, linkTo);
}
// return the result from subBranchs
return ret;
}
// ***************************************************************************
bool CTessFace::isRectangular() const
{
return LevelSquareLimitLevel;
}
// ***************************************************************************
// ***************************************************************************
// For changePatchTexture.
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTessFace::deleteTileUvs()
{
// NB: NearVertices are removed from renderlist with deleteTileUv (called in releaseTileMaterial()).
if(!isLeaf())
{
// Must delete the materials of leaves first.
SonLeft->deleteTileUvs();
SonRight->deleteTileUvs();
if(SonLeft->Level== Patch->TileLimitLevel)
{
// Square patch assumption: the sons are not of the same TileId/Patch.
nlassert(!sameTile(SonLeft, SonRight));
// release tiles.
SonLeft->releaseTileMaterial();
SonRight->releaseTileMaterial();
}
else if(SonLeft->Level > Patch->TileLimitLevel)
{
nlassert(!FBase || !FBase->isLeaf());
// Delete Uv, only if not already done by the neighbor (ie neighbor has yet TileFaces!!).
// But Always delete if neighbor exist and has not same tile as me.
// NB: this work with rectangular neigbor patch, since sameTile() will return false if different patch.
if(!FBase || !FBase->SonLeft->emptyTileFaces() || !sameTile(this, FBase))
{
SonLeft->deleteTileUv(IdUvBase);
}
// In all case, must delete the tilefaces of those face.
SonLeft->deleteTileFaces();
SonRight->deleteTileFaces();
// For createTileUvs, it is important to mark those faces as NO TileMaterial.
SonLeft->TileMaterial= NULL;
SonRight->TileMaterial= NULL;
}
}
else
{
// NB: this is done always BELOW tile creation (see above).
// Do this only for tiles.
if(TileMaterial)
Patch->removeFaceFromTileRenderList(this);
}
}
// ***************************************************************************
void CTessFace::recreateTileUvs()
{
// NB: NearVertices are append to renderlist with allocTileUv (called in computeTileMaterial()/heritTileMaterial()).
if(!isLeaf())
{
// Must recreate the materials of parent first.
// There is no problem with rectangular patch, since tiles are always squares.
// If new tile ....
if(SonLeft->Level==Patch->TileLimitLevel)
{
SonLeft->computeTileMaterial();
SonRight->computeTileMaterial();
}
// else Tile herit.
else if(SonLeft->Level > Patch->TileLimitLevel)
{
heritTileMaterial();
}
SonLeft->recreateTileUvs();
SonRight->recreateTileUvs();
}
else
{
// NB: this is done always AFTER tile creation (see above).
// Do this only for tiles.
if(TileMaterial)
Patch->appendFaceToTileRenderList(this);
}
}
// ***************************************************************************
void CTessFace::heritTileMaterial()
{
SonLeft->TileMaterial= TileMaterial;
SonLeft->TileId= TileId;
SonLeft->buildTileFaces();
SonLeft->copyTileUv(IdUvLeft, this, IdUvBase);
SonLeft->copyTileUv(IdUvRight, this, IdUvLeft);
SonRight->TileMaterial= TileMaterial;
SonRight->TileId= TileId;
SonRight->buildTileFaces();
SonRight->copyTileUv(IdUvLeft, this, IdUvRight);
SonRight->copyTileUv(IdUvRight, this, IdUvBase);
// Create, or link to the tileUv.
// Try to link to a neighbor TileUv.
// Can only work iff exist, and iff FBase is same patch, and same TileId.
if(FBase!=NULL && !FBase->isLeaf() && FBase->SonLeft->TileMaterial!=NULL && sameTile(this, FBase) )
{
// Ok!! link to the (existing) TileUv.
// FBase->SonLeft!=NULL since FBase->isLeaf()==false.
SonLeft->copyTileUv(IdUvBase, FBase->SonLeft, IdUvBase);
SonRight->copyTileUv(IdUvBase, FBase->SonLeft, IdUvBase);
}
else
{
// Allocate a new vertex, and copy it to SonLeft and SonRight.
SonLeft->allocTileUv(IdUvBase);
SonRight->copyTileUv(IdUvBase, SonLeft, IdUvBase);
// Fill the new near vertex, with middle of Left/Right father.
SonLeft->heritTileUv(this);
// UVs are computed, may create and fill VB.
SonLeft->checkCreateFillTileVB(IdUvBase);
}
}
// ***************************************************************************
// ***************************************************************************
// For getTesselatedPos
// ***************************************************************************
// ***************************************************************************
// ***************************************************************************
void CTessFace::getTesselatedPos(const CUV &uv, bool verifInclusion, CVector &ret)
{
CVector uvPos(uv.U, uv.V, 0);
// may verif if uv is In this triangle. supposed true if rectangular branch.
if(verifInclusion && !(isRectangular() && !isLeaf()) )
{
CVector uvs[3];
uvs[0].set( PVBase.getS(), PVBase.getT(), 0);
uvs[1].set( PVLeft.getS(), PVLeft.getT(), 0);
uvs[2].set( PVRight.getS(), PVRight.getT(), 0);
for(sint i=0; i<3; i++)
{
CVector dUv= uvs[(i+1)%3] - uvs[i];
CVector normalUv(dUv.y, -dUv.x, 0);
// if out this 2D plane, uv is out this triangle
if(normalUv * (uvPos-uvs[i]) <0)
return;
}
}
// compute tesselated pos in this face.
if(isLeaf())
// ok, no more sons, let's do it.
computeTesselatedPos(uv, ret);
else
{
// must subdivide.
// if we are rectangular (strange tesselation), must test in both leaves, else, choose only one.
if(isRectangular())
{
SonLeft->getTesselatedPos(uv, true, ret);
SonRight->getTesselatedPos(uv, true, ret);
}
else
{
// Compute the uv plane which separate the 2 leaves.
CVector uvBase, uvMiddle;
uvBase.set ( PVBase.getS(), PVBase.getT(), 0);
uvMiddle.set( SonLeft->PVBase.getS(), SonLeft->PVBase.getT(), 0);
CVector dUv= uvMiddle - uvBase;
CVector normalUv(dUv.y, -dUv.x, 0);
// choose what leaf to recurs.
if(normalUv * (uvPos - uvBase) <0)
SonLeft->getTesselatedPos(uv, false, ret);
else
SonRight->getTesselatedPos(uv, false, ret);
}
}
}
// ***************************************************************************
void CTessFace::computeTesselatedPos(const CUV &uv, CVector &ret)
{
CVector uvPos(uv.U, uv.V, 0);
// compute the UV triangle of this face.
CTriangle uvTri;
uvTri.V0.set( PVBase.getS(), PVBase.getT(), 0);
uvTri.V1.set( PVLeft.getS(), PVLeft.getT(), 0);
uvTri.V2.set( PVRight.getS(), PVRight.getT(), 0);
// must interpolate the position with given UV, so compute XYZ gradients.
CVector Gx;
CVector Gy;
CVector Gz;
// If VertexProgram activated
if( CLandscapeGlobals::VertexProgramEnabled )
{
// then Must update geomorphed position because not done !!
VBase->computeGeomPos();
VLeft->computeGeomPos();
VRight->computeGeomPos();
}
// NB: take geomorphed position.
uvTri.computeGradient(VBase->Pos.x, VLeft->Pos.x, VRight->Pos.x, Gx);
uvTri.computeGradient(VBase->Pos.y, VLeft->Pos.y, VRight->Pos.y, Gy);
uvTri.computeGradient(VBase->Pos.z, VLeft->Pos.z, VRight->Pos.z, Gz);
// Compute interpolated position.
ret= VBase->Pos;
uvPos-= uvTri.V0;
ret.x+= Gx*uvPos;
ret.y+= Gy*uvPos;
ret.z+= Gz*uvPos;
}
// ***************************************************************************
void CTessFace::appendTessellationLeaves(std::vector &leaves) const
{
if(isLeaf())
leaves.push_back(this);
else
{
SonLeft->appendTessellationLeaves(leaves);
SonRight->appendTessellationLeaves(leaves);
}
}
} // NL3D