一、地球背面的一個點,計算它在屏幕上的座標,能獲得嗎? 不是被擋住了嗎?算法
答:計算一個空間點的屏幕座標,使用osgAPEx::GetScreenPosition函數。當空間點處於相機視空間內(無論它是否被別的物體遮擋)時,都是能夠獲得它對應的屏幕座標的。數據庫
如何判斷一個點是否在地球背面?能夠經過計算該點處垂直地面的方向UP,與相機方向的夾角,若是夾角爲銳角,則可認爲該點在地球背面。計算地球上任意點的UP方向使用osgAPEx::ComputeLocalUpVector。數組
二、OSG中能夠播放視頻嗎?app
答:可使用quicktime插件或ffmpeg插件在OSG中播放視頻,quicktime插件的編譯須要QuickTimeSDK,ffmpeg插件的編譯須要ffmpeg庫(www.ffmpeg.org www.ffmpeg.com.cn http://ffmpeg.arrozcru.org/builds/)。OSG的quicktime插件已經中止更新了,如今力推ffmpeg。ide
ffmpeg插件好像是從2.9版開始提供的。函數
利用視頻播放插件能夠讀入avi等視頻文件到OSG紋理中,並把紋理渲染在模型上,好比實現一個立體廣告牌。佈局
三、模型縮放後光照結果錯誤,是什麼緣由?測試
答:OpenGL的放縮矩陣會同時變換模型的法線,進而影響光照效果,所以放大時可能形成模型表面光照過強,須要打開GL_NORMALIZE。優化
在OSG中使用stateset->setMode( GL_NORMALIZE, osg::StateAttribute::ON );動畫
四、OSG中如何打開雙面光照
答:使用osg::LightModel渲染屬性
五、OSG中能夠播放GIF動畫麼?
答:能夠,使用OSG提供的gif插件,能夠讀取動態GIF圖片,並經過設置爲模型紋理來渲染該圖片。經過GIF插件讀取GIF圖片返回的對象派生自osg::ImageStream,所以,可使用paly等接口控制動畫播放。
GIF插件提供了輕量級的動畫紋理播放方式,較quicktime等視頻播放插件更適合一些簡單的應用場合。
另外,OSG還提供osg::ImageSequence類,能夠經過讀入多個靜態圖片造成一個sequence,並經過ImageStream接口函數控制多張圖片的連續播放。
六、什麼是烘培,在OSG中怎樣實現烘培?
答:烘培又叫紋理烘培,是指經過每幀預渲染一副場景到指定紋理上,實現模型貼圖的動態效果。如水面倒影、鏡子、倒計時牌上跳動的文字等效果的實現,均可以使用紋理烘培技術。
在OSG中實現紋理烘培是很是容易的,大體步驟以下:
建立紋理對象,設置紋理的大小、像素格式,並把紋理對象綁定到模型;
建立烘培用的相機,指定渲染順序爲PRE_RENDER,指定實現方式爲FRAME_BUFFER_OBJECT,並把相機的顏色緩衝區綁定到紋理對象;
把要渲染到紋理上的場景加入相機的下級;
把相機加入場景。
具體例子可參考examples\osgprerender或demos\RTT
七、爲何個人OSG窗口看到的圓球是扁的?
答:這是由於投影矩陣的設置有問題,通常狀況下應該根據視口實際的寬高比來設置投影矩陣,如視口寬、高分別爲width和height,則設置相機 投影矩陣爲camera->setProjectionMatrixAsPerspective(30, (double)width/height, 1.0f,10000.0f);
若是沒有按實際視口寬高來設置投影矩陣,就會出現畫面被壓縮或拉伸的效果。
當窗口大小發生變化時,系統經過GraphicsContext對象自動響應WM_SIZE消息,對該GraphicsContext對象相關聯的 全部相機(經過調用osg::Camera::setGraphicsContext將相機與GraphicsContext關聯)自動更新視口和投影矩 陣,更新投影矩陣的方法爲根據寬高比的改變來等比縮放投影矩陣。
能夠經過osg::Camera::setProjectionResizePolicy設置相機在窗口大小發生變化時重置投影矩陣的策略,設置爲FIXED則不會自動重置投影矩陣,缺省設置爲HORIZONTAL。
對於HUD相機,因爲其使用正交投影矩陣,不能經過上述縮放的方式修改投影矩陣(正交投影矩陣不但和寬高比有關,還和寬高值有關)。能夠本身實現一個GUIEventHandler響應RESIZE事件,根據當前主視口寬高重設HUD相機的視口和投影矩陣。
八、什麼是VBO,什麼是DisplayList?
答:VBO和DisplayList是OpenGL提供的兩種用於提升渲染效率的技術。
VBO(Vertex Buffer Object)-頂點緩衝區對象,DisplayList-顯示列表。
在使用VBO時,若是修改頂點數組內容,則須要調用頂點數組的dirty()函數應用修改;使用DisplayList時,若是修改頂點數組內容,則須要調用dirtyDisplayList()來應用修改。
OSG中缺省使用顯示列表而不使用VBO。
九、OSG中分頁數據庫線程、渲染線程和篩選線程是在何時退出的?
答:1.osgViewer::Viewer對象析構時會調用stopThreading同步退出渲染和篩選線程,並調用DatabasePager::cancel通知分頁數據庫線程退出。
2.在收到WM_CLOSE消息時會調用stopThreading同步退渲染和篩選線程。
3.能夠在須要的時候手動調用stopThreading同步退出渲染和篩選線程。
十、OSG處理事件的時機和順序是怎樣的?
答:OSG在Viewer::eventTraversal函數中處理輸入事件,eventTraversal在幀循環的更新遍歷和渲染遍歷以前被調用。
eventTraversal中事件處理的順序爲:
1.爲本幀每個事件遍歷場景樹,調用Node、Drawable、StateSet對象的事件處理回調處理事件。
2.爲本幀每個事件遍歷Viewer的事件處理器(GUIEventHandler)隊列,調用事件處理器處理事件。
3.爲本幀每個事件,調用Viewer的相機操做器(CameraManipulator)處理事件。
十一、Switch(開關)節點的做用是什麼?
答:Switch節點繼承自Group節點,對應於childLis有一個boolList,爲每個子節點保存了一個開關值,當子節點開關關閉時,保證子節點不會被訪問類型爲Active子節點的訪問器訪問到。
當子節點開關關閉時,子節點不參與包圍盒計算。
十二、NodeMask、TraversalMask的做用是什麼?
答:NodeMask(32位無符號整形)做爲Node類的屬性,用於限定節點是否能被指定的節點訪問器訪問。缺省值爲0xffffffff。
TraversalMask(還有OverrideMask)做爲節點訪問器的屬性,用於限定節點訪問器可以訪問哪些節點。缺省值爲0xffffffff。
節點是否能被訪問器訪問取決於:TraversalMask & ( OverrideMask | NodeMask )
1三、osg::ColorMask的做用是什麼?
答:ColorMask封裝了glColorMask函數的功能,用於設置對顏色緩衝區的寫掩碼,即設置是否對R、G、B、A位執行寫入操做。
在只須要寫深度緩衝或模板緩衝,而不實際繪製物體時,可使用ColorMask並設置對RGBA的寫入掩碼均爲false。
1四、怎樣佈局屏幕元素(文字、圖片等)
答:屏幕元素通常使用HUD的方式(即正交投影)渲染,在osg中能夠定義一個正交投影的相機,把屏幕元素做爲子節點添加給這個HUD相機進行渲染。元素的佈局和縮放,通常有兩種處理方式:
1、 按視口實際大小布局,同時跟蹤輸出窗口的大小變化,並按窗口實際大小更新HUD相機的投影矩陣和視口。這種方式不但要動態更新相機的投影矩陣和視口,並且 要本身編寫代碼根據視口大小變化動態更新屏幕元素的位置和大小,如按指定的對齊方式更新位置、按指定的尺寸比例更新大小。也能夠只更新位置不更新大小,這 樣獲得的效果就是屏幕元素的大小不隨窗口縮放。
2、 按指定的固定大小布局,同時設置HUD相機的投影矩陣爲以相同寬高計算的固定值,跟蹤輸出窗口大小變化,按窗口實際大小更新HUD相機的視口。這種方式只 須要編寫代碼更新相機的視口便可,而屏幕元素的大小和位置始終按初始的固定視口大小計算,不需隨窗口變化作調整。這樣獲得的效果就是屏幕元素的大小和佈局 與窗口等比縮放。
1五、setAttribute與setAttributeAndModes的異同
答:setAttribute用於設置渲染屬性和屬性的覆蓋模式,第二個參數的有效值只有OVERRIDE和PROTECTED的組合,不影響屬性狀態的開關(設置ON/OFF是無效的)!
setAttributeAndModes設置渲染屬性和覆蓋模式的同時,也會設置屬性狀態的開關。
setAttribute用於只需設置渲染屬性而不改變狀態的場合,具體是否打開該狀態取決於子節點的設置。
1六、OSG場景篩選中有哪些優化效率的算法
答:OSG場景篩選主要用到的篩選方法有:視景體篩選、最 小像素篩選和遮擋篩選。其中視景體篩選是最基本和最經常使用的篩選方法,它的算法是:使用構成視景體的各個面與節點包圍盒求交,若是節點在某個面之外,則剔除 掉該節點及其子節點;若是節點徹底在某個面之內,則該節點的全部子節點在進行篩選計算的時候不須要再與這個面求交(子節點必然也徹底在這個面之內)。鑑於 這種篩選實現,恰當使用樹形結構場景圖組織場景節點,可以提升篩選效率。
1七、我在MFC的Static控件中使用OSG渲染,但不能響應鼠標事件是爲何?
答:這徹底是MFC控件使用的問題,要使對話框中的Static控件接收鼠標消息,須要把Static控件屬性的Notify設置爲True。
1八、OSG渲染排序是怎麼作的,有哪些接口能夠用來控制渲染順序?
答:OSG經過場景揀選(cull)生成一棵由RenderStage(根節點)、RenderBin(分支節點)和RenderLeaf(渲染葉)構成的渲染樹,同時生成一棵由StateGraph構成的渲染狀態樹,最終的渲染操做經過渲染樹進行,渲染的順序爲:
RenderStage::draw
|
RenderStage::drawPreRenderStages
|
RenderStage::drawInner ------------------------------------------- RenderBin::draw
| |
RenderStage::drawPostRenderStages BinNum小於0的子Bin(按BinNum升序)
|
本Bin的渲染葉列表(排序的)
|
本Bin下掛的全部StateGraph中的渲染葉
|
其餘的子Bin(按BinNum升序)
在 揀選過程當中,經過揀選測試的Drawable對象被添加到當前StateGraph對象的渲染葉列表中,而StateGraph則被當前 RenderBin對象引用,而最終渲染時,部分渲染葉也是經過RenderBin引用的StateGraph對象找到並渲染的,其餘渲染葉(對應 RenderBin的排序模式爲FRONT_TO_BACK/BACK_TO_FRONT/TRAVERSAL_ORDER的)被轉移到 RenderBin的渲染葉列表中排序而後渲染。
咱們能夠經過StateSet的BinNum和BinName設置來控制對應節點的渲染順序。
1)BinName爲"RenderBin"、"StateSortedBin"的缺省排序方式爲SORT_BY_STATE(目前實現爲不排序)
BinName爲"DepthSortedBin"的缺省排序方式爲SORT_BACK_TO_FRONT(由遠到近,用於半透物體的alpha混合)
BinName爲"TraversalOrderBin"的缺省排序方式爲TRAVERSAL_ORDER(即按遍歷到的前後順序排序)。
2)BinNum小的靠前渲染
3)setNestRenderBins設置RenderBin是否容許嵌套,爲true時新的RenderBin被添加爲當前RenderBin的子節點,爲false時新的RenderBin被添加爲RenderStage(根節點)的子節點。
1九、OSG對組合鍵是怎樣映射的?
答:因爲Win32 API的特色,OSG不得不將Ctrl+letter變換到了{1,..,26},也就是說,CTRL+a的getKey()爲1,而CTRL+z的getKey()爲26;Shift+a~z被映射到大寫的'A'~'Z'。
20、爲何OSG自動爲個人紋理對象建立了mipmap?
答:當紋理過濾方式min_filter設 置爲基於mipmap的4種之一時,OSG將自動爲非mipmap貼圖紋理對象建立mipmap。min_filter的缺省值是 LINEAR_MIPMAP_LINEAR,所以若是未顯式設置紋理過濾方式的話,即會啓用自動建立mipmap的功能。
2一、Texture::setUnRefImageDataAfterApply的做用?
答:當該值設置爲true時,OSG在應用過該紋理對象後(apply),自動釋放其對Image對象的引用,以減小內存佔用。
2二、osgDB::SharedStateManager的做用?
答:SharedStateManager 用於對系統中的StateSet、Texture對象執行共享優化,即對系統中內容相同的多個Texture對象,只保留一個實例,其餘使用引用,以減小 內存、顯存的佔用,提升渲染效率。使用方式爲,爲新加載或建立的節點對象調用 osgDB::Registry()->getOrCreateSharedStateManager()->share函數。
2三、如何打開分頁數據庫的預編譯功能?答:須要作兩件事,1)設置 osg::DisplaySettings::instance()->setCompileContextsHint(true),使 Viewer在realize時建立後臺GC用於預編譯操做;2)調用 osgDB::DabasePager::setDoPreCompile(true)開啓預編譯功能。