PagedGeometry
1.3.0
|
A class providing highly optimized methods for rendering massive amounts of geometry. More...
#include <PagedGeometry.h>
Public Member Functions | |
PagedGeometry (Ogre::Camera *cam=NULL, Ogre::Real pageSize=100, Ogre::RenderQueueGroupID queue=Ogre::RENDER_QUEUE_6) | |
Initializes a PagedGeometry object. More... | |
void | setCamera (Ogre::Camera *cam) |
Sets the camera to use when calculating levels of detail. More... | |
void | setTempDir (Ogre::String dir) |
Sets the output directory for the imposter pages. | |
Ogre::String | getTempdir () |
Ogre::Camera * | getCamera () const |
Gets the camera which is used to calculate levels of detail. More... | |
Ogre::SceneManager * | getSceneManager () const |
Gets the scene manager which is being used to display the geometry. More... | |
Ogre::SceneNode * | getSceneNode () const |
Gets the scene node to which all PagedGeometry geometry is attached. More... | |
void | setBounds (const TBounds bounds) |
Switches to bounded mode and uses the given boundaries. More... | |
void | setInfinite () |
Switches to infinite mode. More... | |
const TBounds & | getBounds () const |
Gets the current geometry boundary. More... | |
TBounds | convertAABToTBounds (const Ogre::AxisAlignedBox &aab) const |
Convert an Ogre::AxisAlignedBox to a TBounds coplanar to the plane defined by the UP axis. | |
void | setPageSize (Ogre::Real size) |
Sets the page size. More... | |
Ogre::Real | getPageSize () |
Gets the current page size. More... | |
template<class PageType > | |
GeometryPageManager * | addDetailLevel (Ogre::Real maxRange, Ogre::Real transitionLength=0, const Ogre::Any &data=Ogre::Any(), Ogre::uint32 queryFlag=0) |
Adds a detail level to the PagedGeometry object. More... | |
void | removeDetailLevels () |
Removes all detail levels from the PagedGeometry object. More... | |
const std::list< GeometryPageManager * > & | getDetailLevels () |
Returns a reference to a list of all added detail levels. More... | |
void | setPageLoader (PageLoader *loader) |
Assigns a PageLoader object for the PagedGeometry. More... | |
PageLoader * | getPageLoader () |
Gets the PageLoader currently being used to load geometry. More... | |
void | update () |
Updates this PagedGeometry object. More... | |
void | reloadGeometry () |
Reloads all visible geometry. More... | |
void | reloadGeometryPage (const Ogre::Vector3 &point) |
Reloads geometry at the given location. More... | |
void | reloadGeometryPages (const Ogre::Vector3 ¢er, Ogre::Real radius) |
Reloads geometry in the given radius area. More... | |
void | reloadGeometryPages (const TBounds &area) |
Reloads geometry in the given rect area. More... | |
void | preloadGeometry (const TBounds &area) |
Preloads a region of geometry (loads once and never loads again) More... | |
void | resetPreloadedGeometry () |
Releases geometry preloaded with preloadGeometry() to be unloaded if necessary. More... | |
void | setVisible (bool visible) |
Hides or unhides all geometry managed by this PagedGeometry instance \params visible Whether or not you want this PagedGeometry to be visible. More... | |
bool | getVisible () |
Returns whether or not geometry managed by this PagedGeometry instance is visible. More... | |
void | setShadersEnabled (bool value) |
disables the use of shaders | |
bool | getShadersEnabled () |
const Ogre::String & | getShaderLanguage () const |
const char * | getFragmentProgramName () const |
Ogre::Vector3 | _convertToLocal (const Ogre::Vector3 &globalVec) const |
INTERNAL FUNCTION - DO NOT USE. | |
void | setCustomParam (const Ogre::String &entity, const Ogre::String ¶mName, float paramValue) |
Sets or creates a custom parameter for an entity managed by PagedGeometry This can be used to set custom parameters / data for entities which can be accessed from other PagedGeometry subsystems or your own code. More... | |
void | setCustomParam (const Ogre::String ¶mName, float paramValue) |
Sets or creates a custom parameter for an entity managed by PagedGeometry This can be used to set custom parameters / data for entities which can be accessed from other PagedGeometry subsystems or your own code. More... | |
float | getCustomParam (const Ogre::String &entity, const Ogre::String ¶mName, float defaultParamValue) const |
Returns the value of the custom parameter. More... | |
float | getCustomParam (const Ogre::String ¶mName, float defaultParamValue) const |
Returns the value of the custom parameter. More... | |
Ogre::uint8 | getRenderQueue () const |
Returns the rendering queue that paged geometry was constructed with. | |
Protected Member Functions | |
void | _addDetailLevel (GeometryPageManager *mgr, Ogre::Real maxRange, Ogre::Real transitionLength) |
Protected Attributes | |
Ogre::SceneManager * | sceneMgr |
Ogre::SceneNode * | rootNode |
bool | shadersEnabled |
Ogre::String | shaderLanguage |
bool | geometryAllowedVisible |
Ogre::Camera * | sceneCam |
Ogre::Vector3 | oldCamPos |
Ogre::Camera * | lastSceneCam |
Ogre::Vector3 | lastOldCamPos |
std::list< GeometryPageManager * > | managerList |
PageLoader * | pageLoader |
TBounds | m_bounds |
Ogre::Real | pageSize |
Ogre::uint8 | m_nRenderQueue |
The used rendering queue. | |
Ogre::Timer | timer |
unsigned long | lastTime |
Ogre::String | tempdir |
A class providing highly optimized methods for rendering massive amounts of geometry.
The PagedGeometry class provides highly optimized methods for rendering massive amounts of small meshes, covering a large area. This is especially well suited for dense forests, with thousands of trees, bushes, grass, rocks, etc., etc.
The paged geometry works by loading only the geometry that is visible (or will soon be visible), to save memory. The PagedGeometry engine can display entities using many different methods (static geometry, impostors, etc.) What method is used depends on the entities' distance from the camera, and how you configured these methods (see PagedGeometry::addDetailLevel() for more info about this).
The loading of pages is done through a PageLoader, which you define. This way, you can program the PagedGeometry to load pages any way you want, whether it's from a file on your hard drive, from a procedural generation algorithm, or anything else. See the documentation for the PageLoader class for more information about this.
Forests::PagedGeometry::PagedGeometry | ( | Ogre::Camera * | cam = NULL , |
Ogre::Real | pageSize = 100 , |
||
Ogre::RenderQueueGroupID | queue = Ogre::RENDER_QUEUE_6 |
||
) |
Initializes a PagedGeometry object.
cam | A camera which the PagedGeometry object will use for LOD calculations. |
pageSize | The page size (pages are square) |
pageSize sets the size of a single "page" of geometry. If your pages are too big, you may experience "hiccuping" during the game as these regions are loaded. However, regions that are too small may result in lower frame rates (depending on what detail levels you are using).
Also, using larger pages uses slightly less memory, although you should generally give performance precedence over memory usage.
References Ogre::SceneNode::createChildSceneNode(), Ogre::Camera::getDerivedPosition(), Ogre::SceneManager::getRootSceneNode(), Ogre::Camera::getSceneManager(), getSingleton(), and Ogre::Timer::reset().
void Forests::PagedGeometry::setCamera | ( | Ogre::Camera * | cam | ) |
Sets the camera to use when calculating levels of detail.
cam | The camera to assign to this PagedGeometry object. |
References Ogre::SceneNode::createChildSceneNode(), Ogre::SceneManager::getRootSceneNode(), Ogre::Camera::getSceneManager(), OGRE_EXCEPT, Ogre::Node::setOrientation(), and std::swap().
|
inline |
Gets the camera which is used to calculate levels of detail.
Referenced by Forests::ImpostorPage::update().
|
inline |
Gets the scene manager which is being used to display the geometry.
This function simply returns the SceneManager that this PagedGeometry object is using. If no camera has been set yet, this will return NULL, since PagedGeometry has no way of knowing which SceneManager to use. However, once a camera is set, the SceneManager this function returns will always remain the same - even if the camera is later set to NULL.
Referenced by Forests::BatchPage::init(), Forests::GrassPage::init(), Forests::ImpostorPage::init(), and Forests::WindBatchPage::init().
|
inline |
Gets the scene node to which all PagedGeometry geometry is attached.
This function returns the SceneNode which PagedGeometry uses to render all it's geometry. Everything that PagedGeometry renders to the screen can be found under this scene node.
You don't need to use this function at all to fully make use of PagedGeometry's features - it's primary use is for PagedGeometry's internal subsystems to be able to create geometry using the proper scene node.
Referenced by Forests::BatchPage::init(), Forests::GrassPage::init(), Forests::ImpostorPage::init(), Forests::WindBatchPage::init(), and Forests::ImpostorPage::~ImpostorPage().
void Forests::PagedGeometry::setBounds | ( | const TBounds | bounds | ) |
Switches to bounded mode and uses the given boundaries.
By default, PagedGeometry does not place boundaries on the geometry that can be added to it though the PageLoader. However, there are cases where specifying a strict boundary can improve performance.
When bounded mode is used, PagedGeometry allocates certain lightweight data structures to fill the boundaries. This may result in a slight frame rate boost compared to an infinite world, although it will take a little more memory (especially with large areas).
Since bounded mode requires more memory for larger boundaries, it is best suited for small to medium sized worlds, while infinite mode is best for huge or even near-infinite worlds.
References Ogre::Exception::ERR_RENDERINGAPI_ERROR, Ogre::TRect< class >::height(), OGRE_EXCEPT, and Ogre::TRect< class >::width().
void Forests::PagedGeometry::setInfinite | ( | ) |
Switches to infinite mode.
By default, PagedGeometry will just allocate enough memory to display and cache what is on the screen. This behavior is called "infinite mode", and can be activated by calling this function if not already activated (it will be by default).
Most game worlds have boundaries of some sort, but infinite mode allows you to expand the size of your game world almost to infinity. Since only what's on the screen is actually loaded, it makes little difference if your world is 100 square miles, or 1,000,000 square miles.
The only disadvantage to using infinite mode is that cache efficiency will be slightly reduced in some cases. For example, bounded mode will achieve better performance if you are often switching between multiple cameras, since the cache is more globally based (unlike the locally based cache of infinite mode).
References Ogre::Exception::ERR_RENDERINGAPI_ERROR, and OGRE_EXCEPT.
|
inline |
Gets the current geometry boundary.
This returns a TBounds value, which contains information about the boundaries of the geometry. Since TBounds is simply a typedef for TRect<Real>, accessing the boundary information is easy, for example:
Ogre's documentation should contain more information about TRect members.
Referenced by addDetailLevel(), Forests::TreeLoader2D::TreeLoader2D(), and Forests::TreeLoader3D::TreeLoader3D().
void Forests::PagedGeometry::setPageSize | ( | Ogre::Real | size | ) |
Sets the page size.
This sets the size of a single "page" of geometry. If your pages are too big, you may experience "hiccuping" during the game as these regions are loaded. However, regions that are too small may result in lower frame rates (depending on what detail levels you are using).
Also, using larger pages uses slightly less memory, although you should generally give performance precedence over memory usage.
References Ogre::Exception::ERR_RENDERINGAPI_ERROR, and OGRE_EXCEPT.
|
inline |
Gets the current page size.
Referenced by Forests::GeometryPageManager::initPages(), Forests::TreeLoader2D::TreeLoader2D(), Forests::TreeLoader3D::TreeLoader3D(), Forests::ImpostorPage::update(), and Forests::GeometryPageManager::update().
|
inline |
Adds a detail level to the PagedGeometry object.
PageType | The page type class you want to use for this detail level. |
maxRange | The maximum distance this detail level will be used at. |
transitionLength | The desired length of fade transitions - optional |
data | An extra parameter to pass to the PageType constructor - optional |
On it's own, a plain PagedGeometry object can't display anything. It needs you to add detail levels to it with this function. This way, you can easily customize the behavior of the PagedGeometry.
To use this function, simply use the form:
In the above example, "farRange" specifies the maximum view distance this detail level will be used at. However, the most important part is "<PageType>". Here you substitute "PageType" with the name of a installed page type you want to use for this detail level. A page type is simply a method of displaying or otherwise representing entities.
The PagedGeometry engine currently comes pre-installed with a few page types: BatchPage and ImpostorPage. Refer to their class documentation for detailed information about them (although you never actually access these classes yourself).
You can use these page types in any configuration you want. For example:
That example would set up the PagedGeometry object called pagedTrees to use batched geometry (StaticGeometry) from 0 to 100 units from the camera, and impostors from 100 to 500 units from the camera.
If the included page types aren't adequate, you can fairly easily add your own by simply extending the virtual GeometryPage class properly.
By default, no fade transitions are used. This means that there will be a noticeable "popping" effect when your tree changes from an impostor to a batch, for example. Enabling fade transitions will smooth out the change by slowly "morphing" between the two detail levels. To enable fade transitions, simply add a second parameter to you addDetailLevel() call:
The second parameter seen above will enable transitions between BatchPage and the next LOD (which is ImpostorPage, in this case). The number you supply will specify how long the transition is. Longer transitions result in smoother "morphs", although shorter transitions will give slightly better frame rates.
The transition parameter can also be applied to the last detail level to cause it to fade out:
In the example above, batching is used up to 100 units, where the batched geometry starts transitioning for 50 units into the next level of detail (impostors). The impostors continue up to 400 units, where they begin to fade out for 100 more units.
Transitions can be disabled by omitting the transitionLength parameter, or setting it to 0.
The "data" parameter is entirely optional, so there should never be any requirement that it be used. Just what type of data this parameter accepts, and what it does, depends entirely on the specific GeometryPage implementation you're using. See the appropriate page type documentation for info on how this parameter can be used (if at all).
References getBounds(), Ogre::RenderSystem::getCapabilities(), Ogre::Root::getRenderSystem(), Ogre::Root::getSingletonPtr(), Ogre::RenderSystemCapabilities::hasCapability(), Forests::GeometryPageManager::initPages(), and Ogre::RSC_VERTEX_PROGRAM.
void Forests::PagedGeometry::removeDetailLevels | ( | ) |
Removes all detail levels from the PagedGeometry object.
This removes all detail levels (added with addDetailLevel) from the PagedGeometry object. This also removes all geometry associated with PagedGeometry from the scene. Remember that you will need to re-add all the detail levels again with addDetailLevel() before any of the geometry will be displayed.
|
inline |
Returns a reference to a list of all added detail levels.
This returns a std::list of all detail levels (GeometryPageManager's). These objects can be used to set/get advanced properties, such as view ranges and cache speeds.
Normally you won't ever have to access this data, but it's there in case you need it.
void Forests::PagedGeometry::setPageLoader | ( | PageLoader * | loader | ) |
Assigns a PageLoader object for the PagedGeometry.
loader | A PageLoader object. |
When the page manager decides it should cache a certain region of geometry, it calls on your PageLoader to do the job. This way you can load entities from RAM, a hard-drive, the internet, or even procedurally. Simply create a PageLoader class and use this function to link it to a PagedGeometry object.
|
inline |
Gets the PageLoader currently being used to load geometry.
This can be useful if you want to retrieve the current page loader to delete it, or any other task that needs to be done to the currently used page loader.
void Forests::PagedGeometry::update | ( | void | ) |
Updates this PagedGeometry object.
This function must be called each frame in order for the PagedGeometry object to calculate LODs and perform paging. If this function is not called every frame, none of the geometry managed by this PagedGeometry instance will appear (or if it does, it will appear incorrectly)
References _convertToLocal(), Forests::PageLoader::frameUpdate(), Ogre::Camera::getDerivedDirection(), Ogre::Camera::getDerivedPosition(), Ogre::Timer::getMilliseconds(), Forests::GeometryPageManager::update(), and Forests::StaticBillboardSet::updateAll().
void Forests::PagedGeometry::reloadGeometry | ( | ) |
Reloads all visible geometry.
If your PageLoader changes it's output during runtime, you normally won't see the changes immediately (and in many cases, you will never see the changes). This function provides a way to reload the geometry to force the changes to take effect immediately.
This function will cause ALL visible geometry to be reloaded during the next update. This can take up to several seconds, depending on the complexity of the current scene, so use this function only when absolutely necessary.
References Forests::GeometryPageManager::reloadGeometry().
void Forests::PagedGeometry::reloadGeometryPage | ( | const Ogre::Vector3 & | point | ) |
Reloads geometry at the given location.
point | The point in 3D space where geometry needs to be reloaded. |
If your PageLoader changes it's output during runtime, you normally won't see the changes immediately (and in many cases, you will never see the changes). This function provides a way to reload the geometry to force the changes to take effect immediately.
This function will cause a certain page of visible geometry to be reloaded during the next update. Unlike reloadGeometry(), this function allows pinpoint reloading to take place, resulting in better performance if a small portion of the geometry changes.
Since this doesn't actually reload anything immediately, you can call this function as many times as you need without worrying too much about performance. For example, if you update 150 trees in your game, simply supply this function with the locations of each tree. When the scene is about to be rendered, the appropriate geometry pages will automatically be reloaded.
References _convertToLocal(), and Forests::GeometryPageManager::reloadGeometryPage().
void Forests::PagedGeometry::reloadGeometryPages | ( | const Ogre::Vector3 & | center, |
Ogre::Real | radius | ||
) |
Reloads geometry in the given radius area.
center | The center of the area to be reloaded |
radius | The radius from the center where geometry needs to be reloaded |
If your PageLoader changes it's output during runtime, you normally won't see the changes immediately (and in many cases, you will never see the changes). This function provides a way to reload the geometry to force the changes to take effect immediately.
This function will cause a certain area of visible geometry to be reloaded during the next update. Unlike reloadGeometry(), this function allows selective reloading to take place, resulting in better performance if a small portion of the geometry changes.
Since this doesn't actually reload anything immediately, you can call this function as many times as you need without worrying too much about performance. For example, if you update 150 trees in your game, simply supply this function with the locations of each tree. When the scene is about to be rendered, the appropriate geometry pages will automatically be reloaded.
References _convertToLocal(), and Forests::GeometryPageManager::reloadGeometryPages().
void Forests::PagedGeometry::reloadGeometryPages | ( | const TBounds & | area | ) |
Reloads geometry in the given rect area.
area | A rectangular area that needs reloading |
If your PageLoader changes it's output during runtime, you normally won't see the changes immediately (and in many cases, you will never see the changes). This function provides a way to reload the geometry to force the changes to take effect immediately.
This function will cause a certain area of visible geometry to be reloaded during the next update. Unlike reloadGeometry(), this function allows selective reloading to take place, resulting in better performance if a small portion of the geometry changes.
Since this doesn't actually reload anything immediately, you can call this function as many times as you need without worrying too much about performance. For example, if you update 150 trees in your game, simply supply this function with the locations of each tree. When the scene is about to be rendered, the appropriate geometry pages will automatically be reloaded.
References Ogre::TRect< class >::bottom, Ogre::TRect< class >::left, Forests::GeometryPageManager::reloadGeometryPages(), Ogre::TRect< class >::right, std::swap(), and Ogre::TRect< class >::top.
void Forests::PagedGeometry::preloadGeometry | ( | const TBounds & | area | ) |
Preloads a region of geometry (loads once and never loads again)
area | A rectangular area of the world that needs to be preloaded |
You can use this function to preload entire areas of geometry. Doing this will basically turn off dynamic paging for the given region, since all the pages effecting it will stay loaded forever (until you delete the PagedGeometry object, or if using infinite mode, until you move away from the region).
References Ogre::TRect< class >::bottom, Ogre::TRect< class >::left, Forests::GeometryPageManager::preloadGeometry(), Ogre::TRect< class >::right, std::swap(), and Ogre::TRect< class >::top.
void Forests::PagedGeometry::resetPreloadedGeometry | ( | ) |
Releases geometry preloaded with preloadGeometry() to be unloaded if necessary.
When you call preloadGeometry() to preload region(s) of geometry, it makes those regions un-unloadable; in other words, they will never be unloaded automatically by PagedGeometry (except for some cases in infinite mode). This way you can load a region of your world once and never have to load it again (optimally).
This function allows you to undo all this by allowing all the geometry to be unloaded once again when necessary. It won't unload anything, but it will allow geometry to unload that previously was not allowed.
References Forests::GeometryPageManager::resetPreloadedGeometry().
|
inline |
Hides or unhides all geometry managed by this PagedGeometry instance \params visible Whether or not you want this PagedGeometry to be visible.
By default everything is visible. This can be used to hide an entire PagedGeometry "group" of geometry if desired.
|
inline |
Returns whether or not geometry managed by this PagedGeometry instance is visible.
By default, everything will be visible of course. This function simply returns the visible/invisible state as set by the setVisible() command.
Referenced by Forests::GeometryPageManager::update().
void Forests::PagedGeometry::setCustomParam | ( | const Ogre::String & | entity, |
const Ogre::String & | paramName, | ||
float | paramValue | ||
) |
Sets or creates a custom parameter for an entity managed by PagedGeometry This can be used to set custom parameters / data for entities which can be accessed from other PagedGeometry subsystems or your own code.
Primarily, this is intended for use with GeometryPage implementations or PageLoader implementations.
PagedGeometry includes a GeometryPage implementation, "WindBatchPage", which applies a wind animation shader to your trees, which you can control using these custom parameters: windFactorX and windFactorY.
If you're using 3rd party PagedGeometry "plugins" like GeometryPage implementations, etc., there may be more custom parameters available to you. Check with the appropriate module documentation for info on supported custom parameters and their usage.
entity | Name of the entity |
paramName | Name of the parameter for this entity |
paramValue | Value to assign to the parameter |
void Forests::PagedGeometry::setCustomParam | ( | const Ogre::String & | paramName, |
float | paramValue | ||
) |
Sets or creates a custom parameter for an entity managed by PagedGeometry This can be used to set custom parameters / data for entities which can be accessed from other PagedGeometry subsystems or your own code.
Primarily, this is intended for use with GeometryPage implementations or PageLoader implementations.
PagedGeometry includes a GeometryPage implementation, "WindBatchPage", which applies a wind animation shader to your trees, which you can control using these custom parameters: windFactorX and windFactorY.
If you're using 3rd party PagedGeometry "plugins" like GeometryPage implementations, etc., there may be more custom parameters available to you. Check with the appropriate module documentation for info on supported custom parameters and their usage.
entity | Name of the entity |
paramName | Name of the parameter for this entity |
paramValue | Value to assign to the parameter |
float Forests::PagedGeometry::getCustomParam | ( | const Ogre::String & | entity, |
const Ogre::String & | paramName, | ||
float | defaultParamValue | ||
) | const |
Returns the value of the custom parameter.
entity | Name of the entity |
paramName | Name of the parameter for this entity |
defaultParamValue | Value to return if no entry is found |
Referenced by Forests::WindBatchedGeometry::WindSubBatch::build().
float Forests::PagedGeometry::getCustomParam | ( | const Ogre::String & | paramName, |
float | defaultParamValue | ||
) | const |
Returns the value of the custom parameter.
entity | Name of the entity |
paramName | Name of the parameter for this entity |
defaultParamValue | Value to return if no entry is found |