在D3D10中,通过ID3DX10Font接口对象,我们可以方便的在屏幕上输出文字信息,一个DrawText函数就能解决所有问题,但在D3D11中个,这个变得超级麻烦,因为微软移除了Font接口,要在屏幕上输出文本,用户需要做很多事情。

通常我们可以用以下的方法来输出文本信息:

(1)用纹理贴图的方法,把所有的字体存储在一张纹理上,再做一个字体查询表,对应纹理的相应位置,可以用2D渲染的方式,把文本染出来,但这种方法不是很灵活,英文还好说,字母字符就那么多,但对于汉字就麻烦了,另外字体大小不能任意缩放。

(2)通过和GDI/GDI+进行交互,得到字体,把它们存储在纹理中,然后再用2D渲染的方式渲染出来,这种方法可以得到任意大小的文字,而且对英文、中文都适用。

(3)用direct write,但据说效率不高,http://www.braynzarsoft.net/index.php?p=D3D11FONT 这篇教程中有详细的介绍。

(4)使用一些文本输出控件,比如CEGUI。

(5)用FreeType2动态生成字体纹理,然后用2D渲染的方式显示。

在本篇教程中,我们先看看第一种方法的具体实现,程序的代码是在myTutorialD3D11_61的基础上修改的。

1、准备一副图片,该图片中包含所有的英文字母、标点以及特殊字符(ascii码32-126之间的所有字符),该图片将被当作纹理。

该图片背景是纯黑色,RGB为(0,0,0),用该图片产生font.dds纹理资源文件。

2、建立一个文本索引文件fontdata.txt,该文件索引ascii码32-126之间的所有字符,该文件的格式为:

[Ascii value of character] [The character] [Left Texture U coordinate] [Right Texture U Coordinate] [Pixel Width of Character]

第一列为字符的ascii码,第二列为字符,第三列为字符左边的纹理坐标,第四列为字符右边的纹理坐标,第五列为像素宽度。

32   0.0        0.0         0
33 ! 0.0        0.000976563 1
34 " 0.00195313 0.00488281  3
35 # 0.00585938 0.0136719   8

125 } 0.573242  0.576172    3
126 ~ 0.577148  0.583984    7

通过这几列值,我们就可以把一个字符和纹理中的相应位置对应起来,比如一个字符串"hello”,我们可以动态创建5个四边形,每个四边形有2个三角形组成,三角形的顶点位置为该字符在屏幕上的显示位置,顶点属性中包括纹理坐标,该四边形将通过纹理的方式显示该字符。

下边大致介绍下增加的代码:

我们的2D 文本渲染在TextClass中,可以把该类看作一个sprites class, 主要的功能就是渲染文本。

首先该类用了2个结构来描述文本,第一个是文本句子的结构类型,因为文本都是显示在四边形中,所以该类中包含了顶点缓冲、索引缓冲、顶点数量、顶点颜色等信息。

struct SentenceType
{
    ID3D11Buffer *vertexBuffer, *indexBuffer;
    int vertexCount, indexCount, maxLength;
    float red, green, blue;
};

struct VertexType
{
    D3DXVECTOR3 position;
    D3DXVECTOR2 texture;
};

通过函数InitializeSentence、UpdateSentence我们生成文本句子数据,RenderSentence函数渲染执行最终的渲染,其中的texture就是包含字体的纹理,该函数调用FontShader类实现最终的渲染。

bool TextClass::RenderSentence(ID3D11DeviceContext* deviceContext, SentenceType* sentence, D3DXMATRIX worldMatrix,
                               D3DXMATRIX orthoMatrix,  ID3D11ShaderResourceView* texture)

类FontClass建立文本句子顶点、索引数据,FontShaderClass类用执行渲染。

另外,还有2个shader文件:font.vs和font.ps, ps文件中会根据纹理的颜色来设置alpha的值,这样既可以很好的把字体纹理和背景混合起来。

// 根据选取字符.
color = shaderTexture.Sample(SampleType, input.tex);

//如果纹理为黑色,设置透明,这样可以镂空不需要的背景,只显示字体
if(color.r == 0.0f)
{
    color.a = 0.0f;
}
else
{
    color.rgb = pixelColor.rgb;
    color.a = 1.0f;
}

最终的渲染代码在GraphicsClass类中,主要在开始文本渲染时,要禁用zbuffer,并开启alpha blend功能,渲染完成后,进行相反的操作。

// 关掉z buffer,以便2d渲染
m_D3D->TurnZBufferOff();

// 打开alpha blending
m_D3D->TurnOnAlphaBlending();

//得到正交投影矩阵
m_D3D->GetOrthoMatrix(orthoMatrix);
// 渲染文本
result = m_Text->Render(m_D3D->GetDeviceContext(), worldMatrix, orthoMatrix,m_TexManager->createTex(m_D3D->GetDevice(),string("font.dds")));
if(!result)
    {
    return false;
    }

// 关闭alpha blending
m_D3D->TurnOffAlphaBlending();

// 打开z buffer
m_D3D->TurnZBufferOn();

程序执行后的界面如下,我们用白色和黄色显示了几个字tessellation demo by mikewolf:

完整的代码请参考:

工程文件myTutorialD3D11_63

代码下载:

稍后提供

Directx11教程(66) D3D11屏幕文本输出(1)相关推荐

  1. Directx11教程(14) D3D11管线(2)

    Directx11教程(14) D3D11管线(2) 原文:Directx11教程(14) D3D11管线(2) 下面我们来了解一些GPU memory的知识,主要参考资料:http://fgiese ...

  2. Directx11教程(15) D3D11管线(4)

    本章我们首先了解一下D3D11中的逻辑管线,认识一下管线中每个stage的含义. 参考资料:http://fgiesen.wordpress.com/2011/07/01/a-trip-through ...

  3. Directx11教程(18) D3D11管线(7)

    光栅化阶段(RS)之后,将进入PS/OM阶段. 参考外文资料:http://fgiesen.wordpress.com/2011/07/01/a-trip-through-the-graphics-p ...

  4. python编程在哪里写程序-第一个Python程序——在屏幕上输出文本

    本节我将给大家介绍最简单.最常用的 Python 程序--在屏幕上输出一段文本,包括字符串和数字. Python 使用 print 函数在屏幕上输出一段文本,输出结束后会自动换行. 在屏幕上输出字符串 ...

  5. 区分大小屏幕_第一个Python程序——在屏幕上输出文本

    本节我将给大家介绍最简单.最常用的 Python 程序--在屏幕上输出一段文本,包括字符串和数字.Python 使用 print 函数在屏幕上输出一段文本,输出结束后会自动换行. 在屏幕上输出字符串 ...

  6. python程序-第一个Python程序——在屏幕上输出文本

    本节我将给大家介绍最简单.最常用的 Python 程序--在屏幕上输出一段文本,包括字符串和数字. Python 使用 print 函数在屏幕上输出一段文本,输出结束后会自动换行. 在屏幕上输出字符串 ...

  7. Directx11教程四十六之FBX SDK

    在之前的DX11入门系列文章中,有篇有关 Directx11教程四十之加载OBJ模型 读取obj模型数据的博客.不过在obj读取的那篇博客我有些坑并没有说,就是我写的那个obj解析器只能解析特定的ob ...

  8. windows编程之TextOut与DrawText -- 文本输出实验

    文章目录 windows编程之TextOut与DrawText -- 文本输出实验 实验名称 实验目的 实验要求 实验源码 思考题 windows编程之TextOut与DrawText – 文本输出实 ...

  9. python计算特征的统计值并文本输出

    python计算特征的统计值并文本输出 # 输出统计分位数 df.describe(percentiles=[0.05, 0.25, 0.5, 0.75, 0.95]) featname:A => ...

最新文章

  1. 使用容器的概念理解多维数组和多维空间
  2. 设计模式笔记(2)---单件模式(创建型)
  3. 【干货下载】大数据分析——如何消除金融不确定性
  4. 不敢相信!美国预测10年后的世界!
  5. PHP store session with couchbase
  6. Error:Protocol family unavailable
  7. 超详细Gitlab Runner环境配置中文教程
  8. 一个非常好的C#字符串操作处理类StringHelper.cs
  9. CFGAN--基于生成对抗神经网络的协同过滤推荐
  10. 使用代理服务器哪些风险?
  11. 联盟营销最佳实践:提高联盟计划的投资回报率
  12. DotNetCasClient 如何获取Cas服务器返回的attributes中的数据
  13. python使用pillow生成纯透明png图片
  14. Excel·VBA工作簿拆分所有工作表单独保存
  15. 用matlab实现用Bp神经网络对iris数据集进行分类(以及影响分类性能的参数条件)
  16. SQL渗透与防御——(二)SQL注入
  17. Springboot +AOP日志系统
  18. 一道简单的Python面试题,却涵盖诸多考点,快来试试吧!
  19. GEE6:获取每年水体数据
  20. 跌宕奔流2022,自动驾驶江湖风起雨涌,特斯拉、毫末、华为突破重围

热门文章

  1. mysql组合字段语句_mysql group_concat 使用 (按分组组合字段)
  2. 《Lynda.com XML 必备教程》(Lynda.com XML Essential Training)CFE 教程
  3. 序列的自相关和互相关计算
  4. Visual C++注册
  5. 大文件上传NeatUpload简单用法 (转)
  6. 育碧魁北克-刺客信条:奥德赛
  7. C/C++编程:异步编程入门
  8. java如何创建枚举类型_Java如何创建枚举类型?
  9. 位运算——左移右移运算详解
  10. css清除浮动float