1. 程式人生 > >VC++6.0進行數字影象處理的步驟以及遇到的問題

VC++6.0進行數字影象處理的步驟以及遇到的問題

 

 

 

 

 

 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);
  }
 }