自定义形状

使用编辑点更改形状

以正方形为例。在 PowerPoint 中,使用 编辑点,您可以

  • 将正方形的角向内或向外移动
  • 指定角或点的曲率
  • 向正方形添加新点
  • 操作正方形上的点等

本质上,您可以在任何形状上执行上述任务。使用编辑点,您可以更改一个形状或从现有形状创建新形状。

形状编辑技巧

overview_image

在通过编辑点编辑 PowerPoint 形状之前,您可能需要考虑以下有关形状的要点:

  • 形状(或其路径)可以是闭合的,也可以是开放的。
  • 闭合形状没有起始或结束点。开放形状有起点和终点。
  • 所有形状至少由 2 个锚点组成,锚点之间通过线段相连
  • 线段可以是直线或曲线。锚点决定线段的性质。
  • 锚点可以是拐角点、直线点或平滑点:
    • 拐角点是两条直线在某个角度相交的点。
    • 平滑点是两条控制柄位于同一直线上,且线段以平滑曲线相连的点。此时所有控制柄与锚点的距离相等。
    • 直线点是两条控制柄位于同一直线上,且线段以平滑曲线相连的点。但在此情况下,控制柄与锚点的距离不必相等。
  • 通过移动或编辑锚点(从而改变线段的角度),可以改变形状的外观。

要通过编辑点编辑 PowerPoint 形状,Aspose.Slides 提供了 GeometryPath 类和 GeometryPath 类。

简单编辑操作

此 JavaScript 代码演示如何

添加线段 到路径的末尾

lineTo(point);
lineTo(x, y);

在指定位置添加线段

lineTo(point, index);
lineTo(x, y, index);

在路径末尾添加三次贝塞尔曲线

cubicBezierTo(point1, point2, point3);
cubicBezierTo(x1, y1, x2, y2, x3, y3);

在指定位置添加三次贝塞尔曲线

cubicBezierTo(point1, point2, point3);
cubicBezierTo(x1, y1, x2, y2, x3, y3);

在路径末尾添加二次贝塞尔曲线

quadraticBezierTo(point1, point2);
quadraticBezierTo(x1, y1, x2, y2);

在指定位置添加二次贝塞尔曲线

quadraticBezierTo(point1, point2, index);
quadraticBezierTo(x1, y1, x2, y2, index);

将给定弧线追加到路径

arcTo(width, heigth, startAngle, sweepAngle);

关闭路径的当前图形

closeFigure();

设置下一个点的位置

moveTo(point);
moveTo(x, y);

移除指定索引处的路径段

removeAt(index);

向形状添加自定义点

  1. 创建 GeometryShape 类的实例,并将其类型设置为 ShapeType.Rectangle
  2. 从形状获取 GeometryPath 类的实例。
  3. 在路径的两个顶部点之间添加一个新点。
  4. 在路径的两个底部点之间添加一个新点。
  5. 将路径应用于形状。

此 JavaScript 代码演示如何向形状添加自定义点:

var pres = new aspose.slides.Presentation();
try {
    var shape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Rectangle, 100, 100, 200, 100);
    var geometryPath = shape.getGeometryPaths()[0];
    geometryPath.lineTo(100, 50, 1);
    geometryPath.lineTo(100, 50, 4);
    shape.setGeometryPath(geometryPath);
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

example1_image

从形状中移除点

  1. 创建 GeometryShape 类的实例,并将其类型设置为 ShapeType.Heart
  2. 从形状获取 GeometryPath 类的实例。
  3. 移除路径的段落。
  4. 将路径应用于形状。

此 JavaScript 代码演示如何从形状中移除点:

var pres = new aspose.slides.Presentation();
try {
    var shape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Heart, 100, 100, 300, 300);
    var path = shape.getGeometryPaths()[0];
    path.removeAt(2);
    shape.setGeometryPath(path);
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

example2_image

创建自定义形状

  1. 计算形状的各个点。
  2. 创建 GeometryPath 类的实例。
  3. 使用这些点填充路径。
  4. 创建 GeometryShape 类的实例。
  5. 将路径应用于形状。

此 JavaScript 代码演示如何创建自定义形状:

var points = java.newInstanceSync("java.util.ArrayList");
var R = 100;
var r = 50;
var step = 72;
for (var angle = -90; angle < 270; angle += step) {
    var radians = angle * (java.getStaticFieldValue("java.lang.Math", "PI") / 180.0);
    var x = R * java.callStaticMethodSync("java.lang.Math", "cos", radians);
    var y = R * java.callStaticMethodSync("java.lang.Math", "sin", radians);
    points.add(java.newInstanceSync("com.aspose.slides.Point2DFloat", java.newFloat(x + R), java.newFloat(y + R)));
    radians = (java.getStaticFieldValue("java.lang.Math", "PI") * (angle + (step / 2))) / 180.0;
    x = r * java.callStaticMethodSync("java.lang.Math", "cos", radians);
    y = r * java.callStaticMethodSync("java.lang.Math", "sin", radians);
    points.add(java.newInstanceSync("com.aspose.slides.Point2DFloat", java.newFloat(x + R), java.newFloat(y + R)));
}
var starPath = new aspose.slides.GeometryPath();
starPath.moveTo(points.get(0));
for (var i = 1; i < points.size(); i++) {
    starPath.lineTo(points.get(i));
}
starPath.closeFigure();
var pres = new aspose.slides.Presentation();
try {
    var shape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Rectangle, 100, 100, R * 2, R * 2);
    shape.setGeometryPath(starPath);
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

example3_image

创建复合自定义形状

  1. 创建 GeometryShape 类的实例。
  2. 创建第一个 GeometryPath 类的实例。
  3. 创建第二个 GeometryPath 类的实例。
  4. 将这些路径应用于形状。

此 JavaScript 代码演示如何创建复合自定义形状:

var pres = new aspose.slides.Presentation();
try {
    var shape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Rectangle, 100, 100, 200, 100);
    var geometryPath0 = new aspose.slides.GeometryPath();
    geometryPath0.moveTo(0, 0);
    geometryPath0.lineTo(shape.getWidth(), 0);
    geometryPath0.lineTo(shape.getWidth(), shape.getHeight() / 3);
    geometryPath0.lineTo(0, shape.getHeight() / 3);
    geometryPath0.closeFigure();
    var geometryPath1 = new aspose.slides.GeometryPath();
    geometryPath1.moveTo(0, (shape.getHeight() / 3) * 2);
    geometryPath1.lineTo(shape.getWidth(), (shape.getHeight() / 3) * 2);
    geometryPath1.lineTo(shape.getWidth(), shape.getHeight());
    geometryPath1.lineTo(0, shape.getHeight());
    geometryPath1.closeFigure();
    shape.setGeometryPaths(java.newArray("com.aspose.slides.GeometryPath",[geometryPath0, geometryPath1]));
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

example4_image

创建带有圆角的自定义形状

此 JavaScript 代码演示如何创建带有圆角(向内)的自定义形状;

var shapeX = 20.0;
var shapeY = 20.0;
var shapeWidth = 300.0;
var shapeHeight = 200.0;
var leftTopSize = 50.0;
var rightTopSize = 20.0;
var rightBottomSize = 40.0;
var leftBottomSize = 10.0;
var pres = new aspose.slides.Presentation();
try {
    var childShape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Custom, shapeX, shapeY, shapeWidth, shapeHeight);
    var geometryPath = new aspose.slides.GeometryPath();
    var point1 = java.newInstanceSync("com.aspose.slides.Point2DFloat", leftTopSize, 0);
    var point2 = java.newInstanceSync("com.aspose.slides.Point2DFloat", shapeWidth - rightTopSize, 0);
    var point3 = java.newInstanceSync("com.aspose.slides.Point2DFloat", shapeWidth, shapeHeight - rightBottomSize);
    var point4 = java.newInstanceSync("com.aspose.slides.Point2DFloat", leftBottomSize, shapeHeight);
    var point5 = java.newInstanceSync("com.aspose.slides.Point2DFloat", 0, leftTopSize);
    geometryPath.moveTo(point1);
    geometryPath.lineTo(point2);
    geometryPath.arcTo(rightTopSize, rightTopSize, 180, -90);
    geometryPath.lineTo(point3);
    geometryPath.arcTo(rightBottomSize, rightBottomSize, -90, -90);
    geometryPath.lineTo(point4);
    geometryPath.arcTo(leftBottomSize, leftBottomSize, 0, -90);
    geometryPath.lineTo(point5);
    geometryPath.arcTo(leftTopSize, leftTopSize, 90, -90);
    geometryPath.closeFigure();
    childShape.setGeometryPath(geometryPath);
    pres.save("output.pptx", aspose.slides.SaveFormat.Pptx);
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

确定形状几何是否闭合

闭合形状是指其所有边都相连,形成单一的、没有间隙的边界。该形状可以是简单的几何图形,也可以是复杂的自定义轮廓。下面的代码示例展示了如何检查形状几何是否闭合:

function isGeometryClosed(geometryShape) 
{
    let isClosed = null;

    geometryShape.getGeometryPaths().forEach(geometryPath => {
        const pathData = geometryPath.getPathData();
        const dataLength = pathData.length;

        if (dataLength === 0) return;

        const lastSegment = pathData[dataLength - 1];
        isClosed = lastSegment.getPathCommand() === aspose.slides.PathCommandType.Close;

        if (!isClosed) return false;
    });

    return isClosed === true;
}

将 GeometryPath 转换为 java.awt.Shape

  1. 创建 GeometryShape 类的实例。
  2. 创建 java.awt.Shape 类的实例。
  3. 使用 ShapeUtiljava.awt.Shape 实例转换为 GeometryPath 实例。
  4. 将路径应用于形状。

此 JavaScript 代码——实现上述步骤——演示了 GeometryPathGraphicsPath 的转换过程:

var pres = new aspose.slides.Presentation();
try {
    // 创建新形状
    var shape = pres.getSlides().get_Item(0).getShapes().addAutoShape(aspose.slides.ShapeType.Rectangle, 100, 100, 300, 100);
    // 获取形状的几何路径
    var originalPath = shape.getGeometryPaths()[0];
    originalPath.setFillMode(aspose.slides.PathFillModeType.None);
    // 使用文本创建新的图形路径
    var graphicsPath;
    var font = java.newInstanceSync("java.awt.Font", "Arial", java.getStaticFieldValue("java.awt.Font", "PLAIN"), 40);
    var text = "Text in shape";
    var img = java.newInstanceSync("BufferedImage", 100, 100, java.getStaticFieldValue("BufferedImage", "TYPE_INT_ARGB"));
    var g2 = img.createGraphics();
    try {
        var glyphVector = font.createGlyphVector(g2.getFontRenderContext(), text);
        graphicsPath = glyphVector.getOutline(20.0, -glyphVector.getVisualBounds().getY() + 10);
    } finally {
        g2.dispose();
    }
    // 将图形路径转换为几何路径
    var textPath = aspose.slides.ShapeUtil.graphicsPathToGeometryPath(graphicsPath);
    textPath.setFillMode(aspose.slides.PathFillModeType.Normal);
    // 将新几何路径与原始几何路径的组合设置到形状
    shape.setGeometryPaths(java.newArray("com.aspose.slides.IGeometryPath", [originalPath, textPath]));
} finally {
    if (pres != null) {
        pres.dispose();
    }
}

example5_image

FAQ

替换几何后填充和轮廓会怎样?

样式仍然保留在形状上;仅轮廓会改变。填充和轮廓会自动应用到新的几何上。

如何在旋转自定义形状时一起旋转其几何?

使用形状的 setRotation 方法;几何会随形状一起旋转,因为它绑定在形状自身的坐标系上。

我可以将自定义形状转换为图像以“锁定”结果吗?

可以。将所需的 slide 区域或 shape 本身导出为栅格格式,这样可以简化对复杂几何的后续处理。