OLE(Object Linking and Embedding,对象连接与嵌入)是一种面向对象的技术。OLE分为两种形式,一种是嵌入,一种是链接。前者是将对象嵌入到文档中,不管外部对象怎么更改都是以嵌入操作当时的对象为准,之后不再改变。而后者是以链接形式来保存对象的,外部任何对该对象的更改都会影响文档。Spire.Doc支持Word中的OLE操作,包括添加,修改和提取OLE对象,本文将对此做详细介绍。
添加OLE
C#
//实例化一个Document对象
Document doc = new Document();
//向文档中添加一个Section对象
Section sec = doc.AddSection();
//在这个section上添加一个段落
Paragraph p = sec.AddParagraph();
//实例化一个DocPicture对象
DocPicture picture = new DocPicture(doc);
//目前我们还没有实现自动扫描成图标的功能, 只能手动用图片去设置图标
//事实上,我们每个产品都支持将文档保存成图片,可以巧妙地加以利用。
//这里我加载的图片是通过Spire.XLS 的SaveToImage方法得到的。
Image image = Image.FromFile(@"C:\Users\Administrator\Desktop\ole.png");
picture.LoadImage(image);
//在文档中插入一个工作表, OleLinkType 枚举值控制该OLE是链接还是嵌入
DocOleObject obj = p.AppendOleObject(@"jane.xlsx", picture, OleLinkType.Link);
//DocOleObject obj = p.AppendOleObject(@"jane.xlsx", picture, OleLinkType.Embed);
//保存文档
doc.SaveToFile("添加OLE.docx");
VB.NET
'实例化一个Document对象
Dim doc As New Document()
'向文档中添加一个Section对象
Dim sec As Section = doc.AddSection()
'在这个section上添加一个段落
Dim p As Paragraph = sec.AddParagraph()
'实例化一个DocPicture对象
Dim picture As New DocPicture(doc)
'目前我们还没有实现自动扫描成图标的功能, 只能手动用图片去设置图标
'事实上,我们每个产品都支持将文档保存成图片,可以巧妙地加以利用。
'这里我加载的图片是通过Spire.XLS 的SaveToImage方法得到的。
Dim image As Image = Image.FromFile("C:\Users\Administrator\Desktop\ole.png")
picture.LoadImage(image)
'在文档中插入一个工作表, OleLinkType 枚举值控制该OLE是链接还是嵌入
Dim obj As DocOleObject = p.AppendOleObject("jane.xlsx", picture, OleLinkType.Link)
'DocOleObject obj = p.AppendOleObject(@"jane.xlsx", picture, OleLinkType.Embed)
'保存文档
doc.SaveToFile("添加OLE.docx")
向文档中顺利添加了一个Excel工作表:
修改OLE
C#
//实例化一个Document对象
Document doc = new Document();
//加载一个有OLE对象的文档
doc.LoadFromFile(@"添加OLE.docx");
//获取第一个Section
Section sec = doc.Sections[0];
//遍历这个Section中的所有子元素,找到段落下的OLE对象
foreach (DocumentObject obj in sec.Body.ChildObjects)
{
if (obj is Paragraph)
{
Paragraph par = obj as Paragraph;
foreach (DocumentObject paraObj in par.ChildObjects)
{
//找到OLE对象,根据类型来进行更改操作
if (paraObj.DocumentObjectType == DocumentObjectType.OleObject)
{
DocOleObject Ole = paraObj as DocOleObject;
//如果是链接, 修改对象的链接路径
if (Ole.LinkType == OleLinkType.Link)
{
//同时还要手动去更改OLE的图片
DocPicture pic = Ole.OlePicture;
pic.LoadImage(Image.FromFile(@"WordOle.png"));
Ole.LinkPath = @"替换的Word文档.docx";
}
//如果是嵌入,更改数据即可
byte[] bys = File.ReadAllBytes(@"替换的Word文档.docx");
if (Ole.LinkType == OleLinkType.Embed)
{
DocPicture pic = new DocPicture(doc);
pic.LoadImage(Image.FromFile(@"WordOle.png"));
Ole.ObjectType = "Word.Document.12";
Ole.SetOlePicture(pic);
Ole.SetNativeData(bys);
}
}
}
}
}
doc.SaveToFile("修改OLE.docx", Spire.Doc.FileFormat.Docx2013);
VB.NET
'实例化一个Document对象
Dim doc As New Document()
'加载一个有OLE对象的文档
doc.LoadFromFile("添加OLE.docx")
'获取第一个Section
Dim sec As Section = doc.Sections(0)
'遍历这个Section中的所有子元素,找到段落下的OLE对象
For Each obj As DocumentObject In sec.Body.ChildObjects
If TypeOf obj Is Paragraph Then
Dim par As Paragraph = TryCast(obj, Paragraph)
For Each paraObj As DocumentObject In par.ChildObjects
'找到OLE对象,根据类型来进行更改操作
If paraObj.DocumentObjectType = DocumentObjectType.OleObject Then
Dim Ole As DocOleObject = TryCast(paraObj, DocOleObject)
'如果是链接, 修改对象的链接路径
If Ole.LinkType = OleLinkType.Link Then
'同时还要手动去更改OLE的图片
Dim pic As DocPicture = Ole.OlePicture
pic.LoadImage(Image.FromFile("WordOle.png"))
Ole.LinkPath = "替换的Word文档.docx"
End If
'如果是嵌入,更改数据即可
Dim bys As Byte() = File.ReadAllBytes("替换的Word文档.docx")
If Ole.LinkType = OleLinkType.Embed Then
Dim pic As New DocPicture(doc)
pic.LoadImage(Image.FromFile("WordOle.png"))
Ole.ObjectType = "Word.Document.12"
Ole.SetOlePicture(pic)
Ole.SetNativeData(bys)
End If
End If
Next
End If
Next
doc.SaveToFile("修改OLE.docx", Spire.Doc.FileFormat.Docx2013)
将原来文档中的excel工作表替换成了Word文档:
提取OLE
C#
//实例化一个Document对象
Document doc = new Document();
//加载一个有OLE对象的文档
doc.LoadFromFile(@"添加OLE.docx ");
//遍历文档所有Section
foreach (Section sec in doc.Sections)
{
//遍历Section下面所有的子元素
foreach (DocumentObject obj in sec.Body.ChildObjects)
{
if (obj is Paragraph)
{
Paragraph par = obj as Paragraph;
//遍历这个section下面的段落
foreach (DocumentObject o in par.ChildObjects)
{
//找到OLE对象,并根据类型提取
if (o.DocumentObjectType == DocumentObjectType.OleObject)
{
DocOleObject Ole = o as DocOleObject;
//ObjectType属性可以获取ole对象的具体类型。
//注意,如果是用Spire.Doc添加的ole对象的话,需要在AppendOleObject
//的时候先声明OleObjectType,不然这里得不到具体的类型,只会得到Package
string s = Ole.ObjectType;
//"AcroExch.Document.11"是指PDF对象对应的ProgID
if (s == "AcroExch.Document.11")
{
File.WriteAllBytes("Result.pdf", Ole.NativeData);
}
//"Excel.Sheet.12"是指 Excel03之后的工作表对应的ProgID
else if (s == "Excel.Sheet.12")
{
File.WriteAllBytes("Result.xlsx", Ole.NativeData);
}
//"Word.Document.12"是指03之后的Word对应的ProgID
else if (s == "Word.Document.12")
{
File.WriteAllBytes("Result.docx", Ole.NativeData);
}
}
}
}
}
}
VB.NET
'实例化一个Document对象
Dim doc As New Document()
'加载一个有OLE对象的文档
doc.LoadFromFile("添加OLE.docx ")
'遍历文档所有Section
For Each sec As Section In doc.Sections
'遍历Section下面所有的子元素
For Each obj As DocumentObject In sec.Body.ChildObjects
If TypeOf obj Is Paragraph Then
Dim par As Paragraph = TryCast(obj, Paragraph)
'遍历这个section下面的段落
For Each o As DocumentObject In par.ChildObjects
'找到OLE对象,并根据类型提取
If o.DocumentObjectType = DocumentObjectType.OleObject Then
Dim Ole As DocOleObject = TryCast(o, DocOleObject)
'ObjectType属性可以获取ole对象的具体类型。
'注意,如果是用Spire.Doc添加的ole对象的话,需要在AppendOleObject
'的时候先声明OleObjectType,不然这里得不到具体的类型,只会得到Package
Dim s As String = Ole.ObjectType
'"AcroExch.Document.11"是指PDF对象对应的ProgID
If s = "AcroExch.Document.11" Then
File.WriteAllBytes("Result.pdf", Ole.NativeData)
'"Excel.Sheet.12"是指 Excel03之后的工作表对应的ProgID
ElseIf s = "Excel.Sheet.12" Then
File.WriteAllBytes("Result.xlsx", Ole.NativeData)
'"Word.Document.12"是指03之后的Word对应的ProgID
ElseIf s = "Word.Document.12" Then
File.WriteAllBytes("Result.docx", Ole.NativeData)
End If
End If
Next
End If
Next
Next
提取出来的excel工作表如下: