現在讓我們看看函數CreateNode,象它的名字一樣,他用來建立節點,實際他不僅可以建立一個 節點,還可以建立整個樹,我們隻要調用函數一次, void CreateNode(unsigned int Bounding[4],unsigned int ParentID,unsigned int NodeID) 在一個2D數組中擴展為高度為0的X和Z的面,為發現左上坐標,使用下面的公式
右上為:
:
左下?
:
右下?
:
數學並不困難,現在準備調用:
unsigned int uiBoundingCoordinates[] =
{0,GridWidth,(GridHeight*(GridWidth 1)),((GridHeight)*(GridWidth 1)) GridWidth}; CreateNode(uiBoundingCoordinates,0,0);
父節點已經建立好了,我們可以通過CreateNode來工作了。.
void CreateNode(unsigned int Bounding[4],unsigned int ParentID,unsigned int NodeID)
{
static unsigned int TotalTreeID = 0; unsigned int uiNodeType;
unsigned int uiWidth,uiHeight;
OK,靜態變量TotalTreeID保存了當前的節點數目,我們最後使用他來將子節點與他們的ID聯系起 來,uiNodeType保存節點的類型,uiWidth,uiHeight保存節點的寬和高,由於我們傳送的是包圍坐 標,實際上我們並不知道節點的大小,我們使用uiWidth,uiHeight來告訴節點是葉節點還是普通節點 ,現在需要從包圍坐標中獲得uiWidth,uiHeight:
uiWidth = fVerts[(Bounding[1]*3)] - fVerts[(Bounding[0]*3)];
uiHeight = fVerts[(Bounding[2]*3) 2] - fVerts[(Bounding[0]*3) 2];
T這裡假設fVerts是一個包含頂點列表的數組,每個頂點包含3個部件,X,Y,Z,如果我們有頂點的 索引,就可以獲得指向這個頂點的指針,
Figure 14 如同你看見的一樣,索引0指向element[0],element[0]是頂點0的X部件,依次類推。 現在,我們說我們的葉節點是4*4的三角形,這意味?葉寬為4三角形,由於我們知道節點的寬度(存儲 在uiWidth),如果我們分割寬度的結果為2,那麼意味?這個寬度為4,這個節點就是一個葉節點,
if(0.5*uiWidth==2)
{
uiNodeType = LEAF_TYPE;
}
else
{
uiNodeType = NODE_TYPE;
}
接?,我們想得到一個指向我們節點的指針,pNodeList包含所有我們的節點,我們需要選擇一個。 NODE *pNode = &pNodeList[NodeID]; 向節點內填充內容 pNodeList[NodeID].uID = Whatever; 我們可以簡單的做: pNode->uiID = Whatever; 用我們得到的值填充
pNode->uiID = NodeID;
pNode->uiParentID = ParentID; pNode->vBoundingCoordinates[0].x = fVerts[(Bounding[0]*3)];
pNode->vBoundingCoordinates[0].y = fVerts[(Bounding[0]*3) 1];
pNode->vBoundingCoordinates[0].z = fVerts[(Bounding[0]*3) 2]; pNode->vBoundingCoordinates[1].x = fVerts[(Bounding[1]*3)];
pNode->vBoundingCoordinates[1].y = fVerts[(Bounding[1]*3) 1];
pNode->vBoundingCoordinates[1].z = fVerts[(Bounding[1]*3) 2]; pNode->vBoundingCoordinates[2].x = fVerts[(Bounding[2]*3)];
pNode->vBoundingCoordinates[2].y = fVerts[(Bounding[2]*3) 1];
pNode->vBoundingCoordinates[2].z = fVerts[(Bounding[2]*3) 2]; pNode->vBoundingCoordinates[3].x = fVerts[(Bounding[3]*3)];
pNode->vBoundingCoordinates[3].y = fVerts[(Bounding[3]*3) 1];
pNode->vBoundingCoordinates[3].z = fVerts[(Bounding[3]*3) 2]; pNode->bType = uiNodeType;
現在我們還沒有處理葉節點,一旦我們分配了葉節點,我們將返回調用函數,在真實的世界裡,你 或許希望得到一個指向數組或三角形的葉節點指針,如果你仔細看過NODE結構,你將注意變量 uiVertexStrip1...4,如果你願意的話,可以在裡面填充三角形,.
if(uiNodeType == LEAF_TYPE)
{
return;
}
else
{
下面,我們需要處理節點的子節點
unsigned int BoundingBox[4];
TotalTreeID ;
pNode->uiBranches[0] = TotalTreeID; //Top-Left i.e. b[0]
BoundingBox[0] = Bounding[0];
//Between b[0] and b[1]
BoundingBox[1] = Bounding[0] ((Bounding[1]-Bounding[0])/2);
//[between b[0] and b[2]
BoundingBox[2] = Bounding[0] ((Bounding[2]-Bounding[0])/2);
//middle of node
BoundingBox[3] = Bounding[0] ((Bounding[2]-Bounding[0])/2) ((Bounding[1]-Bounding[0])/2); CreateNode(BoundingBox,NodeID,TotalTreeID);
很簡單,自己看吧,
//****************************************************************************** TotalTreeID ;
pNode->uiBranches[1] = TotalTreeID; // Between b[0] and b[1]
BoundingBox[0] = Bounding[0] ((Bounding[1]-Bounding[0])/2);
//Top-Right i.e. b[1]
BoundingBox[1] = Bounding[1];
//middle of node
BoundingBox[2] = Bounding[0] ((Bounding[2]-Bounding[0])/2) ((Bounding[1]-Bounding[0])/2);
//between b[1] & b[3]
BoundingBox[3] = Bounding[0] ((Bounding[2]-Bounding[0])/2) ((Bounding[1]-Bounding[0])); CreateNode(BoundingBox,NodeID,TotalTreeID); //****************************************************************************** TotalTreeID ;
pNode->uiBranches[2] = TotalTreeID; //between b[0] and b[2]
BoundingBox[0] = Bounding[0] ((Bounding[2]-Bounding[0])/2);
//middle of node
BoundingBox[1] = Bounding[0] ((Bounding[2]-Bounding[0])/2) ((Bounding[1]-Bounding[0])/2);
//Bottom-Left i.e. b[2]
BoundingBox[2] = Bounding[2];
//between b[2] and b[3]
BoundingBox[3] = Bounding[2] ((Bounding[3]-Bounding[2])/2); CreateNode(BoundingBox,NodeID,TotalTreeID); //****************************************************************************** TotalTreeID ;
pNode->uiBranches[3] = TotalTreeID; //middle of node
BoundingBox[0] = Bounding[0] ((Bounding[2]-Bounding[0])/2) ((Bounding[1]-Bounding[0])/2);
//between b[1] and b[3]
BoundingBox[1] = Bounding[0] ((Bounding[2]-Bounding[0])/2) uiWidth;
//between b[2] and b[3]
BoundingBox[2] = Bounding[2] ((Bounding[3]-Bounding[2])/2);
//Bottom-Right i.e. b[3]
BoundingBox[3] = Bounding[3]; CreateNode(BoundingBox,NodeID,TotalTreeID);
} All we need to do is return nothing and end CreateNode's curly brackets: return;
}
And that's about it. You can download the fully annotated source and executable here. 網路志工聯盟----Visita網站http://www.vista.org.tw
---[ 發問前請先找找舊文章 ]---