Tuesday, 18 February 2014

Fixing Word Margins in Reporting Services 08

We don't use Report Services with SQL Server, but the Report Viewer control to render reports from Azure table Storage.

That said, I believe the rendering engine is the same in both cases - and in both cases the 08 version renders header & footer content outside of the document margins.

See this Microsoft Connect discussion on the issue, and Microsoft's good 'ol "its by design" response :)

There really is no workaround you can perform in the Reports themselves - as tricking the layout to work in Word will in turn break the layout in other formats like PDF.

So crazy idea time... since a .docx file is really just a .zip fie containing a bunch of xml settings, I thought I could write a function that could modify the header & footer attributes within the file.

Lo and behold it worked really well. It works for us since we just use the Report Viewer control, and not the full-fledged Reporting Services - as we are manually working with document streams etc.

Below is a code snippet of the function. Note that it requires SharpZipLib for the zip handling. We are still on .NET 4.0 but if you are on 4.5 you could swap this out with the new ZipArchive class in the .NET framework.

Anthony.

private static Stream FixWordMargins(Stream document)
{
// Copy to a seekable stream for manipulation
var zipStream = new MemoryStream();
document.CopyTo(zipStream);
zipStream.Position = 0;
// A docx file is just a zip file, so open it
var zipFile = new ZipFile(zipStream);
// Look for document.xml, which contains the header & footer margins
var documentXmlFilePath = "word/document.xml";
var documentIndex = zipFile.FindEntry(documentXmlFilePath, true);
if (documentIndex > -1)
{
// Load the entry into an XML document for manipulation
var xmlDoc = new XmlDocument();
xmlDoc.Load(zipFile.GetInputStream(zipFile[documentIndex]));
// Update the page margin element
var pgMarEls = xmlDoc.GetElementsByTagName("w:pgMar");
if (pgMarEls.Count == 1)
{
var pgMarEl = pgMarEls[0];
// Set the header margin based on the top margin
var topAttr = pgMarEl.Attributes["w:top"];
if(topAttr != null)
{
var headerAttr = pgMarEl.Attributes["w:header"];
if (headerAttr == null)
{
headerAttr = xmlDoc.CreateAttribute("w:header");
pgMarEl.Attributes.Append(headerAttr);
}
headerAttr.Value = topAttr.Value;
}
// Set the footer margin based on the bottom margin
var bottomAttr = pgMarEl.Attributes["w:bottom"];
if (bottomAttr != null)
{
var footerAttr = pgMarEl.Attributes["w:footer"];
if (footerAttr == null)
{
footerAttr = xmlDoc.CreateAttribute("w:footer");
pgMarEl.Attributes.Append(footerAttr);
}
footerAttr.Value = bottomAttr.Value;
}
}
// Save the updated document.xml back to the zip file
zipFile.BeginUpdate();
var sds = new CustomStaticDataSource();
var outStream = new MemoryStream();
xmlDoc.Save(outStream);
outStream.Position = 0;
sds.SetStream(outStream);
zipFile.Add(sds, documentXmlFilePath);
zipFile.CommitUpdate();
zipFile.IsStreamOwner = false;
zipFile.Close();
zipStream.Position = 0;
}
return zipStream;
}
view raw gistfile1.cs hosted with ❤ by GitHub

No comments: