mirror of https://github.com/apache/poi.git
#63028 - Provide font embedding for slideshows
git-svn-id: https://svn.apache.org/repos/asf/poi/trunk@1849898 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
a78bd71fc1
commit
c039da1b94
|
@ -187,11 +187,11 @@ public class TestAllFiles {
|
||||||
HANDLERS.put(".tif", new NullFileHandler());
|
HANDLERS.put(".tif", new NullFileHandler());
|
||||||
HANDLERS.put(".tiff", new NullFileHandler());
|
HANDLERS.put(".tiff", new NullFileHandler());
|
||||||
HANDLERS.put(".wav", new NullFileHandler());
|
HANDLERS.put(".wav", new NullFileHandler());
|
||||||
HANDLERS.put(".pfx", new NullFileHandler());
|
|
||||||
HANDLERS.put(".xml", new NullFileHandler());
|
HANDLERS.put(".xml", new NullFileHandler());
|
||||||
HANDLERS.put(".csv", new NullFileHandler());
|
HANDLERS.put(".csv", new NullFileHandler());
|
||||||
HANDLERS.put(".ods", new NullFileHandler());
|
HANDLERS.put(".ods", new NullFileHandler());
|
||||||
HANDLERS.put(".ttf", new NullFileHandler());
|
HANDLERS.put(".ttf", new NullFileHandler());
|
||||||
|
HANDLERS.put(".fntdata", new NullFileHandler());
|
||||||
// VBA source files
|
// VBA source files
|
||||||
HANDLERS.put(".vba", new NullFileHandler());
|
HANDLERS.put(".vba", new NullFileHandler());
|
||||||
HANDLERS.put(".bas", new NullFileHandler());
|
HANDLERS.put(".bas", new NullFileHandler());
|
||||||
|
|
|
@ -70,7 +70,7 @@ public enum FontCharset {
|
||||||
/** Specifies the Russian Cyrillic character set. */
|
/** Specifies the Russian Cyrillic character set. */
|
||||||
RUSSIAN(0x000000CC, "Cp1251"),
|
RUSSIAN(0x000000CC, "Cp1251"),
|
||||||
/** Specifies the Thai character set. */
|
/** Specifies the Thai character set. */
|
||||||
THAI_(0x000000DE, "x-windows-874"),
|
THAI(0x000000DE, "x-windows-874"),
|
||||||
/** Specifies a Eastern European character set. */
|
/** Specifies a Eastern European character set. */
|
||||||
EASTEUROPE(0x000000EE, "Cp1250"),
|
EASTEUROPE(0x000000EE, "Cp1250"),
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -0,0 +1,75 @@
|
||||||
|
/* ====================================================================
|
||||||
|
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.common.usermodel.fonts;
|
||||||
|
|
||||||
|
import org.apache.poi.util.Beta;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A FontFacet holds the font data for a shape of a font, i.e. a regular,
|
||||||
|
* italic, bold or bold-italic version of a Font.
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@Beta
|
||||||
|
public interface FontFacet {
|
||||||
|
/**
|
||||||
|
* Get the font weight.<p>
|
||||||
|
*
|
||||||
|
* The weight of the font in the range 0 through 1000.
|
||||||
|
* For example, 400 is normal and 700 is bold.
|
||||||
|
* If this value is zero, a default weight is used.
|
||||||
|
*
|
||||||
|
* @return the font weight
|
||||||
|
*
|
||||||
|
* @since POI 4.1.0
|
||||||
|
*/
|
||||||
|
default int getWeight() {
|
||||||
|
return FontHeader.REGULAR_WEIGHT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the font weight
|
||||||
|
*
|
||||||
|
* @param weight the font weight
|
||||||
|
*/
|
||||||
|
default void setWeight(int weight) {
|
||||||
|
throw new UnsupportedOperationException("FontFacet is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return {@code true}, if the font is italic
|
||||||
|
*/
|
||||||
|
default boolean isItalic() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the font posture
|
||||||
|
*
|
||||||
|
* @param italic {@code true} for italic, {@code false} for regular
|
||||||
|
*/
|
||||||
|
default void setItalic(boolean italic) {
|
||||||
|
throw new UnsupportedOperationException("FontFacet is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the wrapper object holding the font data
|
||||||
|
*/
|
||||||
|
default Object getFontData() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,227 @@
|
||||||
|
/* ====================================================================
|
||||||
|
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.common.usermodel.fonts;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
|
import org.apache.poi.util.IOUtils;
|
||||||
|
import org.apache.poi.util.LittleEndianByteArrayInputStream;
|
||||||
|
import org.apache.poi.util.LittleEndianInput;
|
||||||
|
import org.apache.poi.util.LittleEndianInputStream;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The header data of an EOT font.<p>
|
||||||
|
*
|
||||||
|
* Currently only version 1 fields are read to identify a stream to be embedded.
|
||||||
|
*
|
||||||
|
* @see <a href="http://www.w3.org/Submission/EOT">Embedded OpenType (EOT) File Format</a>
|
||||||
|
*/
|
||||||
|
@SuppressWarnings({"FieldCanBeLocal", "unused", "Duplicates"})
|
||||||
|
public class FontHeader implements FontInfo {
|
||||||
|
/**
|
||||||
|
* Fonts with a font weight of 400 are regarded as regular weighted.
|
||||||
|
* Higher font weights (up to 1000) are bold - lower weights are thin.
|
||||||
|
*/
|
||||||
|
public static final int REGULAR_WEIGHT = 400;
|
||||||
|
|
||||||
|
private int eotSize;
|
||||||
|
private int fontDataSize;
|
||||||
|
private int version;
|
||||||
|
private int flags;
|
||||||
|
private final byte[] panose = new byte[10];
|
||||||
|
private byte charset;
|
||||||
|
private byte italic;
|
||||||
|
private int weight;
|
||||||
|
private int fsType;
|
||||||
|
private int magic;
|
||||||
|
private int unicodeRange1;
|
||||||
|
private int unicodeRange2;
|
||||||
|
private int unicodeRange3;
|
||||||
|
private int unicodeRange4;
|
||||||
|
private int codePageRange1;
|
||||||
|
private int codePageRange2;
|
||||||
|
private int checkSumAdjustment;
|
||||||
|
private String familyName;
|
||||||
|
private String styleName;
|
||||||
|
private String versionName;
|
||||||
|
private String fullName;
|
||||||
|
|
||||||
|
public void init(byte[] source, int offset, int length) {
|
||||||
|
init(new LittleEndianByteArrayInputStream(source, offset, length));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void init(LittleEndianInput leis) {
|
||||||
|
eotSize = leis.readInt();
|
||||||
|
fontDataSize = leis.readInt();
|
||||||
|
version = leis.readInt();
|
||||||
|
if (version != 0x00010000 && version != 0x00020001 && version != 0x00020002) {
|
||||||
|
throw new RuntimeException("not a EOT font data stream");
|
||||||
|
}
|
||||||
|
flags = leis.readInt();
|
||||||
|
leis.readFully(panose);
|
||||||
|
charset = leis.readByte();
|
||||||
|
italic = leis.readByte();
|
||||||
|
weight = leis.readInt();
|
||||||
|
fsType = leis.readUShort();
|
||||||
|
magic = leis.readUShort();
|
||||||
|
if (magic != 0x504C) {
|
||||||
|
throw new RuntimeException("not a EOT font data stream");
|
||||||
|
}
|
||||||
|
unicodeRange1 = leis.readInt();
|
||||||
|
unicodeRange2 = leis.readInt();
|
||||||
|
unicodeRange3 = leis.readInt();
|
||||||
|
unicodeRange4 = leis.readInt();
|
||||||
|
codePageRange1 = leis.readInt();
|
||||||
|
codePageRange2 = leis.readInt();
|
||||||
|
checkSumAdjustment = leis.readInt();
|
||||||
|
int reserved1 = leis.readInt();
|
||||||
|
int reserved2 = leis.readInt();
|
||||||
|
int reserved3 = leis.readInt();
|
||||||
|
int reserved4 = leis.readInt();
|
||||||
|
familyName = readName(leis);
|
||||||
|
styleName = readName(leis);
|
||||||
|
versionName = readName(leis);
|
||||||
|
fullName = readName(leis);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public InputStream bufferInit(InputStream fontStream) throws IOException {
|
||||||
|
LittleEndianInputStream is = new LittleEndianInputStream(fontStream);
|
||||||
|
is.mark(1000);
|
||||||
|
init(is);
|
||||||
|
is.reset();
|
||||||
|
return is;
|
||||||
|
}
|
||||||
|
|
||||||
|
private String readName(LittleEndianInput leis) {
|
||||||
|
// padding
|
||||||
|
leis.readShort();
|
||||||
|
int nameSize = leis.readUShort();
|
||||||
|
byte[] nameBuf = IOUtils.safelyAllocate(nameSize, 1000);
|
||||||
|
leis.readFully(nameBuf);
|
||||||
|
// may be 0-terminated, just trim it away
|
||||||
|
return new String(nameBuf, 0, nameSize, StandardCharsets.UTF_16LE).trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isItalic() {
|
||||||
|
return italic != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getWeight() {
|
||||||
|
return weight;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isBold() {
|
||||||
|
return getWeight() > REGULAR_WEIGHT;
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte getCharsetByte() {
|
||||||
|
return charset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public FontCharset getCharset() {
|
||||||
|
return FontCharset.valueOf(getCharsetByte());
|
||||||
|
}
|
||||||
|
|
||||||
|
public FontPitch getPitch() {
|
||||||
|
byte familyKind = panose[0];
|
||||||
|
switch (familyKind) {
|
||||||
|
default:
|
||||||
|
// Any
|
||||||
|
case 0:
|
||||||
|
// No Fit
|
||||||
|
case 1:
|
||||||
|
return FontPitch.VARIABLE;
|
||||||
|
|
||||||
|
// Latin Text
|
||||||
|
case 2:
|
||||||
|
// Latin Decorative
|
||||||
|
case 4:
|
||||||
|
byte proportion = panose[3];
|
||||||
|
return proportion == 9 ? FontPitch.FIXED : FontPitch.VARIABLE;
|
||||||
|
|
||||||
|
// Latin Hand Written
|
||||||
|
case 3:
|
||||||
|
// Latin Symbol
|
||||||
|
case 5:
|
||||||
|
byte spacing = panose[3];
|
||||||
|
return spacing == 3 ? FontPitch.FIXED : FontPitch.VARIABLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public FontFamily getFamily() {
|
||||||
|
switch (panose[0]) {
|
||||||
|
// Any
|
||||||
|
case 0:
|
||||||
|
// No Fit
|
||||||
|
case 1:
|
||||||
|
return FontFamily.FF_DONTCARE;
|
||||||
|
// Latin Text
|
||||||
|
case 2:
|
||||||
|
byte serifStyle = panose[1];
|
||||||
|
return (10 <= serifStyle && serifStyle <= 15)
|
||||||
|
? FontFamily.FF_SWISS : FontFamily.FF_ROMAN;
|
||||||
|
// Latin Hand Written
|
||||||
|
case 3:
|
||||||
|
return FontFamily.FF_SCRIPT;
|
||||||
|
// Latin Decorative
|
||||||
|
default:
|
||||||
|
case 4:
|
||||||
|
return FontFamily.FF_DECORATIVE;
|
||||||
|
// Latin Symbol
|
||||||
|
case 5:
|
||||||
|
return FontFamily.FF_MODERN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFamilyName() {
|
||||||
|
return familyName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getStyleName() {
|
||||||
|
return styleName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getVersionName() {
|
||||||
|
return versionName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFullName() {
|
||||||
|
return fullName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] getPanose() {
|
||||||
|
return panose;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTypeface() {
|
||||||
|
return getFamilyName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getFlags() {
|
||||||
|
return flags;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,11 @@ limitations under the License.
|
||||||
|
|
||||||
package org.apache.poi.common.usermodel.fonts;
|
package org.apache.poi.common.usermodel.fonts;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.poi.util.Beta;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A FontInfo object holds information about a font configuration.
|
* A FontInfo object holds information about a font configuration.
|
||||||
* It is roughly an equivalent to the LOGFONT structure in Windows GDI.<p>
|
* It is roughly an equivalent to the LOGFONT structure in Windows GDI.<p>
|
||||||
|
@ -30,6 +35,7 @@ package org.apache.poi.common.usermodel.fonts;
|
||||||
*
|
*
|
||||||
* @see <a href="https://msdn.microsoft.com/en-us/library/dd145037.aspx">LOGFONT structure</a>
|
* @see <a href="https://msdn.microsoft.com/en-us/library/dd145037.aspx">LOGFONT structure</a>
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
public interface FontInfo {
|
public interface FontInfo {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,7 +43,9 @@ public interface FontInfo {
|
||||||
* @return unique index number of the underlying record this Font represents
|
* @return unique index number of the underlying record this Font represents
|
||||||
* (probably you don't care unless you're comparing which one is which)
|
* (probably you don't care unless you're comparing which one is which)
|
||||||
*/
|
*/
|
||||||
Integer getIndex();
|
default Integer getIndex() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the index within the collection of Font objects
|
* Sets the index within the collection of Font objects
|
||||||
|
@ -46,7 +54,9 @@ public interface FontInfo {
|
||||||
*
|
*
|
||||||
* @throws UnsupportedOperationException if unsupported
|
* @throws UnsupportedOperationException if unsupported
|
||||||
*/
|
*/
|
||||||
void setIndex(int index);
|
default void setIndex(int index) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -60,36 +70,48 @@ public interface FontInfo {
|
||||||
* @param typeface the full name of the font, when {@code null} removes the font definition -
|
* @param typeface the full name of the font, when {@code null} removes the font definition -
|
||||||
* removal is implementation specific
|
* removal is implementation specific
|
||||||
*/
|
*/
|
||||||
void setTypeface(String typeface);
|
default void setTypeface(String typeface) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the font charset
|
* @return the font charset
|
||||||
*/
|
*/
|
||||||
FontCharset getCharset();
|
default FontCharset getCharset() {
|
||||||
|
return FontCharset.ANSI;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the charset
|
* Sets the charset
|
||||||
*
|
*
|
||||||
* @param charset the charset
|
* @param charset the charset
|
||||||
*/
|
*/
|
||||||
void setCharset(FontCharset charset);
|
default void setCharset(FontCharset charset) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the family class
|
* @return the family class
|
||||||
*/
|
*/
|
||||||
FontFamily getFamily();
|
default FontFamily getFamily() {
|
||||||
|
return FontFamily.FF_DONTCARE;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the font family class
|
* Sets the font family class
|
||||||
*
|
*
|
||||||
* @param family the font family class
|
* @param family the font family class
|
||||||
*/
|
*/
|
||||||
void setFamily(FontFamily family);
|
default void setFamily(FontFamily family) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the font pitch or {@code null} if unsupported
|
* @return the font pitch or {@code null} if unsupported
|
||||||
*/
|
*/
|
||||||
FontPitch getPitch();
|
default FontPitch getPitch() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the font pitch
|
* Set the font pitch
|
||||||
|
@ -98,5 +120,33 @@ public interface FontInfo {
|
||||||
*
|
*
|
||||||
* @throws UnsupportedOperationException if unsupported
|
* @throws UnsupportedOperationException if unsupported
|
||||||
*/
|
*/
|
||||||
void setPitch(FontPitch pitch);
|
default void setPitch(FontPitch pitch) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return panose info in binary form or {@code null} if unknown
|
||||||
|
*/
|
||||||
|
default byte[] getPanose() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the panose in binary form
|
||||||
|
* @param panose the panose bytes
|
||||||
|
*/
|
||||||
|
default void setPanose(byte[] panose) {
|
||||||
|
throw new UnsupportedOperationException("FontInfo is read-only.");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If font facets are embedded in the document, return the list of embedded facets.
|
||||||
|
* The font embedding is experimental, therefore the API can change.
|
||||||
|
* @return the list of embedded EOT font data
|
||||||
|
*/
|
||||||
|
@Beta
|
||||||
|
default List<? extends FontFacet> getFacets() {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -19,10 +19,7 @@
|
||||||
|
|
||||||
package org.apache.poi.sl.draw;
|
package org.apache.poi.sl.draw;
|
||||||
|
|
||||||
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
|
||||||
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
|
||||||
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
|
||||||
import org.apache.poi.util.Internal;
|
import org.apache.poi.util.Internal;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,53 +34,8 @@ import org.apache.poi.util.Internal;
|
||||||
this.typeface = typeface;
|
this.typeface = typeface;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Integer getIndex() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setIndex(int index) {
|
|
||||||
throw new UnsupportedOperationException("DrawFontManagers FontInfo can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getTypeface() {
|
public String getTypeface() {
|
||||||
return typeface;
|
return typeface;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setTypeface(String typeface) {
|
|
||||||
throw new UnsupportedOperationException("DrawFontManagers FontInfo can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public FontCharset getCharset() {
|
|
||||||
return FontCharset.ANSI;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setCharset(FontCharset charset) {
|
|
||||||
throw new UnsupportedOperationException("DrawFontManagers FontInfo can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public FontFamily getFamily() {
|
|
||||||
return FontFamily.FF_SWISS;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFamily(FontFamily family) {
|
|
||||||
throw new UnsupportedOperationException("DrawFontManagers FontInfo can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public FontPitch getPitch() {
|
|
||||||
return FontPitch.VARIABLE;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setPitch(FontPitch pitch) {
|
|
||||||
throw new UnsupportedOperationException("DrawFontManagers FontInfo can't be changed.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,10 +18,14 @@
|
||||||
package org.apache.poi.sl.extractor;
|
package org.apache.poi.sl.extractor;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.BitSet;
|
||||||
|
import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
import java.util.function.Function;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
import org.apache.poi.extractor.POITextExtractor;
|
import org.apache.poi.extractor.POITextExtractor;
|
||||||
import org.apache.poi.sl.usermodel.Comment;
|
|
||||||
import org.apache.poi.sl.usermodel.MasterSheet;
|
import org.apache.poi.sl.usermodel.MasterSheet;
|
||||||
import org.apache.poi.sl.usermodel.Notes;
|
import org.apache.poi.sl.usermodel.Notes;
|
||||||
import org.apache.poi.sl.usermodel.ObjectShape;
|
import org.apache.poi.sl.usermodel.ObjectShape;
|
||||||
|
@ -52,6 +56,10 @@ public class SlideShowExtractor<
|
||||||
> extends POITextExtractor {
|
> extends POITextExtractor {
|
||||||
private static final POILogger LOG = POILogFactory.getLogger(SlideShowExtractor.class);
|
private static final POILogger LOG = POILogFactory.getLogger(SlideShowExtractor.class);
|
||||||
|
|
||||||
|
// placeholder text for slide numbers
|
||||||
|
private static final String SLIDE_NUMBER_PH = "‹#›";
|
||||||
|
|
||||||
|
|
||||||
private SlideShow<S,P> slideshow;
|
private SlideShow<S,P> slideshow;
|
||||||
|
|
||||||
private boolean slidesByDefault = true;
|
private boolean slidesByDefault = true;
|
||||||
|
@ -59,6 +67,7 @@ public class SlideShowExtractor<
|
||||||
private boolean commentsByDefault;
|
private boolean commentsByDefault;
|
||||||
private boolean masterByDefault;
|
private boolean masterByDefault;
|
||||||
|
|
||||||
|
private Predicate<Object> filter = o -> true;
|
||||||
|
|
||||||
public SlideShowExtractor(final SlideShow<S,P> slideshow) {
|
public SlideShowExtractor(final SlideShow<S,P> slideshow) {
|
||||||
setFilesystem(slideshow);
|
setFilesystem(slideshow);
|
||||||
|
@ -115,9 +124,8 @@ public class SlideShowExtractor<
|
||||||
@Override
|
@Override
|
||||||
public String getText() {
|
public String getText() {
|
||||||
final StringBuilder sb = new StringBuilder();
|
final StringBuilder sb = new StringBuilder();
|
||||||
|
|
||||||
for (final Slide<S, P> slide : slideshow.getSlides()) {
|
for (final Slide<S, P> slide : slideshow.getSlides()) {
|
||||||
sb.append(getText(slide));
|
getText(slide, sb::append);
|
||||||
}
|
}
|
||||||
|
|
||||||
return sb.toString();
|
return sb.toString();
|
||||||
|
@ -125,34 +133,37 @@ public class SlideShowExtractor<
|
||||||
|
|
||||||
public String getText(final Slide<S,P> slide) {
|
public String getText(final Slide<S,P> slide) {
|
||||||
final StringBuilder sb = new StringBuilder();
|
final StringBuilder sb = new StringBuilder();
|
||||||
|
getText(slide, sb::append);
|
||||||
|
return sb.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void getText(final Slide<S,P> slide, final Consumer<String> consumer) {
|
||||||
if (slidesByDefault) {
|
if (slidesByDefault) {
|
||||||
printShapeText(slide, sb);
|
printShapeText(slide, consumer);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (masterByDefault) {
|
if (masterByDefault) {
|
||||||
final MasterSheet<S,P> ms = slide.getMasterSheet();
|
final MasterSheet<S,P> ms = slide.getMasterSheet();
|
||||||
printSlideMaster(ms, sb);
|
printSlideMaster(ms, consumer);
|
||||||
|
|
||||||
// only print slide layout, if it's a different instance
|
// only print slide layout, if it's a different instance
|
||||||
final MasterSheet<S,P> sl = slide.getSlideLayout();
|
final MasterSheet<S,P> sl = slide.getSlideLayout();
|
||||||
if (sl != ms) {
|
if (sl != ms) {
|
||||||
printSlideMaster(sl, sb);
|
printSlideMaster(sl, consumer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (commentsByDefault) {
|
if (commentsByDefault) {
|
||||||
printComments(slide, sb);
|
printComments(slide, consumer);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (notesByDefault) {
|
if (notesByDefault) {
|
||||||
printNotes(slide, sb);
|
printNotes(slide, consumer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return sb.toString();
|
private void printSlideMaster(final MasterSheet<S,P> master, final Consumer<String> consumer) {
|
||||||
}
|
|
||||||
|
|
||||||
private void printSlideMaster(final MasterSheet<S,P> master, final StringBuilder sb) {
|
|
||||||
if (master == null) {
|
if (master == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -163,27 +174,49 @@ public class SlideShowExtractor<
|
||||||
if (text == null || text.isEmpty() || "*".equals(text)) {
|
if (text == null || text.isEmpty() || "*".equals(text)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ts.isPlaceholder()) {
|
if (ts.isPlaceholder()) {
|
||||||
// don't bother about boiler plate text on master sheets
|
// don't bother about boiler plate text on master sheets
|
||||||
LOG.log(POILogger.INFO, "Ignoring boiler plate (placeholder) text on slide master:", text);
|
LOG.log(POILogger.INFO, "Ignoring boiler plate (placeholder) text on slide master:", text);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
sb.append(text);
|
|
||||||
if (!text.endsWith("\n")) {
|
|
||||||
sb.append("\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
printTextParagraphs(ts.getTextParagraphs(), consumer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private String printHeaderReturnFooter(final Sheet<S,P> sheet, final StringBuilder sb) {
|
private void printTextParagraphs(final List<P> paras, final Consumer<String> consumer) {
|
||||||
|
printTextParagraphs(paras, consumer, "\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void printTextParagraphs(final List<P> paras, final Consumer<String> consumer, String trailer) {
|
||||||
|
printTextParagraphs(paras, consumer, trailer, SlideShowExtractor::replaceTextCap);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void printTextParagraphs(final List<P> paras, final Consumer<String> consumer, String trailer, final Function<TextRun,String> converter) {
|
||||||
|
for (P p : paras) {
|
||||||
|
for (TextRun r : p) {
|
||||||
|
if (filter.test(r)) {
|
||||||
|
consumer.accept(converter.apply(r));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!trailer.isEmpty() && filter.test(trailer)) {
|
||||||
|
consumer.accept(trailer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void printHeaderFooter(final Sheet<S,P> sheet, final Consumer<String> consumer, final Consumer<String> footerCon) {
|
||||||
final Sheet<S, P> m = (sheet instanceof Slide) ? sheet.getMasterSheet() : sheet;
|
final Sheet<S, P> m = (sheet instanceof Slide) ? sheet.getMasterSheet() : sheet;
|
||||||
final StringBuilder footer = new StringBuilder("\n");
|
addSheetPlaceholderDatails(sheet, Placeholder.HEADER, consumer);
|
||||||
addSheetPlaceholderDatails(sheet, Placeholder.HEADER, sb);
|
addSheetPlaceholderDatails(sheet, Placeholder.FOOTER, footerCon);
|
||||||
addSheetPlaceholderDatails(sheet, Placeholder.FOOTER, footer);
|
|
||||||
|
if (!masterByDefault) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (masterByDefault) {
|
|
||||||
// write header texts and determine footer text
|
// write header texts and determine footer text
|
||||||
for (Shape<S, P> s : m) {
|
for (Shape<S, P> s : m) {
|
||||||
if (!(s instanceof TextShape)) {
|
if (!(s instanceof TextShape)) {
|
||||||
|
@ -196,18 +229,13 @@ public class SlideShowExtractor<
|
||||||
}
|
}
|
||||||
switch (pd.getPlaceholder()) {
|
switch (pd.getPlaceholder()) {
|
||||||
case HEADER:
|
case HEADER:
|
||||||
sb.append(ts.getText());
|
printTextParagraphs(ts.getTextParagraphs(), consumer);
|
||||||
sb.append('\n');
|
|
||||||
break;
|
|
||||||
case SLIDE_NUMBER:
|
|
||||||
if (sheet instanceof Slide) {
|
|
||||||
footer.append(ts.getText().replace("‹#›", Integer.toString(((Slide<S, P>) sheet).getSlideNumber() + 1)));
|
|
||||||
footer.append('\n');
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case FOOTER:
|
case FOOTER:
|
||||||
footer.append(ts.getText());
|
printTextParagraphs(ts.getTextParagraphs(), footerCon);
|
||||||
footer.append('\n');
|
break;
|
||||||
|
case SLIDE_NUMBER:
|
||||||
|
printTextParagraphs(ts.getTextParagraphs(), footerCon, "\n", SlideShowExtractor::replaceSlideNumber);
|
||||||
break;
|
break;
|
||||||
case DATETIME:
|
case DATETIME:
|
||||||
// currently not supported
|
// currently not supported
|
||||||
|
@ -217,109 +245,69 @@ public class SlideShowExtractor<
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (footer.length() > 1) ? footer.toString() : "";
|
|
||||||
}
|
|
||||||
|
|
||||||
private void addSheetPlaceholderDatails(final Sheet<S,P> sheet, final Placeholder placeholder, final StringBuilder sb) {
|
private void addSheetPlaceholderDatails(final Sheet<S,P> sheet, final Placeholder placeholder, final Consumer<String> consumer) {
|
||||||
final PlaceholderDetails headerPD = sheet.getPlaceholderDetails(placeholder);
|
final PlaceholderDetails headerPD = sheet.getPlaceholderDetails(placeholder);
|
||||||
if (headerPD == null) {
|
final String headerStr = (headerPD != null) ? headerPD.getText() : null;
|
||||||
return;
|
if (headerStr != null && filter.test(headerPD)) {
|
||||||
|
consumer.accept(headerStr);
|
||||||
}
|
}
|
||||||
final String headerStr = headerPD.getText();
|
|
||||||
if (headerStr == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
sb.append(headerStr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void printShapeText(final Sheet<S,P> sheet, final StringBuilder sb) {
|
private void printShapeText(final Sheet<S,P> sheet, final Consumer<String> consumer) {
|
||||||
final String footer = printHeaderReturnFooter(sheet, sb);
|
final List<String> footer = new LinkedList<>();
|
||||||
printShapeText((ShapeContainer<S,P>)sheet, sb);
|
printHeaderFooter(sheet, consumer, footer::add);
|
||||||
sb.append(footer);
|
printShapeText((ShapeContainer<S,P>)sheet, consumer);
|
||||||
|
footer.forEach(consumer);
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
private void printShapeText(final ShapeContainer<S,P> container, final StringBuilder sb) {
|
private void printShapeText(final ShapeContainer<S,P> container, final Consumer<String> consumer) {
|
||||||
for (Shape<S,P> shape : container) {
|
for (Shape<S,P> shape : container) {
|
||||||
if (shape instanceof TextShape) {
|
if (shape instanceof TextShape) {
|
||||||
printShapeText((TextShape<S,P>)shape, sb);
|
printTextParagraphs(((TextShape<S,P>)shape).getTextParagraphs(), consumer);
|
||||||
} else if (shape instanceof TableShape) {
|
} else if (shape instanceof TableShape) {
|
||||||
printShapeText((TableShape<S,P>)shape, sb);
|
printShapeText((TableShape<S,P>)shape, consumer);
|
||||||
} else if (shape instanceof ShapeContainer) {
|
} else if (shape instanceof ShapeContainer) {
|
||||||
printShapeText((ShapeContainer<S,P>)shape, sb);
|
printShapeText((ShapeContainer<S,P>)shape, consumer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void printShapeText(final TextShape<S,P> shape, final StringBuilder sb) {
|
|
||||||
final List<P> paraList = shape.getTextParagraphs();
|
|
||||||
if (paraList.isEmpty()) {
|
|
||||||
sb.append('\n');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
for (final P para : paraList) {
|
|
||||||
for (final TextRun tr : para) {
|
|
||||||
final String str = tr.getRawText().replace("\r", "");
|
|
||||||
final String newStr;
|
|
||||||
switch (tr.getTextCap()) {
|
|
||||||
case ALL:
|
|
||||||
newStr = str.toUpperCase(LocaleUtil.getUserLocale());
|
|
||||||
break;
|
|
||||||
case SMALL:
|
|
||||||
newStr = str.toLowerCase(LocaleUtil.getUserLocale());
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
case NONE:
|
|
||||||
newStr = str;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
sb.append(newStr);
|
|
||||||
}
|
|
||||||
sb.append('\n');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@SuppressWarnings("Duplicates")
|
@SuppressWarnings("Duplicates")
|
||||||
private void printShapeText(final TableShape<S,P> shape, final StringBuilder sb) {
|
private void printShapeText(final TableShape<S,P> shape, final Consumer<String> consumer) {
|
||||||
final int nrows = shape.getNumberOfRows();
|
final int nrows = shape.getNumberOfRows();
|
||||||
final int ncols = shape.getNumberOfColumns();
|
final int ncols = shape.getNumberOfColumns();
|
||||||
for (int row = 0; row < nrows; row++){
|
for (int row = 0; row < nrows; row++) {
|
||||||
|
String trailer = "";
|
||||||
for (int col = 0; col < ncols; col++){
|
for (int col = 0; col < ncols; col++){
|
||||||
TableCell<S, P> cell = shape.getCell(row, col);
|
TableCell<S, P> cell = shape.getCell(row, col);
|
||||||
//defensive null checks; don't know if they're necessary
|
//defensive null checks; don't know if they're necessary
|
||||||
if (cell != null){
|
if (cell != null) {
|
||||||
String txt = cell.getText();
|
trailer = col < ncols-1 ? "\t" : "\n";
|
||||||
txt = (txt == null) ? "" : txt;
|
printTextParagraphs(cell.getTextParagraphs(), consumer, trailer);
|
||||||
sb.append(txt);
|
|
||||||
if (col < ncols-1){
|
|
||||||
sb.append('\t');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (!trailer.equals("\n") && filter.test("\n")) {
|
||||||
|
consumer.accept("\n");
|
||||||
}
|
}
|
||||||
sb.append('\n');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void printComments(final Slide<S,P> slide, final StringBuilder sb) {
|
private void printComments(final Slide<S,P> slide, final Consumer<String> consumer) {
|
||||||
for (final Comment comment : slide.getComments()) {
|
slide.getComments().stream().filter(filter).map(c -> c.getAuthor()+" - "+c.getText()).forEach(consumer);
|
||||||
sb.append(comment.getAuthor());
|
|
||||||
sb.append(" - ");
|
|
||||||
sb.append(comment.getText());
|
|
||||||
sb.append("\n");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void printNotes(final Slide<S,P> slide, final StringBuilder sb) {
|
private void printNotes(final Slide<S,P> slide, final Consumer<String> consumer) {
|
||||||
final Notes<S, P> notes = slide.getNotes();
|
final Notes<S, P> notes = slide.getNotes();
|
||||||
if (notes == null) {
|
if (notes == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final String footer = printHeaderReturnFooter(notes, sb);
|
List<String> footer = new LinkedList<>();
|
||||||
|
printHeaderFooter(notes, consumer, footer::add);
|
||||||
printShapeText(notes, sb);
|
printShapeText(notes, consumer);
|
||||||
|
footer.forEach(consumer);
|
||||||
sb.append(footer);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<? extends ObjectShape<S,P>> getOLEShapes() {
|
public List<? extends ObjectShape<S,P>> getOLEShapes() {
|
||||||
|
@ -342,4 +330,83 @@ public class SlideShowExtractor<
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static String replaceSlideNumber(TextRun tr) {
|
||||||
|
String raw = tr.getRawText();
|
||||||
|
|
||||||
|
if (!raw.contains(SLIDE_NUMBER_PH)) {
|
||||||
|
return raw;
|
||||||
|
}
|
||||||
|
|
||||||
|
TextParagraph tp = tr.getParagraph();
|
||||||
|
TextShape ps = (tp != null) ? tp.getParentShape() : null;
|
||||||
|
Sheet sh = (ps != null) ? ps.getSheet() : null;
|
||||||
|
String slideNr = (sh instanceof Slide) ? Integer.toString(((Slide)sh).getSlideNumber() + 1) : "";
|
||||||
|
|
||||||
|
return raw.replace(SLIDE_NUMBER_PH, slideNr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String replaceTextCap(TextRun tr) {
|
||||||
|
final TextParagraph tp = tr.getParagraph();
|
||||||
|
final TextShape sh = (tp != null) ? tp.getParentShape() : null;
|
||||||
|
final Placeholder ph = (sh != null) ? sh.getPlaceholder() : null;
|
||||||
|
|
||||||
|
// 0xB acts like cariage return in page titles and like blank in the others
|
||||||
|
final char sep = (
|
||||||
|
ph == Placeholder.TITLE ||
|
||||||
|
ph == Placeholder.CENTERED_TITLE ||
|
||||||
|
ph == Placeholder.SUBTITLE
|
||||||
|
) ? '\n' : ' ';
|
||||||
|
|
||||||
|
// PowerPoint seems to store files with \r as the line break
|
||||||
|
// The messes things up on everything but a Mac, so translate them to \n
|
||||||
|
String txt = tr.getRawText();
|
||||||
|
txt = txt.replace('\r', '\n');
|
||||||
|
txt = txt.replace((char) 0x0B, sep);
|
||||||
|
|
||||||
|
switch (tr.getTextCap()) {
|
||||||
|
case ALL:
|
||||||
|
txt = txt.toUpperCase(LocaleUtil.getUserLocale());
|
||||||
|
case SMALL:
|
||||||
|
txt = txt.toLowerCase(LocaleUtil.getUserLocale());
|
||||||
|
}
|
||||||
|
|
||||||
|
return txt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract the used codepoints for font embedding / subsetting
|
||||||
|
* @param typeface the typeface/font family of the textruns to examine
|
||||||
|
* @param italic use {@code true} for italic TextRuns, {@code false} for non-italic ones and
|
||||||
|
* {@code null} if it doesn't matter
|
||||||
|
* @param bold use {@code true} for bold TextRuns, {@code false} for non-bold ones and
|
||||||
|
* {@code null} if it doesn't matter
|
||||||
|
* @return a bitset with the marked/used codepoints
|
||||||
|
*/
|
||||||
|
public BitSet getCodepoints(String typeface, Boolean italic, Boolean bold) {
|
||||||
|
final BitSet glyphs = new BitSet();
|
||||||
|
|
||||||
|
Predicate<Object> filterOld = filter;
|
||||||
|
try {
|
||||||
|
filter = o -> filterFonts(o, typeface, italic, bold);
|
||||||
|
slideshow.getSlides().forEach(slide ->
|
||||||
|
getText(slide, s -> s.codePoints().forEach(glyphs::set))
|
||||||
|
);
|
||||||
|
} finally {
|
||||||
|
filter = filterOld;
|
||||||
|
}
|
||||||
|
|
||||||
|
return glyphs;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean filterFonts(Object o, String typeface, Boolean italic, Boolean bold) {
|
||||||
|
if (!(o instanceof TextRun)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
TextRun tr = (TextRun)o;
|
||||||
|
return
|
||||||
|
typeface.equalsIgnoreCase(tr.getFontFamily()) &&
|
||||||
|
(italic == null || tr.isItalic() == italic) &&
|
||||||
|
(bold == null || tr.isBold() == bold);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,22 +0,0 @@
|
||||||
/* ====================================================================
|
|
||||||
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.sl.usermodel;
|
|
||||||
|
|
||||||
public interface FontCollection {
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,29 +0,0 @@
|
||||||
/* ====================================================================
|
|
||||||
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.sl.usermodel;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Common SlideShow resources, such as fonts, pictures
|
|
||||||
* and multimedia data
|
|
||||||
*/
|
|
||||||
public interface Resources {
|
|
||||||
public FontCollection getFontCollection();
|
|
||||||
|
|
||||||
public PictureData[] getPictureData();
|
|
||||||
public int addPictureData(PictureData pict);
|
|
||||||
}
|
|
|
@ -19,6 +19,7 @@ package org.apache.poi.sl.usermodel;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
public interface Slide<
|
public interface Slide<
|
||||||
S extends Shape<S,P>,
|
S extends Shape<S,P>,
|
||||||
P extends TextParagraph<S,P,? extends TextRun>
|
P extends TextParagraph<S,P,? extends TextRun>
|
||||||
|
@ -82,7 +83,7 @@ public interface Slide<
|
||||||
*
|
*
|
||||||
* @since POI 4.0.0
|
* @since POI 4.0.0
|
||||||
*/
|
*/
|
||||||
MasterSheet getSlideLayout();
|
MasterSheet<S,P> getSlideLayout();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the slide name, defaults to "Slide[slideNumber]"
|
* @return the slide name, defaults to "Slide[slideNumber]"
|
||||||
|
|
|
@ -25,6 +25,7 @@ import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
import org.apache.poi.extractor.POITextExtractor;
|
import org.apache.poi.extractor.POITextExtractor;
|
||||||
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
||||||
|
|
||||||
|
@ -44,8 +45,6 @@ public interface SlideShow<
|
||||||
*/
|
*/
|
||||||
List<? extends MasterSheet<S,P>> getSlideMasters();
|
List<? extends MasterSheet<S,P>> getSlideMasters();
|
||||||
|
|
||||||
Resources getResources();
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the current page size
|
* Returns the current page size
|
||||||
*
|
*
|
||||||
|
@ -135,4 +134,30 @@ public interface SlideShow<
|
||||||
* @since POI 4.0.0
|
* @since POI 4.0.0
|
||||||
*/
|
*/
|
||||||
Object getPersistDocument();
|
Object getPersistDocument();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add an EOT font to the slideshow.
|
||||||
|
* An EOT or MTX font is a transformed True-Type (.ttf) or Open-Type (.otf) font.
|
||||||
|
* To transform a True-Type font use the sfntly library (see "see also" below)<p>
|
||||||
|
*
|
||||||
|
* (Older?) Powerpoint versions handle embedded fonts by converting them to .ttf files
|
||||||
|
* and put them into the Windows fonts directory. If the user is not allowed to install
|
||||||
|
* fonts, the slideshow can't be opened. While the slideshow is opened, its possible
|
||||||
|
* to copy the extracted .ttfs from the fonts directory. When the slideshow is closed,
|
||||||
|
* they will be removed.
|
||||||
|
*
|
||||||
|
* @param fontData the EOT font as stream
|
||||||
|
* @return the font info object containing the new font data
|
||||||
|
* @throws IOException if the fontData can't be saved or if the fontData is no EOT font
|
||||||
|
*
|
||||||
|
* @see <a href="http://www.w3.org/Submission/EOT">EOT specification</a>
|
||||||
|
* @see <a href="https://github.com/googlei18n/sfntly">googles sfntly library</a>
|
||||||
|
* @see <a href="https://github.com/kiwiwings/poi-font-mbender">Example on how to subset and embed fonts</a>
|
||||||
|
*/
|
||||||
|
FontInfo addFont(InputStream fontData) throws IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return a list of registered fonts
|
||||||
|
*/
|
||||||
|
List<? extends FontInfo> getFonts();
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,6 +27,7 @@ import org.apache.poi.util.Internal;
|
||||||
/**
|
/**
|
||||||
* Some text.
|
* Some text.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
public interface TextRun {
|
public interface TextRun {
|
||||||
/**
|
/**
|
||||||
* Type of text capitals
|
* Type of text capitals
|
||||||
|
@ -243,4 +244,11 @@ public interface TextRun {
|
||||||
*/
|
*/
|
||||||
@Internal
|
@Internal
|
||||||
FieldType getFieldType();
|
FieldType getFieldType();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return the paragraph which contains this TextRun
|
||||||
|
*
|
||||||
|
* @since POI 4.1.0
|
||||||
|
*/
|
||||||
|
TextParagraph<?,?,?> getParagraph();
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,19 +30,20 @@ import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.OptionalLong;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
import org.apache.poi.ooxml.POIXMLDocument;
|
import org.apache.poi.ooxml.POIXMLDocument;
|
||||||
import org.apache.poi.ooxml.POIXMLDocumentPart;
|
import org.apache.poi.ooxml.POIXMLDocumentPart;
|
||||||
import org.apache.poi.ooxml.POIXMLException;
|
import org.apache.poi.ooxml.POIXMLException;
|
||||||
import org.apache.poi.ooxml.extractor.POIXMLPropertiesTextExtractor;
|
import org.apache.poi.ooxml.extractor.POIXMLPropertiesTextExtractor;
|
||||||
import org.apache.poi.ooxml.util.PackageHelper;
|
import org.apache.poi.ooxml.util.PackageHelper;
|
||||||
import org.apache.poi.openxml4j.exceptions.OpenXML4JException;
|
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
|
||||||
import org.apache.poi.openxml4j.opc.OPCPackage;
|
import org.apache.poi.openxml4j.opc.OPCPackage;
|
||||||
import org.apache.poi.openxml4j.opc.PackagePart;
|
import org.apache.poi.openxml4j.opc.PackagePart;
|
||||||
import org.apache.poi.sl.usermodel.MasterSheet;
|
import org.apache.poi.sl.usermodel.MasterSheet;
|
||||||
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
||||||
import org.apache.poi.sl.usermodel.Resources;
|
|
||||||
import org.apache.poi.sl.usermodel.SlideShow;
|
import org.apache.poi.sl.usermodel.SlideShow;
|
||||||
import org.apache.poi.util.Beta;
|
import org.apache.poi.util.Beta;
|
||||||
import org.apache.poi.util.IOUtils;
|
import org.apache.poi.util.IOUtils;
|
||||||
|
@ -60,7 +61,6 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTNotesMasterIdListE
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTPresentation;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTPresentation;
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdList;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdList;
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdListEntry;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdListEntry;
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideMasterIdListEntry;
|
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideSize;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideSize;
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.PresentationDocument;
|
import org.openxmlformats.schemas.presentationml.x2006.main.PresentationDocument;
|
||||||
|
|
||||||
|
@ -70,6 +70,7 @@ import org.openxmlformats.schemas.presentationml.x2006.main.PresentationDocument
|
||||||
* they are reading or writing a slideshow. It is also the
|
* they are reading or writing a slideshow. It is also the
|
||||||
* top level object for creating new slides/etc.
|
* top level object for creating new slides/etc.
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("WeakerAccess")
|
||||||
@Beta
|
@Beta
|
||||||
public class XMLSlideShow extends POIXMLDocument
|
public class XMLSlideShow extends POIXMLDocument
|
||||||
implements SlideShow<XSLFShape, XSLFTextParagraph> {
|
implements SlideShow<XSLFShape, XSLFTextParagraph> {
|
||||||
|
@ -78,10 +79,10 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
private static final int MAX_RECORD_LENGTH = 1_000_000;
|
private static final int MAX_RECORD_LENGTH = 1_000_000;
|
||||||
|
|
||||||
private CTPresentation _presentation;
|
private CTPresentation _presentation;
|
||||||
private List<XSLFSlide> _slides;
|
private final List<XSLFSlide> _slides = new ArrayList<>();
|
||||||
private List<XSLFSlideMaster> _masters;
|
private final List<XSLFSlideMaster> _masters = new ArrayList<>();
|
||||||
private List<XSLFPictureData> _pictures;
|
private final List<XSLFPictureData> _pictures = new ArrayList<>();
|
||||||
private List<XSLFChart> _charts;
|
private final List<XSLFChart> _charts = new ArrayList<>();
|
||||||
private XSLFTableStyles _tableStyles;
|
private XSLFTableStyles _tableStyles;
|
||||||
private XSLFNotesMaster _notesMaster;
|
private XSLFNotesMaster _notesMaster;
|
||||||
private XSLFCommentAuthors _commentAuthors;
|
private XSLFCommentAuthors _commentAuthors;
|
||||||
|
@ -153,27 +154,26 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_charts = new ArrayList<>(chartMap.size());
|
_charts.clear();
|
||||||
for (XSLFChart chart : chartMap.values()) {
|
_charts.addAll(chartMap.values());
|
||||||
_charts.add(chart);
|
|
||||||
|
_masters.clear();
|
||||||
|
if (_presentation.isSetSldMasterIdLst()) {
|
||||||
|
_presentation.getSldMasterIdLst().getSldMasterIdList().forEach(
|
||||||
|
id -> _masters.add(masterMap.get(id.getId2()))
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
_masters = new ArrayList<>(masterMap.size());
|
_slides.clear();
|
||||||
for (CTSlideMasterIdListEntry masterId : _presentation.getSldMasterIdLst().getSldMasterIdList()) {
|
|
||||||
XSLFSlideMaster master = masterMap.get(masterId.getId2());
|
|
||||||
_masters.add(master);
|
|
||||||
}
|
|
||||||
|
|
||||||
_slides = new ArrayList<>(shIdMap.size());
|
|
||||||
if (_presentation.isSetSldIdLst()) {
|
if (_presentation.isSetSldIdLst()) {
|
||||||
for (CTSlideIdListEntry slId : _presentation.getSldIdLst().getSldIdList()) {
|
_presentation.getSldIdLst().getSldIdList().forEach(id -> {
|
||||||
XSLFSlide sh = shIdMap.get(slId.getId2());
|
XSLFSlide sh = shIdMap.get(id.getId2());
|
||||||
if (sh == null) {
|
if (sh == null) {
|
||||||
LOG.log(POILogger.WARN, "Slide with r:id " + slId.getId() + " was defined, but didn't exist in package, skipping");
|
LOG.log(POILogger.WARN, "Slide with r:id " + id.getId() + " was defined, but didn't exist in package, skipping");
|
||||||
continue;
|
} else {
|
||||||
}
|
|
||||||
_slides.add(sh);
|
_slides.add(sh);
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
} catch (XmlException e) {
|
} catch (XmlException e) {
|
||||||
throw new POIXMLException(e);
|
throw new POIXMLException(e);
|
||||||
|
@ -192,7 +192,7 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
* Get the document's embedded files.
|
* Get the document's embedded files.
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public List<PackagePart> getAllEmbeddedParts() throws OpenXML4JException {
|
public List<PackagePart> getAllEmbeddedParts() {
|
||||||
return Collections.unmodifiableList(
|
return Collections.unmodifiableList(
|
||||||
getPackage().getPartsByName(Pattern.compile("/ppt/embeddings/.*?"))
|
getPackage().getPartsByName(Pattern.compile("/ppt/embeddings/.*?"))
|
||||||
);
|
);
|
||||||
|
@ -200,14 +200,12 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public List<XSLFPictureData> getPictureData() {
|
public List<XSLFPictureData> getPictureData() {
|
||||||
if (_pictures == null) {
|
if (_pictures.isEmpty()) {
|
||||||
List<PackagePart> mediaParts = getPackage().getPartsByName(Pattern.compile("/ppt/media/.*?"));
|
getPackage().getPartsByName(Pattern.compile("/ppt/media/.*?")).forEach(part -> {
|
||||||
_pictures = new ArrayList<>(mediaParts.size());
|
|
||||||
for (PackagePart part : mediaParts) {
|
|
||||||
XSLFPictureData pd = new XSLFPictureData(part);
|
XSLFPictureData pd = new XSLFPictureData(part);
|
||||||
pd.setIndex(_pictures.size());
|
pd.setIndex(_pictures.size());
|
||||||
_pictures.add(pd);
|
_pictures.add(pd);
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
return Collections.unmodifiableList(_pictures);
|
return Collections.unmodifiableList(_pictures);
|
||||||
}
|
}
|
||||||
|
@ -219,20 +217,16 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
* @return created slide
|
* @return created slide
|
||||||
*/
|
*/
|
||||||
public XSLFSlide createSlide(XSLFSlideLayout layout) {
|
public XSLFSlide createSlide(XSLFSlideLayout layout) {
|
||||||
int slideNumber = 256, cnt = 1;
|
CTSlideIdList slideList = _presentation.isSetSldIdLst()
|
||||||
CTSlideIdList slideList;
|
? _presentation.getSldIdLst() : _presentation.addNewSldIdLst();
|
||||||
XSLFRelation relationType = XSLFRelation.SLIDE;
|
|
||||||
if (!_presentation.isSetSldIdLst()) {
|
|
||||||
slideList = _presentation.addNewSldIdLst();
|
|
||||||
} else {
|
|
||||||
slideList = _presentation.getSldIdLst();
|
|
||||||
for (CTSlideIdListEntry slideId : slideList.getSldIdArray()) {
|
|
||||||
slideNumber = (int) Math.max(slideId.getId() + 1, slideNumber);
|
|
||||||
cnt++;
|
|
||||||
}
|
|
||||||
|
|
||||||
cnt = findNextAvailableFileNameIndex(relationType, cnt);
|
@SuppressWarnings("deprecation")
|
||||||
}
|
OptionalLong maxId = Stream.of(slideList.getSldIdArray())
|
||||||
|
.mapToLong(CTSlideIdListEntry::getId).max();
|
||||||
|
|
||||||
|
final XSLFRelation relationType = XSLFRelation.SLIDE;
|
||||||
|
final int slideNumber = (int)(Math.max(maxId.orElse(0),255)+1);
|
||||||
|
final int cnt = findNextAvailableFileNameIndex(relationType);
|
||||||
|
|
||||||
RelationPart rp = createRelationship
|
RelationPart rp = createRelationship
|
||||||
(relationType, XSLFFactory.getInstance(), cnt, false);
|
(relationType, XSLFFactory.getInstance(), cnt, false);
|
||||||
|
@ -250,35 +244,16 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
return slide;
|
return slide;
|
||||||
}
|
}
|
||||||
|
|
||||||
private int findNextAvailableFileNameIndex(XSLFRelation relationType, int idx) {
|
private int findNextAvailableFileNameIndex(XSLFRelation relationType) {
|
||||||
// Bug 55791: We also need to check that the resulting file name is not already taken
|
// Bug 55791: We also need to check that the resulting file name is not already taken
|
||||||
// this can happen when removing/adding slides, notes or charts
|
// this can happen when removing/adding slides, notes or charts
|
||||||
while (true) {
|
try {
|
||||||
String fileName = relationType.getFileName(idx);
|
return getPackage().getUnusedPartIndex(relationType.getDefaultFileName());
|
||||||
boolean found = false;
|
} catch (InvalidFormatException e) {
|
||||||
for (POIXMLDocumentPart relation : getRelations()) {
|
throw new RuntimeException(e);
|
||||||
if (relation.getPackagePart() != null &&
|
|
||||||
fileName.equals(relation.getPackagePart().getPartName().getName())) {
|
|
||||||
// name is taken => try next one
|
|
||||||
found = true;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!found &&
|
|
||||||
getPackage().getPartsByName(Pattern.compile(Pattern.quote(fileName))).size() > 0) {
|
|
||||||
// name is taken => try next one
|
|
||||||
found = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!found) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
idx++;
|
|
||||||
}
|
|
||||||
return idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a blank slide using the default (first) master.
|
* Create a blank slide using the default (first) master.
|
||||||
*/
|
*/
|
||||||
|
@ -313,8 +288,8 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
* @since POI 4.1.0
|
* @since POI 4.1.0
|
||||||
*/
|
*/
|
||||||
public XSLFChart createChart() {
|
public XSLFChart createChart() {
|
||||||
int chartIdx = findNextAvailableFileNameIndex(XSLFRelation.CHART, _charts.size() + 1);
|
int chartIdx = findNextAvailableFileNameIndex(XSLFRelation.CHART);
|
||||||
XSLFChart chart = (XSLFChart) createRelationship(XSLFRelation.CHART, XSLFFactory.getInstance(), chartIdx, true).getDocumentPart();
|
XSLFChart chart = createRelationship(XSLFRelation.CHART, XSLFFactory.getInstance(), chartIdx, true).getDocumentPart();
|
||||||
chart.setChartIndex(chartIdx);
|
chart.setChartIndex(chartIdx);
|
||||||
_charts.add(chart);
|
_charts.add(chart);
|
||||||
return chart;
|
return chart;
|
||||||
|
@ -341,10 +316,8 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
createNotesMaster();
|
createNotesMaster();
|
||||||
}
|
}
|
||||||
|
|
||||||
int slideIndex = XSLFRelation.SLIDE.getFileNameIndex(slide);
|
|
||||||
|
|
||||||
XSLFRelation relationType = XSLFRelation.NOTES;
|
XSLFRelation relationType = XSLFRelation.NOTES;
|
||||||
slideIndex = findNextAvailableFileNameIndex(relationType, slideIndex);
|
int slideIndex = findNextAvailableFileNameIndex(relationType);
|
||||||
|
|
||||||
// add notes slide to presentation
|
// add notes slide to presentation
|
||||||
XSLFNotes notesSlide = (XSLFNotes) createRelationship
|
XSLFNotes notesSlide = (XSLFNotes) createRelationship
|
||||||
|
@ -453,6 +426,7 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
|
|
||||||
// fix ordering in the low-level xml
|
// fix ordering in the low-level xml
|
||||||
CTSlideIdList sldIdLst = _presentation.getSldIdLst();
|
CTSlideIdList sldIdLst = _presentation.getSldIdLst();
|
||||||
|
@SuppressWarnings("deprecation")
|
||||||
CTSlideIdListEntry[] entries = sldIdLst.getSldIdArray();
|
CTSlideIdListEntry[] entries = sldIdLst.getSldIdArray();
|
||||||
CTSlideIdListEntry oldEntry = entries[oldIndex];
|
CTSlideIdListEntry oldEntry = entries[oldIndex];
|
||||||
if (oldIndex < newIndex) {
|
if (oldIndex < newIndex) {
|
||||||
|
@ -517,14 +491,21 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
return img;
|
return img;
|
||||||
}
|
}
|
||||||
|
|
||||||
int imageNumber = _pictures.size();
|
|
||||||
XSLFRelation relType = XSLFPictureData.getRelationForType(format);
|
XSLFRelation relType = XSLFPictureData.getRelationForType(format);
|
||||||
if (relType == null) {
|
if (relType == null) {
|
||||||
throw new IllegalArgumentException("Picture type " + format + " is not supported.");
|
throw new IllegalArgumentException("Picture type " + format + " is not supported.");
|
||||||
}
|
}
|
||||||
|
|
||||||
img = createRelationship(relType, XSLFFactory.getInstance(), imageNumber + 1, true).getDocumentPart();
|
int imageNumber;
|
||||||
img.setIndex(imageNumber);
|
try {
|
||||||
|
imageNumber = getPackage().getUnusedPartIndex("/ppt/media/image#\\..+");
|
||||||
|
} catch (InvalidFormatException e) {
|
||||||
|
imageNumber = _pictures.size() + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
img = createRelationship(relType, XSLFFactory.getInstance(), imageNumber, true).getDocumentPart();
|
||||||
|
img.setIndex(_pictures.size());
|
||||||
_pictures.add(img);
|
_pictures.add(img);
|
||||||
|
|
||||||
try (OutputStream out = img.getPackagePart().getOutputStream()) {
|
try (OutputStream out = img.getPackagePart().getOutputStream()) {
|
||||||
|
@ -624,18 +605,13 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("RedundantThrows")
|
||||||
@Override
|
@Override
|
||||||
public MasterSheet<XSLFShape, XSLFTextParagraph> createMasterSheet() throws IOException {
|
public MasterSheet<XSLFShape, XSLFTextParagraph> createMasterSheet() throws IOException {
|
||||||
// TODO: implement!
|
// TODO: implement!
|
||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Resources getResources() {
|
|
||||||
// TODO: implement!
|
|
||||||
throw new UnsupportedOperationException();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public POIXMLPropertiesTextExtractor getMetadataTextExtractor() {
|
public POIXMLPropertiesTextExtractor getMetadataTextExtractor() {
|
||||||
return new POIXMLPropertiesTextExtractor(this);
|
return new POIXMLPropertiesTextExtractor(this);
|
||||||
|
@ -645,4 +621,14 @@ public class XMLSlideShow extends POIXMLDocument
|
||||||
public Object getPersistDocument() {
|
public Object getPersistDocument() {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public XSLFFontInfo addFont(InputStream fontStream) throws IOException {
|
||||||
|
return XSLFFontInfo.addFontToSlideShow(this, fontStream);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<XSLFFontInfo> getFonts() {
|
||||||
|
return XSLFFontInfo.getFonts(this);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,84 @@
|
||||||
|
/*
|
||||||
|
* ====================================================================
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* ====================================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.poi.xslf.usermodel;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import org.apache.poi.ooxml.POIXMLDocumentPart;
|
||||||
|
import org.apache.poi.openxml4j.opc.PackagePart;
|
||||||
|
import org.apache.poi.util.Beta;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A container for fontdata files, i.e. MTX fonts derived from
|
||||||
|
* true (TTF) or open (OTF) type fonts.
|
||||||
|
*
|
||||||
|
* @since POI 4.1.0
|
||||||
|
*/
|
||||||
|
@Beta
|
||||||
|
public class XSLFFontData extends POIXMLDocumentPart {
|
||||||
|
/**
|
||||||
|
* Create a new XSLFFontData node
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
protected XSLFFontData() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Construct XSLFFontData from a package part
|
||||||
|
*
|
||||||
|
* @param part the package part holding the ole data
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public XSLFFontData(final PackagePart part) {
|
||||||
|
super(part);
|
||||||
|
}
|
||||||
|
|
||||||
|
public InputStream getInputStream() throws IOException {
|
||||||
|
return getPackagePart().getInputStream();
|
||||||
|
}
|
||||||
|
|
||||||
|
public OutputStream getOutputStream() {
|
||||||
|
final PackagePart pp = getPackagePart();
|
||||||
|
pp.clear();
|
||||||
|
return pp.getOutputStream();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* XSLFFontData objects store the actual content in the part directly without keeping a
|
||||||
|
* copy like all others therefore we need to handle them differently.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected void prepareForCommit() {
|
||||||
|
// do not clear the part here
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public void setData(final byte[] data) throws IOException {
|
||||||
|
try (final OutputStream os = getPackagePart().getOutputStream()) {
|
||||||
|
os.write(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,285 @@
|
||||||
|
/*
|
||||||
|
* ====================================================================
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* ====================================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.poi.xslf.usermodel;
|
||||||
|
|
||||||
|
import java.awt.Font;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontFacet;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontHeader;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
||||||
|
import org.apache.poi.ooxml.POIXMLDocumentPart;
|
||||||
|
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
|
||||||
|
import org.apache.poi.util.IOUtils;
|
||||||
|
import org.openxmlformats.schemas.drawingml.x2006.main.CTTextFont;
|
||||||
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTEmbeddedFontDataId;
|
||||||
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTEmbeddedFontList;
|
||||||
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTEmbeddedFontListEntry;
|
||||||
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTPresentation;
|
||||||
|
|
||||||
|
@SuppressWarnings("WeakerAccess")
|
||||||
|
public class XSLFFontInfo implements FontInfo {
|
||||||
|
final XMLSlideShow ppt;
|
||||||
|
final String typeface;
|
||||||
|
final CTEmbeddedFontListEntry fontListEntry;
|
||||||
|
|
||||||
|
public XSLFFontInfo(XMLSlideShow ppt, String typeface) {
|
||||||
|
this.ppt = ppt;
|
||||||
|
this.typeface = typeface;
|
||||||
|
|
||||||
|
final CTPresentation pres = ppt.getCTPresentation();
|
||||||
|
CTEmbeddedFontList fontList = pres.isSetEmbeddedFontLst()
|
||||||
|
? pres.getEmbeddedFontLst() : pres.addNewEmbeddedFontLst();
|
||||||
|
|
||||||
|
for (CTEmbeddedFontListEntry fe : fontList.getEmbeddedFontArray()) {
|
||||||
|
if (typeface.equalsIgnoreCase(fe.getFont().getTypeface())) {
|
||||||
|
fontListEntry = fe;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fontListEntry = fontList.addNewEmbeddedFont();
|
||||||
|
fontListEntry.addNewFont().setTypeface(typeface);
|
||||||
|
}
|
||||||
|
|
||||||
|
public XSLFFontInfo(XMLSlideShow ppt, CTEmbeddedFontListEntry fontListEntry) {
|
||||||
|
this.ppt = ppt;
|
||||||
|
this.typeface = fontListEntry.getFont().getTypeface();
|
||||||
|
this.fontListEntry = fontListEntry;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getTypeface() {
|
||||||
|
return getFont().getTypeface();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setTypeface(String typeface) {
|
||||||
|
getFont().setTypeface(typeface);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public FontCharset getCharset() {
|
||||||
|
return FontCharset.valueOf(getFont().getCharset());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setCharset(FontCharset charset) {
|
||||||
|
getFont().setCharset((byte)charset.getNativeId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public FontFamily getFamily() {
|
||||||
|
return FontFamily.valueOfPitchFamily(getFont().getPitchFamily());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setFamily(FontFamily family) {
|
||||||
|
byte pitchAndFamily = getFont().getPitchFamily();
|
||||||
|
FontPitch pitch = FontPitch.valueOfPitchFamily(pitchAndFamily);
|
||||||
|
getFont().setPitchFamily(FontPitch.getNativeId(pitch, family));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public FontPitch getPitch() {
|
||||||
|
return FontPitch.valueOfPitchFamily(getFont().getPitchFamily());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setPitch(FontPitch pitch) {
|
||||||
|
byte pitchAndFamily = getFont().getPitchFamily();
|
||||||
|
FontFamily family = FontFamily.valueOfPitchFamily(pitchAndFamily);
|
||||||
|
getFont().setPitchFamily(FontPitch.getNativeId(pitch, family));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public byte[] getPanose() {
|
||||||
|
return getFont().getPanose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<FontFacet> getFacets() {
|
||||||
|
List<FontFacet> facetList = new ArrayList<>();
|
||||||
|
if (fontListEntry.isSetRegular()) {
|
||||||
|
facetList.add(new XSLFFontFacet((fontListEntry.getRegular())));
|
||||||
|
}
|
||||||
|
if (fontListEntry.isSetItalic()) {
|
||||||
|
facetList.add(new XSLFFontFacet((fontListEntry.getItalic())));
|
||||||
|
}
|
||||||
|
if (fontListEntry.isSetBold()) {
|
||||||
|
facetList.add(new XSLFFontFacet((fontListEntry.getBold())));
|
||||||
|
}
|
||||||
|
if (fontListEntry.isSetBoldItalic()) {
|
||||||
|
facetList.add(new XSLFFontFacet((fontListEntry.getBoldItalic())));
|
||||||
|
}
|
||||||
|
return facetList;
|
||||||
|
}
|
||||||
|
|
||||||
|
public FontFacet addFacet(InputStream fontData) throws IOException {
|
||||||
|
FontHeader header = new FontHeader();
|
||||||
|
InputStream is = header.bufferInit(fontData);
|
||||||
|
|
||||||
|
final CTPresentation pres = ppt.getCTPresentation();
|
||||||
|
pres.setEmbedTrueTypeFonts(true);
|
||||||
|
pres.setSaveSubsetFonts(true);
|
||||||
|
|
||||||
|
final CTEmbeddedFontDataId dataId;
|
||||||
|
final int style =
|
||||||
|
(header.getWeight() > 400 ? Font.BOLD : Font.PLAIN) |
|
||||||
|
(header.isItalic() ? Font.ITALIC : Font.PLAIN);
|
||||||
|
switch (style) {
|
||||||
|
case Font.PLAIN:
|
||||||
|
dataId = fontListEntry.isSetRegular()
|
||||||
|
? fontListEntry.getRegular() : fontListEntry.addNewRegular();
|
||||||
|
break;
|
||||||
|
case Font.BOLD:
|
||||||
|
dataId = fontListEntry.isSetBold()
|
||||||
|
? fontListEntry.getBold() : fontListEntry.addNewBold();
|
||||||
|
break;
|
||||||
|
case Font.ITALIC:
|
||||||
|
dataId = fontListEntry.isSetItalic()
|
||||||
|
? fontListEntry.getItalic() : fontListEntry.addNewItalic();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
dataId = fontListEntry.isSetBoldItalic()
|
||||||
|
? fontListEntry.getBoldItalic() : fontListEntry.addNewBoldItalic();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
XSLFFontFacet facet = new XSLFFontFacet(dataId);
|
||||||
|
facet.setFontData(is);
|
||||||
|
return facet;
|
||||||
|
}
|
||||||
|
|
||||||
|
private final class XSLFFontFacet implements FontFacet {
|
||||||
|
private final CTEmbeddedFontDataId fontEntry;
|
||||||
|
private final FontHeader header = new FontHeader();
|
||||||
|
|
||||||
|
private XSLFFontFacet(CTEmbeddedFontDataId fontEntry) {
|
||||||
|
this.fontEntry = fontEntry;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getWeight() {
|
||||||
|
init();
|
||||||
|
return header.getWeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isItalic() {
|
||||||
|
init();
|
||||||
|
return header.isItalic();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public XSLFFontData getFontData() {
|
||||||
|
return ppt.getRelationPartById(fontEntry.getId()).getDocumentPart();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setFontData(InputStream is) throws IOException {
|
||||||
|
final XSLFRelation fntRel = XSLFRelation.FONT;
|
||||||
|
final String relId = fontEntry.getId();
|
||||||
|
final XSLFFontData fntData;
|
||||||
|
if (relId == null || relId.isEmpty()) {
|
||||||
|
final int fntDataIdx;
|
||||||
|
try {
|
||||||
|
fntDataIdx = ppt.getPackage().getUnusedPartIndex(fntRel.getDefaultFileName());
|
||||||
|
} catch (InvalidFormatException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
POIXMLDocumentPart.RelationPart rp = ppt.createRelationship(fntRel, XSLFFactory.getInstance(), fntDataIdx, false);
|
||||||
|
fntData = rp.getDocumentPart();
|
||||||
|
fontEntry.setId(rp.getRelationship().getId());
|
||||||
|
} else {
|
||||||
|
fntData = (XSLFFontData)ppt.getRelationById(relId);
|
||||||
|
}
|
||||||
|
|
||||||
|
assert (fntData != null);
|
||||||
|
try (OutputStream os = fntData.getOutputStream()) {
|
||||||
|
IOUtils.copy(is, os);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void init() {
|
||||||
|
if (header.getFamilyName() == null) {
|
||||||
|
try (InputStream is = getFontData().getInputStream()) {
|
||||||
|
byte[] buf = IOUtils.toByteArray(is, 1000);
|
||||||
|
header.init(buf, 0, buf.length);
|
||||||
|
} catch (IOException e) {
|
||||||
|
// TODO: better exception class
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private CTTextFont getFont() {
|
||||||
|
return fontListEntry.getFont();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds or updates a (MTX-) font
|
||||||
|
* @param ppt the slideshow which will contain the font
|
||||||
|
* @param fontStream the (MTX) font data as stream
|
||||||
|
* @return a font data object
|
||||||
|
* @throws IOException if the font data can't be stored
|
||||||
|
*
|
||||||
|
* @since POI 4.1.0
|
||||||
|
*/
|
||||||
|
public static XSLFFontInfo addFontToSlideShow(XMLSlideShow ppt, InputStream fontStream)
|
||||||
|
throws IOException {
|
||||||
|
FontHeader header = new FontHeader();
|
||||||
|
InputStream is = header.bufferInit(fontStream);
|
||||||
|
|
||||||
|
XSLFFontInfo fontInfo = new XSLFFontInfo(ppt, header.getFamilyName());
|
||||||
|
fontInfo.addFacet(is);
|
||||||
|
return fontInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return all registered fonts
|
||||||
|
* @param ppt the slideshow containing the fonts
|
||||||
|
* @return the list of registered fonts
|
||||||
|
*/
|
||||||
|
public static List<XSLFFontInfo> getFonts(XMLSlideShow ppt) {
|
||||||
|
final CTPresentation pres = ppt.getCTPresentation();
|
||||||
|
|
||||||
|
//noinspection deprecation
|
||||||
|
return pres.isSetEmbeddedFontLst()
|
||||||
|
? Stream.of(pres.getEmbeddedFontLst().getEmbeddedFontArray())
|
||||||
|
.map(fe -> new XSLFFontInfo(ppt, fe)).collect(Collectors.toList())
|
||||||
|
: Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -260,6 +260,13 @@ public final class XSLFRelation extends POIXMLRelation {
|
||||||
XSLFObjectData.class
|
XSLFObjectData.class
|
||||||
);
|
);
|
||||||
|
|
||||||
|
public static final XSLFRelation FONT = new XSLFRelation(
|
||||||
|
"application/x-fontdata",
|
||||||
|
"http://schemas.openxmlformats.org/officeDocument/2006/relationships/font",
|
||||||
|
"/ppt/fonts/font#.fntdata",
|
||||||
|
XSLFFontData.class
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
private XSLFRelation(String type, String rel, String defaultName, Class<? extends POIXMLDocumentPart> cls) {
|
private XSLFRelation(String type, String rel, String defaultName, Class<? extends POIXMLDocumentPart> cls) {
|
||||||
super(type, rel, defaultName, cls);
|
super(type, rel, defaultName, cls);
|
||||||
|
|
|
@ -28,6 +28,7 @@ import org.apache.poi.openxml4j.opc.PackagePart;
|
||||||
import org.apache.poi.sl.draw.DrawPaint;
|
import org.apache.poi.sl.draw.DrawPaint;
|
||||||
import org.apache.poi.sl.usermodel.PaintStyle;
|
import org.apache.poi.sl.usermodel.PaintStyle;
|
||||||
import org.apache.poi.sl.usermodel.PaintStyle.SolidPaint;
|
import org.apache.poi.sl.usermodel.PaintStyle.SolidPaint;
|
||||||
|
import org.apache.poi.sl.usermodel.TextParagraph;
|
||||||
import org.apache.poi.sl.usermodel.TextRun;
|
import org.apache.poi.sl.usermodel.TextRun;
|
||||||
import org.apache.poi.util.Beta;
|
import org.apache.poi.util.Beta;
|
||||||
import org.apache.poi.util.POILogFactory;
|
import org.apache.poi.util.POILogFactory;
|
||||||
|
@ -628,16 +629,6 @@ public class XSLFTextRun implements TextRun {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Integer getIndex() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setIndex(int index) {
|
|
||||||
throw new UnsupportedOperationException("setIndex not supported by XSLFFontInfo.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getTypeface() {
|
public String getTypeface() {
|
||||||
CTTextFont tf = getXmlObject(false);
|
CTTextFont tf = getXmlObject(false);
|
||||||
|
@ -829,4 +820,9 @@ public class XSLFTextRun implements TextRun {
|
||||||
return font;
|
return font;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public XSLFTextParagraph getParagraph() {
|
||||||
|
return _p;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,7 +59,7 @@ public class TestXSLFPowerPointExtractor {
|
||||||
|
|
||||||
// Check Basics
|
// Check Basics
|
||||||
assertStartsWith(text, "Lorem ipsum dolor sit amet\n");
|
assertStartsWith(text, "Lorem ipsum dolor sit amet\n");
|
||||||
assertContains(text, "amet\n\n");
|
assertContains(text, "amet\n");
|
||||||
|
|
||||||
// Our placeholder master text
|
// Our placeholder master text
|
||||||
// This shouldn't show up in the output
|
// This shouldn't show up in the output
|
||||||
|
@ -96,7 +96,7 @@ public class TestXSLFPowerPointExtractor {
|
||||||
extractor.setSlidesByDefault(false);
|
extractor.setSlidesByDefault(false);
|
||||||
extractor.setNotesByDefault(true);
|
extractor.setNotesByDefault(true);
|
||||||
text = extractor.getText();
|
text = extractor.getText();
|
||||||
assertEquals("\n\n1\n\n\n2\n", text);
|
assertEquals("\n1\n\n2\n", text);
|
||||||
|
|
||||||
// Both
|
// Both
|
||||||
extractor.setSlidesByDefault(true);
|
extractor.setSlidesByDefault(true);
|
||||||
|
@ -105,14 +105,14 @@ public class TestXSLFPowerPointExtractor {
|
||||||
String bothText =
|
String bothText =
|
||||||
"Lorem ipsum dolor sit amet\n" +
|
"Lorem ipsum dolor sit amet\n" +
|
||||||
"Nunc at risus vel erat tempus posuere. Aenean non ante.\n" +
|
"Nunc at risus vel erat tempus posuere. Aenean non ante.\n" +
|
||||||
"\n\n\n1\n" +
|
"\n\n1\n" +
|
||||||
"Lorem ipsum dolor sit amet\n" +
|
"Lorem ipsum dolor sit amet\n" +
|
||||||
"Lorem\n" +
|
"Lorem\n" +
|
||||||
"ipsum\n" +
|
"ipsum\n" +
|
||||||
"dolor\n" +
|
"dolor\n" +
|
||||||
"sit\n" +
|
"sit\n" +
|
||||||
"amet\n" +
|
"amet\n" +
|
||||||
"\n\n\n2\n";
|
"\n\n2\n";
|
||||||
assertEquals(bothText, text);
|
assertEquals(bothText, text);
|
||||||
|
|
||||||
// With Slides and Master Text
|
// With Slides and Master Text
|
||||||
|
@ -141,21 +141,21 @@ public class TestXSLFPowerPointExtractor {
|
||||||
String snmText =
|
String snmText =
|
||||||
"Lorem ipsum dolor sit amet\n" +
|
"Lorem ipsum dolor sit amet\n" +
|
||||||
"Nunc at risus vel erat tempus posuere. Aenean non ante.\n" +
|
"Nunc at risus vel erat tempus posuere. Aenean non ante.\n" +
|
||||||
"\n\n\n1\n" +
|
"\n\n1\n" +
|
||||||
"Lorem ipsum dolor sit amet\n" +
|
"Lorem ipsum dolor sit amet\n" +
|
||||||
"Lorem\n" +
|
"Lorem\n" +
|
||||||
"ipsum\n" +
|
"ipsum\n" +
|
||||||
"dolor\n" +
|
"dolor\n" +
|
||||||
"sit\n" +
|
"sit\n" +
|
||||||
"amet\n" +
|
"amet\n" +
|
||||||
"\n\n\n2\n";
|
"\n\n2\n";
|
||||||
assertEquals(snmText, text);
|
assertEquals(snmText, text);
|
||||||
|
|
||||||
// Via set defaults
|
// Via set defaults
|
||||||
extractor.setSlidesByDefault(false);
|
extractor.setSlidesByDefault(false);
|
||||||
extractor.setNotesByDefault(true);
|
extractor.setNotesByDefault(true);
|
||||||
text = extractor.getText();
|
text = extractor.getText();
|
||||||
assertEquals("\n\n1\n\n\n2\n", text);
|
assertEquals("\n1\n\n2\n", text);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -38,7 +38,7 @@ import org.openxmlformats.schemas.presentationml.x2006.main.CTNotesMasterIdListE
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdListEntry;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideIdListEntry;
|
||||||
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideMasterIdListEntry;
|
import org.openxmlformats.schemas.presentationml.x2006.main.CTSlideMasterIdListEntry;
|
||||||
|
|
||||||
public class TestXMLSlideShow extends BaseTestSlideShow {
|
public class TestXMLSlideShow extends BaseTestSlideShow<XSLFShape,XSLFTextParagraph> {
|
||||||
private OPCPackage pack;
|
private OPCPackage pack;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -81,6 +81,7 @@ public class TestXMLSlideShow extends BaseTestSlideShow {
|
||||||
xml.close();
|
xml.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("deprecation")
|
||||||
@Test
|
@Test
|
||||||
public void testSlideBasics() throws IOException {
|
public void testSlideBasics() throws IOException {
|
||||||
XMLSlideShow xml = new XMLSlideShow(pack);
|
XMLSlideShow xml = new XMLSlideShow(pack);
|
||||||
|
@ -136,7 +137,7 @@ public class TestXMLSlideShow extends BaseTestSlideShow {
|
||||||
assertEquals(0, xml.getProperties().getExtendedProperties().getUnderlyingProperties().getCharacters());
|
assertEquals(0, xml.getProperties().getExtendedProperties().getUnderlyingProperties().getCharacters());
|
||||||
assertEquals(0, xml.getProperties().getExtendedProperties().getUnderlyingProperties().getLines());
|
assertEquals(0, xml.getProperties().getExtendedProperties().getUnderlyingProperties().getLines());
|
||||||
|
|
||||||
assertEquals(null, xml.getProperties().getCoreProperties().getTitle());
|
assertNull(xml.getProperties().getCoreProperties().getTitle());
|
||||||
assertFalse(xml.getProperties().getCoreProperties().getUnderlyingProperties().getSubjectProperty().isPresent());
|
assertFalse(xml.getProperties().getCoreProperties().getUnderlyingProperties().getSubjectProperty().isPresent());
|
||||||
|
|
||||||
xml.close();
|
xml.close();
|
||||||
|
@ -147,7 +148,7 @@ public class TestXMLSlideShow extends BaseTestSlideShow {
|
||||||
// Default sample file has none
|
// Default sample file has none
|
||||||
XMLSlideShow xml = new XMLSlideShow(pack);
|
XMLSlideShow xml = new XMLSlideShow(pack);
|
||||||
|
|
||||||
assertEquals(null, xml.getCommentAuthors());
|
assertNull(xml.getCommentAuthors());
|
||||||
|
|
||||||
for (XSLFSlide slide : xml.getSlides()) {
|
for (XSLFSlide slide : xml.getSlides()) {
|
||||||
assertTrue(slide.getComments().isEmpty());
|
assertTrue(slide.getComments().isEmpty());
|
||||||
|
@ -186,11 +187,8 @@ public class TestXMLSlideShow extends BaseTestSlideShow {
|
||||||
xml.close();
|
xml.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
public SlideShow<?, ?> reopen(SlideShow<?, ?> show) {
|
@Override
|
||||||
return reopen((XMLSlideShow)show);
|
public XMLSlideShow reopen(SlideShow<XSLFShape,XSLFTextParagraph> show) {
|
||||||
}
|
|
||||||
|
|
||||||
private static XMLSlideShow reopen(XMLSlideShow show) {
|
|
||||||
try {
|
try {
|
||||||
BufAccessBAOS bos = new BufAccessBAOS();
|
BufAccessBAOS bos = new BufAccessBAOS();
|
||||||
show.write(bos);
|
show.write(bos);
|
||||||
|
|
|
@ -87,9 +87,9 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
* Master Slides
|
* Master Slides
|
||||||
*/
|
*/
|
||||||
public SlideListWithText getMasterSlideListWithText() {
|
public SlideListWithText getMasterSlideListWithText() {
|
||||||
for (int i = 0; i < slwts.length; i++) {
|
for (SlideListWithText slwt : slwts) {
|
||||||
if(slwts[i].getInstance() == SlideListWithText.MASTER) {
|
if (slwt.getInstance() == SlideListWithText.MASTER) {
|
||||||
return slwts[i];
|
return slwt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -100,9 +100,9 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
* Slides, or null if there isn't one
|
* Slides, or null if there isn't one
|
||||||
*/
|
*/
|
||||||
public SlideListWithText getSlideSlideListWithText() {
|
public SlideListWithText getSlideSlideListWithText() {
|
||||||
for (int i = 0; i < slwts.length; i++) {
|
for (SlideListWithText slwt : slwts) {
|
||||||
if(slwts[i].getInstance() == SlideListWithText.SLIDES) {
|
if (slwt.getInstance() == SlideListWithText.SLIDES) {
|
||||||
return slwts[i];
|
return slwt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -112,9 +112,9 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
* notes, or null if there isn't one
|
* notes, or null if there isn't one
|
||||||
*/
|
*/
|
||||||
public SlideListWithText getNotesSlideListWithText() {
|
public SlideListWithText getNotesSlideListWithText() {
|
||||||
for (int i = 0; i < slwts.length; i++) {
|
for (SlideListWithText slwt : slwts) {
|
||||||
if(slwts[i].getInstance() == SlideListWithText.NOTES) {
|
if (slwt.getInstance() == SlideListWithText.NOTES) {
|
||||||
return slwts[i];
|
return slwt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
@ -124,7 +124,7 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
/**
|
/**
|
||||||
* Set things up, and find our more interesting children
|
* Set things up, and find our more interesting children
|
||||||
*/
|
*/
|
||||||
protected Document(byte[] source, int start, int len) {
|
/* package */ Document(byte[] source, int start, int len) {
|
||||||
// Grab the header
|
// Grab the header
|
||||||
_header = new byte[8];
|
_header = new byte[8];
|
||||||
System.arraycopy(source,start,_header,0,8);
|
System.arraycopy(source,start,_header,0,8);
|
||||||
|
@ -186,7 +186,7 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
// The new SlideListWithText should go in
|
// The new SlideListWithText should go in
|
||||||
// just before the EndDocumentRecord
|
// just before the EndDocumentRecord
|
||||||
Record endDoc = _children[_children.length - 1];
|
Record endDoc = _children[_children.length - 1];
|
||||||
if(endDoc.getRecordType() == RecordTypes.RoundTripCustomTableStyles12Atom.typeID) {
|
if(endDoc.getRecordType() == RecordTypes.RoundTripCustomTableStyles12.typeID) {
|
||||||
// last record can optionally be a RoundTripCustomTableStyles12Atom
|
// last record can optionally be a RoundTripCustomTableStyles12Atom
|
||||||
endDoc = _children[_children.length - 2];
|
endDoc = _children[_children.length - 2];
|
||||||
}
|
}
|
||||||
|
@ -213,7 +213,7 @@ public final class Document extends PositionDependentRecordContainer
|
||||||
removeChild(slwt);
|
removeChild(slwt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
slwts = lst.toArray(new SlideListWithText[lst.size()]);
|
slwts = lst.toArray(new SlideListWithText[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -17,25 +17,25 @@
|
||||||
|
|
||||||
package org.apache.poi.hslf.record;
|
package org.apache.poi.hslf.record;
|
||||||
|
|
||||||
import org.apache.poi.util.IOUtils;
|
|
||||||
import org.apache.poi.util.LittleEndian;
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import org.apache.poi.util.IOUtils;
|
||||||
|
import org.apache.poi.util.LittleEndianByteArrayInputStream;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A Document Atom (type 1001). Holds misc information on the PowerPoint
|
* A Document Atom (type 1001). Holds misc information on the PowerPoint
|
||||||
* document, lots of them size and scale related.
|
* document, lots of them size and scale related.
|
||||||
*
|
|
||||||
* @author Nick Burch
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@SuppressWarnings({"WeakerAccess", "unused"})
|
||||||
public final class DocumentAtom extends RecordAtom
|
public final class DocumentAtom extends RecordAtom
|
||||||
{
|
{
|
||||||
//arbitrarily selected; may need to increase
|
//arbitrarily selected; may need to increase
|
||||||
private static final int MAX_RECORD_LENGTH = 1_000_000;
|
private static final int MAX_RECORD_LENGTH = 1_000_000;
|
||||||
|
|
||||||
private byte[] _header;
|
private final byte[] _header = new byte[8];
|
||||||
private static long _type = 1001l;
|
private static long _type = RecordTypes.DocumentAtom.typeID;
|
||||||
|
|
||||||
private long slideSizeX; // PointAtom, assume 1st 4 bytes = X
|
private long slideSizeX; // PointAtom, assume 1st 4 bytes = X
|
||||||
private long slideSizeY; // PointAtom, assume 2nd 4 bytes = Y
|
private long slideSizeY; // PointAtom, assume 2nd 4 bytes = Y
|
||||||
|
@ -87,6 +87,11 @@ public final class DocumentAtom extends RecordAtom
|
||||||
return saveWithFonts != 0;
|
return saveWithFonts != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Set the font embedding state */
|
||||||
|
public void setSaveWithFonts(boolean saveWithFonts) {
|
||||||
|
this.saveWithFonts = (byte)(saveWithFonts ? 1 : 0);
|
||||||
|
}
|
||||||
|
|
||||||
/** Have the placeholders on the title slide been omitted? */
|
/** Have the placeholders on the title slide been omitted? */
|
||||||
public boolean getOmitTitlePlace() {
|
public boolean getOmitTitlePlace() {
|
||||||
return omitTitlePlace != 0;
|
return omitTitlePlace != 0;
|
||||||
|
@ -108,41 +113,41 @@ public final class DocumentAtom extends RecordAtom
|
||||||
/**
|
/**
|
||||||
* For the Document Atom
|
* For the Document Atom
|
||||||
*/
|
*/
|
||||||
protected DocumentAtom(byte[] source, int start, int len) {
|
/* package */ DocumentAtom(byte[] source, int start, int len) {
|
||||||
// Sanity Checking
|
final int maxLen = Math.max(len, 48);
|
||||||
if(len < 48) { len = 48; }
|
LittleEndianByteArrayInputStream leis =
|
||||||
|
new LittleEndianByteArrayInputStream(source, start, maxLen);
|
||||||
|
|
||||||
// Get the header
|
// Get the header
|
||||||
_header = new byte[8];
|
leis.readFully(_header);
|
||||||
System.arraycopy(source,start,_header,0,8);
|
|
||||||
|
|
||||||
// Get the sizes and zoom ratios
|
// Get the sizes and zoom ratios
|
||||||
slideSizeX = LittleEndian.getInt(source,start+0+8);
|
slideSizeX = leis.readInt();
|
||||||
slideSizeY = LittleEndian.getInt(source,start+4+8);
|
slideSizeY = leis.readInt();
|
||||||
notesSizeX = LittleEndian.getInt(source,start+8+8);
|
notesSizeX = leis.readInt();
|
||||||
notesSizeY = LittleEndian.getInt(source,start+12+8);
|
notesSizeY = leis.readInt();
|
||||||
serverZoomFrom = LittleEndian.getInt(source,start+16+8);
|
serverZoomFrom = leis.readInt();
|
||||||
serverZoomTo = LittleEndian.getInt(source,start+20+8);
|
serverZoomTo = leis.readInt();
|
||||||
|
|
||||||
// Get the master persists
|
// Get the master persists
|
||||||
notesMasterPersist = LittleEndian.getInt(source,start+24+8);
|
notesMasterPersist = leis.readInt();
|
||||||
handoutMasterPersist = LittleEndian.getInt(source,start+28+8);
|
handoutMasterPersist = leis.readInt();
|
||||||
|
|
||||||
// Get the ID of the first slide
|
// Get the ID of the first slide
|
||||||
firstSlideNum = LittleEndian.getShort(source,start+32+8);
|
firstSlideNum = leis.readShort();
|
||||||
|
|
||||||
// Get the slide size type
|
// Get the slide size type
|
||||||
slideSizeType = LittleEndian.getShort(source,start+34+8);
|
slideSizeType = leis.readShort();
|
||||||
|
|
||||||
// Get the booleans as bytes
|
// Get the booleans as bytes
|
||||||
saveWithFonts = source[start+36+8];
|
saveWithFonts = leis.readByte();
|
||||||
omitTitlePlace = source[start+37+8];
|
omitTitlePlace = leis.readByte();
|
||||||
rightToLeft = source[start+38+8];
|
rightToLeft = leis.readByte();
|
||||||
showComments = source[start+39+8];
|
showComments = leis.readByte();
|
||||||
|
|
||||||
// If there's any other bits of data, keep them about
|
// If there's any other bits of data, keep them about
|
||||||
reserved = IOUtils.safelyAllocate(len-40-8, MAX_RECORD_LENGTH);
|
reserved = IOUtils.safelyAllocate(maxLen-48, MAX_RECORD_LENGTH);
|
||||||
System.arraycopy(source,start+48,reserved,0,reserved.length);
|
leis.readFully(reserved);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -18,13 +18,19 @@
|
||||||
package org.apache.poi.hslf.record;
|
package org.apache.poi.hslf.record;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontHeader;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
||||||
import org.apache.poi.hslf.usermodel.HSLFFontInfo;
|
import org.apache.poi.hslf.usermodel.HSLFFontInfo;
|
||||||
import org.apache.poi.hslf.usermodel.HSLFFontInfoPredefined;
|
import org.apache.poi.hslf.usermodel.HSLFFontInfoPredefined;
|
||||||
|
import org.apache.poi.util.IOUtils;
|
||||||
import org.apache.poi.util.POILogger;
|
import org.apache.poi.util.POILogger;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -32,11 +38,12 @@ import org.apache.poi.util.POILogger;
|
||||||
* about all the fonts in the presentation.
|
* about all the fonts in the presentation.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
@SuppressWarnings("WeakerAccess")
|
||||||
public final class FontCollection extends RecordContainer {
|
public final class FontCollection extends RecordContainer {
|
||||||
private final Map<String,HSLFFontInfo> fonts = new LinkedHashMap<>();
|
private final Map<String,HSLFFontInfo> fonts = new LinkedHashMap<>();
|
||||||
private byte[] _header;
|
private byte[] _header;
|
||||||
|
|
||||||
protected FontCollection(byte[] source, int start, int len) {
|
/* package */ FontCollection(byte[] source, int start, int len) {
|
||||||
_header = new byte[8];
|
_header = new byte[8];
|
||||||
System.arraycopy(source,start,_header,0,8);
|
System.arraycopy(source,start,_header,0,8);
|
||||||
|
|
||||||
|
@ -44,8 +51,13 @@ public final class FontCollection extends RecordContainer {
|
||||||
|
|
||||||
for (Record r : _children){
|
for (Record r : _children){
|
||||||
if(r instanceof FontEntityAtom) {
|
if(r instanceof FontEntityAtom) {
|
||||||
HSLFFontInfo fi = new HSLFFontInfo((FontEntityAtom)r);
|
HSLFFontInfo fi = new HSLFFontInfo((FontEntityAtom) r);
|
||||||
fonts.put(fi.getTypeface(), fi);
|
fonts.put(fi.getTypeface(), fi);
|
||||||
|
} else if (r instanceof FontEmbeddedData) {
|
||||||
|
FontEmbeddedData fed = (FontEmbeddedData)r;
|
||||||
|
FontHeader fontHeader = fed.getFontHeader();
|
||||||
|
HSLFFontInfo fi = addFont(fontHeader);
|
||||||
|
fi.addFacet(fed);
|
||||||
} else {
|
} else {
|
||||||
logger.log(POILogger.WARN, "Warning: FontCollection child wasn't a FontEntityAtom, was " + r.getClass().getSimpleName());
|
logger.log(POILogger.WARN, "Warning: FontCollection child wasn't a FontEntityAtom, was " + r.getClass().getSimpleName());
|
||||||
}
|
}
|
||||||
|
@ -98,6 +110,58 @@ public final class FontCollection extends RecordContainer {
|
||||||
return fi;
|
return fi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public HSLFFontInfo addFont(InputStream fontData) throws IOException {
|
||||||
|
FontHeader fontHeader = new FontHeader();
|
||||||
|
InputStream is = fontHeader.bufferInit(fontData);
|
||||||
|
|
||||||
|
HSLFFontInfo fi = addFont(fontHeader);
|
||||||
|
|
||||||
|
// always overwrite the font info properties when a font data given
|
||||||
|
// as the font info properties are assigned generically when only a typeface is given
|
||||||
|
FontEntityAtom fea = fi.getFontEntityAtom();
|
||||||
|
assert (fea != null);
|
||||||
|
fea.setCharSet(fontHeader.getCharsetByte());
|
||||||
|
fea.setPitchAndFamily(FontPitch.getNativeId(fontHeader.getPitch(),fontHeader.getFamily()));
|
||||||
|
|
||||||
|
// always activate subsetting
|
||||||
|
fea.setFontFlags(1);
|
||||||
|
// true type font and no font substitution
|
||||||
|
fea.setFontType(12);
|
||||||
|
|
||||||
|
Record after = fea;
|
||||||
|
|
||||||
|
final int insertIdx = getFacetIndex(fontHeader.isItalic(), fontHeader.isBold());
|
||||||
|
|
||||||
|
FontEmbeddedData newChild = null;
|
||||||
|
for (FontEmbeddedData fed : fi.getFacets()) {
|
||||||
|
FontHeader fh = fed.getFontHeader();
|
||||||
|
final int curIdx = getFacetIndex(fh.isItalic(), fh.isBold());
|
||||||
|
|
||||||
|
if (curIdx == insertIdx) {
|
||||||
|
newChild = fed;
|
||||||
|
break;
|
||||||
|
} else if (curIdx > insertIdx) {
|
||||||
|
// the new facet needs to be inserted before the current facet
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
after = fed;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newChild == null) {
|
||||||
|
newChild = new FontEmbeddedData();
|
||||||
|
addChildAfter(newChild, after);
|
||||||
|
fi.addFacet(newChild);
|
||||||
|
}
|
||||||
|
|
||||||
|
newChild.setFontData(IOUtils.toByteArray(is));
|
||||||
|
return fi;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int getFacetIndex(boolean isItalic, boolean isBold) {
|
||||||
|
return (isItalic ? 2 : 0) | (isBold ? 1 : 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lookup a FontInfo object by its typeface
|
* Lookup a FontInfo object by its typeface
|
||||||
|
@ -132,4 +196,8 @@ public final class FontCollection extends RecordContainer {
|
||||||
public int getNumberOfFonts() {
|
public int getNumberOfFonts() {
|
||||||
return fonts.size();
|
return fonts.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public List<HSLFFontInfo> getFonts() {
|
||||||
|
return new ArrayList<>(fonts.values());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,116 @@
|
||||||
|
/* ====================================================================
|
||||||
|
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.hslf.record;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontFacet;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontHeader;
|
||||||
|
import org.apache.poi.util.IOUtils;
|
||||||
|
import org.apache.poi.util.LittleEndian;
|
||||||
|
|
||||||
|
@SuppressWarnings("WeakerAccess")
|
||||||
|
public class FontEmbeddedData extends RecordAtom implements FontFacet {
|
||||||
|
//arbitrarily selected; may need to increase
|
||||||
|
private static final int MAX_RECORD_LENGTH = 1_000_000;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Record header.
|
||||||
|
*/
|
||||||
|
private byte[] _header;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Record data - An EOT Font
|
||||||
|
*/
|
||||||
|
private byte[] _data;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a brand new font embedded record.
|
||||||
|
*/
|
||||||
|
/* package */ FontEmbeddedData() {
|
||||||
|
_header = new byte[8];
|
||||||
|
_data = new byte[4];
|
||||||
|
|
||||||
|
LittleEndian.putShort(_header, 2, (short)getRecordType());
|
||||||
|
LittleEndian.putInt(_header, 4, _data.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs the font embedded record from its source data.
|
||||||
|
*
|
||||||
|
* @param source the source data as a byte array.
|
||||||
|
* @param start the start offset into the byte array.
|
||||||
|
* @param len the length of the slice in the byte array.
|
||||||
|
*/
|
||||||
|
/* package */ FontEmbeddedData(byte[] source, int start, int len) {
|
||||||
|
// Get the header.
|
||||||
|
_header = new byte[8];
|
||||||
|
System.arraycopy(source,start,_header,0,8);
|
||||||
|
|
||||||
|
// Get the record data.
|
||||||
|
_data = IOUtils.safelyAllocate(len-8, MAX_RECORD_LENGTH);
|
||||||
|
System.arraycopy(source,start+8,_data,0,len-8);
|
||||||
|
|
||||||
|
// Must be at least 4 bytes long
|
||||||
|
if(_data.length < 4) {
|
||||||
|
throw new IllegalArgumentException("The length of the data for a ExObjListAtom must be at least 4 bytes, but was only " + _data.length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getRecordType() {
|
||||||
|
return RecordTypes.FontEmbeddedData.typeID;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeOut(OutputStream out) throws IOException {
|
||||||
|
out.write(_header);
|
||||||
|
out.write(_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFontData(byte[] fontData) {
|
||||||
|
_data = fontData.clone();
|
||||||
|
LittleEndian.putInt(_header, 4, _data.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
public FontHeader getFontHeader() {
|
||||||
|
FontHeader h = new FontHeader();
|
||||||
|
h.init(_data, 0, _data.length);
|
||||||
|
return h;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getWeight() {
|
||||||
|
return getFontHeader().getWeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isItalic() {
|
||||||
|
return getFontHeader().isItalic();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getTypeface() {
|
||||||
|
return getFontHeader().getFamilyName();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object getFontData() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
|
@ -43,7 +43,7 @@ public final class FontEntityAtom extends RecordAtom {
|
||||||
/**
|
/**
|
||||||
* record header
|
* record header
|
||||||
*/
|
*/
|
||||||
private byte[] _header;
|
private final byte[] _header = new byte[8];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* record data
|
* record data
|
||||||
|
@ -53,9 +53,8 @@ public final class FontEntityAtom extends RecordAtom {
|
||||||
/**
|
/**
|
||||||
* Build an instance of <code>FontEntityAtom</code> from on-disk data
|
* Build an instance of <code>FontEntityAtom</code> from on-disk data
|
||||||
*/
|
*/
|
||||||
protected FontEntityAtom(byte[] source, int start, int len) {
|
/* package */ FontEntityAtom(byte[] source, int start, int len) {
|
||||||
// Get the header
|
// Get the header
|
||||||
_header = new byte[8];
|
|
||||||
System.arraycopy(source,start,_header,0,8);
|
System.arraycopy(source,start,_header,0,8);
|
||||||
|
|
||||||
// Grab the record data
|
// Grab the record data
|
||||||
|
@ -69,7 +68,6 @@ public final class FontEntityAtom extends RecordAtom {
|
||||||
public FontEntityAtom() {
|
public FontEntityAtom() {
|
||||||
_recdata = new byte[68];
|
_recdata = new byte[68];
|
||||||
|
|
||||||
_header = new byte[8];
|
|
||||||
LittleEndian.putShort(_header, 2, (short)getRecordType());
|
LittleEndian.putShort(_header, 2, (short)getRecordType());
|
||||||
LittleEndian.putInt(_header, 4, _recdata.length);
|
LittleEndian.putInt(_header, 4, _recdata.length);
|
||||||
}
|
}
|
||||||
|
@ -108,7 +106,7 @@ public final class FontEntityAtom extends RecordAtom {
|
||||||
byte[] bytes = StringUtil.getToUnicodeLE(name);
|
byte[] bytes = StringUtil.getToUnicodeLE(name);
|
||||||
System.arraycopy(bytes, 0, _recdata, 0, bytes.length);
|
System.arraycopy(bytes, 0, _recdata, 0, bytes.length);
|
||||||
// null the remaining bytes
|
// null the remaining bytes
|
||||||
Arrays.fill(_recdata, 64-bytes.length, 64, (byte)0);
|
Arrays.fill(_recdata, bytes.length, 64, (byte)0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setFontIndex(int idx){
|
public void setFontIndex(int idx){
|
||||||
|
|
|
@ -62,7 +62,18 @@ public enum RecordTypes {
|
||||||
NamedShow(1041,null),
|
NamedShow(1041,null),
|
||||||
NamedShowSlides(1042,null),
|
NamedShowSlides(1042,null),
|
||||||
SheetProperties(1044,null),
|
SheetProperties(1044,null),
|
||||||
RoundTripCustomTableStyles12Atom(1064,null),
|
OriginalMainMasterId(1052,null),
|
||||||
|
CompositeMasterId(1052,null),
|
||||||
|
RoundTripContentMasterInfo12(1054,null),
|
||||||
|
RoundTripShapeId12(1055,null),
|
||||||
|
RoundTripHFPlaceholder12(1056,RoundTripHFPlaceholder12::new),
|
||||||
|
RoundTripContentMasterId(1058,null),
|
||||||
|
RoundTripOArtTextStyles12(1059,null),
|
||||||
|
RoundTripShapeCheckSumForCustomLayouts12(1062,null),
|
||||||
|
RoundTripNotesMasterTextStyles12(1063,null),
|
||||||
|
RoundTripCustomTableStyles12(1064,null),
|
||||||
|
|
||||||
|
|
||||||
List(2000,DocInfoListContainer::new),
|
List(2000,DocInfoListContainer::new),
|
||||||
FontCollection(2005,FontCollection::new),
|
FontCollection(2005,FontCollection::new),
|
||||||
BookmarkCollection(2019,null),
|
BookmarkCollection(2019,null),
|
||||||
|
@ -92,7 +103,7 @@ public enum RecordTypes {
|
||||||
DefaultRulerAtom(4011,null),
|
DefaultRulerAtom(4011,null),
|
||||||
StyleTextProp9Atom(4012, StyleTextProp9Atom::new), //0x0FAC RT_StyleTextProp9Atom
|
StyleTextProp9Atom(4012, StyleTextProp9Atom::new), //0x0FAC RT_StyleTextProp9Atom
|
||||||
FontEntityAtom(4023,FontEntityAtom::new),
|
FontEntityAtom(4023,FontEntityAtom::new),
|
||||||
FontEmbeddedData(4024,null),
|
FontEmbeddedData(4024,FontEmbeddedData::new),
|
||||||
CString(4026,CString::new),
|
CString(4026,CString::new),
|
||||||
MetaFile(4033,null),
|
MetaFile(4033,null),
|
||||||
ExOleObjAtom(4035,ExOleObjAtom::new),
|
ExOleObjAtom(4035,ExOleObjAtom::new),
|
||||||
|
@ -159,17 +170,6 @@ public enum RecordTypes {
|
||||||
// Records ~12050 seem to be related to Document Encryption
|
// Records ~12050 seem to be related to Document Encryption
|
||||||
DocumentEncryptionAtom(12052,DocumentEncryptionAtom::new),
|
DocumentEncryptionAtom(12052,DocumentEncryptionAtom::new),
|
||||||
|
|
||||||
OriginalMainMasterId(1052,null),
|
|
||||||
CompositeMasterId(1052,null),
|
|
||||||
RoundTripContentMasterInfo12(1054,null),
|
|
||||||
RoundTripShapeId12(1055,null),
|
|
||||||
RoundTripHFPlaceholder12(1056,RoundTripHFPlaceholder12::new),
|
|
||||||
RoundTripContentMasterId(1058,null),
|
|
||||||
RoundTripOArtTextStyles12(1059,null),
|
|
||||||
RoundTripShapeCheckSumForCustomLayouts12(1062,null),
|
|
||||||
RoundTripNotesMasterTextStyles12(1063,null),
|
|
||||||
RoundTripCustomTableStyles12(1064,null),
|
|
||||||
|
|
||||||
// records greater then 0xF000 belong to with Microsoft Office Drawing format also known as Escher
|
// records greater then 0xF000 belong to with Microsoft Office Drawing format also known as Escher
|
||||||
EscherDggContainer(0xF000,null),
|
EscherDggContainer(0xF000,null),
|
||||||
EscherDgg(0xf006,null),
|
EscherDgg(0xf006,null),
|
||||||
|
|
|
@ -17,13 +17,18 @@
|
||||||
|
|
||||||
package org.apache.poi.hslf.usermodel;
|
package org.apache.poi.hslf.usermodel;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
||||||
|
import org.apache.poi.hslf.record.FontEmbeddedData;
|
||||||
import org.apache.poi.hslf.record.FontEntityAtom;
|
import org.apache.poi.hslf.record.FontEntityAtom;
|
||||||
import org.apache.poi.util.BitField;
|
import org.apache.poi.util.BitField;
|
||||||
import org.apache.poi.util.BitFieldFactory;
|
import org.apache.poi.util.BitFieldFactory;
|
||||||
|
import org.apache.poi.util.Internal;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a Font used in a presentation.<p>
|
* Represents a Font used in a presentation.<p>
|
||||||
|
@ -32,6 +37,7 @@ import org.apache.poi.util.BitFieldFactory;
|
||||||
*
|
*
|
||||||
* @since POI 3.17-beta2
|
* @since POI 3.17-beta2
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings("WeakerAccess")
|
||||||
public class HSLFFontInfo implements FontInfo {
|
public class HSLFFontInfo implements FontInfo {
|
||||||
|
|
||||||
public enum FontRenderType {
|
public enum FontRenderType {
|
||||||
|
@ -53,6 +59,8 @@ public class HSLFFontInfo implements FontInfo {
|
||||||
private FontPitch pitch = FontPitch.VARIABLE;
|
private FontPitch pitch = FontPitch.VARIABLE;
|
||||||
private boolean isSubsetted;
|
private boolean isSubsetted;
|
||||||
private boolean isSubstitutable = true;
|
private boolean isSubstitutable = true;
|
||||||
|
private final List<FontEmbeddedData> facets = new ArrayList<>();
|
||||||
|
private FontEntityAtom fontEntityAtom;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a new instance of HSLFFontInfo with more or sensible defaults.<p>
|
* Creates a new instance of HSLFFontInfo with more or sensible defaults.<p>
|
||||||
|
@ -70,6 +78,7 @@ public class HSLFFontInfo implements FontInfo {
|
||||||
* Creates a new instance of HSLFFontInfo and initialize it from the supplied font atom
|
* Creates a new instance of HSLFFontInfo and initialize it from the supplied font atom
|
||||||
*/
|
*/
|
||||||
public HSLFFontInfo(FontEntityAtom fontAtom){
|
public HSLFFontInfo(FontEntityAtom fontAtom){
|
||||||
|
fontEntityAtom = fontAtom;
|
||||||
setIndex(fontAtom.getFontIndex());
|
setIndex(fontAtom.getFontIndex());
|
||||||
setTypeface(fontAtom.getFontName());
|
setTypeface(fontAtom.getFontName());
|
||||||
setCharset(FontCharset.valueOf(fontAtom.getCharSet()));
|
setCharset(FontCharset.valueOf(fontAtom.getCharSet()));
|
||||||
|
@ -187,7 +196,11 @@ public class HSLFFontInfo implements FontInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
public FontEntityAtom createRecord() {
|
public FontEntityAtom createRecord() {
|
||||||
|
assert(fontEntityAtom == null);
|
||||||
|
|
||||||
FontEntityAtom fnt = new FontEntityAtom();
|
FontEntityAtom fnt = new FontEntityAtom();
|
||||||
|
fontEntityAtom = fnt;
|
||||||
|
|
||||||
fnt.setFontIndex(getIndex() << 4);
|
fnt.setFontIndex(getIndex() << 4);
|
||||||
fnt.setFontName(getTypeface());
|
fnt.setFontName(getTypeface());
|
||||||
fnt.setCharSet(getCharset().getNativeId());
|
fnt.setCharSet(getCharset().getNativeId());
|
||||||
|
@ -212,4 +225,18 @@ public class HSLFFontInfo implements FontInfo {
|
||||||
fnt.setPitchAndFamily(FontPitch.getNativeId(pitch, family));
|
fnt.setPitchAndFamily(FontPitch.getNativeId(pitch, family));
|
||||||
return fnt;
|
return fnt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void addFacet(FontEmbeddedData facet) {
|
||||||
|
facets.add(facet);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<FontEmbeddedData> getFacets() {
|
||||||
|
return facets;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Internal
|
||||||
|
public FontEntityAtom getFontEntityAtom() {
|
||||||
|
return fontEntityAtom;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,53 +45,23 @@ public enum HSLFFontInfoPredefined implements FontInfo {
|
||||||
this.family = family;
|
this.family = family;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Integer getIndex() {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setIndex(int index) {
|
|
||||||
throw new UnsupportedOperationException("Predefined enum can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getTypeface() {
|
public String getTypeface() {
|
||||||
return typeface;
|
return typeface;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setTypeface(String typeface) {
|
|
||||||
throw new UnsupportedOperationException("Predefined enum can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FontCharset getCharset() {
|
public FontCharset getCharset() {
|
||||||
return charset;
|
return charset;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setCharset(FontCharset charset) {
|
|
||||||
throw new UnsupportedOperationException("Predefined enum can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FontFamily getFamily() {
|
public FontFamily getFamily() {
|
||||||
return family;
|
return family;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFamily(FontFamily family) {
|
|
||||||
throw new UnsupportedOperationException("Predefined enum can't be changed.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FontPitch getPitch() {
|
public FontPitch getPitch() {
|
||||||
return pitch;
|
return pitch;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setPitch(FontPitch pitch) {
|
|
||||||
throw new UnsupportedOperationException("Predefined enum can't be changed.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,7 +50,6 @@ import org.apache.poi.poifs.filesystem.Ole10Native;
|
||||||
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
|
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
|
||||||
import org.apache.poi.sl.usermodel.MasterSheet;
|
import org.apache.poi.sl.usermodel.MasterSheet;
|
||||||
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
||||||
import org.apache.poi.sl.usermodel.Resources;
|
|
||||||
import org.apache.poi.sl.usermodel.SlideShow;
|
import org.apache.poi.sl.usermodel.SlideShow;
|
||||||
import org.apache.poi.util.IOUtils;
|
import org.apache.poi.util.IOUtils;
|
||||||
import org.apache.poi.util.Internal;
|
import org.apache.poi.util.Internal;
|
||||||
|
@ -891,6 +890,21 @@ public final class HSLFSlideShow implements SlideShow<HSLFShape,HSLFTextParagrap
|
||||||
return getDocumentRecord().getEnvironment().getFontCollection().addFont(fontInfo);
|
return getDocumentRecord().getEnvironment().getFontCollection().addFont(fontInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a font in this presentation and also embed its font data
|
||||||
|
*
|
||||||
|
* @param fontData the EOT font data as stream
|
||||||
|
*
|
||||||
|
* @return the registered HSLFFontInfo - the font info object is unique based on the typeface
|
||||||
|
*
|
||||||
|
* @since POI 4.1.0
|
||||||
|
*/
|
||||||
|
public HSLFFontInfo addFont(InputStream fontData) throws IOException {
|
||||||
|
Document doc = getDocumentRecord();
|
||||||
|
doc.getDocumentAtom().setSaveWithFonts(true);
|
||||||
|
return doc.getEnvironment().getFontCollection().addFont(fontData);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a font by index
|
* Get a font by index
|
||||||
*
|
*
|
||||||
|
@ -912,6 +926,11 @@ public final class HSLFSlideShow implements SlideShow<HSLFShape,HSLFTextParagrap
|
||||||
return getDocumentRecord().getEnvironment().getFontCollection().getNumberOfFonts();
|
return getDocumentRecord().getEnvironment().getFontCollection().getNumberOfFonts();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<HSLFFontInfo> getFonts() {
|
||||||
|
return getDocumentRecord().getEnvironment().getFontCollection().getFonts();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return Header / Footer settings for slides
|
* Return Header / Footer settings for slides
|
||||||
*
|
*
|
||||||
|
@ -1127,12 +1146,6 @@ public final class HSLFSlideShow implements SlideShow<HSLFShape,HSLFTextParagrap
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Resources getResources() {
|
|
||||||
// TODO implement or throw exception if not supported
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the handler class which holds the hslf records
|
* @return the handler class which holds the hslf records
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -44,8 +44,9 @@ import org.apache.poi.util.POILogger;
|
||||||
* Represents a run of text, all with the same style
|
* Represents a run of text, all with the same style
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings({"WeakerAccess", "Duplicates", "unused"})
|
||||||
public final class HSLFTextRun implements TextRun {
|
public final class HSLFTextRun implements TextRun {
|
||||||
protected POILogger logger = POILogFactory.getLogger(this.getClass());
|
private static final POILogger logger = POILogFactory.getLogger(HSLFTextRun.class);
|
||||||
|
|
||||||
/** The TextRun we belong to */
|
/** The TextRun we belong to */
|
||||||
private HSLFTextParagraph parentParagraph;
|
private HSLFTextParagraph parentParagraph;
|
||||||
|
@ -132,17 +133,17 @@ public final class HSLFTextRun implements TextRun {
|
||||||
return getFlag(index);
|
return getFlag(index);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected boolean getFlag(int index) {
|
boolean getFlag(int index) {
|
||||||
BitMaskTextProp prop = (characterStyle == null) ? null : characterStyle.findByName(CharFlagsTextProp.NAME);
|
BitMaskTextProp prop = (characterStyle == null) ? null : characterStyle.findByName(CharFlagsTextProp.NAME);
|
||||||
|
|
||||||
if (prop == null || !prop.getSubPropMatches()[index]) {
|
if (prop == null || !prop.getSubPropMatches()[index]) {
|
||||||
prop = getMasterProp(CharFlagsTextProp.NAME);
|
prop = getMasterProp();
|
||||||
}
|
}
|
||||||
|
|
||||||
return prop == null ? false : prop.getSubValue(index);
|
return prop != null && prop.getSubValue(index);
|
||||||
}
|
}
|
||||||
|
|
||||||
private <T extends TextProp> T getMasterProp(final String name) {
|
private <T extends TextProp> T getMasterProp() {
|
||||||
final int txtype = parentParagraph.getRunType();
|
final int txtype = parentParagraph.getRunType();
|
||||||
final HSLFSheet sheet = parentParagraph.getSheet();
|
final HSLFSheet sheet = parentParagraph.getSheet();
|
||||||
if (sheet == null) {
|
if (sheet == null) {
|
||||||
|
@ -156,6 +157,7 @@ public final class HSLFTextRun implements TextRun {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String name = CharFlagsTextProp.NAME;
|
||||||
final TextPropCollection col = master.getPropCollection(txtype, parentParagraph.getIndentLevel(), name, true);
|
final TextPropCollection col = master.getPropCollection(txtype, parentParagraph.getIndentLevel(), name, true);
|
||||||
return (col == null) ? null : col.findByName(name);
|
return (col == null) ? null : col.findByName(name);
|
||||||
}
|
}
|
||||||
|
@ -302,7 +304,7 @@ public final class HSLFTextRun implements TextRun {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setFontFamily(String typeface) {
|
public void setFontFamily(String typeface) {
|
||||||
setFontInfo(new HSLFFontInfo(typeface), FontGroup.LATIN);
|
setFontFamily(typeface, FontGroup.LATIN);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -330,7 +332,7 @@ public final class HSLFTextRun implements TextRun {
|
||||||
switch (fg) {
|
switch (fg) {
|
||||||
default:
|
default:
|
||||||
case LATIN:
|
case LATIN:
|
||||||
propName = "font.index";
|
propName = "ansi.font.index";
|
||||||
break;
|
break;
|
||||||
case COMPLEX_SCRIPT:
|
case COMPLEX_SCRIPT:
|
||||||
// TODO: implement TextCFException10 structure
|
// TODO: implement TextCFException10 structure
|
||||||
|
@ -350,6 +352,7 @@ public final class HSLFTextRun implements TextRun {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
setCharTextPropVal("font.index", fontIdx);
|
||||||
setCharTextPropVal(propName, fontIdx);
|
setCharTextPropVal(propName, fontIdx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -435,8 +438,8 @@ public final class HSLFTextRun implements TextRun {
|
||||||
setFontColor(rgb);
|
setFontColor(rgb);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void setFlag(int index, boolean value) {
|
private void setFlag(int index, boolean value) {
|
||||||
BitMaskTextProp prop = (BitMaskTextProp)characterStyle.addWithName(CharFlagsTextProp.NAME);
|
BitMaskTextProp prop = characterStyle.addWithName(CharFlagsTextProp.NAME);
|
||||||
prop.setSubValue(value, index);
|
prop.setSubValue(value, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -469,7 +472,7 @@ public final class HSLFTextRun implements TextRun {
|
||||||
*
|
*
|
||||||
* @param link the hyperlink
|
* @param link the hyperlink
|
||||||
*/
|
*/
|
||||||
protected void setHyperlink(HSLFHyperlink link) {
|
/* package */ void setHyperlink(HSLFHyperlink link) {
|
||||||
this.link = link;
|
this.link = link;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -521,4 +524,9 @@ public final class HSLFTextRun implements TextRun {
|
||||||
private FontGroup safeFontGroup(FontGroup fontGroup) {
|
private FontGroup safeFontGroup(FontGroup fontGroup) {
|
||||||
return (fontGroup != null) ? fontGroup : FontGroup.getFontGroupFirst(getRawText());
|
return (fontGroup != null) ? fontGroup : FontGroup.getFontGroupFirst(getRawText());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public HSLFTextParagraph getParagraph() {
|
||||||
|
return parentParagraph;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,6 +22,7 @@ import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
import org.apache.poi.common.usermodel.fonts.FontCharset;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
import org.apache.poi.common.usermodel.fonts.FontFamily;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontHeader;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
import org.apache.poi.common.usermodel.fonts.FontPitch;
|
||||||
import org.apache.poi.util.BitField;
|
import org.apache.poi.util.BitField;
|
||||||
|
@ -32,6 +33,7 @@ import org.apache.poi.util.LittleEndianInputStream;
|
||||||
/**
|
/**
|
||||||
* The Font object specifies the attributes of a logical font
|
* The Font object specifies the attributes of a logical font
|
||||||
*/
|
*/
|
||||||
|
@SuppressWarnings({"unused", "Duplicates"})
|
||||||
public class HwmfFont implements FontInfo {
|
public class HwmfFont implements FontInfo {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -289,7 +291,7 @@ public class HwmfFont implements FontInfo {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* An 8-bit unsigned integer that defines the character set.
|
* An 8-bit unsigned integer that defines the character set.
|
||||||
* It SHOULD be set to a value in the {@link WmfCharset} Enumeration.
|
* It SHOULD be set to a value in the {@link FontCharset} Enumeration.
|
||||||
*
|
*
|
||||||
* The DEFAULT_CHARSET value MAY be used to allow the name and size of a font to fully
|
* The DEFAULT_CHARSET value MAY be used to allow the name and size of a font to fully
|
||||||
* describe the logical font. If the specified font name does not exist, a font in another character
|
* describe the logical font. If the specified font name does not exist, a font in another character
|
||||||
|
@ -373,7 +375,7 @@ public class HwmfFont implements FontInfo {
|
||||||
height = -12;
|
height = -12;
|
||||||
width = 0;
|
width = 0;
|
||||||
escapement = 0;
|
escapement = 0;
|
||||||
weight = 400;
|
weight = FontHeader.REGULAR_WEIGHT;
|
||||||
italic = false;
|
italic = false;
|
||||||
underline = false;
|
underline = false;
|
||||||
strikeOut = false;
|
strikeOut = false;
|
||||||
|
@ -437,51 +439,21 @@ public class HwmfFont implements FontInfo {
|
||||||
return FontFamily.valueOf(pitchAndFamily & 0xF);
|
return FontFamily.valueOf(pitchAndFamily & 0xF);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFamily(FontFamily family) {
|
|
||||||
throw new UnsupportedOperationException("setCharset not supported by HwmfFont.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FontPitch getPitch() {
|
public FontPitch getPitch() {
|
||||||
return FontPitch.valueOf((pitchAndFamily >>> 6) & 3);
|
return FontPitch.valueOf((pitchAndFamily >>> 6) & 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setPitch(FontPitch pitch) {
|
|
||||||
throw new UnsupportedOperationException("setPitch not supported by HwmfFont.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Integer getIndex() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setIndex(int index) {
|
|
||||||
throw new UnsupportedOperationException("setIndex not supported by HwmfFont.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getTypeface() {
|
public String getTypeface() {
|
||||||
return facename;
|
return facename;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setTypeface(String typeface) {
|
|
||||||
throw new UnsupportedOperationException("setTypeface not supported by HwmfFont.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FontCharset getCharset() {
|
public FontCharset getCharset() {
|
||||||
return charSet;
|
return charSet;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setCharset(FontCharset charset) {
|
|
||||||
throw new UnsupportedOperationException("setCharset not supported by HwmfFont.");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return "{ height: "+height+
|
return "{ height: "+height+
|
||||||
|
|
|
@ -29,7 +29,9 @@ import java.io.ByteArrayOutputStream;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
import java.util.BitSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import org.apache.poi.POIDataSamples;
|
import org.apache.poi.POIDataSamples;
|
||||||
import org.apache.poi.hslf.usermodel.HSLFObjectShape;
|
import org.apache.poi.hslf.usermodel.HSLFObjectShape;
|
||||||
|
@ -52,12 +54,22 @@ public final class TestExtractor {
|
||||||
/**
|
/**
|
||||||
* Extractor primed on the 2 page basic test data
|
* Extractor primed on the 2 page basic test data
|
||||||
*/
|
*/
|
||||||
private static final String expectText = "This is a test title\nThis is a test subtitle\nThis is on page 1\nThis is the title on page 2\nThis is page two\nIt has several blocks of text\nNone of them have formatting\n";
|
private static final String EXPECTED_PAGE1 =
|
||||||
|
"This is a test title\n" +
|
||||||
|
"This is a test subtitle\n\n" +
|
||||||
|
"This is on page 1\n";
|
||||||
|
|
||||||
/**
|
private static final String EXPECTED_PAGE2 =
|
||||||
* Extractor primed on the 1 page but text-box'd test data
|
"This is the title on page 2\n" +
|
||||||
*/
|
"This is page two\n\n" +
|
||||||
private static final String expectText2 = "Hello, World!!!\nI am just a poor boy\nThis is Times New Roman\nPlain Text \n";
|
"It has several blocks of text\n\n" +
|
||||||
|
"None of them have formatting\n";
|
||||||
|
|
||||||
|
private static final String NOTES_PAGE1 =
|
||||||
|
"\nThese are the notes for page 1\n";
|
||||||
|
|
||||||
|
private static final String NOTES_PAGE2 =
|
||||||
|
"\nThese are the notes on page two, again lacking formatting\n";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Where our embeded files live
|
* Where our embeded files live
|
||||||
|
@ -75,9 +87,16 @@ public final class TestExtractor {
|
||||||
public void testReadSheetText() throws IOException {
|
public void testReadSheetText() throws IOException {
|
||||||
// Basic 2 page example
|
// Basic 2 page example
|
||||||
try (SlideShowExtractor ppe = openExtractor("basic_test_ppt_file.ppt")) {
|
try (SlideShowExtractor ppe = openExtractor("basic_test_ppt_file.ppt")) {
|
||||||
assertEquals(expectText, ppe.getText());
|
assertEquals(EXPECTED_PAGE1+EXPECTED_PAGE2, ppe.getText());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Extractor primed on the 1 page but text-box'd test data
|
||||||
|
final String expectText2 =
|
||||||
|
"Hello, World!!!\n" +
|
||||||
|
"I am just a poor boy\n" +
|
||||||
|
"This is Times New Roman\n" +
|
||||||
|
"Plain Text \n";
|
||||||
|
|
||||||
// 1 page example with text boxes
|
// 1 page example with text boxes
|
||||||
try (SlideShowExtractor ppe = openExtractor("with_textbox.ppt")) {
|
try (SlideShowExtractor ppe = openExtractor("with_textbox.ppt")) {
|
||||||
assertEquals(expectText2, ppe.getText());
|
assertEquals(expectText2, ppe.getText());
|
||||||
|
@ -92,8 +111,7 @@ public final class TestExtractor {
|
||||||
ppe.setSlidesByDefault(false);
|
ppe.setSlidesByDefault(false);
|
||||||
ppe.setMasterByDefault(false);
|
ppe.setMasterByDefault(false);
|
||||||
String notesText = ppe.getText();
|
String notesText = ppe.getText();
|
||||||
String expText = "\nThese are the notes for page 1\n\nThese are the notes on page two, again lacking formatting\n";
|
assertEquals(NOTES_PAGE1+NOTES_PAGE2, notesText);
|
||||||
assertEquals(expText, notesText);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Other one doesn't have notes
|
// Other one doesn't have notes
|
||||||
|
@ -109,14 +127,8 @@ public final class TestExtractor {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testReadBoth() throws IOException {
|
public void testReadBoth() throws IOException {
|
||||||
String[] slText = new String[]{
|
String[] slText = { EXPECTED_PAGE1, EXPECTED_PAGE2 };
|
||||||
"This is a test title\nThis is a test subtitle\nThis is on page 1\n",
|
String[] ntText = { NOTES_PAGE1, NOTES_PAGE2 };
|
||||||
"This is the title on page 2\nThis is page two\nIt has several blocks of text\nNone of them have formatting\n"
|
|
||||||
};
|
|
||||||
String[] ntText = new String[]{
|
|
||||||
"\nThese are the notes for page 1\n",
|
|
||||||
"\nThese are the notes on page two, again lacking formatting\n"
|
|
||||||
};
|
|
||||||
|
|
||||||
try (SlideShowExtractor ppe = openExtractor("basic_test_ppt_file.ppt")) {
|
try (SlideShowExtractor ppe = openExtractor("basic_test_ppt_file.ppt")) {
|
||||||
ppe.setSlidesByDefault(true);
|
ppe.setSlidesByDefault(true);
|
||||||
|
@ -165,8 +177,8 @@ public final class TestExtractor {
|
||||||
final DirectoryNode root = fs.getRoot();
|
final DirectoryNode root = fs.getRoot();
|
||||||
|
|
||||||
final String[] TEST_SET = {
|
final String[] TEST_SET = {
|
||||||
"MBD0000A3B6", "Sample PowerPoint file\nThis is the 1st file\nNot much too it\n",
|
"MBD0000A3B6", "Sample PowerPoint file\nThis is the 1st file\n\nNot much too it\n",
|
||||||
"MBD0000A3B3", "Sample PowerPoint file\nThis is the 2nd file\nNot much too it either\n"
|
"MBD0000A3B3", "Sample PowerPoint file\nThis is the 2nd file\n\nNot much too it either\n"
|
||||||
};
|
};
|
||||||
|
|
||||||
for (int i=0; i<TEST_SET.length; i+=2) {
|
for (int i=0; i<TEST_SET.length; i+=2) {
|
||||||
|
@ -386,7 +398,7 @@ public final class TestExtractor {
|
||||||
// Open directly
|
// Open directly
|
||||||
try (SlideShow<?,?> ppt = SlideShowFactory.create(npoifs.getRoot());
|
try (SlideShow<?,?> ppt = SlideShowFactory.create(npoifs.getRoot());
|
||||||
SlideShowExtractor<?,?> extractor = new SlideShowExtractor<>(ppt)) {
|
SlideShowExtractor<?,?> extractor = new SlideShowExtractor<>(ppt)) {
|
||||||
assertEquals(expectText, extractor.getText());
|
assertEquals(EXPECTED_PAGE1+EXPECTED_PAGE2, extractor.getText());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -457,4 +469,19 @@ public final class TestExtractor {
|
||||||
private static int countMatches(final String base, final String find) {
|
private static int countMatches(final String base, final String find) {
|
||||||
return base.split(find).length-1;
|
return base.split(find).length-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void glyphCounting() throws IOException {
|
||||||
|
String[] expected = {
|
||||||
|
"Times New Roman", "\t\n ,-./01234679:ABDEFGILMNOPRSTVWabcdefghijklmnoprstuvwxyz\u00F3\u201C\u201D",
|
||||||
|
"Arial", " Lacdilnost"
|
||||||
|
};
|
||||||
|
try (SlideShowExtractor ppt = openExtractor("45543.ppt")) {
|
||||||
|
for (int i=0; i<expected.length; i+=2) {
|
||||||
|
BitSet l = ppt.getCodepoints(expected[i], null, null);
|
||||||
|
String s = l.stream().mapToObj(Character::toChars).map(String::valueOf).collect(Collectors.joining());
|
||||||
|
assertEquals(expected[i+1], s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,7 +27,7 @@ import org.apache.poi.sl.usermodel.BaseTestSlideShow;
|
||||||
import org.apache.poi.sl.usermodel.SlideShow;
|
import org.apache.poi.sl.usermodel.SlideShow;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
public class TestHSLFSlideShow extends BaseTestSlideShow {
|
public class TestHSLFSlideShow extends BaseTestSlideShow<HSLFShape,HSLFTextParagraph> {
|
||||||
@Override
|
@Override
|
||||||
public HSLFSlideShow createSlideShow() {
|
public HSLFSlideShow createSlideShow() {
|
||||||
return new HSLFSlideShow();
|
return new HSLFSlideShow();
|
||||||
|
@ -39,11 +39,7 @@ public class TestHSLFSlideShow extends BaseTestSlideShow {
|
||||||
assertNotNull(createSlideShow());
|
assertNotNull(createSlideShow());
|
||||||
}
|
}
|
||||||
|
|
||||||
public SlideShow<?, ?> reopen(SlideShow<?, ?> show) {
|
public HSLFSlideShow reopen(SlideShow<HSLFShape,HSLFTextParagraph> show) {
|
||||||
return reopen((HSLFSlideShow)show);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static HSLFSlideShow reopen(HSLFSlideShow show) {
|
|
||||||
try {
|
try {
|
||||||
BufAccessBAOS bos = new BufAccessBAOS();
|
BufAccessBAOS bos = new BufAccessBAOS();
|
||||||
show.write(bos);
|
show.write(bos);
|
||||||
|
@ -55,7 +51,7 @@ public class TestHSLFSlideShow extends BaseTestSlideShow {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class BufAccessBAOS extends ByteArrayOutputStream {
|
private static class BufAccessBAOS extends ByteArrayOutputStream {
|
||||||
public byte[] getBuf() {
|
byte[] getBuf() {
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
package org.apache.poi.sl.usermodel;
|
package org.apache.poi.sl.usermodel;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertNotNull;
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertNull;
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.assertSame;
|
import static org.junit.Assert.assertSame;
|
||||||
|
@ -29,20 +30,24 @@ import java.io.InputStream;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.poi.POIDataSamples;
|
import org.apache.poi.POIDataSamples;
|
||||||
|
import org.apache.poi.common.usermodel.fonts.FontInfo;
|
||||||
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
import org.apache.poi.sl.usermodel.PictureData.PictureType;
|
||||||
import org.apache.poi.sl.usermodel.TabStop.TabStopType;
|
import org.apache.poi.sl.usermodel.TabStop.TabStopType;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
public abstract class BaseTestSlideShow {
|
public abstract class BaseTestSlideShow<
|
||||||
|
S extends Shape<S,P>,
|
||||||
|
P extends TextParagraph<S,P,? extends TextRun>
|
||||||
|
> {
|
||||||
protected static final POIDataSamples slTests = POIDataSamples.getSlideShowInstance();
|
protected static final POIDataSamples slTests = POIDataSamples.getSlideShowInstance();
|
||||||
|
|
||||||
public abstract SlideShow<?, ?> createSlideShow();
|
public abstract SlideShow<S,P> createSlideShow();
|
||||||
|
|
||||||
public abstract SlideShow<?, ?> reopen(SlideShow<?, ?> show);
|
public abstract SlideShow<S,P> reopen(SlideShow<S,P> show);
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void addPicture_File() throws IOException {
|
public void addPicture_File() throws IOException {
|
||||||
SlideShow<?,?> show = createSlideShow();
|
SlideShow<S,P> show = createSlideShow();
|
||||||
File f = slTests.getFile("clock.jpg");
|
File f = slTests.getFile("clock.jpg");
|
||||||
|
|
||||||
assertEquals(0, show.getPictureData().size());
|
assertEquals(0, show.getPictureData().size());
|
||||||
|
@ -55,26 +60,18 @@ public abstract class BaseTestSlideShow {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void addPicture_Stream() throws IOException {
|
public void addPicture_Stream() throws IOException {
|
||||||
SlideShow<?,?> show = createSlideShow();
|
try (SlideShow<S,P> show = createSlideShow();
|
||||||
try {
|
InputStream stream = slTests.openResourceAsStream("clock.jpg")) {
|
||||||
InputStream stream = slTests.openResourceAsStream("clock.jpg");
|
|
||||||
try {
|
|
||||||
assertEquals(0, show.getPictureData().size());
|
assertEquals(0, show.getPictureData().size());
|
||||||
PictureData picture = show.addPicture(stream, PictureType.JPEG);
|
PictureData picture = show.addPicture(stream, PictureType.JPEG);
|
||||||
assertEquals(1, show.getPictureData().size());
|
assertEquals(1, show.getPictureData().size());
|
||||||
assertSame(picture, show.getPictureData().get(0));
|
assertSame(picture, show.getPictureData().get(0));
|
||||||
|
|
||||||
} finally {
|
|
||||||
stream.close();
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
show.close();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void addPicture_ByteArray() throws IOException {
|
public void addPicture_ByteArray() throws IOException {
|
||||||
SlideShow<?,?> show = createSlideShow();
|
SlideShow<S,P> show = createSlideShow();
|
||||||
byte[] data = slTests.readFile("clock.jpg");
|
byte[] data = slTests.readFile("clock.jpg");
|
||||||
|
|
||||||
assertEquals(0, show.getPictureData().size());
|
assertEquals(0, show.getPictureData().size());
|
||||||
|
@ -87,7 +84,7 @@ public abstract class BaseTestSlideShow {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void findPicture() throws IOException {
|
public void findPicture() throws IOException {
|
||||||
SlideShow<?,?> show = createSlideShow();
|
SlideShow<S,P> show = createSlideShow();
|
||||||
byte[] data = slTests.readFile("clock.jpg");
|
byte[] data = slTests.readFile("clock.jpg");
|
||||||
|
|
||||||
assertNull(show.findPictureData(data));
|
assertNull(show.findPictureData(data));
|
||||||
|
@ -101,11 +98,11 @@ public abstract class BaseTestSlideShow {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void addTabStops() throws IOException {
|
public void addTabStops() throws IOException {
|
||||||
try (final SlideShow<?,?> show1 = createSlideShow()) {
|
try (final SlideShow<S,P> show1 = createSlideShow()) {
|
||||||
// first set the TabStops in the Master sheet
|
// first set the TabStops in the Master sheet
|
||||||
final MasterSheet<?, ?> master1 = show1.getSlideMasters().get(0);
|
final MasterSheet<S,P> master1 = show1.getSlideMasters().get(0);
|
||||||
final AutoShape<?, ?> master1_as = (AutoShape<?,?>)master1.getPlaceholder(Placeholder.BODY);
|
final AutoShape<S,P> master1_as = (AutoShape<S,P>)master1.getPlaceholder(Placeholder.BODY);
|
||||||
final TextParagraph<?, ?, ? extends TextRun> master1_tp = master1_as.getTextParagraphs().get(0);
|
final P master1_tp = master1_as.getTextParagraphs().get(0);
|
||||||
master1_tp.clearTabStops();
|
master1_tp.clearTabStops();
|
||||||
int i1 = 0;
|
int i1 = 0;
|
||||||
for (final TabStopType tst : TabStopType.values()) {
|
for (final TabStopType tst : TabStopType.values()) {
|
||||||
|
@ -114,11 +111,11 @@ public abstract class BaseTestSlideShow {
|
||||||
}
|
}
|
||||||
|
|
||||||
// then set it on a normal slide
|
// then set it on a normal slide
|
||||||
final Slide<?,?> slide1 = show1.createSlide();
|
final Slide<S,P> slide1 = show1.createSlide();
|
||||||
final AutoShape<?, ?> slide1_as = slide1.createAutoShape();
|
final AutoShape<S,P> slide1_as = slide1.createAutoShape();
|
||||||
slide1_as.setText("abc");
|
slide1_as.setText("abc");
|
||||||
slide1_as.setAnchor(new Rectangle2D.Double(100,100,100,100));
|
slide1_as.setAnchor(new Rectangle2D.Double(100,100,100,100));
|
||||||
final TextParagraph<?, ?, ? extends TextRun> slide1_tp = slide1_as.getTextParagraphs().get(0);
|
final P slide1_tp = slide1_as.getTextParagraphs().get(0);
|
||||||
slide1_tp.getTextRuns().get(0).setFontColor(new Color(0x563412));
|
slide1_tp.getTextRuns().get(0).setFontColor(new Color(0x563412));
|
||||||
slide1_tp.clearTabStops();
|
slide1_tp.clearTabStops();
|
||||||
int i2 = 0;
|
int i2 = 0;
|
||||||
|
@ -127,10 +124,10 @@ public abstract class BaseTestSlideShow {
|
||||||
i2++;
|
i2++;
|
||||||
}
|
}
|
||||||
|
|
||||||
try (final SlideShow<?, ?> show2 = reopen(show1)) {
|
try (final SlideShow<S,P> show2 = reopen(show1)) {
|
||||||
final MasterSheet<?, ?> master2 = show2.getSlideMasters().get(0);
|
final MasterSheet<S,P> master2 = show2.getSlideMasters().get(0);
|
||||||
final AutoShape<?, ?> master2_as = (AutoShape<?,?>)master2.getPlaceholder(Placeholder.BODY);
|
final AutoShape<S,P> master2_as = (AutoShape<S,P>)master2.getPlaceholder(Placeholder.BODY);
|
||||||
final TextParagraph<?, ?, ? extends TextRun> master2_tp = master2_as.getTextParagraphs().get(0);
|
final P master2_tp = master2_as.getTextParagraphs().get(0);
|
||||||
final List<? extends TabStop> master2_tabStops = master2_tp.getTabStops();
|
final List<? extends TabStop> master2_tabStops = master2_tp.getTabStops();
|
||||||
assertNotNull(master2_tabStops);
|
assertNotNull(master2_tabStops);
|
||||||
int i3 = 0;
|
int i3 = 0;
|
||||||
|
@ -142,9 +139,10 @@ public abstract class BaseTestSlideShow {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
final Slide<?,?> slide2 = show2.getSlides().get(0);
|
final Slide<S,P> slide2 = show2.getSlides().get(0);
|
||||||
final AutoShape<?,?> slide2_as = (AutoShape<?,?>)slide2.getShapes().get(0);
|
@SuppressWarnings("unchecked")
|
||||||
final TextParagraph<?, ?, ? extends TextRun> slide2_tp = slide2_as.getTextParagraphs().get(0);
|
final AutoShape<S,P> slide2_as = (AutoShape<S,P>)slide2.getShapes().get(0);
|
||||||
|
final P slide2_tp = slide2_as.getTextParagraphs().get(0);
|
||||||
final List<? extends TabStop> slide2_tabStops = slide2_tp.getTabStops();
|
final List<? extends TabStop> slide2_tabStops = slide2_tp.getTabStops();
|
||||||
assertNotNull(slide2_tabStops);
|
assertNotNull(slide2_tabStops);
|
||||||
int i4 = 0;
|
int i4 = 0;
|
||||||
|
@ -162,18 +160,35 @@ public abstract class BaseTestSlideShow {
|
||||||
public void shapeAndSlideName() throws IOException {
|
public void shapeAndSlideName() throws IOException {
|
||||||
final String file = "SampleShow.ppt"+(getClass().getSimpleName().contains("XML")?"x":"");
|
final String file = "SampleShow.ppt"+(getClass().getSimpleName().contains("XML")?"x":"");
|
||||||
try (final InputStream is = slTests.openResourceAsStream(file);
|
try (final InputStream is = slTests.openResourceAsStream(file);
|
||||||
final SlideShow<? extends Shape, ?> ppt = SlideShowFactory.create(is)) {
|
final SlideShow<S,P> ppt = SlideShowFactory.create(is)) {
|
||||||
final List<? extends Shape> shapes1 = ppt.getSlides().get(0).getShapes();
|
final List<S> shapes1 = ppt.getSlides().get(0).getShapes();
|
||||||
assertEquals("The Title", shapes1.get(0).getShapeName());
|
assertEquals("The Title", shapes1.get(0).getShapeName());
|
||||||
assertEquals("Another Subtitle", shapes1.get(1).getShapeName());
|
assertEquals("Another Subtitle", shapes1.get(1).getShapeName());
|
||||||
final List<? extends Shape> shapes2 = ppt.getSlides().get(1).getShapes();
|
final List<S> shapes2 = ppt.getSlides().get(1).getShapes();
|
||||||
assertEquals("Title 1", shapes2.get(0).getShapeName());
|
assertEquals("Title 1", shapes2.get(0).getShapeName());
|
||||||
assertEquals("Content Placeholder 2", shapes2.get(1).getShapeName());
|
assertEquals("Content Placeholder 2", shapes2.get(1).getShapeName());
|
||||||
|
|
||||||
for (final Slide<?,?> slide : ppt.getSlides()) {
|
for (final Slide<S,P> slide : ppt.getSlides()) {
|
||||||
final String expected = slide.getSlideNumber()==1 ? "FirstSlide" : "Slide2";
|
final String expected = slide.getSlideNumber()==1 ? "FirstSlide" : "Slide2";
|
||||||
assertEquals(expected, slide.getSlideName());
|
assertEquals(expected, slide.getSlideName());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void addFont() throws IOException {
|
||||||
|
try (SlideShow<S,P> ppt = createSlideShow()) {
|
||||||
|
ppt.createSlide();
|
||||||
|
try (InputStream fontData = slTests.openResourceAsStream("font.fntdata")) {
|
||||||
|
ppt.addFont(fontData);
|
||||||
|
}
|
||||||
|
|
||||||
|
try (SlideShow<S, P> ppt2 = reopen(ppt)) {
|
||||||
|
List<? extends FontInfo> fonts = ppt2.getFonts();
|
||||||
|
assertFalse(fonts.isEmpty());
|
||||||
|
FontInfo fi = fonts.get(fonts.size()-1);
|
||||||
|
assertEquals("Harlow Solid Italic", fi.getTypeface());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Binary file not shown.
Loading…
Reference in New Issue