#include "terrain.hpp" #include #include #include #include #include #include #include "storage.hpp" #include "quadtreenode.hpp" namespace { bool isPowerOfTwo(int x) { return ( (x > 0) && ((x & (x - 1)) == 0) ); } int nextPowerOfTwo (int v) { if (isPowerOfTwo(v)) return v; int depth=0; while(v) { v >>= 1; depth++; } return 1 << depth; } Terrain::QuadTreeNode* findNode (const Ogre::Vector2& center, Terrain::QuadTreeNode* node) { if (center == node->getCenter()) return node; if (center.x > node->getCenter().x && center.y > node->getCenter().y) return findNode(center, node->getChild(Terrain::NE)); else if (center.x > node->getCenter().x && center.y < node->getCenter().y) return findNode(center, node->getChild(Terrain::SE)); else if (center.x < node->getCenter().x && center.y > node->getCenter().y) return findNode(center, node->getChild(Terrain::NW)); else //if (center.x < node->getCenter().x && center.y < node->getCenter().y) return findNode(center, node->getChild(Terrain::SW)); } } namespace Terrain { Terrain::Terrain(Ogre::SceneManager* sceneMgr, Storage* storage, int visibilityFlags, bool distantLand, bool shaders) : mStorage(storage) , mMinBatchSize(1) , mMaxBatchSize(64) , mSceneMgr(sceneMgr) , mVisibilityFlags(visibilityFlags) , mDistantLand(distantLand) , mShaders(shaders) , mVisible(true) { mCompositeMapSceneMgr = Ogre::Root::getSingleton().createSceneManager(Ogre::ST_GENERIC); Ogre::Camera* compositeMapCam = mCompositeMapSceneMgr->createCamera("a"); mCompositeMapRenderTexture = Ogre::TextureManager::getSingleton().createManual( "terrain/comp/rt", Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, 128, 128, 0, Ogre::PF_A8B8G8R8, Ogre::TU_RENDERTARGET); mCompositeMapRenderTarget = mCompositeMapRenderTexture->getBuffer()->getRenderTarget(); mCompositeMapRenderTarget->setAutoUpdated(false); mCompositeMapRenderTarget->addViewport(compositeMapCam); mBounds = storage->getBounds(); int origSizeX = mBounds.getSize().x; int origSizeY = mBounds.getSize().y; // Dividing a quad tree only works well for powers of two, so round up to the nearest one int size = nextPowerOfTwo(std::max(origSizeX, origSizeY)); // Adjust the center according to the new size Ogre::Vector3 center = mBounds.getCenter() + Ogre::Vector3((size-origSizeX)/2.f, (size-origSizeY)/2.f, 0); mRootSceneNode = mSceneMgr->getRootSceneNode()->createChildSceneNode(); mRootNode = new QuadTreeNode(this, Root, size, Ogre::Vector2(center.x, center.y), NULL); buildQuadTree(mRootNode); mRootNode->initAabb(); mRootNode->initNeighbours(); } Terrain::~Terrain() { delete mRootNode; delete mStorage; } void Terrain::buildQuadTree(QuadTreeNode *node) { float halfSize = node->getSize()/2.f; if (node->getSize() <= mMinBatchSize) { // We arrived at a leaf float minZ,maxZ; Ogre::Vector2 center = node->getCenter(); if (mStorage->getMinMaxHeights(node->getSize(), center, minZ, maxZ)) node->setBoundingBox(Ogre::AxisAlignedBox(Ogre::Vector3(-halfSize*8192, -halfSize*8192, minZ), Ogre::Vector3(halfSize*8192, halfSize*8192, maxZ))); else node->markAsDummy(); // no data available for this node, skip it return; } if (node->getCenter().x - halfSize > mBounds.getMaximum().x || node->getCenter().x + halfSize < mBounds.getMinimum().x || node->getCenter().y - halfSize > mBounds.getMaximum().y || node->getCenter().y + halfSize < mBounds.getMinimum().y ) // Out of bounds of the actual terrain - this will happen because // we rounded the size up to the next power of two { node->markAsDummy(); return; } // Not a leaf, create its children node->createChild(SW, halfSize, node->getCenter() - halfSize/2.f); node->createChild(SE, halfSize, node->getCenter() + Ogre::Vector2(halfSize/2.f, -halfSize/2.f)); node->createChild(NW, halfSize, node->getCenter() + Ogre::Vector2(-halfSize/2.f, halfSize/2.f)); node->createChild(NE, halfSize, node->getCenter() + halfSize/2.f); buildQuadTree(node->getChild(SW)); buildQuadTree(node->getChild(SE)); buildQuadTree(node->getChild(NW)); buildQuadTree(node->getChild(NE)); // if all children are dummy, we are also dummy for (int i=0; i<4; ++i) { if (!node->getChild((ChildDirection)i)->isDummy()) return; } node->markAsDummy(); } void Terrain::update(const Ogre::Vector3& cameraPos) { if (!mVisible) return; mRootNode->update(cameraPos); mRootNode->updateIndexBuffers(); } Ogre::AxisAlignedBox Terrain::getWorldBoundingBox (const Ogre::Vector2& center) { if (center.x > mBounds.getMaximum().x || center.x < mBounds.getMinimum().x || center.y > mBounds.getMaximum().y || center.y < mBounds.getMinimum().y) return Ogre::AxisAlignedBox::BOX_NULL; QuadTreeNode* node = findNode(center, mRootNode); Ogre::AxisAlignedBox box = node->getBoundingBox(); box.setExtents(box.getMinimum() + Ogre::Vector3(center.x, center.y, 0) * 8192, box.getMaximum() + Ogre::Vector3(center.x, center.y, 0) * 8192); return box; } Ogre::HardwareVertexBufferSharedPtr Terrain::getVertexBuffer(int numVertsOneSide) { if (mUvBufferMap.find(numVertsOneSide) != mUvBufferMap.end()) { return mUvBufferMap[numVertsOneSide]; } int vertexCount = numVertsOneSide * numVertsOneSide; std::vector uvs; uvs.reserve(vertexCount*2); for (int col = 0; col < numVertsOneSide; ++col) { for (int row = 0; row < numVertsOneSide; ++row) { uvs.push_back(col / static_cast(numVertsOneSide-1)); // U uvs.push_back(row / static_cast(numVertsOneSide-1)); // V } } Ogre::HardwareBufferManager* mgr = Ogre::HardwareBufferManager::getSingletonPtr(); Ogre::HardwareVertexBufferSharedPtr buffer = mgr->createVertexBuffer( Ogre::VertexElement::getTypeSize(Ogre::VET_FLOAT2), vertexCount, Ogre::HardwareBuffer::HBU_STATIC); buffer->writeData(0, buffer->getSizeInBytes(), &uvs[0], true); mUvBufferMap[numVertsOneSide] = buffer; return buffer; } Ogre::HardwareIndexBufferSharedPtr Terrain::getIndexBuffer(int flags, size_t& numIndices) { if (mIndexBufferMap.find(flags) != mIndexBufferMap.end()) { numIndices = mIndexBufferMap[flags]->getNumIndexes(); return mIndexBufferMap[flags]; } // LOD level n means every 2^n-th vertex is kept size_t lodLevel = (flags >> (4*4)); size_t lodDeltas[4]; for (int i=0; i<4; ++i) lodDeltas[i] = (flags >> (4*i)) & (0xf); bool anyDeltas = (lodDeltas[North] || lodDeltas[South] || lodDeltas[West] || lodDeltas[East]); size_t increment = std::pow(2, lodLevel); assert((int)increment < ESM::Land::LAND_SIZE); std::vector indices; indices.reserve((ESM::Land::LAND_SIZE-1)*(ESM::Land::LAND_SIZE-1)*2*3 / increment); size_t rowStart = 0, colStart = 0, rowEnd = ESM::Land::LAND_SIZE-1, colEnd = ESM::Land::LAND_SIZE-1; // If any edge needs stitching we'll skip all edges at this point, // mainly because stitching one edge would have an effect on corners and on the adjacent edges if (anyDeltas) { colStart += increment; colEnd -= increment; rowEnd -= increment; rowStart += increment; } for (size_t row = rowStart; row < rowEnd; row += increment) { for (size_t col = colStart; col < colEnd; col += increment) { indices.push_back(ESM::Land::LAND_SIZE*col+row); indices.push_back(ESM::Land::LAND_SIZE*(col+increment)+row+increment); indices.push_back(ESM::Land::LAND_SIZE*col+row+increment); indices.push_back(ESM::Land::LAND_SIZE*col+row); indices.push_back(ESM::Land::LAND_SIZE*(col+increment)+row); indices.push_back(ESM::Land::LAND_SIZE*(col+increment)+row+increment); } } size_t innerStep = increment; if (anyDeltas) { // Now configure LOD transitions at the edges - this is pretty tedious, // and some very long and boring code, but it works great // South size_t row = 0; size_t outerStep = std::pow(2, lodDeltas[South] + lodLevel); for (size_t col = 0; col < ESM::Land::LAND_SIZE-1; col += outerStep) { indices.push_back(ESM::Land::LAND_SIZE*col+row); indices.push_back(ESM::Land::LAND_SIZE*(col+outerStep)+row); // Make sure not to touch the right edge if (col+outerStep == ESM::Land::LAND_SIZE-1) indices.push_back(ESM::Land::LAND_SIZE*(col+outerStep-innerStep)+row+innerStep); else indices.push_back(ESM::Land::LAND_SIZE*(col+outerStep)+row+innerStep); for (size_t i = 0; i < outerStep; i += innerStep) { // Make sure not to touch the left or right edges if (col+i == 0 || col+i == ESM::Land::LAND_SIZE-1-innerStep) continue; indices.push_back(ESM::Land::LAND_SIZE*(col)+row); indices.push_back(ESM::Land::LAND_SIZE*(col+i+innerStep)+row+innerStep); indices.push_back(ESM::Land::LAND_SIZE*(col+i)+row+innerStep); } } // North row = ESM::Land::LAND_SIZE-1; outerStep = std::pow(2, lodDeltas[North] + lodLevel); for (size_t col = 0; col < ESM::Land::LAND_SIZE-1; col += outerStep) { indices.push_back(ESM::Land::LAND_SIZE*(col+outerStep)+row); indices.push_back(ESM::Land::LAND_SIZE*col+row); // Make sure not to touch the left edge if (col == 0) indices.push_back(ESM::Land::LAND_SIZE*(col+innerStep)+row-innerStep); else indices.push_back(ESM::Land::LAND_SIZE*col+row-innerStep); for (size_t i = 0; i < outerStep; i += innerStep) { // Make sure not to touch the left or right edges if (col+i == 0 || col+i == ESM::Land::LAND_SIZE-1-innerStep) continue; indices.push_back(ESM::Land::LAND_SIZE*(col+i)+row-innerStep); indices.push_back(ESM::Land::LAND_SIZE*(col+i+innerStep)+row-innerStep); indices.push_back(ESM::Land::LAND_SIZE*(col+outerStep)+row); } } // West size_t col = 0; outerStep = std::pow(2, lodDeltas[West] + lodLevel); for (size_t row = 0; row < ESM::Land::LAND_SIZE-1; row += outerStep) { indices.push_back(ESM::Land::LAND_SIZE*col+row+outerStep); indices.push_back(ESM::Land::LAND_SIZE*col+row); // Make sure not to touch the top edge if (row+outerStep == ESM::Land::LAND_SIZE-1) indices.push_back(ESM::Land::LAND_SIZE*(col+innerStep)+row+outerStep-innerStep); else indices.push_back(ESM::Land::LAND_SIZE*(col+innerStep)+row+outerStep); for (size_t i = 0; i < outerStep; i += innerStep) { // Make sure not to touch the top or bottom edges if (row+i == 0 || row+i == ESM::Land::LAND_SIZE-1-innerStep) continue; indices.push_back(ESM::Land::LAND_SIZE*col+row); indices.push_back(ESM::Land::LAND_SIZE*(col+innerStep)+row+i); indices.push_back(ESM::Land::LAND_SIZE*(col+innerStep)+row+i+innerStep); } } // East col = ESM::Land::LAND_SIZE-1; outerStep = std::pow(2, lodDeltas[East] + lodLevel); for (size_t row = 0; row < ESM::Land::LAND_SIZE-1; row += outerStep) { indices.push_back(ESM::Land::LAND_SIZE*col+row); indices.push_back(ESM::Land::LAND_SIZE*col+row+outerStep); // Make sure not to touch the bottom edge if (row == 0) indices.push_back(ESM::Land::LAND_SIZE*(col-innerStep)+row+innerStep); else indices.push_back(ESM::Land::LAND_SIZE*(col-innerStep)+row); for (size_t i = 0; i < outerStep; i += innerStep) { // Make sure not to touch the top or bottom edges if (row+i == 0 || row+i == ESM::Land::LAND_SIZE-1-innerStep) continue; indices.push_back(ESM::Land::LAND_SIZE*col+row+outerStep); indices.push_back(ESM::Land::LAND_SIZE*(col-innerStep)+row+i+innerStep); indices.push_back(ESM::Land::LAND_SIZE*(col-innerStep)+row+i); } } } numIndices = indices.size(); Ogre::HardwareBufferManager* mgr = Ogre::HardwareBufferManager::getSingletonPtr(); Ogre::HardwareIndexBufferSharedPtr buffer = mgr->createIndexBuffer(Ogre::HardwareIndexBuffer::IT_16BIT, numIndices, Ogre::HardwareBuffer::HBU_STATIC); buffer->writeData(0, buffer->getSizeInBytes(), &indices[0], true); mIndexBufferMap[flags] = buffer; return buffer; } void Terrain::renderCompositeMap(Ogre::TexturePtr target) { mCompositeMapRenderTarget->update(); target->getBuffer()->blit(mCompositeMapRenderTexture->getBuffer()); } void Terrain::clearCompositeMapSceneManager() { mCompositeMapSceneMgr->destroyAllManualObjects(); mCompositeMapSceneMgr->clearScene(); } float Terrain::getHeightAt(const Ogre::Vector3 &worldPos) { return mStorage->getHeightAt(worldPos); } void Terrain::applyMaterials(bool shadows, bool splitShadows) { mShadows = shadows; mSplitShadows = splitShadows; mRootNode->applyMaterials(); } void Terrain::setVisible(bool visible) { if (visible && !mVisible) mSceneMgr->getRootSceneNode()->addChild(mRootSceneNode); else if (!visible && mVisible) mSceneMgr->getRootSceneNode()->removeChild(mRootSceneNode); mVisible = visible; } bool Terrain::getVisible() { return mVisible; } }