// 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/landscape_collision_grid.h" #include "nel/misc/fast_floor.h" #include using namespace std; using namespace NLMISC; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** CLandscapeCollisionGrid::CLandscapeCollisionGrid(CVisualCollisionManager *owner) { _Owner= owner; // reset list to NULL. memset(_Grid, 0, NL_COLGRID_SIZE*NL_COLGRID_SIZE * sizeof(CVisualTileDescNode*)); _Cleared= true; // sizepower. nlassert(isPowerOf2(NL_COLGRID_SIZE)); _SizePower= getPowerOf2(NL_COLGRID_SIZE); } // *************************************************************************** CLandscapeCollisionGrid::~CLandscapeCollisionGrid() { clear(); } // *************************************************************************** void CLandscapeCollisionGrid::clear() { // already cleared? do nothing. if(_Cleared) return; // Parse all quads. sint i; for(i=0;iNext; _Owner->deleteVisualTileDescNode(ptr); ptr= next; } // reset root. _Grid[i]= NULL; } _Cleared= true; } // *************************************************************************** struct CVector2i { sint x,y; }; // *************************************************************************** void CLandscapeCollisionGrid::build(const std::vector &quadBlocks, const CVector &delta) { sint x,y; static CVector2i floorVals[NL_PATCH_BLOCK_MAX_VERTEX*NL_PATCH_BLOCK_MAX_VERTEX]; // first clear clear(); // init for fast floor. OptFastFloorBegin(); // then fill. _Cleared= false; _Delta= delta; // parse all quad blocks. for(sint i=0; i<(sint)quadBlocks.size();i++) { CPatchQuadBlock &qb= *quadBlocks[i]; sint lenS= qb.PatchBlockId.S1 - qb.PatchBlockId.S0; sint lenT= qb.PatchBlockId.T1 - qb.PatchBlockId.T0; // First, floor all vertices of interest. for(y=0; y=0 && x1>=x0); nlassert(y0>=0 && y1>=y0); // first, transform coordinate (in meters) in quadgrid eltSize (ie 2 meters). x0= (x0>>1); // floor(). x1= (x1>>1) + 1; // equivalent of ceil(). y0= (y0>>1); // floor(). y1= (y1>>1) + 1; // equivalent of ceil(). // setup bounds in quadgrid coordinate. if(x1-x0>=NL_COLGRID_SIZE) x0=0, x1= NL_COLGRID_SIZE; else { x0&= NL_COLGRID_SIZE-1; x1&= NL_COLGRID_SIZE-1; if(x1<=x0) x1+=NL_COLGRID_SIZE; } if(y1-y0>=NL_COLGRID_SIZE) y0=0, y1= NL_COLGRID_SIZE; else { y0&= NL_COLGRID_SIZE-1; y1&= NL_COLGRID_SIZE-1; if(y1<=y0) y1+=NL_COLGRID_SIZE; } // fill all cases with element. sint x,y; for(y= y0;ynewVisualTileDescNode(); // fill elt. elt->PatchQuadBlocId= paBlockId; elt->QuadId= quadId; // bind elt to the list. elt->Next= _Grid[gridId]; _Grid[gridId]= elt; } } } // *************************************************************************** CVisualTileDescNode *CLandscapeCollisionGrid::select(const NLMISC::CVector &pos) { // compute pos in the quadgrid. CVector localPos; localPos= pos + _Delta; // cases are 2x2 meters. localPos/=2; // floor, bound in quadgrid coordinate. sint x,y; x= (sint)floor(localPos.x); y= (sint)floor(localPos.y); x&= NL_COLGRID_SIZE-1; y&= NL_COLGRID_SIZE-1; return _Grid[y*NL_COLGRID_SIZE+x]; } } // NL3D