之前说过,D2D主要为了绘制有三种类型的数据:几何图形,图片,文字。这几种对象也叫做资源,资源就是要D2D流水线中要被加工的对象。

几何图形包括

  • 简单几何图形

    • 直线,DrawLine,由起点和终点构成,点用结构体:D2D1_POINT_2F描述。
    • 矩形,ID2D1RectangleGeometry,DrawRectangle,由距四条边的距离表示,用结构体:D2D1_RECT_F描述。
    • 圆角矩形,ID2D1RoundedRectangleGeometry ,DrawRoundedRectangle,同矩形,用结构体:D2D1_ROUNDED_RECT描述,另外还要加入圆角参数。
    • 椭圆,包括圆,ID2D1EllipseGeometry ,DrawEllipse,用结构体:D2D1_ELLIPSE 描述。
  • 路径,ID2D1PathGeometry
  • 组合图形,
    • 几何图形组合,ID2D1GeometryGroup
    • 变幻几何图形,ID2D1TransformedGeometry

准备绘制:

在开始绘制之前,至少要准备好3样东西:
  • D2D工厂对象,ID2D1Factory接口。用于创建绘制目标,用方法CreateHwndRenderTarget。
  • 渲染窗口,ID2D1HwndRenderTarget接口。如果是Win32程序就是默认的HWND窗口,如果是商店应用则是一个绘制表面的控件。
    渲染窗口同时还负责着绘制几何图形,文字和图片。
  • 画刷,ID2D1SolidColorBrush接口。绘制几何图形时用于指定图形的样式。
  • 绘制区域,如果是win32程序还需要指定渲染窗口中要绘制的区域。
首先要引入头文件和命名空间:
#include "windows.h"
#include "D2D1.h" using namespace D2D1;
别忘了加要链接的Lib,要input的lib如下:
d2d1.lib;
dxgi.lib;
dwrite.lib;
dxguid.lib;
windowscodecs.lib;

声明以上4个对象的代码如下:

	// Direct2D factory
ID2D1Factory* pD2DFactory;
// Render target
ID2D1HwndRenderTarget* pRenderTarget;
// A black brush, reflect the line color
ID2D1SolidColorBrush* pBlackBrush;
// Render area
RECT rc ;

初始化这些值的代码如下:

	//create d2d factory
hr = D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, &pD2DFactory) ;
//create render target (hwnd window)
GetClientRect(hWnd, &rc);
hr = pD2DFactory->CreateHwndRenderTarget( RenderTargetProperties(), HwndRenderTargetProperties(hWnd, SizeU(rc.right - rc.left, rc.bottom - rc.top)), &pRenderTarget);
//create brush
hr = pRenderTarget->CreateSolidColorBrush(ColorF(ColorF::Blue), &pBlackBrush);

然后就可以开始绘制了。笔者使用的程序是win32应用程序,绘制的代码都必须写在WndProc回调函数的case WM_PAINT:之后,并且要将所有绘制代码都包含在pRenderTarget->BeginDraw();和pRenderTarget->EndDraw();之间。例如:

	pRenderTarget->BeginDraw();

	//clear screen
pRenderTarget->Clear(D2D1::ColorF(D2D1::ColorF::White)); //draw line with 2 points
D2D1_POINT_2F ptStart = Point2F(0,0);
D2D1_POINT_2F ptEnd = Point2F(100,100);
pRenderTarget->DrawLine(ptStart,ptEnd,pBlackBrush); //draw rectangle
D2D1_RECT_F rect = RectF(rc.left + 100.0f, rc.top + 100.0f, rc.right - 100.0f, rc.bottom - 100.0f);
pRenderTarget->DrawRectangle(rect, pBlackBrush); //draw round rect
D2D1_ROUNDED_RECT roundrect = RoundedRect(RectF(rc.left + 100.0f, rc.top + 100.0f, rc.right - 100.0f, rc.bottom - 100.0f), 30.0f, 50.0f);
pRenderTarget->DrawRoundedRectangle(roundrect, pBlackBrush, 1.0f); pRenderTarget->EndDraw();

WM_PAINT是在窗口需要重绘时由系统调用的,所以我们不用主动地去调用它,它们不断地刷新的。

当然了,把代码都积压在WM_PAINT之后也太SB了,最好自己封装一个类,如下:

头文件如下:
#pragma once

#include "stdafx.h"
#include "windows.h"
#include "D2D1.h" using namespace D2D1; class D2DRender
{
public:
D2DRender(void);
~D2DRender(void); HRESULT CreateDeviceIndependentResources();
HRESULT CreateDeviceResources(HWND hWnd);
void OnRender();
void DiscardDeviceResources();
void OnResize(UINT width, UINT height); private:
// Direct2D factory
ID2D1Factory* pD2DFactory;
// Render target
ID2D1HwndRenderTarget* pRenderTarget;
// A black brush, reflect the line color
ID2D1SolidColorBrush* pBlackBrush;
// Render area
RECT rc ;
//result
HRESULT hr;
};

实现如下:

#include "stdafx.h"
#include "D2DRender.h" D2DRender::D2DRender()
{
// Direct2D factory
pD2DFactory = NULL;
// Render target
pRenderTarget = NULL;
// A black brush, reflect the line color
pBlackBrush = NULL;
} D2DRender::~D2DRender()
{
pD2DFactory->Release();
pRenderTarget->Release();
pBlackBrush->Release(); } void D2DRender::DiscardDeviceResources()
{
pD2DFactory->Release();
pRenderTarget->Release();
pBlackBrush->Release();
} void D2DRender::OnResize(UINT width, UINT height)
{
if(pRenderTarget)
{
pRenderTarget->Resize(SizeU(width,height));
}
} HRESULT D2DRender::CreateDeviceIndependentResources()
{
//create d2d factory
hr = D2D1CreateFactory(D2D1_FACTORY_TYPE_SINGLE_THREADED, &pD2DFactory) ; return hr;
} HRESULT D2DRender::CreateDeviceResources(HWND hWnd)
{
//create render target (hwnd window)
GetClientRect(hWnd, &rc);
hr = pD2DFactory->CreateHwndRenderTarget( RenderTargetProperties(), HwndRenderTargetProperties(hWnd, SizeU(rc.right - rc.left, rc.bottom - rc.top)), &pRenderTarget); //create brush
hr = pRenderTarget->CreateSolidColorBrush(ColorF(ColorF::Blue), &pBlackBrush); return hr;
} void D2DRender::OnRender()
{
pRenderTarget->BeginDraw(); //clear screen
pRenderTarget->Clear(D2D1::ColorF(D2D1::ColorF::White)); //draw line with 2 points
D2D1_POINT_2F ptStart = Point2F(0,0);
D2D1_POINT_2F ptEnd = Point2F(100,100);
pRenderTarget->DrawLine(ptStart,ptEnd,pBlackBrush); //draw rectangle
D2D1_RECT_F rect = RectF(rc.left + 100.0f, rc.top + 100.0f, rc.right - 100.0f, rc.bottom - 100.0f);
pRenderTarget->DrawRectangle(rect, pBlackBrush); //draw round rect
D2D1_ROUNDED_RECT roundrect = RoundedRect(RectF(rc.left + 100.0f, rc.top + 100.0f, rc.right - 100.0f, rc.bottom - 100.0f), 30.0f, 50.0f);
pRenderTarget->DrawRoundedRectangle(roundrect, pBlackBrush, 1.0f); pRenderTarget->EndDraw();
}

然后在WM_PANIT里调用,默认生成的gdi绘制用到的hdc相关代码可以删掉。

	case WM_PAINT:
//hdc = BeginPaint(hWnd, &ps);
D2DRender pRender;
pRender.CreateDeviceIndependentResources();
pRender.CreateDeviceResources(hWnd);
pRender.OnRender();
//EndPaint(hWnd, &ps);
break;

还是比较简单。


Direct2D 几何图形绘制基础的更多相关文章

  1. android Graphics(一):概述及基本几何图形绘制

    前言:我最近想抽空研究研究android的各种特效,android的特效真是其它平台无法比拟的,而且一个漂亮的UI交互,会给APP增色不少,而学习特效之前,有关graphics绘图的基础知识是必不可少 ...

  2. 自定义View入门-绘制基础(1)

    ### 前言 说道自定义View,我们一定会想到,自定义View的绘制流程 - 测量阶段(measure) - 布局阶段(layout) - 绘制阶段(draw) 我们看到的一些炫酷的view效果,都 ...

  3. [Direct2D开发] 绘制网格

    转载请注明出处:http://www.cnblogs.com/Ray1024 一.引言 最近在使用Direct2D进行绘制工作中,需要实现使用Direct2D绘制网格的功能.在网上查了很多资料,终于实 ...

  4. android Graphics类:概述及基本几何图形绘制

    当须要在Android上绘制图形时.就会用到Graphics类.Paint类.Paint就是相当于笔,而Canvas就是 纸.这里叫画布. 所以,凡有跟要要画的东西的设置相关的.比方大小,粗细,画笔颜 ...

  5. 2019-10-23-C#-从零开始写-SharpDx-应用-绘制基础图形

    title author date CreateTime categories C# 从零开始写 SharpDx 应用 绘制基础图形 lindexi 2019-10-23 21:16:35 +0800 ...

  6. Android自定控件基础(一)——几何图形绘制

    虽然本人有几年开发经验,但是自定义控件这一块儿,研究的很少,惭愧--用到的时候就是百度查找,复制粘贴.工时紧,总是想的快点完工就好.(都是借口啦,想学总会有时间哒) 作为一个Android开发 要说自 ...

  7. 《MATLAB从入门到放弃》二维曲线和图形绘制基础(二):使用Help文档学习line、plot、plotyy、subplot、hold绘图函数

    目录: »  plot 最常用的二维曲线绘图函数 >  帮助文档 >  基本使用语法 >  线条的样式.符号和颜色调整 >  图形属性调整 >  使用图形句柄进行设置 » ...

  8. Flutter自定义绘制(1)- 绘制基础

    CustomPainter Flutter 中实现绘制的主要是CustomPainter类. 我们一般继承这个类,来使用它: class MyPainter extends CustomPainter ...

  9. 自定义控件之Canvas图形绘制基础练习-青春痘笑脸^_^

    对于自定义控件的意义不言而喻,所以对它的深入研究是很有必要的,前些年写过几篇关于UI效果的学习过程,但是中途比较懒一直就停滞了,而对于实际工作还是面试来说系统深入的了解自定义控件那是很有必要的,所以接 ...

随机推荐

  1. (搬运工)国内顺利使用Google的另类技巧

    在特殊的地方和特殊的时间,流畅顺利使用Google的方法也会变得很特殊.分享一些奇葩的Google使用方法,通过下列网址也可以使用Google来搜索:http://www.GoogleStable.c ...

  2. nodejs 保存 payload 发送过来的文件

    1:接受文件 http://stackoverflow.com/questions/24610996/how-to-get-uploaded-file-in-node-js-express-app-u ...

  3. jquery select三级联动

    需求:对地区进行选择,选择相应的省,就会出现相应范围的市,然后出现相应的范围的县区:如果县不存在,就不现实,自我要求是自己写个简单的插件,方便以后调用: 逻辑:1.通过div的类名来获取,其下的sel ...

  4. [linux]linux下开启wifi热点

    网上说用 ap-hotspot可以,但是我测试的时候总是在start的时候卡主,试了以前的版本也是,最后找到了下面的方法. 第一步:安装 plasma-nm 可以使用下面的命令 sudo apt-ge ...

  5. CloudStack核心类ApiServlet、ApiServer、ApiDispatcher、GenericDaoBase源码分析

    ApiServlet 首先从整体上看下ApiServlet,Outline视图如下, 一.注意@Inject依赖的是javax.inject.jar,它和spring的@Autowired的区别在于使 ...

  6. hdu 4578 Transformation

    http://acm.hdu.edu.cn/showproblem.php?pid=4578 题意:1,a,b,c代表在a,b区间的每一个数加上c:2,a,b,c代表在a,b区间的每一个数乘上c: 3 ...

  7. 关于NGINX变量的一些测试结果

    作为NGINX变量,不像正规语言那么正式. 但处理自定义和内部变量时,这些操作,也是少不了的. geo $dollar { default "$"; } server { list ...

  8. 《Programming WPF》翻译 第9章 2.选择一个基类

    原文:<Programming WPF>翻译 第9章 2.选择一个基类 WPF提供了很多类,当创建一个自定义元素时,你可以从这些类中派生.图9-1显示了一组可能作为类--可能是合适的基类, ...

  9. heritrix 3.2.0 -- 环境搭建

    heritrix作为一个比较经典的开源爬虫,写这篇文章目的是因为,3.X之后的heritrix的介绍以及配置的文章比较少了. heritrix 3.x 以后使用maven 2配置jar包引用,但是总是 ...

  10. CString 的一些事

    MFC Visual Studio 2008 CString 的 Format 中不能这样存在str.Format(_T("Cool(\%)"));  或者 str.Format( ...