关于Unity中的Bmpfont的使用

系统字体,不占空间,效果一般。

自己拖的.TTF文件形成的字体,占空间很大,有特殊效果。一个TTF字库差不多占用3M左右,之所以这么大,是因为里面包含了所有的字,就像一本字典一样,所以字符都在里面。

但是我们开发游戏,并不需要用到所有的字,有的时候可能就只用到其中的几十个字,所以可以用一种解决方案,使得文字的效果好,同时占很小的空间。

这种方案就是Bmpfont,前提是有限的字符和好的字体。

用工具GlyphDesigner (Mac系统),把有限的字符添加字体效果后,绘制在一张png大图上,同时附加一个.fnt的配置文件,里面记录了文字在图片中的位置坐标信息,用于匹配,如Advance表示的是下一个字符开始的位置

就是说有了这个fnt文件,系统才能找到这个字在图片中的哪个位置,从而复制出来显示。

 

一、Bmpfont

1.把fnt文件和png文件拖进Resources文件夹下面

2.创建一个Canvas

3.对Canvas进行初始化,记得把Game视图的分辨率调成和Canvas里面设置的一样的分辨率640X960

4.创建一个Image的UI节点作为Canvas的子节点,名字叫bg,拖进背景图片到这个节点中。

5.创建一个Text类型的UI节点叫text

6.创建一个类似TTF类型的字库资源文件在Resources文件下create--->custom font,名字改成和fnt文件一样

7.创建一个材质球create--->material,名字改成和fnt文件一样,Shader属性改成GUI/Text Shader(用默认颜色覆盖原来颜色)或者Moboile/Diffuse(原样),把png图片设置为Sprite后拖进材质球的贴图属性。

8.把类似TTF类型的字库资源文件的Default Material属性拖进第7步的材质球,再手动计算和输入每一个字符的参数属性,size=1表示只有一个字符,其他的参数在fnt文件里面找。

计算公式:

char id=52 x=2 y=2 width=25 height=33 xoffset=0 yoffset=0 xadvance=20 page=0 chnl=0 letter="4"
UV
X = x / 贴图宽(2/128 = 0.015625) ; Y = 1 - y / 贴图高(1- 2/128 = 0.984375)
W = width / 贴图宽(25/128 = 0.1953125) ; H = -1 * height / 贴图高 (-33/128 = -0.2578125)
Vert
X = xoffset (0) ; Y = yoffset (-0.5 * height)
W = width(25) ; H = height(33) advance = xadvance (20);

 

9.把类似TTF类型的字库资源文件拖进text的UI节点的Character--->font字体属性中。

10.text节点中的text属性里面输入想显示的字符,必须是在png图片中有的且被我们手动录入的字符。

 

二、扩展编辑器

上面第七步需要手动一个一个计算字符参数和添加字符进去,非常麻烦,有一种方案可以便捷一点,就是扩展编辑器

1.在Assets文件夹下面创建一个叫做Editor的文件夹,必须叫这个,不然等下里面放的脚本会和平时写的脚本混在一起。

2.把字符处理脚本(网上下载的),放进Editor文件夹下

3.字符处理脚本CreateFontEditor:

using UnityEngine;
using UnityEditor;
using System.Collections.Generic;
using System.IO;
using System.Text.RegularExpressions;

// 创建bmfont  
// 继承自编辑器的扩展,用来扩展我们的编辑器的,应该放到Editor这个目录下;

public class CreateFontEditor : Editor
{
    // 制定我们入口的菜单
    [MenuItem("Assets/Create/CreateBMFont")]
    static void CreateFont()
    {
        // 当前选择的物体
        Object obj = Selection.activeObject;
        // Unity API 返回当前你选择的资源的路径
        string fntPath = AssetDatabase.GetAssetPath(obj);
        Debug.Log("#####" + fntPath);

        // 程序需要从fnt文件里面导入我们字模信息;
        if (fntPath.IndexOf(".fnt") == -1)
        {
            // 不是字体文件  
            return;
        }

        // your_name.fnt --> your_name.fontsetting;名字一致
        // new path --> .fnt --> .fontsettings;
        string customFontPath = fntPath.Replace(".fnt", ".fontsettings");
        if (!File.Exists(customFontPath))
        {
            return;
        }

        Debug.Log(fntPath);
        StreamReader reader = new StreamReader(new FileStream(fntPath, FileMode.Open));

        List<CharacterInfo> charList = new List<CharacterInfo>();

        Regex reg = new Regex(@"char id=(?<id>\d+)\s+x=(?<x>\d+)\s+y=(?<y>\d+)\s+width=(?<width>\d+)\s+height=(?<height>\d+)\s+xoffset=(?<xoffset>\d+)\s+yoffset=(?<yoffset>\d+)\s+xadvance=(?<xadvance>\d+)\s+");
        string line = reader.ReadLine();
        int lineHeight = 0;
        int texWidth = 1;
        int texHeight = 1;

        while (line != null)
        {
            if (line.IndexOf("char id=") != -1)
            {
                Match match = reg.Match(line);
                if (match != Match.Empty)
                {
                    var id = System.Convert.ToInt32(match.Groups["id"].Value);
                    var x = System.Convert.ToInt32(match.Groups["x"].Value);
                    var y = System.Convert.ToInt32(match.Groups["y"].Value);
                    var width = System.Convert.ToInt32(match.Groups["width"].Value);
                    var height = System.Convert.ToInt32(match.Groups["height"].Value);
                    var xoffset = System.Convert.ToInt32(match.Groups["xoffset"].Value);
                    var yoffset = System.Convert.ToInt32(match.Groups["yoffset"].Value);
                    var xadvance = System.Convert.ToInt32(match.Groups["xadvance"].Value);

                    CharacterInfo info = new CharacterInfo();
                    info.index = id;
                    float uvx = 1f * x / texWidth;
                    float uvy = 1 - (1f * y / texHeight);
                    float uvw = 1f * width / texWidth;
                    float uvh = -1f * height / texHeight;

                    info.uvBottomLeft = new Vector2(uvx, uvy);
                    info.uvBottomRight = new Vector2(uvx + uvw, uvy);
                    info.uvTopLeft = new Vector2(uvx, uvy + uvh);
                    info.uvTopRight = new Vector2(uvx + uvw, uvy + uvh);

                    info.minX = xoffset;
                    info.minY = yoffset + height / 2;   // 这样调出来的效果是ok的,原理未知  
                    info.glyphWidth = width;
                    info.glyphHeight = -height; // 同上,不知道为什么要用负的,可能跟unity纹理uv有关  
                    info.advance = xadvance;

                    charList.Add(info);
                }
            }
            else if (line.IndexOf("scaleW=") != -1)
            {
                Regex reg2 = new Regex(@"common lineHeight=(?<lineHeight>\d+)\s+.*scaleW=(?<scaleW>\d+)\s+scaleH=(?<scaleH>\d+)");
                Match match = reg2.Match(line);
                if (match != Match.Empty)
                {
                    lineHeight = System.Convert.ToInt32(match.Groups["lineHeight"].Value);
                    texWidth = System.Convert.ToInt32(match.Groups["scaleW"].Value);
                    texHeight = System.Convert.ToInt32(match.Groups["scaleH"].Value);
                }
            }
            line = reader.ReadLine();
        }

        Font customFont = AssetDatabase.LoadAssetAtPath<Font>(customFontPath);
        customFont.characterInfo = charList.ToArray();
        AssetDatabase.SaveAssets();
        AssetDatabase.Refresh();
        Debug.Log(customFont);
    }
}

 

4.添加好custom font资源文件的文字材质球后,对想要手动添加字符的custom font资源文件使用右键--->Create--->CreateBMFont就可以把fnt文件里面的每一个字符和它的参数都计算添加到custom font资源文件里面

5.把custom font资源文件拖进text的UI节点的font属性,就可以使用了。

 

posted @ 2017-05-29 09:58  杭者  阅读(1269)  评论(0编辑  收藏  举报