讀完此章之后,你將能夠做到:
建立允許用戶選擇(select)屏幕區域或拾取(pick)繪制在屏幕上的物體的應用程序
利用OpenGL的反饋(feedback)模式獲取絢染計算結果
有些圖形應用程序只繪制兩維和三維物體構成的靜態圖形,另一些允許用戶識別屏幕上的物體并移動、修改、刪除或用其它方法操縱這些物體。OpenGL正是設計用于支持這些交互式應用程序的。因為繪制在屏幕上的物體通常經過多次旋轉、移動和透視變換,所以確定用戶選中了三維場景中的哪個物體會很困難。為了幫助你,OpenGL提供了一個選取機制可惟自動告訴你哪個物體被繪制在窗口的提定區域里。你可以用這個機制與一個工具例程(a special utility routine)一道決定哪個物體在用戶說明或用光標選取的區域里。
選擇(selection)實際上是OpenGL的一個操作模式;反饋(feedback)是這類模式中的別一個。在反饋模式中,你用你的圖形硬件和OpenGL完成通常的絢染計算。但與用這個計算結果去在屏幕上繪制圖形相反,OpenGL返回(或反饋(feeds back))這些繪制信息給你。如果你想在繪圖儀而不是屏幕上繪制圖形,舉個例子,你就得在反饋模式繪制它們,收集繪制指令,然后將這些指令轉換為繪圖儀可以理解的命令。
在選擇和反饋模式中,繪制信息返回給應用程序而不是象在絢染模式中那樣送往幀緩沖。因此,當OpenGL處于選擇或反饋模式時,屏幕將被凍結-沒有圖形出現。這一章將會在各自的節中解釋這些模式:
“選擇(Selection)” 討論怎樣使用選擇模式和相關的例程以使你程序的用戶能拾取畫在屏幕上的物體。
“反饋(Feedback)” 描述了怎樣獲取有關什么將被畫在屏幕上的信息和這些信息是以什么格式組織的。
---------------------------------------------------
Section
通常,當你打算使用OpenGL的選擇機制時,你首先把你的場景畫進幀緩沖,然后進入選擇模式并重新繪制這個場景。然而,一旦你進入了選擇模式,幀緩沖的內容將保存不變,直到你退出選擇模式。當你退出時,OpenGL返回一個圖元(premitives)清單,圖元可能被視見體(viewing volume)分割(記住,視見體是由當前模式視見和投影矩陣及你定義的所有裁剪面定義,裁剪面詳見"Additional Clipping Planes.")。每個被視見體圖元引出一資選擇命中(hit)。確切的說,圖元清單是作為一個取整數值的名字(integer-valued names)數組和相關的數據-命中記錄(hit record)-對應名字棧(name stack)的當前內容。當你在選擇模式下發布圖元繪制命令時向名字棧中加入名字就可建立起名字棧。這樣,當名字清單被返回后,你就可以用它來確定屏幕上的哪個圖元可能被用戶選中了。
除了這個選擇機制之外,OpenGL提供了一個工具例程,以便在某些情況下通過限定在視口(viewport)一個小區域內繪制來簡化選擇。通常你可以用這個例程決定哪個物體被畫在光標附近了,這樣你就能識別用戶拾取了哪個物體。你也可以通過指定附加的裁剪面來界定一個選擇區域;詳見"Additional Clipping Planes"。因為拾取是選擇的一個特殊情況,所以本章選講選擇,然后講拾取。
基本步驟
建立名字矩陣
命中記錄
一個選擇的例子
拾取
關于編寫使用選擇的程序的提示
------------------------------------------------------------------------
基本步驟
為使用選擇機制,你得作以下幾步:
1、用glSelectBuffer()指定用于返回命中記錄的數組。
2、以GL_SELECT為參數調用glRenderMode()進入選擇模式。
3、用glInitName()和glPushName()初始化名字棧。
4、定義用于選擇的視見體。通常它與你原來用于繪制場景的視見體不同。因此你或許會想用glPushMatrix()和glPopMatrix()來保存和恢復當前的變換矩陣。
5、交替發布圖元繪制命令和名字??刂泼?,這樣每個感興趣的圖元都會被指定適當的名字。
6、退出選擇模式并處理返回的選擇數據(命中記錄)。
后面的段落將描述glSelectBuffer()和glRenderMode()。下一節則講解名字棧的控制。
void glSelectBuffer(GLsizei size, GLuint *buffer);
指定用于返回選擇數據的數組。參數buffer是指向無符號整數(unsigned integer)數組的指針,數據就存在這個數組中,size參數說明數組中最多能夠保存的值的個數。要在進入選擇模式之前調用glSelectBuffer()!
GLint glRenderMode(GLenum mode);
控制應用程序是否進入絢染(rendering)、選擇或反饋模式。mode參數可以是GL_RENDER(默認)、GL_SELECT或GL_FEEDBACK之一。應用程序將保持處于給定模式,直到再次以不同的參數調用glRenderMode()。在進入選擇模式之前必須調用glSelectBuffer()指定選擇數組。類似的,進入反饋模式之前要調用glFeedbackBuffer()指定反饋數組。如果當前模式是GL_SELECT或GL_FEEDBACK之一,那么glRenderMode()的返回值有意義。返回值是當前退出當前模式時,選擇命中數或放在反饋數組中的值的個數。(譯者注:調用此函數就會退出當前模式);負值意味著選擇或反饋數組溢出(overflowed)。你可以用GL_RENDER_MODE調用glGetIntegerv()獲取當前模式。
-------------------------------------------------------------------------------
建立名字矩陣
正如前面提到的,名字棧是返回給你的選擇信息的基礎。要建立名字棧,首先用glInitNames()初始化它,這將簡單地清空棧。然后當你發布相應的繪制命令時向其中加入整數名字。正如你可能想象,??刂泼钤试S你壓入名字(glPushName()),彈出名字(glPopName()),替換棧頂的名字(glLoadName())。
/********************************************************************/
Example 12-1: Creating a Name Stack
glInitNames();
glPushName(-1);
glPushMatrix(); /* save the current transformation state */
/*to do: create your desired viewing volume here */
glLoadName(1);
drawSomeObject();
glLoadName(2);
drawAnotherObject();
glLoadName(3);
drawYetAnotherObject();
drawJustOneMoreObject();
glPopMatrix (); /* restore the previous transformation state*/
/********************************************************************/
在這個例子中,前兩個被繪制的物體有自己的名字,第三和第四個共用一個名字。這樣,如果第三或第四個物體中的一個或全部引起一個選擇命中,只有一個命中記錄返回給你。如果處理命中記錄時不想區分各個物體的話,可以讓多個物體共享一個名字。
void glInitNames(void);
清空名字棧。
void glPushName(GLuint name);
將name壓入名字棧。壓入名字超過棧容量時將生成一個GL_STACK_OVERFLOW錯誤。名字棧深度因OpenGL實現(implementations)不同而不同,但最少要能容納64個名字。你可以用參數GL_NAME_STACK_DEPTH調用glGetIntegerv()以獲取名字棧深度。
void glPopName(void);
彈出名字棧棧頂的那一個名字。從空棧中彈出名字引發GL_STACK_UNDERFLOW錯誤。
void glLoadName(GLuint name);
用name取代名字棧棧頂的那個名字。如果棧是空的,剛調用過glInitName()后就是這樣,glLoadName()生成一個GL_INVALID_OPRATION錯。為避免這種情況,如果棧初始時是空的,那么在調用glLoadName()之前至少調用一次glPushName()以在名字棧中放上點東西。
如果不是在選擇模式下,對glPushName()、glPopName()、glLoadName()的調用將被忽略。這使得在選擇模式和正常的絢染模式下用相同的繪制代碼大為簡化。
-------------------------------------------------------------------------------
命中記錄
在選擇模式下,被視見體裁剪的每個圖元引起一個選擇命中。當前一個名字??刂泼畋粓绦谢騡lRenderMode()被調用后,OpenGL將一個命中記錄寫進選擇數組,如果從上一次名字棧操縱或glRenderMode()調用以來有了一個命中記錄的話。這個過程中,共用同樣名字的物體-例如:由多個圖元組成的物體-不生成多個命中記錄。當然,命中記錄不保證會被寫進數組中直到glRenderMode()被調用。
除圖元之外,glRasterPos()產生的有效坐標也可以引起選擇命中。在多邊形的情況下,如果它已經被消隱掉的話不會有命中記錄出現。
每個命中記錄由四項組成,依次是:
當命中出現時名字棧中的名字數
至上次記錄的命中以來,被視見體裁剪后的圖元的所有頂點的窗口Z坐標的 最大和最小值
本次命中時名字棧的內容,最底元素最前。
當前你進入選擇模式時,OpenGL初始化一個指針指向選擇數組的起點。每寫入一個命中記錄,指針相應更新。如果寫入一個命中記錄會使數組中值的個數超過glSelectBuffer()的size參數時,OpenGL會寫入盡可能多的記錄并設置一個溢出標志。當用glRenderMode()退出選擇模式時,這條命令返回被寫入的記錄的個數(包括一條部分記錄如果有的話),清除名字棧,復位溢出標識,重置棧指針。如設定溢了出標識則返回值是-1。
本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/Crazyjumper/archive/2007/10/18/1830865.aspx