如果你是一個使用VB編程的程序員,要在程序中顯示JPG或者GIF圖像簡直易如反掌,將圖像控件拖到Form中,分分鐘即可搞掂。但是C++程序員要顯示同樣的圖形卻沒有那么輕松,那么是不是要自己編寫JPG解壓縮代碼呢?當然不用那么復雜啦!本文將針對這個問題討論如何在MFC中顯示JPG或者GIF圖像。
用VB寫圖像顯示程序之所以如此輕松,完全是利用了琳瑯滿目的圖像處理控件,把你想要做的事情都一一搞掂。而C++程序員為了實現(xiàn)相同的功能必須忙乎半天。其實,C/C++程序員也能使用那些VB程序員所用的(或者說幾乎一樣的)圖像控件。VB用的圖像控件實際上都基于一個系統(tǒng)級COM類——IPicture。下面是有關 IPicture 的方法描述:
方法 | 描述 |
get_Handle | 返回圖像對象的Windows GDI句柄 |
get_Hpal | 返回圖像對象當前使用的調(diào)色板拷貝 |
get_Type | 返回當前圖像對象的的圖像類型 |
get_Width | 返回當前圖像對象的圖像寬度 |
get_Height | 返回當前圖像對象的圖像高度 |
Render | 在指定的位置、指定的設備上下文上繪制指定的圖像部分 |
set_Hpal | 設置當前圖像的調(diào)色板 |
get_CurDC | 返回當前選中這個圖像的設備上下文 |
SelectPicture | 將一個位圖圖像選入給定的設備上下文,返回選中圖像的設備上下文和圖像的GDI句柄 |
get_KeepOriginalForma | 返回圖像對象KeepOriginalFormat 屬性的當前值 |
put_KeepOriginalFormat | 設置圖像對象的KeepOriginalFormat 屬性 |
PictureChanged | 通知圖像對象它的圖像資源改變了 |
SaveAsFile | 將圖像數(shù)據(jù)存儲到流中,格式與存成文件格式相同 |
get_Attributes | 返回圖像位屬性當前的設置 |
從上面這個表可以看出,IPicture操縱著圖像對象及其屬性。圖像對象提供對位圖的抽象,而Windows負責BMP、JPG和GIF位圖的標準實現(xiàn)。程序員要做的只是實例化IPicture,然后調(diào)用其Render函數(shù)。與通常使用接口的方式不同,這里實例的創(chuàng)建我們不用CoCreateInstance函數(shù),而是用一個專門的函數(shù)OleLoadPicture。
IStream* pstm = // 需要一個流(stream)IPicture* pIPicture;hr = OleLoadPicture(pstm, 0, FALSE, IID_IPicture, (void**)&pIPicture);OleLoadPicture從流中加載圖像并創(chuàng)建一個可用來顯示圖像的新IPicture對象。
rc = // 顯示圖像的矩形// 將rc 轉換為 HIMETRICspIPicture->Render(pDC, rc);IPicture 負責處理所有瑣事,以便確定圖形之格式,如 Windows 位圖、JPEG或者GIF文件——甚至是圖標和元文件(metafiles)。當然啦,所有這些的實現(xiàn)細節(jié)是需要技巧的,為此我寫了一個Demo程序Myimgapp(如圖二)來示范這些IPicture的使用方法。
CPicture pic(ID_MYPIC); // 加載圖像 CRect rc(0,0,0,0); // 使用缺省的rc pic.Render(pDC, rc); // 顯示圖像CPicture::Render提供一個顯示圖片的矩形。IPicture 對圖像進行延伸處理。如果傳遞一個空矩形,則CPicture用圖像本身的大小--不進行延伸處理。對于圖像本身而言,CPicture查找"IMAGE"類型的資源,所以在資源文件中你必須要加入下面的代碼:
IDR_MYPIC IMAGE MOVEABLE PURE "res\\MyPic.jpg"CPicture是個很棒的傻瓜類,它具備一個 ATL 智能指針CComQIPtr
class CPictureDoc : public CDocument {protected: CPicture m_pict; // the picture};并且CPictureDoc::Serialize 調(diào)用CPicture::Load 從MFC存檔的數(shù)據(jù)中讀取圖像。
void CPictureDoc::Serialize(CArchive& ar){ if (ar.IsLoading()) { m_pict.Load(ar); }}為了使Myimgapp程序更實用,CPictureDoc::OnNewDocument從程序資源數(shù)據(jù)加載了一幅圖像。為了顯示這幅圖像,CPictureView::OnDraw要調(diào)用CPicture::Render。這樣程序一啟動便會顯示一幅默認的圖像。
void CPictureView::OnDraw(CDC* pDC){ CPictureDoc* pDoc = GetDocument(); CPicture* ppic = pDoc->GetPicture(); CRect rc; GetImageRect(rc); ppic->Render(pDC,rc);}GetImageRect是CPictureView類的一個成員函數(shù),作用是根據(jù)當前Myimgapp的縮放比率(可用25%、33%、50%、75%、100%或自適應方式)獲取圖像矩形。GetImageRect調(diào)用CPicture::GetImageSize來獲得真正的圖像大小,然后根據(jù)比率顯示。 CPictureView其余的部分完全和CScrollView的做法差不多,初始化視圖并設置滾動大小,處理命令等等。唯一讓人操心的是IPicture::Render中HIMETRIC的處理問題,因為標準的MFC應用程序都使用MM_TEXT映射模型。不用擔心,CPicture::Render和CPicture::GetImageSize會將這一切轉換過來,所以你不必為這些事情傷神。 CPictureView有一個消息處理器值得一提:它就是OnEraseBkgnd,當要顯示的圖像比客戶區(qū)小的時候,這個函數(shù)必須繪制空白區(qū)域,如圖二,OnEraseBkgnd創(chuàng)建一個與圖像大小相等的切邊(clip)矩形,然后將客戶區(qū)填成黑色。之所以要創(chuàng)建切邊矩形,主要是避免當改變窗口大小時出現(xiàn)的抖動——FillRect不繪制切邊矩形內(nèi)的區(qū)域,此乃Windows圖形處理的常識。
class CAboutDialog : public CDialog {protected: CPictureCtrl m_wndPict; virtual BOOL OnInitDialog();};BOOL CAboutDialog::OnInitDialog(){ m_wndPict.SubclassDlgItem(IDC_MYIMAGE,this); return CDialog::OnInitDialog();}假設你的對話框中有一個靜態(tài)控制,它的ID=IDC_IMAGE,并且有一幅IMAGE資源的ID與之相同。則從CStaticLink派生出的CPictureCtrl還可以指定一個URL超鏈接(或者創(chuàng)建一個ID與此控制或圖像的ID相同的串資源)。如果你指定了一個URL,則在圖像上單擊鼠標將啟動默認瀏覽器訪問URL。真是酷呆了。CPicture控制著CPicture對象并改寫WM_PAINT消息處理例程,調(diào)用CPicture::Render代替通常的靜態(tài)控制處理例程。處理細節(jié)請參見代碼。打開Myimgapp程序的“關于”對話框就知道了。