本系列文章由zhmxy555編寫,轉載請注明出處。
http://blog.csdn.net/zhmxy555/article/details/7338082
透明效果
由于所有的圖檔案都是以矩形來儲存的,我們也許會需要把一張怪獸圖檔貼到視窗的背景圖上,而這種情況下如果直接進行貼圖,結果如下圖:
![]()
MFC GDI繪圖(3) 圖檔透明化 這似乎不是我們想要的結果。
為了得到透明效果,我們需要運用到BitBlt()貼圖函數以及其參數Raster的值來将圖檔中不必要的部分去掉(又稱去背),使得圖中的主題可以與背景完美融合。
制作透明效果有很多種方法,但是基本上都是利用貼圖時不同的Raster運算,通過轉換而産生相同的透明效果。在這裡先來介紹一種透明運算的方法。
我們以圖中的恐龍為例子,首先準備一張位圖,如下圖。
![]()
MFC GDI繪圖(3) 圖檔透明化 圖中的左邊的圖是要去背并貼到背景上的前景圖。右邊的黑白圖稱為“屏蔽圖”,在透明的過程中會用到它。要把去背的位圖與屏蔽圖合并成同一張圖,透明的時候再按照需要來進行裁切。可以把它分成兩張圖,但是這樣程式必須運作兩次圖檔案加載的操作。
有了屏蔽圖就可以利用貼圖函數來産生透明效果了,所需的貼圖步驟如下:
<1>将屏蔽圖與背景圖做"AND"運算,Raster值為SRCAND,貼到目的地DC中。
<2>将前景圖與背景圖做"OR"運算,Raster值為SRCPAINT,貼到目的地DC中。
為什麼經過上面兩個操作就能産生透明的效果呢?看下圖就了解了:
![]()
MFC GDI繪圖(3) 圖檔透明化 下面具體說明上面兩個步驟所産生的圖點色彩的變化。
1.屏蔽圖與背景圖做"AND"運算
<1>屏蔽圖中的黑色部分與背景圖做"AND"運算:
<2>屏蔽圖中的白色部分與背景圖做"AND"運算:![]()
MFC GDI繪圖(3) 圖檔透明化 進過這一運算所産生的結果如下圖![]()
MFC GDI繪圖(3) 圖檔透明化 ![]()
MFC GDI繪圖(3) 圖檔透明化 2.前景圖與背景圖做"OR"運算
<1>前景圖中的彩色部分與圖第一步得到的“黑色恐龍”圖做"OR"運算:
<2>前景圖中的黑色部分與第一步得到的“黑色恐龍”圖做"OR"運算:![]()
MFC GDI繪圖(3) 圖檔透明化 經過這一運算後所顯示的畫面就是所需的透明圖了,如下圖所示:![]()
MFC GDI繪圖(3) 圖檔透明化 ![]()
MFC GDI繪圖(3) 圖檔透明化 下面我們來看看實作上述透明貼圖效果的源代碼
[cpp] view plain copy print ?
![]()
MFC GDI繪圖(3) 圖檔透明化 ![]()
MFC GDI繪圖(3) 圖檔透明化 最後程式的運作結果為:
- #include "stdafx.h"
- //全局變量聲明
- HINSTANCE hInst;
- HBITMAP bg,dra; //聲明兩個位圖對象,分别存儲背景圖與前景恐龍圖
- HDC mdc; //聲明一個記憶體DC"mdc",用來暫存位圖
- //全局函數聲明
- ATOM MyRegisterClass(HINSTANCE hInstance);
- BOOL InitInstance(HINSTANCE, int);
- LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
- void MyPaint(HDC hdc);
- ****Winmain函數,程式入口點函數**************************************
- int APIENTRY WinMain(HINSTANCE hInstance,
- HINSTANCE hPrevInstance,
- LPSTR lpCmdLine,
- int nCmdShow)
- {
- MSG msg;
- MyRegisterClass(hInstance);
- if (!InitInstance (hInstance, nCmdShow))
- {
- return FALSE;
- }
- //消息循環
- while (GetMessage(&msg, NULL, 0, 0))
- {
- TranslateMessage(&msg);
- DispatchMessage(&msg);
- }
- return msg.wParam;
- }
- //****設計一個視窗類,類似填空題,使用視窗結構體*************************
- ATOM MyRegisterClass(HINSTANCE hInstance)
- {
- WNDCLASSEX wcex;
- wcex.cbSize = sizeof(WNDCLASSEX);
- wcex.style = CS_HREDRAW | CS_VREDRAW;
- wcex.lpfnWndProc = (WNDPROC)WndProc;
- wcex.cbClsExtra = 0;
- wcex.cbWndExtra = 0;
- wcex.hInstance = hInstance;
- wcex.hIcon = NULL;
- wcex.hCursor = NULL;
- wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
- wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
- wcex.lpszMenuName = NULL;
- wcex.lpszClassName = "canvas";
- wcex.hIconSm = NULL;
- return RegisterClassEx(&wcex);
- }
- //****初始化函數*************************************
- // 1.建立與視窗DC相容的記憶體DC
- // 2.從檔案加載背景圖與恐龍圖
- BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
- {
- HWND hWnd;
- HDC hdc;
- hInst = hInstance;
- hWnd = CreateWindow("canvas", "繪圖視窗" , WS_OVERLAPPEDWINDOW,
- CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
- if (!hWnd)
- {
- return FALSE;
- }
- MoveWindow(hWnd,10,10,600,450,true);
- ShowWindow(hWnd, nCmdShow);
- UpdateWindow(hWnd);
- hdc = GetDC(hWnd); //獲得視窗DC
- mdc = CreateCompatibleDC(hdc); //建立與視窗相容的記憶體DC(mdc)
- bg = (HBITMAP)LoadImage(NULL,"bg.bmp",IMAGE_BITMAP,600,450,LR_LOADFROMFILE);
- //J加載背景圖到bg中
- dra = (HBITMAP)LoadImage(NULL,"dra.bmp",IMAGE_BITMAP,170,99,LR_LOADFROMFILE);
- //加載恐龍圖到dra中
- MyPaint(hdc);
- ReleaseDC(hWnd,hdc);
- return TRUE;
- }
- //****自定義繪圖函數*********************************
- //透明貼圖
- void MyPaint(HDC hdc)
- {
- SelectObject(mdc,bg);
- BitBlt(hdc,0,0,600,450,mdc,0,0,SRCCOPY); //先将背景圖貼到顯示視窗中
- SelectObject(mdc,dra); //選用恐龍圖到"mdc"中
- BitBlt(hdc,280,320,85,99,mdc,85,0,SRCAND);//進行制作貼圖的第一步驟,即将屏蔽圖與背景圖做"AND"運算,屏蔽圖在整張恐龍圖中,最左上角起始位置點得坐标為(85,0),BitBlt()函數中最後一個Raster參數值設定為SRCAND。
- BitBlt(hdc,280,320,85,99,mdc,0,0,SRCPAINT);//進行制作透明貼圖的第二步驟,即将前景圖與背景圖做"OR"運算,前景圖在整張恐龍圖中,最左上角起始位置的坐标為(0,0),BitBlt()函數最後一個參數值設定為SRCPAINT。
- }
- //****消息處理函數**********************************
- LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
- {
- PAINTSTRUCT ps;
- HDC hdc;
- switch (message)
- {
- case WM_PAINT: //視窗重繪消息
- hdc = BeginPaint(hWnd, &ps);
- MyPaint(hdc);
- EndPaint(hWnd, &ps);
- break;
- case WM_DESTROY: //視窗結束消息
- DeleteDC(mdc);
- DeleteObject(bg);
- DeleteObject(dra);
- PostQuitMessage(0);
- break;
- default: //其他消息
- return DefWindowProc(hWnd, message, wParam, lParam);
- }
- return 0;
- }
![]()
MFC GDI繪圖(3) 圖檔透明化 通過BitBlt()貼圖函數及Raster運算值的設定,很簡單地就做出了想要的透明效果,這種方法在設計2D遊戲的一些畫面内容時使用相當頻繁。
最後我說明一個非常關鍵的問題(多謝yao050421103 的提醒),而這點由恰恰依賴于美工設計師們。我們在準備位圖資源的時候,前景圖部分絕對不能包含背景圖的顔色,否則,就不會得到我們預期的結果。
還要指出的一個地方是,前景圖的需要還原為背景色的部分一定要為黑色