#ifndef COMPONENTS_TERRAIN_QUADTREENODE_H #define COMPONENTS_TERRAIN_QUADTREENODE_H #include #include #include namespace Ogre { class Rectangle2D; } namespace Terrain { class Terrain; class Chunk; class MaterialGenerator; enum Direction { North = 0, East = 1, South = 2, West = 3 }; enum ChildDirection { NW = 0, NE = 1, SW = 2, SE = 3, Root }; /** * @brief A node in the quad tree for our terrain. Depending on LOD, * a node can either choose to render itself in one batch (merging its children), * or delegate the render process to its children, rendering each child in at least one batch. */ class QuadTreeNode { public: /// @param terrain /// @param dir relative to parent, or Root if we are the root node /// @param size size (in *cell* units!) /// @param center center (in *cell* units!) /// @param parent parent node QuadTreeNode (Terrain* terrain, ChildDirection dir, float size, const Ogre::Vector2& center, QuadTreeNode* parent); ~QuadTreeNode(); /// @note takes ownership of \a child void createChild (ChildDirection id, float size, const Ogre::Vector2& center); /// Mark this node as a dummy node. This can happen if the terrain size isn't a power of two. /// For the QuadTree to work, we need to round the size up to a power of two, which means we'll /// end up with empty nodes that don't actually render anything. void markAsDummy() { mIsDummy = true; } bool isDummy() { return mIsDummy; } QuadTreeNode* getParent() { return mParent; } int getSize() { return mSize; } Ogre::Vector2 getCenter() { return mCenter; } bool hasChildren() { return mChildren[0] != 0; } QuadTreeNode* getChild(ChildDirection dir) { return mChildren[dir]; } /// Search for a neighbour node in this direction QuadTreeNode* searchNeighbour (Direction dir); /// Returns our direction relative to the parent node, or Root if we are the root node. ChildDirection getDirection() { return mDirection; } /// Set bounding box in local coordinates. Should be done at load time for leaf nodes. /// Other nodes can merge AABB of child nodes. void setBoundingBox (const Ogre::AxisAlignedBox& box) { mBounds = box; } /// Get bounding box in local coordinates const Ogre::AxisAlignedBox& getBoundingBox(); Terrain* getTerrain() { return mTerrain; } /// Adjust LODs for the given camera position, possibly splitting up chunks or merging them. void update (const Ogre::Vector3& cameraPos); /// Adjust index buffers of chunks to stitch together chunks of different LOD, so that cracks are avoided. /// Call after QuadTreeNode::update! void updateIndexBuffers(); /// Remove chunks rendered by this node and all its children void removeChunks(); /// Get the effective LOD level if this node was rendered in one chunk /// with ESM::Land::LAND_SIZE^2 vertices size_t getNativeLodLevel() { return mLodLevel; } /// Get the effective current LOD level used by the chunk rendering this node size_t getActualLodLevel(); /// Is this node currently configured to render itself? bool hasChunk(); bool hasCompositeMap(); /// Add a textured quad to a specific 2d area in the composite map scenemanager. /// Only nodes with size <= 1 can be rendered with alpha blending, so larger nodes will simply /// call this method on their children. /// @param area area in image space to put the quad /// @param quads collect quads here so they can be deleted later void prepareForCompositeMap(Ogre::TRect area); private: // Stored here for convenience in case we need layer list again MaterialGenerator* mMaterialGenerator; bool mIsDummy; float mSize; size_t mLodLevel; // LOD if we were to render this node in one chunk Ogre::AxisAlignedBox mBounds; ChildDirection mDirection; Ogre::Vector2 mCenter; Ogre::SceneNode* mSceneNode; QuadTreeNode* mParent; QuadTreeNode* mChildren[4]; Chunk* mChunk; Terrain* mTerrain; Ogre::TexturePtr mCompositeMap; void ensureLayerInfo(); void ensureCompositeMap(); }; } #endif