OGRE根據高度圖創建Terrain地形流程分析

OGRE可以通過兩個接口來生成地形,分別是void TerrainSceneManager::setWorldGeometry( const String& filename )以及void TerrainSceneManager::setWorldGeometry(DataStreamPtr& stream, const String& typeName ),兩者的根本區別就是一個是傳遞地形信息文件名一個是傳遞地形信息數據流,設計兩個接口的目的是可以讓用戶使用自己的地形信息配置文件,而不必侷限於OGRE定義的地形信息配置文件格式。在分析流程之前,先說明一點,爲了提高渲染性能以及地形查找性能,OGRE把整個地形分成若干個地形頁,每個地形頁又分成了多個地形小塊,到最後會把地形頁及地形小塊tile掛接到場景節點上。以下是生成地形的主要流程:

一:清除地形分級索引緩衝及地形分頁,根據地形信息數據流加載地形信息配置文件,

void TerrainSceneManager::loadConfig(DataStreamPtr& stream),其先將地形配置信息從數據流中逐一讀出到map中,然後通過void TerrainSceneManager::selectPageSource(const String& typeName,  TerrainPageSourceOptionList& optionList)設置地形數據源(目前只有高度圖數據源)。在地形信息配置文件中可以配置多個地形數據源,然後根據一種數據源類型生成地形,參數typeName就用來指定數據源類型,目前就是HeightMap,找到指定的數據源後,對指定的數據源進行初始化,

 void HeightmapTerrainPageSource::initialise(TerrainSceneManager* tsm,  ushort tileSize, ushort pageSize, bool asyncLoading, TerrainPageSourceOptionList& optionList)。初始化的過程主要是調用void HeightmapTerrainPageSource::loadHeightmap(void)將高度圖灰度圖像數據加載到內存中,如果是Raw數據,就加載到mRawData中,否則加載到mImage中。整個加載過程其實完成了兩大工作,首先加載地形配置信息,然後加載高度圖數據。

 

二:初始化分級索引緩衝,void TerrainSceneManager::initLevelIndexes();

 

三:void OctreeSceneManager::resize( const AxisAlignedBox &box )

 

四:設置地形材質,void TerrainSceneManager::setupTerrainMaterial(void);

 

五:設置地形分頁, void TerrainSceneManager::setupTerrainPages(void)

首先創建一個名爲Terrain的場景根節點的子節點

mTerrainRoot = getRootSceneNode() -> createChildSceneNode( "Terrain" );

然後初始化TerrainPage2D mTerrainPages;

最後調用 void HeightmapTerrainPageSource::requestPage(ushort x, ushort y)requestPage只支持一個Page,首先將圖像數據進行縮放

然後調用“TerrainPageSource::firePageConstructed()”通知Listener;然後調用“TerrainPage* TerrainPageSource::buildPage(Real*heightData, const MaterialPtr& pMaterial)”創建一個新的TerrainPage對象。

buildPage()是一個比較核心的函數。它首先構造一個TerrainPage對象,然後創建一個用於容納該TerrainPage對象的場景節點:“page->pageSceneNode = mSceneManager->createSceneNode(name);”,然後根據對地形的分割,循環創建子SceneNode,並且創建子場景節點對應的可渲染體TerrainRenderable,將該可渲染體attach到這個子節點上。通過“TerrainRenderable::initialise()”來創建頂點數據、渲染方式等。

 

然後調用void TerrainSceneManager::attachPage(ushort pageX, ushort pageZ, TerrainPage* page)加入到mTerrainPages中;

然後在attachPage中調用“mTerrainRoot->addChild(page->pageSceneNode);”加入到SceneGraph中。

 

注:當前只支持一個Page,該分頁被掛接到一個稱謂Terrain的場景節點上,該場景節點下面又創建了很多子場景節點,每一個子場景節點對應一個tile,也就是一個獨立的可渲染體TerrainRenderable

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章