mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-15 18:49:54 +00:00
e197f5318b
conversion from 'const float' to 'int', possible loss of data conversion from 'double' to 'int', possible loss of data conversion from 'float' to 'int', possible loss of data
189 lines
6.8 KiB
C++
189 lines
6.8 KiB
C++
/*
|
|
* Copyright (c) 2015 scrawl <scrawl@baseoftrash.de>
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
#ifndef COMPONENTS_TERRAIN_QUADTREENODE_H
|
|
#define COMPONENTS_TERRAIN_QUADTREENODE_H
|
|
|
|
#include <OgreAxisAlignedBox.h>
|
|
#include <OgreVector2.h>
|
|
#include <OgreTexture.h>
|
|
|
|
#include "defs.hpp"
|
|
|
|
namespace Ogre
|
|
{
|
|
class Rectangle2D;
|
|
}
|
|
|
|
namespace Terrain
|
|
{
|
|
class DefaultWorld;
|
|
class Chunk;
|
|
class MaterialGenerator;
|
|
struct LoadResponseData;
|
|
|
|
enum ChildDirection
|
|
{
|
|
NW = 0,
|
|
NE = 1,
|
|
SW = 2,
|
|
SE = 3,
|
|
Root
|
|
};
|
|
|
|
enum LoadState
|
|
{
|
|
LS_Unloaded,
|
|
LS_Loading,
|
|
LS_Loaded
|
|
};
|
|
|
|
/**
|
|
* @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 (DefaultWorld* terrain, ChildDirection dir, float size, const Ogre::Vector2& center, QuadTreeNode* parent);
|
|
~QuadTreeNode();
|
|
|
|
/// Rebuild all materials
|
|
void applyMaterials();
|
|
|
|
/// Initialize neighbours - do this after the quadtree is built
|
|
void initNeighbours();
|
|
/// Initialize bounding boxes of non-leafs by merging children bounding boxes.
|
|
/// Do this after the quadtree is built - note that leaf bounding boxes
|
|
/// need to be set first via setBoundingBox!
|
|
void initAabb();
|
|
|
|
/// @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; }
|
|
|
|
Ogre::SceneNode* getSceneNode() { return mSceneNode; }
|
|
|
|
int getSize() { return static_cast<int>(mSize); }
|
|
Ogre::Vector2 getCenter() { return mCenter; }
|
|
|
|
bool hasChildren() { return mChildren[0] != 0; }
|
|
QuadTreeNode* getChild(ChildDirection dir) { return mChildren[dir]; }
|
|
|
|
/// Get neighbour node in this direction
|
|
QuadTreeNode* getNeighbour (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);
|
|
|
|
/// Get bounding box in local coordinates
|
|
const Ogre::AxisAlignedBox& getBoundingBox();
|
|
|
|
const Ogre::AxisAlignedBox& getWorldBoundingBox();
|
|
|
|
DefaultWorld* getTerrain() { return mTerrain; }
|
|
|
|
/// Adjust LODs for the given camera position, possibly splitting up chunks or merging them.
|
|
/// @return Did we (or all of our children) choose to render?
|
|
bool 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();
|
|
|
|
/// Destroy chunks rendered by this node *and* its children (if param is true)
|
|
void destroyChunks(bool children);
|
|
|
|
/// Get the effective LOD level if this node was rendered in one chunk
|
|
/// with Storage::getCellVertices^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();
|
|
|
|
/// 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.
|
|
/// @note Do not call this before World::areLayersLoaded() == true
|
|
/// @param area area in image space to put the quad
|
|
void prepareForCompositeMap(Ogre::TRect<float> area);
|
|
|
|
/// Create a chunk for this node from the given data.
|
|
void load (const LoadResponseData& data);
|
|
void unload(bool recursive=false);
|
|
void loadLayers (const LayerCollection& collection);
|
|
/// This is recursive! Call it once on the root node after all leafs have loaded layers.
|
|
void loadMaterials();
|
|
|
|
LoadState getLoadState() { return mLoadState; }
|
|
|
|
private:
|
|
// Stored here for convenience in case we need layer list again
|
|
MaterialGenerator* mMaterialGenerator;
|
|
|
|
LoadState mLoadState;
|
|
|
|
bool mIsDummy;
|
|
float mSize;
|
|
size_t mLodLevel; // LOD if we were to render this node in one chunk
|
|
Ogre::AxisAlignedBox mBounds;
|
|
Ogre::AxisAlignedBox mWorldBounds;
|
|
ChildDirection mDirection;
|
|
Ogre::Vector2 mCenter;
|
|
|
|
Ogre::SceneNode* mSceneNode;
|
|
|
|
QuadTreeNode* mParent;
|
|
QuadTreeNode* mChildren[4];
|
|
QuadTreeNode* mNeighbours[4];
|
|
|
|
Chunk* mChunk;
|
|
|
|
DefaultWorld* mTerrain;
|
|
|
|
Ogre::TexturePtr mCompositeMap;
|
|
|
|
void ensureCompositeMap();
|
|
};
|
|
|
|
}
|
|
|
|
#endif
|