diff --git a/OpenXmlPowerTools.Tests/Html/HtmlToWmlReadAsXElement.cs b/OpenXmlPowerTools.Tests/Html/HtmlToWmlReadAsXElement.cs
index 9a2e2c85..930da5e8 100644
--- a/OpenXmlPowerTools.Tests/Html/HtmlToWmlReadAsXElement.cs
+++ b/OpenXmlPowerTools.Tests/Html/HtmlToWmlReadAsXElement.cs
@@ -63,7 +63,7 @@ public static XElement ReadAsXElement(FileInfo sourceHtmlFi)
html = XElement.Parse(sb.ToString());
}
#else
- catch (XmlException e)
+ catch (XmlException)
{
throw;
}
diff --git a/OpenXmlPowerTools.Tests/Word/DocumentAssemblerTests.cs b/OpenXmlPowerTools.Tests/Word/DocumentAssemblerTests.cs
index f0241bf0..49387d26 100644
--- a/OpenXmlPowerTools.Tests/Word/DocumentAssemblerTests.cs
+++ b/OpenXmlPowerTools.Tests/Word/DocumentAssemblerTests.cs
@@ -118,7 +118,44 @@ public DocumentAssemblerTests(ITestOutputHelper log) : base(log)
[InlineData("DA264-InvalidRunLevelRepeat.docx", "DA-Data.xml", true)]
[InlineData("DA265-RunLevelRepeatWithWhiteSpaceBefore.docx", "DA-Data.xml", false)]
[InlineData("DA266-RunLevelRepeat-NoData.docx", "DA-Data.xml", true)]
-
+ [InlineData("DA267-Repeat-HorizontalAlignType.docx", "DA-Data.xml", false)]
+ [InlineData("DA268-Repeat-VerticalAlignType.docx", "DA-Data.xml", false)]
+ [InlineData("DA269-Repeat-InvalidAlignType.docx", "DA-Data.xml", true)]
+ [InlineData("DA270-ImageSelect.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA270A-ImageSelect.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA271-ImageSelectWithRepeat.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA271A-ImageSelectWithRepeat.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA272-ImageSelectWithRepeatHorizontalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA272A-ImageSelectWithRepeatHorizontalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA273-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA273A-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA274-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA274A-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA275-ImageSelectWithRepeatInvalidAlign.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA275A-ImageSelectWithRepeatInvalidAlign.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA276-ImageSelectInsideTable.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA276A-ImageSelectInsideTable.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA277-ImageSelectMissingOrInvalidPictureContent.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA277A-ImageSelectMissingOrInvalidPictureContent.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA278-ImageSelect.docx", "DA-Data-WithImagesInvalidPath.xml", true)]
+ [InlineData("DA278A-ImageSelect.docx", "DA-Data-WithImagesInvalidPath.xml", true)]
+ [InlineData("DA279-ImageSelectWithRepeat.docx", "DA-Data-WithImagesInvalidMIMEType.xml", true)]
+ [InlineData("DA279A-ImageSelectWithRepeat.docx", "DA-Data-WithImagesInvalidMIMEType.xml", true)]
+ [InlineData("DA280-ImageSelectWithRepeat.docx", "DA-Data-WithImagesInvalidImageDataFormat.xml", true)]
+ [InlineData("DA280A-ImageSelectWithRepeat.docx", "DA-Data-WithImagesInvalidImageDataFormat.xml", true)]
+ [InlineData("DA281-ImageSelectExtraWhitespaceBeforeImageContent.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA281A-ImageSelectExtraWhitespaceBeforeImageContent.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA282-ImageSelectWithHeader.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA282A-ImageSelectWithHeader.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA282-ImageSelectWithHeader.docx", "DA-Data-WithImagesInvalidPath.xml", true)]
+ [InlineData("DA282A-ImageSelectWithHeader.docx", "DA-Data-WithImagesInvalidPath.xml", true)]
+ [InlineData("DA283-ImageSelectWithFooter.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA283A-ImageSelectWithFooter.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA284-ImageSelectWithHeaderAndFooter.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA284A-ImageSelectWithHeaderAndFooter.docx", "DA-Data-WithImages.xml", false)]
+ [InlineData("DA285-ImageSelectNoParagraphFollowedAfterMetadata.docx", "DA-Data-WithImages.xml", true)]
+ [InlineData("DA285A-ImageSelectNoParagraphFollowedAfterMetadata.docx", "DA-Data-WithImages.xml", true)]
+
public void DA101(string name, string data, bool err)
{
var templateDocx = new FileInfo(Path.Combine(_sourceDir.FullName, name));
diff --git a/OpenXmlPowerTools/Word/DocumentAssembler.cs b/OpenXmlPowerTools/Word/DocumentAssembler.cs
index f766ce7a..a688e88e 100644
--- a/OpenXmlPowerTools/Word/DocumentAssembler.cs
+++ b/OpenXmlPowerTools/Word/DocumentAssembler.cs
@@ -15,46 +15,75 @@
using DocumentFormat.OpenXml.Packaging;
using Clippit;
using System.Collections;
+using DocumentFormat.OpenXml.Drawing.Wordprocessing;
+using Path = System.IO.Path;
namespace Clippit.Word
{
- public class DocumentAssembler
+ public static class DocumentAssembler
{
public static WmlDocument AssembleDocument(WmlDocument templateDoc, XmlDocument data, out bool templateError)
{
- XDocument xDoc = data.GetXDocument();
+ var xDoc = data.GetXDocument();
return AssembleDocument(templateDoc, xDoc.Root, out templateError);
}
public static WmlDocument AssembleDocument(WmlDocument templateDoc, XElement data, out bool templateError)
{
- byte[] byteArray = templateDoc.DocumentByteArray;
- using (MemoryStream mem = new MemoryStream())
+ var byteArray = templateDoc.DocumentByteArray;
+ using var mem = new MemoryStream();
+ mem.Write(byteArray, 0, (int)byteArray.Length);
+
+ using (var wordDoc = WordprocessingDocument.Open(mem, true))
{
- mem.Write(byteArray, 0, (int)byteArray.Length);
- using (WordprocessingDocument wordDoc = WordprocessingDocument.Open(mem, true))
- {
- if (RevisionAccepter.HasTrackedRevisions(wordDoc))
- throw new OpenXmlPowerToolsException("Invalid DocumentAssembler template - contains tracked revisions");
+ if (RevisionAccepter.HasTrackedRevisions(wordDoc))
+ throw new OpenXmlPowerToolsException("Invalid DocumentAssembler template - contains tracked revisions");
- var te = new TemplateError();
- foreach (var part in wordDoc.ContentParts())
- {
- ProcessTemplatePart(data, te, part);
- }
- templateError = te.HasError;
+ // calculate and store the max docPr id for later use when adding image objects
+ var macDocPrId = GetMaxDocPrId(wordDoc);
+
+ var te = new TemplateError();
+ foreach (var part in wordDoc.ContentParts())
+ {
+ ProcessTemplatePart(data, te, part);
}
- WmlDocument assembledDocument = new WmlDocument("TempFileName.docx", mem.ToArray());
- return assembledDocument;
+ templateError = te.HasError;
+
+ // update image docPr ids for the whole document
+ FixUpDocPrIds(wordDoc, macDocPrId);
}
+
+ var assembledDocument = new WmlDocument("TempFileName.docx", mem.ToArray());
+ return assembledDocument;
}
private static void ProcessTemplatePart(XElement data, TemplateError te, OpenXmlPart part)
{
- XDocument xDoc = part.GetXDocument();
+ var xDoc = part.GetXDocument();
var xDocRoot = RemoveGoBackBookmarks(xDoc.Root);
+ // process diagrams part
+ // TODO: consider splitting this method into two for clarity, so, for example, there will be
+ // TODO: pipeline-like processing: first the diagram, then the document part, or vice versa
+ var diagramPart = part.GetPartsOfType().FirstOrDefault();
+ if (diagramPart != null)
+ {
+ var diagramDoc = diagramPart.GetXDocument();
+ if (diagramDoc != null)
+ {
+ var dataPartRoot = diagramDoc.Root;
+ if (dataPartRoot != null)
+ {
+ dataPartRoot = (XElement)TransformToMetadata(dataPartRoot, data, te);
+ // do the actual content replacement
+ dataPartRoot = (XElement)ContentReplacementTransform(dataPartRoot, data, te, part);
+ diagramDoc.Elements().First().ReplaceWith(dataPartRoot);
+ diagramPart.PutXDocument();
+ }
+ }
+ }
+
// content controls in cells can surround the W.tc element, so transform so that such content controls are within the cell content
xDocRoot = (XElement)NormalizeContentControlsInCells(xDocRoot);
@@ -72,7 +101,7 @@ private static void ProcessTemplatePart(XElement data, TemplateError te, OpenXml
ProcessOrphanEndRepeatEndConditional(xDocRoot, te);
// do the actual content replacement
- xDocRoot = (XElement)ContentReplacementTransform(xDocRoot, data, te);
+ xDocRoot = (XElement)ContentReplacementTransform(xDocRoot, data, te, part);
xDoc.Elements().First().ReplaceWith(xDocRoot);
part.PutXDocument();
@@ -85,12 +114,12 @@ private static void ProcessTemplatePart(XElement data, TemplateError te, OpenXml
PA.Repeat,
PA.EndRepeat,
PA.Table,
+ PA.Image
};
private static object ForceBlockLevelAsAppropriate(XNode node, TemplateError te)
{
- XElement element = node as XElement;
- if (element != null)
+ if (node is XElement element)
{
if (element.Name == W.p)
{
@@ -102,7 +131,7 @@ private static object ForceBlockLevelAsAppropriate(XNode node, TemplateError te)
if (otherTextInParagraph != "")
{
var newPara = new XElement(element);
- var newMeta = newPara.Elements().Where(n => s_MetaToForceToBlock.Contains(n.Name)).First();
+ var newMeta = newPara.Elements().First(n => s_MetaToForceToBlock.Contains(n.Name));
newMeta.ReplaceWith(CreateRunErrorMessage("Error: Unmatched metadata can't be in paragraph with other text", te));
return newPara;
}
@@ -117,9 +146,9 @@ private static object ForceBlockLevelAsAppropriate(XNode node, TemplateError te)
var count = childMeta.Count();
if (count % 2 == 0)
{
- if (childMeta.Where(c => c.Name == PA.Repeat).Count() != childMeta.Where(c => c.Name == PA.EndRepeat).Count())
+ if (childMeta.Count(c => c.Name == PA.Repeat) != childMeta.Count(c => c.Name == PA.EndRepeat))
return CreateContextErrorMessage(element, "Error: Mismatch Repeat / EndRepeat at run level", te);
- if (childMeta.Where(c => c.Name == PA.Conditional).Count() != childMeta.Where(c => c.Name == PA.EndConditional).Count())
+ if (childMeta.Count(c => c.Name == PA.Conditional) != childMeta.Count(c => c.Name == PA.EndConditional))
return CreateContextErrorMessage(element, "Error: Mismatch Conditional / EndConditional at run level", te);
return new XElement(element.Name,
element.Attributes(),
@@ -157,7 +186,7 @@ private static XElement RemoveGoBackBookmarks(XElement xElement)
while (true)
{
var bm = cloneXDoc.DescendantsAndSelf(W.bookmarkStart).FirstOrDefault(b => (string)b.Attribute(W.name) == "_GoBack");
- if (bm == null)
+ if (bm is null)
break;
var id = (string)bm.Attribute(W.id);
var endBm = cloneXDoc.DescendantsAndSelf(W.bookmarkEnd).FirstOrDefault(b => (string)b.Attribute(W.id) == id);
@@ -171,8 +200,7 @@ private static XElement RemoveGoBackBookmarks(XElement xElement)
// the content control, which contains the paragraph content of the cell.
private static object NormalizeContentControlsInCells(XNode node)
{
- XElement element = node as XElement;
- if (element != null)
+ if (node is XElement element)
{
if (element.Name == W.sdt && element.Parent.Name == W.tr)
{
@@ -187,7 +215,7 @@ private static object NormalizeContentControlsInCells(XNode node)
}
return new XElement(element.Name,
element.Attributes(),
- element.Nodes().Select(n => NormalizeContentControlsInCells(n)));
+ element.Nodes().Select(NormalizeContentControlsInCells));
}
return node;
}
@@ -200,7 +228,7 @@ private static void NormalizeTablesRepeatAndConditional(XElement xDoc, TemplateE
var tables = xDoc.Descendants(PA.Table).ToList();
foreach (var table in tables)
{
- var followingElement = table.ElementsAfterSelf().Where(e => e.Name == W.tbl || e.Name == W.p).FirstOrDefault();
+ var followingElement = table.ElementsAfterSelf().FirstOrDefault(e => e.Name == W.tbl || e.Name == W.p);
if (followingElement == null || followingElement.Name != W.tbl)
{
table.ReplaceWith(CreateParaErrorMessage("Table metadata is not immediately followed by a table", te));
@@ -213,6 +241,52 @@ private static void NormalizeTablesRepeatAndConditional(XElement xDoc, TemplateE
table.Add(followingElement);
}
+ var images = xDoc.Descendants(PA.Image).ToList();
+ foreach (var image in images)
+ {
+ var followingElement = image.ElementsAfterSelf().FirstOrDefault(e => e.Name == W.sdt || e.Name == W.p);
+
+ if (followingElement == null)
+ {
+ image.ReplaceWith(CreateParaErrorMessage("Image metadata is not immediately followed by an image", te));
+ continue;
+ }
+
+ // get sdt element (can also be within a paragraph) and check it's contents
+ var sdt = followingElement.Name == W.p ? followingElement.Elements().FirstOrDefault(e => e.Name == W.sdt) : followingElement;
+
+ if (sdt != null && sdt.Name == W.sdt)
+ {
+ // get sdt properties
+ var sdtPr = sdt.Elements().FirstOrDefault(e => e.Name == W.sdtPr);
+ if (sdtPr != null)
+ {
+ // check for properties if contain picture
+ var picture = sdtPr.Elements().FirstOrDefault(e => e.Name == W.picture);
+ if (picture == null)
+ {
+ image.ReplaceWith(
+ CreateParaErrorMessage("Image metadata does not contain picture element", te));
+ continue;
+ }
+ }
+ }
+ else
+ {
+ // there might be the image without surrounding content control
+ image.RemoveNodes();
+ followingElement.Remove();
+ image.Add(followingElement);
+ continue;
+ }
+
+ // remove superflous paragraph from Image metadata
+ image.RemoveNodes();
+ // detach w:sdt from parent, and add to Image metadata
+ followingElement.Remove();
+ image.Add(followingElement);
+ }
+
int repeatDepth = 0;
int conditionalDepth = 0;
foreach (var metadata in xDoc.Descendants().Where(d =>
@@ -249,8 +323,8 @@ private static void NormalizeTablesRepeatAndConditional(XElement xDoc, TemplateE
while (true)
{
- bool didReplace = false;
- foreach (var metadata in xDoc.Descendants().Where(d => (d.Name == PA.Repeat || d.Name == PA.Conditional) && d.Attribute(PA.Depth) != null).ToList())
+ var didReplace = false;
+ foreach (var metadata in xDoc.Descendants().Where(d => (d.Name == PA.Repeat || d.Name == PA.Conditional || d.Name == PA.Image) && d.Attribute(PA.Depth) != null).ToList())
{
var depth = (int)metadata.Attribute(PA.Depth);
XName matchingEndName = null;
@@ -282,8 +356,9 @@ private static void NormalizeTablesRepeatAndConditional(XElement xDoc, TemplateE
}
}
- private static List s_AliasList = new List()
+ private static List s_AliasList = new List
{
+ "Image",
"Content",
"Table",
"Repeat",
@@ -294,13 +369,12 @@ private static void NormalizeTablesRepeatAndConditional(XElement xDoc, TemplateE
private static object TransformToMetadata(XNode node, XElement data, TemplateError te)
{
- XElement element = node as XElement;
- if (element != null)
+ if (node is XElement element)
{
if (element.Name == W.sdt)
{
var alias = (string)element.Elements(W.sdtPr).Elements(W.alias).Attributes(W.val).FirstOrDefault();
- if (alias == null || alias == "" || s_AliasList.Contains(alias))
+ if (string.IsNullOrEmpty(alias) || s_AliasList.Contains(alias))
{
var ccContents = element
.DescendantsTrimmed(W.txbxContent)
@@ -312,7 +386,7 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
.Replace('”', '"');
if (ccContents.StartsWith("<"))
{
- XElement xml = TransformXmlTextToMetadata(te, ccContents);
+ var xml = TransformXmlTextToMetadata(te, ccContents);
if (xml.Name == W.p || xml.Name == W.r) // this means there was an error processing the XML.
{
if (element.Parent.Name == W.p)
@@ -337,6 +411,92 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
element.Attributes(),
element.Nodes().Select(n => TransformToMetadata(n, data, te)));
}
+ if (element.Name == A.r)
+ {
+ var paraContents = element
+ .DescendantsTrimmed(W.txbxContent)
+ .Where(e => e.Name == A.t)
+ .Select(t => (string)t)
+ .StringConcatenate()
+ .Trim();
+ int occurances = paraContents.Select((c, i) => paraContents.Substring(i)).Count(sub => sub.StartsWith("<#"));
+ if (paraContents.StartsWith("<#") && paraContents.EndsWith("#>") && occurances == 1)
+ {
+ var xmlText = paraContents.Substring(2, paraContents.Length - 4).Trim();
+ var xml = TransformXmlTextToMetadata(te, xmlText);
+ if (xml.Name == W.p || xml.Name == W.r)
+ return xml;
+ xml.Add(element);
+ return xml;
+ }
+ if (paraContents.Contains("<#"))
+ {
+ var runReplacementInfo = new List();
+ var thisGuid = Guid.NewGuid().ToString();
+ var r = new Regex("<#.*?#>");
+ XElement xml = null;
+ OpenXmlRegex.Replace(new[] { element }, r, thisGuid, (para, match) =>
+ {
+ var matchString = match.Value.Trim();
+ var xmlText = matchString.Substring(2, matchString.Length - 4).Trim().Replace('“', '"').Replace('”', '"');
+ try
+ {
+ xml = XElement.Parse(xmlText);
+ }
+ catch (XmlException e)
+ {
+ var rri = new RunReplacementInfo
+ {
+ Xml = null,
+ XmlExceptionMessage = "XmlException: " + e.Message,
+ SchemaValidationMessage = null,
+ };
+ runReplacementInfo.Add(rri);
+ return true;
+ }
+ var schemaError = ValidatePerSchema(xml);
+ if (schemaError != null)
+ {
+ var rri = new RunReplacementInfo
+ {
+ Xml = null,
+ XmlExceptionMessage = null,
+ SchemaValidationMessage = "Schema Validation Error: " + schemaError,
+ };
+ runReplacementInfo.Add(rri);
+ return true;
+ }
+ var rri2 = new RunReplacementInfo
+ {
+ Xml = xml,
+ XmlExceptionMessage = null,
+ SchemaValidationMessage = null,
+ };
+ runReplacementInfo.Add(rri2);
+ return true;
+ }, false);
+
+ var newPara = new XElement(element);
+ foreach (var rri in runReplacementInfo)
+ {
+ var runToReplace = newPara.Descendants(W.r).FirstOrDefault(rn => rn.Value == thisGuid && rn.Parent.Name != PA.Content);
+ if (runToReplace == null)
+ throw new OpenXmlPowerToolsException("Internal error");
+ if (rri.XmlExceptionMessage != null)
+ runToReplace.ReplaceWith(CreateRunErrorMessage(rri.XmlExceptionMessage, te));
+ else if (rri.SchemaValidationMessage != null)
+ runToReplace.ReplaceWith(CreateRunErrorMessage(rri.SchemaValidationMessage, te));
+ else
+ {
+ var newXml = new XElement(rri.Xml);
+ newXml.Add(runToReplace);
+ runToReplace.ReplaceWith(newXml);
+ }
+ }
+ var coalescedParagraph = WordprocessingMLUtil.CoalesceAdjacentRunsWithIdenticalFormatting(newPara);
+ return coalescedParagraph;
+ }
+ }
if (element.Name == W.p)
{
var paraContents = element
@@ -345,11 +505,11 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
.Select(t => (string)t)
.StringConcatenate()
.Trim();
- int occurances = paraContents.Select((c, i) => paraContents.Substring(i)).Count(sub => sub.StartsWith("<#"));
+ var occurances = paraContents.Select((c, i) => paraContents.Substring(i)).Count(sub => sub.StartsWith("<#"));
if (paraContents.StartsWith("<#") && paraContents.EndsWith("#>") && occurances == 1)
{
var xmlText = paraContents.Substring(2, paraContents.Length - 4).Trim();
- XElement xml = TransformXmlTextToMetadata(te, xmlText);
+ var xml = TransformXmlTextToMetadata(te, xmlText);
if (xml.Name == W.p || xml.Name == W.r)
return xml;
xml.Add(element);
@@ -357,7 +517,7 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
}
if (paraContents.Contains("<#"))
{
- List runReplacementInfo = new List();
+ var runReplacementInfo = new List();
var thisGuid = Guid.NewGuid().ToString();
var r = new Regex("<#.*?#>");
XElement xml = null;
@@ -371,7 +531,7 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
}
catch (XmlException e)
{
- RunReplacementInfo rri = new RunReplacementInfo()
+ var rri = new RunReplacementInfo
{
Xml = null,
XmlExceptionMessage = "XmlException: " + e.Message,
@@ -380,10 +540,10 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
runReplacementInfo.Add(rri);
return true;
}
- string schemaError = ValidatePerSchema(xml);
+ var schemaError = ValidatePerSchema(xml);
if (schemaError != null)
{
- RunReplacementInfo rri = new RunReplacementInfo()
+ var rri = new RunReplacementInfo
{
Xml = null,
XmlExceptionMessage = null,
@@ -392,7 +552,7 @@ private static object TransformToMetadata(XNode node, XElement data, TemplateErr
runReplacementInfo.Add(rri);
return true;
}
- RunReplacementInfo rri2 = new RunReplacementInfo()
+ var rri2 = new RunReplacementInfo
{
Xml = xml,
XmlExceptionMessage = null,
@@ -442,7 +602,7 @@ private static XElement TransformXmlTextToMetadata(TemplateError te, string xmlT
{
return CreateParaErrorMessage("XmlException: " + e.Message, te);
}
- string schemaError = ValidatePerSchema(xml);
+ var schemaError = ValidatePerSchema(xml);
if (schemaError != null)
return CreateParaErrorMessage("Schema Validation Error: " + schemaError, te);
return xml;
@@ -450,20 +610,21 @@ private static XElement TransformXmlTextToMetadata(TemplateError te, string xmlT
private class RunReplacementInfo
{
- public XElement Xml;
- public string XmlExceptionMessage;
- public string SchemaValidationMessage;
+ public XElement Xml { get; set; }
+ public string XmlExceptionMessage { get; set; }
+ public string SchemaValidationMessage { get; set; }
}
private static string ValidatePerSchema(XElement element)
{
if (s_PASchemaSets == null)
{
- s_PASchemaSets = new Dictionary()
+ s_PASchemaSets = new Dictionary
{
{
PA.Content,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
@@ -477,7 +638,8 @@ private static string ValidatePerSchema(XElement element)
},
{
PA.Table,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
@@ -490,13 +652,15 @@ private static string ValidatePerSchema(XElement element)
},
{
PA.Repeat,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
+
",
@@ -504,7 +668,8 @@ private static string ValidatePerSchema(XElement element)
},
{
PA.EndRepeat,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
@@ -513,7 +678,8 @@ private static string ValidatePerSchema(XElement element)
},
{
PA.Conditional,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
@@ -528,81 +694,542 @@ private static string ValidatePerSchema(XElement element)
},
{
PA.EndConditional,
- new PASchemaSet() {
+ new PASchemaSet
+ {
XsdMarkup =
@"
",
}
},
+ {
+ PA.Image,
+ new PASchemaSet
+ {
+ XsdMarkup =
+ @"
+
+
+
+
+
+ ",
+ }
+ }
};
foreach (var item in s_PASchemaSets)
{
var itemPAss = item.Value;
- XmlSchemaSet schemas = new XmlSchemaSet();
+ var schemas = new XmlSchemaSet();
schemas.Add("", XmlReader.Create(new StringReader(itemPAss.XsdMarkup)));
itemPAss.SchemaSet = schemas;
}
}
if (!s_PASchemaSets.ContainsKey(element.Name))
{
- return string.Format("Invalid XML: {0} is not a valid element", element.Name.LocalName);
+ return $"Invalid XML: {element.Name.LocalName} is not a valid element";
}
var paSchemaSet = s_PASchemaSets[element.Name];
- XDocument d = new XDocument(element);
+ var d = new XDocument(element);
string message = null;
d.Validate(paSchemaSet.SchemaSet, (sender, e) =>
{
- if (message == null)
- message = e.Message;
+ message ??= e.Message;
}, true);
- if (message != null)
- return message;
- return null;
+ return message;
}
private class PA
{
- public static XName Content = "Content";
- public static XName Table = "Table";
- public static XName Repeat = "Repeat";
- public static XName EndRepeat = "EndRepeat";
- public static XName Conditional = "Conditional";
- public static XName EndConditional = "EndConditional";
-
- public static XName Select = "Select";
- public static XName Optional = "Optional";
- public static XName Match = "Match";
- public static XName NotMatch = "NotMatch";
- public static XName Depth = "Depth";
+ public static readonly XName Image = "Image";
+ public static readonly XName Content = "Content";
+ public static readonly XName Table = "Table";
+ public static readonly XName Repeat = "Repeat";
+ public static readonly XName EndRepeat = "EndRepeat";
+ public static readonly XName Conditional = "Conditional";
+ public static readonly XName EndConditional = "EndConditional";
+
+ public static readonly XName Select = "Select";
+ public static readonly XName Optional = "Optional";
+ public static readonly XName Match = "Match";
+ public static readonly XName NotMatch = "NotMatch";
+ public static readonly XName Depth = "Depth";
+ public static readonly XName Align = "Align";
}
private class PASchemaSet
{
- public string XsdMarkup;
- public XmlSchemaSet SchemaSet;
+ public string XsdMarkup { get; set; }
+ public XmlSchemaSet SchemaSet { get; set; }
}
- private static Dictionary s_PASchemaSets = null;
+ private static Dictionary s_PASchemaSets;
private class TemplateError
{
- public bool HasError = false;
+ public bool HasError { get; set; }
+ }
+
+ ///
+ /// Gets the next image relationship identifier of given part. The
+ /// parts can be either header, footer or main document part. The method
+ /// scans for already present relationship identifiers, then increments and
+ /// returns the next available value.
+ ///
+ /// The part.
+ /// System.String.
+ private static string GetNextImageRelationshipId(OpenXmlPart part)
+ {
+ if (part is MainDocumentPart mainDocumentPart)
+ {
+ var imageId = mainDocumentPart.Parts
+ .Select(p => Regex.Match(p.RelationshipId, @"rId(?\d+)").Groups["rId"].Value)
+ .Max(Convert.ToDecimal);
+
+ return $"rId{++imageId}";
+ }
+
+ if (part is HeaderPart headerPart)
+ {
+ var imageId = headerPart.Parts
+ .Select(p => Regex.Match(p.RelationshipId, @"rId(?\d+)").Groups["rId"].Value)
+ .Max(Convert.ToDecimal);
+
+ return $"rId{++imageId}";
+ }
+
+ if (part is FooterPart footerPart)
+ {
+ var imageId = footerPart.Parts
+ .Select(p => Regex.Match(p.RelationshipId, @"rId(?\d+)").Groups["rId"].Value)
+ .Max(Convert.ToDecimal);
+
+ return $"rId{++imageId}";
+ }
+
+ return null;
+ }
+
+ ///
+ /// Calculates the maximum docPr id. The identifier is
+ /// unique throughout the document. This method
+ /// scans the whole document, finds and stores the max number (id is signed
+ /// 23 bit integer).
+ ///
+ /// The word document.
+ /// System.Decimal.
+ private static decimal GetMaxDocPrId(WordprocessingDocument wordDoc)
+ {
+ var idsList = new List();
+ foreach (var part in wordDoc.ContentParts())
+ {
+ idsList.AddRange(part.GetXDocument().Descendants(WP.docPr)
+ .SelectMany(e => e.Attributes().Where(a => a.Name == NoNamespace.id)).Select(v => v.Value));
+ }
+ return idsList.Count == 0 ? 0 : idsList.Max(Convert.ToDecimal);
+ }
+
+ private const string InvalidImageId = "InvalidImageId";
+
+ ///
+ /// Fixes docPrIds for the document. The identifier is unique throughout the
+ /// document. This method scans the whole document, finds and replaces the
+ /// image ids which were marked as invalid with incremental id
+ /// (id is signed 23 bit integer).
+ ///
+ /// The word processing document.
+ /// The current maximum document pr identifier calculated
+ /// before the document has been processed.
+ private static void FixUpDocPrIds(WordprocessingDocument wDoc, decimal maxDocPrId)
+ {
+ var elementToFind = WP.docPr;
+ var docPrToChange = wDoc
+ .ContentParts()
+ .Select(cp => cp.GetXDocument())
+ .Select(xd => xd.Descendants().Where(d => d.Name == elementToFind))
+ .SelectMany(m => m);
+ var nextId = maxDocPrId;
+ foreach (var item in docPrToChange)
+ {
+ var idAtt = item.Attribute(NoNamespace.id);
+ if (idAtt != null && idAtt.Value == InvalidImageId)
+ idAtt.Value = $"{++nextId}";
+ }
+ foreach (var cp in wDoc.ContentParts())
+ cp.PutXDocument();
+ }
+
+ // shape type identifier
+ private static int s_shapeTypeId = 1;
+ private static int GetNextShapeType() => s_shapeTypeId++;
+
+ // shape identifier
+ private static int s_shapeId = 2000;
+ private static string GetNextShapeId() => $"_x0000_s{s_shapeId++}";
+
+ ///
+ /// Creates and returns the image part inside the given part. The
+ /// part can be either header, footer or main document part.
+ ///
+ /// The part.
+ /// Type of the image part.
+ /// The relationship identifier.
+ /// ImagePart.
+ private static ImagePart GetImagePart(OpenXmlPart part, ImagePartType imagePartType, string relationshipId)
+ {
+ if (part is MainDocumentPart mainDocumentPart)
+ return mainDocumentPart.AddImagePart(imagePartType, relationshipId);
+
+ if (part is HeaderPart headerPart)
+ return headerPart.AddImagePart(imagePartType, relationshipId);
+
+ if (part is FooterPart footerPart)
+ return footerPart.AddImagePart(imagePartType, relationshipId);
+
+ return null;
+ }
+
+ ///
+ /// Method processes the image content and generates image element
+ ///
+ /// Source element
+ /// Data element with content
+ /// Error indicator
+ /// The part where the image is getting processed.
+ /// Image element
+ private static object ProcessImageContent(XElement element, XElement data, TemplateError templateError, OpenXmlPart part)
+ {
+ // check for misplaced sdt content, should contain the paragraph and not vice versa
+ var sdt = element.Descendants(W.sdt).FirstOrDefault();
+ // get the original element with all the formatting
+ var orig = sdt == null ? element.Descendants(W.p).FirstOrDefault() : sdt.Descendants(W.p).FirstOrDefault();
+
+ // check for first run having image element in it
+ if (orig == null || !orig.Descendants(W.r).FirstOrDefault().Descendants(W.drawing).Any())
+ {
+ return CreateContextErrorMessage(element, "Image metadata is not immediately followed by an image", templateError);
+ }
+
+ // clone the paragraph, so repeating elements won't be overridden
+ var para = new XElement(orig);
+
+ // get the xpath of of the element
+ var xPath = (string)element.Attribute(PA.Select);
+ // get image path
+ var imagePath = EvaluateXPathToString(data, xPath, false);
+
+ // assign unique image and paragraph ids. Image id is document property Id (wp:docPr)
+ // and relationship id is rId. Their numbering is different.
+ const string imageId = InvalidImageId; // Ids will be replaced with real ones later, after transform is done
+ var relationshipId = GetNextImageRelationshipId(part);
+
+ var inline =
+ para.Descendants(W.drawing)
+ .Descendants(WP.inline).FirstOrDefault();
+ if (inline == null)
+ {
+ return CreateContextErrorMessage(element, "Image: invalid picture control", templateError);
+ }
+
+ // get aspect ratio option
+ var ratioAttr = inline
+ .Descendants(WP.cNvGraphicFramePr)
+ .Descendants(A.graphicFrameLocks).FirstOrDefault().Attribute(NoNamespace.noChangeAspect);
+
+ var keepSourceImageAspect = (ratioAttr == null);
+ var keepOriginalImageSizeElement = inline.Descendants(Pic.cNvPicPr).FirstOrDefault();
+ var keepOriginalImageSize = false;
+
+ if (keepOriginalImageSizeElement != null)
+ {
+ var attr = keepOriginalImageSizeElement.Attribute("preferRelativeResize");
+ if (attr != null)
+ {
+ keepOriginalImageSize = attr.Value == "0";
+ }
+ }
+
+ // get extent
+ var extent = inline
+ .Descendants(WP.extent)
+ .FirstOrDefault();
+ var pictureExtent = inline
+ .Descendants(A.graphic)
+ .Descendants(A.graphicData)
+ .Descendants(Pic._pic)
+ .Descendants(Pic.spPr)
+ .Descendants(A.xfrm)
+ .Descendants(A.ext).
+ FirstOrDefault();
+
+ if (extent == null || pictureExtent == null)
+ {
+ return CreateContextErrorMessage(element, "Image: missing element in picture control - extent(s)", templateError);
+ }
+
+ // get docPr
+ var docPr = inline.Descendants(WP.docPr).FirstOrDefault();
+ if (docPr == null)
+ {
+ return CreateContextErrorMessage(element, "Image: missing element in picture control - docPtr", templateError);
+ }
+
+ docPr.SetAttributeValue(NoNamespace.id, imageId);
+ docPr.SetAttributeValue(NoNamespace.name, "Templated Image Content");
+
+ var blip = inline
+ .Descendants(A.graphic)
+ .Descendants(A.graphicData)
+ .Descendants(Pic.blipFill)
+ .Descendants(A.blip)
+ .FirstOrDefault();
+
+ if (blip != null)
+ {
+ // Add the image to main document part
+ var stream = Image2Stream(imagePath, out var imagePartType, out var error);
+ if (stream != null)
+ {
+ var ip = GetImagePart(part, imagePartType, relationshipId);
+
+ if (ip == null)
+ {
+ error = "Failed to get image part";
+ return CreateContextErrorMessage(element, string.Concat("Image: ", error), templateError);
+ }
+ ip.FeedData(stream);
+ stream.Close();
+
+ // access the saved image and get the dimensions
+ using (var savedStream = ip.GetStream(FileMode.Open))
+ using (var image = System.Drawing.Image.FromStream(savedStream))
+ {
+ // one inch is 914400 EMUs
+ // 96dpi where dot is pixel
+ var pixelInEMU = 914400 / 96;
+ var width = image.Width;
+ var height = image.Height;
+
+ if (keepSourceImageAspect)
+ {
+ var ratio = height / (width * 1.0);
+ if (!int.TryParse(extent.Attribute(NoNamespace.cx).Value, out width))
+ {
+ return CreateContextErrorMessage(element, "Image: Invalid image attributes",
+ templateError);
+ }
+ height = (int)(width * ratio);
+
+ // replace attributes
+ extent.SetAttributeValue(NoNamespace.cy, height);
+ pictureExtent.SetAttributeValue(NoNamespace.cx, width);
+ pictureExtent.SetAttributeValue(NoNamespace.cy, height);
+ }
+
+ if (keepOriginalImageSize)
+ {
+ width = image.Width * pixelInEMU;
+ height = image.Height * pixelInEMU;
+
+ // replace attributes
+ extent.SetAttributeValue(NoNamespace.cx, width);
+ extent.SetAttributeValue(NoNamespace.cy, height);
+ pictureExtent.SetAttributeValue(NoNamespace.cx, width);
+ pictureExtent.SetAttributeValue(NoNamespace.cy, height);
+ }
+ }
+ }
+ else
+ {
+ return CreateContextErrorMessage(element, string.Concat("Image: ", error), templateError);
+ }
+
+ blip.SetAttributeValue(R.embed, relationshipId);
+ }
+
+ return para;
+ }
+
+ ///
+ /// Determines whether the input image is base64 encoded string or path
+ ///
+ /// Input image (either image path or base64 encoded string). Base 64 encoded string
+ /// should start with MIME data type identifier followed by raw data. Example:
+ /// ...
+ /// Image Part Type to be embedded in the document and to be
+ /// referenced by image control
+ /// Error message
+ private static Stream Image2Stream(string inputImage, out ImagePartType imagePartType, out string error)
+ {
+ string imageType;
+ Stream stream;
+ if (inputImage.StartsWith("data:image"))
+ {
+ // assume the image is base64 encoded format. See https://en.wikipedia.org/wiki/Data_URI_scheme
+
+ // get the image type and data
+ imageType = Regex.Match(inputImage, @"data:image/(?.+?);").Groups["type"].Value;
+ var base64Data = Regex.Match(inputImage, @"data:image/(?.+?),(?.+)").Groups["data"].Value;
+
+ try
+ {
+ var imageBytes = Base64.ConvertFromBase64(string.Empty, base64Data);
+
+ stream = new MemoryStream(imageBytes, 0, imageBytes.Length);
+ }
+ catch (Exception)
+ {
+ imagePartType = default(ImagePartType);
+ error = "Invalid Image data format";
+ return null;
+ }
+ }
+ else
+ {
+ // assume this is path fo file, so get the extension
+ imageType = Path.GetExtension(inputImage).Trim('.');
+
+ try
+ {
+ stream = File.Open(inputImage, FileMode.Open);
+ }
+ catch
+ {
+ imagePartType = default;
+ error = "Invalid Image path";
+ return null;
+ }
+ }
+
+ switch (imageType)
+ {
+ case "jpg":
+ case "jpeg":
+ imagePartType = ImagePartType.Jpeg;
+ break;
+ case "png":
+ imagePartType = ImagePartType.Png;
+ break;
+ case "tif":
+ case "tiff":
+ imagePartType = ImagePartType.Tiff;
+ break;
+ case "bmp":
+ imagePartType = ImagePartType.Bmp;
+ break;
+ default:
+ imagePartType = default;
+ error = "Invalid image type";
+ return null;
+ }
+
+ error = string.Empty;
+ return stream;
+ }
+
+ ///
+ /// Method processes internal paragraphs (marked with a prefix)
+ ///
+ /// Source element
+ /// Data element with content
+ /// Error indicator
+ /// Processed element
+ private static object ProcessAParagraph(XElement element, XElement data, TemplateError templateError)
+ {
+ var para = element.Descendants(A.r).FirstOrDefault();
+
+ var xPath = (string)element.Attribute(PA.Select);
+ var optionalString = (string)element.Attribute(PA.Optional);
+ var optional = (optionalString != null && optionalString.ToLower() == "true");
+
+ string newValue;
+ try
+ {
+ newValue = EvaluateXPathToString(data, xPath, optional);
+ }
+ catch (XPathException e)
+ {
+ return CreateContextErrorMessage(element, "XPathException: " + e.Message, templateError);
+ }
+
+ if (para != null)
+ {
+ var p = new XElement(A.r, para.Elements(A.rPr));
+ foreach (var line in newValue.Split('\n'))
+ {
+ p.Add(new XElement(A.t,
+ para.Elements(A.t).Elements(A.rPr).FirstOrDefault(),
+ line));
+ }
+ return p;
+ }
+
+ return null;
}
- static object ContentReplacementTransform(XNode node, XElement data, TemplateError templateError)
+ static object ContentReplacementTransform(XNode node, XElement data, TemplateError templateError, OpenXmlPart part)
{
- XElement element = node as XElement;
- if (element != null)
+ if (node is XElement element)
{
+ // TODO: need to figure out potentially better place for handling Alternate Content
+ if (element.Name == MC.AlternateContent)
+ {
+ // assign new DrawingML object id (for repeated content)
+ var docProperties = element
+ .Descendants(W.drawing)
+ .Descendants(WP.docPr)
+ .FirstOrDefault();
+ docProperties?.SetAttributeValue(NoNamespace.id, InvalidImageId);
+
+ // get the fallback picture element
+ var picture = element
+ .Descendants(MC.Fallback)
+ .Descendants(W.pict)
+ .FirstOrDefault();
+ if (picture != null)
+ {
+ // get the shape type element (it's okay not to have it,
+ // as the shape might use the type defined previously and left
+ // in other shape after copy-paste operation in the editor)
+ var shapeType = picture.Descendants(VML.shapetype).FirstOrDefault();
+ var shape = picture.Descendants(VML.shape).FirstOrDefault();
+
+ if (shape != null)
+ {
+ shape.SetAttributeValue(NoNamespace.id, GetNextShapeId());
+
+ if (shapeType != null)
+ {
+ // get next available shape type
+ var spt = GetNextShapeType();
+ var shapeTypeId = string.Format("_x0000_t{0}", spt);
+
+ // replace the attribute in shape type and in the corresponding shapes
+ shapeType.SetAttributeValue(O.spt, string.Format("{0}", spt));
+ shapeType.SetAttributeValue(NoNamespace.id, shapeTypeId);
+
+ shape.SetAttributeValue(NoNamespace.type, string.Format("#{0}", shapeTypeId));
+ }
+ }
+ }
+ }
+ if (element.Name == PA.Image)
+ {
+ return ProcessImageContent(element, data, templateError, part);
+ }
if (element.Name == PA.Content)
{
- XElement para = element.Descendants(W.p).FirstOrDefault();
- XElement run = element.Descendants(W.r).FirstOrDefault();
+ if (element.Descendants(A.r).FirstOrDefault() != null)
+ {
+ return ProcessAParagraph(element, data, templateError);
+ }
+
+ var para = element.Descendants(W.p).FirstOrDefault();
+ var run = element.Descendants(W.r).FirstOrDefault();
var xPath = (string) element.Attribute(PA.Select);
var optionalString = (string) element.Attribute(PA.Optional);
- bool optional = (optionalString != null && optionalString.ToLower() == "true");
+ var optional = (optionalString != null && optionalString.ToLower() == "true");
string newValue;
try
@@ -617,7 +1244,7 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
if (para != null)
{
- XElement p = new XElement(W.p, para.Elements(W.pPr));
+ var p = new XElement(W.p, para.Elements(W.pPr));
foreach(string line in newValue.Split('\n'))
{
p.Add(new XElement(W.r,
@@ -629,8 +1256,8 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
}
else
{
- List list = new List();
- foreach(string line in newValue.Split('\n'))
+ var list = new List();
+ foreach(var line in newValue.Split('\n'))
{
list.Add(new XElement(W.r,
run.Elements().Where(e => e.Name != W.t),
@@ -642,9 +1269,10 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
}
if (element.Name == PA.Repeat)
{
- string selector = (string)element.Attribute(PA.Select);
+ var selector = (string)element.Attribute(PA.Select);
var optionalString = (string)element.Attribute(PA.Optional);
- bool optional = (optionalString != null && optionalString.ToLower() == "true");
+ var optional = (optionalString != null && optionalString.ToLower() == "true");
+ var alignmentOption = (string)element.Attribute(PA.Align) ?? "vertical";
IEnumerable repeatingData;
try
@@ -672,12 +1300,32 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
{
var content = element
.Elements()
- .Select(e => ContentReplacementTransform(e, d, templateError))
+ .Select(e => ContentReplacementTransform(e, d, templateError, part))
.ToList();
return content;
})
.ToList();
- return newContent;
+ switch (alignmentOption.ToLower())
+ {
+ case "horizontal":
+ // keep the properties of first paragraph
+ var pPr = new XElement(W.p, newContent.First())
+ .Elements(W.p)
+ .FirstOrDefault()
+ .Elements(W.pPr)
+ .FirstOrDefault();
+ // create runs from repeated content
+ var runs = newContent.Select(x =>
+ {
+ var run = new XElement(W.p, x);
+ return run.Descendants(W.r).FirstOrDefault();
+ });
+ return pPr == null ? new XElement(W.p, runs) : new XElement(W.p, pPr, runs);
+ case "vertical":
+ return newContent;
+ default:
+ return CreateContextErrorMessage(element, "Repeat: Invalid Align option", templateError);
+ }
}
if (element.Name == PA.Table)
{
@@ -690,22 +1338,22 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
{
return CreateContextErrorMessage(element, "XPathException: " + e.Message, templateError);
}
- if (tableData.Count() == 0)
+ if (!tableData.Any())
return CreateContextErrorMessage(element, "Table Select returned no data", templateError);
- XElement table = element.Element(W.tbl);
- XElement protoRow = table.Elements(W.tr).Skip(1).FirstOrDefault();
+ var table = element.Element(W.tbl);
+ var protoRow = table.Elements(W.tr).Skip(1).FirstOrDefault();
var footerRowsBeforeTransform = table
.Elements(W.tr)
.Skip(2)
.ToList();
var footerRows = footerRowsBeforeTransform
- .Select(x => ContentReplacementTransform(x, data, templateError))
+ .Select(x => ContentReplacementTransform(x, data, templateError, part))
.ToList();
if (protoRow == null)
return CreateContextErrorMessage(element, string.Format("Table does not contain a prototype row"), templateError);
protoRow.Descendants(W.bookmarkStart).Remove();
protoRow.Descendants(W.bookmarkEnd).Remove();
- XElement newTable = new XElement(W.tbl,
+ var newTable = new XElement(W.tbl,
table.Elements().Where(e => e.Name != W.tr),
table.Elements(W.tr).FirstOrDefault(),
tableData.Select(d =>
@@ -715,16 +1363,30 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
.Select(tc =>
{
XElement paragraph = tc.Elements(W.p).FirstOrDefault();
- XElement cellRun = paragraph.Elements(W.r).FirstOrDefault();
- string xPath = paragraph.Value;
- string newValue = null;
+
+ // TODO: to check for other types (if needed, of course). Also, would be nice to refactor it, say, with
+ // TODO: different condition, for example, with switch case which checks the type of content.
+ if (paragraph == null)
+ {
+ // check if this is emebedded image
+ var image = tc.Elements(PA.Image).FirstOrDefault();
+ if (image != null)
+ {
+ // has to be wrapped as table cell element, since we are re-formatting the table
+ return new XElement(W.tc, ProcessImageContent(image, d, templateError, part));
+ }
+ }
+
+ var cellRun = paragraph.Elements(W.r).FirstOrDefault();
+ var xPath = paragraph.Value;
+ string newValue;
try
{
newValue = EvaluateXPathToString(d, xPath, false);
}
catch (XPathException e)
{
- XElement errorCell = new XElement(W.tc,
+ var errorCell = new XElement(W.tc,
tc.Elements().Where(z => z.Name != W.p),
new XElement(W.p,
paragraph.Element(W.pPr),
@@ -732,7 +1394,7 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
return errorCell;
}
- XElement newCell = new XElement(W.tc,
+ var newCell = new XElement(W.tc,
tc.Elements().Where(z => z.Name != W.p),
new XElement(W.p,
paragraph.Element(W.pPr),
@@ -747,7 +1409,7 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
}
if (element.Name == PA.Conditional)
{
- string xPath = (string)element.Attribute(PA.Select);
+ var xPath = (string)element.Attribute(PA.Select);
var match = (string)element.Attribute(PA.Match);
var notMatch = (string)element.Attribute(PA.NotMatch);
@@ -756,8 +1418,7 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
if (match != null && notMatch != null)
return CreateContextErrorMessage(element, "Conditional: Cannot specify both Match and NotMatch", templateError);
- string testValue = null;
-
+ string testValue;
try
{
testValue = EvaluateXPathToString(data, xPath, false);
@@ -769,27 +1430,24 @@ static object ContentReplacementTransform(XNode node, XElement data, TemplateErr
if ((match != null && testValue == match) || (notMatch != null && testValue != notMatch))
{
- var content = element.Elements().Select(e => ContentReplacementTransform(e, data, templateError));
+ var content = element.Elements().Select(e => ContentReplacementTransform(e, data, templateError, part));
return content;
}
return null;
}
return new XElement(element.Name,
element.Attributes(),
- element.Nodes().Select(n => ContentReplacementTransform(n, data, templateError)));
+ element.Nodes().Select(n => ContentReplacementTransform(n, data, templateError, part)));
}
return node;
}
private static object CreateContextErrorMessage(XElement element, string errorMessage, TemplateError templateError)
{
- XElement para = element.Descendants(W.p).FirstOrDefault();
- XElement run = element.Descendants(W.r).FirstOrDefault();
+ var para = element.Descendants(W.p).FirstOrDefault();
+ var run = element.Descendants(W.r).FirstOrDefault();
var errorRun = CreateRunErrorMessage(errorMessage, templateError);
- if (para != null)
- return new XElement(W.p, errorRun);
- else
- return errorRun;
+ return para != null ? new XElement(W.p, errorRun) : errorRun;
}
private static XElement CreateRunErrorMessage(string errorMessage, TemplateError templateError)
@@ -821,7 +1479,7 @@ private static string EvaluateXPathToString(XElement element, string xPath, bool
try
{
//support some cells in the table may not have an xpath expression.
- if (String.IsNullOrWhiteSpace(xPath)) return String.Empty;
+ if (string.IsNullOrWhiteSpace(xPath)) return string.Empty;
xPathSelectResult = element.XPathEvaluate(xPath);
}
@@ -830,28 +1488,29 @@ private static string EvaluateXPathToString(XElement element, string xPath, bool
throw new XPathException("XPathException: " + e.Message, e);
}
- if ((xPathSelectResult is IEnumerable) && !(xPathSelectResult is string))
+ if (xPathSelectResult is IEnumerable enumerable and not string)
{
- var selectedData = ((IEnumerable) xPathSelectResult).Cast();
+ var selectedData = enumerable.Cast();
if (!selectedData.Any())
{
if (optional) return string.Empty;
- throw new XPathException(string.Format("XPath expression ({0}) returned no results", xPath));
+ throw new XPathException($"XPath expression ({xPath}) returned no results");
}
if (selectedData.Count() > 1)
{
- throw new XPathException(string.Format("XPath expression ({0}) returned more than one node", xPath));
+ throw new XPathException($"XPath expression ({xPath}) returned more than one node");
}
- XObject selectedDatum = selectedData.First();
-
- if (selectedDatum is XElement) return ((XElement) selectedDatum).Value;
-
- if (selectedDatum is XAttribute) return ((XAttribute) selectedDatum).Value;
+ switch (selectedData.First())
+ {
+ case XElement xElement:
+ return xElement.Value;
+ case XAttribute attribute:
+ return attribute.Value;
+ }
}
return xPathSelectResult.ToString();
-
}
}
}
diff --git a/RELEASE_NOTES.md b/RELEASE_NOTES.md
index 70d43b60..ebb14baa 100644
--- a/RELEASE_NOTES.md
+++ b/RELEASE_NOTES.md
@@ -1,3 +1,6 @@
+#### 1.7.0-beta1 - Jul 5, 2021
+- DocumentAssembler: Support for images [#31](https://github.com/sergey-tihon/Clippit/pull/31)
+
#### 1.6.1 - Jul 11, 2021
- New docs site generated by DocFx [#33](https://github.com/sergey-tihon/Clippit/pull/33)
- Tests suite cleanup, samples converted to tests [#32](https://github.com/sergey-tihon/Clippit/pull/32)
diff --git a/TestFiles/DA-Data-WithImages.xml b/TestFiles/DA-Data-WithImages.xml
new file mode 100644
index 00000000..b0222b91
--- /dev/null
+++ b/TestFiles/DA-Data-WithImages.xml
@@ -0,0 +1,28 @@
+
+
+ 1
+ Cheryl
+ True
+ ../../../../TestFiles/img.png
+ ../../../../TestFiles/T0936_files/image001.png
+
+
+ Unicycle
+ 3
+ 9/5/2001
+ ../../../../TestFiles/img2.png
+
+
+ Tricycle
+ 3
+ 8/6/2000
+ 
+
+
+ 6
+
+
\ No newline at end of file
diff --git a/TestFiles/DA-Data-WithImagesInvalidImageDataFormat.xml b/TestFiles/DA-Data-WithImagesInvalidImageDataFormat.xml
new file mode 100644
index 00000000..7b35fc41
--- /dev/null
+++ b/TestFiles/DA-Data-WithImagesInvalidImageDataFormat.xml
@@ -0,0 +1,28 @@
+
+
+ 1
+ Cheryl
+ True
+ ../../../../TestFiles/img.png
+
+
+ Unicycle
+ 3
+ 9/5/2001
+ ../../../../TestFiles/img2.png
+
+
+ Tricycle
+ 3
+ 8/6/2000
+ 
+
+
+
+ 6
+
+
\ No newline at end of file
diff --git a/TestFiles/DA-Data-WithImagesInvalidMIMEType.xml b/TestFiles/DA-Data-WithImagesInvalidMIMEType.xml
new file mode 100644
index 00000000..1a765357
--- /dev/null
+++ b/TestFiles/DA-Data-WithImagesInvalidMIMEType.xml
@@ -0,0 +1,27 @@
+
+
+ 1
+ Cheryl
+ True
+ ../../../../TestFiles/img.png
+
+
+ Unicycle
+ 3
+ 9/5/2001
+ ../../../../TestFiles/img2.png
+
+
+ Tricycle
+ 3
+ 8/6/2000
+ 
+
+
+ 6
+
+
\ No newline at end of file
diff --git a/TestFiles/DA-Data-WithImagesInvalidPath.xml b/TestFiles/DA-Data-WithImagesInvalidPath.xml
new file mode 100644
index 00000000..85129613
--- /dev/null
+++ b/TestFiles/DA-Data-WithImagesInvalidPath.xml
@@ -0,0 +1,28 @@
+
+
+ 1
+ Cheryl
+ True
+ ../../../../TestFiles/invalid.png
+ ../../../../TestFiles/invalid.png
+
+
+ Unicycle
+ 3
+ 9/5/2001
+ ../../../../TestFiles/img2.png
+
+
+ Tricycle
+ 3
+ 8/6/2000
+ 
+
+
+ 6
+
+
\ No newline at end of file
diff --git a/TestFiles/DA267-Repeat-HorizontalAlignType.docx b/TestFiles/DA267-Repeat-HorizontalAlignType.docx
new file mode 100644
index 00000000..90da4cf6
Binary files /dev/null and b/TestFiles/DA267-Repeat-HorizontalAlignType.docx differ
diff --git a/TestFiles/DA268-Repeat-VerticalAlignType.docx b/TestFiles/DA268-Repeat-VerticalAlignType.docx
new file mode 100644
index 00000000..2ef83762
Binary files /dev/null and b/TestFiles/DA268-Repeat-VerticalAlignType.docx differ
diff --git a/TestFiles/DA269-Repeat-InvalidAlignType.docx b/TestFiles/DA269-Repeat-InvalidAlignType.docx
new file mode 100644
index 00000000..1139aeac
Binary files /dev/null and b/TestFiles/DA269-Repeat-InvalidAlignType.docx differ
diff --git a/TestFiles/DA270-ImageSelect.docx b/TestFiles/DA270-ImageSelect.docx
new file mode 100644
index 00000000..fac3e305
Binary files /dev/null and b/TestFiles/DA270-ImageSelect.docx differ
diff --git a/TestFiles/DA270A-ImageSelect.docx b/TestFiles/DA270A-ImageSelect.docx
new file mode 100644
index 00000000..2adf96f3
Binary files /dev/null and b/TestFiles/DA270A-ImageSelect.docx differ
diff --git a/TestFiles/DA271-ImageSelectWithRepeat.docx b/TestFiles/DA271-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..65ee59a0
Binary files /dev/null and b/TestFiles/DA271-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA271A-ImageSelectWithRepeat.docx b/TestFiles/DA271A-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..7c75c293
Binary files /dev/null and b/TestFiles/DA271A-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA272-ImageSelectWithRepeatHorizontalAlign.docx b/TestFiles/DA272-ImageSelectWithRepeatHorizontalAlign.docx
new file mode 100644
index 00000000..740130c9
Binary files /dev/null and b/TestFiles/DA272-ImageSelectWithRepeatHorizontalAlign.docx differ
diff --git a/TestFiles/DA272A-ImageSelectWithRepeatHorizontalAlign.docx b/TestFiles/DA272A-ImageSelectWithRepeatHorizontalAlign.docx
new file mode 100644
index 00000000..3bb1bbfa
Binary files /dev/null and b/TestFiles/DA272A-ImageSelectWithRepeatHorizontalAlign.docx differ
diff --git a/TestFiles/DA273-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx b/TestFiles/DA273-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx
new file mode 100644
index 00000000..c717101f
Binary files /dev/null and b/TestFiles/DA273-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx differ
diff --git a/TestFiles/DA273A-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx b/TestFiles/DA273A-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx
new file mode 100644
index 00000000..aa8e15d6
Binary files /dev/null and b/TestFiles/DA273A-ImageSelectInsideTextBoxWithRepeatVerticalAlign.docx differ
diff --git a/TestFiles/DA274-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx b/TestFiles/DA274-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx
new file mode 100644
index 00000000..cd2ccab4
Binary files /dev/null and b/TestFiles/DA274-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx differ
diff --git a/TestFiles/DA274A-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx b/TestFiles/DA274A-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx
new file mode 100644
index 00000000..dba63a74
Binary files /dev/null and b/TestFiles/DA274A-ImageSelectInsideTextBoxWithRepeatHorizontalAlign.docx differ
diff --git a/TestFiles/DA275-ImageSelectWithRepeatInvalidAlign.docx b/TestFiles/DA275-ImageSelectWithRepeatInvalidAlign.docx
new file mode 100644
index 00000000..e05e5583
Binary files /dev/null and b/TestFiles/DA275-ImageSelectWithRepeatInvalidAlign.docx differ
diff --git a/TestFiles/DA275A-ImageSelectWithRepeatInvalidAlign.docx b/TestFiles/DA275A-ImageSelectWithRepeatInvalidAlign.docx
new file mode 100644
index 00000000..8e7bf525
Binary files /dev/null and b/TestFiles/DA275A-ImageSelectWithRepeatInvalidAlign.docx differ
diff --git a/TestFiles/DA276-ImageSelectInsideTable.docx b/TestFiles/DA276-ImageSelectInsideTable.docx
new file mode 100644
index 00000000..552e873b
Binary files /dev/null and b/TestFiles/DA276-ImageSelectInsideTable.docx differ
diff --git a/TestFiles/DA276A-ImageSelectInsideTable.docx b/TestFiles/DA276A-ImageSelectInsideTable.docx
new file mode 100644
index 00000000..e8a6261f
Binary files /dev/null and b/TestFiles/DA276A-ImageSelectInsideTable.docx differ
diff --git a/TestFiles/DA277-ImageSelectMissingOrInvalidPictureContent.docx b/TestFiles/DA277-ImageSelectMissingOrInvalidPictureContent.docx
new file mode 100644
index 00000000..72127c40
Binary files /dev/null and b/TestFiles/DA277-ImageSelectMissingOrInvalidPictureContent.docx differ
diff --git a/TestFiles/DA277A-ImageSelectMissingOrInvalidPictureContent.docx b/TestFiles/DA277A-ImageSelectMissingOrInvalidPictureContent.docx
new file mode 100644
index 00000000..8d6be62b
Binary files /dev/null and b/TestFiles/DA277A-ImageSelectMissingOrInvalidPictureContent.docx differ
diff --git a/TestFiles/DA278-ImageSelect.docx b/TestFiles/DA278-ImageSelect.docx
new file mode 100644
index 00000000..1e578579
Binary files /dev/null and b/TestFiles/DA278-ImageSelect.docx differ
diff --git a/TestFiles/DA278A-ImageSelect.docx b/TestFiles/DA278A-ImageSelect.docx
new file mode 100644
index 00000000..d6315ea3
Binary files /dev/null and b/TestFiles/DA278A-ImageSelect.docx differ
diff --git a/TestFiles/DA279-ImageSelectWithRepeat.docx b/TestFiles/DA279-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..65ee59a0
Binary files /dev/null and b/TestFiles/DA279-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA279A-ImageSelectWithRepeat.docx b/TestFiles/DA279A-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..7752f197
Binary files /dev/null and b/TestFiles/DA279A-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA280-ImageSelectWithRepeat.docx b/TestFiles/DA280-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..65ee59a0
Binary files /dev/null and b/TestFiles/DA280-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA280A-ImageSelectWithRepeat.docx b/TestFiles/DA280A-ImageSelectWithRepeat.docx
new file mode 100644
index 00000000..aff28980
Binary files /dev/null and b/TestFiles/DA280A-ImageSelectWithRepeat.docx differ
diff --git a/TestFiles/DA281-ImageSelectExtraWhitespaceBeforeImageContent.docx b/TestFiles/DA281-ImageSelectExtraWhitespaceBeforeImageContent.docx
new file mode 100644
index 00000000..5e6c19fc
Binary files /dev/null and b/TestFiles/DA281-ImageSelectExtraWhitespaceBeforeImageContent.docx differ
diff --git a/TestFiles/DA281A-ImageSelectExtraWhitespaceBeforeImageContent.docx b/TestFiles/DA281A-ImageSelectExtraWhitespaceBeforeImageContent.docx
new file mode 100644
index 00000000..3531057f
Binary files /dev/null and b/TestFiles/DA281A-ImageSelectExtraWhitespaceBeforeImageContent.docx differ
diff --git a/TestFiles/DA282-ImageSelectWithHeader.docx b/TestFiles/DA282-ImageSelectWithHeader.docx
new file mode 100644
index 00000000..1082a1bb
Binary files /dev/null and b/TestFiles/DA282-ImageSelectWithHeader.docx differ
diff --git a/TestFiles/DA282A-ImageSelectWithHeader.docx b/TestFiles/DA282A-ImageSelectWithHeader.docx
new file mode 100644
index 00000000..b7c30c26
Binary files /dev/null and b/TestFiles/DA282A-ImageSelectWithHeader.docx differ
diff --git a/TestFiles/DA283-ImageSelectWithFooter.docx b/TestFiles/DA283-ImageSelectWithFooter.docx
new file mode 100644
index 00000000..0008d8d4
Binary files /dev/null and b/TestFiles/DA283-ImageSelectWithFooter.docx differ
diff --git a/TestFiles/DA283A-ImageSelectWithFooter.docx b/TestFiles/DA283A-ImageSelectWithFooter.docx
new file mode 100644
index 00000000..d8715c10
Binary files /dev/null and b/TestFiles/DA283A-ImageSelectWithFooter.docx differ
diff --git a/TestFiles/DA284-ImageSelectWithHeaderAndFooter.docx b/TestFiles/DA284-ImageSelectWithHeaderAndFooter.docx
new file mode 100644
index 00000000..3b0a2266
Binary files /dev/null and b/TestFiles/DA284-ImageSelectWithHeaderAndFooter.docx differ
diff --git a/TestFiles/DA284A-ImageSelectWithHeaderAndFooter.docx b/TestFiles/DA284A-ImageSelectWithHeaderAndFooter.docx
new file mode 100644
index 00000000..2b157c45
Binary files /dev/null and b/TestFiles/DA284A-ImageSelectWithHeaderAndFooter.docx differ
diff --git a/TestFiles/DA285-ImageSelectNoParagraphFollowedAfterMetadata.docx b/TestFiles/DA285-ImageSelectNoParagraphFollowedAfterMetadata.docx
new file mode 100644
index 00000000..123b8152
Binary files /dev/null and b/TestFiles/DA285-ImageSelectNoParagraphFollowedAfterMetadata.docx differ
diff --git a/TestFiles/DA285A-ImageSelectNoParagraphFollowedAfterMetadata.docx b/TestFiles/DA285A-ImageSelectNoParagraphFollowedAfterMetadata.docx
new file mode 100644
index 00000000..9de4810a
Binary files /dev/null and b/TestFiles/DA285A-ImageSelectNoParagraphFollowedAfterMetadata.docx differ
diff --git a/docs/images/word/documentassembler/image1.jpg b/docs/images/word/documentassembler/image1.jpg
new file mode 100644
index 00000000..8f3f4046
Binary files /dev/null and b/docs/images/word/documentassembler/image1.jpg differ
diff --git a/docs/images/word/documentassembler/image2.png b/docs/images/word/documentassembler/image2.png
new file mode 100644
index 00000000..6a97b599
Binary files /dev/null and b/docs/images/word/documentassembler/image2.png differ
diff --git a/docs/images/word/documentassembler/image3.png b/docs/images/word/documentassembler/image3.png
new file mode 100644
index 00000000..0faca5ce
Binary files /dev/null and b/docs/images/word/documentassembler/image3.png differ
diff --git a/docs/images/word/documentassembler/image4.png b/docs/images/word/documentassembler/image4.png
new file mode 100644
index 00000000..0450d8d9
Binary files /dev/null and b/docs/images/word/documentassembler/image4.png differ
diff --git a/docs/tutorials/word/DocumentAssembler_ImagesSupport.md b/docs/tutorials/word/DocumentAssembler_ImagesSupport.md
new file mode 100644
index 00000000..ab28e77f
--- /dev/null
+++ b/docs/tutorials/word/DocumentAssembler_ImagesSupport.md
@@ -0,0 +1,30 @@
+---
+uid: Tutorial.Word.DocumentAssembler.ImagesSupport
+---
+
+# Key highlights from [#31](https://github.com/sergey-tihon/Clippit/pull/31#issuecomment-874335292)
+
+1. Image can be provided either by base64-encoded string or by specifying the filename. The Assembler will have to figure out the image type, based on either MIME or file extension. Here are both examples:
+
+ - `../../md-logo.png`
+ - `…`
+
+2. The image content control will be surrounded by new type of select (in similar to repeat or conditional) – `Image Select` and `EndImage`. I found this approach easier than dealing just with the image content control. Here is the example:
+
+
+
+3. When used in repeated context, the image is used with relative XPath (in similar to other fields):
+
+
+
+4. Very similar situation with the Table:
+
+
+
+5. Here is example of data xml file:
+
+
+
+6. There is still a lot to be improved and some issues to be resolved, such as using templated image in header/footer, managing image size (preserve original/modify based on template/maintain aspect ratio etc.)
+
+7. Samples can be found in `DocumentAssemblerTests.cs` merged in [#31](https://github.com/sergey-tihon/Clippit/pull/31)
\ No newline at end of file
diff --git a/docs/tutorials/word/toc.yml b/docs/tutorials/word/toc.yml
index 7ebe3abc..e013d9a4 100644
--- a/docs/tutorials/word/toc.yml
+++ b/docs/tutorials/word/toc.yml
@@ -4,3 +4,7 @@
href: DocumentBuilder_CustomISource.md
- name: TableCellSource
href: DocumentBuilder_TableCellSource.md
+- name: DocumentAssembler
+ items:
+ - name: Images Support
+ href: DocumentAssembler_ImagesSupport.md
\ No newline at end of file