转 解决CImage加载png背景变黑

阅读: 评论:0

  VC7.1 MFC图形处理类里有一个强大的成员---CImage,这个类提供了从外部磁盘中调入一个JPEGGIFBMPPNG格式的图像文件加以显示,而 且这些文件格式可以相互转换。由于CImage在不同的Windows操作系统中其某些性能是不一样的,因此在使用时要特别注意。例 如,CImage::PlgBltCImage::MaskBlt只能在 Windows NT 4.0 或更高版本中使用,但不能运行在Windows 95/98 应用程序中。CImage::AlphaBlendCImage::TransparentBlt也只能在 Windows 2000/98或其更高版本中使用。即使在Windows 2000运行程序还必须将stdafx.h文件中的WINVER_WIN32_WINNT的预定义修改成0x0500才能正常使用
使用CImage的一般方法

  使用CImage的一般方法是这样的过程:

  (1) 打开应用程序的stdafx.h文件添加CImage类的包含文件:


#include atlimage.h

  (2) 定义一个CImage类对象,然后调用CImage::Load方法装载一个外部图像文件。
烷基铝
  (3) 调用CImage::Draw方法绘制图像。Draw方法具有如下定义:

BOOL Draw( HDC hDestDC, int xDest, int yDest,
int nDestWidth, int nDestHeight, int xSrc, int ySrc,
int nSrcWidth, int nSrcHeight );
BOOL Draw( HDC hDestDC, const RECT& rectDest, const RECT& rectSrc );
BOOL Draw( HDC hDestDC, int xDest, int yDest );
BOOL Draw( HDC hDestDC, const POINT& pointDest );
BOOL Draw( HDC hDestDC, int xDest, int yDest,
int nDestWidth, int nDestHeight );
BOOL Draw( HDC hDestDC, const RECT& rectDest );

   其中,hDestDC用来指定绘制的目标设备环境句柄,(xDest, yDest)pointDest用来指定图像显示的位置,这个位置和源图像的左上角点相对应。nDestWidthnDestHeight分别指定图 像要显示的高度和宽度,xSrcySrcnSrcWidthnSrcHeight用来指定要显示的源图像的某个部分所在的位置和大小。 rectDestrectSrc分别用来指定目标设备环境上和源图像所要显示的某个部分的位置和大小。

  需要说明的是,Draw方法 综合了StretchBltTransparentBltAlphaBlend函数的功能。默认时,Draw的功能和StretchBlt相同。但当 图像含有透明或Alpha通道时,它的功能又和TransparentBltAlphaBlend相同。因此,在一般情况下,我们都应该尽量调用 CImage::Draw方法来绘制图像。
现在简单介绍一下CImage图像的提取和显示,这个在很多的网站上都有讲述,随便都能到的东西,我这里就不再详述,随便了一个代码
(1) 创建一个默认的单文档程序项目Ex_Image


  (2) 打开stdafx.h文件中添加CImage类的包含文件atlimage.h

  (3) CEx_ImageView类添加异氰酸甲酯ID_FILE_OPENCOMMAND事件映射程序,并添加下列代码:

void CEx_ImageView::OnFileOpen()
{
模具石膏粉 CString strFilter;
 CSimpleArray秸秆炭化GUID aguidFileTypes;
 HRESULT hResult;

 // 获取CImage支持的图像文件的过滤字符串
 hResult = m_Image.GetExporterFilterString(strFilter,aguidFileTypes,
_T( "All Image Files") );
 if (FAILED(hResult)) {
  MessageBox("GetExporterFilter调用失败!");
  return;
 }
 CFileDialog dlg(TRUE, NULL, NULL, OFN_FILEMUSTEXIST, strFilter);
 if(IDOK != dlg.DoModal())
  return;

 m_Image.Destroy();
 // 将外部图像文件装载到CImage对象中
 hResult = m_Image.Load(dlg.GetFileName());
 if (FAILED(hResult)) {
  MessageBox("调用图像文件失败!");
  return;
 }

 // 设置主窗口标题栏内容
 CString str;
 str.LoadString(AFX_IDS_APP_TITLE);
 AfxGetMainWnd()-SetWindowText(str + " - " +dlg.GetFileName());

 Invalidate(); // 强制调用OnDraw
}

  (4) 定位到CEx_ImageView::OnDraw函数处,添加下列代码:

void CEx_ImageView::OnDraw(CDC* pDC)
{
 CEx_ImageDoc* pDoc = GetDocument();
 ASSERT_VALID(pDoc);
 if (!m_Image.IsNull()) {
  m_Image.Draw(pDC-电视升降机>m_hDC,0,0);
 }
}

  (5) 打开Ex_ImageView.h文件,添加一个公共的成员数据m_Image

public:
CImage m_Image;

  顶空瓶(6) 编译并运行。单击"打开"工具按钮,在弹出的对话框中指定一个图像文件后,单击"打开"按钮,其结果如图7.21所示。
如果这个时候加载透明图层,一般情况是不会显示透明背景的(我想能看到这里的朋友都是为透明所困的人吧),我也是卡在这里很长时间,查阅了大量的资料和网上跑了很多网站,都没有得到满意的答复,以下有些网站资料可以参考:
供CImage类显示的半透明PNG文件处理方法
在高兴之余,也很不理解
使用CImage类如何显示背景是透明的png格式图片
在网上搜索的结果让人感到很失望,我也开始思考是不是要用其他的方法来处理图片的透明,因为我原来写 游戏都一直使用SDK的,在9.0SDK中绘制透明PNG格式是支持透明的,为什么在2003里他就不支持了呢?我觉得这个问题并不是 CImage本身的处理能力问题,而是有的地方我们并设置好。先做一个实验:我调入一张带有透明背景图片,然后通过CImage保存成新的一张PNG,用 PHOTOSHOP打开发现背景依然是透明的,这证明我的猜想是正确的,但是为什么就是不显示透明呢?我开始反复查阅MSDN,试图到问题的根本原因。
    MSDN种介绍CImage::Draw 的段落并不多,如果在编译应用程序时将 _WIN32_WINNT 的值设置为等于或大于 0x0500,则 Draw 将在运行 Windows 2000 Windows 98 以及更高版本的系统上自动处理透明。它在 Windows NT 4.0 Windows 95 上同样有效,但没有透明支持。我们现用的系统是XP,按照MSDN说的,应该是自动处理透明,为什么我们就不自动呢?其中一句话引起了我的注意  如果编译应用程序时将 _WIN32
_WINNT 的值设置为小于 0x0500,则 Draw 有效,但它在运行 Windows 2000 Windows 98 以及更高版本的系统上将不自动处理透明  这句话在本文开篇的时候我已经用蓝标注出来了,这行字是Draw是否自动处理透明的关键,但是我早就把_WIN32_WINNT改成0x0500了,为什么还不行呢?我又重新查看了stdafx.h中需要修改的部分:
WINVER        // 允许使用 Windows 95 Windows NT 4 或更高版本的特定功能。
_WIN32_WINNT  // 允许使用 Windows NT 4 或更高版本的特定功能。
_WIN32_WINDOWS // 允许使用 Windows 98 或更高版本的特定功能。
这里的3个版本相关的内容,但是MSDN上只让我们修改了一出处,是不是问题出在这里呢?修改一下就知道了。
#ifndef WINVER    // 允许使用 Windows 95 Windows NT 4 或更高版本的特定功能。
#define WINVER 0x0500  // Windows98 Windows 2000 及更新版本改变为适当的值。
#endif
#ifndef _WIN32_WINNT  // 允许使用 Windows NT 4 或更高版本的特定功能。
#define _WIN32_WINNT 0x0500 // Windows98 Windows 2000 及更新版本改变为适当的值。
#endif     
#ifndef _WIN32_WINDOWS  // 允许使用 Windows 98 或更高版本的特定功能。
#define _WIN32_WINDOWS 0x0510 // Windows Me 及更新版本改变为适当的值。
#endif
编译原来的程序通过,打开带有半透明背景的图片,马上显示背面的网格,OK!问题到了!打开透明的 怎么样呢?结果令人失望,大片的透明背景被绘制成了白背景,似乎还是没有解决好啊,再看看MSDN,里面有一个TransparentBlt绘制透明, 他不是把透明变成白的了吗?TransparentBlt里面的关键改成白绘制,哈哈,透明的图片出来了。

本文发布于:2023-08-11 22:04:35,感谢您对本站的认可!

本文链接:https://patent.en369.cn/patent/4/214276.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

标签:透明   显示   处理   背景   使用
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 369专利查询检索平台 豫ICP备2021025688号-20 网站地图