/* | |
* ==================================================================== | |
* Licensed to the Apache Software Foundation (ASF) under one or more | |
* contributor license agreements. See the NOTICE file distributed with | |
* this work for additional information regarding copyright ownership. | |
* The ASF licenses this file to You under the Apache License, Version 2.0 | |
* (the "License"); you may not use this file except in compliance with | |
* the License. You may obtain a copy of the License at | |
* | |
* http://www.apache.org/licenses/LICENSE-2.0 | |
* | |
* Unless required by applicable law or agreed to in writing, software | |
* distributed under the License is distributed on an "AS IS" BASIS, | |
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
* See the License for the specific language governing permissions and | |
* limitations under the License. | |
* ==================================================================== | |
*/ | |
package org.apache.poi.xslf.usermodel; | |
import java.awt.geom.AffineTransform; | |
import java.awt.geom.Path2D; | |
import java.awt.geom.PathIterator; | |
import java.awt.geom.Rectangle2D; | |
import org.apache.poi.sl.usermodel.FreeformShape; | |
import org.apache.poi.util.Beta; | |
import org.apache.poi.util.Units; | |
import org.apache.xmlbeans.XmlObject; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTAdjPoint2D; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTCustomGeometry2D; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTGeomRect; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTNonVisualDrawingProps; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2D; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2DClose; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2DCubicBezierTo; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2DLineTo; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2DMoveTo; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTPath2DQuadBezierTo; | |
import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeProperties; | |
import org.openxmlformats.schemas.presentationml.x2006.main.CTShape; | |
import org.openxmlformats.schemas.presentationml.x2006.main.CTShapeNonVisual; | |
/** | |
* Represents a custom geometric shape. | |
* This shape will consist of a series of lines and curves described within a creation path. | |
*/ | |
@Beta | |
public class XSLFFreeformShape extends XSLFAutoShape | |
implements FreeformShape<XSLFShape,XSLFTextParagraph> { | |
/*package*/ XSLFFreeformShape(CTShape shape, XSLFSheet sheet) { | |
super(shape, sheet); | |
} | |
@Override | |
public int setPath(Path2D.Double path) { | |
CTPath2D ctPath = CTPath2D.Factory.newInstance(); | |
Rectangle2D bounds = path.getBounds2D(); | |
int x0 = Units.toEMU(bounds.getX()); | |
int y0 = Units.toEMU(bounds.getY()); | |
PathIterator it = path.getPathIterator(new AffineTransform()); | |
int numPoints = 0; | |
ctPath.setH(Units.toEMU(bounds.getHeight())); | |
ctPath.setW(Units.toEMU(bounds.getWidth())); | |
while (!it.isDone()) { | |
double[] vals = new double[6]; | |
int type = it.currentSegment(vals); | |
switch (type) { | |
case PathIterator.SEG_MOVETO: | |
CTAdjPoint2D mv = ctPath.addNewMoveTo().addNewPt(); | |
mv.setX(Units.toEMU(vals[0]) - x0); | |
mv.setY(Units.toEMU(vals[1]) - y0); | |
numPoints++; | |
break; | |
case PathIterator.SEG_LINETO: | |
CTAdjPoint2D ln = ctPath.addNewLnTo().addNewPt(); | |
ln.setX(Units.toEMU(vals[0]) - x0); | |
ln.setY(Units.toEMU(vals[1]) - y0); | |
numPoints++; | |
break; | |
case PathIterator.SEG_QUADTO: | |
CTPath2DQuadBezierTo qbez = ctPath.addNewQuadBezTo(); | |
CTAdjPoint2D qp1 = qbez.addNewPt(); | |
qp1.setX(Units.toEMU(vals[0]) - x0); | |
qp1.setY(Units.toEMU(vals[1]) - y0); | |
CTAdjPoint2D qp2 = qbez.addNewPt(); | |
qp2.setX(Units.toEMU(vals[2]) - x0); | |
qp2.setY(Units.toEMU(vals[3]) - y0); | |
numPoints += 2; | |
break; | |
case PathIterator.SEG_CUBICTO: | |
CTPath2DCubicBezierTo bez = ctPath.addNewCubicBezTo(); | |
CTAdjPoint2D p1 = bez.addNewPt(); | |
p1.setX(Units.toEMU(vals[0]) - x0); | |
p1.setY(Units.toEMU(vals[1]) - y0); | |
CTAdjPoint2D p2 = bez.addNewPt(); | |
p2.setX(Units.toEMU(vals[2]) - x0); | |
p2.setY(Units.toEMU(vals[3]) - y0); | |
CTAdjPoint2D p3 = bez.addNewPt(); | |
p3.setX(Units.toEMU(vals[4]) - x0); | |
p3.setY(Units.toEMU(vals[5]) - y0); | |
numPoints += 3; | |
break; | |
case PathIterator.SEG_CLOSE: | |
numPoints++; | |
ctPath.addNewClose(); | |
break; | |
default: | |
throw new IllegalStateException("Unrecognized path segment type: " + type); | |
} | |
it.next(); | |
} | |
XmlObject xo = getShapeProperties(); | |
if (!(xo instanceof CTShapeProperties)) { | |
return -1; | |
} | |
((CTShapeProperties)xo).getCustGeom().getPathLst().setPathArray(new CTPath2D[]{ctPath}); | |
setAnchor(bounds); | |
return numPoints; | |
} | |
@Override | |
public Path2D.Double getPath() { | |
Path2D.Double path = new Path2D.Double(); | |
Rectangle2D bounds = getAnchor(); | |
XmlObject xo = getShapeProperties(); | |
if (!(xo instanceof CTShapeProperties)) { | |
return null; | |
} | |
CTCustomGeometry2D geom = ((CTShapeProperties)xo).getCustGeom(); | |
for(CTPath2D spPath : geom.getPathLst().getPathArray()){ | |
double scaleW = bounds.getWidth() / Units.toPoints(spPath.getW()); | |
double scaleH = bounds.getHeight() / Units.toPoints(spPath.getH()); | |
for(XmlObject ch : spPath.selectPath("*")){ | |
if(ch instanceof CTPath2DMoveTo){ | |
CTAdjPoint2D pt = ((CTPath2DMoveTo)ch).getPt(); | |
path.moveTo( | |
(float) (Units.toPoints((Long) pt.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt.getY()) * scaleH)); | |
} else if (ch instanceof CTPath2DLineTo){ | |
CTAdjPoint2D pt = ((CTPath2DLineTo)ch).getPt(); | |
path.lineTo((float)Units.toPoints((Long)pt.getX()), | |
(float)Units.toPoints((Long)pt.getY())); | |
} else if (ch instanceof CTPath2DQuadBezierTo){ | |
CTPath2DQuadBezierTo bez = ((CTPath2DQuadBezierTo)ch); | |
CTAdjPoint2D pt1 = bez.getPtArray(0); | |
CTAdjPoint2D pt2 = bez.getPtArray(1); | |
path.quadTo( | |
(float) (Units.toPoints((Long) pt1.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt1.getY()) * scaleH), | |
(float) (Units.toPoints((Long) pt2.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt2.getY()) * scaleH)); | |
} else if (ch instanceof CTPath2DCubicBezierTo){ | |
CTPath2DCubicBezierTo bez = ((CTPath2DCubicBezierTo)ch); | |
CTAdjPoint2D pt1 = bez.getPtArray(0); | |
CTAdjPoint2D pt2 = bez.getPtArray(1); | |
CTAdjPoint2D pt3 = bez.getPtArray(2); | |
path.curveTo( | |
(float) (Units.toPoints((Long) pt1.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt1.getY()) * scaleH), | |
(float) (Units.toPoints((Long) pt2.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt2.getY()) * scaleH), | |
(float) (Units.toPoints((Long) pt3.getX()) * scaleW), | |
(float) (Units.toPoints((Long) pt3.getY()) * scaleH)); | |
} else if (ch instanceof CTPath2DClose){ | |
path.closePath(); | |
} | |
} | |
} | |
// the created path starts at (x=0, y=0). | |
// The returned path should fit in the bounding rectangle | |
AffineTransform at = new AffineTransform(); | |
at.translate(bounds.getX(), bounds.getY()); | |
return new Path2D.Double(at.createTransformedShape(path)); | |
} | |
/** | |
* @param shapeId 1-based shapeId | |
*/ | |
static CTShape prototype(int shapeId) { | |
CTShape ct = CTShape.Factory.newInstance(); | |
CTShapeNonVisual nvSpPr = ct.addNewNvSpPr(); | |
CTNonVisualDrawingProps cnv = nvSpPr.addNewCNvPr(); | |
cnv.setName("Freeform " + shapeId); | |
cnv.setId(shapeId + 1); | |
nvSpPr.addNewCNvSpPr(); | |
nvSpPr.addNewNvPr(); | |
CTShapeProperties spPr = ct.addNewSpPr(); | |
CTCustomGeometry2D geom = spPr.addNewCustGeom(); | |
geom.addNewAvLst(); | |
geom.addNewGdLst(); | |
geom.addNewAhLst(); | |
geom.addNewCxnLst(); | |
CTGeomRect rect = geom.addNewRect(); | |
rect.setR("r"); | |
rect.setB("b"); | |
rect.setT("t"); | |
rect.setL("l"); | |
geom.addNewPathLst(); | |
return ct; | |
} | |
} |