1)
2)
3)
添加CDIB類時,如果沒有你要選的那個類,可以先隨便選個基類繼承,然后自己在代碼里把基類修改成要繼承的,把一些消息映射的注釋掉就可以了,這樣的話在建立類向導里也可以找到新建的類。
// TODO: Add your command handler code here
static char szFilter[]="BMP文件(*.bmp)|*bmp||";
//定義過濾文件的類型
CFileDialog dlg(TRUE,"bmp",NULL,OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT,sz
Filter); //定義對話框對象
if(ret==IDOK)
{
filename=dlg.GetFileName(); //獲取所選擇圖像的路徑
m_dib.LoadFromFile(filename);//加載圖像
if(!m_dib.m_bLoaded) //判斷是否加載圖像成功
{
AfxMessageBox(圖像打不開);
return;
}
}
Invalidate(1);
Visual C++ MFC中沒有提供一個專門的類來處理DIB位圖,因此,為了方便地使用位圖文件,我們有必要派生一個CDib類。類的源代碼如下:
(1) CDib類的聲明
// DIB.h:類CDib聲明頭文件
#ifndef __DIB_H__
#define __DIB_H__
#include <wingdi.h>
class CDib
{
public:
CDib();
~CDib();
BOOL Load( const char * );
BOOL Save( const char * );
BOOL Draw( CDC *, int nX = 0, int nY = 0, int nWidth = -1, int nHeight = -1, int mode = SRCCOPY);
BOOL SetPalette( CDC * );
private:
CPalette m_Palette;
unsigned char *m_pDib, *m_pDibBits;
DWORD m_dwDibSize;
BITMAPINFOHEADER *m_pBIH;
RGBQUAD *m_pPalette;
int m_nPaletteEntries;
};
#endif
(2) CDib類的實現
// DIB.cpp:類CDib實現文件
#include "stdafx.h"
#include "DIB.h"
CDib::CDib()
{
m_pDib = NULL;
}
CDib::~CDib()
{
// 如果位圖已經被加載,釋放內存
if (m_pDib != NULL)
delete []m_pDib;
}
下面這個函數非常重要,其功能為加載位圖,類似於CBitmap類的LoadBitmap函數:
BOOL CDib::Load(const char *pszFilename)
{
CFile cf;
// 打開位圖文件
if (!cf.Open(pszFilename, CFile::modeRead))
return (FALSE);
// 獲得位圖文件大小,並減去BITMAPFILEHEADER的長度
DWORD dwDibSize;
dwDibSize = cf.GetLength() - sizeof(BITMAPFILEHEADER);
// 為DIB位圖分配內存
unsigned char *pDib;
pDib = new unsigned char[dwDibSize];
if (pDib == NULL)
return (FALSE);
BITMAPFILEHEADER BFH;
// 讀取位圖文件數據
try
{
// 文件格式是否正確有效
if ( cf.Read(&BFH, sizeof(BITMAPFILEHEADER)) != sizeof(BITMAPFILEHEADER) ||
BFH.bfType != ’MB’ || cf.Read(pDib, dwDibSize) != dwDibSize)
{
delete []pDib;
return (FALSE);
}
}
catch (CFileException *e)
{
e->Delete();
delete []pDib;
return (FALSE);
}
// delete先前加載的位圖
if (m_pDib != NULL)
delete m_pDib;
// 將臨時Dib數據指針和Dib大小變量賦給類成員變量
m_pDib = pDib;
m_dwDibSize = dwDibSize;
// 為相應類成員變量賦BITMAPINFOHEADER和調色板指針
m_pBIH = (BITMAPINFOHEADER*)m_pDib;
m_pPalette = (RGBQUAD*) &m_pDib[sizeof(BITMAPINFOHEADER)];
// 計算調色板中實際顏色數量
m_nPaletteEntries = 1 << m_pBIH->biBitCount;
if (m_pBIH->biBitCount > 8)
m_nPaletteEntries = 0;
else if (m_pBIH->biClrUsed != 0)
m_nPaletteEntries = m_pBIH->biClrUsed;
// 為相應類成員變量賦image data指針
m_pDibBits = &m_pDib[sizeof(BITMAPINFOHEADER) + m_nPaletteEntries * sizeof (RGBQUAD)];
// delete先前的調色板
if (m_Palette.GetSafeHandle() != NULL)
m_Palette.DeleteObject();
// 如果位圖中存在調色板,創建LOGPALETTE 及CPalette
if (m_nPaletteEntries != 0)
{
LOGPALETTE *pLogPal = (LOGPALETTE*)new char[sizeof(LOGPALETTE) + m_nPaletteEntries *sizeof(PALETTEENTRY)];
if (pLogPal != NULL)
{
pLogPal->palVersion = 0x300;
pLogPal->palNumEntries = m_nPaletteEntries;
for (int i = 0; i < m_nPaletteEntries; i++)
{
pLogPal->palPalEntry[i].peRed = m_pPalette[i].rgbRed;
pLogPal->palPalEntry[i].peGreen = m_pPalette[i].rgbGreen;
pLogPal->palPalEntry[i].peBlue = m_pPalette[i].rgbBlue;
}
//創建CPalette並釋放LOGPALETTE的內存
m_Palette.CreatePalette(pLogPal);
delete []pLogPal;
}
}
return (TRUE);
}
//函數功能:保存位圖入BMP文件
BOOL CDib::Save(const char *pszFilename)
{
if (m_pDib == NULL)
return (FALSE);
CFile cf;
if (!cf.Open(pszFilename, CFile::modeCreate | CFile::modeWrite))
return (FALSE);
try
{
BITMAPFILEHEADER BFH;
memset(&BFH, 0, sizeof(BITMAPFILEHEADER));
BFH.bfType = ’MB’;
BFH.bfSize = sizeof(BITMAPFILEHEADER) + m_dwDibSize;
BFH.bfOffBits = sizeof(BITMAPFILEHEADER) +
sizeof(BITMAPINFOHEADER) + m_nPaletteEntries *sizeof(RGBQUAD);
cf.Write(&BFH, sizeof(BITMAPFILEHEADER));
cf.Write(m_pDib, m_dwDibSize);
}
catch (CFileException *e)
{
e->Delete();
return (FALSE);
}
return (TRUE);
}
下面這個函數也非常重要,其功能為在pDC指向的CDC中繪制位圖,起點坐標為(nX,nY),繪制寬度和高度為nWidth、nHeight,最后一個參數是光柵模式:
BOOL CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)
{
if (m_pDib == NULL)
return (FALSE);
// 獲取位圖寬度和高度賦值
if (nWidth == - 1)
nWidth = m_pBIH->biWidth;
if (nHeight == - 1)
nHeight = m_pBIH->biHeight;
// 繪制位圖
StretchDIBits(pDC->m_hDC, nX, nY, nWidth, nHeight, 0, 0, m_pBIH->biWidth, m_pBIH->biHeight, m_pDibBits, (BITMAPINFO*)m_pBIH, BI_RGB, mode);
return (TRUE);
}
//函數功能:設置調色板
BOOL CDib::SetPalette(CDC *pDC)
{
if (m_pDib == NULL)
return (FALSE);
// 檢查當前是否有一個調色板句柄,對於大於256色的位圖,為NULL
if (m_Palette.GetSafeHandle() == NULL)
return (TRUE);
// 選擇調色板,接着實施之,最后恢復老的調色板
CPalette *pOldPalette;
pOldPalette = pDC->SelectPalette(&m_Palette, FALSE);
pDC->RealizePalette();
pDC->SelectPalette(pOldPalette, FALSE);
return (TRUE);
}
從整個CDib類的代碼中我們可以看出,DIB位圖的顯示需遵循如下步驟:
(1)讀取位圖,本類中使用pDib = new unsigned char[dwDibSize]為位圖中的信息分配內存,另一種方法是調用API函數CreateDIBSection,譬如:
m_hBitmap = ::CreateDIBSection(pDC->GetSafeHdc(),
(LPBITMAPINFO) m_lpBMPHdr, DIB_RGB_COLORS,
(LPVOID*) &m_lpDIBits, NULL, 0);
m_hBitmap定義為:
HBITMAP m_hBitmap;
(2)根據讀取的位圖信息,計算出調色板大小,然后創建調色板;
(3)調用CDib::SetPalette( CDC *pDC )設置調色板,需要用到CDC::SelectPalette及CDC::RealizePalette兩個函數;
(4)調用CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)函數繪制位圖。在此函數中,真正發揮顯示位圖作用的是對StretchDIBits API函數的調用。StretchDIBits函數具有縮放功能,其最后一個參數也是光柵操作的模式。
下面給出DIB位圖的打開及顯示並在其中加入天極網logo的函數源代碼。"DIB位圖"父菜單下"打開"子菜單的單擊事件消息處理函數為(其功能為打開位圖並顯示之):
void CBitMapExampleDlg::OnOpendibpic()
{
// 彈出文件對話框,讓用戶選擇位圖文件
CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL,"位圖文件(*.BMP)|*.bmp;*.BMP|");
if (IDOK == fileDialog.DoModal())
{
// 加載位圖並顯示之
CDib dib;
if (dib.Load(fileDialog.GetPathName()))
{
CClientDC dc(this);
dib.SetPalette(&dc);
dib.Draw(&dc);
}
}
}
"DIB位圖"父菜單下"標記"子菜單的單擊事件消息處理函數為(其功能為給位圖加上天極網logo):
void CBitMapExampleDlg::OnMarkDibpic()
{
// 彈出文件對話框,讓用戶選擇標記logo
CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL, "標記位圖文件(*.BMP)|*.bmp;*.BMP|");
if (IDOK == fileDialog.DoModal())
{
// 加載標記logo位圖並與目標位圖相與
CDib dib;
if (dib.Load(fileDialog.GetPathName()))
{
CClientDC dc(this);
dib.SetPalette(&dc);
dib.Draw(&dc, 0, 0, - 1, - 1, SRCAND);
}
}