Pptx的形状转为WPF的Geometry
本文是将演示如何解析pptx文件的形状到WPF当中,并且绘制显示出来
安装Openxml sdk
首先,我们先安装nuget的openxml sdk,下面两种方式都可以安装:
- nuget包管理器控制台:
Install-Package DocumentFormat.OpenXml -Version 2.13.0
- csproj引用:
<PackageReference Include="DocumentFormat.OpenXml" Version="2.13.0" />
解析Pptx
我打算解析pptx中的五边形来作为演示效果,直接上代码:
MainWindow.xaml:
<Grid>
<Grid.RowDefinitions>
<RowDefinition />
<RowDefinition Height="2*"/>
</Grid.RowDefinitions>
<StackPanel Orientation="Horizontal" VerticalAlignment="Center" HorizontalAlignment="Center">
<TextBlock Text="pptx的文件路径:" VerticalAlignment="Center" FontSize="15" Margin="10"/>
<TextBox x:Name="FilePathText" Height="50" Width="300" Margin="0,0,10,0" TextWrapping="Wrap"/>
<Button x:Name="Button" Content="解析PPT" Click="Button_OnClick" Width="120" Height="40"/>
</StackPanel>
<Path x:Name="Path" Grid.Row="1" HorizontalAlignment="Center" VerticalAlignment="Center" Stroke="Blue"/>
</Grid>
MainWindow.xaml.cs:
private void PptxToGeometry(string filePath)
{
if (!File.Exists(filePath) || !filePath.EndsWith(".pptx", StringComparison.OrdinalIgnoreCase))
{
return;
}
using (var presentationDocument = PresentationDocument.Open(filePath, false))
{
var presentationPart = presentationDocument.PresentationPart;
var presentation = presentationPart?.Presentation;
var slideIdList = presentation?.SlideIdList;
if (slideIdList == null)
{
return;
}
foreach (var slideId in slideIdList.ChildElements.OfType<SlideId>())
{
var slidePart = (SlidePart)presentationPart.GetPartById(slideId.RelationshipId);
var slide = slidePart.Slide;
foreach (var shapeProperties in slide.Descendants<ShapeProperties>())
{
var presetGeometry = shapeProperties.GetFirstChild<PresetGeometry>();
if (presetGeometry != null && presetGeometry.Preset.HasValue)
{
if (presetGeometry.Preset == ShapeTypeValues.Pentagon)
{
var transform2D = shapeProperties.GetFirstChild<Transform2D>();
var extents = transform2D?.GetFirstChild<Extents>();
if (extents != null)
{
var width = extents.Cx;
var height = extents.Cy;
if (width.HasValue && height.HasValue)
{
var points = GetPentagonPoints(width.Value.EmuToPixel(), height.Value.EmuToPixel());
RenderGeometry(points);
}
}
}
}
}
}
}
}
/// <summary>
/// 获取五边形顶点坐标
/// </summary>
/// <param name="width"></param>
/// <param name="height"></param>
/// <returns></returns>
/// 该五边形定义出自ECMA-376-Fifth-Edition-Part-1-Fundamentals-And-Markup-Language-Reference
/// \OfficeOpenXML-DrawingMLGeometries文档的presetShapeDefinitions.xml
private List<Point> GetPentagonPoints(double width, double height)
{
var properties = new FormulaProperties(width, height);
//<avLst xmlns="http://schemas.openxmlformats.org/drawingml/2006/main">
// <gd name="hf" fmla="val 105146" />
// <gd name="vf" fmla="val 110557" />
//</avLst>
var hf = 105146d;
var vf = 110557d;
//<gdLst xmlns="http://schemas.openxmlformats.org/drawingml/2006/main">
// <gd name="swd2" fmla="*/ wd2 hf 100000" />
// <gd name="shd2" fmla="*/ hd2 vf 100000" />
// <gd name="svc" fmla="*/ vc vf 100000" />
// <gd name="dx1" fmla="cos swd2 1080000" />
// <gd name="dx2" fmla="cos swd2 18360000" />
// <gd name="dy1" fmla="sin shd2 1080000" />
// <gd name="dy2" fmla="sin shd2 18360000" />
// <gd name="x1" fmla="+- hc 0 dx1" />
// <gd name="x2" fmla="+- hc 0 dx2" />
// <gd name="x3" fmla="+- hc dx2 0" />
// <gd name="x4" fmla="+- hc dx1 0" />
// <gd name="y1" fmla="+- svc 0 dy1" />
// <gd name="y2" fmla="+- svc 0 dy2" />
// <gd name="it" fmla="*/ y1 dx2 dx1" />
//</gdLst>
// <gd name="swd2" fmla="*/ wd2 hf 100000" />
var swd2 = properties.wd2 * hf / 100000;
// <gd name="shd2" fmla="*/ hd2 vf 100000" />
var shd2 = properties.hd2 * vf / 100000;
// <gd name="svc" fmla="*/ vc vf 100000" />
var svc = properties.vc * vf / 100000;
// <gd name="dx1" fmla="cos swd2 1080000" />
var dx1 = Cos(swd2, 1080000);
// <gd name="dx2" fmla="cos swd2 18360000" />
var dx2 = Cos(swd2, 18360000);
// <gd name="dy1" fmla="sin shd2 1080000" />
var dy1 = Sin(shd2, 1080000);
// <gd name="dy2" fmla="sin shd2 18360000" />
var dy2 = Sin(shd2, 18360000);
// <gd name="x1" fmla="+- hc 0 dx1" />
var x1 = properties.hc - dx1;
// <gd name="x2" fmla="+- hc 0 dx2" />
var x2 = properties.hc - dx2;
// <gd name="x3" fmla="+- hc dx2 0" />
var x3 = properties.hc + dx2;
// <gd name="x4" fmla="+- hc dx1 0" />
var x4 = properties.hc + dx1;
// <gd name="y1" fmla="+- svc 0 dy1" />
var y1 = svc - dy1;
// <gd name="y2" fmla="+- svc 0 dy2" />
var y2 = svc - dy2;
// <gd name="it" fmla="*/ y1 dx2 dx1" />
// <pathLst xmlns="http://schemas.openxmlformats.org/drawingml/2006/main">
// <path>
// <moveTo>
// <pt x="x1" y="y1" />
// </moveTo>
// <lnTo>
// <pt x="hc" y="t" />
// </lnTo>
// <lnTo>
// <pt x="x4" y="y1" />
// </lnTo>
// <lnTo>
// <pt x="x3" y="y2" />
// </lnTo>
// <lnTo>
// <pt x="x2" y="y2" />
// </lnTo>
// <close />
// </path>
//</pathLst>
var points = new List<Point>(5)
{
new Point(x1, y1),
new Point(properties.hc,properties.t),
new Point(x4, y1),
new Point(x3, y2),
new Point(x2, y2),
};
return points;
}
private void RenderGeometry(List<Point> points)
{
if (points.Count > 0)
{
var streamGeometry = new StreamGeometry();
using var context = streamGeometry.Open();
context.BeginFigure(points[0], true, true);
context.PolyLineTo(points, true, true);
this.Path.Data = streamGeometry;
}
}
private void Button_OnClick(object sender, RoutedEventArgs e)
{
var filePath = @"C:\Users\Ryzen\Desktop\测验\五边形.pptx";
if (!string.IsNullOrEmpty(FilePathText.Text))
{
filePath = FilePathText.Text.Trim();
}
PptxToGeometry(filePath);
}
ShapeGeometryHelper.cs:
public static class ShapeGeometryHelper
{
public readonly struct FormulaProperties
{
public readonly double t;
public readonly double h;
public readonly double w;
public readonly double hd2;
public readonly double wd2;
public readonly double vc;
public readonly double hc;
public FormulaProperties(double width, double height)
{
t = 0;
w = width;
h = height;
hd2 = h / 2;
wd2 = w / 2;
vc = height / 2;
hc = width / 2;
}
}
public static long EmuToPixel(this long eum)
{
const long defaultDpi = 96;
return eum / 914400 * defaultDpi;
}
/// <summary>
/// OpenXml 三角函数的Sin函数:sin x y = (x * sin( y )) = (x * Math.Sin(y))
/// </summary>
/// <param name="x">ppt的数值</param>
/// <param name="y">ppt表示角度的值</param>
/// <returns></returns>
public static double Sin(double x, int y)
{
var angle = GetAngle(y);
return x * Math.Sin(angle);
}
/// <summary>
/// OpenXml 三角函数的Cos函数:cos x y = (x * cos( y )) = (x * Math.Cos(y))
/// </summary>
/// <param name="x">ppt的数值</param>
/// <param name="y">ppt表示角度的值</param>
/// <returns></returns>
public static double Cos(double x, int y)
{
var angle = GetAngle(y);
return x * Math.Cos(angle);
}
/// <summary>
/// OpenXml 三角函数的Tan函数:tan x y = (x * tan( y )) = (x * Math.Tan(y))
/// </summary>
/// <param name="x">ppt的数值</param>
/// <param name="y">ppt表示角度的值</param>
/// <returns></returns>
public static double Tan(double x, int y)
{
var angle = GetAngle(y);
return x * Math.Tan(angle);
}
/// <summary>
/// ppt的值转为角度
/// </summary>
/// <param name="value">ppt表示角度的值</param>
/// <returns></returns>
private static double GetAngle(int value)
{
var degree = value / 60000.0;
var angle = degree * Math.PI / 180;
return angle;
}
}
效果如下:

源码
BlogCodeSample/PptPolygonToWPFGeometry at main · ZhengDaoWang/BlogCodeSample
参考
C# dontet Office Open XML Unit Converter
C# dotnet 使用 OpenXml 解析 PPT 元素的坐标和宽度高度
C# dotnet 使用 OpenXml 解析 PPT 文件
Pptx的形状转为WPF的Geometry的更多相关文章
- Pptx的多路径形状转为WPF的Path
本文是将演示如何解析pptx文件的多路径的形状转换到WPF,绘制多个Shape的Path Shape Path 这是Pptx的[标注:弯曲曲线(无边框)]形状的OpenXml定义部分: <cal ...
- 【OpenXml】Pptx的边框虚线转为WPF的边框虚线
安装Openxml sdk 首先,我们先安装nuget的需要的有关的Openxml sdk,我们开源了解析pptx的Openxml拍平层,下面两种方式都可以安装: nuget包管理器控制台: Inst ...
- 使用不安全代码将 Bitmap 位图转为 WPF 的 ImageSource 以获得高性能和持续小的内存占用
在 WPF 中将一个现成的 Bitmap 位图转换成 ImageSource 用于显示一个麻烦的事儿,因为 WPF 并没有提供多少可以转过来的方法.不过产生 Bitmap 来源却非常多,比如屏幕截图. ...
- 【Openxml】将Openxml的椭圆弧线arcTo转为Svg的椭圆弧线
本文将介绍如何将OpenXml的actTo转为Svg的弧线(a) OpenXml的artTo 首先下面是一段OpenXml的arcTo弧线 <arcTo wR="152403" ...
- WPF 中的形状和基本绘图概述
本主题概述如何使用 Shape 对象绘图. Shape 是一种允许您在屏幕中绘制形状的 UIElement 类型. 由于它们是 UI 元素,因此 Shape 对象可以在 Panel 元素和大多数控件中 ...
- WPF 动画(形状、画刷)
一:形状 在WPF用户界面中,可以通过形状(Shape)来绘制直线.椭圆.矩形及一些多边形的类.通过这些基本的图像,组合成为复杂的图形. Shape类中,主要的形状有Rectangle(),Ellip ...
- 使用 WPF 做个 PowerPoint 系列 基于 OpenXML 解析实现 PPT 文本描边效果
本文是使用 WPF 做个 PowerPoint 系列的博客,本文来告诉大家如何解析 PPT 里面的文本描边效果,在 WPF 应用中绘制出来,实现像素级相同 背景知识 在开始之前,期望你了解了 PPT ...
- 【WPF学习】第四十三章 路径和几何图形
前面四章介绍了继承自Shape的类,包括Rectangle.Ellipse.Polygon以及Polyline.但还有一个继承自Shape的类尚未介绍,而且该类是到现在为止功能最强大的形状类,即Pat ...
- Three.js基础探寻六——文字形状与自定义形状
1.文字形状 说起3d文字想起了早年word里的一些艺术字: 时间真快. 那么TextGeometry可以用来创建三维的文字形状. 使用文字形状需要下载和引用额外的字体库.这里,我们以 helveti ...
随机推荐
- 🍎
江湖中有一本练了就能天下无敌的葵花宝典,大家都想得到它.如果有一天葵花宝典被公开了,人人都有机会练,到底是好事还是坏事呢? 这会成为一个灾难. 因为一个人拥有时,练不练是一个人的事.大家都拥有,练不练 ...
- SLAM导航及控制部分,源码公布
欢迎Fork,觉得不错就点个小星星吧 ROS源码 https://github.com/ShieldQiQi/TX2_StereoSLAM STM32驱动板源码 https://github.com/ ...
- POJ 2135 简单费用流
题意: 题意是一个人他要从牧场1走到牧场n然后在走回来,每条路径只走一次,问全程的最短路径是多少. 思路: 这个题目挺简单的吧,首先要保证每条边只能走一次,然后还要要求费用最 ...
- Photoshop 第二课 工具-钢笔的使用
钢笔的使用 钢笔→ 是一个非常实用(主要用于)但是非常难操作(会者不难哦~)的工具. 钢笔属性中有三种状态:1.路径:2.形状:3.像素.其中路径和形状是我们最常用的状态.路径是一条用来圈定需要操作的 ...
- ColyseusJS 轻量级多人游戏服务器开发框架 - 中文手册(上)
快速上手多人游戏服务器开发.后续会基于 Google Agones,更新相关 K8S 运维.大规模快速扩展专用游戏服务器的文章.拥抱️原生 Cloud-Native! 快速开始 在开始之前,让我们确保 ...
- 【python】Leetcode每日一题-丑数
[python]Leetcode每日一题-丑数 [题目描述] 给你一个整数 n ,请你判断 n 是否为 丑数 .如果是,返回 true :否则,返回 false . 丑数 就是只包含质因数 2.3 和 ...
- MyBatis Plus 实现多表分页模糊查询
项目中使用springboot+mybatis-plus来实现. 但是之前处理的时候都是一个功能,比如分页查询,条件查询,模糊查询. 这次将这个几个功能合起来就有点头疼,写下这边博客来记录自己碰到的问 ...
- C#中的类、方法和属性
这节讲C#中的类,方法,属性.这是编码中我们最直接打交道的三个结构. 类: 类(class)是面向对象中最基本的单元,它是一种抽象,对现实世界中事物的抽象,在C#中使用class关键字声明一 ...
- Codeforces Round #660 (Div. 2)
A. Captain Flint and Crew Recruitment 题意:定义了一种数(接近质数),这种数可以写成p*q并且p和q都是素数,问n是否可以写成四个不同的数的和,并且保证至少三个数 ...
- C++ primer plus读书笔记——第5章 循环和关系表达式
第5章 循环和关系表达式 1. cout.setf(ios_base::boolalpha); cout << (100 > 3) << endl;将输出true,而不是 ...