2014-06-29 00:42:36 +00:00
|
|
|
#include "terraingrid.hpp"
|
|
|
|
|
2015-06-02 23:18:36 +00:00
|
|
|
#include <memory>
|
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
#include <osg/ComputeBoundsVisitor>
|
2017-03-06 19:41:02 +00:00
|
|
|
#include <osg/Group>
|
2017-03-03 17:26:40 +00:00
|
|
|
|
2017-03-06 19:41:02 +00:00
|
|
|
#include "chunkmanager.hpp"
|
2022-07-18 16:11:37 +00:00
|
|
|
#include "heightcull.hpp"
|
2019-06-13 13:37:00 +00:00
|
|
|
#include "storage.hpp"
|
2022-07-18 16:11:37 +00:00
|
|
|
#include "view.hpp"
|
2019-06-13 13:37:00 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2022-07-18 16:11:37 +00:00
|
|
|
|
2014-08-07 18:43:33 +00:00
|
|
|
namespace Terrain
|
2014-06-29 00:42:36 +00:00
|
|
|
{
|
|
|
|
|
2017-03-09 19:52:50 +00:00
|
|
|
class MyView : public View
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
osg::ref_ptr<osg::Node> mLoaded;
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void reset() override {}
|
2017-03-09 19:52:50 +00:00
|
|
|
};
|
|
|
|
|
2021-04-20 06:42:06 +00:00
|
|
|
TerrainGrid::TerrainGrid(osg::Group* parent, osg::Group* compileRoot, Resource::ResourceSystem* resourceSystem,
|
2023-09-09 17:29:26 +00:00
|
|
|
Storage* storage, unsigned int nodeMask, ESM::RefId worldspace, double expiryDelay, unsigned int preCompileMask,
|
2023-05-14 16:47:09 +00:00
|
|
|
unsigned int borderMask)
|
2023-09-09 17:29:26 +00:00
|
|
|
: Terrain::World(
|
|
|
|
parent, compileRoot, resourceSystem, storage, nodeMask, preCompileMask, borderMask, worldspace, expiryDelay)
|
2020-01-12 07:42:47 +00:00
|
|
|
, mNumSplits(4)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-05-14 16:47:09 +00:00
|
|
|
TerrainGrid::TerrainGrid(osg::Group* parent, Storage* storage, ESM::RefId worldspace, unsigned int nodeMask)
|
|
|
|
: Terrain::World(parent, storage, nodeMask, worldspace)
|
2018-06-14 10:27:22 +00:00
|
|
|
, mNumSplits(4)
|
2014-06-29 00:42:36 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-03-09 19:52:50 +00:00
|
|
|
TerrainGrid::~TerrainGrid()
|
2016-02-09 19:57:30 +00:00
|
|
|
{
|
2018-10-09 06:21:12 +00:00
|
|
|
while (!mGrid.empty())
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2018-10-09 06:21:12 +00:00
|
|
|
TerrainGrid::unloadCell(mGrid.begin()->first.first, mGrid.begin()->first.second);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2016-02-09 19:57:30 +00:00
|
|
|
}
|
|
|
|
|
2015-11-06 19:14:57 +00:00
|
|
|
void TerrainGrid::cacheCell(View* view, int x, int y)
|
|
|
|
{
|
|
|
|
osg::Vec2f center(x + 0.5f, y + 0.5f);
|
|
|
|
static_cast<MyView*>(view)->mLoaded = buildTerrain(nullptr, 1.f, center);
|
|
|
|
}
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2018-10-09 06:21:12 +00:00
|
|
|
osg::ref_ptr<osg::Node> TerrainGrid::buildTerrain(
|
2015-11-06 19:14:57 +00:00
|
|
|
osg::Group* parent, float chunkSize, const osg::Vec2f& chunkCenter)
|
|
|
|
{
|
|
|
|
if (chunkSize * mNumSplits > 1.f)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2015-11-06 19:14:57 +00:00
|
|
|
// keep splitting
|
|
|
|
osg::ref_ptr<osg::Group> group(new osg::Group);
|
|
|
|
if (parent)
|
|
|
|
parent->addChild(group);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-11-06 19:14:57 +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;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-04-30 13:37:00 +00:00
|
|
|
osg::ref_ptr<osg::Node> node
|
|
|
|
= mChunkManager->getChunk(chunkSize, chunkCenter, 0, 0, false, osg::Vec3f(), true);
|
2017-03-06 19:41:02 +00:00
|
|
|
if (!node)
|
2018-10-09 06:21:12 +00:00
|
|
|
return nullptr;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2023-05-18 09:56:45 +00:00
|
|
|
const float cellWorldSize = mStorage->getCellWorldSize(mWorldspace);
|
2019-06-13 13:37:00 +00:00
|
|
|
osg::ref_ptr<SceneUtil::PositionAttitudeTransform> pat = new SceneUtil::PositionAttitudeTransform;
|
|
|
|
pat->setPosition(osg::Vec3f(chunkCenter.x() * cellWorldSize, chunkCenter.y() * cellWorldSize, 0.f));
|
|
|
|
pat->addChild(node);
|
2015-11-06 19:14:57 +00:00
|
|
|
if (parent)
|
2019-06-13 13:37:00 +00:00
|
|
|
parent->addChild(pat);
|
|
|
|
return pat;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2015-06-02 23:18:36 +00:00
|
|
|
}
|
2015-11-06 19:14:57 +00:00
|
|
|
|
|
|
|
void TerrainGrid::loadCell(int x, int y)
|
|
|
|
{
|
|
|
|
if (mGrid.find(std::make_pair(x, y)) != mGrid.end())
|
|
|
|
return; // already loaded
|
2014-06-29 00:42:36 +00:00
|
|
|
|
2017-03-06 19:41:02 +00:00
|
|
|
osg::Vec2f center(x + 0.5f, y + 0.5f);
|
2018-10-09 06:21:12 +00:00
|
|
|
osg::ref_ptr<osg::Node> terrainNode = buildTerrain(nullptr, 1.f, center);
|
2015-11-06 19:14:57 +00:00
|
|
|
if (!terrainNode)
|
2017-03-06 19:41:02 +00:00
|
|
|
return; // no terrain defined
|
2015-11-06 19:14:57 +00:00
|
|
|
|
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);
|
2014-06-29 00:42:36 +00:00
|
|
|
|
2016-02-09 19:23:53 +00:00
|
|
|
mGrid[std::make_pair(x, y)] = terrainNode;
|
2019-06-13 13:37:00 +00:00
|
|
|
updateWaterCulling();
|
2014-06-29 00:42:36 +00:00
|
|
|
}
|
|
|
|
|
2015-06-02 23:18:36 +00:00
|
|
|
void TerrainGrid::unloadCell(int x, int y)
|
2014-06-29 00:42:36 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
CellBorder::CellGrid::iterator it = mGrid.find(std::make_pair(x, y));
|
2014-06-29 00:42:36 +00:00
|
|
|
if (it == mGrid.end())
|
2015-06-02 23:18:36 +00:00
|
|
|
return;
|
2014-06-29 00:42:36 +00:00
|
|
|
|
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);
|
2016-02-09 14:30:53 +00:00
|
|
|
|
2015-06-02 23:18:36 +00:00
|
|
|
mGrid.erase(it);
|
2019-06-13 13:37:00 +00:00
|
|
|
updateWaterCulling();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TerrainGrid::updateWaterCulling()
|
|
|
|
{
|
2020-01-12 07:42:47 +00:00
|
|
|
if (!mHeightCullCallback)
|
|
|
|
return;
|
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
osg::ComputeBoundsVisitor computeBoundsVisitor;
|
|
|
|
mTerrainRoot->accept(computeBoundsVisitor);
|
|
|
|
float lowZ = computeBoundsVisitor.getBoundingBox()._min.z();
|
|
|
|
mHeightCullCallback->setLowZ(lowZ);
|
2014-06-29 00:42:36 +00:00
|
|
|
}
|
|
|
|
|
2017-03-09 19:52:50 +00:00
|
|
|
View* TerrainGrid::createView()
|
|
|
|
{
|
|
|
|
return new MyView;
|
|
|
|
}
|
|
|
|
|
2014-06-29 00:42:36 +00:00
|
|
|
}
|