二維填充圖元的生成_第1頁
已閱讀1頁,還剩77頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)

文檔簡介

1、1,第4章 二維填充圖元的生成,2,本章目標,掌握如何繪制填充圖元(矩形、多邊形等)掃面轉(zhuǎn)換算法(掃描線算法)填充算法學會使用OpenGL的繪制函數(shù),3,主要內(nèi)容,填充圖元掃描轉(zhuǎn)換矩形掃描轉(zhuǎn)換多邊形掃描轉(zhuǎn)換扇形區(qū)域區(qū)域填充以圖像填充區(qū)域字符的表示與輸出OpenGL相關(guān)函數(shù),4,4.1 填充圖元,填充(Filling)矩形(Rectangle)多邊形(Polygon)扇形(Ellipse Arc)步驟確定

2、那些像素位于填充圖元的內(nèi)部用指定顏色繪制這些像素兩類方法掃描轉(zhuǎn)換(Scan Converting): 參數(shù)->點陣填充(Filling) : 點陣->點陣,,,,,,,,5,4.2 掃描轉(zhuǎn)換矩形,含義用指定顏色填充矩形內(nèi)部區(qū)域定義矩形的參數(shù)左下角坐標(xmin, ymin)右上角坐標(xmax, ymax),void FillRectangle(Rectangle *rect,int color){ in

3、t x,y; for(y = rect->ymin; y ymax; y++) for(x = rect->xmin; x xmax; x++) SetPixel(x, y, color);}/*end of FillRectangle() */,,(xmin, ymin),(xmax, ymax),6,4.2 掃描轉(zhuǎn)換矩形,問題矩形是簡單的多邊形,那么為什么要單獨處理矩形?比一般多邊形可簡化

4、計算應(yīng)用非常多,如窗口系統(tǒng)共享邊界如何處理?原則:左閉右開,下閉上開,,,,屬于誰?,7,4.3 掃描轉(zhuǎn)換多邊形,多邊形的表示方法頂點表示頂點序列P0P1P2…Pn點陣表示掃描轉(zhuǎn)換多邊形:將頂點表示形式轉(zhuǎn)換成點陣表示形式三種方法:逐點判斷法;掃描線算法;邊緣填充法多邊形分類(只考慮:簡單多邊形,即多邊形邊不自相交)凸多邊形(convex):內(nèi)角小于180度凹多邊形(concave):存在內(nèi)角大于180度,8,

5、4.3 掃描轉(zhuǎn)換多邊形,如何識別多邊形的凸凹性方法1:觀察多邊形邊的延長線是否劃分頂點在兩側(cè)方法2:向量的叉積每條邊建立一個向量,測試相鄰邊的叉積z坐標的正負(1)如果叉積同號,那么是凸多邊形(2)如果叉積不同號,那么是凹多邊形,(E1×E2 )z > 0(E2×E3 )z > 0(E3×E4 )z 0(E5×E6 )z > 0(E6×E1 )z &

6、gt; 0,9,4.3 掃描轉(zhuǎn)換多邊形,向量叉積(Cross Product of Two Vector),當a與b為二維向量時,a×b 矢量中x,y 分量為0,10,4.3.1 逐點判斷法,基本原理判斷繪圖窗口內(nèi)的像素是否位于多邊形內(nèi),若是,則用指定顏色繪制該像素問題如何判斷點在多邊形的內(nèi)外關(guān)系?射線法累計角度法*編碼法*,,(xmin, ymin),(xmax, ymax),11,4.3.1 逐點判斷法,算

7、法假設(shè)判斷點是否在多邊形內(nèi)的函數(shù)為IsInside(),void FillPolygon(Polygon *P, int polygonColor){ int x,y; for(y = ymin;y <= ymax;y++) for(x = xmin;x <= xmax;x++)if(IsInside(P,x,y)) SetPixel(x, y, polygonColor);el

8、se SetPixel(x,y,backgroundColor);}/*end of FillPolygonPbyP()*/,#define MAX 100typedef struct { // 多邊形頂點個數(shù) int PolygonNum; //多邊形頂點數(shù)組 Point vertexces[MAX]} Polygon; // 多邊形結(jié)構(gòu),12,4.3.1 逐點判斷法,

9、判斷點是否在多邊形內(nèi)-射線法 步驟從待判別點 v 發(fā)出射線求與多邊形交點個數(shù) kk 的奇偶性決定了點與多邊形的內(nèi)外關(guān)系偶數(shù):外奇數(shù):內(nèi),v2,13,4.3.1 逐點判斷法,判斷點是否在多邊形內(nèi)-射線法 奇異情況射線在邊上:無數(shù)個點 判斷是否與邊同線交點為頂點:算幾個?異側(cè):1個同側(cè):2個,14,4.3.1 逐點判斷法,逐點判斷掃描轉(zhuǎn)換方法特點程序簡單測試點是否在多邊形內(nèi)的算法速度太慢,

10、效率低改進逐點判斷法孤立考慮各個像素與多邊形的內(nèi)外關(guān)系利用內(nèi)部點的連續(xù)性,15,4.3.1 逐點判斷法,思考題 下圖是某油田油井分布圖,已知每口油井的位置(x, y坐標值)和產(chǎn)油量,如何求任意多邊形(虛線所示)中的總產(chǎn)油量?,,利用射線法判斷油井是否在多邊形內(nèi)?,16,4.3.2 掃描線算法,英文:Scan-Line Algorithm目標利用相鄰像素之間的連貫性,提高算法效率處理對象:簡單多邊形非自交多邊形 (邊與

11、邊之間除了頂點外無其它交點)掃描線(Scanning Line)平行于坐標軸的直線一般取平行于X軸區(qū)間:掃描線與邊的交點間的線段,17,4.3.2 掃描線算法,連貫性(Coherence)邊的連貫性(Edge Coherence)某條邊與當前掃描線相交,也可能與下一條掃描線相交掃描線的連貫性(Scan-line Coherence)當前掃描線與各邊的交點順序與 下一條掃描線與各邊的交點順序可能相同或類似區(qū)間的連貫性(S

12、pan Coherence)同一區(qū)間上的像素取同一顏色屬性,18,4.3.2 掃描線算法,基本原理將整個繪圖窗口內(nèi)掃描多邊形的問題分解到一條條掃描線,只要完成每條掃描線的繪制就實現(xiàn)了多邊形的掃描轉(zhuǎn)換一條掃描線與多邊形的邊有偶數(shù)個交點,每2個點形成一區(qū)間步驟(對于每一條掃描線)(1)計算掃描線與邊的交點(2)交點按x坐標從小到大排序(3)交點兩兩配對,填充區(qū)間,19,4.3.2 掃描線算法,計算交點分類第一類交點:位于

13、同一條邊上的后繼交點--(P2, P4)第二類交點:新出現(xiàn)的邊與掃描線的交點--(P3)計算:由掃描線y=e與多邊形的交點遞推計算掃描線 y=e+1的交點第一類交點:x’=x+1/m第二類交點: 線段的下端點即為交點,P3,P4,P0,P2,P1,20,4.3.2 掃描線算法,計算交點(續(xù))交點取整規(guī)則:要求:使生成的像素全部位于多邊形之內(nèi)用于線畫圖元掃描轉(zhuǎn)換

14、的四舍五入原則導致部分像素位于多邊形之外,從而不可用,掃描轉(zhuǎn)換,位于多邊形內(nèi),21,4.3.2 掃描線算法,取整規(guī)則假定非水平變與掃描線 y=e 相交,交點的橫坐標為x規(guī)則1 X為小數(shù),即交點落于掃描線上兩個相鄰像素之間 (a)交點位于左邊之上,向右取整 (b)交點位于右邊之上,向左取整,22,4.3.2 掃描線算法,規(guī)則2落在右上邊界的像素不予填充。 具體實現(xiàn)時,只要對掃描線與多邊形的相交

15、區(qū)間左閉右開,23,4.3.2 掃描線算法,規(guī)則3 掃描線與多邊形的頂點相交時,采用上開下閉及右開左閉取交點,保證交點正確配對。 檢查兩相鄰邊在掃描線的哪一側(cè)。 只要檢查頂點的兩條邊的另外兩個端點的Y值,兩個Y值中大于交點Y值的個數(shù)是0,1,2,來決定取0,1,2個交點,24,4.3.2 掃描線算法,計算交點(續(xù))水平邊不考慮排序掃描線連貫性采用插入排序交點兩兩配對與區(qū)間繪制區(qū)間連續(xù)性連續(xù)繪制區(qū)

16、間上的像素,,25,4.3.2 掃描線算法,算法實現(xiàn)-數(shù)據(jù)結(jié)構(gòu)(1)邊的分類表ET (Edge Table)(又稱新邊表)按照邊的下端點 y 坐標,對非水平邊進行分類的鏈表下端點 y 坐標值等于i 的邊屬于第i類作用:避免盲目求交,26,4.3.2 掃描線算法,ET定義每條掃描線,對應(yīng)一個鏈表鏈表中每個結(jié)點的結(jié)構(gòu),typedef struct {int ymax; float x, deltax; Edge *ne

17、xtEdge; }Edge;,ET的結(jié)點信息:ymax: 邊的上端點的y坐標值x:邊的下端點的 x 坐標deltax:邊的斜率的倒數(shù)nextEdge: 下一條邊的指針,27,4.3.2 掃描線算法,結(jié)點結(jié)構(gòu)解釋,typedef struct {int ymax; float x, deltax; Edge *nextEdge; }Edge;,float x, deltax; 用于遞推計算交點 x’=x+1/m,int

18、 ymax;當掃描線 y = e+1 == ymax,說明下一條掃描線與此邊不相交,28,4.3.2 掃描線算法,(2)活性邊表AEL(Active Edge List) 存放活性邊的順序鏈表,且按交點 x 的值從小到大排序 活性邊:與當前掃描線相交的邊 邊結(jié)構(gòu)定義:,typedef struct {int ymax; float x, deltax; Edge *nextEdge; }Edge;,

19、AEL 的結(jié)點信息:ymax: 所交邊的最高掃描線號x:當前掃描線與邊的交點的x坐標deltax:邊的斜率的倒數(shù)nextEdge: 下一條邊的指針,29,4.3.2 掃描線算法,實例,(a) Y=6對應(yīng)的活性邊表,(b) Y=7對應(yīng)的活性邊表,30,4.3.2 掃描線算法,算法(Scan-Line Algorithm)1、建立ET;2、將掃描線縱坐標y的初值置為ET中非空 元素的最小序號,如圖中,y=1;3

20、、置AEL為空;4、執(zhí)行下列步驟直至ET和AEL都為空.4.1、如ET中的第y類非空,則將其中的所有 邊取出并插入AEL中;4.2、如果有新邊插入AEL,則對AEL中各邊排序;4.3、對AEL中的邊兩兩配對,(1和2為一對,3和4為一對,…),將每對邊中x坐標按規(guī)則取整,獲得有效的填充區(qū)段,再填充.4.4、將當前掃描線縱坐標 y 值遞值1;4.5、將AEL中滿足y = ymax邊刪去(因為每條邊被看作下閉上開的)

21、;4.6、對AEL中剩下的每一條邊的x遞增deltax,即x = x+deltax.,31,4.3.2 掃描線算法,4.3.2 掃描線算法,例子,y=5 y=6 y=7 y=8,AET:y=1 y=2 y=3 y=4,4 4 -1,5 33/4 5/4 ^,5 19/2 5/4 ^,4 3 -1

22、,8 2 0,,,,,,,,,,5 43/4 5/4 ^,8 2 0,8 2 0,8 2 0,11 12 0 ^,11 12 0 ^,8 7 -5,11 7 5/4,11 12 0 ^,11 33/4 5/4,11 12 0 ^,32,4.3.2 掃描線算法,思考問題算法如何體現(xiàn)連貫性?對凸多邊形而言,算法是否可以簡化?如何簡化?對三角形而言,算法如何簡化?,

23、三角形廣泛應(yīng)用于物體建模,33,思考題,思考問題 如何高效計算平面上 n 條線段的交點?,利用掃描線和邊的連貫性,時間復雜性m×n’ (m為掃描線條數(shù), n’為與掃描線相交邊的平均條數(shù), n’ << n),一般方法:兩兩求交點,時間復雜性 n2,34,4.3.3 邊緣填充算法*,寫像素的邏輯操作主要包括:拷貝、異或(求余)等寫模型:像素的顏色與源像素及像素當前顏色相關(guān),35,4.

24、3.3 邊緣填充算法*,求余運算假定A為一個正整數(shù),則 M 的余定義為 A – M, 記為 求余運算可用異或邏輯運算實現(xiàn)例性質(zhì),36,4.3.3 邊緣填充算法*,求余運算(續(xù))應(yīng)用:光標移動、橡皮筋線和加亮菜單等操作如:交互繪制線段時的橡皮筋線,交互方式:(1)點擊鼠標左鍵輸入線段的一個端點(2)點擊右鍵(或左鍵)輸入另一端點(3)鼠標移動過程中繪制瞬時線段,注意:不能采用直接繪制(復制、拷貝)方法

25、,37,4.3.3 邊緣填充算法*,邊緣填充算法光柵圖形中,如果某區(qū)域已著上值為M的顏色值后,做偶數(shù)次求余運算,該區(qū)域顏色不變;而做奇數(shù)次求余運算,則該區(qū)域顏色變?yōu)橹禐?的顏色這一規(guī)律應(yīng)用于多邊形掃描轉(zhuǎn)換,稱為邊緣填充算法。算法基本思想對于每條掃描線和每條多邊形邊的交點,將該掃描線上交點右方的所有像素取余M為填充色,A為當前背景色,38,4.3.3 邊緣填充算法*,算法1(以掃描線為中心的邊緣填充算法)1、將當前掃

26、描線上的所有像素著上值為 顏色;2、求余: for(i=0; i<=m; i++) 在當前掃描線上,從橫坐標為xi的交點向右求余;,圖中次序:x0, x1, x2, x3,次序可以任意,39,4.3.3 邊緣填充算法*,算法2(以邊為中心的邊緣填充算法)1、將繪圖窗口的背景色置為 ;2、對多邊形的每一條非水平邊做:從該邊上的每個像素開始向右求余,40,特點適合用于具有幀緩存的圖形系

27、統(tǒng)。處理后,按掃描線順序讀出幀緩存的內(nèi)容,送入顯示設(shè)備優(yōu)點:算法簡單缺點:對于復雜圖形,每一像素可能被訪問多次,輸入/輸出的量比掃描線算法大得多,4.3.3 邊緣填充算法*,41,扇形區(qū)域的描述圓的半徑R起始角度:?1終止角度:?2原理:同掃描轉(zhuǎn)換多邊形對每條掃描線,首先計算與扇形區(qū)域邊界的交點,再用指定顏色繪制繪制配對交點間的像素問題如何確定掃描線與直線段和圓弧段的交點及相交順序?,4.4 掃描轉(zhuǎn)換扇形區(qū)域,42,方

28、法:分類按點 P1(x, y) 和P2(x, y) 點所處象限的不同,需要將扇形區(qū)域分成4×4=16種情況 假設(shè) P1 點落在第一象限 掃描線和區(qū)域邊界只有2個交點扇形區(qū)域的掃描轉(zhuǎn)換分四種情況(1) P2落在第一象限區(qū)域:OP1A和AP1P2,4.4 掃描轉(zhuǎn)換扇形區(qū)域,43,(2)P2落在第二象限,此時又分為兩種情況 當 時三個區(qū)域: OAP1、AP1BP2和P2BC當

29、 時三個區(qū)域: OAP2、AP1BP2和P1BC,4.4 掃描轉(zhuǎn)換扇形區(qū)域,44,(3) P2 落在第三象限三個區(qū)域: P1CA、BOP1A和P2OB(4) P2落在第四象限三個區(qū)域: AP1D、BOP1 A、 CP2OB和CEP2,4.4 掃描轉(zhuǎn)換扇形區(qū)域,45,遺留問題:當 P1 落在其它區(qū)域時?P1 落在第二、三、四象限時,將扇形區(qū)域繞坐標原點順時針旋轉(zhuǎn)90度

30、(180,270),使 P1 落在第1象限掃描轉(zhuǎn)換再逆時針旋轉(zhuǎn)90度(180,270),4.4 掃描轉(zhuǎn)換扇形區(qū)域,46,區(qū)域:點陣表示的圖形,像素集合表示方法:內(nèi)點表示、邊界表示內(nèi)點表示枚舉出區(qū)域內(nèi)部的所有像素內(nèi)部的所有像素為同一個顏色邊界像素與內(nèi)部像素的顏色不同邊界表示枚舉出邊界上所有的像素邊界上的所有像素為同一顏色內(nèi)部像素與邊界像素的顏色不同,4.5 區(qū)域填充,47,種子填充法對區(qū)域重新著色的過程

31、 將指定的顏色從種子點開始擴展到整個區(qū)域區(qū)域填充算法要求區(qū)域是連通的,4.5 區(qū)域填充,48,連通性4連通區(qū)域:區(qū)域中任意兩點可通過上下左右四個方向互相 到達8連通區(qū)域:區(qū)域中任意兩點可通過上下左右和對角線八 個方向互相到達,4.5 區(qū)域填充,49,4.5 區(qū)域填充,4連通與8連通區(qū)域的區(qū)別連通性: 4連通可看作8連通區(qū)域,但對邊界有要求不同依據(jù)區(qū)域內(nèi)點能否訪問到區(qū)域外的點,對邊界的要求是4連通區(qū)域,邊界只要

32、8連通即可8連通區(qū)域,邊界必須是4連通例:如左圖(1)4連通區(qū)域,邊界為 像素(2)8連通區(qū)域,邊界為 和 像素,,,,50,4.5.1 種子填充法,種子填充算法(泛濫填充:flood-fill) (1)內(nèi)點表示的4連通區(qū)域種子P(x,y),原色oldColor,新顏色newColor方法:先判斷P(x, y)的顏色,若其值不等于oldColor,說明該像素位于區(qū)域外,或已設(shè)置為newC

33、olor,算法結(jié)束;否則,置像素顏色為newColor,再對其相鄰的上下左右四個像素分別作為種子作上述遞歸處理。,void FloodFill4(int x,int y,int oldColor,int newColor){ if(GetPixel(x,y) == oldColor) { SetPixel(x,y,newColor); FloodFill4(x,y+1,oldColor,newColor);

34、 FloodFill4(x,y-1,oldColor,newColor); FloodFill4(x-1,y,oldColor,newColor); FloodFill4(x+1,y,oldColor,newColor); }}/*end of FloodFill4()*/,51,,4.5.1 種子填充法,①FloodFill4(x, y+1, oldColor, newColor);②FloodFill

35、4(x, y-1, oldColor, newColor);③FloodFill4(x-1, y, oldColor, newColor);④FloodFill4(x+1, y, oldColor, newColor);,52,(2)邊界表示的4連通區(qū)域,4.5.1 種子填充法,void BoundaryFill4(int x, int y, int oldColor, int newColor){//oldColor邊界像素顏色

36、 int color; color = GetPixel(x, y); if((color != oldColor) && (color != newColor)) { SetPixel(x, y, newColor); BoundaryFill4(x,y+1,oldColor,newColor); BoundaryFill4(x,y-1,oldColor,ne

37、wColor); BoundaryFill4(x-1,y,oldColor,newColor); BoundaryFill4(x+1,y,oldColor,newColor); }}/*end of BoundaryFill4( ) */,53,缺點有些像素需要重復判斷,降低算法效率棧結(jié)構(gòu)占空間遞歸執(zhí)行,算法簡單,但效率不高,區(qū)域內(nèi)每一像素都引起一次遞歸,進/出棧,費時費內(nèi)存改進減少遞歸次數(shù),提高效

38、率方法之一使用掃描線填充算法,4.5.1 種子填充法,54,4.5.2 掃描線算法*,掃描線算法目標:減少遞歸層次適用于內(nèi)點表示的4連通區(qū)域基本過程:當給定種子點時,首先填充種子點所在的掃描線上的位于給定區(qū)域的一個區(qū)段,然后確定與這一區(qū)段相通的上下兩條掃描線上位于給定區(qū)域內(nèi)的區(qū)段,并依次保存下來。反復這個過程,直到填充結(jié)束,55,4.5.2 掃描線算法*,算法步驟1、初始化 :將種子區(qū)段壓入堆棧2、出棧:如果堆棧為空,

39、則算法結(jié)束;否則取棧頂元素(x,y)作為種子點3、區(qū)段填充:從種子點(x, y)開始沿縱坐標為y的掃描行向左右兩個方向逐個像素填色,直到邊界為止4、定范圍:以 [xLeft, xRight]為(3)得到的區(qū)段 5、進棧:分別在與當前掃描線相鄰的上下掃描線上,確定位于區(qū)間[xLeft, xRight]內(nèi)的區(qū)段。如果區(qū)段內(nèi)的像素顏色值為newColor或boundaryColor,則轉(zhuǎn)到(2)區(qū)段壓入堆棧,轉(zhuǎn)到(2),56,例,

40、4.5.2 掃描線算法*,像素中的序號標指它所在區(qū)段位于堆棧中的位置,57,多邊形掃描轉(zhuǎn)換與區(qū)域填充方法比較基本思想不同前者:頂點表示轉(zhuǎn)換成點陣表示后者:只改變區(qū)域內(nèi)填充顏色,沒有改變表示方法對邊界的要求不同前者:掃描線與多邊形邊界交點個數(shù)為偶數(shù)后者:區(qū)域封閉,防止遞歸填充跨界基本的條件不同前者:從邊界頂點信息出發(fā)后者:區(qū)域內(nèi)種子點,4.5.3 掃描轉(zhuǎn)換與區(qū)域填充的比較,58,填充方式(1)均勻著色(2)位圖不透明(

41、3)位圖透明(4)像素圖填充在確定區(qū)域內(nèi)的像素后,查詢它對應(yīng)的位圖或圖像中的單元,再以該單元的值按填充方式的不同顯示該像素方法(1)均勻著色方法:將圖元內(nèi)部像素置成同一顏色(2)位圖不透明:若像素對應(yīng)的位圖單元為1,則以前景色顯示該像素;若為0,則以背景色顯示該像素;(3)位圖透明:若像素對應(yīng)的位圖單元為1,則以前景色顯示該像素;若為0,則不做任何處理。(4)像素圖填充:以像素對應(yīng)的像素圖單元的顏色值顯示該像素。,4.6 以

42、圖像填充區(qū)域,59,基本問題建立區(qū)域與圖像間的對應(yīng)關(guān)系方法1:建立整個繪圖空間與圖像空間的1-1映射,4.6 以圖像填充區(qū)域,圖像(紋理),,,繪圖空間,60,例 適用:動畫中漫游圖像,如透過車窗看外景,4.6 以圖像填充區(qū)域,61,方法2:建立區(qū)域局部坐標空間與圖像空間的1-1映射,4.6 以圖像填充區(qū)域,適用:圖像作為區(qū)域表面屬性的情況,例如,桌面與其上的木紋,62,字符集ASCII碼:128個字符GB23

43、12-80:漢字6763個,682個圖形符號點陣字體存儲(壓縮與非壓縮)顯示:檢索與寫緩沖矢量字體表示:筆畫組成曲線(參數(shù))掃描轉(zhuǎn)換:參數(shù)到點陣顯示:1、由編碼檢索 2、掃描轉(zhuǎn)換存儲:空間少 (windows下的仿宋體庫: SIMFANG.TTF 為3904K),4.7 字符的表示與輸出,63,4.8 OpenGL相關(guān)函數(shù),區(qū)域函數(shù)(填充作用)矩形函數(shù):glRect[i|s|f|d]{v}()glBegin()

44、中的參數(shù) GL_POLYGON、GL_TRIANGLES、 GL_TRIANGLE_STRIP、 GL_TRIANGLE_FAN三角形的三種繪制參數(shù),int p1[]={20,100},p2[]={50,50};int p3[]={110,50},p4[]={140,100};int p5[]={110,150},p6[]={50,150};glBegin(GL_TRIANGLES); glVertex2i

45、v(p1); glVertex2iv(p2); glVertex2iv(p3); glVertex2iv(p4); glVertex2iv(p5); glVertex2iv(p6);glEnd();,GL_TRIANGLE_STRIP,,,GL_TRIANGLES,GL_TRIANGLE_FAN,,p1,p1,p1,64,菜單函數(shù)int glutCreateMenu(void (*func)(int value));//

46、創(chuàng)建菜單void glutAddMenuEntry(char *name, int value);//添加菜單項void glutAddSubMenu(char *name, int menu);//添加子菜單glutAttachMenu(int button);例:,4.8 OpenGL相關(guān)函數(shù),void MenuInit(){ int sub_menu1 = glutCreateMenu(objectMenu);

47、 glutAddMenuEntry("直線",11); glutAddMenuEntry("多邊形",12); glutCreateMenu(topMenu); glutAddSubMenu("圖形",sub_menu1); glutAddMenuEntry("退出",2); glutAttachMenu(GL

48、UT_RIGHT_BUTTON);},void objectMenu(int id){ if (id==11) GraphicsType = 11; else if (id==12) GraphicsType = 12; else GraphicsType = 0;}void topMenu(int id){ if (id==2) exit(0);},65,線框圖與填充圖函數(shù):glPloygonMode(f

49、ace,displayMode)face:指定前后面??蛇x值:GL_FRONT、GL_BACK和GL_FRONT_AND_BACKdisplayMode:GL_FILL和GL_LINE,4.8 OpenGL相關(guān)函數(shù),void display (void){ glClearColor (1.0, 1.0, 1.0, 0.0); glClear (GL_COLOR_BUFFER_BIT); glPolygonMode

50、(GL_FRONT, GL_LINE); glBegin(GL_POLYGON); glColor3f(1.0,0,0); glVertex2f(0.5,0); glVertex2f(-0.5,0.5); glVertex2f(-0.5,-0.5); glVertex2f(0.6,-0.5); glEnd(); glFlush();},前向面:

51、頂點序列逆時針順序排列,66,顏色插值模式函數(shù):glShadeModel(mode)Mode: GL_FLAT和GL_SMOOTH例:,4.8 OpenGL相關(guān)函數(shù),void display (void){ glClearColor (1.0, 1.0, 1.0, 0.0); glClear (GL_COLOR_BUFFER_BIT); glShadeModel(GL_FLAT);//默認為GL_SMO

52、OTH glBegin(GL_POLYGON); glColor3f(1,0,0); glVertex2f(-0.5,0.5); glColor3f(0,1,0); glVertex2f(-0.5,-0.5); glColor3f(0,0,1); glVertex2f(0.5,-0.5); glColor3f(1,1,1); glVertex2f(0.5,0.5); glEnd();

53、 glFlush();},GL_FLAT方式下以第1個點顏色填充,67,頂點數(shù)組原因:復雜圖形需要很多坐標描述,導致函數(shù)調(diào)用量大增,影響程序的執(zhí)行效率采用頂點數(shù)組減少函數(shù)調(diào)用方法調(diào)用函數(shù)glEnableClientState(GL_VERTEX_ARRAY), 激活頂點數(shù)組特性調(diào)用glVertexPointer指定頂點坐標的位置和數(shù)據(jù)格式使用與數(shù)組相關(guān)的函數(shù)繪制圖形,4.8 OpenGL相關(guān)函數(shù),68,例:glCl

54、ear (GL_COLOR_BUFFER_BIT);glColor3f (1.0, 0.0, 0.0);vertex2 pt[]={{20,100},{50,50},{110,50},{140,100},{110,150},{50,150}};glEnableClientState(GL_VERTEX_ARRAY);glVertexPointer(2,GL_INT,0,pt);GLubyte vertIndex[]={0

55、,1,2,3,4,5};glDrawElements(GL_TRIANGLES,24,GL_UNSIGNED_BYTE, vertIndex); glFlush ( );,4.8 OpenGL相關(guān)函數(shù),69,4.8 OpenGL相關(guān)函數(shù),寫像素模式16種,默認方式為GL_COPY模式改變時,需要調(diào)用glEnable(GL_COLOR_LOGIC_OP)設(shè)置模式:glLogicOp(GL_XOR)例:異或模式,背景,

56、正方形,70,4.8 OpenGL相關(guān)函數(shù),填充圖案函數(shù)啟用函數(shù):glEnable(GL_POLYGON_STIPPLE)關(guān)閉函數(shù):glDisable( GL_POLYGON_STIPPLE)設(shè)置掩模(mask):glPolygonStipple(fillPattern) 注意:filePattern類型為GLubyte,大小為32×32,GLubyte fillPattern[]=

57、 {0x96,0x96,0x96,0x96, 0x84,0x84,0x84,0x84, 0x48,0x48,0x48,0x48, 0x20,0x20,0x20,0x20, 0x10,0x10,0x10,0x10, 0x10,0x10,0x10,0x10, 0x18,0x18,0x18,0x18, 0x64,0

58、x64,0x64,0x64, 。。。。(重復3次) },71,4.8 OpenGL相關(guān)函數(shù),OpenGL位圖函數(shù)void glRasterPos{234}{SIFD}[V](TYPE x, TYPE y, TYPE z, TYPE w);設(shè)置當前所畫位圖或圖像的原點glBitmap(width, height, x0,y0,xO

59、ffset,yOffset,bitShape)設(shè)定當前光柵位置該函數(shù)顯示二值位圖。bitShape中的元素值為0或1,當為1時,用設(shè)定的顏色繪制對應(yīng)的像素,否則對應(yīng)像素不受影響位圖的存儲模式:字節(jié)glPixelStorei (GL_UNPACK_ALIGNMENT, 1 ) ;,72,4.8 OpenGL相關(guān)函數(shù),程序,void display(){ // 最好放在初始化函數(shù)中 glPixelStorei (GL_

60、UNPACK_ALIGNMENT, 1); glColor3f (0.0, 0.0, 0.0); glRasterPos2i (100, 200); glBitmap (8, 12, 0.0, 0.0, 20.0, 20.0, rasters); glColor3f (1.0, 0.0, 0.0); glRasterPos2i (120, 220); glBitmap (8, 12,

61、 0.0, 0.0, 0.0, 0.0, rasters); glColor3f (0.0, 0.0, 1.0); glRasterPos2i (150, 200); glBitmap (8, 12, 0.0, 0.0, 0.0, 0.0, rasters); glFlush ( );},,,,,,,,,,rasters,73,4.8 OpenGL相關(guān)函數(shù),繪制字符OpenGL基本庫:位圖方法,

62、需要自己創(chuàng)建字庫GLUT工具包:位圖和輪廓字體位圖字符函數(shù):glutBitmapCharacter(font, character)font:指定字型,常量(如GLUT_BITMAP_8_BY_13)Character:字符的ASCII碼值輪廓字符函數(shù):glutStrokeCharacter(font, character)font:指定字型,常量(如GLUT_STROKE_ROMAN)Character:字符的ASCII

63、碼值位圖和輪廓字符比較前者速度快,后者放大縮小不變形,74,4.8 OpenGL相關(guān)函數(shù),程序,void display(){ glColor3f (1.0, 0.0, 0.0); glRasterPos2i (50, 50); glutBitmapCharacter (GLUT_BITMAP_9_BY_15,66); glRasterPos2i (80, 50); glutBitmapCharact

64、er (GLUT_BITMAP_TIMES_ROMAN_10,66); glColor3f (0.0, 0.0, 1.0); glPushMatrix(); glTranslatef(50,100,0);glScalef(0.2,0.2,1); glutStrokeCharacter(GLUT_STROKE_ROMAN,66); glPopMatrix(); glTranslatef(100,1

65、00,0); glScalef(0.6,0.6,1); glutStrokeCharacter(GLUT_STROKE_ROMAN,66); glutSwapBuffers();},75,小結(jié),多邊形的掃描轉(zhuǎn)換點是否在區(qū)域內(nèi)的判斷方法(射線法)掃描線算法(思想、步驟和算法實現(xiàn))多邊形的區(qū)域填充種子填充(連通性)扇形區(qū)域的掃描轉(zhuǎn)換字符的繪制方法OpenGL函數(shù),76,作業(yè),1 、多邊形掃描轉(zhuǎn)換的

66、掃描線算法利用了掃描線的連貫性和邊的連貫性,在數(shù)據(jù)結(jié)構(gòu)和算法中各體現(xiàn)在何處?2、一個多邊形的頂點坐標如下圖所示。 (a)寫出邊的分類表 (b)說明掃描線y=6,7,8,9和10時,哪些邊是活性邊。并給出活性邊表。,77,作業(yè),3、內(nèi)點表示的區(qū)域和邊界表示的區(qū)域在填充時有什么不同?4、根據(jù)三角形的特殊性,如何簡化掃描轉(zhuǎn)換三角形的掃描線算法。,78,上機題1:補充,添加填充圖元的功能填充多邊形多邊形交互輸入、顏色定義和填

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
  • 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論