2012-01-21 16:59:08 +00:00
|
|
|
#include <OgreTerrain.h>
|
|
|
|
#include <OgreTerrainGroup.h>
|
2012-01-24 13:29:31 +00:00
|
|
|
#include <OgreTerrainMaterialGeneratorA.h>
|
2012-01-21 16:59:08 +00:00
|
|
|
|
|
|
|
#include "terrain.hpp"
|
|
|
|
|
|
|
|
#include "components/esm/loadland.hpp"
|
|
|
|
|
|
|
|
namespace MWRender
|
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
|
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
TerrainManager::TerrainManager(Ogre::SceneManager* mgr)
|
|
|
|
{
|
|
|
|
mTerrainGlobals = OGRE_NEW Ogre::TerrainGlobalOptions();
|
|
|
|
|
|
|
|
mTerrainGlobals->setMaxPixelError(8);
|
2012-02-07 12:41:08 +00:00
|
|
|
mTerrainGlobals->setLayerBlendMapSize(SPLIT_TERRAIN ? 256 : 1024);
|
|
|
|
mTerrainGlobals->setLightMapSize(SPLIT_TERRAIN ? 256 : 1024);
|
|
|
|
mTerrainGlobals->setCompositeMapSize(SPLIT_TERRAIN ? 256 : 1024);
|
|
|
|
mTerrainGlobals->setDefaultGlobalColourMapSize(256);
|
|
|
|
|
|
|
|
//10 (default) didn't seem to be quite enough
|
|
|
|
mTerrainGlobals->setSkirtSize(32);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Here we are pushing the composite map distance beyond the edge
|
|
|
|
* of the rendered terrain due to light issues (the lighting differs
|
|
|
|
* so much that it is very noticable
|
|
|
|
*/
|
|
|
|
mTerrainGlobals->setCompositeMapDistance(ESM::Land::REAL_SIZE*4);
|
2012-01-24 13:29:31 +00:00
|
|
|
|
|
|
|
Ogre::TerrainMaterialGenerator::Profile* const activeProfile =
|
|
|
|
mTerrainGlobals->getDefaultMaterialGenerator()
|
|
|
|
->getActiveProfile();
|
|
|
|
Ogre::TerrainMaterialGeneratorA::SM2Profile* matProfile =
|
|
|
|
static_cast<Ogre::TerrainMaterialGeneratorA::SM2Profile*>(activeProfile);
|
|
|
|
|
|
|
|
matProfile->setLightmapEnabled(false);
|
|
|
|
matProfile->setReceiveDynamicShadowsEnabled(false);
|
|
|
|
|
2012-02-07 12:41:08 +00:00
|
|
|
//scale the land size if required
|
2012-01-24 13:29:31 +00:00
|
|
|
mLandSize = ESM::Land::LAND_SIZE;
|
|
|
|
mRealSize = ESM::Land::REAL_SIZE;
|
|
|
|
if ( SPLIT_TERRAIN )
|
|
|
|
{
|
|
|
|
mLandSize = (mLandSize - 1)/2 + 1;
|
|
|
|
mRealSize /= 2;
|
|
|
|
}
|
2012-01-21 16:59:08 +00:00
|
|
|
|
|
|
|
mTerrainGroup = OGRE_NEW Ogre::TerrainGroup(mgr,
|
2012-01-24 13:29:31 +00:00
|
|
|
Ogre::Terrain::ALIGN_X_Z,
|
|
|
|
mLandSize,
|
|
|
|
mRealSize);
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
mTerrainGroup->setOrigin(Ogre::Vector3(mRealSize/2,
|
|
|
|
0,
|
|
|
|
-mRealSize/2));
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
Ogre::Terrain::ImportData& importSettings =
|
2012-01-21 16:59:08 +00:00
|
|
|
mTerrainGroup->getDefaultImportSettings();
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
importSettings.inputBias = 0;
|
|
|
|
importSettings.terrainSize = mLandSize;
|
|
|
|
importSettings.worldSize = mRealSize;
|
2012-01-21 16:59:08 +00:00
|
|
|
importSettings.minBatchSize = 9;
|
2012-01-24 13:29:31 +00:00
|
|
|
importSettings.maxBatchSize = mLandSize;
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
importSettings.deleteInputData = true;
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
TerrainManager::~TerrainManager()
|
|
|
|
{
|
|
|
|
OGRE_DELETE mTerrainGroup;
|
|
|
|
OGRE_DELETE mTerrainGlobals;
|
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
void TerrainManager::cellAdded(MWWorld::Ptr::CellStore *store)
|
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
const int cellX = store->cell->getGridX();
|
|
|
|
const int cellY = store->cell->getGridY();
|
|
|
|
|
|
|
|
Ogre::Terrain::ImportData terrainData =
|
|
|
|
mTerrainGroup->getDefaultImportSettings();
|
|
|
|
|
|
|
|
if ( SPLIT_TERRAIN )
|
|
|
|
{
|
|
|
|
//split the cell terrain into four segments
|
|
|
|
const int numTextures = ESM::Land::LAND_TEXTURE_SIZE/2;
|
|
|
|
|
|
|
|
for ( int x = 0; x < 2; x++ )
|
|
|
|
{
|
|
|
|
for ( int y = 0; y < 2; y++ )
|
|
|
|
{
|
|
|
|
const int terrainX = cellX * 2 + x;
|
|
|
|
const int terrainY = cellY * 2 + y;
|
|
|
|
|
|
|
|
terrainData.inputFloat = OGRE_ALLOC_T(float,
|
|
|
|
mLandSize*mLandSize,
|
|
|
|
Ogre::MEMCATEGORY_GEOMETRY);
|
|
|
|
|
|
|
|
//copy the height data row by row
|
|
|
|
for ( int terrainCopyY = 0; terrainCopyY < mLandSize; terrainCopyY++ )
|
|
|
|
{
|
|
|
|
//the offset of the current segment
|
|
|
|
const size_t yOffset = y * (mLandSize-1) * ESM::Land::LAND_SIZE +
|
|
|
|
//offset of the row
|
|
|
|
terrainCopyY * ESM::Land::LAND_SIZE;
|
|
|
|
const size_t xOffset = x * (mLandSize-1);
|
|
|
|
|
|
|
|
memcpy(&terrainData.inputFloat[terrainCopyY*mLandSize],
|
|
|
|
&store->land[1][1]->landData->heights[yOffset + xOffset],
|
|
|
|
mLandSize*sizeof(float));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::map<uint16_t, int> indexes;
|
|
|
|
initTerrainTextures(&terrainData, store,
|
|
|
|
x * numTextures, y * numTextures,
|
|
|
|
numTextures, indexes);
|
|
|
|
|
2012-02-07 12:41:08 +00:00
|
|
|
assert( mTerrainGroup->getTerrain(cellX, cellY) == NULL &&
|
|
|
|
"The terrain for this cell already existed" );
|
2012-01-24 13:29:31 +00:00
|
|
|
mTerrainGroup->defineTerrain(terrainX, terrainY, &terrainData);
|
|
|
|
|
|
|
|
mTerrainGroup->loadTerrain(terrainX, terrainY, true);
|
|
|
|
Ogre::Terrain* terrain = mTerrainGroup->getTerrain(terrainX, terrainY);
|
|
|
|
initTerrainBlendMaps(terrain, store,
|
|
|
|
x * numTextures, y * numTextures,
|
|
|
|
numTextures, indexes);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//one cell is one terrain segment
|
|
|
|
terrainData.inputFloat = OGRE_ALLOC_T(float,
|
|
|
|
mLandSize*mLandSize,
|
|
|
|
Ogre::MEMCATEGORY_GEOMETRY);
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
memcpy(&terrainData.inputFloat[0],
|
|
|
|
&store->land[1][1]->landData->heights[0],
|
|
|
|
mLandSize*mLandSize*sizeof(float));
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
std::map<uint16_t, int> indexes;
|
|
|
|
initTerrainTextures(&terrainData, store, 0, 0,
|
|
|
|
ESM::Land::LAND_TEXTURE_SIZE, indexes);
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
mTerrainGroup->defineTerrain(cellX, cellY, &terrainData);
|
|
|
|
|
|
|
|
mTerrainGroup->loadTerrain(cellX, cellY, true);
|
|
|
|
Ogre::Terrain* terrain = mTerrainGroup->getTerrain(cellX, cellY);
|
|
|
|
initTerrainBlendMaps(terrain, store, 0, 0,
|
|
|
|
ESM::Land::LAND_TEXTURE_SIZE, indexes);
|
|
|
|
}
|
2012-01-21 16:59:08 +00:00
|
|
|
|
2012-02-07 12:41:08 +00:00
|
|
|
mTerrainGroup->freeTemporaryResources();
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
void TerrainManager::cellRemoved(MWWorld::Ptr::CellStore *store)
|
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
if ( SPLIT_TERRAIN )
|
|
|
|
{
|
|
|
|
for ( int x = 0; x < 2; x++ )
|
|
|
|
{
|
|
|
|
for ( int y = 0; y < 2; y++ )
|
|
|
|
{
|
2012-02-07 12:41:08 +00:00
|
|
|
mTerrainGroup->unloadTerrain(store->cell->getGridX() * 2 + x,
|
2012-01-24 13:29:31 +00:00
|
|
|
store->cell->getGridY() * 2 + y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-02-07 12:41:08 +00:00
|
|
|
mTerrainGroup->unloadTerrain(store->cell->getGridX(),
|
2012-01-24 13:29:31 +00:00
|
|
|
store->cell->getGridY());
|
|
|
|
}
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 17:59:12 +00:00
|
|
|
void TerrainManager::initTerrainTextures(Ogre::Terrain::ImportData* terrainData,
|
|
|
|
MWWorld::Ptr::CellStore* store,
|
2012-01-23 17:19:54 +00:00
|
|
|
int fromX, int fromY, int size,
|
2012-01-21 17:59:12 +00:00
|
|
|
std::map<uint16_t, int>& indexes)
|
|
|
|
{
|
2012-01-23 17:19:54 +00:00
|
|
|
assert(store != NULL && "store must be a valid pointer");
|
|
|
|
assert(terrainData != NULL && "Must have valid terrain data");
|
|
|
|
assert(fromX >= 0 && fromY >= 0 &&
|
|
|
|
"Can't get a terrain texture on terrain outside the current cell");
|
|
|
|
assert(fromX+size <= ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
fromY+size <= ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
"Can't get a terrain texture on terrain outside the current cell");
|
|
|
|
|
2012-01-21 17:59:12 +00:00
|
|
|
//have a base texture for now, but this is probably not needed on most cells
|
|
|
|
terrainData->layerList.resize(1);
|
|
|
|
terrainData->layerList[0].worldSize = 256;
|
|
|
|
terrainData->layerList[0].textureNames.push_back("textures\\_land_default.dds");
|
|
|
|
terrainData->layerList[0].textureNames.push_back("textures\\_land_default.dds");
|
|
|
|
|
2012-01-23 17:19:54 +00:00
|
|
|
for ( int y = fromY - 1; y < fromY + size + 1; y++ )
|
2012-01-21 17:59:12 +00:00
|
|
|
{
|
2012-01-23 17:19:54 +00:00
|
|
|
for ( int x = fromX - 1; x < fromX + size + 1; x++ )
|
2012-01-21 17:59:12 +00:00
|
|
|
{
|
2012-01-23 17:19:54 +00:00
|
|
|
const uint16_t ltexIndex = getLtexIndexAt(store, x, y);
|
|
|
|
//this is the base texture, so we can ignore this at present
|
2012-01-21 17:59:12 +00:00
|
|
|
if ( ltexIndex == 0 )
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::map<uint16_t, int>::const_iterator it = indexes.find(ltexIndex);
|
|
|
|
|
|
|
|
if ( it == indexes.end() )
|
|
|
|
{
|
|
|
|
//NB: All vtex ids are +1 compared to the ltex ids
|
2012-01-23 17:19:54 +00:00
|
|
|
assert((int)ltexIndex >= 0 &&
|
2012-01-21 17:59:12 +00:00
|
|
|
store->landTextures->ltex.size() > (size_t)ltexIndex - 1 &&
|
|
|
|
"LAND.VTEX must be within the bounds of the LTEX array");
|
|
|
|
|
|
|
|
std::string texture = store->landTextures->ltex[ltexIndex-1].texture;
|
|
|
|
//TODO this is needed due to MWs messed up texture handling
|
|
|
|
texture = texture.substr(0, texture.rfind(".")) + ".dds";
|
|
|
|
|
|
|
|
const size_t position = terrainData->layerList.size();
|
|
|
|
terrainData->layerList.push_back(Ogre::Terrain::LayerInstance());
|
|
|
|
|
|
|
|
terrainData->layerList[position].worldSize = 256;
|
|
|
|
terrainData->layerList[position].textureNames.push_back("textures\\" + texture);
|
|
|
|
|
|
|
|
//Normal map. This should be removed but it would require alterations to
|
|
|
|
//the material generator. Another option would be to use a 1x1 blank texture
|
|
|
|
terrainData->layerList[position].textureNames.push_back("textures\\" + texture);
|
|
|
|
|
|
|
|
indexes[ltexIndex] = position;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
|
|
|
|
2012-01-21 17:59:12 +00:00
|
|
|
void TerrainManager::initTerrainBlendMaps(Ogre::Terrain* terrain,
|
|
|
|
MWWorld::Ptr::CellStore* store,
|
2012-01-23 17:19:54 +00:00
|
|
|
int fromX, int fromY, int size,
|
2012-01-21 17:59:12 +00:00
|
|
|
const std::map<uint16_t, int>& indexes)
|
|
|
|
{
|
2012-01-23 17:19:54 +00:00
|
|
|
assert(store != NULL && "store must be a valid pointer");
|
|
|
|
assert(terrain != NULL && "Must have valid terrain");
|
|
|
|
assert(fromX >= 0 && fromY >= 0 &&
|
|
|
|
"Can't get a terrain texture on terrain outside the current cell");
|
|
|
|
assert(fromX+size <= ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
fromY+size <= ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
"Can't get a terrain texture on terrain outside the current cell");
|
|
|
|
|
|
|
|
//size must be a power of 2 as we do divisions with a power of 2 number
|
|
|
|
//that need to result in an integer for correct splatting
|
|
|
|
assert( (size & (size - 1)) == 0 && "Size must be a power of 2");
|
|
|
|
|
2012-01-21 17:59:12 +00:00
|
|
|
const int blendSize = terrain->getLayerBlendMapSize();
|
2012-01-23 17:19:54 +00:00
|
|
|
const int blendDist = TERRAIN_SHADE_DISTANCE * (blendSize / size);
|
2012-01-21 17:59:12 +00:00
|
|
|
|
|
|
|
//zero out every map
|
|
|
|
std::map<uint16_t, int>::const_iterator iter;
|
|
|
|
for ( iter = indexes.begin(); iter != indexes.end(); ++iter )
|
|
|
|
{
|
|
|
|
float* pBlend = terrain->getLayerBlendMap(iter->second)
|
|
|
|
->getBlendPointer();
|
|
|
|
memset(pBlend, 0, sizeof(float) * blendSize * blendSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
//covert the ltex data into a set of blend maps
|
2012-01-23 17:19:54 +00:00
|
|
|
for ( int texY = fromY - 1; texY < fromY + size + 1; texY++ )
|
2012-01-21 17:59:12 +00:00
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
for ( int texX = fromX - 1; texX < fromX + size + 1; texX++ )
|
2012-01-21 17:59:12 +00:00
|
|
|
{
|
2012-01-23 17:19:54 +00:00
|
|
|
const uint16_t ltexIndex = getLtexIndexAt(store, texX, texY);
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//whilte texX is the splat index relative to the entire cell,
|
|
|
|
//relX is relative to the current segment we are splatting
|
|
|
|
const int relX = texX - fromX;
|
|
|
|
const int relY = texY - fromY;
|
|
|
|
|
2012-01-23 17:19:54 +00:00
|
|
|
//this is the ground texture, which is currently the base texture
|
|
|
|
//so don't alter the splatting map
|
2012-01-21 17:59:12 +00:00
|
|
|
if ( ltexIndex == 0 ){
|
|
|
|
continue;
|
|
|
|
}
|
2012-01-23 17:19:54 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
assert (indexes.find(ltexIndex) != indexes.end() &&
|
|
|
|
"Texture layer must exist");
|
|
|
|
|
2012-01-21 17:59:12 +00:00
|
|
|
const int layerIndex = indexes.find(ltexIndex)->second;
|
|
|
|
|
|
|
|
float* const pBlend = terrain->getLayerBlendMap(layerIndex)
|
|
|
|
->getBlendPointer();
|
|
|
|
|
2012-01-23 17:19:54 +00:00
|
|
|
const int splatSize = blendSize / size;
|
2012-01-21 17:59:12 +00:00
|
|
|
|
|
|
|
//setup the bounds for the shading of this texture splat
|
2012-01-24 13:29:31 +00:00
|
|
|
const int startX = std::max(0, relX*splatSize - blendDist);
|
|
|
|
const int endX = std::min(blendSize, (relX+1)*splatSize + blendDist);
|
2012-01-21 17:59:12 +00:00
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
const int startY = std::max(0, relY*splatSize - blendDist);
|
|
|
|
const int endY = std::min(blendSize, (relY+1)*splatSize + blendDist);
|
2012-01-21 17:59:12 +00:00
|
|
|
|
|
|
|
for ( int blendX = startX; blendX < endX; blendX++ )
|
|
|
|
{
|
|
|
|
for ( int blendY = startY; blendY < endY; blendY++ )
|
|
|
|
{
|
|
|
|
assert(blendX >= 0 && blendX < blendSize &&
|
|
|
|
"index must be within texture bounds");
|
|
|
|
|
|
|
|
assert(blendY >= 0 && blendY < blendSize &&
|
|
|
|
"index must be within texture bounds");
|
|
|
|
|
|
|
|
//calculate the distance from the edge of the square
|
|
|
|
// to the point we are shading
|
2012-01-24 13:29:31 +00:00
|
|
|
int distX = relX*splatSize - blendX;
|
2012-01-21 17:59:12 +00:00
|
|
|
if ( distX < 0 )
|
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
distX = std::max(0, blendX - (relX+1)*splatSize);
|
2012-01-21 17:59:12 +00:00
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
int distY = relY*splatSize - blendY;
|
2012-01-21 17:59:12 +00:00
|
|
|
if ( distY < 0 )
|
|
|
|
{
|
2012-01-24 13:29:31 +00:00
|
|
|
distY = std::max(0, blendY - (relY+1)*splatSize);
|
2012-01-21 17:59:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float blendAmount = blendDist - std::sqrt((float)distX*distX + distY*distY);
|
|
|
|
blendAmount /= blendDist;
|
|
|
|
|
|
|
|
//this is required as blendDist < sqrt(blendDist*blendDist + blendDist*blendDist)
|
|
|
|
//this means that the corners are slightly the "wrong" shape but totaly smooth
|
|
|
|
//shading for the edges
|
|
|
|
blendAmount = std::max( (float) 0.0, blendAmount);
|
|
|
|
|
|
|
|
assert(blendAmount >= 0 && "Blend should never be negative");
|
|
|
|
|
|
|
|
//flips the y
|
|
|
|
const int index = blendSize*(blendSize - 1 - blendY) + blendX;
|
|
|
|
pBlend[index] += blendAmount;
|
|
|
|
pBlend[index] = std::min((float)1, pBlend[index]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//update the maps
|
|
|
|
for ( iter = indexes.begin(); iter != indexes.end(); ++iter )
|
|
|
|
{
|
|
|
|
Ogre::TerrainLayerBlendMap* blend = terrain->getLayerBlendMap(iter->second);
|
|
|
|
blend->dirty();
|
|
|
|
blend->update();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-01-24 13:29:31 +00:00
|
|
|
//----------------------------------------------------------------------------------------------
|
2012-01-23 17:19:54 +00:00
|
|
|
|
|
|
|
int TerrainManager::getLtexIndexAt(MWWorld::Ptr::CellStore* store,
|
|
|
|
int x, int y)
|
|
|
|
{
|
|
|
|
//check texture index falls within the 9 cell bounds
|
|
|
|
//as this function can't cope with anything above that
|
|
|
|
assert(x >= -ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
y >= -ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
"Trying to get land textures that are out of bounds");
|
|
|
|
|
|
|
|
assert(x < 2*ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
y < 2*ESM::Land::LAND_TEXTURE_SIZE &&
|
|
|
|
"Trying to get land textures that are out of bounds");
|
|
|
|
|
|
|
|
assert(store != NULL && "Store pointer must be valid");
|
|
|
|
|
|
|
|
//default center cell is indexed at (1,1)
|
|
|
|
int cellX = 1;
|
|
|
|
int cellY = 1;
|
|
|
|
|
|
|
|
if ( x < 0 )
|
|
|
|
{
|
|
|
|
cellX--;
|
|
|
|
x += ESM::Land::LAND_TEXTURE_SIZE;
|
|
|
|
}
|
|
|
|
else if ( x >= ESM::Land::LAND_TEXTURE_SIZE )
|
|
|
|
{
|
|
|
|
cellX++;
|
|
|
|
x -= ESM::Land::LAND_TEXTURE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( y < 0 )
|
|
|
|
{
|
|
|
|
cellY--;
|
|
|
|
y += ESM::Land::LAND_TEXTURE_SIZE;
|
|
|
|
}
|
|
|
|
else if ( y >= ESM::Land::LAND_TEXTURE_SIZE )
|
|
|
|
{
|
|
|
|
cellY++;
|
|
|
|
y -= ESM::Land::LAND_TEXTURE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return store->land[cellX][cellY]
|
|
|
|
->landData
|
|
|
|
->textures[y * ESM::Land::LAND_TEXTURE_SIZE + x];
|
|
|
|
}
|
|
|
|
|
2012-01-21 16:59:08 +00:00
|
|
|
}
|