VC6.0使用GDI+的方法及初始化工作

时间:2021-05-18 03:55:57

GDI+是Windows XP中的一个子系统,它主要负责在显示屏幕和打印设备输出有关信息,它是一组通过C++类实现的应用程序编程接口。顾名思义,GDI+是以前版本GDI的继承者,出于兼容性考虑,Windows XP仍然支持以前版本的GDI,但是在开发新应用程序的时候,开发人员为了满足图形输出需要应该使用GDI+,因为GDI+对以前的Windows版本中GDI进行了优化,并添加了许多新的功能。

     遗憾的是,VC6.0并不能直接使用GDI+,下面介绍一种很方便的在VC6.0中使用GDI+的方法。步骤如下:

1.将GDI+.rar将其中的Includes和Lib中的文件拷到vc目录下的Includes和Lib文件夹中;

  GDI+ for VC6.0 SDK下载地址:

  http://www.codeguru.com/code/legacy/gdi/GDIPlus.zip

2.在你将要使用GDI+的工程中,完成初始化工作:

将以下代码加入StdAfx.h头文件中:

//加入的头文件
#include <comdef.h>//初始化一下com接口

#ifndef ULONG_PTR
#define ULONG_PTR unsigned long*
#include "GdiPlus.h"
using namespace Gdiplus;
#endif
//结束


3.在程序初始化部分加入

//gdi+用到的两个变量
    GdiplusStartupInput m_gdiplusStartupInput;
    ULONG_PTR m_pGdiToken;
    //装载gdi+
    GdiplusStartup(&m_pGdiToken,&m_gdiplusStartupInput,NULL);

4在程序退出部分前加入

    //卸载gdi+
            GdiplusShutdown(m_pGdiToken);

5在Project->stting->Link->Object/libary中加入gdiplus.lib

这样你的工程就可以使用GDI+了,没什么技术含量吧!

 

GDI+编程基础(一)GDI+ Vs GDI

作者:李昊

下载源代码

一、GDI
  GDI是位于应用程序与不同硬件之间的中间层,这种结构让程序员从直接处理不同硬件的工作中解放出来,把硬件间的差异交给了GDI处理。GDI通过将应用程序与不同输出设备特性相隔离,使Windows应用程序能够毫无障碍地在Windows支持的任何图形输出设备上运行。例如,我们可以在不改变程序的前提下,让能在Epson点式打印机上工作的程序也能在激光打印机上工作。它把windows系统中的图形输出转换成硬件命令然后发送给硬件设备。GDI是以文件的形式存储在系统中,系统需要输出图形时把它载入内存,如果转换成硬件命令时遇到非GDI命令,系统还可能载入硬件驱动程序,驱动程序辅助GDI把图形命令转换成硬件命令。

 

二、设备环境
  Windows系统是用来给应用程序提供设备独立性的工具,它是windows系统为了处理输出设备而使用的一种内部数据结构,设备环境是windos程序,驱动程序,和输出设备(如打印机,绘图仪)之间的纽带,GDI是一组C++类,它在驱动程序的协助下把数据描绘在硬件上,它位于应用程序与硬件之间,把数据从一方传到另一方。在Visual Studio .NET中Micro$oft解决了GDI中的许多问题,并让它变得易用,GDI的.net版本叫做GDI+。

三、GDI+
  GDI+是GDI的下一个版本,它进行了很好的改进,并且易用性更好。GDI的一个好处就是你不必知道任何关于数据怎样在设备上渲染的细节,GDI+更好的实现了这个优点,也就是说,GDI是一个中低层API,你还可能要知道设备,而GDI+是一个高层的API,你不必知道设备。例如你如果要设置某个控件的前景和背景色,只需设置BackColor和ForeColor属性。

四、编程模式的变化
“GDI uses a stateful model, whereas GDI+ uses a stateless”——GDI是有状态的,GDI+是无无状态的。

1、不再使用设备环境或句柄
  在使用GDI绘图时,必须要指定一个设备环境(DC),用来将某个窗口或设备与设备环境类的句柄指针关联起来,所有的绘图操作都与该句柄有关。而GDI+不再使用这个设备环境或句柄,取而代之是使用Graphics对象。与设备环境相类似,Graphics对象也是将屏幕的某一个窗口与之相关联,并包含绘图操作所需要的相关属性。但是,只有这个Graphics对象与设备环境句柄还存在着联系,其余的如Pen、Brush、Image和Font等对象均不再使用设备环境。

2、Pen、Brush,Font,Image等对象是图形对象独立的
  画笔对象能与用于提供绘制方法的图形对象分开创建于维护,Graphics绘图方法直接将Pen对象作为自己的参数,从而避免了在GDI使用SelectObject进行繁琐的切换,类似的还有Brush、Path、Image和Font等。

3、“当前位置”
  GDI绘图操作(如画线)中总存在一个被称为"当前位置"的特殊位置。每次画线都是以此当前位置为起始点,画线操作结束之后,直线的结束点位置又成为了当前位置。设置当前位置的理由是为了提高画线操作的效率,因为在一些场合下,总是一条直线连着另一条直线,首尾相接。有了当前位置的自动更新,就可避免每次画线时都要给出两点的坐标。尽管有其必要性,但是单独绘制一条直线的场合总是比较多的,因此GDI+取消这个"当前位置"以避免当无法确定"当前位置"时所造成的绘图的差错,取而代之的是直接在DrawLine中指定直线起止点的坐标。   

4、绘制和填充
  GDI总是让形状轮廓绘制和填充使用同一个绘图函数,例如Rectangle。轮廓绘制需要一个画笔,而填充一个区域需要一个画刷。也就是说,不管我们是否需要填充所绘制的形状,我们都需要指定一个画刷,否则GDI采用默认的画刷进行填充。这种方式确实给我们带来了许多不便,现在GDI+将形状轮廓绘制和填充操作分开而采用不同的方法,例如DrawRectangle和FillRectangle分别用来绘制和填充一个矩形。   

5、区域的操作
  GDI提供了许多区域创建函数,如:CreateRectRgn、CreateEllpticRgn、CreateRoundRectRgn、CreatePolygonRgn和CreatePolyPolygonRgn等。诚然,这些函数给我们带来了许多方便。但在GDI+中,由于为了便于将区域引入矩阵变换操作,GDI+简化一般区域创建的方法,而将更复杂的区域创建交由Path接管。由于Path对象是与设备环境分离开来的,因而可以直接在Region构造函数中加以指定。

五、GDI+新特色
GDI+与GDI相比,增加了下列新的特性:

1、渐变画刷
  以往GDI实现颜色渐变区域的方法是通过使用不同颜色的线条来填充一个裁剪区域而达到的。现在GDI+拓展了GDI功能,提供线型渐变和路径渐变画刷来填充一个图形、路径和区域,甚至也可用来绘制直线、曲线等。这里的路径可以视为由各种绘图函数产生的轨迹。   

2、样条曲线
  对于曲线而言,最具实际意义的莫过于样条曲线。样条曲线是在生产实践的基础上产生和发展起来的。模线间的设计人员在绘制模线时,先按给定的数据将型值点准确地"点"到图板上。然后,采用一种称为"样条"的工具(一根富有弹性的有机玻璃条或木条),用压铁强迫它通过这些型值点,再适当调整这些压铁,让样条的形态发生变化,直至取得合适的形状,才沿着样条画出所需的曲线。如果我们把样条看成弹性细梁,那么压铁就可看成作用在这梁上的某些点上的集中力。GDI+的Graphics:: DrawCurve函数中就有一个这样的参数用来调整集中力的大小。除了样条曲线外,GDI+还支持原来GDI中的Bezier曲线。

3、独立的路径对象
  在GDI中,路径是隶属于一个设备环境(上下文),也就是说一旦设备环境指针超过它的有效期,路径也会被删除。而GDI+是使用Graphics对象来进行绘图操作,并将路径操作从Graphics对象分离出来,提供一个GraphicsPath类供用户使用。这就是说,我们不必担心路径对象会受到Graphics对象操作的影响,从而可以使用同一个路径对象进行多次的路径绘制操作。

4、矩阵和矩阵变换
  在图形处理过程中常需要对其几何信息进行变换以便产生复杂的新图形,矩阵是这种图形几何变换最常用的方法。为了满足人们对图形变换的需求,GDI+提供了功能强大的Matrix类来实现矩阵的旋转、错切、平移、比例等变换操作,并且GDI+还支持Graphics图形和区域(Region)的矩阵变换。

5、Alpha通道合成运算
  在图像处理中,Alpha用来衡量一个像素或图像的透明度。在非压缩的32位RGB图像中,每个像素是由四个部分组成:一个Alpha通道和三个颜色分量(R、G和B)。当Alpha值为0时,该像素是完全透明的,而当Alpha值为255时,则该像素是完全不透明。   
  Alpha混色是将源像素和背景像素的颜色进行混合,最终显示的颜色取决于其RGB颜色分量和Alpha值。它们之间的关系可用下列公式来表示
  显示颜色 = 源像素颜色 X alpha / 255 + 背景颜色 X (255 - alpha) / 255
GDI+的Color类定义了ARGB颜色数据类型,从而可以通过调整Alpha值来改变线条、图像等与背景色混合后的实际效果。

6、多图片格式的支持
  GDI+提供了对各种图片的打开,存储功能。通过GDI+,我们能够直接将一幅BMP文件存储成JPG或其它格式的图片文件。
  除了上述新特性外,GDI+还将支持重新着色、色彩修正、消除走样、元数据以及Graphics容器等特性。

六、VC.net中使用GDI+的方法  
在Visual C++.NET使用GDI+一般遵循下列步骤:   

(1)、在应用程序中添加GDI+的包含文件gdiplus.h以及附加的类库gdiplus.lib。通常gdiplus.h包含文件添加在应用程序的stdafx.h文件中,而gdiplus.lib可用两种进行添加:第一种是直接在stdafx.h文件中添加下列语句:

#pragma comment( lib, "gdiplus.lib" )
  另一种方法是:选择"项目->属性"菜单命令,在弹出的对话框中选中左侧的"链接器->输入"选项,在右侧的"附加依赖项"框中键入gdiplus.lib,

(2)、在应用程序项目的应用类中,添加一个成员变量,如下列代码:

ULONG_PTR m_gdiplusToken;
其中,ULONG_PTR是一个DWORD数据类型,该成员变量用来保存GDI+被初始化后在应用程序中的GDI+标识,以便能在应用程序退出后,引用该标识来调用Gdiplus:: GdiplusShutdown来关闭GDI+。

(3)、在应用类中添加ExitInstance的重载,并添加下列代码用来关闭GDI+:


int CGDIPlusApp::ExitInstance()
{
 Gdiplus::GdiplusShutdown(m_gdiplusToken);
 return CWinApp::ExitInstance();
}  
(4)、在应用类的InitInstance函数中添加GDI+的初始化代码: BOOL CGDIPlusApp::InitInstance()
{
 CWinApp::InitInstance();
 Gdiplus::GdiplusStartupInput gdiplusStartupInput;
 Gdiplus::GdiplusStartup(&m_gdiplusToken, &gdiplusStartupInput, NULL);
 ...
}  
(5)、在需要绘图的窗口或视图类中添加GDI+的绘制代码: void CGDIPlusView::onDraw(CDC *pDC)
{
 Graphics graphics( pDC->m_hDC );

 GraphicsPath path; // 构造一个路径
 path.AddEllipse(50, 50, 200, 100);

 // 使用路径构造一个画刷
 PathGradientBrush pthGrBrush(&path);

 // 将路径中心颜色设为蓝色
 pthGrBrush.SetCenterColor(Color(255, 0, 0, 255));

 // 设置路径周围的颜色为蓝芭,但alpha值为0
 Color colors[] = {Color(0, 0, 0, 255)};
 INT count = 1;
 pthGrBrush.SetSurroundColors(colors, &count);

 graphics.FillRectangle(&pthGrBrush, 50, 50, 200, 100);

 LinearGradientBrush linGrBrush(
  Point(300, 50),
  Point(500, 150),
  Color(255, 255, 0, 0), // 红色
  Color(255, 0, 0, 255)); // 蓝色

 graphics.FillRectangle(&linGrBrush, 300, 50, 200, 100);
}


GDI+编程(二)使用画笔

作者:李昊

  画笔常用于绘制图形的轮廓.GDI+的画笔除了具有常见的色彩和宽度属性外,还具有对齐方式,线帽,变换方式等属性。GDI+中通过Pen类来定义画笔对象。

(一)、构造与使用画笔

      Pen(brush, width); //用颜色与线宽构造一个画笔
      Pen(color, width); //用画刷与宽度构造一个画笔
例子:       Pen pen(Color(255, 0, 0, 0),1);//用第一个构造函数.构造宽度为1的黑色画刷
      graphics.DrawLine(&pen, 20, 10, 300, 100);
     
      Image         image(L"Texture1.jpg");
      TextureBrush  tBrush(&image);
      graphics.DrawImage(&image, 0, 0, image.GetWidth(), image.GetHeight());
      Pen           texturedPen(&tBrush, 30);//用第二个构造函数,用一个纹理画刷
      graphics.DrawEllipse(&texturedPen, 100, 20, 200, 100);

     
     

(二)、设值宽度与对齐方式
  创建画笔时,可以把宽度当作参数传给构造函数,我们也可以使用SetWidth()方法来改变画笔的线宽。一个理想的线宽度为0, 我们绘制一条直线时,像素位于直线的正中,下面的例子中我们用宽度为1的先用黑画笔绘制一条直线,在用绿色的宽度为10的画笔再绘制一次。       Pen blackPen(Color(255, 0, 0, 0), 1);
      Pen greenPen(Color(255, 0, 255, 0), 10);
      greenPen.SetAlignment(PenAlignmentCenter);
      graphics.DrawLine(&greenPen, 10, 100, 100, 50);
      graphics.DrawLine(&blackPen, 10, 100, 100, 50);

        

     我们把绿色画笔设为中对齐时:
     graphics.DrawRectangle(&greenPen, 10, 100, 50, 50);
     graphics.DrawRectangle(&blackPen, 10, 100, 50, 50);

    

     我们把绿色画笔设为内对齐时:    
     greenPen.SetAlignment(PenAlignmentInset);

    

     这样我们可以按需要来设置对齐方式。    
(三)、设置笔帽
  我们可以用多种方式来绘制线条的头部与尾部形状,GDI+支持圆形,方形,菱形,与箭头等样式的笔帽。       Pen pen(Color(255, 0, 0, 255), 8);
      pen.SetStartCap(LineCapArrowAnchor);
      pen.SetEndCap(LineCapRoundAnchor);
      graphics.DrawLine(&pen, 20, 175, 300, 175);
      效果如下:
        
(四)、设置两条直线的连接形
  GDI+为我们提供了当两条直线连接时连接处形状的设置,有四种方式:斜接(miter)、斜切(bevel),圆形(round),剪裁斜接(miter clipped)。       GraphicsPath path;
      Pen penJoin(Color(255, 0, 0, 255), 8);
      path.StartFigure();
      path.AddLine(Point(50, 200), Point(100, 200));
      path.AddLine(Point(100, 200), Point(100, 250));
      penJoin.SetLineJoin(LineJoinBevel);
      graphics.DrawPath(&penJoin, &path);

         
(五)、自定义线型
  GDI+为我们提供了好多线型,如果满足不了我们的需求,我们可以用成员函数SetDashPattern可以使用一个预定义的数组来描述画笔的虚实, 下面的例子用自定义风格绘制了一条直线,所用数组为{5, 2, 15, 4},如果你用画笔宽度去乘数组得{25, 10, 75, 20},显示的曲线在25与75间变换, 空白在10与20间变换。       REAL dashValues[4] = {5, 2, 15, 4};
      Pen blackPen(Color(255, 0, 0, 0), 5);
      blackPen.SetDashPattern(dashValues, 4);
      graphics.DrawLine(&blackPen, Point(5, 5), Point(405, 5));

           
有一点要明白,最后的那条虚线要比25单位少,这样它才能在405处结束。

(六)、画笔的旋转变换
  我们可以在程序中修改画笔在水平与垂直方向上的宽度的,假设我们有一个画笔的宽度为5,那么我们用它来绘制的矩形在四条边上长度都是一样的, 如果想让在水平与数值方向上不一致,我们可以使用变换,有三种方式可以实现上面的要求:      Pen pen(Color(255,0,0,255));
     pen.SetWidth(5);
     Matrix matrix(1,0,0,2,0,0);
     pen.MutiplyTransform(&matrix,MatrixOrderPrepend);//方法一
     pen.SetTransform(&matrix);//方法二
     pen.ScaleTransform(1,4);
     graphics.DrawRectange(&pen,50,50,200,200);  
  还可以对画笔进行旋转变换,旋转是相对在水平宽度与垂直位置上不一致的画笔而言的左图为缩放变换,右图为旋转变换。