我想创建一个包含来自数据库的数据的演示文稿。我设法获得在PowerPoint中打开的有效演示文稿(SDK 2.5的Open XML Productivity Tool对此有很大帮助)。但是,如何计算文本框形状的大小?我看到放置值的位置,但new Extents()
默认为零宽度和高度。当我从现有演示文稿中获取Shape
的某些值时,我可能会获得正确的高度(至少对于带有一行或固定行数的文本),但文本将溢出到右侧或在底部(取决于自动换行设置)。
同样在NormalAutoFit
的{{1}}中插入BodyProperties
无效 - TextBody
和FontScale
的必要值不会在OpenXML中计算
那么,使用LineSpaceReduction
设置Extents
Shape
的最佳做法是什么?
是否有内置方法来计算给定TextBody
或Extents
的{{1}}? (使用内置方法的一些经验法则会比没有更好)
我知道PowerPoint会在进行任何更改后重新计算TextBody
的值(至少在更改之前和之后的一堆幻灯片中),但这在演示文稿开始之前启动时无效进行了更改(或者如果它是使用PowerPoint查看器启动的)。
答案 0 :(得分:4)
这是一项非常重要(但可行)的任务。
经过多次实验,我发现System.Windows.Forms.TextRenderer中的文本度量方法给了我最好的结果。这是WmlToHtmlConverter使用的文本度量标准功能。您可以查看WmlToHtmlConverter中的代码,作为使用TextRenderer的一个示例。
以下是我根据Eric White的WmlToHtmlConverter,this post和this为我的目的提出的代码。我用它来计算文本水印的TextBox和OpenXml for Word文档的图像水印的尺寸。
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);
}