2017-03-08 00:53:13 +00:00
|
|
|
#include "quadtreeworld.hpp"
|
|
|
|
|
2021-10-11 09:27:50 +00:00
|
|
|
#include <osg/Material>
|
2020-04-23 09:12:10 +00:00
|
|
|
#include <osg/PolygonMode>
|
|
|
|
#include <osg/ShapeDrawable>
|
2017-03-08 23:01:13 +00:00
|
|
|
#include <osgUtil/CullVisitor>
|
|
|
|
|
2020-04-23 08:10:50 +00:00
|
|
|
#include <limits>
|
2017-03-09 20:15:12 +00:00
|
|
|
|
2021-09-05 15:43:46 +00:00
|
|
|
#include <components/loadinglistener/reporter.hpp>
|
2018-09-17 10:52:43 +00:00
|
|
|
#include <components/misc/constants.hpp>
|
2023-02-05 20:30:38 +00:00
|
|
|
#include <components/misc/mathutil.hpp>
|
2021-09-27 19:25:39 +00:00
|
|
|
#include <components/resource/resourcesystem.hpp>
|
2019-06-13 13:37:00 +00:00
|
|
|
#include <components/sceneutil/positionattitudetransform.hpp>
|
2018-02-05 23:03:18 +00:00
|
|
|
|
2017-03-08 21:19:48 +00:00
|
|
|
#include "chunkmanager.hpp"
|
2017-03-09 00:56:43 +00:00
|
|
|
#include "compositemaprenderer.hpp"
|
2022-07-18 16:11:37 +00:00
|
|
|
#include "heightcull.hpp"
|
2017-03-08 00:53:13 +00:00
|
|
|
#include "quadtreenode.hpp"
|
|
|
|
#include "storage.hpp"
|
2020-04-23 09:12:10 +00:00
|
|
|
#include "terraindrawable.hpp"
|
2017-03-08 21:19:48 +00:00
|
|
|
#include "viewdata.hpp"
|
2017-03-08 00:53:13 +00:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
unsigned int Log2(unsigned int n)
|
2017-03-08 21:19:48 +00:00
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
unsigned int targetlevel = 0;
|
2017-03-08 21:19:48 +00:00
|
|
|
while (n >>= 1)
|
|
|
|
++targetlevel;
|
|
|
|
return targetlevel;
|
|
|
|
}
|
|
|
|
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace Terrain
|
|
|
|
{
|
|
|
|
|
2017-03-08 21:19:48 +00:00
|
|
|
class DefaultLodCallback : public LodCallback
|
2017-03-12 21:39:53 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2023-05-19 08:19:16 +00:00
|
|
|
DefaultLodCallback(float factor, float minSize, float viewDistance, const osg::Vec4i& grid, int cellSizeInUnits,
|
|
|
|
float distanceModifier = 0.f)
|
2019-02-27 15:41:07 +00:00
|
|
|
: mFactor(factor)
|
|
|
|
, mMinSize(minSize)
|
2020-05-10 13:37:00 +00:00
|
|
|
, mViewDistance(viewDistance)
|
2020-04-30 13:37:00 +00:00
|
|
|
, mActiveGrid(grid)
|
2021-09-27 19:32:18 +00:00
|
|
|
, mDistanceModifier(distanceModifier)
|
2023-05-19 08:19:16 +00:00
|
|
|
, mCellSizeInUnits(cellSizeInUnits)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-12 21:39:53 +00:00
|
|
|
}
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
ReturnValue isSufficientDetail(QuadTreeNode* node, float dist) override
|
2017-03-08 21:19:48 +00:00
|
|
|
{
|
2020-05-10 13:37:00 +00:00
|
|
|
const osg::Vec2f& center = node->getCenter();
|
|
|
|
bool activeGrid = (center.x() > mActiveGrid.x() && center.y() > mActiveGrid.y()
|
|
|
|
&& center.x() < mActiveGrid.z() && center.y() < mActiveGrid.w());
|
2021-08-09 20:56:04 +00:00
|
|
|
|
2020-04-30 13:37:00 +00:00
|
|
|
if (node->getSize() > 1)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2020-04-30 13:37:00 +00:00
|
|
|
float halfSize = node->getSize() / 2;
|
|
|
|
osg::Vec4i nodeBounds(static_cast<int>(center.x() - halfSize), static_cast<int>(center.y() - halfSize),
|
|
|
|
static_cast<int>(center.x() + halfSize), static_cast<int>(center.y() + halfSize));
|
2020-05-06 13:37:00 +00:00
|
|
|
bool intersects = (std::max(nodeBounds.x(), mActiveGrid.x()) < std::min(nodeBounds.z(), mActiveGrid.z())
|
|
|
|
&& std::max(nodeBounds.y(), mActiveGrid.y()) < std::min(nodeBounds.w(), mActiveGrid.w()));
|
2019-06-13 13:37:00 +00:00
|
|
|
// to prevent making chunks who will cross the activegrid border
|
2020-04-30 13:37:00 +00:00
|
|
|
if (intersects)
|
2021-11-08 09:27:42 +00:00
|
|
|
return Deeper;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-09-27 19:32:18 +00:00
|
|
|
dist = std::max(0.f, dist + mDistanceModifier);
|
2021-08-09 20:56:04 +00:00
|
|
|
if (dist > mViewDistance && !activeGrid) // for Scene<->ObjectPaging sync the activegrid must remain loaded
|
|
|
|
return StopTraversal;
|
2023-05-19 08:19:16 +00:00
|
|
|
return getNativeLodLevel(node, mMinSize)
|
|
|
|
<= convertDistanceToLodLevel(dist, mMinSize, mFactor, mCellSizeInUnits)
|
2021-11-08 09:27:42 +00:00
|
|
|
? StopTraversalAndUse
|
|
|
|
: Deeper;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-11-08 09:27:42 +00:00
|
|
|
static unsigned int getNativeLodLevel(const QuadTreeNode* node, float minSize)
|
2019-06-13 13:37:00 +00:00
|
|
|
{
|
2020-04-30 13:37:00 +00:00
|
|
|
return Log2(static_cast<unsigned int>(node->getSize() / minSize));
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2023-05-19 08:19:16 +00:00
|
|
|
static unsigned int convertDistanceToLodLevel(float dist, float minSize, float factor, int cellSize)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2023-05-19 08:19:16 +00:00
|
|
|
return Log2(static_cast<unsigned int>(dist / (cellSize * minSize * factor)));
|
2019-06-13 13:37:00 +00:00
|
|
|
}
|
2017-03-12 21:39:53 +00:00
|
|
|
|
|
|
|
private:
|
2019-02-27 15:41:07 +00:00
|
|
|
float mFactor;
|
2017-03-12 21:39:53 +00:00
|
|
|
float mMinSize;
|
2020-05-10 13:37:00 +00:00
|
|
|
float mViewDistance;
|
2020-04-30 13:37:00 +00:00
|
|
|
osg::Vec4i mActiveGrid;
|
2021-09-27 19:32:18 +00:00
|
|
|
float mDistanceModifier;
|
2023-05-19 08:19:16 +00:00
|
|
|
int mCellSizeInUnits;
|
2017-03-08 21:19:48 +00:00
|
|
|
};
|
2017-03-08 00:53:13 +00:00
|
|
|
|
|
|
|
class RootNode : public QuadTreeNode
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2017-03-08 00:53:13 +00:00
|
|
|
RootNode(float size, const osg::Vec2f& center)
|
2018-10-09 06:21:12 +00:00
|
|
|
: QuadTreeNode(nullptr, Root, size, center)
|
|
|
|
, mWorld(nullptr)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void setWorld(QuadTreeWorld* world) { mWorld = world; }
|
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void accept(osg::NodeVisitor& nv) override
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2017-03-09 01:17:25 +00:00
|
|
|
if (!nv.validNodeMask(*this))
|
|
|
|
return;
|
2017-03-08 00:53:13 +00:00
|
|
|
nv.pushOntoNodePath(this);
|
|
|
|
mWorld->accept(nv);
|
|
|
|
nv.popFromNodePath();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QuadTreeWorld* mWorld;
|
|
|
|
};
|
|
|
|
|
2017-03-09 03:01:27 +00:00
|
|
|
class QuadTreeBuilder
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2023-05-14 18:00:02 +00:00
|
|
|
QuadTreeBuilder(Terrain::Storage* storage, float minSize, ESM::RefId worldspace)
|
2017-03-08 00:53:13 +00:00
|
|
|
: mStorage(storage)
|
|
|
|
, mMinX(0.f)
|
|
|
|
, mMaxX(0.f)
|
|
|
|
, mMinY(0.f)
|
|
|
|
, mMaxY(0.f)
|
|
|
|
, mMinSize(minSize)
|
2023-05-14 18:00:02 +00:00
|
|
|
, mWorldspace(worldspace)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
2017-03-09 03:01:27 +00:00
|
|
|
void build()
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2023-05-18 23:07:34 +00:00
|
|
|
mStorage->getBounds(mMinX, mMaxX, mMinY, mMaxY, mWorldspace);
|
2017-03-08 00:53:13 +00:00
|
|
|
|
|
|
|
int origSizeX = static_cast<int>(mMaxX - mMinX);
|
|
|
|
int origSizeY = static_cast<int>(mMaxY - mMinY);
|
|
|
|
|
|
|
|
// Dividing a quad tree only works well for powers of two, so round up to the nearest one
|
2023-02-05 20:30:38 +00:00
|
|
|
int size = Misc::nextPowerOfTwo(std::max(origSizeX, origSizeY));
|
2017-03-08 00:53:13 +00:00
|
|
|
|
|
|
|
float centerX = (mMinX + mMaxX) / 2.f + (size - origSizeX) / 2.f;
|
|
|
|
float centerY = (mMinY + mMaxY) / 2.f + (size - origSizeY) / 2.f;
|
|
|
|
|
|
|
|
mRootNode = new RootNode(size, osg::Vec2f(centerX, centerY));
|
|
|
|
addChildren(mRootNode);
|
|
|
|
|
|
|
|
mRootNode->initNeighbours();
|
2023-05-18 09:56:45 +00:00
|
|
|
float cellWorldSize = mStorage->getCellWorldSize(mWorldspace);
|
2020-05-06 13:37:00 +00:00
|
|
|
mRootNode->setInitialBound(
|
|
|
|
osg::BoundingSphere(osg::BoundingBox(osg::Vec3(mMinX * cellWorldSize, mMinY * cellWorldSize, 0),
|
|
|
|
osg::Vec3(mMaxX * cellWorldSize, mMaxY * cellWorldSize, 0))));
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void addChildren(QuadTreeNode* parent)
|
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
float halfSize = parent->getSize() / 2.f;
|
|
|
|
osg::BoundingBox boundingBox;
|
2017-03-08 00:53:13 +00:00
|
|
|
for (unsigned int i = 0; i < 4; ++i)
|
2019-02-20 13:37:00 +00:00
|
|
|
{
|
|
|
|
osg::ref_ptr<QuadTreeNode> child = addChild(parent, static_cast<ChildDirection>(i), halfSize);
|
2022-09-22 18:26:05 +00:00
|
|
|
if (child)
|
|
|
|
{
|
2017-03-08 00:53:13 +00:00
|
|
|
boundingBox.expandBy(child->getBoundingBox());
|
2019-03-07 10:50:29 +00:00
|
|
|
parent->addChildNode(child);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-02-20 13:37:00 +00:00
|
|
|
}
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
if (!boundingBox.valid())
|
|
|
|
parent->removeChildren(0, 4);
|
|
|
|
else
|
|
|
|
parent->setBoundingBox(boundingBox);
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
osg::ref_ptr<QuadTreeNode> addChild(QuadTreeNode* parent, ChildDirection direction, float size)
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
float halfSize = size / 2.f;
|
2017-03-08 00:53:13 +00:00
|
|
|
osg::Vec2f center;
|
|
|
|
switch (direction)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-08 00:53:13 +00:00
|
|
|
case SW:
|
2019-02-20 13:37:00 +00:00
|
|
|
center = parent->getCenter() + osg::Vec2f(-halfSize, -halfSize);
|
2017-03-08 00:53:13 +00:00
|
|
|
break;
|
|
|
|
case SE:
|
2019-02-20 13:37:00 +00:00
|
|
|
center = parent->getCenter() + osg::Vec2f(halfSize, -halfSize);
|
2017-03-08 00:53:13 +00:00
|
|
|
break;
|
|
|
|
case NW:
|
2019-02-20 13:37:00 +00:00
|
|
|
center = parent->getCenter() + osg::Vec2f(-halfSize, halfSize);
|
2017-03-08 00:53:13 +00:00
|
|
|
break;
|
|
|
|
case NE:
|
2019-02-20 13:37:00 +00:00
|
|
|
center = parent->getCenter() + osg::Vec2f(halfSize, halfSize);
|
2017-03-08 00:53:13 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
osg::ref_ptr<QuadTreeNode> node = new QuadTreeNode(parent, direction, size, center);
|
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
if (center.x() - halfSize > mMaxX || center.x() + halfSize < mMinX || center.y() - halfSize > mMaxY
|
|
|
|
|| center.y() + halfSize < mMinY)
|
2019-02-28 18:10:13 +00:00
|
|
|
// Out of bounds of the actual terrain - this will happen because
|
|
|
|
// we rounded the size up to the next power of two
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2019-02-28 18:10:13 +00:00
|
|
|
// Still create and return an empty node so as to not break the assumption that each QuadTreeNode has
|
|
|
|
// either 4 or 0 children.
|
2017-03-08 00:53:13 +00:00
|
|
|
return node;
|
|
|
|
}
|
2018-05-22 08:40:01 +00:00
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
// Do not add child nodes for default cells without data.
|
|
|
|
// size = 1 means that the single shape covers the whole cell.
|
2023-05-14 18:00:02 +00:00
|
|
|
if (node->getSize() == 1
|
|
|
|
&& !mStorage->hasData(ESM::ExteriorCellLocation(center.x() - 0.5, center.y() - 0.5, mWorldspace)))
|
2019-02-20 13:37:00 +00:00
|
|
|
return node;
|
2018-05-22 08:40:01 +00:00
|
|
|
|
2019-02-28 18:10:13 +00:00
|
|
|
if (node->getSize() <= mMinSize)
|
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
// We arrived at a leaf.
|
2019-02-20 13:37:00 +00:00
|
|
|
// Since the tree is used for LOD level selection instead of culling, we do not need to load the actual
|
|
|
|
// height data here.
|
2021-05-04 21:03:14 +00:00
|
|
|
constexpr float minZ = -std::numeric_limits<float>::max();
|
|
|
|
constexpr float maxZ = std::numeric_limits<float>::max();
|
2023-05-18 09:56:45 +00:00
|
|
|
float cellWorldSize = mStorage->getCellWorldSize(mWorldspace);
|
2019-02-20 13:37:00 +00:00
|
|
|
osg::BoundingBox boundingBox(
|
|
|
|
osg::Vec3f((center.x() - halfSize) * cellWorldSize, (center.y() - halfSize) * cellWorldSize, minZ),
|
|
|
|
osg::Vec3f((center.x() + halfSize) * cellWorldSize, (center.y() + halfSize) * cellWorldSize, maxZ));
|
|
|
|
node->setBoundingBox(boundingBox);
|
2019-02-28 18:10:13 +00:00
|
|
|
return node;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
addChildren(node);
|
|
|
|
return node;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-02-28 18:10:13 +00:00
|
|
|
}
|
2017-03-08 00:53:13 +00:00
|
|
|
|
|
|
|
osg::ref_ptr<RootNode> getRootNode() { return mRootNode; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Terrain::Storage* mStorage;
|
|
|
|
|
|
|
|
float mMinX, mMaxX, mMinY, mMaxY;
|
|
|
|
float mMinSize;
|
|
|
|
|
|
|
|
osg::ref_ptr<RootNode> mRootNode;
|
2023-05-14 18:00:02 +00:00
|
|
|
ESM::RefId mWorldspace;
|
2017-03-08 00:53:13 +00:00
|
|
|
};
|
|
|
|
|
2021-09-27 19:25:39 +00:00
|
|
|
class DebugChunkManager : public QuadTreeWorld::ChunkManager
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
public:
|
2023-05-14 18:00:02 +00:00
|
|
|
DebugChunkManager(
|
|
|
|
Resource::SceneManager* sceneManager, Storage* storage, unsigned int nodeMask, ESM::RefId worldspace)
|
|
|
|
: QuadTreeWorld::ChunkManager(worldspace)
|
|
|
|
, mSceneManager(sceneManager)
|
2021-09-27 19:25:39 +00:00
|
|
|
, mStorage(storage)
|
|
|
|
, mNodeMask(nodeMask)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
|
|
|
}
|
2021-09-27 19:25:39 +00:00
|
|
|
osg::ref_ptr<osg::Node> getChunk(float size, const osg::Vec2f& chunkCenter, unsigned char lod,
|
|
|
|
unsigned int lodFlags, bool activeGrid, const osg::Vec3f& viewPoint, bool compile)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-09-27 19:25:39 +00:00
|
|
|
osg::Vec3f center = { chunkCenter.x(), chunkCenter.y(), 0 };
|
|
|
|
auto chunkBorder = CellBorder::createBorderGeometry(center.x() - size / 2.f, center.y() - size / 2.f, size,
|
2023-05-14 18:00:02 +00:00
|
|
|
mStorage, mSceneManager, mNodeMask, mWorldspace, 5.f, { 1, 0, 0, 0 });
|
2021-10-11 09:27:50 +00:00
|
|
|
osg::ref_ptr<SceneUtil::PositionAttitudeTransform> pat = new SceneUtil::PositionAttitudeTransform;
|
2023-05-15 21:48:22 +00:00
|
|
|
pat->setPosition(-center * ESM::getCellSize(mWorldspace));
|
2021-10-11 09:27:50 +00:00
|
|
|
pat->addChild(chunkBorder);
|
|
|
|
return pat;
|
2021-09-27 19:25:39 +00:00
|
|
|
}
|
|
|
|
unsigned int getNodeMask() { return mNodeMask; }
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-09-27 19:25:39 +00:00
|
|
|
private:
|
|
|
|
Resource::SceneManager* mSceneManager;
|
|
|
|
Storage* mStorage;
|
|
|
|
unsigned int mNodeMask;
|
|
|
|
};
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-09-29 15:10:58 +00:00
|
|
|
QuadTreeWorld::QuadTreeWorld(osg::Group* parent, osg::Group* compileRoot, Resource::ResourceSystem* resourceSystem,
|
|
|
|
Storage* storage, unsigned int nodeMask, unsigned int preCompileMask, unsigned int borderMask,
|
|
|
|
int compMapResolution, float compMapLevel, float lodFactor, int vertexLodMod, float maxCompGeometrySize,
|
2023-10-28 13:14:00 +00:00
|
|
|
bool debugChunks, ESM::RefId worldspace, double expiryDelay)
|
|
|
|
: TerrainGrid(
|
|
|
|
parent, compileRoot, resourceSystem, storage, nodeMask, worldspace, expiryDelay, preCompileMask, borderMask)
|
2017-03-09 01:03:51 +00:00
|
|
|
, mViewDataMap(new ViewDataMap)
|
2017-03-08 00:53:13 +00:00
|
|
|
, mQuadTreeBuilt(false)
|
2019-02-27 15:41:07 +00:00
|
|
|
, mLodFactor(lodFactor)
|
2019-02-20 13:37:00 +00:00
|
|
|
, mVertexLodMod(vertexLodMod)
|
2019-02-20 13:37:00 +00:00
|
|
|
, mViewDistance(std::numeric_limits<float>::max())
|
2023-05-18 22:16:58 +00:00
|
|
|
, mMinSize(ESM::isEsm4Ext(worldspace) ? 1 / 4.f : 1 / 8.f)
|
2021-09-29 15:10:58 +00:00
|
|
|
, mDebugTerrainChunks(debugChunks)
|
2021-09-27 19:25:39 +00:00
|
|
|
{
|
2022-05-29 11:24:32 +00:00
|
|
|
mChunkManager->setCompositeMapSize(compMapResolution);
|
|
|
|
mChunkManager->setCompositeMapLevel(compMapLevel);
|
|
|
|
mChunkManager->setMaxCompositeGeometrySize(maxCompGeometrySize);
|
2021-09-27 19:25:39 +00:00
|
|
|
mChunkManagers.push_back(mChunkManager.get());
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
if (mDebugTerrainChunks)
|
|
|
|
{
|
2023-05-14 18:00:02 +00:00
|
|
|
mDebugChunkManager = std::make_unique<DebugChunkManager>(
|
|
|
|
mResourceSystem->getSceneManager(), mStorage, borderMask, mWorldspace);
|
2019-06-13 13:37:00 +00:00
|
|
|
addChunkManager(mDebugChunkManager.get());
|
2019-02-20 13:37:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QuadTreeWorld::~QuadTreeWorld() {}
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
/// get the level of vertex detail to render this node at, expressed relative to the native resolution of the vertex
|
2021-11-08 09:27:42 +00:00
|
|
|
/// data set, NOT relative to mMinSize as is the case with node LODs.
|
|
|
|
unsigned int getVertexLod(QuadTreeNode* node, int vertexLodMod)
|
2017-03-08 23:45:31 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
unsigned int vertexLod = DefaultLodCallback::getNativeLodLevel(node, 1);
|
2021-11-08 09:27:42 +00:00
|
|
|
if (vertexLodMod > 0)
|
2017-03-08 23:45:31 +00:00
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
vertexLod = static_cast<unsigned int>(std::max(0, static_cast<int>(vertexLod) - vertexLodMod));
|
2017-03-08 23:45:31 +00:00
|
|
|
}
|
2019-02-20 13:37:00 +00:00
|
|
|
else if (vertexLodMod < 0)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
float size = node->getSize();
|
|
|
|
// Stop to simplify at this level since with size = 1 the node already covers the whole cell and has
|
|
|
|
// getCellVertices() vertices.
|
|
|
|
while (size < 1)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
size *= 2;
|
2021-11-08 09:27:42 +00:00
|
|
|
vertexLodMod = std::min(0, vertexLodMod + 1);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-11-08 09:27:42 +00:00
|
|
|
vertexLod += std::abs(vertexLodMod);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-11-08 09:27:42 +00:00
|
|
|
return vertexLod;
|
2017-03-08 23:45:31 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 09:27:42 +00:00
|
|
|
/// get the flags to use for stitching in the index buffer so that chunks of different LOD connect seamlessly
|
|
|
|
unsigned int getLodFlags(QuadTreeNode* node, unsigned int ourVertexLod, int vertexLodMod, const ViewData* vd)
|
2017-03-09 03:14:07 +00:00
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
unsigned int lodFlags = 0;
|
2017-03-09 03:14:07 +00:00
|
|
|
for (unsigned int i = 0; i < 4; ++i)
|
2017-03-12 19:17:16 +00:00
|
|
|
{
|
2018-10-09 06:21:12 +00:00
|
|
|
QuadTreeNode* neighbour = node->getNeighbour(static_cast<Direction>(i));
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2018-10-09 06:21:12 +00:00
|
|
|
// If the neighbour isn't currently rendering itself,
|
2017-03-08 23:45:31 +00:00
|
|
|
// go up until we find one. NOTE: We don't need to go down,
|
2018-10-09 06:21:12 +00:00
|
|
|
// because in that case neighbour's detail would be higher than
|
2017-03-08 23:45:31 +00:00
|
|
|
// our detail and the neighbour would handle stitching by itself.
|
|
|
|
while (neighbour && !vd->contains(neighbour))
|
2017-03-12 19:17:16 +00:00
|
|
|
neighbour = neighbour->getParent();
|
2021-11-08 09:27:42 +00:00
|
|
|
unsigned int lod = 0;
|
2017-03-08 23:45:31 +00:00
|
|
|
if (neighbour)
|
2021-11-08 09:27:42 +00:00
|
|
|
lod = getVertexLod(neighbour, vertexLodMod);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2017-03-12 19:17:16 +00:00
|
|
|
if (lod <= ourVertexLod) // We only need to worry about neighbours less detailed than we are -
|
2017-03-08 23:45:31 +00:00
|
|
|
lod = 0; // neighbours with more detail will do the stitching themselves
|
2017-03-12 19:17:16 +00:00
|
|
|
// Use 4 bits for each LOD delta
|
2017-03-08 23:45:31 +00:00
|
|
|
if (lod > 0)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-12 19:17:16 +00:00
|
|
|
lodFlags |= (lod - ourVertexLod) << (4 * i);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2017-03-12 19:17:16 +00:00
|
|
|
}
|
2021-11-08 09:27:42 +00:00
|
|
|
// Use the remaining bits for our vertex LOD
|
|
|
|
lodFlags |= (ourVertexLod << (4 * 4));
|
2017-03-08 23:45:31 +00:00
|
|
|
return lodFlags;
|
2017-03-09 03:14:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void QuadTreeWorld::loadRenderingNode(
|
|
|
|
ViewDataEntry& entry, ViewData* vd, float cellWorldSize, const osg::Vec4i& gridbounds, bool compile)
|
2019-06-13 13:37:00 +00:00
|
|
|
{
|
|
|
|
if (!vd->hasChanged() && entry.mRenderingNode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (vd->hasChanged())
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2022-10-04 23:05:09 +00:00
|
|
|
vd->buildNodeIndex();
|
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
unsigned int ourVertexLod = getVertexLod(entry.mNode, mVertexLodMod);
|
|
|
|
// have to recompute the lodFlags in case a neighbour has changed LOD.
|
|
|
|
unsigned int lodFlags = getLodFlags(entry.mNode, ourVertexLod, mVertexLodMod, vd);
|
2020-05-10 13:37:00 +00:00
|
|
|
if (lodFlags != entry.mLodFlags)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2020-05-10 13:37:00 +00:00
|
|
|
entry.mRenderingNode = nullptr;
|
|
|
|
entry.mLodFlags = lodFlags;
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
|
|
|
}
|
2019-06-13 13:37:00 +00:00
|
|
|
|
2021-10-31 11:59:34 +00:00
|
|
|
if (!entry.mRenderingNode)
|
2019-06-13 13:37:00 +00:00
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
osg::ref_ptr<SceneUtil::PositionAttitudeTransform> pat = new SceneUtil::PositionAttitudeTransform;
|
2019-06-13 13:37:00 +00:00
|
|
|
pat->setPosition(osg::Vec3f(
|
2021-11-08 09:27:42 +00:00
|
|
|
entry.mNode->getCenter().x() * cellWorldSize, entry.mNode->getCenter().y() * cellWorldSize, 0.f));
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-11-08 09:27:42 +00:00
|
|
|
const osg::Vec2f& center = entry.mNode->getCenter();
|
2020-05-10 13:37:00 +00:00
|
|
|
bool activeGrid = (center.x() > gridbounds.x() && center.y() > gridbounds.y() && center.x() < gridbounds.z()
|
|
|
|
&& center.y() < gridbounds.w());
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2021-10-31 11:59:34 +00:00
|
|
|
for (QuadTreeWorld::ChunkManager* m : mChunkManagers)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-11-08 09:27:42 +00:00
|
|
|
osg::ref_ptr<osg::Node> n = m->getChunk(entry.mNode->getSize(), entry.mNode->getCenter(),
|
|
|
|
DefaultLodCallback::getNativeLodLevel(entry.mNode, mMinSize), entry.mLodFlags, activeGrid,
|
|
|
|
vd->getViewPoint(), compile);
|
2019-06-13 13:37:00 +00:00
|
|
|
if (n)
|
|
|
|
pat->addChild(n);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-06-13 13:37:00 +00:00
|
|
|
entry.mRenderingNode = pat;
|
|
|
|
}
|
|
|
|
}
|
2017-03-09 03:14:07 +00:00
|
|
|
|
2021-09-27 19:25:39 +00:00
|
|
|
void updateWaterCullingView(
|
|
|
|
HeightCullCallback* callback, ViewData* vd, osgUtil::CullVisitor* cv, float cellworldsize, bool outofworld)
|
2019-06-13 13:37:00 +00:00
|
|
|
{
|
2020-04-23 08:10:50 +00:00
|
|
|
if (!(cv->getTraversalMask() & callback->getCullMask()))
|
2022-09-22 18:26:05 +00:00
|
|
|
return;
|
2020-04-23 08:10:50 +00:00
|
|
|
float lowZ = std::numeric_limits<float>::max();
|
2021-09-27 19:25:39 +00:00
|
|
|
float highZ = callback->getHighZ();
|
2019-06-13 13:37:00 +00:00
|
|
|
if (cv->getEyePoint().z() <= highZ || outofworld)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2020-04-23 09:12:10 +00:00
|
|
|
callback->setLowZ(-std::numeric_limits<float>::max());
|
2022-09-22 18:26:05 +00:00
|
|
|
return;
|
|
|
|
}
|
2020-04-23 09:12:10 +00:00
|
|
|
cv->pushCurrentMask();
|
|
|
|
static bool debug = getenv("OPENMW_WATER_CULLING_DEBUG") != nullptr;
|
|
|
|
for (unsigned int i = 0; i < vd->getNumEntries(); ++i)
|
|
|
|
{
|
2021-10-31 11:59:34 +00:00
|
|
|
ViewDataEntry& entry = vd->getEntry(i);
|
2021-09-27 19:25:39 +00:00
|
|
|
osg::BoundingBox bb
|
|
|
|
= static_cast<TerrainDrawable*>(entry.mRenderingNode->asGroup()->getChild(0))->getWaterBoundingBox();
|
2020-04-23 09:12:10 +00:00
|
|
|
if (!bb.valid())
|
2019-06-13 13:37:00 +00:00
|
|
|
continue;
|
2021-09-05 15:43:46 +00:00
|
|
|
osg::Vec3f ofs(
|
2019-06-13 13:37:00 +00:00
|
|
|
entry.mNode->getCenter().x() * cellworldsize, entry.mNode->getCenter().y() * cellworldsize, 0.f);
|
|
|
|
bb._min += ofs;
|
|
|
|
bb._max += ofs;
|
|
|
|
bb._min.z() = highZ;
|
|
|
|
bb._max.z() = highZ;
|
|
|
|
if (cv->isCulled(bb))
|
|
|
|
continue;
|
2020-04-23 09:12:10 +00:00
|
|
|
lowZ = bb._min.z();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2020-04-23 09:12:10 +00:00
|
|
|
if (!debug)
|
2022-09-22 18:26:05 +00:00
|
|
|
break;
|
2020-04-23 09:12:10 +00:00
|
|
|
osg::Box* b = new osg::Box;
|
|
|
|
b->set(bb.center(), bb._max - bb.center());
|
|
|
|
osg::ShapeDrawable* drw = new osg::ShapeDrawable(b);
|
|
|
|
static osg::ref_ptr<osg::StateSet> stateset = nullptr;
|
|
|
|
if (!stateset)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2020-04-23 09:12:10 +00:00
|
|
|
stateset = new osg::StateSet;
|
|
|
|
stateset->setMode(GL_CULL_FACE, osg::StateAttribute::OFF);
|
|
|
|
stateset->setMode(GL_DEPTH_TEST, osg::StateAttribute::OFF);
|
|
|
|
stateset->setAttributeAndModes(
|
|
|
|
new osg::PolygonMode(osg::PolygonMode::FRONT_AND_BACK, osg::PolygonMode::LINE),
|
|
|
|
osg::StateAttribute::ON);
|
|
|
|
osg::Material* m = new osg::Material;
|
|
|
|
m->setEmission(osg::Material::FRONT_AND_BACK, osg::Vec4f(0, 0, 1, 1));
|
|
|
|
m->setDiffuse(osg::Material::FRONT_AND_BACK, osg::Vec4f(0, 0, 0, 1));
|
|
|
|
m->setAmbient(osg::Material::FRONT_AND_BACK, osg::Vec4f(0, 0, 0, 1));
|
|
|
|
stateset->setAttributeAndModes(m, osg::StateAttribute::ON);
|
|
|
|
stateset->setRenderBinDetails(100, "RenderBin");
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2020-04-23 09:12:10 +00:00
|
|
|
drw->setStateSet(stateset);
|
|
|
|
drw->accept(*cv);
|
|
|
|
}
|
|
|
|
callback->setLowZ(lowZ);
|
|
|
|
cv->popCurrentMask();
|
2019-06-13 13:37:00 +00:00
|
|
|
}
|
|
|
|
|
2017-03-08 00:53:13 +00:00
|
|
|
void QuadTreeWorld::accept(osg::NodeVisitor& nv)
|
2017-03-08 23:01:13 +00:00
|
|
|
{
|
2020-01-12 07:42:47 +00:00
|
|
|
bool isCullVisitor = nv.getVisitorType() == osg::NodeVisitor::CULL_VISITOR;
|
2021-10-04 20:00:31 +00:00
|
|
|
if (!isCullVisitor && nv.getVisitorType() != osg::NodeVisitor::INTERSECTION_VISITOR)
|
|
|
|
return;
|
2019-02-20 13:37:00 +00:00
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
osg::Object* viewer = isCullVisitor ? static_cast<osgUtil::CullVisitor*>(&nv)->getCurrentCamera() : nullptr;
|
|
|
|
bool needsUpdate = true;
|
|
|
|
osg::Vec3f viewPoint = viewer ? nv.getViewPoint() : nv.getEyePoint();
|
|
|
|
ViewData* vd = mViewDataMap->getViewData(viewer, viewPoint, mActiveGrid, needsUpdate);
|
2019-02-20 13:37:00 +00:00
|
|
|
if (needsUpdate)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2019-02-20 13:37:00 +00:00
|
|
|
vd->reset();
|
2023-05-19 08:19:16 +00:00
|
|
|
DefaultLodCallback lodCallback(
|
|
|
|
mLodFactor, mMinSize, mViewDistance, mActiveGrid, ESM::getCellSize(mWorldspace));
|
2019-06-13 13:37:00 +00:00
|
|
|
mRootNode->traverseNodes(vd, viewPoint, &lodCallback);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-06-13 13:37:00 +00:00
|
|
|
|
2023-05-18 10:26:25 +00:00
|
|
|
const float cellWorldSize = ESM::getCellSize(mWorldspace);
|
2017-03-08 21:19:48 +00:00
|
|
|
|
2020-01-12 07:42:47 +00:00
|
|
|
for (unsigned int i = 0; i < vd->getNumEntries(); ++i)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2021-09-27 19:25:39 +00:00
|
|
|
ViewDataEntry& entry = vd->getEntry(i);
|
|
|
|
loadRenderingNode(entry, vd, cellWorldSize, mActiveGrid, false);
|
|
|
|
entry.mRenderingNode->accept(nv);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-06-13 13:37:00 +00:00
|
|
|
|
2020-01-12 07:42:47 +00:00
|
|
|
if (mHeightCullCallback && isCullVisitor)
|
2021-10-31 11:59:34 +00:00
|
|
|
updateWaterCullingView(mHeightCullCallback, vd, static_cast<osgUtil::CullVisitor*>(&nv),
|
2023-05-18 09:56:45 +00:00
|
|
|
mStorage->getCellWorldSize(mWorldspace), !isGridEmpty());
|
2019-03-21 13:50:49 +00:00
|
|
|
|
2022-10-04 23:05:09 +00:00
|
|
|
vd->resetChanged();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2019-03-21 13:50:49 +00:00
|
|
|
double referenceTime = nv.getFrameStamp() ? nv.getFrameStamp()->getReferenceTime() : 0.0;
|
|
|
|
if (referenceTime != 0.0)
|
|
|
|
{
|
|
|
|
vd->setLastUsageTimeStamp(referenceTime);
|
|
|
|
mViewDataMap->clearUnusedViews(referenceTime);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-03-21 13:50:49 +00:00
|
|
|
}
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2017-03-09 02:31:30 +00:00
|
|
|
void QuadTreeWorld::ensureQuadTreeBuilt()
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2020-06-25 19:46:07 +00:00
|
|
|
std::lock_guard<std::mutex> lock(mQuadTreeMutex);
|
2017-03-09 03:01:27 +00:00
|
|
|
if (mQuadTreeBuilt)
|
|
|
|
return;
|
2017-03-09 02:31:30 +00:00
|
|
|
|
2023-05-14 18:00:02 +00:00
|
|
|
QuadTreeBuilder builder(mStorage, mMinSize, mWorldspace);
|
2017-03-09 03:01:27 +00:00
|
|
|
builder.build();
|
|
|
|
|
|
|
|
mRootNode = builder.getRootNode();
|
|
|
|
mRootNode->setWorld(this);
|
|
|
|
mQuadTreeBuilt = true;
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
|
|
|
|
2017-03-09 02:31:30 +00:00
|
|
|
void QuadTreeWorld::enable(bool enabled)
|
2017-03-08 00:53:13 +00:00
|
|
|
{
|
2017-03-09 02:31:30 +00:00
|
|
|
if (enabled)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
2017-03-09 02:31:30 +00:00
|
|
|
ensureQuadTreeBuilt();
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2017-03-09 02:31:30 +00:00
|
|
|
if (!mRootNode->getNumParents())
|
|
|
|
mTerrainRoot->addChild(mRootNode);
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2021-10-16 19:48:13 +00:00
|
|
|
else if (mRootNode)
|
|
|
|
mTerrainRoot->removeChild(mRootNode);
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|
2017-03-09 03:17:25 +00:00
|
|
|
|
2021-09-05 15:43:46 +00:00
|
|
|
View* QuadTreeWorld::createView()
|
2017-03-09 03:17:25 +00:00
|
|
|
{
|
|
|
|
return mViewDataMap->createIndependentView();
|
2022-09-22 18:26:05 +00:00
|
|
|
}
|
2019-06-13 13:37:00 +00:00
|
|
|
|
2021-09-27 19:32:18 +00:00
|
|
|
void QuadTreeWorld::preload(View* view, const osg::Vec3f& viewPoint, const osg::Vec4i& grid,
|
|
|
|
std::atomic<bool>& abort, Loading::Reporter& reporter)
|
2017-03-09 03:17:25 +00:00
|
|
|
{
|
2021-09-27 19:32:18 +00:00
|
|
|
ensureQuadTreeBuilt();
|
2023-05-18 09:56:45 +00:00
|
|
|
const float cellWorldSize = mStorage->getCellWorldSize(mWorldspace);
|
2021-09-09 21:10:22 +00:00
|
|
|
|
2021-09-27 19:32:18 +00:00
|
|
|
ViewData* vd = static_cast<ViewData*>(view);
|
|
|
|
vd->setViewPoint(viewPoint);
|
|
|
|
vd->setActiveGrid(grid);
|
2021-09-09 21:10:22 +00:00
|
|
|
|
2023-08-27 14:03:47 +00:00
|
|
|
DefaultLodCallback lodCallback(mLodFactor, mMinSize, mViewDistance, grid, cellWorldSize);
|
|
|
|
mRootNode->traverseNodes(vd, viewPoint, &lodCallback);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2023-08-27 14:03:47 +00:00
|
|
|
reporter.addTotal(vd->getNumEntries());
|
2021-09-09 21:10:22 +00:00
|
|
|
|
2023-08-27 14:03:47 +00:00
|
|
|
for (unsigned int i = 0, n = vd->getNumEntries(); i < n && !abort; ++i)
|
|
|
|
{
|
|
|
|
ViewDataEntry& entry = vd->getEntry(i);
|
|
|
|
loadRenderingNode(entry, vd, cellWorldSize, grid, true);
|
|
|
|
reporter.addProgress(1);
|
2021-09-27 19:32:18 +00:00
|
|
|
}
|
2017-03-09 03:17:25 +00:00
|
|
|
}
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2017-03-09 18:52:30 +00:00
|
|
|
void QuadTreeWorld::reportStats(unsigned int frameNumber, osg::Stats* stats)
|
|
|
|
{
|
2020-01-12 07:42:47 +00:00
|
|
|
if (mCompositeMapRenderer)
|
|
|
|
stats->setAttribute(frameNumber, "Composite", mCompositeMapRenderer->getCompileSetSize());
|
2017-03-09 18:52:30 +00:00
|
|
|
}
|
|
|
|
|
2019-02-28 18:19:48 +00:00
|
|
|
void QuadTreeWorld::loadCell(int x, int y)
|
|
|
|
{
|
|
|
|
// fallback behavior only for undefined cells (every other is already handled in quadtree)
|
|
|
|
float dummy;
|
2023-05-14 18:00:02 +00:00
|
|
|
if (mChunkManager && !mStorage->getMinMaxHeights(1, osg::Vec2f(x + 0.5, y + 0.5), mWorldspace, dummy, dummy))
|
2019-02-28 18:19:48 +00:00
|
|
|
TerrainGrid::loadCell(x, y);
|
|
|
|
else
|
|
|
|
World::loadCell(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
void QuadTreeWorld::unloadCell(int x, int y)
|
|
|
|
{
|
|
|
|
// fallback behavior only for undefined cells (every other is already handled in quadtree)
|
|
|
|
float dummy;
|
2023-05-14 18:00:02 +00:00
|
|
|
if (mChunkManager && !mStorage->getMinMaxHeights(1, osg::Vec2f(x + 0.5, y + 0.5), mWorldspace, dummy, dummy))
|
2019-02-28 18:19:48 +00:00
|
|
|
TerrainGrid::unloadCell(x, y);
|
|
|
|
else
|
|
|
|
World::unloadCell(x, y);
|
|
|
|
}
|
|
|
|
|
2019-06-13 13:37:00 +00:00
|
|
|
void QuadTreeWorld::addChunkManager(QuadTreeWorld::ChunkManager* m)
|
|
|
|
{
|
|
|
|
mChunkManagers.push_back(m);
|
|
|
|
mTerrainRoot->setNodeMask(mTerrainRoot->getNodeMask() | m->getNodeMask());
|
2021-10-31 11:59:34 +00:00
|
|
|
if (m->getViewDistance())
|
2023-05-19 08:19:16 +00:00
|
|
|
m->setMaxLodLevel(
|
|
|
|
DefaultLodCallback::convertDistanceToLodLevel(m->getViewDistance() + mViewDataMap->getReuseDistance(),
|
|
|
|
mMinSize, ESM::getCellSize(mWorldspace), mLodFactor));
|
2019-06-13 13:37:00 +00:00
|
|
|
}
|
2017-03-08 00:53:13 +00:00
|
|
|
|
2020-05-08 13:37:00 +00:00
|
|
|
void QuadTreeWorld::rebuildViews()
|
2020-05-08 13:37:00 +00:00
|
|
|
{
|
2020-05-08 13:37:00 +00:00
|
|
|
mViewDataMap->rebuildViews();
|
2020-05-08 13:37:00 +00:00
|
|
|
}
|
|
|
|
|
2021-10-31 11:59:34 +00:00
|
|
|
void QuadTreeWorld::setViewDistance(float viewDistance)
|
|
|
|
{
|
|
|
|
if (mViewDistance == viewDistance)
|
|
|
|
return;
|
|
|
|
mViewDistance = viewDistance;
|
|
|
|
mViewDataMap->rebuildViews();
|
|
|
|
}
|
|
|
|
|
2017-03-08 00:53:13 +00:00
|
|
|
}
|