diff --git a/build.xml b/build.xml
index 62c208bf1d..84c2bccbfd 100644
--- a/build.xml
+++ b/build.xml
@@ -108,6 +108,7 @@ under the License.
+
@@ -553,6 +554,9 @@ under the License.
+
+
+
diff --git a/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step1.java b/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step1.java
new file mode 100644
index 0000000000..a83a17e4b8
--- /dev/null
+++ b/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step1.java
@@ -0,0 +1,68 @@
+/*
+ * ====================================================================
+ * 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.tutorial;
+
+import org.apache.poi.xslf.usermodel.XMLSlideShow;
+import org.apache.poi.xslf.usermodel.XSLFShape;
+import org.apache.poi.xslf.usermodel.XSLFSlide;
+import org.apache.poi.xslf.usermodel.XSLFTextParagraph;
+import org.apache.poi.xslf.usermodel.XSLFTextRun;
+import org.apache.poi.xslf.usermodel.XSLFTextShape;
+
+import java.io.FileInputStream;
+
+/**
+ * Reading a .pptx presentation and printing basic shape properties
+ *
+ * @author Yegor Kozlov
+ */
+public class Step1 {
+
+ public static void main(String[] args) throws Exception {
+ if(args.length == 0) {
+ System.out.println("Input file is required");
+ return;
+ }
+
+ XMLSlideShow ppt = new XMLSlideShow(new FileInputStream(args[0]));
+
+ for(XSLFSlide slide : ppt.getSlides()){
+ System.out.println("Title: " + slide.getTitle());
+
+ for(XSLFShape shape : slide.getShapes()){
+ if(shape instanceof XSLFTextShape) {
+ XSLFTextShape tsh = (XSLFTextShape)shape;
+ for(XSLFTextParagraph p : tsh){
+ System.out.println("Paragraph level: " + p.getLevel());
+ for(XSLFTextRun r : p){
+ System.out.println(r.getText());
+ System.out.println(" bold: " + r.isBold());
+ System.out.println(" italic: " + r.isItalic());
+ System.out.println(" underline: " + r.isUnderline());
+ System.out.println(" font.family: " + r.getFontFamily());
+ System.out.println(" font.size: " + r.getFontSize());
+ System.out.println(" font.color: " + r.getFontColor());
+ }
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step2.java b/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step2.java
new file mode 100644
index 0000000000..16b155d3f2
--- /dev/null
+++ b/src/examples/src/org/apache/poi/xslf/usermodel/tutorial/Step2.java
@@ -0,0 +1,80 @@
+/*
+ * ====================================================================
+ * 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.tutorial;
+
+import org.apache.poi.xslf.usermodel.SlideLayout;
+import org.apache.poi.xslf.usermodel.XMLSlideShow;
+import org.apache.poi.xslf.usermodel.XSLFSlide;
+import org.apache.poi.xslf.usermodel.XSLFSlideLayout;
+import org.apache.poi.xslf.usermodel.XSLFSlideMaster;
+import org.apache.poi.xslf.usermodel.XSLFTextShape;
+
+import java.io.FileOutputStream;
+
+/**
+ * Create slides from pre-defined slide layouts
+ *
+ * @author Yegor Kozlov
+ */
+public class Step2 {
+ public static void main(String[] args) throws Exception{
+ XMLSlideShow ppt = new XMLSlideShow();
+
+
+ // first see what slide layouts are available by default
+ System.out.println("Available slide layouts:");
+ for(XSLFSlideMaster master : ppt.getSlideMasters()){
+ for(XSLFSlideLayout layout : master.getSlideLayouts()){
+ System.out.println(layout.getType());
+ }
+ }
+
+ // blank slide
+ XSLFSlide blankSlide = ppt.createSlide();
+
+ XSLFSlideMaster defaultMaster = ppt.getSlideMasters()[0];
+
+ // title slide
+ XSLFSlideLayout titleLayout = defaultMaster.getLayout(SlideLayout.TITLE);
+ XSLFSlide slide1 = ppt.createSlide(titleLayout);
+ XSLFTextShape title1 = slide1.getPlaceholder(0);
+ title1.setText("First Title");
+
+ // title and content
+ XSLFSlideLayout titleBodyLayout = defaultMaster.getLayout(SlideLayout.TITLE_AND_CONTENT);
+ XSLFSlide slide2 = ppt.createSlide(titleBodyLayout);
+
+ XSLFTextShape title2 = slide2.getPlaceholder(0);
+ title2.setText("Second Title");
+
+ XSLFTextShape body2 = slide2.getPlaceholder(1);
+ body2.clearText(); // unset any existing text
+ body2.addNewTextParagraph().addNewTextRun().setText("First paragraph");
+ body2.addNewTextParagraph().addNewTextRun().setText("Second paragraph");
+ body2.addNewTextParagraph().addNewTextRun().setText("Third paragraph");
+
+
+
+ FileOutputStream out = new FileOutputStream("step2.pptx");
+ ppt.write(out);
+ out.close();
+
+ }
+}
diff --git a/src/ooxml/java/org/apache/poi/xslf/extractor/XSLFPowerPointExtractor.java b/src/ooxml/java/org/apache/poi/xslf/extractor/XSLFPowerPointExtractor.java
index 812c7ad0e6..0f9a26fe21 100644
--- a/src/ooxml/java/org/apache/poi/xslf/extractor/XSLFPowerPointExtractor.java
+++ b/src/ooxml/java/org/apache/poi/xslf/extractor/XSLFPowerPointExtractor.java
@@ -128,7 +128,7 @@ public class XSLFPowerPointExtractor extends POIXMLTextExtractor {
XSLFNotes notes = slide.getNotes();
XSLFComments comments = slide.getComments();
XSLFSlideLayout layout = slide.getSlideLayout();
- XSLFSlideMaster master = slide.getMasterSheet();
+ XSLFSlideMaster master = layout.getSlideMaster();
// TODO Do the slide's name
// (Stored in docProps/app.xml)
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/ArcToCommand.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/ArcToCommand.java
index 9b2e1bcf31..b1ea0defc8 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/ArcToCommand.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/ArcToCommand.java
@@ -57,15 +57,11 @@ public class ArcToCommand implements PathCommand {
double x0 = pt.getX() - rx - rx * Math.cos(Math.toRadians(start));
double y0 = pt.getY() - ry - ry * Math.sin(Math.toRadians(start));
- if(start == 180 && extent == 180) {
- x0 -= rx*2; //YK: TODO revisit the code and get rid of this hack
- }
-
Arc2D arc = new Arc2D.Double(
x0,
y0,
2 * rx, 2 * ry,
- -start, -extent, // negate angles because DrawingML rotates counter-clockwise
+ -start, -extent,
Arc2D.OPEN);
path.append(arc, true);
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/Context.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/Context.java
index 230524cfde..ea86c51002 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/Context.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/Context.java
@@ -29,17 +29,19 @@ import java.util.Map;
* @author Yegor Kozlov
*/
public class Context {
- Map _ctx = new HashMap();
- IAdjustableShape _props;
-
- public Context(CustomGeometry geom, IAdjustableShape props){
+ final Map _ctx = new HashMap();
+ final IAdjustableShape _props;
+ final Rectangle2D _anchor;
+
+ public Context(CustomGeometry geom, Rectangle2D anchor, IAdjustableShape props){
_props = props;
+ _anchor = anchor;
for(Guide gd : geom.adjusts) evaluate(gd);
for(Guide gd : geom.guides) evaluate(gd);
}
public Rectangle2D getShapeAnchor(){
- return _props.getAnchor();
+ return _anchor;
}
public Guide getAdjustValue(String name){
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/CustomGeometry.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/CustomGeometry.java
index 7507f55b21..6745ccd513 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/CustomGeometry.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/CustomGeometry.java
@@ -24,6 +24,7 @@ import org.openxmlformats.schemas.drawingml.x2006.main.*;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
+import java.awt.geom.Rectangle2D;
/**
* Definition of a custom geometric shape
@@ -34,8 +35,9 @@ public class CustomGeometry implements Iterable{
List adjusts = new ArrayList();
List guides = new ArrayList();
List paths = new ArrayList();
+ Path textBounds;
- public CustomGeometry(CTCustomGeometry2D geom){
+ public CustomGeometry(CTCustomGeometry2D geom) {
CTGeomGuideList avLst = geom.getAvLst();
if(avLst != null) for(CTGeomGuide gd : avLst.getGdList()){
adjusts.add(new AdjustValue(gd));
@@ -50,6 +52,21 @@ public class CustomGeometry implements Iterable{
if(pathLst != null) for(CTPath2D spPath : pathLst.getPathList()){
paths.add(new Path(spPath));
}
+
+ if(geom.isSetRect()) {
+ CTGeomRect rect = geom.getRect();
+ textBounds = new Path();
+ textBounds.addCommand(
+ new MoveToCommand(rect.getL().toString(), rect.getT().toString()));
+ textBounds.addCommand(
+ new LineToCommand(rect.getR().toString(), rect.getT().toString()));
+ textBounds.addCommand(
+ new LineToCommand(rect.getR().toString(), rect.getB().toString()));
+ textBounds.addCommand(
+ new LineToCommand(rect.getL().toString(), rect.getB().toString()));
+ textBounds.addCommand(
+ new ClosePathCommand());
+ }
}
@@ -58,4 +75,7 @@ public class CustomGeometry implements Iterable{
return paths.iterator();
}
+ public Path getTextBounds(){
+ return textBounds;
+ }
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/IAdjustableShape.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/IAdjustableShape.java
index 174af3d34c..44f5a562f7 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/IAdjustableShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/IAdjustableShape.java
@@ -19,7 +19,6 @@
package org.apache.poi.xslf.model.geom;
-import java.awt.geom.Rectangle2D;
/**
* A bridge to the consumer application.
@@ -29,12 +28,6 @@ import java.awt.geom.Rectangle2D;
* @author Yegor Kozlov
*/
public interface IAdjustableShape {
- /**
- *
- * @return bounds of the shape
- */
- Rectangle2D getAnchor();
-
/**
*
* @param name name of a adjust value, e.g. adj1
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/LineToCommand.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/LineToCommand.java
index 3df2e499e1..5142dd234a 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/LineToCommand.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/LineToCommand.java
@@ -36,6 +36,11 @@ public class LineToCommand implements PathCommand {
arg2 = pt.getY().toString();
}
+ LineToCommand(String s1, String s2){
+ arg1 = s1;
+ arg2 = s2;
+ }
+
public void execute(GeneralPath path, Context ctx){
double x = ctx.getValue(arg1);
double y = ctx.getValue(arg2);
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/MoveToCommand.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/MoveToCommand.java
index a5a4b5ce69..9d9575ace1 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/MoveToCommand.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/MoveToCommand.java
@@ -36,6 +36,11 @@ public class MoveToCommand implements PathCommand {
arg2 = pt.getY().toString();
}
+ MoveToCommand(String s1, String s2){
+ arg1 = s1;
+ arg2 = s2;
+ }
+
public void execute(GeneralPath path, Context ctx){
double x = ctx.getValue(arg1);
double y = ctx.getValue(arg2);
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/Outline.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/Outline.java
new file mode 100644
index 0000000000..dbf9f1f9c9
--- /dev/null
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/Outline.java
@@ -0,0 +1,45 @@
+/*
+ * ====================================================================
+ * 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.model.geom;
+
+import java.awt.Shape;
+
+/**
+* Date: 11/6/11
+*
+* @author Yegor Kozlov
+*/
+public class Outline {
+ private Shape shape;
+ private Path path;
+
+ public Outline(Shape shape, Path path){
+ this.shape = shape;
+ this.path = path;
+ }
+
+ public Path getPath(){
+ return path;
+ }
+
+ public Shape getOutline(){
+ return shape;
+ }
+}
diff --git a/src/ooxml/java/org/apache/poi/xslf/model/geom/Path.java b/src/ooxml/java/org/apache/poi/xslf/model/geom/Path.java
index 9d5e9f71d3..94ab37662c 100644
--- a/src/ooxml/java/org/apache/poi/xslf/model/geom/Path.java
+++ b/src/ooxml/java/org/apache/poi/xslf/model/geom/Path.java
@@ -39,11 +39,26 @@ import java.util.List;
public class Path {
private final List commands;
boolean _fill, _stroke;
+ long _w, _h;
+
+ public Path(){
+ this(true, true);
+ }
+
+ public Path(boolean fill, boolean stroke){
+ commands = new ArrayList();
+ _w = -1;
+ _h = -1;
+ _fill = fill;
+ _stroke = stroke;
+ }
public Path(CTPath2D spPath){
_fill = spPath.getFill() != STPathFillMode.NONE;
_stroke = spPath.getStroke();
-
+ _w = spPath.isSetW() ? spPath.getW() : -1;
+ _h = spPath.isSetH() ? spPath.getH() : -1;
+
commands = new ArrayList();
for(XmlObject ch : spPath.selectPath("*")){
if(ch instanceof CTPath2DMoveTo){
@@ -74,6 +89,13 @@ public class Path {
}
}
+ public void addCommand(PathCommand cmd){
+ commands.add(cmd);
+ }
+
+ /**
+ * Convert the internal represenation to java.awt.GeneralPath
+ */
public GeneralPath getPath(Context ctx) {
GeneralPath path = new GeneralPath();
for(PathCommand cmd : commands)
@@ -88,4 +110,12 @@ public class Path {
public boolean isFilled(){
return _fill;
}
+
+ public long getW(){
+ return _w;
+ }
+
+ public long getH(){
+ return _h;
+ }
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/Placeholder.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/Placeholder.java
index a96e88778d..087b28c054 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/Placeholder.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/Placeholder.java
@@ -30,7 +30,7 @@ public enum Placeholder {
SLIDE_NUMBER,
FOOTER,
HEADER,
- OBJECT,
+ CONTENT,
CHART,
TABLE,
CLIP_ART,
@@ -38,5 +38,4 @@ public enum Placeholder {
MEDIA,
SLIDE_IMAGE,
PICTURE
-
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/RenderableShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/RenderableShape.java
index 393ba333e6..bbd0de9ea6 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/RenderableShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/RenderableShape.java
@@ -1,11 +1,563 @@
+/*
+ * ====================================================================
+ * 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 org.apache.poi.openxml4j.opc.PackagePart;
+import org.apache.poi.openxml4j.opc.PackageRelationship;
+import org.apache.poi.util.Internal;
+import org.apache.poi.util.Units;
+import org.apache.poi.xslf.model.PropertyFetcher;
+import org.apache.poi.xslf.model.geom.Context;
+import org.apache.poi.xslf.model.geom.CustomGeometry;
+import org.apache.poi.xslf.model.geom.Guide;
+import org.apache.poi.xslf.model.geom.IAdjustableShape;
+import org.apache.poi.xslf.model.geom.Outline;
+import org.apache.poi.xslf.model.geom.Path;
+import org.apache.xmlbeans.XmlObject;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTBlip;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTBlipFillProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTGeomGuide;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTGradientFillProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTGradientStop;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTLineProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTNoFillProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPathShadeProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPresetGeometry2D;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSchemeColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeStyle;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSolidColorFillProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTStyleMatrixReference;
+import org.openxmlformats.schemas.drawingml.x2006.main.STPathShadeType;
+
+import java.awt.AlphaComposite;
+import java.awt.BasicStroke;
+import java.awt.Color;
+import java.awt.GradientPaint;
+import java.awt.Graphics2D;
+import java.awt.Paint;
+import java.awt.Shape;
+import java.awt.Stroke;
+import java.awt.TexturePaint;
+import java.awt.geom.AffineTransform;
+import java.awt.geom.Point2D;
+import java.awt.geom.Rectangle2D;
+import java.awt.image.BufferedImage;
+import java.lang.reflect.Constructor;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Comparator;
+
/**
- * Created by IntelliJ IDEA.
- * User: yegor
- * Date: Oct 27, 2011
- * Time: 4:50:08 PM
- * To change this template use File | Settings | File Templates.
+ * Encapsulates logic to translate DrawingML objects to Java2D
*/
+@Internal
class RenderableShape {
+ public final static Color NO_PAINT = new Color(0xFF, 0xFF, 0xFF, 0);
+
+ private XSLFSimpleShape _shape;
+
+ public RenderableShape(XSLFSimpleShape shape){
+ _shape = shape;
+ }
+
+ /**
+ * Convert shape fill into java.awt.Paint. The result is either Color or
+ * TexturePaint or GradientPaint or null
+ *
+ * @param graphics the target graphics
+ * @param obj the xml to read. Must contain elements from the EG_ColorChoice group:
+ *
+ * a:scrgbClr RGB Color Model - Percentage Variant
+ * a:srgbClr RGB Color Model - Hex Variant
+ * a:hslClr Hue, Saturation, Luminance Color Model
+ * a:sysClr System Color
+ * a:schemeClr Scheme Color
+ * a:prstClr Preset Color
+ *
+ *
+ * @param phClr context color
+ * @param parentPart the parent package part. Any external references (images, etc.) are resolved relative to it.
+ *
+ * @return the applied Paint or null if none was applied
+ */
+ public Paint selectPaint(Graphics2D graphics, XmlObject obj, CTSchemeColor phClr, PackagePart parentPart) {
+ XSLFTheme theme = _shape.getSheet().getTheme();
+ Rectangle2D anchor = _shape.getAnchor();
+
+ Paint paint = null;
+ if (obj instanceof CTNoFillProperties) {
+ paint = NO_PAINT;
+
+ }
+ else if (obj instanceof CTSolidColorFillProperties) {
+ CTSolidColorFillProperties solidFill = (CTSolidColorFillProperties) obj;
+ XSLFColor c = new XSLFColor(solidFill, theme, phClr);
+ paint = c.getColor();
+ }
+ else if (obj instanceof CTBlipFillProperties) {
+ CTBlipFillProperties blipFill = (CTBlipFillProperties)obj;
+ paint = createTexturePaint(blipFill, graphics, parentPart);
+ }
+ else if (obj instanceof CTGradientFillProperties) {
+ CTGradientFillProperties gradFill = (CTGradientFillProperties) obj;
+ if (gradFill.isSetLin()) {
+ paint = createLinearGradientPaint(gradFill, anchor, theme, phClr);
+ } else if (gradFill.isSetPath()){
+ CTPathShadeProperties ps = gradFill.getPath();
+ if(ps.getPath() == STPathShadeType.CIRCLE){
+ paint = createRadialGradientPaint(gradFill, anchor, theme, phClr);
+ }
+ }
+ }
+
+ return paint;
+ }
+
+ private Paint createTexturePaint(CTBlipFillProperties blipFill, Graphics2D graphics,
+ PackagePart parentPart){
+ Paint paint = null;
+ CTBlip blip = blipFill.getBlip();
+ String blipId = blip.getEmbed();
+ PackageRelationship rel = parentPart.getRelationship(blipId);
+ if (rel != null) {
+ XSLFImageRendener renderer = null;
+ if (graphics != null)
+ renderer = (XSLFImageRendener) graphics.getRenderingHint(XSLFRenderingHint.IMAGE_RENDERER);
+ if (renderer == null) renderer = new XSLFImageRendener();
+
+ try {
+ BufferedImage img = renderer.readImage(parentPart.getRelatedPart(rel));
+ if (blip.sizeOfAlphaModFixArray() > 0) {
+ float alpha = blip.getAlphaModFixArray(0).getAmt() / 100000.f;
+ AlphaComposite ac = AlphaComposite.getInstance(
+ AlphaComposite.SRC_OVER, alpha);
+ if (graphics != null) graphics.setComposite(ac);
+ }
+
+ if(img != null) {
+ paint = new TexturePaint(
+ img, new Rectangle2D.Double(0, 0, img.getWidth(), img.getHeight()));
+ }
+ }
+ catch (Exception e) {
+ e.printStackTrace();
+ }
+ }
+ return paint;
+ }
+
+ private static Paint createLinearGradientPaint(
+ CTGradientFillProperties gradFill, Rectangle2D anchor,
+ XSLFTheme theme, CTSchemeColor phClr) {
+ double angle = gradFill.getLin().getAng() / 60000;
+ CTGradientStop[] gs = gradFill.getGsLst().getGsArray();
+
+ Arrays.sort(gs, new Comparator() {
+ public int compare(CTGradientStop o1, CTGradientStop o2) {
+ Integer pos1 = o1.getPos();
+ Integer pos2 = o2.getPos();
+ return pos1.compareTo(pos2);
+ }
+ });
+
+ Color[] colors = new Color[gs.length];
+ float[] fractions = new float[gs.length];
+
+ AffineTransform at = AffineTransform.getRotateInstance(
+ Math.toRadians(angle),
+ anchor.getX() + anchor.getWidth() / 2,
+ anchor.getY() + anchor.getHeight() / 2);
+
+ double diagonal = Math.sqrt(anchor.getHeight() * anchor.getHeight() + anchor.getWidth() * anchor.getWidth());
+ Point2D p1 = new Point2D.Double(anchor.getX() + anchor.getWidth() / 2 - diagonal / 2,
+ anchor.getY() + anchor.getHeight() / 2);
+ p1 = at.transform(p1, null);
+
+ Point2D p2 = new Point2D.Double(anchor.getX() + anchor.getWidth(), anchor.getY() + anchor.getHeight() / 2);
+ p2 = at.transform(p2, null);
+
+ snapToAnchor(p1, anchor);
+ snapToAnchor(p2, anchor);
+
+ for (int i = 0; i < gs.length; i++) {
+ CTGradientStop stop = gs[i];
+ colors[i] = new XSLFColor(stop, theme, phClr).getColor();
+ fractions[i] = stop.getPos() / 100000.f;
+ }
+
+ // Trick to return GradientPaint on JDK 1.5 and LinearGradientPaint on JDK 1.6+
+ Paint paint;
+ try {
+ Class clz = Class.forName("java.awt.LinearGradientPaint");
+ Constructor c =
+ clz.getConstructor(Point2D.class, Point2D.class, float[].class, Color[].class);
+ paint = (Paint) c.newInstance(p1, p2, fractions, colors);
+ } catch (ClassNotFoundException e) {
+ paint = new GradientPaint(p1, colors[0], p2, colors[colors.length - 1]);
+ } catch (Exception e) {
+ throw new RuntimeException(e);
+ }
+ return paint;
+ }
+
+ private static Paint createRadialGradientPaint(
+ CTGradientFillProperties gradFill, Rectangle2D anchor,
+ XSLFTheme theme, CTSchemeColor phClr) {
+ CTGradientStop[] gs = gradFill.getGsLst().getGsArray();
+
+ Point2D pCenter = new Point2D.Double(anchor.getX() + anchor.getWidth()/2,
+ anchor.getY() + anchor.getHeight()/2);
+
+ float radius = (float)Math.max(anchor.getWidth(), anchor.getHeight());
+
+ Arrays.sort(gs, new Comparator() {
+ public int compare(CTGradientStop o1, CTGradientStop o2) {
+ Integer pos1 = o1.getPos();
+ Integer pos2 = o2.getPos();
+ return pos1.compareTo(pos2);
+ }
+ });
+
+ Color[] colors = new Color[gs.length];
+ float[] fractions = new float[gs.length];
+
+
+ for (int i = 0; i < gs.length; i++) {
+ CTGradientStop stop = gs[i];
+ colors[i] = new XSLFColor(stop, theme, phClr).getColor();
+ fractions[i] = stop.getPos() / 100000.f;
+ }
+
+ // Trick to return GradientPaint on JDK 1.5 and RadialGradientPaint on JDK 1.6+
+ Paint paint;
+ try {
+ Class clz = Class.forName("java.awt.RadialGradientPaint");
+ Constructor c =
+ clz.getConstructor(Point2D.class, float.class,
+ float[].class, Color[].class);
+ paint = (Paint) c.newInstance(pCenter, radius, fractions, colors);
+ } catch (ClassNotFoundException e) {
+ // the result on JDK 1.5 is incorrect, but it is better than nothing
+ paint = new GradientPaint(
+ new Point2D.Double(anchor.getX(), anchor.getY()),
+ colors[0], pCenter, colors[colors.length - 1]);
+ } catch (Exception e) {
+ throw new RuntimeException(e);
+ }
+ return paint;
+ }
+
+ private static void snapToAnchor(Point2D p, Rectangle2D anchor) {
+ if (p.getX() < anchor.getX()) {
+ p.setLocation(anchor.getX(), p.getY());
+ } else if (p.getX() > (anchor.getX() + anchor.getWidth())) {
+ p.setLocation(anchor.getX() + anchor.getWidth(), p.getY());
+ }
+
+ if (p.getY() < anchor.getY()) {
+ p.setLocation(p.getX(), anchor.getY());
+ } else if (p.getY() > (anchor.getY() + anchor.getHeight())) {
+ p.setLocation(p.getX(), anchor.getY() + anchor.getHeight());
+ }
+ }
+
+
+ @SuppressWarnings("deprecation") // getXYZArray() array accessors are deprecated
+ Paint getPaint(Graphics2D graphics, XmlObject spPr, CTSchemeColor phClr) {
+
+ Paint paint = null;
+ for (XmlObject obj : spPr.selectPath("*")) {
+ paint = selectPaint(graphics, obj, phClr, _shape.getSheet().getPackagePart());
+ if(paint != null) break;
+ }
+ return paint == NO_PAINT ? null : paint;
+ }
+
+
+ /**
+ * fetch shape fill as a java.awt.Paint
+ *
+ * @return either Color or GradientPaint or TexturePaint or null
+ */
+ Paint getFillPaint(final Graphics2D graphics) {
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
+ CTShapeProperties spPr = shape.getSpPr();
+ if (spPr.isSetNoFill()) {
+ setValue(RenderableShape.NO_PAINT); // use it as 'nofill' value
+ return true;
+ }
+ Paint paint = getPaint(graphics, spPr, null);
+ if (paint != null) {
+ setValue(paint);
+ return true;
+ }
+ return false;
+ }
+ };
+ _shape.fetchShapeProperty(fetcher);
+
+ Paint paint = fetcher.getValue();
+ if (paint == null) {
+ // fill color was not found, check if it is defined in the theme
+ CTShapeStyle style = _shape.getSpStyle();
+ if (style != null) {
+ // get a reference to a fill style within the style matrix.
+ CTStyleMatrixReference fillRef = style.getFillRef();
+ // The idx attribute refers to the index of a fill style or
+ // background fill style within the presentation's style matrix, defined by the fmtScheme element.
+ // value of 0 or 1000 indicates no background,
+ // values 1-999 refer to the index of a fill style within the fillStyleLst element
+ // values 1001 and above refer to the index of a background fill style within the bgFillStyleLst element.
+ int idx = (int)fillRef.getIdx();
+ CTSchemeColor phClr = fillRef.getSchemeClr();
+ XSLFSheet sheet = _shape.getSheet();
+ XSLFTheme theme = sheet.getTheme();
+ XmlObject fillProps = null;
+ if(idx >= 1 && idx <= 999){
+ fillProps = theme.getXmlObject().
+ getThemeElements().getFmtScheme().getFillStyleLst().selectPath("*")[idx - 1];
+ } else if (idx >= 1001 ){
+ fillProps = theme.getXmlObject().
+ getThemeElements().getFmtScheme().getBgFillStyleLst().selectPath("*")[idx - 1001];
+ }
+ if(fillProps != null) {
+ paint = selectPaint(graphics, fillProps, phClr, sheet.getPackagePart());
+ }
+ }
+ }
+ return paint == RenderableShape.NO_PAINT ? null : paint;
+ }
+
+ public Paint getLinePaint(final Graphics2D graphics) {
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
+ CTLineProperties spPr = shape.getSpPr().getLn();
+ if (spPr != null) {
+ if (spPr.isSetNoFill()) {
+ setValue(NO_PAINT); // use it as 'nofill' value
+ return true;
+ }
+ Paint paint = getPaint(graphics, spPr, null);
+ if (paint != null) {
+ setValue(paint);
+ return true;
+ }
+ }
+ return false;
+
+ }
+ };
+ _shape.fetchShapeProperty(fetcher);
+
+ Paint paint = fetcher.getValue();
+ if (paint == null) {
+ // line color was not found, check if it is defined in the theme
+ CTShapeStyle style = _shape.getSpStyle();
+ if (style != null) {
+ // get a reference to a line style within the style matrix.
+ CTStyleMatrixReference lnRef = style.getLnRef();
+ int idx = (int)lnRef.getIdx();
+ CTSchemeColor phClr = lnRef.getSchemeClr();
+ if(idx > 0){
+ XSLFTheme theme = _shape.getSheet().getTheme();
+ XmlObject lnProps = theme.getXmlObject().
+ getThemeElements().getFmtScheme().getLnStyleLst().selectPath("*")[idx - 1];
+ paint = getPaint(graphics, lnProps, phClr);
+ }
+ }
+ }
+
+ return paint == NO_PAINT ? null : paint;
+ }
+
+ /**
+ * convert PPT dash into java.awt.BasicStroke
+ *
+ * The mapping is derived empirically on PowerPoint 2010
+ */
+ private static float[] getDashPattern(LineDash lineDash, float lineWidth) {
+ float[] dash = null;
+ switch (lineDash) {
+ case SYS_DOT:
+ dash = new float[]{lineWidth, lineWidth};
+ break;
+ case SYS_DASH:
+ dash = new float[]{2 * lineWidth, 2 * lineWidth};
+ break;
+ case DASH:
+ dash = new float[]{3 * lineWidth, 4 * lineWidth};
+ break;
+ case DASH_DOT:
+ dash = new float[]{4 * lineWidth, 3 * lineWidth, lineWidth,
+ 3 * lineWidth};
+ break;
+ case LG_DASH:
+ dash = new float[]{8 * lineWidth, 3 * lineWidth};
+ break;
+ case LG_DASH_DOT:
+ dash = new float[]{8 * lineWidth, 3 * lineWidth, lineWidth,
+ 3 * lineWidth};
+ break;
+ case LG_DASH_DOT_DOT:
+ dash = new float[]{8 * lineWidth, 3 * lineWidth, lineWidth,
+ 3 * lineWidth, lineWidth, 3 * lineWidth};
+ break;
+ }
+ return dash;
+ }
+
+
+ public Stroke applyStroke(Graphics2D graphics) {
+
+ float lineWidth = (float) _shape.getLineWidth();
+ LineDash lineDash = _shape.getLineDash();
+ float[] dash = null;
+ float dash_phase = 0;
+ if (lineDash != null) {
+ dash = getDashPattern(lineDash, lineWidth);
+ }
+
+ int cap = BasicStroke.CAP_BUTT;
+ LineCap lineCap = _shape.getLineCap();
+ if (lineCap != null) {
+ switch (lineCap) {
+ case ROUND:
+ cap = BasicStroke.CAP_ROUND;
+ break;
+ case SQUARE:
+ cap = BasicStroke.CAP_SQUARE;
+ break;
+ default:
+ cap = BasicStroke.CAP_BUTT;
+ break;
+ }
+ }
+
+ int meter = BasicStroke.JOIN_ROUND;
+
+ Stroke stroke = new BasicStroke(lineWidth, cap, meter, Math.max(1, lineWidth), dash,
+ dash_phase);
+ graphics.setStroke(stroke);
+ return stroke;
+ }
+
+ public void render(Graphics2D graphics){
+ Collection elems = computeOutlines();
+
+ // shadow
+ XSLFShadow shadow = _shape.getShadow();
+
+ // first fill
+ Paint fill = getFillPaint(graphics);
+ if(fill != null) for(Outline o : elems){
+ if(o.getPath().isFilled()){
+ if(shadow != null) shadow.fill(graphics, o.getOutline());
+
+ graphics.setPaint(fill);
+ graphics.fill(o.getOutline());
+ }
+ }
+
+ // then draw any content within this shape (text, image, etc.)
+ _shape.drawContent(graphics);
+
+ // then stroke the shape outline
+ Paint line = getLinePaint(graphics);
+ if(line != null) for(Outline o : elems){
+ if(o.getPath().isStroked()){
+ applyStroke(graphics); // the stroke applies both to the shadow and the shape
+
+ if(shadow != null) shadow.draw(graphics, o.getOutline());
+
+ graphics.setPaint(line);
+ graphics.draw(o.getOutline());
+ }
+ }
+ }
+
+ private Collection computeOutlines() {
+ CustomGeometry geom = _shape.getGeometry();
+
+ Collection lst = new ArrayList();
+
+ Rectangle2D anchor = _shape.getAnchor();
+ for (Path p : geom) {
+
+ double w = p.getW() == -1 ? anchor.getWidth() * Units.EMU_PER_POINT : p.getW();
+ double h = p.getH() == -1 ? anchor.getHeight() * Units.EMU_PER_POINT : p.getH();
+
+ // the guides in the shape definitions are all defined relative to each other,
+ // so we build the path starting from (0,0).
+ final Rectangle2D pathAnchor = new Rectangle2D.Double(
+ 0,
+ 0,
+ w,
+ h
+ );
+
+ Context ctx = new Context(geom, pathAnchor, new IAdjustableShape() {
+
+ public Guide getAdjustValue(String name) {
+ CTPresetGeometry2D prst = _shape.getSpPr().getPrstGeom();
+ if (prst.isSetAvLst()) {
+ for (CTGeomGuide g : prst.getAvLst().getGdList()) {
+ if (g.getName().equals(name)) {
+ return new Guide(g);
+ }
+ }
+ }
+ return null;
+ }
+ }) ;
+
+ Shape gp = p.getPath(ctx);
+
+ // translate the result to the canvas coordinates in points
+ AffineTransform at = new AffineTransform();
+ at.translate(anchor.getX(), anchor.getY());
+
+ double scaleX, scaleY;
+ if (p.getW() != -1) {
+ scaleX = anchor.getWidth() / p.getW();
+ } else {
+ scaleX = 1.0 / Units.EMU_PER_POINT;
+ }
+ if (p.getH() != -1) {
+ scaleY = anchor.getHeight() / p.getH();
+ } else {
+ scaleY = 1.0 / Units.EMU_PER_POINT;
+ }
+
+ at.scale(scaleX, scaleY);
+
+ Shape canvasShape = at.createTransformedShape(gp);
+
+ lst.add(new Outline(canvasShape, p));
+ }
+
+ // add any shape-specific stuff here (line decorations, etc.)
+ lst.addAll(_shape.getCustomOutlines());
+ return lst;
+ }
+
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/SlideLayout.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/SlideLayout.java
new file mode 100644
index 0000000000..729057b13b
--- /dev/null
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/SlideLayout.java
@@ -0,0 +1,99 @@
+/*
+ * ====================================================================
+ * 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;
+
+/**
+ * Date: 11/5/11
+ *
+ * @author Yegor Kozlov
+ */
+public enum SlideLayout {
+ /**
+ * Title layout with centered title and subtitle placeholders
+ */
+ TITLE,
+ /**
+ * Title and text
+ */
+ TEXT,
+
+ TWO_COL_TX,
+ TBL,
+ TEXT_AND_CHART,
+
+ /**
+ * Title, chart on left and text on right
+ */
+ CHART_AND_TEXT,
+
+ DGM,
+
+ /**
+ * Title and chart
+ */
+ CHART,
+
+ TX_AND_CLIP_ART,
+ /**
+ * Title, clipart on left, text on right
+ */
+ CLIP_ART_AND_TEXT,
+
+ /**
+ * Title only
+ */
+ TITLE_ONLY,
+
+ /**
+ * Blank
+ */
+ BLANK,
+
+ TX_AND_OBJ,
+ OBJ_AND_TX,
+ OBJ_ONLY,
+ /**
+ * title and content
+ */
+ TITLE_AND_CONTENT,
+ TX_AND_MEDIA,
+ MEDIA_AND_TX,
+ OBJ_OVER_TX,
+ TX_OVER_OBJ,
+ TX_AND_TWO_OBJ,
+ TWO_OBJ_AND_TX,
+ TWO_OBJ_OVER_TX,
+ FOUR_OBJ,
+ VERT_TX,
+ CLIP_ART_AND_VERT_TX,
+ VERT_TITLE_AND_TX,
+ VERT_TITLE_AND_TX_OVER_CHART,
+ TWO_OBJ,
+ OBJ_AND_TWO_OBJ,
+ TWO_OBJ_AND_OBJ,
+ CUST,
+ /**
+ * Section Header
+ */
+ SECTION_HEADER,
+ TWO_TX_TWO_OBJ,
+ OBJ_TX,
+ PIC_TX,
+}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/TextAlign.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/TextAlign.java
index c712dbacef..2570e6cfc9 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/TextAlign.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/TextAlign.java
@@ -43,5 +43,8 @@ public enum TextAlign {
* is smart in the sense that it will not justify sentences
* which are short
*/
- JUSTIFY
+ JUSTIFY,
+ JUSTIFY_LOW,
+ DIST,
+ THAI_DIST
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/TextCap.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/TextCap.java
new file mode 100644
index 0000000000..2e998efcf0
--- /dev/null
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/TextCap.java
@@ -0,0 +1,33 @@
+/*
+ * ====================================================================
+ * 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;
+
+/**
+ * Created by IntelliJ IDEA.
+ * User: yegor
+ * Date: 11/3/11
+ * Time: 5:07 PM
+ * To change this template use File | Settings | File Templates.
+ */
+public enum TextCap {
+ NONE,
+ SMALL,
+ ALL
+}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XMLSlideShow.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XMLSlideShow.java
index 8f25beeede..efec667a93 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XMLSlideShow.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XMLSlideShow.java
@@ -40,7 +40,7 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdListEntry;
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideSize;
import org.openxmlformats.schemas.presentationml.x2006.main.PresentationDocument;
-import java.awt.*;
+import java.awt.Dimension;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
@@ -190,7 +190,13 @@ public class XMLSlideShow extends POIXMLDocument {
return Collections.unmodifiableList(_pictures);
}
- public XSLFSlide createSlide() {
+ /**
+ * Create a slide and initialize it from the specified layout.
+ *
+ * @param layout
+ * @return created slide
+ */
+ public XSLFSlide createSlide(XSLFSlideLayout layout) {
int slideNumber = 256, cnt = 1;
CTSlideIdList slideList;
if (!_presentation.isSetSldIdLst()) slideList = _presentation.addNewSldIdLst();
@@ -209,12 +215,7 @@ public class XMLSlideShow extends POIXMLDocument {
slideId.setId(slideNumber);
slideId.setId2(slide.getPackageRelationship().getId());
- String masterId = _presentation.getSldMasterIdLst().getSldMasterIdArray(0).getId2();
- XSLFSlideMaster master = _masters.get(masterId);
-
- XSLFSlideLayout layout = master.getLayout("blank");
- if(layout == null) throw new IllegalArgumentException("Blank layout was not found");
-
+ layout.copyLayout(slide);
slide.addRelation(layout.getPackageRelationship().getId(), layout);
PackagePartName ppName = layout.getPackagePart().getPartName();
@@ -224,7 +225,20 @@ public class XMLSlideShow extends POIXMLDocument {
_slides.add(slide);
return slide;
}
-
+
+ /**
+ * Create a blank slide.
+ */
+ public XSLFSlide createSlide() {
+ String masterId = _presentation.getSldMasterIdLst().getSldMasterIdArray(0).getId2();
+ XSLFSlideMaster master = _masters.get(masterId);
+
+ XSLFSlideLayout layout = master.getLayout(SlideLayout.BLANK);
+ if(layout == null) throw new IllegalArgumentException("Blank layout was not found");
+
+ return createSlide(layout);
+ }
+
/**
* Return the Notes Master, if there is one.
* (May not be present if no notes exist)
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFAutoShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFAutoShape.java
index 2b8a1e08b8..5f4e838a0c 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFAutoShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFAutoShape.java
@@ -28,16 +28,14 @@ import org.openxmlformats.schemas.drawingml.x2006.main.STShapeType;
import org.openxmlformats.schemas.presentationml.x2006.main.CTShape;
import org.openxmlformats.schemas.presentationml.x2006.main.CTShapeNonVisual;
-import java.util.regex.Pattern;
/**
- * Represents a preset geometric shape.
+ * Represents a shape with a preset geometry.
*
* @author Yegor Kozlov
*/
@Beta
public class XSLFAutoShape extends XSLFTextShape {
- private static final Pattern adjPtrn = Pattern.compile("val\\s+(\\d+)");
/*package*/ XSLFAutoShape(CTShape shape, XSLFSheet sheet) {
super(shape, sheet);
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFBackground.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFBackground.java
index cd82eabead..7d8cb061ef 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFBackground.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFBackground.java
@@ -17,16 +17,16 @@
package org.apache.poi.xslf.usermodel;
-import org.openxmlformats.schemas.presentationml.x2006.main.CTBackground;
-import org.openxmlformats.schemas.presentationml.x2006.main.CTBackgroundProperties;
-import org.openxmlformats.schemas.drawingml.x2006.main.CTStyleMatrixReference;
-import org.openxmlformats.schemas.drawingml.x2006.main.CTStyleMatrix;
-import org.openxmlformats.schemas.drawingml.x2006.main.CTBackgroundFillStyleList;
import org.apache.xmlbeans.XmlObject;
-import org.apache.xmlbeans.XmlCursor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTBackgroundFillStyleList;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSchemeColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTStyleMatrixReference;
+import org.openxmlformats.schemas.presentationml.x2006.main.CTBackground;
-import javax.xml.namespace.QName;
-import java.awt.*;
+import java.awt.Color;
+import java.awt.Dimension;
+import java.awt.Graphics2D;
+import java.awt.Paint;
import java.awt.geom.Rectangle2D;
/**
@@ -49,31 +49,46 @@ public class XSLFBackground extends XSLFSimpleShape {
public void draw(Graphics2D graphics) {
Rectangle2D anchor = getAnchor();
- XmlObject spPr = null;
- CTBackground bg = (CTBackground)getXmlObject();
- if(bg.isSetBgPr()){
- spPr = bg.getBgPr();
- } else if (bg.isSetBgRef()){
- CTStyleMatrixReference bgRef= bg.getBgRef();
- int idx = (int)bgRef.getIdx() - 1000;
- XSLFTheme theme = getSheet().getTheme();
- CTBackgroundFillStyleList bgStyles =
- theme.getXmlObject().getThemeElements().getFmtScheme().getBgFillStyleLst();
-
- // TODO pass this to getPaint
- XmlObject bgStyle = bgStyles.selectPath("*")[idx];
- }
-
- if(spPr == null){
- return;
- }
-
- Paint fill = getPaint(graphics, spPr);
+ Paint fill = getPaint(graphics);
if(fill != null) {
graphics.setPaint(fill);
graphics.fill(anchor);
}
}
-
+ /**
+ * @return the Paint object to fill
+ */
+ Paint getPaint(Graphics2D graphics){
+ RenderableShape rShape = new RenderableShape(this);
+
+ Paint fill = null;
+ CTBackground bg = (CTBackground)getXmlObject();
+ if(bg.isSetBgPr()){
+ XmlObject spPr = bg.getBgPr();
+ fill = rShape.getPaint(graphics, spPr, null);
+ } else if (bg.isSetBgRef()){
+ CTStyleMatrixReference bgRef= bg.getBgRef();
+ CTSchemeColor phClr = bgRef.getSchemeClr();
+
+ int idx = (int)bgRef.getIdx() - 1001;
+ XSLFTheme theme = getSheet().getTheme();
+ CTBackgroundFillStyleList bgStyles =
+ theme.getXmlObject().getThemeElements().getFmtScheme().getBgFillStyleLst();
+
+ XmlObject bgStyle = bgStyles.selectPath("*")[idx];
+ fill = rShape.selectPaint(graphics, bgStyle, phClr, theme.getPackagePart());
+ }
+
+ return fill;
+ }
+
+ @Override
+ public Color getFillColor(){
+ Paint p = getPaint(null);
+ if(p instanceof Color){
+ return (Color)p;
+ }
+ return null;
+ }
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFColor.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFColor.java
index 8ca8dbde9f..c0f4762c7e 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFColor.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFColor.java
@@ -30,7 +30,7 @@ import org.openxmlformats.schemas.drawingml.x2006.main.CTSchemeColor;
import org.openxmlformats.schemas.drawingml.x2006.main.CTSystemColor;
import org.w3c.dom.Node;
-import java.awt.*;
+import java.awt.Color;
import java.util.HashMap;
import java.util.Map;
@@ -40,12 +40,15 @@ import java.util.Map;
* @author Yegor Kozlov
*/
@Beta
+@Internal
public class XSLFColor {
private XmlObject _xmlObject;
private Color _color;
+ private CTSchemeColor _phClr;
- XSLFColor(XmlObject obj, XSLFTheme theme) {
+ public XSLFColor(XmlObject obj, XSLFTheme theme, CTSchemeColor phClr) {
_xmlObject = obj;
+ _phClr = phClr;
_color = toColor(obj, theme);
}
@@ -94,7 +97,7 @@ public class XSLFColor {
return result;
}
- static Color toColor(XmlObject obj, XSLFTheme theme) {
+ Color toColor(XmlObject obj, XSLFTheme theme) {
Color color = null;
for (XmlObject ch : obj.selectPath("*")) {
if (ch instanceof CTHslColor) {
@@ -102,7 +105,8 @@ public class XSLFColor {
int h = hsl.getHue2();
int s = hsl.getSat2();
int l = hsl.getLum2();
- // is it correct ?
+ // This conversion is not correct and differs from PowerPoint.
+ // TODO: Revisit and improve.
color = Color.getHSBColor(h / 60000f, s / 100000f, l / 100000f);
} else if (ch instanceof CTPresetColor) {
CTPresetColor prst = (CTPresetColor)ch;
@@ -111,12 +115,13 @@ public class XSLFColor {
} else if (ch instanceof CTSchemeColor) {
CTSchemeColor schemeColor = (CTSchemeColor)ch;
String colorRef = schemeColor.getVal().toString();
+ if(_phClr != null) {
+ // context color overrides the theme
+ colorRef = _phClr.getVal().toString();
+ }
// find referenced CTColor in the theme and convert it to java.awt.Color via a recursive call
CTColor ctColor = theme.getCTColor(colorRef);
if(ctColor != null) color = toColor(ctColor, null);
- else {
- color = Color.black;
- }
} else if (ch instanceof CTScRgbColor) {
// same as CTSRgbColor but with values expressed in percents
CTScRgbColor scrgb = (CTScRgbColor)ch;
@@ -145,21 +150,59 @@ public class XSLFColor {
return color;
}
+ /**
+ * Read a perecentage value from the supplied xml bean.
+ * Example:
+ *
+ *
+ * the returned value is 45
+ *
+ * @return the percentage value in the range [0 .. 100]
+ */
private int getPercentageValue(String elem){
- XmlObject[] obj = _xmlObject.selectPath(
- "declare namespace a='http://schemas.openxmlformats.org/drawingml/2006/main' $this//a:" + elem);
+ String query = "declare namespace a='http://schemas.openxmlformats.org/drawingml/2006/main' $this//a:" + elem;
+
+ XmlObject[] obj;
+
+ // first ask the context color and if not found, ask the actual color bean
+ if(_phClr != null){
+ obj = _phClr.selectPath(query);
+ if(obj.length == 1){
+ Node attr = obj[0].getDomNode().getAttributes().getNamedItem("val");
+ if(attr != null) {
+ return Integer.parseInt(attr.getNodeValue()) / 1000;
+ }
+ }
+ }
+
+ obj = _xmlObject.selectPath(query);
if(obj.length == 1){
Node attr = obj[0].getDomNode().getAttributes().getNamedItem("val");
if(attr != null) {
return Integer.parseInt(attr.getNodeValue()) / 1000;
}
}
+
+
return -1;
}
private int getAngleValue(String elem){
- XmlObject[] obj = _xmlObject.selectPath(
- "declare namespace a='http://schemas.openxmlformats.org/drawingml/2006/main' $this//a:" + elem);
+ String color = "declare namespace a='http://schemas.openxmlformats.org/drawingml/2006/main' $this//a:" + elem;
+ XmlObject[] obj;
+
+ // first ask the context color and if not found, ask the actual color bean
+ if(_phClr != null){
+ obj = _xmlObject.selectPath( color );
+ if(obj.length == 1){
+ Node attr = obj[0].getDomNode().getAttributes().getNamedItem("val");
+ if(attr != null) {
+ return Integer.parseInt(attr.getNodeValue()) / 60000;
+ }
+ }
+ }
+
+ obj = _xmlObject.selectPath( color );
if(obj.length == 1){
Node attr = obj[0].getDomNode().getAttributes().getNamedItem("val");
if(attr != null) {
@@ -341,7 +384,7 @@ public class XSLFColor {
* A 10% shade is 10% of the input color combined with 90% black.
*
* @return the value of the shade specified as a
- * percentage with 0% indicating minimal blue and 100% indicating maximum
+ * percentage with 0% indicating minimal shade and 100% indicating maximum
* or -1 if the value is not set
*/
int getShade(){
@@ -353,7 +396,7 @@ public class XSLFColor {
* A 10% tint is 10% of the input color combined with 90% white.
*
* @return the value of the tint specified as a
- * percentage with 0% indicating minimal blue and 100% indicating maximum
+ * percentage with 0% indicating minimal tint and 100% indicating maximum
* or -1 if the value is not set
*/
int getTint(){
@@ -389,6 +432,10 @@ public class XSLFColor {
return color;
}
+ /**
+ * This algorithm returns result different from PowerPoint.
+ * TODO: revisit and improve
+ */
private static Color shade(Color c, int shade) {
return new Color(
(int)(c.getRed() * shade * 0.01),
@@ -397,6 +444,10 @@ public class XSLFColor {
c.getAlpha());
}
+ /**
+ * This algorithm returns result different from PowerPoint.
+ * TODO: revisit and improve
+ */
private static Color tint(Color c, int tint) {
int r = c.getRed();
int g = c.getGreen();
@@ -414,7 +465,7 @@ public class XSLFColor {
/**
* Preset colors defined in DrawingML
*/
- static Map presetColors;
+ static final Map presetColors;
static {
presetColors = new HashMap();
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFConnectorShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFConnectorShape.java
index 2e28774a4c..647b414c53 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFConnectorShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFConnectorShape.java
@@ -20,6 +20,8 @@
package org.apache.poi.xslf.usermodel;
import org.apache.poi.util.Beta;
+import org.apache.poi.xslf.model.geom.Outline;
+import org.apache.poi.xslf.model.geom.Path;
import org.openxmlformats.schemas.drawingml.x2006.main.CTLineEndProperties;
import org.openxmlformats.schemas.drawingml.x2006.main.CTLineProperties;
import org.openxmlformats.schemas.drawingml.x2006.main.CTNonVisualDrawingProps;
@@ -32,11 +34,14 @@ import org.openxmlformats.schemas.drawingml.x2006.main.STShapeType;
import org.openxmlformats.schemas.presentationml.x2006.main.CTConnector;
import org.openxmlformats.schemas.presentationml.x2006.main.CTConnectorNonVisual;
-import java.awt.*;
+import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.awt.geom.Ellipse2D;
import java.awt.geom.GeneralPath;
import java.awt.geom.Rectangle2D;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
/**
* Specifies a connection shape.
@@ -197,36 +202,7 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
return len == null ? LineEndLength.MEDIUM : LineEndLength.values()[len.intValue() - 1];
}
- @Override
- public void draw(Graphics2D graphics) {
- java.awt.Shape outline = getOutline();
-
- // shadow
- XSLFShadow shadow = getShadow();
-
- //border
- Paint line = getLinePaint(graphics);
- if (line != null) {
- if (shadow != null) shadow.draw(graphics);
-
- graphics.setPaint(line);
- applyStroke(graphics);
- graphics.draw(outline);
-
- Shape tailDecoration = getTailDecoration();
- if (tailDecoration != null) {
- graphics.draw(tailDecoration);
- }
-
- Shape headDecoration = getHeadDecoration();
- if (headDecoration != null) {
- graphics.draw(headDecoration);
-
- }
- }
- }
-
- Shape getTailDecoration() {
+ Outline getTailDecoration() {
LineEndLength tailLength = getLineTailLength();
LineEndWidth tailWidth = getLineTailWidth();
@@ -239,17 +215,20 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
AffineTransform at = new AffineTransform();
Shape shape = null;
+ Path p = null;
Rectangle2D bounds;
double scaleY = Math.pow(2, tailWidth.ordinal());
double scaleX = Math.pow(2, tailLength.ordinal());
- switch (getLineHeadDecoration()) {
+ switch (getLineTailDecoration()) {
case OVAL:
+ p = new Path();
shape = new Ellipse2D.Double(0, 0, lineWidth * scaleX, lineWidth * scaleY);
bounds = shape.getBounds2D();
at.translate(x2 - bounds.getWidth() / 2, y2 - bounds.getHeight() / 2);
at.rotate(alpha, bounds.getX() + bounds.getWidth() / 2, bounds.getY() + bounds.getHeight() / 2);
break;
case ARROW:
+ p = new Path();
GeneralPath arrow = new GeneralPath();
arrow.moveTo((float) (-lineWidth * 3), (float) (-lineWidth * 2));
arrow.lineTo(0, 0);
@@ -259,6 +238,7 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
at.rotate(alpha);
break;
case TRIANGLE:
+ p = new Path();
scaleY = tailWidth.ordinal() + 1;
scaleX = tailLength.ordinal() + 1;
GeneralPath triangle = new GeneralPath();
@@ -277,10 +257,10 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
if (shape != null) {
shape = at.createTransformedShape(shape);
}
- return shape;
+ return shape == null ? null : new Outline(shape, p);
}
- Shape getHeadDecoration() {
+ Outline getHeadDecoration() {
LineEndLength headLength = getLineHeadLength();
LineEndWidth headWidth = getLineHeadWidth();
@@ -293,11 +273,13 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
AffineTransform at = new AffineTransform();
Shape shape = null;
+ Path p = null;
Rectangle2D bounds;
double scaleY = 1;
double scaleX = 1;
switch (getLineHeadDecoration()) {
case OVAL:
+ p = new Path();
shape = new Ellipse2D.Double(0, 0, lineWidth * scaleX, lineWidth * scaleY);
bounds = shape.getBounds2D();
at.translate(x1 - bounds.getWidth() / 2, y1 - bounds.getHeight() / 2);
@@ -305,6 +287,7 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
break;
case STEALTH:
case ARROW:
+ p = new Path();
GeneralPath arrow = new GeneralPath();
arrow.moveTo((float) (lineWidth * 3 * scaleX), (float) (-lineWidth * scaleY * 2));
arrow.lineTo(0, 0);
@@ -314,6 +297,7 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
at.rotate(alpha);
break;
case TRIANGLE:
+ p = new Path();
scaleY = headWidth.ordinal() + 1;
scaleX = headLength.ordinal() + 1;
GeneralPath triangle = new GeneralPath();
@@ -332,7 +316,19 @@ public class XSLFConnectorShape extends XSLFSimpleShape {
if (shape != null) {
shape = at.createTransformedShape(shape);
}
- return shape;
+ return shape == null ? null : new Outline(shape, p);
+ }
+
+ @Override
+ List getCustomOutlines(){
+ List lst = new ArrayList();
+
+ Outline head = getHeadDecoration();
+ if(head != null) lst.add(head);
+
+ Outline tail = getTailDecoration();
+ if(tail != null) lst.add(tail);
+ return lst;
}
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFDrawing.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFDrawing.java
index 6de6e4d775..c3b0b1a511 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFDrawing.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFDrawing.java
@@ -25,7 +25,8 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTGroupShape;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPicture;
import org.openxmlformats.schemas.presentationml.x2006.main.CTShape;
-import java.awt.*;
+import java.awt.Color;
+import java.awt.Rectangle;
/**
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFFreeformShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFFreeformShape.java
index 10490c0c9a..36afac5f8c 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFFreeformShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFFreeformShape.java
@@ -188,10 +188,4 @@ public class XSLFFreeformShape extends XSLFAutoShape {
geom.addNewPathLst();
return ct;
}
-
- @Override
- protected java.awt.Shape getOutline(){
- return getPath();
- }
-
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGraphicFrame.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGraphicFrame.java
index 8e6809bba2..5beb53850c 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGraphicFrame.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGraphicFrame.java
@@ -26,7 +26,7 @@ import org.openxmlformats.schemas.drawingml.x2006.main.CTPositiveSize2D;
import org.openxmlformats.schemas.drawingml.x2006.main.CTTransform2D;
import org.openxmlformats.schemas.presentationml.x2006.main.CTGraphicalObjectFrame;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
/**
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGroupShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGroupShape.java
index 95826af0e4..3159ee98a4 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGroupShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFGroupShape.java
@@ -35,7 +35,7 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTGroupShape;
import org.openxmlformats.schemas.presentationml.x2006.main.CTGroupShapeNonVisual;
import org.openxmlformats.schemas.presentationml.x2006.main.CTShape;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.AffineTransform;
import java.awt.geom.Rectangle2D;
import java.util.List;
@@ -62,10 +62,12 @@ public class XSLFGroupShape extends XSLFShape {
_spPr = shape.getGrpSpPr();
}
+ @Override
public CTGroupShape getXmlObject(){
return _shape;
}
+ @Override
public Rectangle2D getAnchor(){
CTGroupTransform2D xfrm = _spPr.getXfrm();
CTPoint2D off = xfrm.getOff();
@@ -79,6 +81,7 @@ public class XSLFGroupShape extends XSLFShape {
Units.toPoints(cx), Units.toPoints(cy));
}
+ @Override
public void setAnchor(Rectangle2D anchor){
CTGroupTransform2D xfrm = _spPr.isSetXfrm() ? _spPr.getXfrm() : _spPr.addNewXfrm();
CTPoint2D off = xfrm.isSetOff() ? xfrm.getOff() : xfrm.addNewOff();
@@ -93,6 +96,12 @@ public class XSLFGroupShape extends XSLFShape {
ext.setCy(cy);
}
+ /**
+ *
+ * @return the coordinates of the child extents rectangle
+ * used for calculations of grouping, scaling, and rotation
+ * behavior of shapes placed within a group.
+ */
public Rectangle2D getInteriorAnchor(){
CTGroupTransform2D xfrm = _spPr.getXfrm();
CTPoint2D off = xfrm.getChOff();
@@ -106,6 +115,12 @@ public class XSLFGroupShape extends XSLFShape {
Units.toPoints(cx), Units.toPoints(cy));
}
+ /**
+ *
+ * @param anchor the coordinates of the child extents rectangle
+ * used for calculations of grouping, scaling, and rotation
+ * behavior of shapes placed within a group.
+ */
public void setInteriorAnchor(Rectangle2D anchor){
CTGroupTransform2D xfrm = _spPr.isSetXfrm() ? _spPr.getXfrm() : _spPr.addNewXfrm();
CTPoint2D off = xfrm.isSetChOff() ? xfrm.getChOff() : xfrm.addNewChOff();
@@ -120,10 +135,17 @@ public class XSLFGroupShape extends XSLFShape {
ext.setCy(cy);
}
+ /**
+ *
+ * @return child shapes contained witin this group
+ */
public XSLFShape[] getShapes(){
return _shapes.toArray(new XSLFShape[_shapes.size()]);
}
+ /**
+ * Remove the specified shape from this group
+ */
public boolean removeShape(XSLFShape xShape) {
XmlObject obj = xShape.getXmlObject();
if(obj instanceof CTShape){
@@ -138,10 +160,12 @@ public class XSLFGroupShape extends XSLFShape {
return _shapes.remove(xShape);
}
+ @Override
public String getShapeName(){
return _shape.getNvGrpSpPr().getCNvPr().getName();
}
+ @Override
public int getShapeId(){
return (int)_shape.getNvGrpSpPr().getCNvPr().getId();
}
@@ -216,53 +240,37 @@ public class XSLFGroupShape extends XSLFShape {
return sh;
}
-
+ @Override
public void setFlipHorizontal(boolean flip){
_spPr.getXfrm().setFlipH(flip);
}
+ @Override
public void setFlipVertical(boolean flip){
_spPr.getXfrm().setFlipV(flip);
}
- /**
- * Whether the shape is horizontally flipped
- *
- * @return whether the shape is horizontally flipped
- */
+
+ @Override
public boolean getFlipHorizontal(){
return _spPr.getXfrm().getFlipH();
}
+ @Override
public boolean getFlipVertical(){
return _spPr.getXfrm().getFlipV();
}
- /**
- * Rotate this shape.
- *
- * Positive angles are clockwise (i.e., towards the positive y axis);
- * negative angles are counter-clockwise (i.e., towards the negative y axis).
- *
- *
- * @param theta the rotation angle in degrees.
- */
+ @Override
public void setRotation(double theta){
_spPr.getXfrm().setRot((int)(theta*60000));
}
- /**
- * Rotation angle in degrees
- *
- * Positive angles are clockwise (i.e., towards the positive y axis);
- * negative angles are counter-clockwise (i.e., towards the negative y axis).
- *
- *
- * @return rotation angle in degrees
- */
+ @Override
public double getRotation(){
return (double)_spPr.getXfrm().getRot()/60000;
}
+ @Override
public void draw(Graphics2D graphics){
// the coordinate system of this group of shape
@@ -278,16 +286,14 @@ public class XSLFGroupShape extends XSLFShape {
for (XSLFShape shape : getShapes()) {
// remember the initial transform and restore it after we are done with the drawing
- AffineTransform at0 = graphics.getTransform();
+ AffineTransform at = graphics.getTransform();
graphics.setRenderingHint(XSLFRenderingHint.GSAVE, true);
- // apply rotation and flipping
- shape.applyTransform(graphics);
-
+ shape.applyTransform(graphics);
shape.draw(graphics);
// restore the coordinate system
- graphics.setTransform(at0);
+ graphics.setTransform(at);
graphics.setRenderingHint(XSLFRenderingHint.GRESTORE, true);
}
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFImageRendener.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFImageRendener.java
index 8f763622f4..a8bbe8b6a2 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFImageRendener.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFImageRendener.java
@@ -19,10 +19,11 @@
package org.apache.poi.xslf.usermodel;
+import org.apache.poi.openxml4j.opc.PackagePart;
import org.apache.poi.util.Beta;
import javax.imageio.ImageIO;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
@@ -76,11 +77,10 @@ public class XSLFImageRendener {
public boolean drawImage(Graphics2D graphics, XSLFPictureData data,
Rectangle2D anchor) {
try {
- BufferedImage img = readImage(new ByteArrayInputStream(data.getData()));
- if (img != null){
- graphics.drawImage(img, (int) anchor.getX(), (int) anchor.getY(),
- (int) anchor.getWidth(), (int) anchor.getHeight(), null);
- }
+ BufferedImage img = ImageIO.read(data.getPackagePart().getInputStream());
+ graphics.drawImage(img,
+ (int) anchor.getX(), (int) anchor.getY(),
+ (int) anchor.getWidth(), (int) anchor.getHeight(), null);
return true;
} catch (Exception e) {
return false;
@@ -89,12 +89,13 @@ public class XSLFImageRendener {
}
/**
- * create a buffered image from input stream
+ * Create a buffered image from the supplied package part.
+ * This method is called to create texture paints.
*
* @return a BufferedImage containing the decoded
* contents of the input, or null.
*/
- public BufferedImage readImage(InputStream is) throws IOException {
- return ImageIO.read(is);
+ public BufferedImage readImage(PackagePart packagePart) throws IOException {
+ return ImageIO.read(packagePart.getInputStream());
}
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotes.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotes.java
index 3025afcc40..ac4a4d2870 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotes.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotes.java
@@ -75,4 +75,9 @@ public final class XSLFNotes extends XSLFSheet {
protected String getRootElementName(){
return "notes";
}
+
+ @Override
+ public XSLFSheet getMasterSheet() {
+ return null;
+ }
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotesMaster.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotesMaster.java
index 05813bdb96..77ebf05956 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotesMaster.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFNotesMaster.java
@@ -24,7 +24,6 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTNotesMaster;
import org.openxmlformats.schemas.presentationml.x2006.main.NotesMasterDocument;
import java.io.IOException;
-import java.util.Map;
/**
* Notes master object associated with this layout.
@@ -45,8 +44,6 @@ import java.util.Map;
@Beta
public class XSLFNotesMaster extends XSLFSheet {
private CTNotesMaster _slide;
- private Map _layouts;
- private XSLFTheme _theme;
XSLFNotesMaster() {
super();
@@ -70,4 +67,10 @@ import java.util.Map;
protected String getRootElementName(){
return "notesMaster";
}
+
+ @Override
+ public XSLFSheet getMasterSheet() {
+ return null;
+ }
+
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFPictureShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFPictureShape.java
index 0ceb5c5636..445c16e9b0 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFPictureShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFPictureShape.java
@@ -33,12 +33,14 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTPicture;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPictureNonVisual;
import javax.imageio.ImageIO;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayInputStream;
/**
+ * Represents a picture shape
+ *
* @author Yegor Kozlov
*/
@Beta
@@ -114,37 +116,14 @@ public class XSLFPictureShape extends XSLFSimpleShape {
}
@Override
- public void draw(Graphics2D graphics){
- java.awt.Shape outline = getOutline();
-
- // shadow
- XSLFShadow shadow = getShadow();
-
- Paint fill = getFill(graphics);
- Paint line = getLinePaint(graphics);
- if(shadow != null) {
- shadow.draw(graphics);
- }
-
- if(fill != null) {
- graphics.setPaint(fill);
- graphics.fill(outline);
- }
-
+ public void drawContent(Graphics2D graphics) {
XSLFPictureData data = getPictureData();
if(data == null) return;
-
+
XSLFImageRendener renderer = (XSLFImageRendener)graphics.getRenderingHint(XSLFRenderingHint.IMAGE_RENDERER);
if(renderer == null) renderer = new XSLFImageRendener();
renderer.drawImage(graphics, data, getAnchor());
-
- if (line != null){
- graphics.setPaint(line);
- applyStroke(graphics);
- graphics.draw(outline);
- }
}
-
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFRenderingHint.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFRenderingHint.java
index cc0df0fbfd..345d57c4b3 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFRenderingHint.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFRenderingHint.java
@@ -19,7 +19,7 @@
package org.apache.poi.xslf.usermodel;
-import java.awt.*;
+import java.awt.RenderingHints;
/**
*
@@ -38,5 +38,37 @@ public class XSLFRenderingHint extends RenderingHints.Key {
public static final XSLFRenderingHint GSAVE = new XSLFRenderingHint(1);
public static final XSLFRenderingHint GRESTORE = new XSLFRenderingHint(2);
+
+ /**
+ * Use a custom image rendener
+ *
+ * @see XSLFImageRendener
+ */
public static final XSLFRenderingHint IMAGE_RENDERER = new XSLFRenderingHint(3);
+
+ /**
+ * how to render text:
+ *
+ * {@link #TEXT_MODE_CHARACTERS} (default) means to draw via
+ * {@link java.awt.Graphics2D#drawString(java.text.AttributedCharacterIterator, float, float)}.
+ * This mode draws text as characters. Use it if the target graphics writes the actual
+ * character codes instead of glyph outlines (PDFGraphics2D, SVGGraphics2D, etc.)
+ *
+ * {@link #TEXT_MODE_GLYPHS} means to render via
+ * {@link java.awt.font.TextLayout#draw(java.awt.Graphics2D, float, float)}.
+ * This mode draws glyphs as shapes and provides some advanced capabilities such as
+ * justification and font substitution. Use it if the target graphics is an image.
+ *
+ */
+ public static final XSLFRenderingHint TEXT_RENDERING_MODE = new XSLFRenderingHint(4);
+
+ /**
+ * draw text via {@link java.awt.Graphics2D#drawString(java.text.AttributedCharacterIterator, float, float)}
+ */
+ public static final int TEXT_MODE_CHARACTERS = 1;
+
+ /**
+ * draw text via {@link java.awt.font.TextLayout#draw(java.awt.Graphics2D, float, float)}
+ */
+ public static final int TEXT_MODE_GLYPHS = 2;
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShadow.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShadow.java
index 8838fa3e41..70d42ce5c3 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShadow.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShadow.java
@@ -19,8 +19,11 @@ package org.apache.poi.xslf.usermodel;
import org.apache.poi.util.Units;
import org.openxmlformats.schemas.drawingml.x2006.main.CTOuterShadowEffect;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSchemeColor;
-import java.awt.*;
+import java.awt.Color;
+import java.awt.Graphics2D;
+import java.awt.Shape;
import java.awt.geom.Rectangle2D;
/**
@@ -37,36 +40,44 @@ public class XSLFShadow extends XSLFSimpleShape {
_parent = parentShape;
}
- @Override
- public void draw(Graphics2D graphics) {
- Shape outline = _parent.getOutline();
- Paint parentFillColor = _parent.getFill(graphics);
- Paint parentLineColor = _parent.getLinePaint(graphics);
+ public void fill(Graphics2D graphics, Shape outline) {
double angle = getAngle();
double dist = getDistance();
- double dx = dist * Math.cos( Math.toRadians(angle));
- double dy = dist * Math.sin( Math.toRadians(angle));
+ double dx = dist * Math.cos(Math.toRadians(angle));
+ double dy = dist * Math.sin(Math.toRadians(angle));
graphics.translate(dx, dy);
Color fillColor = getFillColor();
if (fillColor != null) {
graphics.setColor(fillColor);
- }
-
- if(parentFillColor != null) {
graphics.fill(outline);
- }
- if(parentLineColor != null) {
- _parent.applyStroke(graphics);
- graphics.draw(outline);
- }
+ }
graphics.translate(-dx, -dy);
}
+ public void draw(Graphics2D graphics, Shape outline) {
+
+ double angle = getAngle();
+ double dist = getDistance();
+ double dx = dist * Math.cos(Math.toRadians(angle));
+ double dy = dist * Math.sin(Math.toRadians(angle));
+
+ graphics.translate(dx, dy);
+
+ Color fillColor = getFillColor();
+ if (fillColor != null) {
+ graphics.setColor(fillColor);
+ graphics.draw(outline);
+ }
+
+ graphics.translate(-dx, -dy);
+ }
+
+
@Override
public Rectangle2D getAnchor(){
return _parent.getAnchor();
@@ -112,6 +123,11 @@ public class XSLFShadow extends XSLFSimpleShape {
public Color getFillColor() {
XSLFTheme theme = getSheet().getTheme();
CTOuterShadowEffect ct = (CTOuterShadowEffect)getXmlObject();
- return ct == null ? null : new XSLFColor(ct, theme).getColor();
+ if(ct == null) {
+ return null;
+ } else {
+ CTSchemeColor phClr = ct.getSchemeClr();
+ return new XSLFColor(ct, theme, phClr).getColor();
+ }
}
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShape.java
index 992973c518..9ff619257c 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFShape.java
@@ -22,24 +22,54 @@ package org.apache.poi.xslf.usermodel;
import org.apache.poi.util.Beta;
import org.apache.xmlbeans.XmlObject;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
/**
+ * Base super-class class for all shapes in PresentationML
+ *
* @author Yegor Kozlov
*/
@Beta
public abstract class XSLFShape {
-
+ /**
+ *
+ * @return the position of this shape within the drawing canvas.
+ * The coordinates are expressed in points
+ */
public abstract Rectangle2D getAnchor();
+ /**
+ *
+ * @param anchor the position of this shape within the drawing canvas.
+ * The coordinates are expressed in points
+ */
public abstract void setAnchor(Rectangle2D anchor);
+ /**
+ *
+ * @return the xml bean holding this shape's data
+ */
public abstract XmlObject getXmlObject();
+ /**
+ *
+ * @return human-readable name of this shape, e.g. "Rectange 3"
+ */
public abstract String getShapeName();
+ /**
+ * Returns a unique identifier for this shape within the current document.
+ * This ID may be used to assist in uniquely identifying this object so that it can
+ * be referred to by other parts of the document.
+ *
+ * If multiple objects within the same document share the same id attribute value,
+ * then the document shall be considered non-conformant.
+ *
+ *
+ * @return unique id of this shape
+ */
public abstract int getShapeId();
/**
@@ -64,8 +94,16 @@ public abstract class XSLFShape {
*/
public abstract double getRotation();
+ /**
+ * @param flip whether the shape is horizontally flipped
+ */
public abstract void setFlipHorizontal(boolean flip);
+ /**
+ * Whether the shape is vertically flipped
+ *
+ * @param flip whether the shape is vertically flipped
+ */
public abstract void setFlipVertical(boolean flip);
/**
@@ -75,14 +113,25 @@ public abstract class XSLFShape {
*/
public abstract boolean getFlipHorizontal();
+ /**
+ * Whether the shape is vertically flipped
+ *
+ * @return whether the shape is vertically flipped
+ */
public abstract boolean getFlipVertical();
+ /**
+ * Draw this shape into the supplied canvas
+ *
+ * @param graphics the graphics to draw into
+ */
public abstract void draw(Graphics2D graphics);
- protected java.awt.Shape getOutline(){
- return getAnchor();
- }
-
+ /**
+ * Apply 2-D transforms before drawing this shape. This includes rotation and flipping.
+ *
+ * @param graphics the graphics whos transform matrix will be modified
+ */
protected void applyTransform(Graphics2D graphics){
Rectangle2D anchor = getAnchor();
@@ -112,5 +161,5 @@ public abstract class XSLFShape {
graphics.translate(-anchor.getX(), -anchor.getY());
}
}
-
+
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSheet.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSheet.java
index 87da995d49..964df2df96 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSheet.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSheet.java
@@ -21,9 +21,9 @@ import org.apache.poi.openxml4j.opc.PackagePart;
import org.apache.poi.openxml4j.opc.PackageRelationship;
import org.apache.poi.openxml4j.opc.TargetMode;
import org.apache.poi.util.Beta;
+import org.apache.poi.util.Internal;
import org.apache.xmlbeans.XmlObject;
import org.apache.xmlbeans.XmlOptions;
-import org.openxmlformats.schemas.drawingml.x2006.main.CTTextListStyle;
import org.openxmlformats.schemas.officeDocument.x2006.relationships.STRelationshipId;
import org.openxmlformats.schemas.presentationml.x2006.main.CTCommonSlideData;
import org.openxmlformats.schemas.presentationml.x2006.main.CTConnector;
@@ -34,22 +34,25 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
import org.openxmlformats.schemas.presentationml.x2006.main.CTShape;
import javax.xml.namespace.QName;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.AffineTransform;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.HashMap;
+import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.regex.Pattern;
@Beta
-public abstract class XSLFSheet extends POIXMLDocumentPart {
+public abstract class XSLFSheet extends POIXMLDocumentPart implements Iterable {
private XSLFCommonSlideData _commonSlideData;
private XSLFDrawing _drawing;
private List _shapes;
private CTGroupShape _spTree;
+
+ private List_placeholders;
private Map _placeholderByIdMap;
private Map _placeholderByTypeMap;
@@ -61,6 +64,10 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
super(part, rel);
}
+ /**
+ *
+ * @return the XMLSlideShow this sheet belongs to
+ */
public XMLSlideShow getSlideShow() {
POIXMLDocumentPart p = getParent();
while(p != null) {
@@ -69,7 +76,7 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
}
p = p.getParent();
}
- return null;
+ throw new IllegalStateException("SlideShow was not found");
}
protected List buildShapes(CTGroupShape spTree){
@@ -92,11 +99,16 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
return shapes;
}
+ /**
+ * @return top-level Xml bean representing this sheet
+ */
public abstract XmlObject getXmlObject();
+ @Internal
public XSLFCommonSlideData getCommonSlideData() {
return _commonSlideData;
}
+
protected void setCommonSlideData(CTCommonSlideData data) {
if(data == null) {
_commonSlideData = null;
@@ -180,10 +192,34 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
return sh;
}
+ /**
+ * Returns an array containing all of the shapes in this sheet
+ *
+ * @return an array of all shapes in this sheet
+ */
public XSLFShape[] getShapes(){
return getShapeList().toArray(new XSLFShape[_shapes.size()]);
}
+ /**
+ * Returns an iterator over the shapes in this sheet
+ *
+ * @return an iterator over the shapes in this sheet
+ */
+ public Iterator iterator(){
+ return getShapeList().iterator();
+ }
+
+ /**
+ * Removes the specified shape from this sheet, if it is present
+ * (optional operation). If this sheet does not contain the element,
+ * it is unchanged.
+ *
+ * @param xShape shape to be removed from this sheet, if present
+ * @return true if this sheet contained the specified element
+ * @throws IllegalArgumentException if the type of the specified shape
+ * is incompatible with this sheet (optional)
+ */
public boolean removeShape(XSLFShape xShape) {
XmlObject obj = xShape.getXmlObject();
CTGroupShape spTree = getSpTree();
@@ -199,10 +235,6 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
return getShapeList().remove(xShape);
}
- public XSLFBackground getBackground(){
- return null;
- }
-
protected abstract String getRootElementName();
protected CTGroupShape getSpTree(){
@@ -248,22 +280,22 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
getXmlObject().set(src.getXmlObject());
}
- public XSLFTheme getTheme(){
+ /**
+ * @return theme (shared styles) associated with this theme.
+ * By default returns null which means that this sheet is theme-less.
+ * Sheets that support the notion of themes (slides, masters, layouts, etc.) should override this
+ * method and return the corresposnding package part.
+ */
+ XSLFTheme getTheme(){
return null;
}
- public XSLFSlideMaster getSlideMaster(){
- return null;
- }
+ /**
+ *
+ * @return master of this sheet.
+ */
+ public abstract XSLFSheet getMasterSheet();
- public XSLFSlideLayout getSlideLayout(){
- return null;
- }
-
- protected CTTextListStyle getTextProperties(Placeholder textType) {
- return null;
- }
-
protected XSLFTextShape getTextShapeByType(Placeholder type){
for(XSLFShape shape : this.getShapes()){
if(shape instanceof XSLFTextShape) {
@@ -286,72 +318,107 @@ public abstract class XSLFSheet extends POIXMLDocumentPart {
return shape;
}
- XSLFSimpleShape getPlaceholderById(int id) {
- if(_placeholderByIdMap == null) {
+ void initPlaceholders() {
+ if(_placeholders == null) {
+ _placeholders = new ArrayList();
_placeholderByIdMap = new HashMap();
+ _placeholderByTypeMap = new HashMap();
+
for(XSLFShape sh : getShapes()){
- if(sh instanceof XSLFSimpleShape){
- XSLFSimpleShape sShape = (XSLFSimpleShape)sh;
+ if(sh instanceof XSLFTextShape){
+ XSLFTextShape sShape = (XSLFTextShape)sh;
CTPlaceholder ph = sShape.getCTPlaceholder();
- if(ph != null && ph.isSetIdx()){
- int idx = (int)ph.getIdx();
- _placeholderByIdMap.put(idx, sShape);
+ if(ph != null) {
+ _placeholders.add(sShape);
+ if(ph.isSetIdx()) {
+ int idx = (int)ph.getIdx();
+ _placeholderByIdMap.put(idx, sShape);
+ }
+ if(ph.isSetType()){
+ _placeholderByTypeMap.put(ph.getType().intValue(), sShape);
+ }
}
}
}
}
+ }
+
+ XSLFSimpleShape getPlaceholderById(int id) {
+ initPlaceholders();
return _placeholderByIdMap.get(id);
}
XSLFSimpleShape getPlaceholderByType(int ordinal) {
- if(_placeholderByTypeMap == null) {
- _placeholderByTypeMap = new HashMap();
- for(XSLFShape sh : getShapes()){
- if(sh instanceof XSLFSimpleShape){
- XSLFSimpleShape sShape = (XSLFSimpleShape)sh;
- CTPlaceholder ph = sShape.getCTPlaceholder();
- if(ph != null && ph.isSetType()){
- _placeholderByTypeMap.put(ph.getType().intValue(), sShape);
- }
- }
- }
- }
+ initPlaceholders();
return _placeholderByTypeMap.get(ordinal);
}
+ /**
+ *
+ * @param idx 0-based index of a placeholder in the sheet
+ * @return placeholder
+ */
+ public XSLFTextShape getPlaceholder(int idx) {
+ initPlaceholders();
+ return _placeholders.get(idx);
+ }
+
+ /**
+ *
+ * @return all placeholder shapes in this sheet
+ */
+ public XSLFTextShape[] getPlaceholders() {
+ initPlaceholders();
+ return _placeholders.toArray(new XSLFTextShape[_placeholders.size()]);
+ }
+
/**
* Checks if this sheet displays the specified shape.
*
- * Subclasses can override it and skip certain shapes from drawings.
+ * Subclasses can override it and skip certain shapes from drawings,
+ * for instance, slide masters and layouts don't display placeholders
*/
protected boolean canDraw(XSLFShape shape){
return true;
}
+ /**
+ *
+ * @return whether shapes on the master sheet should be shown. By default master graphics is turned off.
+ * Sheets that support the notion of master (slide, slideLayout) should override it and
+ * check this setting in the sheet XML
+ */
+ public boolean getFollowMasterGraphics(){
+ return false;
+ }
+
/**
* Render this sheet into the supplied graphics object
*
* @param graphics
*/
public void draw(Graphics2D graphics){
- XSLFBackground bg = getBackground();
- if(bg != null) bg.draw(graphics);
+ XSLFSheet master = getMasterSheet();
+ if(getFollowMasterGraphics() && master != null) master.draw(graphics);
for(XSLFShape shape : getShapeList()) {
if(!canDraw(shape)) continue;
// remember the initial transform and restore it after we are done with drawing
- AffineTransform at0 = graphics.getTransform();
+ AffineTransform at = graphics.getTransform();
+ // concrete implementations can make sense of this hint,
+ // for example PSGraphics2D or PDFGraphics2D would call gsave() / grestore
graphics.setRenderingHint(XSLFRenderingHint.GSAVE, true);
// apply rotation and flipping
shape.applyTransform(graphics);
-
+ // draw stuff
shape.draw(graphics);
// restore the coordinate system
- graphics.setTransform(at0);
+ graphics.setTransform(at);
+
graphics.setRenderingHint(XSLFRenderingHint.GRESTORE, true);
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSimpleShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSimpleShape.java
index 10bad70d03..c67f39e793 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSimpleShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSimpleShape.java
@@ -19,8 +19,6 @@
package org.apache.poi.xslf.usermodel;
-import org.apache.poi.openxml4j.opc.PackagePart;
-import org.apache.poi.openxml4j.opc.PackageRelationship;
import org.apache.poi.util.Beta;
import org.apache.poi.util.Units;
import org.apache.poi.xslf.model.PropertyFetcher;
@@ -28,36 +26,50 @@ import org.apache.poi.xslf.model.geom.Context;
import org.apache.poi.xslf.model.geom.CustomGeometry;
import org.apache.poi.xslf.model.geom.Guide;
import org.apache.poi.xslf.model.geom.IAdjustableShape;
+import org.apache.poi.xslf.model.geom.Outline;
import org.apache.poi.xslf.model.geom.Path;
import org.apache.poi.xslf.model.geom.PresetGeometries;
import org.apache.xmlbeans.XmlObject;
-import org.openxmlformats.schemas.drawingml.x2006.main.*;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTEffectStyleItem;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTGeomGuide;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTLineProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTNonVisualDrawingProps;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTOuterShadowEffect;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPoint2D;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPositiveSize2D;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPresetGeometry2D;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTPresetLineDashProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSRgbColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeStyle;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSolidColorFillProperties;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTStyleMatrix;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTTransform2D;
+import org.openxmlformats.schemas.drawingml.x2006.main.STLineCap;
+import org.openxmlformats.schemas.drawingml.x2006.main.STPresetLineDashVal;
+import org.openxmlformats.schemas.drawingml.x2006.main.STShapeType;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
import org.openxmlformats.schemas.presentationml.x2006.main.STPlaceholderType;
+import java.awt.Color;
+import java.awt.Graphics2D;
+import java.awt.Paint;
+import java.awt.Shape;
import java.awt.geom.AffineTransform;
import java.awt.geom.GeneralPath;
-import java.awt.geom.Point2D;
import java.awt.geom.Rectangle2D;
-import java.awt.image.BufferedImage;
-import java.awt.Paint;
-import java.awt.Graphics2D;
-import java.awt.Color;
-import java.awt.TexturePaint;
-import java.awt.AlphaComposite;
-import java.awt.GradientPaint;
-import java.awt.BasicStroke;
-import java.awt.Stroke;
-import java.util.Arrays;
-import java.util.Comparator;
-import java.lang.reflect.InvocationTargetException;
-import java.lang.reflect.Constructor;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.List;
/**
+ * Represents a single (non-group) shape in a .pptx slide show
+ *
* @author Yegor Kozlov
*/
@Beta
public abstract class XSLFSimpleShape extends XSLFShape {
+
private final XmlObject _shape;
private final XSLFSheet _sheet;
private CTShapeProperties _spPr;
@@ -70,10 +82,15 @@ public abstract class XSLFSimpleShape extends XSLFShape {
_sheet = sheet;
}
+ @Override
public XmlObject getXmlObject() {
return _shape;
}
+ /**
+ *
+ * @return the sheet this shape belongs to
+ */
public XSLFSheet getSheet() {
return _sheet;
}
@@ -88,10 +105,12 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return stEnum == null ? 0 : stEnum.intValue();
}
+ @Override
public String getShapeName() {
return getNvPr().getName();
}
+ @Override
public int getShapeId() {
return (int) getNvPr().getId();
}
@@ -132,22 +151,22 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return _spStyle;
}
- protected CTPlaceholder getCTPlaceholder(){
- if(_ph == null){
+ protected CTPlaceholder getCTPlaceholder() {
+ if (_ph == null) {
XmlObject[] obj = _shape.selectPath(
"declare namespace p='http://schemas.openxmlformats.org/presentationml/2006/main' .//*/p:nvPr/p:ph");
- if(obj.length == 1){
- _ph = (CTPlaceholder)obj[0];
+ if (obj.length == 1) {
+ _ph = (CTPlaceholder) obj[0];
}
}
return _ph;
}
- private CTTransform2D getXfrm(){
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
+ private CTTransform2D getXfrm() {
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
CTShapeProperties pr = shape.getSpPr();
- if(pr.isSetXfrm()){
+ if (pr.isSetXfrm()) {
setValue(pr.getXfrm());
return true;
}
@@ -158,10 +177,11 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return fetcher.getValue();
}
+ @Override
public Rectangle2D getAnchor() {
CTTransform2D xfrm = getXfrm();
-
+
CTPoint2D off = xfrm.getOff();
long x = off.getX();
long y = off.getY();
@@ -173,6 +193,7 @@ public abstract class XSLFSimpleShape extends XSLFShape {
Units.toPoints(cx), Units.toPoints(cy));
}
+ @Override
public void setAnchor(Rectangle2D anchor) {
CTShapeProperties spPr = getSpPr();
CTTransform2D xfrm = spPr.isSetXfrm() ? spPr.getXfrm() : spPr.addNewXfrm();
@@ -189,64 +210,46 @@ public abstract class XSLFSimpleShape extends XSLFShape {
ext.setCy(cy);
}
- /**
- * Rotate this shape.
- *
- * Positive angles are clockwise (i.e., towards the positive y axis);
- * negative angles are counter-clockwise (i.e., towards the negative y
- * axis).
- *
- * Positive angles are clockwise (i.e., towards the positive y axis);
- * negative angles are counter-clockwise (i.e., towards the negative y
- * axis).
- *
- *
- * @return rotation angle in degrees
- */
+ @Override
public double getRotation() {
CTTransform2D xfrm = getXfrm();
return (double) xfrm.getRot() / 60000;
}
+ @Override
public void setFlipHorizontal(boolean flip) {
CTShapeProperties spPr = getSpPr();
CTTransform2D xfrm = spPr.isSetXfrm() ? spPr.getXfrm() : spPr.addNewXfrm();
xfrm.setFlipH(flip);
}
+ @Override
public void setFlipVertical(boolean flip) {
CTShapeProperties spPr = getSpPr();
CTTransform2D xfrm = spPr.isSetXfrm() ? spPr.getXfrm() : spPr.addNewXfrm();
xfrm.setFlipV(flip);
}
- /**
- * Whether the shape is horizontally flipped
- *
- * @return whether the shape is horizontally flipped
- */
+ @Override
public boolean getFlipHorizontal() {
return getXfrm().getFlipH();
}
+ @Override
public boolean getFlipVertical() {
return getXfrm().getFlipV();
}
/**
- * Get line properties defined in the theme (if any)
+ * Get default line properties defined in the theme (if any).
+ * Used internally to resolve shape properties.
*
* @return line propeties from the theme of null
*/
@@ -262,6 +265,10 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return ln;
}
+ /**
+ * @param color the color to paint the shape outline.
+ * A null value turns off the shape outline.
+ */
public void setLineColor(Color color) {
CTShapeProperties spPr = getSpPr();
if (color == null) {
@@ -281,48 +288,24 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
}
+ /**
+ *
+ * @return the color of the shape outline or null
+ * if outline is turned off
+ */
public Color getLineColor() {
- Paint paint = getLinePaint(null);
- if(paint instanceof Color){
- return (Color)paint;
+ RenderableShape rShape = new RenderableShape(this);
+ Paint paint = rShape.getLinePaint(null);
+ if (paint instanceof Color) {
+ return (Color) paint;
}
return null;
}
- public Paint getLinePaint(final Graphics2D graphics) {
- final XSLFTheme theme = _sheet.getTheme();
- final Color nofill = new Color(0,0,0,0);
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
- CTLineProperties spPr = shape.getSpPr().getLn();
- if (spPr != null) {
- if (spPr.isSetNoFill()) {
- setValue(nofill); // use it as 'nofill' value
- return true;
- }
- Paint paint = getPaint(graphics, spPr);
- if (paint != null) {
- setValue( paint );
- return true;
- }
- }
- return false;
-
- }
- };
- fetchShapeProperty(fetcher);
-
- Paint color = fetcher.getValue();
- if(color == null){
- // line color was not found, check if it is defined in the theme
- CTShapeStyle style = getSpStyle();
- if (style != null) {
- color = new XSLFColor(style.getLnRef(), theme).getColor();
- }
- }
- return color == nofill ? null : color;
- }
-
+ /**
+ *
+ * @param width line width in points. 0 means no line
+ */
public void setLineWidth(double width) {
CTShapeProperties spPr = getSpPr();
if (width == 0.) {
@@ -335,9 +318,13 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
}
+ /**
+ *
+ * @return line width in points. 0 means no line.
+ */
public double getLineWidth() {
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
CTShapeProperties spPr = shape.getSpPr();
CTLineProperties ln = spPr.getLn();
if (ln != null) {
@@ -347,7 +334,7 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
if (ln.isSetW()) {
- setValue( Units.toPoints(ln.getW()) );
+ setValue(Units.toPoints(ln.getW()));
return true;
}
}
@@ -357,7 +344,7 @@ public abstract class XSLFSimpleShape extends XSLFShape {
fetchShapeProperty(fetcher);
double lineWidth = 0;
- if(fetcher.getValue() == null) {
+ if (fetcher.getValue() == null) {
CTLineProperties defaultLn = getDefaultLineProperties();
if (defaultLn != null) {
if (defaultLn.isSetW()) lineWidth = Units.toPoints(defaultLn.getW());
@@ -369,6 +356,10 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return lineWidth;
}
+ /**
+ *
+ * @param dash a preset line dashing scheme to stroke thr shape outline
+ */
public void setLineDash(LineDash dash) {
CTShapeProperties spPr = getSpPr();
if (dash == null) {
@@ -384,16 +375,19 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
}
+ /**
+ * @return a preset line dashing scheme to stroke thr shape outline
+ */
public LineDash getLineDash() {
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
CTShapeProperties spPr = shape.getSpPr();
CTLineProperties ln = spPr.getLn();
if (ln != null) {
CTPresetLineDashProperties ctDash = ln.getPrstDash();
if (ctDash != null) {
- setValue( LineDash.values()[ctDash.getVal().intValue() - 1] );
+ setValue(LineDash.values()[ctDash.getVal().intValue() - 1]);
return true;
}
}
@@ -403,7 +397,7 @@ public abstract class XSLFSimpleShape extends XSLFShape {
fetchShapeProperty(fetcher);
LineDash dash = fetcher.getValue();
- if(dash == null){
+ if (dash == null) {
CTLineProperties defaultLn = getDefaultLineProperties();
if (defaultLn != null) {
CTPresetLineDashProperties ctDash = defaultLn.getPrstDash();
@@ -415,6 +409,10 @@ public abstract class XSLFSimpleShape extends XSLFShape {
return dash;
}
+ /**
+ *
+ * @param cap the line end cap style
+ */
public void setLineCap(LineCap cap) {
CTShapeProperties spPr = getSpPr();
if (cap == null) {
@@ -427,15 +425,19 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
}
+ /**
+ *
+ * @return the line end cap style
+ */
public LineCap getLineCap() {
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
CTShapeProperties spPr = shape.getSpPr();
CTLineProperties ln = spPr.getLn();
if (ln != null) {
STLineCap.Enum stCap = ln.getCap();
if (stCap != null) {
- setValue( LineCap.values()[stCap.intValue() - 1] );
+ setValue(LineCap.values()[stCap.intValue() - 1]);
return true;
}
}
@@ -445,7 +447,7 @@ public abstract class XSLFSimpleShape extends XSLFShape {
fetchShapeProperty(fetcher);
LineCap cap = fetcher.getValue();
- if(cap == null){
+ if (cap == null) {
CTLineProperties defaultLn = getDefaultLineProperties();
if (defaultLn != null) {
STLineCap.Enum stCap = defaultLn.getCap();
@@ -469,10 +471,10 @@ public abstract class XSLFSimpleShape extends XSLFShape {
if (color == null) {
if (spPr.isSetSolidFill()) spPr.unsetSolidFill();
- if(!spPr.isSetNoFill()) spPr.addNewNoFill();
+ if (!spPr.isSetNoFill()) spPr.addNewNoFill();
} else {
- if(spPr.isSetNoFill()) spPr.unsetNoFill();
-
+ if (spPr.isSetNoFill()) spPr.unsetNoFill();
+
CTSolidColorFillProperties fill = spPr.isSetSolidFill() ? spPr
.getSolidFill() : spPr.addNewSolidFill();
@@ -485,55 +487,24 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
/**
- * @return solid fill color of null if not set
+ * @return solid fill color of null if not set or fill color
+ * is not solid (pattern or gradient)
*/
public Color getFillColor() {
- Paint paint = getFill(null);
- if(paint instanceof Color){
- return (Color)paint;
+ RenderableShape rShape = new RenderableShape(this);
+ Paint paint = rShape.getFillPaint(null);
+ if (paint instanceof Color) {
+ return (Color) paint;
}
return null;
}
/**
- * fetch shape fill as a java.awt.Paint
- *
- * @return either Color or GradientPaint or TexturePaint or null
+ * @return shadow of this shape or null if shadow is disabled
*/
- Paint getFill(final Graphics2D graphics) {
- final XSLFTheme theme = _sheet.getTheme();
- final Color nofill = new Color(0xFF,0xFF,0xFF, 0);
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
- CTShapeProperties spPr = shape.getSpPr();
- if (spPr.isSetNoFill()) {
- setValue(nofill); // use it as 'nofill' value
- return true;
- }
- Paint paint = getPaint(graphics, spPr);
- if (paint != null) {
- setValue( paint );
- return true;
- }
- return false;
- }
- };
- fetchShapeProperty(fetcher);
-
- Paint paint = fetcher.getValue();
- if(paint == null){
- // fill color was not found, check if it is defined in the theme
- CTShapeStyle style = getSpStyle();
- if (style != null) {
- paint = new XSLFColor(style.getFillRef(), theme).getColor();
- }
- }
- return paint == nofill ? null : paint;
- }
-
- public XSLFShadow getShadow(){
- PropertyFetcher fetcher = new PropertyFetcher(){
- public boolean fetch(XSLFSimpleShape shape){
+ public XSLFShadow getShadow() {
+ PropertyFetcher fetcher = new PropertyFetcher() {
+ public boolean fetch(XSLFSimpleShape shape) {
CTShapeProperties spPr = shape.getSpPr();
if (spPr.isSetEffectLst()) {
CTOuterShadowEffect obj = spPr.getEffectLst().getOuterShdw();
@@ -546,233 +517,48 @@ public abstract class XSLFSimpleShape extends XSLFShape {
fetchShapeProperty(fetcher);
CTOuterShadowEffect obj = fetcher.getValue();
- if(obj == null){
+ if (obj == null) {
// fill color was not found, check if it is defined in the theme
CTShapeStyle style = getSpStyle();
if (style != null) {
// 1-based index of a shadow style within the style matrix
int idx = (int) style.getEffectRef().getIdx();
-
- CTStyleMatrix styleMatrix = _sheet.getTheme().getXmlObject().getThemeElements().getFmtScheme();
- CTEffectStyleItem ef = styleMatrix.getEffectStyleLst().getEffectStyleArray(idx - 1);
- obj = ef.getEffectLst().getOuterShdw();
+ if(idx != 0) {
+ CTStyleMatrix styleMatrix = _sheet.getTheme().getXmlObject().getThemeElements().getFmtScheme();
+ CTEffectStyleItem ef = styleMatrix.getEffectStyleLst().getEffectStyleArray(idx - 1);
+ obj = ef.getEffectLst().getOuterShdw();
+ }
}
}
return obj == null ? null : new XSLFShadow(obj, this);
}
+ @Override
public void draw(Graphics2D graphics) {
-
+ RenderableShape rShape = new RenderableShape(this);
+ rShape.render(graphics);
}
- @SuppressWarnings("deprecation") // getXYZArray() array accessors are deprecated
- protected Paint getPaint(Graphics2D graphics, XmlObject spPr) {
- XSLFTheme theme = getSheet().getTheme();
- Rectangle2D anchor = getAnchor();
-
- Paint paint = null;
- for(XmlObject obj : spPr.selectPath("*")){
- if(obj instanceof CTNoFillProperties){
- paint = null;
- break;
- }
- if(obj instanceof CTSolidColorFillProperties){
- CTSolidColorFillProperties solidFill = (CTSolidColorFillProperties)obj;
- XSLFColor c = new XSLFColor(solidFill, theme);
- paint = c.getColor();
- }
- if(obj instanceof CTBlipFillProperties){
- CTBlipFillProperties blipFill = (CTBlipFillProperties)obj;
- CTBlip blip = blipFill.getBlip();
- String blipId = blip.getEmbed();
- PackagePart p = getSheet().getPackagePart();
- PackageRelationship rel = p.getRelationship(blipId);
- if (rel != null) {
- XSLFImageRendener renderer = null;
- if(graphics != null) renderer = (XSLFImageRendener)graphics.getRenderingHint(XSLFRenderingHint.IMAGE_RENDERER);
- if(renderer == null) renderer = new XSLFImageRendener();
-
- try {
- BufferedImage img = renderer.readImage(p.getRelatedPart(rel).getInputStream());
- if(blip.sizeOfAlphaModFixArray() > 0){
- float alpha = blip.getAlphaModFixArray(0).getAmt()/100000.f;
- AlphaComposite ac = AlphaComposite.getInstance(
- AlphaComposite.SRC_OVER, alpha);
- if(graphics != null) graphics.setComposite(ac);
- }
-
- paint = new TexturePaint(
- img, new Rectangle2D.Double(0, 0, img.getWidth(), img.getHeight()));
- }
- catch (Exception e) {
- return null;
- }
- }
- }
- if(obj instanceof CTGradientFillProperties){
- CTGradientFillProperties gradFill = (CTGradientFillProperties)obj;
- double angle;
- if(gradFill.isSetLin()) {
- angle = gradFill.getLin().getAng() / 60000;
- } else {
- // XSLF only supports linear gradient fills. Other types are filled as liner with angle=90 degrees
- angle = 90;
- }
- CTGradientStop[] gs = gradFill.getGsLst().getGsArray();
-
- Arrays.sort(gs, new Comparator(){
- public int compare(CTGradientStop o1, CTGradientStop o2){
- Integer pos1 = o1.getPos();
- Integer pos2 = o2.getPos();
- return pos1.compareTo(pos2);
- }
- });
-
- Color[] colors = new Color[gs.length];
- float[] fractions = new float[gs.length];
-
- AffineTransform at = AffineTransform.getRotateInstance(
- Math.toRadians(angle),
- anchor.getX() + anchor.getWidth()/2,
- anchor.getY() + anchor.getHeight()/2);
-
- double diagonal = Math.sqrt(anchor.getHeight()*anchor.getHeight() + anchor.getWidth()*anchor.getWidth());
- Point2D p1 = new Point2D.Double(anchor.getX() + anchor.getWidth()/2 - diagonal/2,
- anchor.getY() + anchor.getHeight()/2);
- p1 = at.transform(p1, null);
-
- Point2D p2 = new Point2D.Double(anchor.getX() + anchor.getWidth(), anchor.getY() + anchor.getHeight()/2);
- p2 = at.transform(p2, null);
-
- norm(p1, anchor);
- norm(p2, anchor);
-
- for(int i = 0; i < gs.length; i++){
- CTGradientStop stop = gs[i];
- colors[i] = new XSLFColor(stop, theme).getColor();
- fractions[i] = stop.getPos() / 100000.f;
- }
-
- paint = createGradientPaint(p1, p2, fractions, colors);
- }
- }
- return paint;
- }
/**
- * Trick to return GradientPaint on JDK 1.5 and LinearGradientPaint on JDK 1.6+
- */
- private Paint createGradientPaint(Point2D p1, Point2D p2, float[] fractions, Color[] colors){
- Paint paint;
- try {
- Class clz = Class.forName("java.awt.LinearGradientPaint");
- Constructor c =
- clz.getConstructor(Point2D.class, Point2D.class, float[].class, Color[].class);
- paint = (Paint)c.newInstance(p1, p2, fractions, colors);
- } catch (ClassNotFoundException e){
- paint = new GradientPaint(p1, colors[0], p2, colors[colors.length - 1]);
- } catch (Exception e){
- throw new RuntimeException(e);
- }
- return paint;
- }
-
- void norm(Point2D p, Rectangle2D anchor){
- if(p.getX() < anchor.getX()){
- p.setLocation(anchor.getX(), p.getY());
- } else if(p.getX() > (anchor.getX() + anchor.getWidth())){
- p.setLocation(anchor.getX() + anchor.getWidth(), p.getY());
- }
-
- if(p.getY() < anchor.getY()){
- p.setLocation(p.getX(), anchor.getY());
- } else if (p.getY() > (anchor.getY() + anchor.getHeight())){
- p.setLocation(p.getX(), anchor.getY() + anchor.getHeight());
- }
- }
-
- protected float[] getDashPattern(LineDash lineDash, float lineWidth) {
- float[] dash = null;
- switch (lineDash) {
- case SYS_DOT:
- dash = new float[]{lineWidth, lineWidth};
- break;
- case SYS_DASH:
- dash = new float[]{2 * lineWidth, 2 * lineWidth};
- break;
- case DASH:
- dash = new float[]{3 * lineWidth, 4 * lineWidth};
- break;
- case DASH_DOT:
- dash = new float[]{4 * lineWidth, 3 * lineWidth, lineWidth,
- 3 * lineWidth};
- break;
- case LG_DASH:
- dash = new float[]{8 * lineWidth, 3 * lineWidth};
- break;
- case LG_DASH_DOT:
- dash = new float[]{8 * lineWidth, 3 * lineWidth, lineWidth,
- 3 * lineWidth};
- break;
- case LG_DASH_DOT_DOT:
- dash = new float[]{8 * lineWidth, 3 * lineWidth, lineWidth,
- 3 * lineWidth, lineWidth, 3 * lineWidth};
- break;
- }
- return dash;
- }
-
- protected void applyStroke(Graphics2D graphics) {
-
- float lineWidth = (float) getLineWidth();
- LineDash lineDash = getLineDash();
- float[] dash = null;
- float dash_phase = 0;
- if (lineDash != null) {
- dash = getDashPattern(lineDash, lineWidth);
- }
-
- int cap = BasicStroke.CAP_BUTT;
- LineCap lineCap = getLineCap();
- if (lineCap != null) {
- switch (lineCap) {
- case ROUND:
- cap = BasicStroke.CAP_ROUND;
- break;
- case SQUARE:
- cap = BasicStroke.CAP_SQUARE;
- break;
- default:
- cap = BasicStroke.CAP_BUTT;
- break;
- }
- }
-
- int meter = BasicStroke.JOIN_ROUND;
-
- Stroke stroke = new BasicStroke(lineWidth, cap, meter, Math.max(1, lineWidth), dash,
- dash_phase);
- graphics.setStroke(stroke);
- }
-
- /**
- * Walk up the inheritance tree and fetch properties.
+ * Walk up the inheritance tree and fetch shape properties.
*
- * slide <-- slideLayout <-- slideMaster
+ * The following order of inheritance is assumed:
+ *
+ * slide <-- slideLayout <-- slideMaster <-- default styles in the slideMaster
+ *
*
- *
- * @param visitor the object that collects the desired property
+ * @param visitor the object that collects the desired property
* @return true if the property was fetched
*/
- boolean fetchShapeProperty(PropertyFetcher visitor){
+ boolean fetchShapeProperty(PropertyFetcher visitor) {
boolean ok = visitor.fetch(this);
XSLFSimpleShape masterShape;
- if(!ok){
-
- // first try to fetch from the slide layout
- XSLFSlideLayout layout = getSheet().getSlideLayout();
- if(layout != null) {
+ XSLFSheet layout = getSheet().getMasterSheet();
+ if (layout != null) {
+ if (!ok) {
+ // first try to fetch from the slide layout
CTPlaceholder ph = getCTPlaceholder();
if (ph != null) {
masterShape = layout.getPlaceholder(ph);
@@ -781,96 +567,77 @@ public abstract class XSLFSimpleShape extends XSLFShape {
}
}
}
- }
- // try slide master
- if (!ok) {
- int textType;
- CTPlaceholder ph = getCTPlaceholder();
- if(ph == null || !ph.isSetType()) textType = STPlaceholderType.INT_BODY;
- else {
- switch(ph.getType().intValue()){
- case STPlaceholderType.INT_TITLE:
- case STPlaceholderType.INT_CTR_TITLE:
- textType = STPlaceholderType.INT_TITLE;
- break;
- case STPlaceholderType.INT_FTR:
- case STPlaceholderType.INT_SLD_NUM:
- case STPlaceholderType.INT_DT:
- textType = ph.getType().intValue();
- break;
- default:
- textType = STPlaceholderType.INT_BODY;
- break;
+ // try slide master
+ if (!ok) {
+ int textType;
+ CTPlaceholder ph = getCTPlaceholder();
+ if (ph == null || !ph.isSetType()) textType = STPlaceholderType.INT_BODY;
+ else {
+ switch (ph.getType().intValue()) {
+ case STPlaceholderType.INT_TITLE:
+ case STPlaceholderType.INT_CTR_TITLE:
+ textType = STPlaceholderType.INT_TITLE;
+ break;
+ case STPlaceholderType.INT_FTR:
+ case STPlaceholderType.INT_SLD_NUM:
+ case STPlaceholderType.INT_DT:
+ textType = ph.getType().intValue();
+ break;
+ default:
+ textType = STPlaceholderType.INT_BODY;
+ break;
+ }
}
- }
- XSLFSlideMaster master = getSheet().getSlideMaster();
- if(master != null) {
- masterShape = master.getPlaceholderByType(textType);
- if (masterShape != null) {
- ok = visitor.fetch(masterShape);
+ XSLFSheet master = layout.getMasterSheet();
+ if (master != null) {
+ masterShape = master.getPlaceholderByType(textType);
+ if (masterShape != null) {
+ ok = visitor.fetch(masterShape);
+ }
}
}
}
-
return ok;
}
-
- @Override
- protected java.awt.Shape getOutline(){
- PresetGeometries dict = PresetGeometries.getInstance();
+ /**
+ *
+ * @return definition of the shape geometry
+ */
+ CustomGeometry getGeometry(){
CTShapeProperties spPr = getSpPr();
- String name;
- if(spPr.isSetPrstGeom()) {
- name = spPr.getPrstGeom().getPrst().toString();
- } else {
- name = "rect";
- }
- CustomGeometry geom = dict.get(name);
- Rectangle2D anchor = getAnchor();
- if(geom != null) {
- // the guides in the shape definitions are all defined relative to each other,
- // so we build the path starting from (0,0).
- final Rectangle2D anchorEmu = new Rectangle2D.Double(
- 0,
- 0,
- Units.toEMU(anchor.getWidth()),
- Units.toEMU(anchor.getHeight())
- );
-
- GeneralPath path = new GeneralPath();
- Context ctx = new Context(geom, new IAdjustableShape() {
- public Rectangle2D getAnchor() {
- return anchorEmu;
- }
-
- public Guide getAdjustValue(String name) {
- CTPresetGeometry2D prst = getSpPr().getPrstGeom();
- if(prst.isSetAvLst()) {
- for(CTGeomGuide g : prst.getAvLst().getGdList()){
- if(g.getName().equals(name)) {
- return new Guide(g);
- }
- }
- }
- return null;
- }
- });
-
- for(Path p : geom){
- path.append( p.getPath(ctx) , false);
+ CustomGeometry geom;
+ PresetGeometries dict = PresetGeometries.getInstance();
+ if(spPr.isSetPrstGeom()){
+ String name = spPr.getPrstGeom().getPrst().toString();
+ geom = dict.get(name);
+ if(geom == null) {
+ throw new IllegalStateException("Unknown shape geometry: " + name);
}
-
- // translate the result to the canvas coordinates in points
- AffineTransform at = new AffineTransform();
- at.scale(
- 1.0/Units.EMU_PER_POINT, 1.0/Units.EMU_PER_POINT);
- at.translate(Units.toEMU(anchor.getX()), Units.toEMU(anchor.getY()));
- return at.createTransformedShape(path);
+ } else if (spPr.isSetCustGeom()){
+ geom = new CustomGeometry(spPr.getCustGeom());
} else {
- return anchor;
+ geom = dict.get("rect");
}
- }
+ return geom;
+ }
+
+ /**
+ * @return any shape-specific geometry that is not included in presetShapeDefinitions.xml
+ * (line decorations, etc)
+ */
+ List getCustomOutlines(){
+ return Collections.emptyList();
+ }
+
+ /**
+ * draw any content within this shape (image, text, etc.).
+ *
+ * @param graphics the graphics to draw into
+ */
+ public void drawContent(Graphics2D graphics){
+
+ }
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlide.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlide.java
index f6f92d4f59..598baaeee4 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlide.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlide.java
@@ -32,7 +32,7 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTGroupShapeNonVisua
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlide;
import org.openxmlformats.schemas.presentationml.x2006.main.SldDocument;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.io.IOException;
@Beta
@@ -109,11 +109,11 @@ public final class XSLFSlide extends XSLFSheet {
return "sld";
}
- public XSLFSlideMaster getMasterSheet(){
- return getSlideLayout().getSlideMaster();
+ @Override
+ public XSLFSlideLayout getMasterSheet(){
+ return getSlideLayout();
}
- @Override
public XSLFSlideLayout getSlideLayout(){
if(_layout == null){
for (POIXMLDocumentPart p : getRelations()) {
@@ -128,7 +128,6 @@ public final class XSLFSlide extends XSLFSheet {
return _layout;
}
- @Override
public XSLFSlideMaster getSlideMaster(){
return getSlideLayout().getSlideMaster();
}
@@ -159,20 +158,12 @@ public final class XSLFSlide extends XSLFSheet {
}
if(_notes == null) {
// This slide lacks notes
- // Not al have them, sorry...
+ // Not all have them, sorry...
return null;
}
return _notes;
}
- public void setFollowMasterBackground(boolean value){
- _slide.setShowMasterSp(value);
- }
-
- public boolean getFollowMasterBackground(){
- return !_slide.isSetShowMasterSp() || _slide.getShowMasterSp();
- }
-
/**
*
* @return title of this slide or empty string if title is not set
@@ -182,27 +173,59 @@ public final class XSLFSlide extends XSLFSheet {
return txt == null ? "" : txt.getText();
}
+ @Override
public XSLFTheme getTheme(){
return getSlideLayout().getSlideMaster().getTheme();
}
- @Override
- public void draw(Graphics2D graphics){
+ /**
+ *
+ * @return the information about background appearance of this slide
+ */
+ public XSLFBackground getBackground() {
- if (getFollowMasterBackground()){
- XSLFSlideLayout layout = getSlideLayout();
- layout.draw(graphics);
- }
- super.draw(graphics);
- }
-
- @Override
- public XSLFBackground getBackground(){
if(_slide.getCSld().isSetBg()) {
return new XSLFBackground(_slide.getCSld().getBg(), this);
}
+
+ XSLFSlideLayout layout = getMasterSheet();
+ if(layout.getXmlObject().getCSld().isSetBg()) {
+ return new XSLFBackground(layout.getXmlObject().getCSld().getBg(), this);
+ }
+
+ XSLFSlideMaster master = layout.getMasterSheet();
+ if(master.getXmlObject().getCSld().isSetBg()) {
+ return new XSLFBackground(master.getXmlObject().getCSld().getBg(), this);
+ }
return null;
}
+ /**
+ *
+ * @return whether shapes on the master slide should be shown or not.
+ */
+ public boolean getFollowMasterGraphics(){
+ return !_slide.isSetShowMasterSp() || _slide.getShowMasterSp();
+ }
+
+ /**
+ *
+ * @param value whether shapes on the master slide should be shown or not.
+ */
+ public void setFollowMasterGraphics(boolean value){
+ _slide.setShowMasterSp(value);
+ }
+
+
+ @Override
+ public void draw(Graphics2D graphics){
+
+ XSLFBackground bg = getBackground();
+ if(bg != null) bg.draw(graphics);
+
+ super.draw(graphics);
+ }
+
+
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideLayout.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideLayout.java
index 699d153345..4a505972dc 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideLayout.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideLayout.java
@@ -22,7 +22,6 @@ import org.apache.poi.openxml4j.opc.PackageRelationship;
import org.apache.poi.util.Beta;
import org.apache.poi.util.Internal;
import org.apache.xmlbeans.XmlException;
-import org.openxmlformats.schemas.drawingml.x2006.main.CTTextListStyle;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideLayout;
import org.openxmlformats.schemas.presentationml.x2006.main.SldLayoutDocument;
@@ -42,13 +41,13 @@ public class XSLFSlideLayout extends XSLFSheet {
public XSLFSlideLayout(PackagePart part, PackageRelationship rel) throws IOException, XmlException {
super(part, rel);
SldLayoutDocument doc =
- SldLayoutDocument.Factory.parse(getPackagePart().getInputStream());
+ SldLayoutDocument.Factory.parse(getPackagePart().getInputStream());
_layout = doc.getSldLayout();
setCommonSlideData(_layout.getCSld());
}
- public String getName(){
+ public String getName() {
return _layout.getCSld().getName();
}
@@ -61,79 +60,92 @@ public class XSLFSlideLayout extends XSLFSheet {
}
@Override
- protected String getRootElementName(){
+ protected String getRootElementName() {
return "sldLayout";
}
/**
* Slide master object associated with this layout.
- *
- * Within a slide master slide are contained all elements
- * that describe the objects and their corresponding formatting
- * for within a presentation slide.
- *
- *
- * Within a slide master slide are two main elements.
- * The cSld element specifies the common slide elements such as shapes and
- * their attached text bodies. Then the txStyles element specifies the
- * formatting for the text within each of these shapes. The other properties
- * within a slide master slide specify other properties for within a presentation slide
- * such as color information, headers and footers, as well as timing and
- * transition information for all corresponding presentation slides.
- *
*
* @return slide master. Never null.
* @throws IllegalStateException if slide master was not found
*/
- @Override
- public XSLFSlideMaster getSlideMaster(){
- if(_master == null){
+ public XSLFSlideMaster getSlideMaster() {
+ if (_master == null) {
for (POIXMLDocumentPart p : getRelations()) {
- if (p instanceof XSLFSlideMaster){
- _master = (XSLFSlideMaster)p;
- }
- }
+ if (p instanceof XSLFSlideMaster) {
+ _master = (XSLFSlideMaster) p;
+ }
+ }
}
- if(_master == null) {
+ if (_master == null) {
throw new IllegalStateException("SlideMaster was not found for " + this.toString());
}
return _master;
}
-
- public XSLFTheme getTheme(){
- return getSlideMaster().getTheme();
- }
+ @Override
+ public XSLFSlideMaster getMasterSheet() {
+ return getSlideMaster();
+ }
+
+ @Override
+ public XSLFTheme getTheme() {
+ return getSlideMaster().getTheme();
+ }
@Override
- protected CTTextListStyle getTextProperties(Placeholder textType) {
- XSLFTextShape lp = getTextShapeByType(textType);
- CTTextListStyle props = lp.getTextBody(false).getLstStyle();
- return props;
+ public boolean getFollowMasterGraphics() {
+ return !_layout.isSetShowMasterSp() || _layout.getShowMasterSp();
}
/**
* Render this sheet into the supplied graphics object
- *
*/
@Override
- protected boolean canDraw(XSLFShape shape){
- if(shape instanceof XSLFSimpleShape){
- XSLFSimpleShape txt = (XSLFSimpleShape)shape;
+ protected boolean canDraw(XSLFShape shape) {
+ if (shape instanceof XSLFSimpleShape) {
+ XSLFSimpleShape txt = (XSLFSimpleShape) shape;
CTPlaceholder ph = txt.getCTPlaceholder();
- if(ph != null) {
+ if (ph != null) {
return false;
}
}
return true;
}
- @Override
- public XSLFBackground getBackground(){
- if(_layout.getCSld().isSetBg()) {
- return new XSLFBackground(_layout.getCSld().getBg(), this);
+ /**
+ * Copy placeholders from this layout to the destination slide
+ *
+ * @param slide destination slide
+ */
+ public void copyLayout(XSLFSlide slide) {
+ for (XSLFShape sh : getShapes()) {
+ if (sh instanceof XSLFTextShape) {
+ XSLFTextShape tsh = (XSLFTextShape) sh;
+ Placeholder ph = tsh.getTextType();
+ if (ph == null) continue;
+
+ switch (ph) {
+ // these are special and not copied by default
+ case DATETIME:
+ case SLIDE_NUMBER:
+ case FOOTER:
+ break;
+ default:
+ slide.getSpTree().addNewSp().set(tsh.getXmlObject().copy());
+ }
+ }
}
- return getSlideMaster().getBackground();
+ }
+
+ /**
+ *
+ * @return type of this layout
+ */
+ public SlideLayout getType(){
+ int ordinal = _layout.getType().intValue() - 1;
+ return SlideLayout.values()[ordinal];
}
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideMaster.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideMaster.java
index 2ec83609a8..e73ee7a4c1 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideMaster.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFSlideMaster.java
@@ -21,7 +21,9 @@ import org.apache.poi.openxml4j.opc.PackagePart;
import org.apache.poi.openxml4j.opc.PackageRelationship;
import org.apache.poi.util.Beta;
import org.apache.xmlbeans.XmlException;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTColorMapping;
import org.openxmlformats.schemas.drawingml.x2006.main.CTTextListStyle;
+import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideMaster;
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideMasterTextStyles;
import org.openxmlformats.schemas.presentationml.x2006.main.SldMasterDocument;
@@ -78,7 +80,12 @@ import java.util.Map;
return "sldMaster";
}
- public XSLFSlideLayout getLayout(String name){
+ @Override
+ public XSLFSheet getMasterSheet() {
+ return null;
+ }
+
+ private Map getLayouts(){
if(_layouts == null){
_layouts = new HashMap();
for (POIXMLDocumentPart p : getRelations()) {
@@ -88,14 +95,36 @@ import java.util.Map;
}
}
}
- return _layouts.get(name);
+ return _layouts;
}
+ /**
+ *
+ * @return all slide layouts referencing this master
+ */
+ public XSLFSlideLayout[] getSlideLayouts() {
+ return getLayouts().values().toArray(new XSLFSlideLayout[_layouts.size()]);
+ }
+
+ public XSLFSlideLayout getLayout(SlideLayout type){
+ for(XSLFSlideLayout layout : getLayouts().values()){
+ if(layout.getType() == type) {
+ return layout;
+ }
+ }
+ return null;
+ }
+
+ @Override
public XSLFTheme getTheme(){
if(_theme == null){
for (POIXMLDocumentPart p : getRelations()) {
if (p instanceof XSLFTheme){
_theme = (XSLFTheme)p;
+ CTColorMapping cmap = _slide.getClrMap();
+ if(cmap != null){
+ _theme.initColorMap(cmap);
+ }
break;
}
}
@@ -122,11 +151,20 @@ import java.util.Map;
return props;
}
+ /**
+ * Render this sheet into the supplied graphics object
+ *
+ */
@Override
- public XSLFBackground getBackground(){
- if(_slide.getCSld().isSetBg()) {
- return new XSLFBackground(_slide.getCSld().getBg(), this);
+ protected boolean canDraw(XSLFShape shape){
+ if(shape instanceof XSLFSimpleShape){
+ XSLFSimpleShape txt = (XSLFSimpleShape)shape;
+ CTPlaceholder ph = txt.getCTPlaceholder();
+ if(ph != null) {
+ return false;
+ }
}
- return null;
+ return true;
}
+
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTableCell.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTableCell.java
index 335e77d593..f3d4400aa0 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTableCell.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTableCell.java
@@ -35,7 +35,7 @@ import org.openxmlformats.schemas.drawingml.x2006.main.STLineEndWidth;
import org.openxmlformats.schemas.drawingml.x2006.main.STPenAlignment;
import org.openxmlformats.schemas.drawingml.x2006.main.STPresetLineDashVal;
-import java.awt.*;
+import java.awt.Color;
/**
* Represents a cell of a table in a .pptx presentation
@@ -77,7 +77,7 @@ public class XSLFTableCell extends XSLFTextShape {
}
@Override
- public void setMarginLeft(double margin){
+ public void setLeftInset(double margin){
CTTableCellProperties pr = getXmlObject().getTcPr();
if(pr == null) pr = getXmlObject().addNewTcPr();
@@ -85,7 +85,7 @@ public class XSLFTableCell extends XSLFTextShape {
}
@Override
- public void setMarginRight(double margin){
+ public void setRightInset(double margin){
CTTableCellProperties pr = getXmlObject().getTcPr();
if(pr == null) pr = getXmlObject().addNewTcPr();
@@ -93,7 +93,7 @@ public class XSLFTableCell extends XSLFTextShape {
}
@Override
- public void setMarginTop(double margin){
+ public void setTopInset(double margin){
CTTableCellProperties pr = getXmlObject().getTcPr();
if(pr == null) pr = getXmlObject().addNewTcPr();
@@ -101,7 +101,7 @@ public class XSLFTableCell extends XSLFTextShape {
}
@Override
- public void setMarginBottom(double margin){
+ public void setBottomInset(double margin){
CTTableCellProperties pr = getXmlObject().getTcPr();
if(pr == null) pr = getXmlObject().addNewTcPr();
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextBox.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextBox.java
index 177e8099eb..ef523b4fc8 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextBox.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextBox.java
@@ -63,23 +63,4 @@ public class XSLFTextBox extends XSLFAutoShape {
return ct;
}
-
- /**
- * Specifies that the corresponding shape should be represented by the generating application
- * as a placeholder. When a shape is considered a placeholder by the generating application
- * it can have special properties to alert the user that they may enter content into the shape.
- * Different types of placeholders are allowed and can be specified by using the placeholder
- * type attribute for this element
- *
- * @param placeholder
- */
- public void setPlaceholder(Placeholder placeholder){
- CTShape sh = (CTShape)getXmlObject();
- CTApplicationNonVisualDrawingProps nv = sh.getNvSpPr().getNvPr();
- if(placeholder == null) {
- if(nv.isSetPh()) nv.unsetPh();
- } else {
- nv.addNewPh().setType(STPlaceholderType.Enum.forInt(placeholder.ordinal() + 1));
- }
- }
}
\ No newline at end of file
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextParagraph.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextParagraph.java
index a67dc0d34a..7134246bbb 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextParagraph.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextParagraph.java
@@ -30,7 +30,8 @@ import org.openxmlformats.schemas.drawingml.x2006.main.STTextAlignType;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
import org.openxmlformats.schemas.presentationml.x2006.main.STPlaceholderType;
-import java.awt.*;
+import java.awt.Color;
+import java.awt.Graphics2D;
import java.awt.font.LineBreakMeasurer;
import java.awt.font.TextAttribute;
import java.awt.font.TextLayout;
@@ -80,6 +81,23 @@ public class XSLFTextParagraph implements Iterable{
return out.toString();
}
+ private String getVisibleText(){
+ StringBuilder out = new StringBuilder();
+ for (XSLFTextRun r : _runs) {
+ String txt = r.getText();
+ switch (r.getTextCap()){
+ case ALL:
+ txt = txt.toUpperCase();
+ break;
+ case SMALL:
+ txt = txt.toLowerCase();
+ break;
+ }
+ out.append(txt);
+ }
+ return out.toString();
+ }
+
@Internal
public CTTextParagraph getXmlObject(){
return _p;
@@ -89,6 +107,7 @@ public class XSLFTextParagraph implements Iterable{
return _shape;
}
+
public List getTextRuns(){
return _runs;
}
@@ -186,7 +205,7 @@ public class XSLFTextParagraph implements Iterable{
ParagraphPropertyFetcher fetcher = new ParagraphPropertyFetcher(getLevel()){
public boolean fetch(CTTextParagraphProperties props){
if(props.isSetBuClr()){
- XSLFColor c = new XSLFColor(props.getBuClr(), theme);
+ XSLFColor c = new XSLFColor(props.getBuClr(), theme, null);
setValue(c.getColor());
return true;
}
@@ -496,17 +515,18 @@ public class XSLFTextParagraph implements Iterable{
return "[" + getClass() + "]" + getText();
}
- public List getTextLines(){
+ List getTextLines(){
return _lines;
}
public double draw(Graphics2D graphics, double x, double y){
- double marginLeft = _shape.getMarginLeft();
- double marginRight = _shape.getMarginRight();
+ double marginLeft = _shape.getLeftInset();
+ double marginRight = _shape.getRightInset();
Rectangle2D anchor = _shape.getAnchor();
double penY = y;
double textOffset = getLeftMargin();
+ boolean firstLine = true;
for(TextFragment line : _lines){
double penX = x;
switch (getTextAlign()) {
@@ -521,7 +541,7 @@ public class XSLFTextParagraph implements Iterable{
break;
}
- if(_bullet != null){
+ if(_bullet != null && firstLine){
_bullet.draw(graphics, penX + getIndent(), penY);
}
line.draw(graphics, penX, penY);
@@ -535,6 +555,8 @@ public class XSLFTextParagraph implements Iterable{
// positive value means absolute spacing in points
penY += -spacing;
}
+
+ firstLine = false;
}
return penY - y;
}
@@ -551,8 +573,12 @@ public class XSLFTextParagraph implements Iterable{
void draw(Graphics2D graphics, double x, double y){
double yBaseline = y + _layout.getAscent();
- graphics.drawString(_str.getIterator(), (float)x, (float)yBaseline );
-
+ Integer textMode = (Integer)graphics.getRenderingHint(XSLFRenderingHint.TEXT_RENDERING_MODE);
+ if(textMode != null && textMode == XSLFRenderingHint.TEXT_MODE_GLYPHS){
+ _layout.draw(graphics, (float)x, (float)yBaseline);
+ } else {
+ graphics.drawString(_str.getIterator(), (float)x, (float)yBaseline );
+ }
}
public float getHeight(){
@@ -564,8 +590,8 @@ public class XSLFTextParagraph implements Iterable{
}
- public AttributedString getAttributedString(){
- String text = getText();
+ AttributedString getAttributedString(Graphics2D graphics){
+ String text = getVisibleText();
AttributedString string = new AttributedString(text);
@@ -579,7 +605,10 @@ public class XSLFTextParagraph implements Iterable{
int endIndex = startIndex + length;
string.addAttribute(TextAttribute.FOREGROUND, run.getFontColor(), startIndex, endIndex);
+
+ // user can pass an object to convert fonts via a rendering hint
string.addAttribute(TextAttribute.FAMILY, run.getFontFamily(), startIndex, endIndex);
+
string.addAttribute(TextAttribute.SIZE, (float)run.getFontSize(), startIndex, endIndex);
if(run.isBold()) {
string.addAttribute(TextAttribute.WEIGHT, TextAttribute.WEIGHT_BOLD, startIndex, endIndex);
@@ -601,6 +630,7 @@ public class XSLFTextParagraph implements Iterable{
string.addAttribute(TextAttribute.SUPERSCRIPT, TextAttribute.SUPERSCRIPT_SUPER, startIndex, endIndex);
}
+
startIndex = endIndex;
}
@@ -610,7 +640,7 @@ public class XSLFTextParagraph implements Iterable{
void breakText(Graphics2D graphics){
_lines = new ArrayList();
- AttributedString at = getAttributedString();
+ AttributedString at = getAttributedString(graphics);
AttributedCharacterIterator it = at.getIterator();
if(it.getBeginIndex() == it.getEndIndex()) {
return;
@@ -628,7 +658,8 @@ public class XSLFTextParagraph implements Iterable{
int endIndex = measurer.getPosition();
- if(getTextAlign() == TextAlign.JUSTIFY) {
+ TextAlign hAlign = getTextAlign();
+ if(hAlign == TextAlign.JUSTIFY || hAlign == TextAlign.JUSTIFY_LOW) {
layout = layout.getJustifiedLayout((float)wrappingWidth);
}
@@ -673,7 +704,7 @@ public class XSLFTextParagraph implements Iterable{
width = _shape.getSheet().getSlideShow().getPageSize().getWidth();
} else {
width = _shape.getAnchor().getWidth() -
- _shape.getMarginLeft() - _shape.getMarginRight() - getLeftMargin();
+ _shape.getLeftInset() - _shape.getRightInset() - getLeftMargin();
}
return width;
}
@@ -700,7 +731,13 @@ public class XSLFTextParagraph implements Iterable{
}
int level = getLevel();
- XmlObject[] o = _shape.getSheet().getSlideMaster().getXmlObject().selectPath(
+
+ XSLFSheet masterSheet = _shape.getSheet();
+ while (masterSheet.getMasterSheet() != null){
+ masterSheet = masterSheet.getMasterSheet();
+ }
+
+ XmlObject[] o = masterSheet.getXmlObject().selectPath(
"declare namespace p='http://schemas.openxmlformats.org/presentationml/2006/main' " +
"declare namespace a='http://schemas.openxmlformats.org/drawingml/2006/main' " +
".//p:txStyles/p:" + defaultStyleSelector +"/a:lvl" +(level+1)+ "pPr");
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextRun.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextRun.java
index e4ebf9db58..b4aa3ca8bf 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextRun.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextRun.java
@@ -20,14 +20,17 @@ import org.apache.poi.util.Beta;
import org.apache.poi.xslf.model.CharacterPropertyFetcher;
import org.openxmlformats.schemas.drawingml.x2006.main.CTRegularTextRun;
import org.openxmlformats.schemas.drawingml.x2006.main.CTSRgbColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTSchemeColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTShapeStyle;
import org.openxmlformats.schemas.drawingml.x2006.main.CTSolidColorFillProperties;
import org.openxmlformats.schemas.drawingml.x2006.main.CTTextCharacterProperties;
import org.openxmlformats.schemas.drawingml.x2006.main.CTTextFont;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTTextNormalAutofit;
import org.openxmlformats.schemas.drawingml.x2006.main.CTTextParagraphProperties;
import org.openxmlformats.schemas.drawingml.x2006.main.STTextStrikeType;
import org.openxmlformats.schemas.drawingml.x2006.main.STTextUnderlineType;
-import java.awt.*;
+import java.awt.Color;
/**
* Represents a run of text within the containing text body. The run element is the
@@ -70,12 +73,14 @@ public class XSLFTextRun {
public Color getFontColor(){
final XSLFTheme theme = _p.getParentShape().getSheet().getTheme();
+ CTShapeStyle style = _p.getParentShape().getSpStyle();
+ final CTSchemeColor shapeStyle = style == null ? null : style.getFontRef().getSchemeClr();
CharacterPropertyFetcher fetcher = new CharacterPropertyFetcher(_p.getLevel()){
public boolean fetch(CTTextCharacterProperties props){
CTSolidColorFillProperties solidFill = props.getSolidFill();
- if(solidFill != null){
- Color c = new XSLFColor(solidFill, theme).getColor();
+ if(solidFill != null) {
+ Color c = new XSLFColor(solidFill, theme, shapeStyle).getColor();
setValue(c);
return true;
}
@@ -104,6 +109,10 @@ public class XSLFTextRun {
* @return font size in points or -1 if font size is not set.
*/
public double getFontSize(){
+ double scale = 1;
+ CTTextNormalAutofit afit = getParentParagraph().getParentShape().getTextBodyPr().getNormAutofit();
+ if(afit != null) scale = (double)afit.getFontScale() / 100000;
+
CharacterPropertyFetcher fetcher = new CharacterPropertyFetcher(_p.getLevel()){
public boolean fetch(CTTextCharacterProperties props){
if(props.isSetSz()){
@@ -114,7 +123,27 @@ public class XSLFTextRun {
}
};
fetchCharacterProperty(fetcher);
- return fetcher.getValue() == null ? -1 : fetcher.getValue();
+ return fetcher.getValue() == null ? -1 : fetcher.getValue()*scale;
+ }
+
+ /**
+ *
+ * @return the spacing between characters within a text run,
+ * If this attribute is omitted than a value of 0 or no adjustment is assumed.
+ */
+ public double getCharacterSpacing(){
+
+ CharacterPropertyFetcher fetcher = new CharacterPropertyFetcher(_p.getLevel()){
+ public boolean fetch(CTTextCharacterProperties props){
+ if(props.isSetSpc()){
+ setValue(props.getSpc()*0.01);
+ return true;
+ }
+ return false;
+ }
+ };
+ fetchCharacterProperty(fetcher);
+ return fetcher.getValue() == null ? 0 : fetcher.getValue();
}
/**
@@ -234,6 +263,24 @@ public class XSLFTextRun {
return fetcher.getValue() == null ? false : fetcher.getValue();
}
+ /**
+ * @return whether a run of text will be formatted as a superscript text. Default is false.
+ */
+ public TextCap getTextCap() {
+ CharacterPropertyFetcher fetcher = new CharacterPropertyFetcher(_p.getLevel()){
+ public boolean fetch(CTTextCharacterProperties props){
+ if(props.isSetCap()){
+ int idx = props.getCap().intValue() - 1;
+ setValue(TextCap.values()[idx]);
+ return true;
+ }
+ return false;
+ }
+ };
+ fetchCharacterProperty(fetcher);
+ return fetcher.getValue() == null ? TextCap.NONE : fetcher.getValue();
+ }
+
/**
* Specifies whether this run of text will be formatted as bold text
*
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextShape.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextShape.java
index 279cac9198..dd49c316a0 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextShape.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTextShape.java
@@ -30,12 +30,16 @@ import org.openxmlformats.schemas.drawingml.x2006.main.CTTextParagraph;
import org.openxmlformats.schemas.drawingml.x2006.main.STTextAnchoringType;
import org.openxmlformats.schemas.drawingml.x2006.main.STTextVerticalType;
import org.openxmlformats.schemas.drawingml.x2006.main.STTextWrappingType;
+import org.openxmlformats.schemas.presentationml.x2006.main.CTApplicationNonVisualDrawingProps;
import org.openxmlformats.schemas.presentationml.x2006.main.CTPlaceholder;
+import org.openxmlformats.schemas.presentationml.x2006.main.CTShape;
+import org.openxmlformats.schemas.presentationml.x2006.main.STPlaceholderType;
-import java.awt.*;
+import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.util.ArrayList;
+import java.util.Iterator;
import java.util.List;
/**
@@ -44,7 +48,7 @@ import java.util.List;
* @author Yegor Kozlov
*/
@Beta
-public abstract class XSLFTextShape extends XSLFSimpleShape {
+public abstract class XSLFTextShape extends XSLFSimpleShape implements Iterable{
private final List _paragraphs;
/*package*/ XSLFTextShape(XmlObject shape, XSLFSheet sheet) {
@@ -59,7 +63,14 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
}
}
- // textual properties
+ public Iterator iterator(){
+ return _paragraphs.iterator();
+ }
+
+ /**
+ *
+ * @return text contained within this shape or empty string
+ */
public String getText() {
StringBuilder out = new StringBuilder();
for (XSLFTextParagraph p : _paragraphs) {
@@ -69,10 +80,34 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
return out.toString();
}
+ /**
+ * unset text from this shape
+ */
+ public void clearText(){
+ _paragraphs.clear();
+ CTTextBody txBody = getTextBody(true);
+ txBody.setPArray(null); // remove any existing paragraphs
+ }
+
+ public void setText(String text){
+ clearText();
+
+ addNewTextParagraph().addNewTextRun().setText(text);
+ }
+
+ /**
+ *
+ * @return text paragraphs in this shape
+ */
public List getTextParagraphs() {
return _paragraphs;
}
+ /**
+ * add a new paragraph run to this shape
+ *
+ * @return created paragraph run
+ */
public XSLFTextParagraph addNewTextParagraph() {
CTTextBody txBody = getTextBody(true);
CTTextParagraph p = txBody.addNewP();
@@ -84,9 +119,9 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Sets the type of vertical alignment for the text.
- * One of the Anchor* constants defined in this class.
*
- * @param anchor - the type of alignment. Default is {@link org.apache.poi.xslf.usermodel.VerticalAlignment#TOP}
+ * @param anchor - the type of alignment.
+ * A null values unsets this property.
*/
public void setVerticalAlignment(VerticalAlignment anchor){
CTTextBodyProperties bodyPr = getTextBodyPr();
@@ -102,7 +137,7 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Returns the type of vertical alignment for the text.
*
- * @return the type of alignment
+ * @return the type of vertical alignment
*/
public VerticalAlignment getVerticalAlignment(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
@@ -147,13 +182,15 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
}
return TextDirection.HORIZONTAL;
}
+
+
/**
* Returns the distance (in points) between the bottom of the text frame
* and the bottom of the inscribed rectangle of the shape that contains the text.
*
- * @return the bottom margin or -1 if not set
+ * @return the bottom inset in points
*/
- public double getMarginBottom(){
+ public double getBottomInset(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
public boolean fetch(CTTextBodyProperties props){
if(props.isSetBIns()){
@@ -173,9 +210,9 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
* and the left edge of the inscribed rectangle of the shape that contains
* the text.
*
- * @return the left margin
+ * @return the left inset in points
*/
- public double getMarginLeft(){
+ public double getLeftInset(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
public boolean fetch(CTTextBodyProperties props){
if(props.isSetLIns()){
@@ -195,9 +232,9 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
* text frame and the right edge of the inscribed rectangle of the shape
* that contains the text.
*
- * @return the right margin
+ * @return the right inset in points
*/
- public double getMarginRight(){
+ public double getRightInset(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
public boolean fetch(CTTextBodyProperties props){
if(props.isSetRIns()){
@@ -216,9 +253,9 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
* Returns the distance (in points) between the top of the text frame
* and the top of the inscribed rectangle of the shape that contains the text.
*
- * @return the top margin
+ * @return the top inset in points
*/
- public double getMarginTop(){
+ public double getTopInset(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
public boolean fetch(CTTextBodyProperties props){
if(props.isSetTIns()){
@@ -235,11 +272,11 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Sets the botom margin.
- * @see #getMarginBottom()
+ * @see #getBottomInset()
*
* @param margin the bottom margin
*/
- public void setMarginBottom(double margin){
+ public void setBottomInset(double margin){
CTTextBodyProperties bodyPr = getTextBodyPr();
if (bodyPr != null) {
if(margin == -1) bodyPr.unsetBIns();
@@ -249,11 +286,11 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Sets the left margin.
- * @see #getMarginLeft()
+ * @see #getLeftInset()
*
* @param margin the left margin
*/
- public void setMarginLeft(double margin){
+ public void setLeftInset(double margin){
CTTextBodyProperties bodyPr = getTextBodyPr();
if (bodyPr != null) {
if(margin == -1) bodyPr.unsetLIns();
@@ -263,11 +300,11 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Sets the right margin.
- * @see #getMarginRight()
+ * @see #getRightInset()
*
* @param margin the right margin
*/
- public void setMarginRight(double margin){
+ public void setRightInset(double margin){
CTTextBodyProperties bodyPr = getTextBodyPr();
if (bodyPr != null) {
if(margin == -1) bodyPr.unsetRIns();
@@ -277,11 +314,11 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
* Sets the top margin.
- * @see #getMarginTop()
+ * @see #getTopInset()
*
* @param margin the top margin
*/
- public void setMarginTop(double margin){
+ public void setTopInset(double margin){
CTTextBodyProperties bodyPr = getTextBodyPr();
if (bodyPr != null) {
if(margin == -1) bodyPr.unsetTIns();
@@ -291,10 +328,7 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
/**
- * Returns the value indicating word wrap.
- * One of the Wrap* constants defined in this class.
- *
- * @return the value indicating word wrap
+ * @return whether to wrap words within the bounding rectangle
*/
public boolean getWordWrap(){
PropertyFetcher fetcher = new TextBodyPropertyFetcher(){
@@ -311,9 +345,8 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
}
/**
- * Specifies how the text should be wrapped
*
- * @param wrap the value indicating how the text should be wrapped
+ * @param wrap whether to wrap words within the bounding rectangle
*/
public void setWordWrap(boolean wrap){
CTTextBodyProperties bodyPr = getTextBodyPr();
@@ -381,33 +414,25 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
}
}
- @Override
- public void draw(Graphics2D graphics){
- java.awt.Shape outline = getOutline();
- // shadow
- XSLFShadow shadow = getShadow();
-
- Paint fill = getFill(graphics);
- Paint line = getLinePaint(graphics);
- if(shadow != null) {
- shadow.draw(graphics);
+ /**
+ * Specifies that the corresponding shape should be represented by the generating application
+ * as a placeholder. When a shape is considered a placeholder by the generating application
+ * it can have special properties to alert the user that they may enter content into the shape.
+ * Different types of placeholders are allowed and can be specified by using the placeholder
+ * type attribute for this element
+ *
+ * @param placeholder
+ */
+ public void setPlaceholder(Placeholder placeholder){
+ CTShape sh = (CTShape)getXmlObject();
+ CTApplicationNonVisualDrawingProps nv = sh.getNvSpPr().getNvPr();
+ if(placeholder == null) {
+ if(nv.isSetPh()) nv.unsetPh();
+ } else {
+ nv.addNewPh().setType(STPlaceholderType.Enum.forInt(placeholder.ordinal() + 1));
}
-
- if(fill != null) {
- graphics.setPaint(fill);
- graphics.fill(outline);
- }
-
- if (line != null){
- graphics.setPaint(line);
- applyStroke(graphics);
- graphics.draw(outline);
- }
-
- // text
- if(getText().length() > 0) drawText(graphics);
- }
+ }
/**
* Compute the cumulative height occupied by the text
@@ -418,15 +443,19 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
return drawParagraphs(img.createGraphics(), 0, 0);
}
- void breakText(Graphics2D graphics){
+ /**
+ * break the contained text into lines
+ */
+ private void breakText(Graphics2D graphics){
for(XSLFTextParagraph p : _paragraphs) p.breakText(graphics);
}
- public void drawText(Graphics2D graphics) {
+ @Override
+ public void drawContent(Graphics2D graphics) {
breakText(graphics);
Rectangle2D anchor = getAnchor();
- double x = anchor.getX() + getMarginLeft();
+ double x = anchor.getX() + getLeftInset();
double y = anchor.getY();
// first dry-run to calculate the total height of the text
@@ -434,21 +463,20 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
switch (getVerticalAlignment()){
case TOP:
- y += getMarginTop();
+ y += getTopInset();
break;
case BOTTOM:
- y += anchor.getHeight() - textHeight - getMarginBottom();
+ y += anchor.getHeight() - textHeight - getBottomInset();
break;
default:
case MIDDLE:
double delta = anchor.getHeight() - textHeight -
- getMarginTop() - getMarginBottom();
- y += getMarginTop() + delta/2;
+ getTopInset() - getBottomInset();
+ y += getTopInset() + delta/2;
break;
}
drawParagraphs(graphics, x, y);
-
}
@@ -461,7 +489,7 @@ public abstract class XSLFTextShape extends XSLFSimpleShape {
double y0 = y;
for(int i = 0; i < _paragraphs.size(); i++){
XSLFTextParagraph p = _paragraphs.get(i);
- java.util.List lines = p.getTextLines();
+ List lines = p.getTextLines();
if(i > 0 && lines.size() > 0) {
// the amount of vertical white space before the paragraph
diff --git a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTheme.java b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTheme.java
index 77dd0a8564..b996dd5fb4 100644
--- a/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTheme.java
+++ b/src/ooxml/java/org/apache/poi/xslf/usermodel/XSLFTheme.java
@@ -26,6 +26,7 @@ import org.apache.xmlbeans.XmlObject;
import org.apache.xmlbeans.XmlOptions;
import org.openxmlformats.schemas.drawingml.x2006.main.CTBaseStyles;
import org.openxmlformats.schemas.drawingml.x2006.main.CTColor;
+import org.openxmlformats.schemas.drawingml.x2006.main.CTColorMapping;
import org.openxmlformats.schemas.drawingml.x2006.main.CTColorScheme;
import org.openxmlformats.schemas.drawingml.x2006.main.CTOfficeStyleSheet;
import org.openxmlformats.schemas.drawingml.x2006.main.ThemeDocument;
@@ -36,6 +37,11 @@ import java.io.OutputStream;
import java.util.HashMap;
import java.util.Map;
+/**
+ * A shared style sheet in a .pptx slide show
+ *
+ * @author Yegor Kozlov
+ */
@Beta
public class XSLFTheme extends POIXMLDocumentPart {
private CTOfficeStyleSheet _theme;
@@ -64,17 +70,33 @@ public class XSLFTheme extends POIXMLDocumentPart {
String name = c.getDomNode().getLocalName();
_schemeColors.put(name, c);
}
+ }
- _schemeColors.put("bg1", _schemeColors.get("lt1"));
- _schemeColors.put("bg2", _schemeColors.get("lt2"));
- _schemeColors.put("tx1", _schemeColors.get("dk1"));
- _schemeColors.put("tx2", _schemeColors.get("dk2"));
+ /**
+ * re-map colors
+ *
+ * @param cmap color map defined in the master slide referencing this theme
+ */
+ void initColorMap(CTColorMapping cmap) {
+ _schemeColors.put("bg1", _schemeColors.get(cmap.getBg1().toString()));
+ _schemeColors.put("bg2", _schemeColors.get(cmap.getBg2().toString()));
+ _schemeColors.put("tx1", _schemeColors.get(cmap.getTx1().toString()));
+ _schemeColors.put("tx2", _schemeColors.get(cmap.getTx2().toString()));
}
+ /**
+ *
+ * @return name of this theme, e.g. "Office Theme"
+ */
public String getName(){
return _theme.getName();
}
+ /**
+ * Set name of this theme
+ *
+ * @param name name of this theme
+ */
public void setName(String name){
_theme.setName(name);
}
@@ -111,10 +133,20 @@ public class XSLFTheme extends POIXMLDocumentPart {
out.close();
}
+ /**
+ * @return typeface of the major font to use in a document.
+ * Typically the major font is used for heading areas of a document.
+ *
+ */
public String getMajorFont(){
return _theme.getThemeElements().getFontScheme().getMajorFont().getLatin().getTypeface();
}
+ /**
+ * @return typeface of the minor font to use in a document.
+ * Typically the monor font is used for normal text or paragraph areas.
+ *
+ */
public String getMinorFont(){
return _theme.getThemeElements().getFontScheme().getMinorFont().getLatin().getTypeface();
}
diff --git a/src/ooxml/java/org/apache/poi/xslf/util/PPTX2PNG.java b/src/ooxml/java/org/apache/poi/xslf/util/PPTX2PNG.java
index f4e6955e6a..76bb08559f 100644
--- a/src/ooxml/java/org/apache/poi/xslf/util/PPTX2PNG.java
+++ b/src/ooxml/java/org/apache/poi/xslf/util/PPTX2PNG.java
@@ -28,21 +28,26 @@ import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
-import java.awt.geom.AffineTransform;
-import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
-import java.io.File;
import java.io.FileOutputStream;
/**
- * Date: 10/11/11
+ * An utulity to convert slides of a .pptx slide show to a PNG image
*
* @author Yegor Kozlov
*/
public class PPTX2PNG {
+
+ static void usage(){
+ System.out.println("Usage: PPTX2PNG [options] ");
+ System.out.println("Options:");
+ System.out.println(" -scale scale factor");
+ System.out.println(" -slide 1-based index of a slide to render");
+ }
+
public static void main(String[] args) throws Exception {
if (args.length == 0) {
- System.out.println("Usage: PPTX2PNG [options] ");
+ usage();
return;
}
@@ -62,6 +67,11 @@ public class PPTX2PNG {
}
}
+ if(file == null){
+ usage();
+ return;
+ }
+
System.out.println("Processing " + file);
XMLSlideShow ppt = new XMLSlideShow(OPCPackage.open(file));
@@ -79,19 +89,23 @@ public class PPTX2PNG {
BufferedImage img = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
Graphics2D graphics = img.createGraphics();
+ // default rendering options
graphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
graphics.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
graphics.setRenderingHint(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BICUBIC);
graphics.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
- graphics.setPaint(Color.white);
- graphics.fill(new Rectangle2D.Float(0, 0, width, height));
+ graphics.setColor(Color.white);
+ graphics.clearRect(0, 0, width, height);
- graphics.scale((double) width / pgsize.width, (double) height / pgsize.height);
+ graphics.scale(scale, scale);
+ // draw stuff
slide[i].draw(graphics);
- String fname = file.replaceAll("\\.pptx", "-" + (i + 1) + ".png");
+ // save the result
+ int sep = file.lastIndexOf(".");
+ String fname = file.substring(0, sep == -1 ? file.length() : sep) + "-" + (i + 1) +".png";
FileOutputStream out = new FileOutputStream(fname);
ImageIO.write(img, "png", out);
out.close();
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/geom/TestFormulaParser.java b/src/ooxml/testcases/org/apache/poi/xslf/geom/TestFormulaParser.java
index b9f15139d5..df3e2ec29d 100644
--- a/src/ooxml/testcases/org/apache/poi/xslf/geom/TestFormulaParser.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/geom/TestFormulaParser.java
@@ -24,7 +24,7 @@ public class TestFormulaParser extends TestCase {
};
CustomGeometry geom = new CustomGeometry(CTCustomGeometry2D.Factory.newInstance());
- Context ctx = new Context(geom, null);
+ Context ctx = new Context(geom, null, null);
for(Formula fmla : ops) {
ctx.evaluate(fmla);
}
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/geom/TestPresetGeometries.java b/src/ooxml/testcases/org/apache/poi/xslf/geom/TestPresetGeometries.java
index 5c19b4e53e..c71a065bb7 100644
--- a/src/ooxml/testcases/org/apache/poi/xslf/geom/TestPresetGeometries.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/geom/TestPresetGeometries.java
@@ -21,11 +21,7 @@ public class TestPresetGeometries extends TestCase {
for(String name : shapes.keySet()) {
CustomGeometry geom = shapes.get(name);
- Context ctx = new Context(geom, new IAdjustableShape() {
- public Rectangle2D getAnchor() {
- return new Rectangle2D.Double(0, 0, 100, 100);
- }
-
+ Context ctx = new Context(geom, new Rectangle2D.Double(0, 0, 100, 100), new IAdjustableShape() {
public Guide getAdjustValue(String name) {
return null;
}
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestPPTX2PNG.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestPPTX2PNG.java
index 1f8826fdad..948b792b8b 100644
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestPPTX2PNG.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestPPTX2PNG.java
@@ -34,7 +34,7 @@ import java.awt.image.BufferedImage;
public class TestPPTX2PNG extends TestCase {
public void testRender(){
String[] testFiles = {"layouts.pptx", "sample.pptx", "shapes.pptx",
- "45541_Header.pptx", "backgrounds.pptx"};
+ "themes.pptx", "backgrounds.pptx"};
for(String sampleFile : testFiles){
XMLSlideShow pptx = XSLFTestDataSamples.openSampleDocument(sampleFile);
Dimension pg = pptx.getPageSize();
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFAutoShape.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFAutoShape.java
index 7cab8ea4e3..d29cb30662 100755
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFAutoShape.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFAutoShape.java
@@ -34,38 +34,38 @@ public class TestXSLFAutoShape extends TestCase {
shape.addNewTextParagraph().addNewTextRun().setText("POI");
// default margins from slide master
- assertEquals(3.6, shape.getMarginBottom());
- assertEquals(3.6, shape.getMarginTop());
- assertEquals(7.2, shape.getMarginLeft());
- assertEquals(7.2, shape.getMarginRight());
+ assertEquals(3.6, shape.getBottomInset());
+ assertEquals(3.6, shape.getTopInset());
+ assertEquals(7.2, shape.getLeftInset());
+ assertEquals(7.2, shape.getRightInset());
- shape.setMarginBottom(1.0);
- assertEquals(1.0, shape.getMarginBottom());
- shape.setMarginTop(2.0);
- assertEquals(2.0, shape.getMarginTop());
- shape.setMarginLeft(3.0);
- assertEquals(3.0, shape.getMarginLeft());
- shape.setMarginRight(4.0);
- assertEquals(4.0, shape.getMarginRight());
+ shape.setBottomInset(1.0);
+ assertEquals(1.0, shape.getBottomInset());
+ shape.setTopInset(2.0);
+ assertEquals(2.0, shape.getTopInset());
+ shape.setLeftInset(3.0);
+ assertEquals(3.0, shape.getLeftInset());
+ shape.setRightInset(4.0);
+ assertEquals(4.0, shape.getRightInset());
- shape.setMarginBottom(0.0);
- assertEquals(0.0, shape.getMarginBottom());
- shape.setMarginTop(0.0);
- assertEquals(0.0, shape.getMarginTop());
- shape.setMarginLeft(0.0);
- assertEquals(0.0, shape.getMarginLeft());
- shape.setMarginRight(0.0);
- assertEquals(0.0, shape.getMarginRight());
+ shape.setBottomInset(0.0);
+ assertEquals(0.0, shape.getBottomInset());
+ shape.setTopInset(0.0);
+ assertEquals(0.0, shape.getTopInset());
+ shape.setLeftInset(0.0);
+ assertEquals(0.0, shape.getLeftInset());
+ shape.setRightInset(0.0);
+ assertEquals(0.0, shape.getRightInset());
// unset to defauls
- shape.setMarginBottom(-1);
- assertEquals(3.6, shape.getMarginBottom());
- shape.setMarginTop(-1);
- assertEquals(3.6, shape.getMarginTop());
- shape.setMarginLeft(-1);
- assertEquals(7.2, shape.getMarginLeft());
- shape.setMarginRight(-1);
- assertEquals(7.2, shape.getMarginRight());
+ shape.setBottomInset(-1);
+ assertEquals(3.6, shape.getBottomInset());
+ shape.setTopInset(-1);
+ assertEquals(3.6, shape.getTopInset());
+ shape.setLeftInset(-1);
+ assertEquals(7.2, shape.getLeftInset());
+ shape.setRightInset(-1);
+ assertEquals(7.2, shape.getRightInset());
// shape
assertTrue(shape.getWordWrap());
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFColor.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFColor.java
index ffb25b37e1..108fca327b 100755
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFColor.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFColor.java
@@ -37,7 +37,7 @@ public class TestXSLFColor extends TestCase {
CTSRgbColor c = xml.addNewSrgbClr();
c.setVal(new byte[]{(byte)0xFF, 0, 0});
- XSLFColor color = new XSLFColor(xml, null);
+ XSLFColor color = new XSLFColor(xml, null, null);
assertEquals(-1, color.getAlpha());
c.addNewAlpha().setVal(50000);
@@ -99,7 +99,7 @@ public class TestXSLFColor extends TestCase {
c.setSat2(100000);
c.setLum2(50000);
- XSLFColor color = new XSLFColor(xml, null);
+ XSLFColor color = new XSLFColor(xml, null, null);
assertEquals(new Color(128, 00, 00), color.getColor());
}
@@ -107,7 +107,7 @@ public class TestXSLFColor extends TestCase {
CTColor xml = CTColor.Factory.newInstance();
xml.addNewSrgbClr().setVal(new byte[]{ (byte)0xFF, (byte)0xFF, 0});
- XSLFColor color = new XSLFColor(xml, null);
+ XSLFColor color = new XSLFColor(xml, null, null);
assertEquals(new Color(0xFF, 0xFF, 0), color.getColor());
}
@@ -118,19 +118,19 @@ public class TestXSLFColor extends TestCase {
CTColor xml = CTColor.Factory.newInstance();
xml.addNewSchemeClr().setVal(STSchemeColorVal.ACCENT_2);
- XSLFColor color = new XSLFColor(xml, theme);
+ XSLFColor color = new XSLFColor(xml, theme, null);
// accent2 is theme1.xml is
assertEquals(Color.decode("0xC0504D"), color.getColor());
xml = CTColor.Factory.newInstance();
xml.addNewSchemeClr().setVal(STSchemeColorVal.LT_1);
- color = new XSLFColor(xml, theme);
+ color = new XSLFColor(xml, theme, null);
//
assertEquals(Color.decode("0xFFFFFF"), color.getColor());
xml = CTColor.Factory.newInstance();
xml.addNewSchemeClr().setVal(STSchemeColorVal.DK_1);
- color = new XSLFColor(xml, theme);
+ color = new XSLFColor(xml, theme, null);
//
assertEquals(Color.decode("0x000000"), color.getColor());
}
@@ -138,7 +138,7 @@ public class TestXSLFColor extends TestCase {
public void testPresetColor() {
CTColor xml = CTColor.Factory.newInstance();
xml.addNewPrstClr().setVal(STPresetColorVal.AQUAMARINE);
- XSLFColor color = new XSLFColor(xml, null);
+ XSLFColor color = new XSLFColor(xml, null, null);
assertEquals(new Color(127, 255, 212), color.getColor());
@@ -147,7 +147,7 @@ public class TestXSLFColor extends TestCase {
STPresetColorVal.Enum val = STPresetColorVal.Enum.forString(colorName);
assertNotNull(colorName, val);
xml.addNewPrstClr().setVal(val);
- color = new XSLFColor(xml, null);
+ color = new XSLFColor(xml, null, null);
assertEquals(XSLFColor.presetColors.get(colorName), color.getColor());
}
}
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlide.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlide.java
index 9ffe69fe98..d8cb5783e4 100755
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlide.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlide.java
@@ -19,9 +19,6 @@ package org.apache.poi.xslf.usermodel;
import junit.framework.TestCase;
import org.apache.poi.xslf.XSLFTestDataSamples;
-import org.apache.poi.POIXMLDocumentPart;
-
-import java.util.List;
/**
* @author Yegor Kozlov
@@ -99,11 +96,11 @@ public class TestXSLFSlide extends TestCase {
assertEquals(0, ppt.getSlides().length);
XSLFSlide slide = ppt.createSlide();
- assertTrue(slide.getFollowMasterBackground());
- slide.setFollowMasterBackground(false);
- assertFalse(slide.getFollowMasterBackground());
- slide.setFollowMasterBackground(true);
- assertTrue(slide.getFollowMasterBackground());
+ assertTrue(slide.getFollowMasterGraphics());
+ slide.setFollowMasterGraphics(false);
+ assertFalse(slide.getFollowMasterGraphics());
+ slide.setFollowMasterGraphics(true);
+ assertTrue(slide.getFollowMasterGraphics());
}
}
\ No newline at end of file
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlideShow.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlideShow.java
index c07493a3f9..46e10e45d8 100755
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlideShow.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFSlideShow.java
@@ -38,7 +38,7 @@ public class TestXSLFSlideShow extends TestCase {
List rels = slide1.getRelations();
assertEquals(1, rels.size());
- assertEquals(slide1.getMasterSheet().getLayout("blank"), rels.get(0));
+ assertEquals(slide1.getSlideMaster().getLayout(SlideLayout.BLANK), rels.get(0));
XSLFSlide slide2 = ppt.createSlide();
assertEquals(2, ppt.getSlides().length);
@@ -91,7 +91,7 @@ public class TestXSLFSlideShow extends TestCase {
assertEquals(1, masters.length);
XSLFSlide slide = ppt.createSlide();
- assertSame(masters[0], slide.getMasterSheet());
+ assertSame(masters[0], slide.getSlideMaster());
}
public void testSlideLayout(){
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTextShape.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTextShape.java
index 7524b25292..6347b63ccf 100644
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTextShape.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTextShape.java
@@ -65,10 +65,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.MIDDLE, shape1.getVerticalAlignment());
// now check text properties
@@ -95,10 +95,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape2.getVerticalAlignment());
assertEquals("subtitle", shape2.getText());
@@ -134,10 +134,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.MIDDLE, shape1.getVerticalAlignment());
// now check text properties
@@ -169,10 +169,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape2.getVerticalAlignment());
XSLFTextRun pr1 = shape2.getTextParagraphs().get(0).getTextRuns().get(0);
@@ -244,10 +244,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape1.getVerticalAlignment());
// now check text properties
@@ -278,10 +278,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.BOTTOM, shape2.getVerticalAlignment());
assertEquals("Section Header", shape2.getText());
@@ -318,10 +318,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.MIDDLE, shape1.getVerticalAlignment());
// now check text properties
@@ -351,10 +351,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape2.getVerticalAlignment());
XSLFTextRun pr1 = shape2.getTextParagraphs().get(0).getTextRuns().get(0);
@@ -425,10 +425,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.MIDDLE, shape1.getVerticalAlignment());
// now check text properties
@@ -449,10 +449,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape2.getVerticalAlignment());
XSLFTextRun pr1 = shape2.getTextParagraphs().get(0).getTextRuns().get(0);
@@ -498,10 +498,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr1.isSetBIns() && !bodyPr1.isSetTIns() &&
!bodyPr1.isSetAnchor()
);
- assertEquals(7.2, shape1.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape1.getMarginRight()); // 0.1"
- assertEquals(3.6, shape1.getMarginTop()); // 0.05"
- assertEquals(3.6, shape1.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape1.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape1.getRightInset()); // 0.1"
+ assertEquals(3.6, shape1.getTopInset()); // 0.05"
+ assertEquals(3.6, shape1.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.BOTTOM, shape1.getVerticalAlignment());
// now check text properties
@@ -532,10 +532,10 @@ public class TestXSLFTextShape extends TestCase {
!bodyPr2.isSetBIns() && !bodyPr2.isSetTIns() &&
!bodyPr2.isSetAnchor()
);
- assertEquals(7.2, shape2.getMarginLeft()); // 0.1"
- assertEquals(7.2, shape2.getMarginRight()); // 0.1"
- assertEquals(3.6, shape2.getMarginTop()); // 0.05"
- assertEquals(3.6, shape2.getMarginBottom()); // 0.05"
+ assertEquals(7.2, shape2.getLeftInset()); // 0.1"
+ assertEquals(7.2, shape2.getRightInset()); // 0.1"
+ assertEquals(3.6, shape2.getTopInset()); // 0.05"
+ assertEquals(3.6, shape2.getBottomInset()); // 0.05"
assertEquals(VerticalAlignment.TOP, shape2.getVerticalAlignment());
XSLFTextRun pr1 = shape2.getTextParagraphs().get(0).getTextRuns().get(0);
diff --git a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTheme.java b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTheme.java
index 81f35d3250..b01c25e1d5 100644
--- a/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTheme.java
+++ b/src/ooxml/testcases/org/apache/poi/xslf/usermodel/TestXSLFTheme.java
@@ -18,22 +18,141 @@ package org.apache.poi.xslf.usermodel;
import junit.framework.TestCase;
+
import org.apache.poi.xslf.XSLFTestDataSamples;
+import java.awt.Color;
+import java.awt.TexturePaint;
+
/**
- * test common properties for sheets (slides, masters, layouts, etc.)
+ * test reading properties from a multi-theme and multi-master document
*
* @author Yegor Kozlov
*/
public class TestXSLFTheme extends TestCase {
public void testRead(){
- XMLSlideShow ppt = new XMLSlideShow();
- XSLFSlide slide = ppt.createSlide();
- XSLFTheme theme = slide.getSlideLayout().getSlideMaster().getTheme();
- assertNotNull(theme);
-
+ XMLSlideShow ppt = XSLFTestDataSamples.openSampleDocument("themes.pptx");
+ XSLFSlide[] slides = ppt.getSlides();
+
+ slide1(slides[0]);
+ slide2(slides[1]);
+ slide3(slides[2]);
+ slide4(slides[3]);
+ slide5(slides[4]);
+ slide6(slides[5]);
+ slide7(slides[6]);
+ slide8(slides[7]);
+ slide9(slides[8]);
+ slide10(slides[9]);
+ }
+
+ private XSLFShape getShape(XSLFSheet sheet, String name){
+ for(XSLFShape sh : sheet.getShapes()){
+ if(sh.getShapeName().equals(name)) return sh;
+ }
+ throw new IllegalArgumentException("Shape not found: " + name);
+ }
+
+ void slide1(XSLFSlide slide){
+ assertEquals(Color.white, slide.getBackground().getFillColor());
+
+ XSLFTheme theme = slide.getTheme();
assertEquals("Office Theme", theme.getName());
- //XSLFColor accent1 = theme.getColor("accent1");
- //assertNotNull(accent1);
+
+ XSLFTextShape sh1 = (XSLFTextShape)getShape(slide, "Rectangle 3");
+ RenderableShape rsh1 = new RenderableShape(sh1);
+ XSLFTextRun run1 = sh1.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(Color.white, run1.getFontColor());
+ assertEquals(new Color(79, 129, 189), sh1.getFillColor());
+ assertTrue(rsh1.getFillPaint(null) instanceof Color) ; // solid fill
+
+ }
+
+ void slide2(XSLFSlide slide){
+ // Background 2, darker 10%
+ // YK: PPT shows slightly different color: new Color(221, 217, 195)
+ assertEquals(new Color(214, 212, 203), slide.getBackground().getFillColor());
+ }
+
+ void slide3(XSLFSlide slide){
+ assertNull(slide.getBackground().getFillColor());
+ assertTrue(slide.getBackground().getPaint(null).getClass().getName().indexOf("Gradient") > 0);
+ }
+
+ void slide4(XSLFSlide slide){
+ assertNull(slide.getBackground().getFillColor());
+ assertTrue(slide.getBackground().getPaint(null).getClass().getName().indexOf("Gradient") > 0);
+
+ XSLFTextShape sh1 = (XSLFTextShape)getShape(slide, "Rectangle 4");
+ RenderableShape rsh1 = new RenderableShape(sh1);
+ XSLFTextRun run1 = sh1.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(Color.white, run1.getFontColor());
+ assertEquals(new Color(148, 198, 0), sh1.getFillColor());
+ assertTrue(rsh1.getFillPaint(null) instanceof Color) ; // solid fill
+
+ XSLFTextShape sh2 = (XSLFTextShape)getShape(slide, "Title 3");
+ XSLFTextRun run2 = sh2.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(new Color(148, 198, 0), run2.getFontColor());
+ assertNull(sh2.getFillColor()); // no fill
+
+ assertTrue(slide.getSlideLayout().getFollowMasterGraphics());
+ }
+
+ void slide5(XSLFSlide slide){
+ assertTrue(slide.getBackground().getPaint(null) instanceof TexturePaint);
+
+ XSLFTextShape sh2 = (XSLFTextShape)getShape(slide, "Title 1");
+ XSLFTextRun run2 = sh2.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(new Color(148, 198, 0), run2.getFontColor());
+ assertNull(sh2.getFillColor()); // no fill
+ // font size is 40pt and scale factor is 90%
+ assertEquals(36.0, run2.getFontSize());
+
+ assertTrue(slide.getSlideLayout().getFollowMasterGraphics());
+ }
+
+ void slide6(XSLFSlide slide){
+
+ XSLFTextShape sh1 = (XSLFTextShape)getShape(slide, "Subtitle 3");
+ XSLFTextRun run1 = sh1.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(new Color(66, 66, 66), run1.getFontColor());
+ assertNull(sh1.getFillColor()); // no fill
+
+ XSLFTextShape sh2 = (XSLFTextShape)getShape(slide, "Title 2");
+ XSLFTextRun run2 = sh2.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(new Color(148, 198, 0), run2.getFontColor());
+ assertNull(sh2.getFillColor()); // no fill
+
+ assertFalse(slide.getSlideLayout().getFollowMasterGraphics());
+ }
+
+ void slide7(XSLFSlide slide){
+
+ //YK: PPT reports a slightly different color: r=189,g=239,b=87
+ assertEquals(new Color(182, 218, 108), slide.getBackground().getFillColor());
+
+ assertFalse(slide.getFollowMasterGraphics());
+ }
+
+ void slide8(XSLFSlide slide){
+ assertTrue(slide.getBackground().getPaint(null) instanceof TexturePaint);
+ }
+
+ void slide9(XSLFSlide slide){
+ assertTrue(slide.getBackground().getPaint(null) instanceof TexturePaint);
+ }
+
+ void slide10(XSLFSlide slide){
+ assertTrue(slide.getBackground().getPaint(null).getClass().getName().indexOf("Gradient") > 0);
+
+ XSLFTextShape sh1 = (XSLFTextShape)getShape(slide, "Title 3");
+ XSLFTextRun run1 = sh1.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(Color.white, run1.getFontColor());
+ assertNull(sh1.getFillColor()); // no fill
+
+ XSLFTextShape sh2 = (XSLFTextShape)getShape(slide, "Subtitle 4");
+ XSLFTextRun run2 = sh2.getTextParagraphs().get(0).getTextRuns().get(0);
+ assertEquals(Color.white, run2.getFontColor());
+ assertNull(sh2.getFillColor()); // no fill
}
}
diff --git a/src/resources/scratchpad/org/apache/poi/xslf/usermodel/empty.pptx b/src/resources/ooxml/org/apache/poi/xslf/usermodel/empty.pptx
similarity index 100%
rename from src/resources/scratchpad/org/apache/poi/xslf/usermodel/empty.pptx
rename to src/resources/ooxml/org/apache/poi/xslf/usermodel/empty.pptx
diff --git a/src/resources/scratchpad/org/apache/poi/xslf/usermodel/presetShapeDefinitions.xml b/src/resources/ooxml/org/apache/poi/xslf/usermodel/presetShapeDefinitions.xml
similarity index 100%
rename from src/resources/scratchpad/org/apache/poi/xslf/usermodel/presetShapeDefinitions.xml
rename to src/resources/ooxml/org/apache/poi/xslf/usermodel/presetShapeDefinitions.xml
diff --git a/test-data/slideshow/pptx2svg.pptx b/test-data/slideshow/pptx2svg.pptx
new file mode 100644
index 0000000000..2f9a3515cb
Binary files /dev/null and b/test-data/slideshow/pptx2svg.pptx differ
diff --git a/test-data/slideshow/pptx2svg.svg b/test-data/slideshow/pptx2svg.svg
new file mode 100644
index 0000000000..9ea3636916
--- /dev/null
+++ b/test-data/slideshow/pptx2svg.svg
@@ -0,0 +1,94 @@
+
+
diff --git a/test-data/slideshow/themes.pptx b/test-data/slideshow/themes.pptx
new file mode 100755
index 0000000000..ff39de3ed4
Binary files /dev/null and b/test-data/slideshow/themes.pptx differ