利用CImage完成图片的显示、拷贝、绘制及转换图片格式

近日一个利用VC编写的程序需要在界面上显示图片,经过查阅资料利用CImage方得实现。将相关技术点整理如下,以供以后参考。

一、             图片的载入

如果需要在界面上显示的是已经存在的图片,那么需要将待显示的图片载入至CImage对象之中,CImage提供了四个载入函数:

 

HRESULT Load(LPCTSTR pszFileName) throw( );

HRESULT Load(IStream* pStream) throw();

void LoadFromResource(HINSTANCE hInstance, LPCTSTR pszResourceName) throw( );

void LoadFromResource(HINSTANCE hInstance, UINT nIDResource) throw( );

 

如果需要显示的图片在程序运行时需要更换则通常使用第一个函数Load(LPCTSTR pszFileName)载入图片,参数pszFileName指定要载入的图片文件;如果需要显示的图片是固定不变的,则通常使用第三个LoadFromResource(HINSTANCE hInstance, LPCTSTR pszResourceName)或第四个函数LoadFromResource(HINSTANCE hInstance, UINT nIDResource)载入图片,这两个函数是从资源中取得图片信息,每一个参数是包含有待载入图片的模块实例句柄,第二个参数是资源的ID或名称。

以下代码分别通过Load和LoadFromResource载入了图片:

 

CImage m_image1;        // 实际代码中,被声明成了类成员a

CImage m_image2;        // 实际代码中,被声明成了类成员

m_image1.Load("G:\\VC显示图片\\PIC\\aa.jpg");

m_image2.LoadFromResource(AfxGetInstanceHandle(), MAKEINTRESOURCE(IDB_BITMAP1));

  

二、             图片的显示

将图片载入CImage对象的目的是将其在界面之中显示出来,显示图像用到的核心函数是Draw,Draw提供了6个重载函数,通过参数名称,很容易了解参数的含义,其原型如下:

 

BOOL Draw(

   HDC hDestDC,

   int xDest,

   int yDest,

   int nDestWidth,

   int nDestHeight,

   int xSrc,

   int ySrc,

   int nSrcWidth,

   int nSrcHeight

) const throw( );

 

BOOL Draw(

   HDC hDestDC,

   const RECT& rectDest,

   const RECT& rectSrc

) const throw( );

 

BOOL Draw(

   HDC hDestDC,

   int xDest,

   int yDest

) const throw( );

 

BOOL Draw(

   HDC hDestDC,

   const POINT& pointDest

) const throw( );

 

BOOL Draw(

   HDC hDestDC,

   int xDest,

   int yDest,

   int nDestWidth,

   int nDestHeight

) const throw( );

 

BOOL Draw(

   HDC hDestDC,

   const RECT& rectDest

) const throw( );

 

如果要以控件的大小来显示图片,可以通过以下代码实现:

 

view plaincopy to clipboardprint?
if(m_image2.IsNull())    //判断有无图像  
        return;  
 
// 取得客户区尺寸  
CRect zcRect;  
GetDlgItem(IDC_STATIC_PIC2)->GetClientRect(&zcRect);  
 
// 将图像显示在界面之上  
m_image2.Draw(GetDlgItem(IDC_STATIC_PIC2)->GetDC()->m_hDC,  
                            zcRect.left,  
                            zcRect.top,  
                            zcRect.Width(),  
                            zcRect.Height()); 
if(m_image2.IsNull())    //判断有无图像
  return;

// 取得客户区尺寸
CRect zcRect;
GetDlgItem(IDC_STATIC_PIC2)->GetClientRect(&zcRect);

// 将图像显示在界面之上
m_image2.Draw(GetDlgItem(IDC_STATIC_PIC2)->GetDC()->m_hDC,
       zcRect.left,
       zcRect.top,
       zcRect.Width(),
       zcRect.Height());

 

如果希望以图像原图大小显示图像,则可以在显示图像之前通过SetWindowPos函数调整控件大小,使得图像可以以原始大小显示。以下代码演示了该情况:

 

view plaincopy to clipboardprint?
if(m_image1.IsNull())  
    return;  
 
// 将整控件调整为与图像同一尺寸   
GetDlgItem(IDC_STATIC_PIC)->SetWindowPos(NULL,  
                    0,0,m_image1.GetWidth(), m_image1.GetHeight(),   
                    SWP_NOMOVE);     
 
CRect zcRect;  
GetDlgItem(IDC_STATIC_PIC)->GetClientRect(&zcRect);  
 
m_image1.Draw(GetDlgItem(IDC_STATIC_PIC)->GetDC()->m_hDC,  
                            zcRect.left,  
                            zcRect.top,  
                            zcRect.Width(),  
                            zcRect.Height()); 
if(m_image1.IsNull())
 return;

// 将整控件调整为与图像同一尺寸
GetDlgItem(IDC_STATIC_PIC)->SetWindowPos(NULL,
     0,0,m_image1.GetWidth(), m_image1.GetHeight(),
     SWP_NOMOVE);  

CRect zcRect;
GetDlgItem(IDC_STATIC_PIC)->GetClientRect(&zcRect);

m_image1.Draw(GetDlgItem(IDC_STATIC_PIC)->GetDC()->m_hDC,
       zcRect.left,
       zcRect.top,
       zcRect.Width(),
       zcRect.Height());

 

 

当然如果不在意控件尺寸必须与图像尺寸一致的话,也可以不调整控件尺寸,而直接在调用Draw方法时以图像尺寸来设定其第四个和第五个参数。

 

注意:显示出的图片在被其它窗口遮住以后,或者当前窗口最小化后,再次显示程序界面时刚才设定的图片将踪影全无,所以必须对它们进行重绘。因此可以将显示图片的代码放在一个单独的函数中,然后在OnPaint中进行调用(不知道有没有更简单的方法,希望知道的网友告诉我)。

 

三、             图片的拷贝

也可以利用CImage的BitBlt函数实现将一个图片拷贝至另一个图片之上,其效果相当于将一张图片放置于另一张图片之上。这个功能似乎不会经常用到。函数BitBlt的原型如下:

 

BOOL BitBlt(

   HDC hDestDC,

   int xDest,

   int yDest,

   DWORD dwROP = SRCCOPY

) const throw( );

BOOL BitBlt(

   HDC hDestDC,

   const POINT& pointDest,

   DWORD dwROP = SRCCOPY

) const throw( );

BOOL BitBlt(

   HDC hDestDC,

   int xDest,

   int yDest,

   int nDestWidth,

   int nDestHeight,

   int xSrc,

   int ySrc,

   DWORD dwROP = SRCCOPY

) const throw( );

BOOL BitBlt(

   HDC hDestDC,

   const RECT& rectDest,

   const POINT& pointSrc,

   DWORD dwROP = SRCCOPY

) const throw( );

 

例如,希望将image2中存储的图像拷贝至image1中,可以通过以下代码实现:

 

m_image2.BitBlt(m_image1.GetDC(),0,0);

 

执行上述代码后,需要重新在界面绘制m_image1,否则只有在重新界面时拷贝的效果才可以显现出来。

 

四、             在内存中绘图

有时候需要在界面上动态绘图,但是如果直接在界面上绘图可能会引发闪烁的问题,可以通过先在CImage对象中绘图,然后将CImage对象显示于界面的方式降低这种感觉。假定m_image3是一个CImage对象,而我们需要画一个三角形然后将其在界面显示,可以通过以下代码完成。

 

view plaincopy to clipboardprint?
if(!m_image3.IsNull())  
{  
    m_image3.Destroy();   
}  
 
m_image3.Create(100, 100, 32);  
 
HDC hdc = m_image3.GetDC();  
CDC *pDC=new CDC;   
pDC->Attach(hdc);  
CPen penRed(PS_SOLID, 2, RGB(255,0,0));  
CPen* oldPen = pDC->SelectObject(&penRed);  
pDC->MoveTo(50, 10);  
pDC->LineTo(10,90);  
pDC->LineTo(90,90);  
pDC->LineTo(50,10);  
DeleteObject(pDC->SelectObject(oldPen));   
 
ReleaseDC(pDC);     
 
CRect zcRect;  
GetDlgItem(IDC_STATIC_PIC3)->GetClientRect(&zcRect);  
m_image3.Draw(GetDlgItem(IDC_STATIC_PIC3)->GetDC()->m_hDC,  
                    zcRect.left,  
                    zcRect.top,  
                    zcRect.Width(),  
                    zcRect.Height()); 
if(!m_image3.IsNull())
{
    m_image3.Destroy();
}

m_image3.Create(100, 100, 32);

HDC hdc = m_image3.GetDC();
CDC *pDC=new CDC;
pDC->Attach(hdc);
CPen penRed(PS_SOLID, 2, RGB(255,0,0));
CPen* oldPen = pDC->SelectObject(&penRed);
pDC->MoveTo(50, 10);
pDC->LineTo(10,90);
pDC->LineTo(90,90);
pDC->LineTo(50,10);
DeleteObject(pDC->SelectObject(oldPen));

ReleaseDC(pDC);  

CRect zcRect;
GetDlgItem(IDC_STATIC_PIC3)->GetClientRect(&zcRect);
m_image3.Draw(GetDlgItem(IDC_STATIC_PIC3)->GetDC()->m_hDC,
        zcRect.left,
        zcRect.top,
        zcRect.Width(),
        zcRect.Height());

 

五、             转换图片格式

通过CImage的Load函数可以导入多种格式的图片文件,或者我们也可以直接在内存中绘制出一张图画,这些图像信息最终可能需要转换成指定的图片压缩格式。利用CImage的Save函数完成这种转换是非常容易的。函数Save的原型如下:

 

HRESULT Save(

   IStream* pStream,

   REFGUID guidFileType

) const throw();

HRESULT Save(

   LPCTSTR pszFileName,

   REFGUID guidFileType= GUID_NULL

) const throw();

 

参数guidFileType指定了需要保存的结果类型,其允许的值及含义如下:

ImageFormatBMP   未压缩的位图

ImageFormatPNG   PNG压缩格式

ImageFormatJPEG  JPEG 压缩格式

ImageFormatGIF   GIF压缩格式

 

对于Save函数的第二种重载形式,也可以将guidFileType设定为GUID_NULL或者不设定(guidFileType将默认为GUID_NULL),此时Save函数将通过第一个参数pszFileName指定的文件名后缀的值来识别需要保存的格式。例如m_image1中导入的是JPEG格式的图片,通过以下代码,可以将其转换成后缀为.bmp的位图及后缀为.gif的GIF压缩格式图。

 

m_image1.Save("G:\\aa.bmp");

m_image1.Save("G:\\aa.gif");

 

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/ishallwin/archive/2009/11/20/4840180.aspx

转载于:https://www.cnblogs.com/okaimee/archive/2010/05/31/CImage.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值