在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设置)。

TextBodyBodyProperties中插入NormalAutoFit也没有帮助- FontScaleLineSpaceReduction的必要值在OpenXML中不计算。

那么,用TextBody设置ShapeExtents的最佳实践是什么?

是否有一个内置的方法来计算给定TextBodyShapeExtents ?(有一些内置方法的经验法则总比没有好)

我知道在做出任何更改后,PowerPoint将重新计算NormalAutoFit的值(至少对于更改之前和之后的一堆幻灯片),但是当演示文稿在进行更改之前开始时(或者如果它是在PowerPoint查看器中开始的),这没有帮助。

在OpenXML (PresentationML) PowerPoint (PPTX)中计算文本(TextBody)的大

选自Eric White's Forum

这是一个重要的(但可行的)任务。

经过多次实验,我发现System.Windows.Forms.TextRenderer中的文本度量方法给了我最好的结果。这是WmlToHtmlConverter使用的文本度量功能。您可以查看WmlToHtmlConverter中的代码,作为使用TextRenderer的一个示例。

以下是我根据Eric White的WmlToHtmlConverter,这篇文章和这篇文章为我的目的编写的代码。我用它来计算用于Word文档的OpenXml文本水印和图像水印的TextBox的尺寸。
    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);
    }