在OpenXML(PresentationML)PowerPoint(PPTX)

时间:2015-09-19 13:03:15

标签: c# powerpoint openxml openxml-sdk presentationml

我想创建一个包含来自数据库的数据的演示文稿。我设法获得在PowerPoint中打开的有效演示文稿(SDK 2.5的Open XML Productivity Tool对此有很大帮助)。但是,如何计算文本框形状的大小?我看到放置值的位置,但new Extents()默认为零宽度和高度。当我从现有演示文稿中获取Shape的某些值时,我可能会获得正确的高度(至少对于带有一行或固定行数的文本),但文本将溢出到右侧或在底部(取决于自动换行设置)。

同样在NormalAutoFit的{​​{1}}中插入BodyProperties无效 - TextBodyFontScale的必要值不会在OpenXML中计算

那么,使用LineSpaceReduction设置Extents Shape的最佳做法是什么?

是否有内置方法来计算给定TextBodyExtents的{​​{1}}? (使用内置方法的一些经验法则会比没有更好)

我知道PowerPoint会在进行任何更改后重新计算TextBody的值(至少在更改之前和之后的一堆幻灯片中),但这在演示文稿开始之前启动时无效进行了更改(或者如果它是使用PowerPoint查看器启动的)。

1 个答案:

答案 0 :(得分:4)

来自Eric White's Forum

  

这是一项非常重要(但可行)的任务。

     

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

以下是我根据Eric White的WmlToHtmlConverter,this postthis为我的目的提出的代码。我用它来计算文本水印的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);
    }