正文

GDI圖形編程(14)

Windows移動(dòng)游戲開發(fā)實(shí)戰(zhàn) 作者:(美)Adam Dawes


 

程序清單3-11  繪制多幀動(dòng)畫中單獨(dú)的一幀

// A variable to store the current animation frame.

int animFrame = 0;

Private void Form1_Paint(object sender, PaintEventArgs e)

{

const int frameWidth = 75; // The width of each animation frame

const int frameHeight = 75; // The height of each animation frame

// Create the source rectangle.

// This will have a width and height of 75 (the size of our animationframe).

// The x-coordinate will specify the position within the source image 

// from which we want to copy. Multiplying the animation frame number by 

the frame width 

// results in a coordinate at the left of the frame that we are going to 

// copy.

Rectangle srcRect = new Rectangle(animFrame * frameWidth, 0, frameWidth, frameHeight);

// Draw the bitmap at coordinate (100, 100)

e.Graphics.DrawImage(myBitmap, 100, 100, srcRect, GraphicsUnit.Pixel);

// Move to the next animation frame for the next Paint

animFrame += 1;

}

接下來對(duì)函數(shù)DrawImage中傳遞的參數(shù)進(jìn)行解釋:

● 第一個(gè)參數(shù)為需要在屏幕上進(jìn)行繪制的Bitmap對(duì)象(myBitmap)。

● 接下來指定圖像所要顯示的位置,用x坐標(biāo)和y坐標(biāo)的形式,在本例中為(100,100)。

● 接下來再指定源圖像中要繪制的區(qū)域的位置及大小(srcRect)。

● 與該函數(shù)的桌面版一樣,其.NET CF版也讓我們提供GraphicsUnit來測量Rectangles。由于現(xiàn)在的.NET CF版本中只支持像素,所以我們只能將GraphicsUnit.Pixel作為該參數(shù)的值。

2. 縮放

除了能夠通過一個(gè)矩形來讀取指定源圖像的某個(gè)區(qū)域,我們還可以提供另一個(gè)矩形來指定繪制目標(biāo)區(qū)域,而不像以前那樣簡單地使用坐標(biāo)。如果目標(biāo)矩形與原始矩形大小不同,那么位圖會(huì)被縮放來適應(yīng)目標(biāo)矩形的尺寸。

這個(gè)功能具有潛在的作用,但是,對(duì)源位圖進(jìn)行縮放是個(gè)相當(dāng)慢的操作,可能會(huì)影響游戲的運(yùn)行速度。偶爾適當(dāng)?shù)厥褂靡幌驴s放會(huì)產(chǎn)生有用的效果,但不能將它作用于大量的圖像上,否則游戲的幀率會(huì)大幅下降。

要使用該功能的話,需要像指定源Rectangle對(duì)象和目標(biāo)Rectangle對(duì)象,如程序清單3-12所示。如果兩個(gè)矩形尺寸不一致,那么圖像就會(huì)根據(jù)情況進(jìn)行縮放。

程序清單3-12  將位圖的寬與高放大到原來的兩倍

private void Form1_Paint(object sender, PaintEventArgs e)

{

// Create the source rectangle to match the size of the bitmap.

Rectangle srcRect = new Rectangle(0, 0, myBitmap.Width, myBitmap.Height);

// Create the destination rectangle at double the size of the source rectangle.

Rectangle destRect = new Rectangle(100, 100, srcRect.Width * 2, srcRect.Height * 3);

// Draw the bitmap

e.Graphics.DrawImage(myBitmap, destRect, srcRect, GraphicsUnit.Pixel);

}

3. 顏色鍵

DrawImage函數(shù)最后要提到的這個(gè)功能可能是最重要的一個(gè)功能:繪制圖像時(shí)使圖像的某個(gè)區(qū)域透明。我們現(xiàn)在為止所看到的示例都是將圖像完全不變地復(fù)制到屏幕的矩形區(qū)域中,將該區(qū)域原來的東西完全覆蓋。絕大部分情況下,我們都會(huì)希望要繪制的圖像與已存在的圖像重疊時(shí),不要出現(xiàn)那些矩形的空白。

圖3-14展示了DrawImage函數(shù)標(biāo)準(zhǔn)的繪圖行為。您可以看到第二個(gè)圓形圖像的左側(cè)將第一個(gè)圖像剪切出了一個(gè)空白區(qū)域。而在右邊的圖中,這兩個(gè)圓互不妨礙地重疊在一起。

 

圖3-14  在繪圖時(shí)沒有采用顏色鍵的效果(左圖)與采用顏色鍵的效果(右圖)

使用一個(gè)顏色鍵可以得到圖3-14右圖的效果,我們將指定源圖像中的某一種想要使之成為透明的顏色,然后將該顏色傳遞給DrawImage函數(shù)。在屏幕上繪圖時(shí)任何一個(gè)與該顏色匹配的像素都會(huì)變?yōu)橥该?。這些透明的像素不只會(huì)像本例中那樣只是出現(xiàn)在圖像的外部,它們可以出現(xiàn)在任何需要進(jìn)行透明處理的地方。因此,您需要選擇一個(gè)圖像中還沒有使用的顏色。

顏色鍵只能使位圖像素完全透明(對(duì)于那些與顏色鍵相匹配的像素)或者完全不透明(對(duì)于所有其他像素)。它不支持GDI提供的透明度或者半透明繪制(在第10章介紹OpenGL ES時(shí)您就會(huì)看到在繪圖中如何處理透明度)。

要指定顏色鍵,需要?jiǎng)?chuàng)建一個(gè)ImageAttributes對(duì)象。這又是一個(gè)桌面版.NET類的精簡版本,實(shí)際上,該類中只保留了SetColorKey和ClearColorKey這兩個(gè)有用的函數(shù)。

通過調(diào)用SetColorKey函數(shù)來指定要透明化的顏色。您會(huì)發(fā)現(xiàn)該函數(shù)實(shí)際要接受兩個(gè)Color類的參數(shù),參數(shù)名分別為colorLow和colorHigh。其原因是要與其桌面版.NET函數(shù)保持一致,在桌面版的.NET函數(shù)中可以指定一個(gè)顏色范圍。然而,.NET CF只支持單個(gè)顏色,所以要為這兩個(gè)參數(shù)指定同一個(gè)顏色。程序清單3-13向屏幕繪制圖像時(shí)令所有白色的像素為透明。


上一章目錄下一章

Copyright ? 讀書網(wǎng) ranfinancial.com 2005-2020, All Rights Reserved.
鄂ICP備15019699號(hào) 鄂公網(wǎng)安備 42010302001612號(hào)