1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-24 23:23:54 +00:00
openmw/components/terrain/viewdata.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

234 lines
6.4 KiB
C++
Raw Normal View History

#include "viewdata.hpp"
#include "quadtreenode.hpp"
#include <algorithm>
namespace Terrain
{
ViewData::ViewData()
: mNumEntries(0)
, mLastUsageTimeStamp(0.0)
, mChanged(false)
, mHasViewPoint(false)
, mWorldUpdateRevision(0)
{
}
ViewData::~ViewData() {}
2017-03-08 23:44:21 +00:00
void ViewData::copyFrom(const ViewData& other)
2022-09-22 18:26:05 +00:00
{
2017-03-08 23:44:21 +00:00
mNumEntries = other.mNumEntries;
mEntries = other.mEntries;
mChanged = other.mChanged;
2017-03-08 23:44:21 +00:00
mHasViewPoint = other.mHasViewPoint;
mViewPoint = other.mViewPoint;
mActiveGrid = other.mActiveGrid;
2017-03-08 23:44:21 +00:00
mWorldUpdateRevision = other.mWorldUpdateRevision;
mNodes = other.mNodes;
2022-09-22 18:26:05 +00:00
}
void ViewData::add(QuadTreeNode* node)
2022-09-22 18:26:05 +00:00
{
unsigned int index = mNumEntries++;
if (index + 1 > mEntries.size())
mEntries.resize(index + 1);
ViewDataEntry& entry = mEntries[index];
if (entry.set(node))
{
mChanged = true;
mNodes.clear();
}
2022-09-22 18:26:05 +00:00
}
void ViewData::setViewPoint(const osg::Vec3f& viewPoint)
2022-09-22 18:26:05 +00:00
{
mViewPoint = viewPoint;
2017-03-09 03:02:58 +00:00
mHasViewPoint = true;
}
// NOTE: As a performance optimisation, we cache mRenderingNodes from previous frames here.
// If this cache becomes invalid (e.g. through mWorldUpdateRevision), we need to use clear() instead of reset().
void ViewData::reset()
{
// clear any unused entries
for (unsigned int i = mNumEntries; i < mEntries.size(); ++i)
mEntries[i].set(nullptr);
// reset index for next frame
mNumEntries = 0;
mChanged = false;
mNodes.clear();
}
void ViewData::clear()
2017-03-08 23:45:31 +00:00
{
for (unsigned int i = 0; i < mEntries.size(); ++i)
mEntries[i].set(nullptr);
mNumEntries = 0;
mLastUsageTimeStamp = 0;
mChanged = false;
mHasViewPoint = false;
mNodes.clear();
2017-03-08 23:45:31 +00:00
}
bool ViewData::suitableToUse(const osg::Vec4i& activeGrid) const
2022-09-22 18:26:05 +00:00
{
return hasViewPoint() && activeGrid == mActiveGrid && getNumEntries();
}
bool ViewData::contains(const QuadTreeNode* node) const
{
return std::binary_search(mNodes.begin(), mNodes.end(), node);
}
void ViewData::buildNodeIndex()
{
if (!mNodes.empty())
return;
mNodes.reserve(mNumEntries);
for (std::size_t i = 0; i < mNumEntries; ++i)
if (const QuadTreeNode* node = mEntries[i].mNode)
mNodes.push_back(node);
std::sort(mNodes.begin(), mNodes.end());
}
void ViewData::removeNodeFromIndex(const QuadTreeNode* node)
{
const auto it = std::lower_bound(mNodes.begin(), mNodes.end(), node);
if (it == mNodes.end() || *it != node)
return;
mNodes.erase(it);
2022-09-22 18:26:05 +00:00
}
ViewDataEntry::ViewDataEntry()
2018-10-09 06:21:12 +00:00
: mNode(nullptr)
, mLodFlags(0)
{
}
bool ViewDataEntry::set(QuadTreeNode* node)
{
if (node == mNode)
return false;
2022-09-22 18:26:05 +00:00
else
{
mNode = node;
// clear cached data
mRenderingNode = nullptr;
return true;
2022-09-22 18:26:05 +00:00
}
}
ViewData* ViewDataMap::getViewData(
osg::Object* viewer, const osg::Vec3f& viewPoint, const osg::Vec4i& activeGrid, bool& needsUpdate)
{
ViewerMap::const_iterator found = mViewers.find(viewer);
ViewData* vd = nullptr;
if (found == mViewers.end())
{
vd = createOrReuseView();
mViewers[viewer] = vd;
2022-09-22 18:26:05 +00:00
}
else
vd = found->second;
needsUpdate = false;
2022-09-22 18:26:05 +00:00
if (!(vd->suitableToUse(activeGrid)
&& (vd->getViewPoint() - viewPoint).length2() < mReuseDistance * mReuseDistance
&& vd->getWorldUpdateRevision() >= mWorldUpdateRevision))
2022-09-22 18:26:05 +00:00
{
float shortestDist = viewer ? mReuseDistance * mReuseDistance : std::numeric_limits<float>::max();
const ViewData* mostSuitableView = nullptr;
for (const ViewData* other : mUsedViews)
{
if (other->suitableToUse(activeGrid) && other->getWorldUpdateRevision() >= mWorldUpdateRevision)
{
float dist = (viewPoint - other->getViewPoint()).length2();
if (dist < shortestDist)
2022-09-22 18:26:05 +00:00
{
shortestDist = dist;
mostSuitableView = other;
2022-09-22 18:26:05 +00:00
}
}
}
if (mostSuitableView && mostSuitableView != vd)
{
vd->copyFrom(*mostSuitableView);
return vd;
}
else if (!mostSuitableView)
{
if (vd->getWorldUpdateRevision() != mWorldUpdateRevision)
2022-09-22 18:26:05 +00:00
{
vd->setWorldUpdateRevision(mWorldUpdateRevision);
vd->clear();
2022-09-22 18:26:05 +00:00
}
vd->setViewPoint(viewPoint);
vd->setActiveGrid(activeGrid);
needsUpdate = true;
}
}
return vd;
}
ViewData* ViewDataMap::createOrReuseView()
{
ViewData* vd = nullptr;
if (mUnusedViews.size())
2022-09-22 18:26:05 +00:00
{
vd = mUnusedViews.front();
mUnusedViews.pop_front();
}
else
{
2020-10-17 08:26:35 +00:00
mViewVector.emplace_back();
vd = &mViewVector.back();
2022-09-22 18:26:05 +00:00
}
mUsedViews.push_back(vd);
vd->setWorldUpdateRevision(mWorldUpdateRevision);
return vd;
}
ViewData* ViewDataMap::createIndependentView() const
{
ViewData* vd = new ViewData;
vd->setWorldUpdateRevision(mWorldUpdateRevision);
return vd;
}
2022-09-22 18:26:05 +00:00
void ViewDataMap::clearUnusedViews(double referenceTime)
{
for (ViewerMap::iterator it = mViewers.begin(); it != mViewers.end();)
{
if (it->second->getLastUsageTimeStamp() + mExpiryDelay < referenceTime)
mViewers.erase(it++);
2022-09-22 18:26:05 +00:00
else
++it;
}
for (std::deque<ViewData*>::iterator it = mUsedViews.begin(); it != mUsedViews.end();)
2022-09-22 18:26:05 +00:00
{
if ((*it)->getLastUsageTimeStamp() + mExpiryDelay < referenceTime)
2022-09-22 18:26:05 +00:00
{
(*it)->clear();
mUnusedViews.push_back(*it);
it = mUsedViews.erase(it);
2022-09-22 18:26:05 +00:00
}
else
++it;
}
}
void ViewDataMap::rebuildViews()
{
++mWorldUpdateRevision;
}
}