前言 ofd文件的作用就是保证信息能如实的存储、传递、显示。保证ofd文件的真实性靠的是签名;ofd 的显示需要专用软件。ofd标准是新的国家标准,应用范围远不如pdf;现有浏览器不能解析ofd、支持ofd显示的软件也寥寥无几。ofd转图片程序下载。
专用软件读取ofd过程包括对ofd文件解压缩、分析每页的文字和图元、关联对应的资源、解析签章等复杂操作。将ofd文件转成图片,可以避免这些重复的操作;ofd转成图片后,就可以方便的在浏览器、各类app中显示。所以,ofd转图片是ofd系统中必不可少的一个功能。
ofd转图片实现思路
ofd转图片没有捷径可走。实现ofd转图片,就是在图片上输出文字、图元、各类曲线等,这些操作和输出到屏幕上并没有多大区别。ofd转图片和ofd阅读器在技术上是有很大重合的。要完成ofd转图片,需要你对ofd文件有充分的理解。参见我的文章《采用WPF技术,开发OFD电子文档阅读器》。
本文ofd转图片功能没用到wpf相关类,而是使用了System.Drawing暨GDI+。.net core 3.0 已经实现了System.Drawing;理论上,本文所涉及的代码可以很方便的移植到.net core下,在linux下实现ofd转图只有一步之遥。
实现ofd转图片包括以下几个步骤:
1 创建ofd页面信息模型
ofd页面由文本、图片、曲线等组成。首先将页面解析成各类object,它们的父类为PageObject:
public class PageObject
{
public string ID { get; set; }
public PageLayer ParentLayer { get; set; }
public string PageFileLoc => ParentLayer.ParentPage.PageFileLoc;
XmlNode _xmlNode;
public string Boundary { get; set; }
public string CTM { get; set; }
public OfdClipsGroup ClipsGroup { get; set; }
public void SetPageObject(PageLayer layer, XmlNode xmlNode)
{
_xmlNode = xmlNode;
ID = XmlHelper.GetXmlAttributeValue(xmlNode, "ID");
ParentLayer = layer;
Boundary = XmlHelper.GetXmlAttributeValue(xmlNode, "Boundary");
CTM = XmlHelper.GetXmlAttributeValue(xmlNode, "CTM");
foreach (XmlNode childNode in xmlNode.ChildNodes)
{
if (childNode.Name == OfdClipsGroup.XML_Name)
{
ClipsGroup = OfdClipsGroup.FromXml(childNode);
break;
}
}
}
public string GetAttributeValue(string name)
{
string result = XmlHelper.GetXmlAttributeValue(_xmlNode, name);
return result;
}
}
从此类派生出 PageObjectText、PageObjectPath、PageObjectImage等,代表ofd页面文本、曲线、图片等类型信息。
2 由信息模型PageObject 创建显示模型OfdVisual;
信息模型包含了显示需要的各类信息,显示模型就是利用这类信息显示。显示模型的父类为OfdVisual:
public class OfdVisual
{
public OfdVisual()
{
}
public virtual void Show(bool visiable, bool even = false)
{
}
public virtual void ShowSelect(bool visiable, Rectangle selectRegion, bool even = false)
{
}
public PointF BoundaryLocation { get; set; }
public SizeF BoundarySize { get; set; }
public Matrix ObjectTransform { get; protected set; }
public Graphics VisualGraphics { get; set; }
public PageObject PageObject { get; protected set; }
public string PageItemId
{
get
{
if (PageObject == null)
return string.Empty;
return PageObject.ID;
}
}public PointF ToPageLocation(PointF pt)
{
return new PointF(pt.X + BoundaryLocation.X, pt.Y + BoundaryLocation.Y);
}
protected void PutBoundary()
{
VisualGraphics.TranslateTransform(BoundaryLocation.X, BoundaryLocation.Y);
VisualGraphics.SetClip(ClipRect);
}
GraphicsState graphicsState;
protected void DrawSave()
{
graphicsState = VisualGraphics.Save();
}
protected void DrawRestore()
{
VisualGraphics.Restore(graphicsState);
}
protected void PutTransform()
{
if (ObjectTransform != null)
{
VisualGraphics.MultiplyTransform(ObjectTransform);
}
}
public double XZoomRate
{
get
{
if (ObjectTransform == null)
return 1;
float result = ObjectTransform.Elements[0];
return result;
}
}
public double YZoomRate
{
get
{
if (ObjectTransform == null)
return 1;
float result = ObjectTransform.Elements[4];
return result;
}
}
}
有三类显示模型OfdVisualText、OfdVisualImage、OfdVisualPath等。
3 创建bitmap
根据页面大小创建bitmap,由bitmap获取Graphics。后面的显示操作就是利用Graphics各类函数处理。
_ofdBitmap = new Bitmap((int)(width* scale), (int)(height* scale));
Graphics = Graphics.FromImage(_ofdBitmap);
ofd转图片程序功能说明
操作步骤:选择生成图片的缩放比例,点击“ofd转图片”按钮,选择ofd文件。转成图片后,ofd文件在左侧列表显示,点击列表中文件,右侧显示对应的图片。程序底部为日志,显示ofd文件的页数、转换耗时等信息。
为了方便别的程序调用,程序可以做成web 服务形式,对外接口为web api。客户端传入ofd文件,输出为图片。
后记 对于ofd转图片,网上有一种方法是采用虚拟打印。这是一种捷径,也是没有办法的办法。但是采用虚拟打印,必须依赖第三方ofd阅读器,使用不方便,输出效果也增加了不确定性。要实现ofd转图片,必须对ofd标准吃透,建立相应显示模型,按部就班的实现转图功能;只有这样,程序的功能才是自主可控、效率采有保证。