1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-18 04:10:06 +00:00
OpenMW/components/terrain/terraingrid.cpp

105 lines
2.7 KiB
C++
Raw Normal View History

#include "terraingrid.hpp"
2015-06-02 23:18:36 +00:00
#include <memory>
#include <osg/Group>
#include "chunkmanager.hpp"
namespace Terrain
{
class MyView : public View
{
public:
osg::ref_ptr<osg::Node> mLoaded;
virtual void reset(unsigned int frame) {}
};
2018-06-12 23:48:31 +00:00
TerrainGrid::TerrainGrid(osg::Group* parent, osg::Group* compileRoot, Resource::ResourceSystem* resourceSystem, Storage* storage, int nodeMask, int preCompileMask, int borderMask)
: Terrain::World(parent, compileRoot, resourceSystem, storage, nodeMask, preCompileMask, borderMask)
2018-06-14 10:27:22 +00:00
, mNumSplits(4)
{
}
TerrainGrid::~TerrainGrid()
{
while (!mGrid.empty())
{
unloadCell(mGrid.begin()->first.first, mGrid.begin()->first.second);
}
}
void TerrainGrid::cacheCell(View* view, int x, int y)
2016-02-09 19:57:30 +00:00
{
osg::Vec2f center(x+0.5f, y+0.5f);
static_cast<MyView*>(view)->mLoaded = buildTerrain(NULL, 1.f, center);
2016-02-09 19:57:30 +00:00
}
osg::ref_ptr<osg::Node> TerrainGrid::buildTerrain (osg::Group* parent, float chunkSize, const osg::Vec2f& chunkCenter)
{
if (chunkSize * mNumSplits > 1.f)
{
// keep splitting
osg::ref_ptr<osg::Group> group (new osg::Group);
if (parent)
parent->addChild(group);
2015-11-06 19:21:39 +00:00
float newChunkSize = chunkSize/2.f;
buildTerrain(group, newChunkSize, chunkCenter + osg::Vec2f(newChunkSize/2.f, newChunkSize/2.f));
buildTerrain(group, newChunkSize, chunkCenter + osg::Vec2f(newChunkSize/2.f, -newChunkSize/2.f));
buildTerrain(group, newChunkSize, chunkCenter + osg::Vec2f(-newChunkSize/2.f, newChunkSize/2.f));
buildTerrain(group, newChunkSize, chunkCenter + osg::Vec2f(-newChunkSize/2.f, -newChunkSize/2.f));
return group;
}
else
{
2017-03-08 23:45:31 +00:00
osg::ref_ptr<osg::Node> node = mChunkManager->getChunk(chunkSize, chunkCenter, 0, 0);
if (!node)
return NULL;
if (parent)
parent->addChild(node);
return node;
2015-06-02 23:18:36 +00:00
}
}
void TerrainGrid::loadCell(int x, int y)
{
if (mGrid.find(std::make_pair(x, y)) != mGrid.end())
return; // already loaded
osg::Vec2f center(x+0.5f, y+0.5f);
osg::ref_ptr<osg::Node> terrainNode = buildTerrain(NULL, 1.f, center);
if (!terrainNode)
return; // no terrain defined
2018-06-14 10:27:22 +00:00
TerrainGrid::World::loadCell(x,y);
2018-06-12 23:48:31 +00:00
2016-02-09 19:23:53 +00:00
mTerrainRoot->addChild(terrainNode);
2016-02-09 19:23:53 +00:00
mGrid[std::make_pair(x,y)] = terrainNode;
}
2015-06-02 23:18:36 +00:00
void TerrainGrid::unloadCell(int x, int y)
{
2018-06-12 23:48:31 +00:00
MWRender::CellBorder::CellGrid::iterator it = mGrid.find(std::make_pair(x,y));
if (it == mGrid.end())
2015-06-02 23:18:36 +00:00
return;
2018-06-14 10:27:22 +00:00
Terrain::World::unloadCell(x,y);
2018-06-14 10:01:09 +00:00
2016-02-09 19:57:30 +00:00
osg::ref_ptr<osg::Node> terrainNode = it->second;
2016-02-09 19:23:53 +00:00
mTerrainRoot->removeChild(terrainNode);
2015-06-02 23:18:36 +00:00
mGrid.erase(it);
}
View *TerrainGrid::createView()
{
return new MyView;
}
}