WPF解析Fnt字体
偶遇需要再 WPF中加载Fnt字体,此做。。。
using System;
using System.Collections.Generic;
using System.Drawing;
using System.IO;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using Helper.JilyImage; namespace Helper.JilyData
{ public class FntInfo
{ public FileInfo FntFileInfo { get; set; } /// <summary> 字体名称 </summary>
public string Name { get; set; } /// <summary> 大小为32像素 </summary>
public int Size { get; set; } /// <summary> 加粗 </summary>
public bool Bold { get; set; } /// <summary> 斜体 </summary>
public bool Italic { get; set; } /// <summary> 编码字符集,没有填写值即使用默认 </summary>
public string Charset { get; set; } /// <summary> 使用Unicode </summary>
public bool Unicode { get; set; } /// <summary> 纵向缩放百分比 </summary>
public double StretchH { get; set; } /// <summary> 开启平滑 </summary>
public bool Smooth { get; set; } /// <summary> 开启抗锯齿 </summary>
public bool Aa { get; set; } /// <summary> 内边距,文字与边框的空隙 </summary>
public FntPadding Padding { get; set; } /// <summary> 外边距,就是相临边缘的距离 </summary>
public FntPadding Spacing { get; set; } public FntCommon Common { get; set; } /// <summary> 此种字体共用到图数 </summary>
public List<FntPage> Pages { get; set; } public List<FntChar> Chars { get; set; } public List<FntKerning> Kernings { get; set; } public Rect Bounds { get; private set; } private FntInfo()
{
this.Common = new FntCommon();
this.Pages = new List<FntPage>();
this.Chars = new List<FntChar>();
this.Kernings = new List<FntKerning>();
} public FntInfo(string fntpath, Encoding encoding = null)
: this()
{
this.FntFileInfo = new FileInfo(fntpath);
if (this.FntFileInfo.Exists)
{
var fntlines = File.ReadAllLines(this.FntFileInfo.FullName, encoding ?? Encoding.Default);
foreach (var item in fntlines)
{
if (item.StartsWith("info"))
{
DecodeInfo(item);
}
else if (item.StartsWith("common"))
{
DecodeCommon(item);
}
else if (item.StartsWith("page"))
{
DecodePage(item);
}
else if (item.StartsWith("char") && !item.StartsWith("chars"))
{
DecodeChar(item);
}
else if (item.StartsWith("kerning") && !item.StartsWith("kernings"))
{
DecodeKerning(item);
}
}
}
} private void DecodeInfo(string str)
{
this.Name = GetValueAfterTittle(str, "face=\"", '"');
this.Size = int.Parse(GetValueAfterTittle(str, "size="));
this.Bold = int.Parse(GetValueAfterTittle(str, "bold=")) != ;
this.Italic = int.Parse(GetValueAfterTittle(str, "italic=")) != ;
this.Charset = GetValueAfterTittle(str, "charset=\"", '"');
this.Unicode = int.Parse(GetValueAfterTittle(str, "unicode=")) != ;
this.StretchH = int.Parse(GetValueAfterTittle(str, "stretchH="));
this.Smooth = int.Parse(GetValueAfterTittle(str, "smooth=")) != ;
this.Aa = int.Parse(GetValueAfterTittle(str, "aa=")) != ;
this.Padding = new FntPadding(GetValueAfterTittle(str, "padding="));
this.Spacing = new FntPadding(GetValueAfterTittle(str, "spacing="));
} private void DecodeCommon(string str)
{
this.Common.LineHeight = int.Parse(GetValueAfterTittle(str, "lineHeight="));
this.Common.Base = int.Parse(GetValueAfterTittle(str, "base="));
this.Common.ScaleW = int.Parse(GetValueAfterTittle(str, "scaleW="));
this.Common.ScaleH = int.Parse(GetValueAfterTittle(str, "scaleH="));
this.Common.Packed = int.Parse(GetValueAfterTittle(str, "lineHeight=")) != ;
} private void DecodePage(string str)
{
var page = new FntPage();
page.Id = int.Parse(GetValueAfterTittle(str, "id="));
page.FilePath = GetValueAfterTittle(str, "file=\"", '"');
page.RealPath = this.FntFileInfo.DirectoryName + "\\" + page.FilePath;
this.Pages.Add(page);
} private void DecodeChar(string str)
{
var fntchar = new FntChar();
fntchar.Id = int.Parse(GetValueAfterTittle(str, "id="));
fntchar.X = int.Parse(GetValueAfterTittle(str, "x="));
fntchar.Y = int.Parse(GetValueAfterTittle(str, "y="));
fntchar.Width = int.Parse(GetValueAfterTittle(str, "width="));
fntchar.Height = int.Parse(GetValueAfterTittle(str, "height="));
fntchar.Xoffset = int.Parse(GetValueAfterTittle(str, "xoffset="));
fntchar.Yoffset = int.Parse(GetValueAfterTittle(str, "yoffset="));
fntchar.Xadvance = int.Parse(GetValueAfterTittle(str, "xadvance="));
fntchar.Page = int.Parse(GetValueAfterTittle(str, "page="));
fntchar.Chnl = int.Parse(GetValueAfterTittle(str, "chnl=")); fntchar.InitData(this); this.Chars.Add(fntchar);
} private void DecodeKerning(string str)
{
var kerning = new FntKerning();
kerning.First = int.Parse(GetValueAfterTittle(str, "first="));
kerning.Second = int.Parse(GetValueAfterTittle(str, "second="));
kerning.Amount = int.Parse(GetValueAfterTittle(str, "amount="));
this.Kernings.Add(kerning);
} public ImageSource GetStrImage(string text)
{
//1 移除不在范围内的字符
List<char> chars = new List<char>();
foreach (var item in text)
{
if (this.Chars.FirstOrDefault(fc => fc.Id == (int)item) != null)
{
chars.Add(item);
}
} int viewwidth = ;
int viewheight = ;
int count = chars.Count - ;
for (int i = ; i <= count; i++)
{
char item = chars[i];
var c = this.Chars.FirstOrDefault(fc => fc.Id == (int)item);
if (c != null)
{
//设置宽高,用最大高度
viewheight = (c.Height + c.Yoffset) > viewheight ? (c.Height + c.Yoffset) : viewheight;
viewwidth += (c.Xadvance); if (i != )
{
var kerning = this.Kernings.FirstOrDefault(k => k.First == (int)item && k.Second == (int)chars[i - ]);
if (kerning != null)
{
//补齐长度,在存在偏移的时候,偏移量也计算在总长度
viewwidth += kerning.Amount;
}
}
}
} int currentxoffset = ;
var img = new Bitmap(viewwidth, viewheight);
using (var g = Graphics.FromImage(img))
{
for (int i = ; i <= count; i++)
{
char item = chars[i];
var c = this.Chars.FirstOrDefault(fc => fc.Id == (int)item);
if (c != null)
{
var xposition = currentxoffset + c.Xoffset;
int yposition = c.Yoffset; if (i != )
{
//前后两个字符进行偏移
var kerning = this.Kernings.FirstOrDefault(k => k.First == (int)item && k.Second == (int)chars[i - ]);
if (kerning != null)
{
//确定位置,和前一个相比,加上需要偏移的位置
xposition += kerning.Amount;
}
}
//当显示文字为空格 char 32 时不需要绘制
if (c.Data != null)
{
g.DrawImage(c.Data, xposition, yposition, c.Width, c.Height);
}
//叠加当前绘制过的长度
currentxoffset += (c.Xadvance);
}
}
} Bounds = new Rect(, , viewwidth, viewheight); return img.ConvertImageSource();
} private string GetValueAfterTittle(string s, string tittle, char endwith = ' ')
{
//补齐一个空格,用来判断最后的字符
var str = s + " ";
if (!string.IsNullOrWhiteSpace(str))
{
int firstindex = str.IndexOf(tittle) + tittle.Length;
int lastindex = str.IndexOf(endwith, firstindex);
if (firstindex != - && lastindex != -)
{
return new string(str.Skip(firstindex).Take(lastindex - firstindex).ToArray());
}
}
return "";
} } public class FntCommon
{
/// <summary> 行高,如果遇到换行符时,绘制字的位置坐标的Y值在换行后增加的像素值 </summary>
public int LineHeight { get; set; } /// <summary> 字的基本大小 </summary>
public int Base { get; set; } /// <summary> 图片宽 </summary>
public int ScaleW { get; set; } /// <summary>
/// 图片高
/// </summary>
public int ScaleH { get; set; } //没搞明白什么用
//alphaChnl = int.Parse(GetValueAfterTittle(str, "alphaChnl="));
//redChnl = int.Parse(GetValueAfterTittle(str, "redChnl="));
//greenChnl = int.Parse(GetValueAfterTittle(str, "greenChnl="));
//blueChnl = int.Parse(GetValueAfterTittle(str, "blueChnl=")); /// <summary> 图片压缩 </summary>
public bool Packed { get; set; } public FntCommon()
{ }
} public class FntPage
{
/// <summary> 当前页数 </summary>
public int Id { get; set; } /// <summary> 图片相对fnt文件的路径 </summary>
public string FilePath { get; set; } private string realPath;
public string RealPath
{
get { return realPath; }
set
{
realPath = value;
FntImage = BitmapHelper.GetBitmap(realPath);
}
} public Bitmap FntImage { get; private set; } } public class FntChar
{ /// <summary> 文字的 ID </summary>
public int Id { get; set; } /// <summary> 所在图片的X </summary>
public int X { get; set; }
/// <summary> 所在图片的Y </summary>
public int Y { get; set; } /// <summary> 宽 </summary>
public int Width { get; set; }
/// <summary> 高 </summary>
public int Height { get; set; } /// <summary> 像素偏移X </summary>
public int Xoffset { get; set; }
/// <summary> 像素偏移Y </summary>
public int Yoffset { get; set; } /// <summary> 绘制完后相应位置的x往后移 ... 像素再画下一个字 </summary>
public int Xadvance { get; set; } /// <summary> 所在图片页 </summary>
public int Page { get; set; } /// <summary> 未知 </summary>
public int Chnl { get; set; } public Bitmap Data { get; private set; } static int count = ;
public void InitData(FntInfo fntinfo)
{
count++;
var page = fntinfo.Pages.FirstOrDefault(i => i.Id == this.Page); if (page != null)
{
var rect = new Rectangle(this.X, this.Y, this.Width, this.Height);
if (Rectangle.Empty != rect && !(this.Width == && this.Height == ))
{
this.Data = page.FntImage.Clone(rect, page.FntImage.PixelFormat);
}
}
} public override string ToString()
{
return ((char)this.Id).ToString();
}
} public class FntKerning
{
/// <summary> 位移的前字符 </summary>
public int First { get; set; } /// <summary> 位移的后字符 </summary>
public int Second { get; set; } /// <summary> X右移位置,正为右 </summary>
public int Amount { get; set; } } public struct FntPadding
{
public int V1 { get; set; }
public int V2 { get; set; }
public int V3 { get; set; }
public int V4 { get; set; } public FntPadding(int v1, int v2, int v3, int v4)
: this()
{
this.V1 = v1;
this.V2 = v2;
this.V3 = v3;
this.V4 = v4;
} public FntPadding(string value)
: this()
{
var values = value.Split(',');
if (values.Length == )
{
this.V1 = int.Parse(values[]);
this.V2 = int.Parse(values[]);
this.V3 = int.Parse(values[]);
this.V4 = int.Parse(values[]);
}
else if (value.Length == )
{
this.V1 = int.Parse(values[]);
this.V2 = int.Parse(values[]);
this.V3 = int.Parse(values[]);
this.V4 = int.Parse(values[]);
}
}
}
}
WPF解析Fnt字体的更多相关文章
- WPF解析TTF 字体
偶遇需要自己解析 TTF 字体并显示,此做... using System; using System.Collections.Generic; using System.Drawing.Text; ...
- cocosstdio之字体之文本和FNT字体
FNT字体和文本字体的作用是:导入字体资源可以使用字体资源便可以使用其资源内的字体来在程序中使用 不同的是FNT字体资源内容比较少,所以个人猜想可以在特定情况下使用: 两种字体资源对比: 赋值过程对比 ...
- WPF设置全局字体和字体嵌入
原文:WPF设置全局字体和字体嵌入 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/CLeopard/article/details/40590373 ...
- WPF使用矢量字体图标(阿里巴巴iconfont)
原文:WPF使用矢量字体图标(阿里巴巴iconfont) 版权声明:本文为博主原创文章,转载请注明出处. https://blog.csdn.net/lwwl12/article/details/78 ...
- Wpf 获取指定字体和大小的字符的长宽
Wpf 获取指定字体和大小的字符的长宽 运行环境:Win10 x64, NetFrameWork 4.8, 作者:乌龙哈里,日期:2019-05-09 参考: 章节: 比如一个 Consolas 字体 ...
- 在WPF中使用字体图标
一.源码描述 这是一款基于WPF窗体应用程序的字体图标示例源码, 该源码简单易懂使用于初学者和实战项目应用, 感兴趣的朋友们可以下载看看哦. 二.功能介绍 1.用ICO字体代替 ...
- WPF中应用字体图标
一.什么是字体图标 我们在进行GDI(图形界面)编程的过程中图标是不可少的.近些年随着网络的繁荣和移动应用的繁荣,矢量图的应用越来越火. 矢量图是一种用数学方法描述的.由一系列点和线组成的图,因此相比 ...
- 【WPF】添加自定义字体
需求:在WPF项目中使用幼圆字体. 步骤: 1.首先要有幼圆TTF字体文件.在C:\Windows\Fonts目录下找,如果系统字体库中没有,就上网下一份,如这里或这里. 2.将字体文件复制到WPF项 ...
- 为WPF程序添加字体
很多时候我们开发的程序可能会在多个版本的Windows上运行,比如XP.Win7.Win8. 为了程序美观,现在很多公司会使用WPF作为程序的界面设计. 跨版本的操作的操作系统往往有一些字体上的问题, ...
随机推荐
- CSS hacker(兼容IE6、7、8)
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">这行代码是永远以最新的 ...
- Activity之间传递数据或数据包Bundle,传递对象,对象序列化,对象实现Parcelable接口
package com.gaojinhua.android.activitymsg; import android.content.Intent; import android.os.Bundle; ...
- 面向对象的异常处理之深入理解java异常处理机制
什么是异常? 异常是对问题的描述,将问题的对象进行封装: 异常体系的特点:异常体系中的所有类以及建立的对象: 都具有可抛性,也就是说可以被throw和throws关键字所操作,只有异常体系具有该特点: ...
- PictureBox控件鼠标进入的手形改变和提示
PictureBox控件载入了图片后,如果要设置其为链接作用的功能,一般需要当鼠标移动到其上时鼠标自动变为手形以提示用户此时可以点击,如果图形无法方便辨识链接的具体功能,最好此时给与一定提示. 1.鼠 ...
- Mac下配置cocos2dx2.2.6的Android环境
最近在学习cocos2dx在MAC上如何打包apk,今天先把安装JDK和ANT的过程记来. 首先,打开终端,输入”java -version” 点击回车后,出现如下提示: 我们的MAC上并没有安装过J ...
- S(tuple)类及可选(Optional)类型型
元组将多个值组合为单个值.元组内的值可以是任意 类型,各元素不必是相同的类型.元组在作为函数返 回值时尤其有用. 1.定义方法1 let http404Error= (404,"Not Fo ...
- UI2_视图切换
// // ViewController.m // UI2_视图切换 // // Created by zhangxueming on 15/7/1. // Copyright (c) 2015年 z ...
- 4月12日学习笔记——jQuery操作属性和样式
区分 DOM 属性和元素属性 <img src="images/image.1.jpg" id="hibiscus" alt="Hibiscus ...
- 分享8款最新HTML5/CSS3功能插件及源码下载
1.HTML5/CSS3鬼脸表情下拉菜单 超级可爱 这款HTML5/CSS3鬼脸表情下拉菜单真的很特别,虽然菜单的实现并没有利用复杂的HTML5/CSS3技术,但是创意的确不错. 在线演示 源码下载 ...
- call与apply函数
call与apply函数 1.为什么需要call与apply函数 Javascript中,每一个函数内部都有一个特殊的关键词this,其随着所处环境的不同其指向也是不同的. 函数的内部其this也是指 ...