mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-27 21:35:24 +00:00
d684f1a78f
Signed-off-by: Bret Curtis <psi29a@gmail.com>
303 lines
12 KiB
C++
303 lines
12 KiB
C++
#include "objectpaging.hpp"
|
|
|
|
#include <osg/Version>
|
|
#include <osg/LOD>
|
|
#include <osg/Switch>
|
|
#include <osg/MatrixTransform>
|
|
#include <osgUtil/IncrementalCompileOperation>
|
|
|
|
#include <components/esm/esmreader.hpp>
|
|
#include <components/misc/resourcehelpers.hpp>
|
|
#include <components/resource/scenemanager.hpp>
|
|
#include <components/sceneutil/optimizer.hpp>
|
|
#include <components/sceneutil/clone.hpp>
|
|
|
|
#include <osgParticle/ParticleProcessor>
|
|
#include <osgParticle/ParticleSystemUpdater>
|
|
#include <osgParticle/Emitter>
|
|
|
|
#include <components/sceneutil/morphgeometry.hpp>
|
|
#include <components/sceneutil/riggeometry.hpp>
|
|
#include <components/settings/settings.hpp>
|
|
|
|
#include "apps/openmw/mwworld/esmstore.hpp"
|
|
#include "apps/openmw/mwbase/environment.hpp"
|
|
#include "apps/openmw/mwbase/world.hpp"
|
|
|
|
#include "vismask.hpp"
|
|
|
|
namespace MWRender
|
|
{
|
|
|
|
bool typeFilter(int type, bool far)
|
|
{
|
|
switch (type)
|
|
{
|
|
case ESM::REC_STAT:
|
|
case ESM::REC_ACTI:
|
|
case ESM::REC_DOOR:
|
|
return true;
|
|
case ESM::REC_CONT:
|
|
return far ? false : true;
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
const std::string& getModel(int type, const std::string& id, const MWWorld::ESMStore& store)
|
|
{
|
|
switch (type)
|
|
{
|
|
case ESM::REC_STAT:
|
|
return store.get<ESM::Static>().searchStatic(id)->mModel;
|
|
case ESM::REC_ACTI:
|
|
return store.get<ESM::Activator>().searchStatic(id)->mModel;
|
|
case ESM::REC_DOOR:
|
|
return store.get<ESM::Door>().searchStatic(id)->mModel;
|
|
case ESM::REC_CONT:
|
|
return store.get<ESM::Container>().searchStatic(id)->mModel;
|
|
default: throw std::exception();
|
|
}
|
|
}
|
|
|
|
osg::ref_ptr<osg::Node> ObjectPaging::getChunk(float size, const osg::Vec2f& center, unsigned char lod, unsigned int lodFlags, bool far, const osg::Vec3f& viewPoint)
|
|
{
|
|
if (!far)return nullptr;
|
|
ChunkId id = std::make_tuple(center, size);
|
|
|
|
osg::ref_ptr<osg::Object> obj = mCache->getRefFromObjectCache(id);
|
|
if (obj)
|
|
return obj->asNode();
|
|
else
|
|
{
|
|
osg::ref_ptr<osg::Node> node = createChunk(size, center, viewPoint);
|
|
mCache->addEntryToObjectCache(id, node.get());
|
|
return node;
|
|
}
|
|
}
|
|
|
|
class CanOptimizeCallback : public SceneUtil::Optimizer::IsOperationPermissibleForObjectCallback
|
|
{
|
|
public:
|
|
virtual bool isOperationPermissibleForObjectImplementation(const SceneUtil::Optimizer* optimizer, const osg::Drawable* node,unsigned int option) const
|
|
{
|
|
return true;
|
|
}
|
|
virtual bool isOperationPermissibleForObjectImplementation(const SceneUtil::Optimizer* optimizer, const osg::Node* node,unsigned int option) const
|
|
{
|
|
return true;
|
|
}
|
|
};
|
|
|
|
class CopyOp : public osg::CopyOp
|
|
{
|
|
public:
|
|
CopyOp() : mDistance(0.f) {
|
|
setCopyFlags(osg::CopyOp::DEEP_COPY_NODES|osg::CopyOp::DEEP_COPY_DRAWABLES
|
|
#if OSG_MIN_VERSION_REQUIRED(3,5,6)
|
|
|osg::CopyOp::DEEP_COPY_ARRAYS|osg::CopyOp::DEEP_COPY_PRIMITIVES // damned vbogarbage racing
|
|
#endif
|
|
);
|
|
}
|
|
|
|
float mDistance;
|
|
|
|
virtual osg::Node* operator() (const osg::Node* node) const
|
|
{
|
|
if (dynamic_cast<const osgParticle::ParticleProcessor*>(node))
|
|
return nullptr;
|
|
if (dynamic_cast<const osgParticle::ParticleSystemUpdater*>(node))
|
|
return nullptr;
|
|
|
|
if (const osg::Switch* sw = node->asSwitch())
|
|
{
|
|
osg::Group* n = new osg::Group;
|
|
for (unsigned int i=0; i<sw->getNumChildren(); ++i)
|
|
if (sw->getValue(i))
|
|
n->addChild(operator()(sw->getChild(i)));
|
|
n->setDataVariance(osg::Object::STATIC);
|
|
return n;
|
|
}
|
|
if (const osg::LOD* lod = dynamic_cast<const osg::LOD*>(node))
|
|
{
|
|
osg::Group* n = new osg::Group;
|
|
for (unsigned int i=0; i<lod->getNumChildren(); ++i)
|
|
if (lod->getMinRange(i) <= mDistance && mDistance < lod->getMaxRange(i))
|
|
n->addChild(operator()(lod->getChild(i)));
|
|
n->setDataVariance(osg::Object::STATIC);
|
|
return n;
|
|
}
|
|
|
|
osg::Node* n = osg::CopyOp::operator()(node);
|
|
if (n) {
|
|
n->setDataVariance(osg::Object::STATIC);
|
|
n->setUserDataContainer(nullptr);
|
|
n->setName("");
|
|
}
|
|
return n;
|
|
}
|
|
virtual osg::Drawable* operator() (const osg::Drawable* drawable) const
|
|
{
|
|
if (dynamic_cast<const osgParticle::ParticleSystem*>(drawable))
|
|
return nullptr;
|
|
|
|
if (const SceneUtil::RigGeometry* rig = dynamic_cast<const SceneUtil::RigGeometry*>(drawable))
|
|
return osg::CopyOp::operator()(rig->getSourceGeometry());
|
|
if (const SceneUtil::MorphGeometry* morph = dynamic_cast<const SceneUtil::MorphGeometry*>(drawable))
|
|
return osg::CopyOp::operator()(morph->getSourceGeometry());
|
|
|
|
return osg::CopyOp::operator()(drawable);
|
|
}
|
|
virtual osg::Callback* operator() (const osg::Callback* callback) const
|
|
{
|
|
return nullptr;
|
|
}
|
|
};
|
|
|
|
ObjectPaging::ObjectPaging(Resource::SceneManager* sceneManager)
|
|
: GenericResourceManager<ChunkId>(nullptr)
|
|
, mSceneManager(sceneManager)
|
|
{
|
|
mMergeGeometry = Settings::Manager::getBool("object paging merge geometry", "Terrain");
|
|
mMinSize = Settings::Manager::getFloat("object paging min size", "Terrain");
|
|
}
|
|
|
|
osg::ref_ptr<osg::Node> ObjectPaging::createChunk(float size, const osg::Vec2f& center, const osg::Vec3f& viewPoint)
|
|
{
|
|
osg::Vec2i startCell = osg::Vec2i(std::floor(center.x() - size/2.f), std::floor(center.y() - size/2.f));
|
|
|
|
osg::ref_ptr<osg::Group> group = new osg::Group;
|
|
|
|
osg::Vec3f worldCenter = osg::Vec3f(center.x(), center.y(), 0)*ESM::Land::REAL_SIZE;
|
|
osg::Vec3f relativeViewPoint = viewPoint - worldCenter;
|
|
|
|
std::vector<ESM::CellRef> refs;
|
|
std::vector<ESM::ESMReader> esm;
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
for (int cellX = startCell.x(); cellX < startCell.x() + size; ++cellX)
|
|
{
|
|
for (int cellY = startCell.y(); cellY < startCell.y() + size; ++cellY)
|
|
{
|
|
const ESM::Cell* cell = store.get<ESM::Cell>().searchStatic(cellX, cellY);
|
|
if (!cell) continue;
|
|
for (size_t i=0; i<cell->mContextList.size(); ++i)
|
|
{
|
|
try
|
|
{
|
|
unsigned int index = cell->mContextList.at(i).index;
|
|
if (esm.size()<=index)
|
|
esm.resize(index+1);
|
|
cell->restore(esm[index], i);
|
|
ESM::CellRef ref;
|
|
ref.mRefNum.mContentFile = ESM::RefNum::RefNum_NoContentFile;
|
|
bool deleted = false;
|
|
while(cell->getNextRef(esm[index], ref, deleted))
|
|
{
|
|
if (std::find(cell->mMovedRefs.begin(), cell->mMovedRefs.end(), ref.mRefNum) != cell->mMovedRefs.end()) continue;
|
|
int type = store.findStatic(Misc::StringUtils::lowerCase(ref.mRefID));
|
|
if (!typeFilter(type,size>=2)) continue;
|
|
if (deleted) continue;
|
|
refs.push_back(ref);
|
|
}
|
|
}
|
|
catch (std::exception& e)
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
for (ESM::CellRefTracker::const_iterator it = cell->mLeasedRefs.begin(); it != cell->mLeasedRefs.end(); ++it)
|
|
{
|
|
ESM::CellRef ref = it->first;
|
|
bool deleted = it->second;
|
|
if (deleted) continue;
|
|
int type = store.findStatic(Misc::StringUtils::lowerCase(ref.mRefID));
|
|
if (!typeFilter(type,size>=2)) continue;
|
|
refs.push_back(ref);
|
|
}
|
|
}
|
|
}
|
|
|
|
osg::Vec2f minBound = (center - osg::Vec2f(size/2.f, size/2.f));
|
|
osg::Vec2f maxBound = (center + osg::Vec2f(size/2.f, size/2.f));
|
|
for (const ESM::CellRef& ref : refs)
|
|
{
|
|
std::string id = Misc::StringUtils::lowerCase(ref.mRefID);
|
|
if (id == "prisonmarker" || id == "divinemarker" || id == "templemarker" || id == "northmarker")
|
|
continue; // marker objects that have a hardcoded function in the game logic, should be hidden from the player
|
|
|
|
int type = store.findStatic(id);
|
|
std::string model = "meshes/" + getModel(type, id, store);
|
|
/*
|
|
bool useAnim = type != ESM::REC_STAT;
|
|
if (useAnim)
|
|
model = Misc::ResourceHelpers::correctActorModelPath(model, mSceneManager->getVFS());
|
|
*/
|
|
if (model.empty()) continue;
|
|
|
|
osg::Vec3f pos = ref.mPos.asVec3();
|
|
if (size < 1.f)
|
|
{
|
|
osg::Vec3f cellPos = pos / ESM::Land::REAL_SIZE;
|
|
cellPos.x() = std::max(cellPos.x(), std::floor(minBound.x()));
|
|
cellPos.x() = std::min(cellPos.x(), std::ceil(maxBound.x()));
|
|
cellPos.y() = std::max(cellPos.y(), std::floor(minBound.y()));
|
|
cellPos.y() = std::min(cellPos.y(), std::ceil(maxBound.y()));
|
|
if (cellPos.x() < minBound.x() || cellPos.x() > maxBound.x() || cellPos.y() < minBound.y() || cellPos.y() > maxBound.y())
|
|
continue;
|
|
}
|
|
|
|
osg::ref_ptr<const osg::Node> cnode = mSceneManager->getTemplate(model, false);
|
|
|
|
float d = (viewPoint - pos).length();
|
|
|
|
if (cnode->getBound().radius() * ref.mScale < d*mMinSize)
|
|
continue;
|
|
|
|
CopyOp co = CopyOp();
|
|
co.mDistance = d;
|
|
osg::ref_ptr<osg::Node> node = osg::clone(cnode.get(), co);
|
|
node->setUserDataContainer(nullptr);
|
|
|
|
osg::Matrixf matrix;
|
|
matrix.preMultTranslate(pos - worldCenter);
|
|
matrix.preMultRotate( osg::Quat(ref.mPos.rot[2], osg::Vec3f(0,0,-1)) *
|
|
osg::Quat(ref.mPos.rot[1], osg::Vec3f(0,-1,0)) *
|
|
osg::Quat(ref.mPos.rot[0], osg::Vec3f(-1,0,0)) );
|
|
matrix.preMultScale(osg::Vec3f(ref.mScale, ref.mScale, ref.mScale));
|
|
osg::ref_ptr<osg::MatrixTransform> trans = new osg::MatrixTransform(matrix);
|
|
trans->addChild(node);
|
|
trans->setDataVariance(osg::Object::STATIC);
|
|
|
|
group->addChild(trans);
|
|
}
|
|
|
|
if (mMergeGeometry)
|
|
{
|
|
SceneUtil::Optimizer optimizer;
|
|
if ((relativeViewPoint - group->getBound().center()).length2() > group->getBound().radius2())
|
|
{
|
|
optimizer.setViewPoint(relativeViewPoint);
|
|
optimizer.setMergeAlphaBlending(true);
|
|
}
|
|
optimizer.setIsOperationPermissibleForObjectCallback(new CanOptimizeCallback);
|
|
unsigned int options = SceneUtil::Optimizer::FLATTEN_STATIC_TRANSFORMS|SceneUtil::Optimizer::REMOVE_REDUNDANT_NODES|SceneUtil::Optimizer::MERGE_GEOMETRY;
|
|
optimizer.optimize(group, options);
|
|
}
|
|
|
|
auto ico = mSceneManager->getIncrementalCompileOperation();
|
|
if (ico) ico->add(group);
|
|
else group->getBound();
|
|
|
|
group->setNodeMask(Mask_Static);
|
|
|
|
return group;
|
|
}
|
|
|
|
unsigned int ObjectPaging::getNodeMask()
|
|
{
|
|
return Mask_Static;
|
|
}
|
|
|
|
}
|