1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-10-24 06:26:36 +00:00

Merge branch 'terrain' of https://github.com/Yacoby/openmw into terrain_next

Conflicts:
	apps/openmw/mwrender/terrain.cpp
This commit is contained in:
scrawl 2012-03-02 13:49:57 +01:00
commit 04c141ddd5
4 changed files with 88 additions and 9 deletions

View file

@ -123,6 +123,9 @@ namespace MWRender
const int terrainX = cellX * 2 + x; const int terrainX = cellX * 2 + x;
const int terrainY = cellY * 2 + y; const int terrainY = cellY * 2 + y;
//it makes far more sense to reallocate the memory here,
//and let Ogre deal with it due to the issues with deleting
//it at the wrong time if using threads (Which Ogre::Terrain does)
terrainData.inputFloat = OGRE_ALLOC_T(float, terrainData.inputFloat = OGRE_ALLOC_T(float,
mLandSize*mLandSize, mLandSize*mLandSize,
Ogre::MEMCATEGORY_GEOMETRY); Ogre::MEMCATEGORY_GEOMETRY);
@ -151,10 +154,19 @@ namespace MWRender
mTerrainGroup->defineTerrain(terrainX, terrainY, &terrainData); mTerrainGroup->defineTerrain(terrainX, terrainY, &terrainData);
mTerrainGroup->loadTerrain(terrainX, terrainY, true); mTerrainGroup->loadTerrain(terrainX, terrainY, true);
Ogre::Terrain* terrain = mTerrainGroup->getTerrain(terrainX, terrainY); Ogre::Terrain* terrain = mTerrainGroup->getTerrain(terrainX, terrainY);
initTerrainBlendMaps(terrain, store, initTerrainBlendMaps(terrain, store,
x * numTextures, y * numTextures, x * numTextures, y * numTextures,
numTextures, indexes); numTextures,
indexes);
if ( store->land[1][1]->landData->usingColours )
{
Ogre::Image vertex = getVertexColours(store, x*32, y*32, mLandSize);
terrain->setGlobalColourMapEnabled(true);
terrain->getGlobalColourMap()->loadImage(vertex);
}
} }
} }
} }
@ -181,8 +193,17 @@ namespace MWRender
mTerrainGroup->loadTerrain(cellX, cellY, true); mTerrainGroup->loadTerrain(cellX, cellY, true);
Ogre::Terrain* terrain = mTerrainGroup->getTerrain(cellX, cellY); Ogre::Terrain* terrain = mTerrainGroup->getTerrain(cellX, cellY);
initTerrainBlendMaps(terrain, store, 0, 0, initTerrainBlendMaps(terrain, store, 0, 0,
ESM::Land::LAND_TEXTURE_SIZE, indexes); ESM::Land::LAND_TEXTURE_SIZE,
indexes);
if ( store->land[1][1]->landData->usingColours )
{
Ogre::Image vertex = getVertexColours(store, 0, 0, mLandSize);
terrain->setGlobalColourMapEnabled(true);
terrain->getGlobalColourMap()->loadImage(vertex);
}
} }
mTerrainGroup->freeTemporaryResources(); mTerrainGroup->freeTemporaryResources();
@ -245,8 +266,8 @@ namespace MWRender
if ( it == indexes.end() ) if ( it == indexes.end() )
{ {
//NB: All vtex ids are +1 compared to the ltex ids //NB: All vtex ids are +1 compared to the ltex ids
assert((int)ltexIndex >= 0 &&
(int)store->landTextures->ltex.size() > (int)ltexIndex - 1 && assert( (int)store->landTextures->ltex.size() >= (int)ltexIndex - 1 &&
"LAND.VTEX must be within the bounds of the LTEX array"); "LAND.VTEX must be within the bounds of the LTEX array");
std::string texture; std::string texture;
@ -376,10 +397,9 @@ namespace MWRender
} }
} }
//update the maps for ( int i = 1; i < terrain->getLayerCount(); i++ )
for ( iter = indexes.begin(); iter != indexes.end(); ++iter )
{ {
Ogre::TerrainLayerBlendMap* blend = terrain->getLayerBlendMap(iter->second); Ogre::TerrainLayerBlendMap* blend = terrain->getLayerBlendMap(i);
blend->dirty(); blend->dirty();
blend->update(); blend->update();
} }
@ -473,4 +493,43 @@ namespace MWRender
return tex; return tex;
} }
//----------------------------------------------------------------------------------------------
Ogre::Image TerrainManager::getVertexColours(MWWorld::Ptr::CellStore* store,
int fromX, int fromY, int size)
{
const char* const colours = store->land[1][1]->landData->colours;
Ogre::uchar* imgData = OGRE_ALLOC_T(Ogre::uchar,
size*size*sizeof(Ogre::uchar)*3,
Ogre::MEMCATEGORY_GENERAL);
for ( int y = 0; y < size; y++ )
{
for ( int x = 0; x < size; x++ )
{
const size_t colourOffset = (y+fromY)*3*65 + (x+fromX)*3;
assert( colourOffset >= 0 && colourOffset < 65*65*3 &&
"Colour offset is out of the expected bounds of record" );
const unsigned char r = colours[colourOffset + 0];
const unsigned char g = colours[colourOffset + 1];
const unsigned char b = colours[colourOffset + 2];
//as is the case elsewhere we need to flip the y
const size_t imageOffset = (size - 1 - y)*size*3 + x*3;
imgData[imageOffset + 0] = r;
imgData[imageOffset + 1] = g;
imgData[imageOffset + 2] = b;
}
}
Ogre::Image img;
img.loadDynamicImage(imgData, size, size, 1, Ogre::PF_R8G8B8, true);
return img;
}
} }

View file

@ -121,7 +121,20 @@ namespace MWRender{
* @param fileName the name of the *diffuse* texture * @param fileName the name of the *diffuse* texture
*/ */
Ogre::TexturePtr getNormalDisp(const std::string& fileName); Ogre::TexturePtr getNormalDisp(const std::string& fileName);
/**
* Due to the fact that Ogre terrain doesn't support vertex colours
* we have to generate them manually
*
* @param store the cell store for the given terrain cell
* @param fromX the *vertex* index in the current cell to start making texture from
* @param fromY the *vertex* index in the current cell to start making the texture from
* @param size the size (number of vertexes) to get
*
* @TODO FIXME the return of this function possibly copies the image data
*/
Ogre::Image getVertexColours(MWWorld::Ptr::CellStore* store,
int fromX, int fromY, int size);
}; };
} }

View file

@ -93,7 +93,10 @@ void Land::loadData(ESMReader &esm)
} }
if (esm.isNextSub("VCLR")) if (esm.isNextSub("VCLR"))
{ {
esm.skipHSubSize(12675); landData->usingColours = true;
esm.getHExact(&landData->colours, 3*LAND_NUM_VERTS);
}else{
landData->usingColours = false;
} }
//TODO fix magic numbers //TODO fix magic numbers
uint16_t vtex[512]; uint16_t vtex[512];
@ -108,6 +111,8 @@ void Land::loadData(ESMReader &esm)
} }
else else
{ {
landData->usingColours = false;
memset(&landData->textures, 0, 512 * sizeof(uint16_t));
for (int i = 0; i < LAND_NUM_VERTS; i++) for (int i = 0; i < LAND_NUM_VERTS; i++)
{ {
landData->heights[i] = -256.0f * HEIGHT_SCALE; landData->heights[i] = -256.0f * HEIGHT_SCALE;

View file

@ -58,6 +58,8 @@ struct Land
float heights[LAND_NUM_VERTS]; float heights[LAND_NUM_VERTS];
//float normals[LAND_NUM_VERTS * 3]; //float normals[LAND_NUM_VERTS * 3];
uint16_t textures[LAND_NUM_TEXTURES]; uint16_t textures[LAND_NUM_TEXTURES];
bool usingColours;
char colours[3 * LAND_NUM_VERTS]; char colours[3 * LAND_NUM_VERTS];
}; };