在OpenXML (PresentationML) PowerPoint (PPTX)中计算文本(TextBody)的大
本文关键字:文本 计算 TextBody 的大 PowerPoint OpenXML PresentationML PPTX | 更新日期: 2023-09-27 18:13:42
我想用来自数据库的数据创建一个演示文稿。我设法获得在PowerPoint中打开的有效演示文稿(SDK 2.5的open XML生产力工具在这方面有很大帮助)。但是,如何计算文本框形状的大小呢?我看到在哪里放值,但new Extents()
默认为零宽度和高度。当我从现有的演示文稿中获取Shape
的一些值时,我可能会得到正确的高度(至少对于只有一行或固定行数的文本),但是文本将溢出到右侧或底部(取决于wordwrap设置)。
在TextBody
的BodyProperties
中插入NormalAutoFit
也没有帮助- FontScale
和LineSpaceReduction
的必要值在OpenXML中不计算。
那么,用TextBody
设置Shape
的Extents
的最佳实践是什么?
是否有一个内置的方法来计算给定TextBody
或Shape
的Extents
?(有一些内置方法的经验法则总比没有好)
我知道在做出任何更改后,PowerPoint将重新计算NormalAutoFit
的值(至少对于更改之前和之后的一堆幻灯片),但是当演示文稿在进行更改之前开始时(或者如果它是在PowerPoint查看器中开始的),这没有帮助。
选自Eric White's Forum
以下是我根据Eric White的WmlToHtmlConverter,这篇文章和这篇文章为我的目的编写的代码。我用它来计算用于Word文档的OpenXml文本水印和图像水印的TextBox的尺寸。这是一个重要的(但可行的)任务。
经过多次实验,我发现System.Windows.Forms.TextRenderer中的文本度量方法给了我最好的结果。这是WmlToHtmlConverter使用的文本度量功能。您可以查看WmlToHtmlConverter中的代码,作为使用TextRenderer的一个示例。
private static D.Size pixelsToEmus(int widthPx, int heightPx, double resDpiX, double resDpiY, int zoomX, int zoomY)
{
const int emusPerInch = 914400;
const int emusPerCm = 360000;
const decimal maxWidthCm = 16.51m;
var widthEmus = (int)(widthPx / resDpiX * emusPerInch) * zoomX / 100;
var heightEmus = (int)(heightPx / resDpiY * emusPerInch) * zoomY / 100;
var maxWidthEmus = (int)(maxWidthCm * emusPerCm);
if (widthEmus > maxWidthEmus)
{
var ratio = ((decimal)heightEmus / (decimal)widthEmus);
widthEmus = maxWidthEmus;
heightEmus = (int)(widthEmus * ratio);
}
return new D.Size(widthEmus, heightEmus);
}
public static D.Size GetTextSize(this CWatermarkItemBase watermark, string runText)
{
var fs = watermark.GetFontStyle();
var sz = watermark.FontSize;
var proposedSize = new D.Size(int.MaxValue, int.MaxValue);
D.Size sf;
using (var ff = new D.FontFamily(watermark.FontFamily))
{
try
{
using (var f = new D.Font(ff, (float)sz, fs))
{
const TextFormatFlags tff = TextFormatFlags.NoPadding;
sf = TextRenderer.MeasureText(runText, f, proposedSize, tff);
}
}
catch (ArgumentException)
{
try
{
const D.FontStyle fs2 = D.FontStyle.Regular;
using (D.Font f = new D.Font(ff, (float)sz, fs2))
{
const TextFormatFlags tff = TextFormatFlags.NoPadding;
sf = TextRenderer.MeasureText(runText, f, proposedSize, tff);
}
}
catch (ArgumentException)
{
const D.FontStyle fs2 = D.FontStyle.Bold;
try
{
using (var f = new D.Font(ff, (float)sz, fs2))
{
const TextFormatFlags tff = TextFormatFlags.NoPadding;
sf = TextRenderer.MeasureText(runText, f, proposedSize, tff);
}
}
catch (ArgumentException)
{
// if both regular and bold fail, then get metrics for Times New Roman
// use the original FontStyle (in fs)
using (var ff2 = new D.FontFamily("Times New Roman"))
using (var f = new D.Font(ff2, (float)sz, fs))
{
const TextFormatFlags tff = TextFormatFlags.NoPadding;
sf = TextRenderer.MeasureText(runText, f, proposedSize, tff);
}
}
}
}
}
D.Size s2 = pixelsToEmus(sf.Width, sf.Height, 96, 96, 100, 100);
return s2;
}
public static D.Size GetImageSize(this CWatermarkItemImage watermarkItem)
{
var img = new BitmapImage(new Uri(watermarkItem.FilePath, UriKind.RelativeOrAbsolute));
return pixelsToEmus(img.PixelWidth, img.PixelHeight, img.DpiX, img.DpiY, watermarkItem.ZoomWidth, watermarkItem.ZoomHeight);
}