package com.tom_roush.pdfbox.pdmodel.font; import android.util.Log; import com.tom_roush.fontbox.afm.FontMetrics; import com.tom_roush.fontbox.cmap.CMap; import com.tom_roush.fontbox.util.BoundingBox; import com.tom_roush.pdfbox.cos.COSArray; import com.tom_roush.pdfbox.cos.COSBase; import com.tom_roush.pdfbox.cos.COSDictionary; import com.tom_roush.pdfbox.cos.COSName; import com.tom_roush.pdfbox.cos.COSNumber; import com.tom_roush.pdfbox.cos.COSStream; import com.tom_roush.pdfbox.io.IOUtils; import com.tom_roush.pdfbox.pdmodel.common.COSArrayList; import com.tom_roush.pdfbox.pdmodel.common.COSObjectable; import com.tom_roush.pdfbox.pdmodel.font.encoding.GlyphList; import com.tom_roush.pdfbox.util.Matrix; import com.tom_roush.pdfbox.util.Vector; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.Collections; import java.util.List; /** * This is the base class for all PDF fonts. * * @author Ben Litchfield */ public abstract class PDFont implements COSObjectable, PDFontLike { protected static final Matrix DEFAULT_FONT_MATRIX = new Matrix(0.001f, 0, 0, 0.001f, 0, 0); protected final COSDictionary dict; private final CMap toUnicodeCMap; private final FontMetrics afmStandard14; // AFM for standard 14 fonts private PDFontDescriptor fontDescriptor; private List<Integer> widths; private float avgFontWidth; private float fontWidthOfSpace = -1f; /** * Constructor for embedding. */ PDFont() { dict = new COSDictionary(); dict.setItem(COSName.TYPE, COSName.FONT); toUnicodeCMap = null; fontDescriptor = null; afmStandard14 = null; } /** * Constructor for Standard 14. */ PDFont(String baseFont) { dict = new COSDictionary(); toUnicodeCMap = null; afmStandard14 = Standard14Fonts.getAFM(baseFont); if (afmStandard14 == null) { throw new IllegalArgumentException("No AFM for font " + baseFont); } fontDescriptor = PDType1FontEmbedder.buildFontDescriptor(afmStandard14); } /** * Constructor. * * @param fontDictionary Font dictionary. */ protected PDFont(COSDictionary fontDictionary) throws IOException { dict = fontDictionary; // standard 14 fonts use an AFM afmStandard14 = Standard14Fonts.getAFM(getName()); // may be null (it usually is) // font descriptor COSDictionary fd = (COSDictionary) dict.getDictionaryObject(COSName.FONT_DESC); if (fd != null) { fontDescriptor = new PDFontDescriptor(fd); } else if (afmStandard14 != null) { // build font descriptor from the AFM fontDescriptor = PDType1FontEmbedder.buildFontDescriptor(afmStandard14); } else { fontDescriptor = null; } // ToUnicode CMap COSBase toUnicode = dict.getDictionaryObject(COSName.TO_UNICODE); if (toUnicode != null) { toUnicodeCMap = readCMap(toUnicode); if (toUnicodeCMap != null && !toUnicodeCMap.hasUnicodeMappings()) { Log.w("PdfBox-Android", "Invalid ToUnicode CMap in font " + getName()); } } else { toUnicodeCMap = null; } } /** * Returns the AFM if this is a Standard 14 font. */ protected final FontMetrics getStandard14AFM() { return afmStandard14; } @Override public PDFontDescriptor getFontDescriptor() { return fontDescriptor; } /** * Sets the font descriptor when embedding a font. */ protected final void setFontDescriptor(PDFontDescriptor fontDescriptor) { this.fontDescriptor = fontDescriptor; } /** * Reads a CMap given a COS Stream or Name. May return null if a predefined CMap does not exist. * * @param base COSName or COSStream */ protected final CMap readCMap(COSBase base) throws IOException { if (base instanceof COSName) { // predefined CMap String name = ((COSName)base).getName(); return CMapManager.getPredefinedCMap(name); } else if (base instanceof COSStream) { // embedded CMap InputStream input = null; try { input = ((COSStream)base).getUnfilteredStream(); return CMapManager.parseCMap(input); } finally { IOUtils.closeQuietly(input); } } else { throw new IOException("Expected Name or Stream"); } } @Override public COSDictionary getCOSObject() { return dict; } @Override public Vector getPositionVector(int code) { throw new UnsupportedOperationException("Horizontal fonts have no position vector"); } /** * Returns the displacement vector (w0, w1) in text space, for the given character. * For horizontal text only the x component is used, for vertical text only the y component. * * @param code character code * @return displacement vector */ public Vector getDisplacement(int code) throws IOException { return new Vector(getWidth(code) / 1000, 0); } @Override public float getWidth(int code) throws IOException { // Acrobat overrides the widths in the font program on the conforming reader's system with // the widths specified in the font dictionary." (Adobe Supplement to the ISO 32000) // // Note: The Adobe Supplement says that the override happens "If the font program is not // embedded", however PDFBOX-427 shows that it also applies to embedded fonts. // Type1, Type1C, Type3 if (dict.containsKey(COSName.WIDTHS) || dict.containsKey(COSName.MISSING_WIDTH)) { int firstChar = dict.getInt(COSName.FIRST_CHAR, -1); int lastChar = dict.getInt(COSName.LAST_CHAR, -1); if (getWidths().size() > 0 && code >= firstChar && code <= lastChar) { return getWidths().get(code - firstChar).floatValue(); } PDFontDescriptor fd = getFontDescriptor(); if (fd != null) { // if there's nothing to override with, then obviously we fall back to the font return fd.getMissingWidth(); // default is 0 } } // standard 14 font widths are specified by an AFM if (isStandard14()) { return getStandard14Width(code); } // if there's nothing to override with, then obviously we fall back to the font return getWidthFromFont(code); } /** * Returns the glyph width from the AFM if this is a Standard 14 font. * * @param code character code * @return width in 1/1000 text space */ protected abstract float getStandard14Width(int code); @Override public abstract float getWidthFromFont(int code) throws IOException; @Override public abstract boolean isEmbedded(); @Override public abstract float getHeight(int code) throws IOException; /** * Encodes the given string for use in a PDF content stream. * * @param text Any Unicode text. * @return Array of PDF content stream bytes. * @throws IOException If the text could not be encoded. */ public final byte[] encode(String text) throws IOException { ByteArrayOutputStream out = new ByteArrayOutputStream(); for (int offset = 0; offset < text.length(); ) { int codePoint = text.codePointAt(offset); // multi-byte encoding with 1 to 4 bytes byte[] bytes = encode(codePoint); out.write(bytes); offset += Character.charCount(codePoint); } return out.toByteArray(); } /** * Encodes the given Unicode code point for use in a PDF content stream. * Content streams use a multi-byte encoding with 1 to 4 bytes. * * <p>This method is called when embedding text in PDFs and when filling in fields. * * @param unicode Unicode code point. * @return Array of 1 to 4 PDF content stream bytes. * @throws IOException If the text could not be encoded. */ protected abstract byte[] encode(int unicode) throws IOException; /** * Returns the width of the given Unicode string. * * @param text The text to get the width of. * @return The width of the string in 1/1000 units of text space * @throws IOException If there is an error getting the width information. */ public float getStringWidth(String text) throws IOException { byte[] bytes = encode(text); ByteArrayInputStream in = new ByteArrayInputStream(bytes); float width = 0; while (in.available() > 0) { int code = readCode(in); width += getWidth(code); } return width; } /** * This will get the average font width for all characters. * * @return The width is in 1000 unit of text space, ie 333 or 777 */ // todo: this method is highly suspicious, the average glyph width is not usually a good metric @Override public float getAverageFontWidth() { float average; if (avgFontWidth != 0.0f) { average = avgFontWidth; } else { float totalWidth = 0.0f; float characterCount = 0.0f; COSArray widths = (COSArray) dict.getDictionaryObject(COSName.WIDTHS); if (widths != null) { for (int i = 0; i < widths.size(); i++) { COSNumber fontWidth = (COSNumber) widths.getObject(i); if (fontWidth.floatValue() > 0) { totalWidth += fontWidth.floatValue(); characterCount += 1; } } } if (totalWidth > 0) { average = totalWidth / characterCount; } else { average = 0; } avgFontWidth = average; } return average; } /** * Reads a character code from a content stream string. Codes may be up to 4 bytes long. * * @param in string stream * @return character code * @throws IOException if the CMap or stream cannot be read */ public abstract int readCode(InputStream in) throws IOException; /** * Returns the Unicode character sequence which corresponds to the given character code. * * @param code character code * @param customGlyphList a custom glyph list to use instead of the Adobe Glyph List * @return Unicode character(s) */ public String toUnicode(int code, GlyphList customGlyphList) throws IOException { return toUnicode(code); } /** * Returns the Unicode character sequence which corresponds to the given character code. * * @param code character code * @return Unicode character(s) */ public String toUnicode(int code) throws IOException { // if the font dictionary containsName a ToUnicode CMap, use that CMap if (toUnicodeCMap != null) { if (toUnicodeCMap.getName() != null && toUnicodeCMap.getName().startsWith("Identity-")) { // handle the undocumented case of using Identity-H/V as a ToUnicode CMap, this // isn't actually valid as the Identity-x CMaps are code->CID maps, not // code->Unicode maps. See sample_fonts_solidconvertor.pdf for an example. return new String(new char[] { (char) code }); } else { // proceed as normal return toUnicodeCMap.toUnicode(code); } } // if no value has been produced, there is no way to obtain Unicode for the character. // this behaviour can be overridden is subclasses, but this method *must* return null here return null; } /** * This will always return "Font" for fonts. * * @return The type of object that this is. */ public String getType() { return dict.getNameAsString(COSName.TYPE); } /** * This will get the subtype of font. */ public String getSubType() { return dict.getNameAsString(COSName.SUBTYPE); } @Override public abstract String getName(); @Override public abstract BoundingBox getBoundingBox() throws IOException; /** * The widths of the characters. This will be null for the standard 14 fonts. * * @return The widths of the characters. */ protected final List<Integer> getWidths() { if (widths == null) { COSArray array = (COSArray) dict.getDictionaryObject(COSName.WIDTHS); if (array != null) { widths = COSArrayList.convertIntegerCOSArrayToList(array); } else { widths = Collections.emptyList(); } } return widths; } @Override public Matrix getFontMatrix() { return DEFAULT_FONT_MATRIX; } /** * Determines the width of the space character. * * @return the width of the space character */ public float getSpaceWidth() { if (fontWidthOfSpace == -1f) { COSBase toUnicode = dict.getDictionaryObject(COSName.TO_UNICODE); try { if (toUnicode != null) { int spaceMapping = toUnicodeCMap.getSpaceMapping(); if (spaceMapping > -1) { fontWidthOfSpace = getWidth(spaceMapping); } } else { fontWidthOfSpace = getWidth(32); } // use the average font width as fall back if (fontWidthOfSpace <= 0) { fontWidthOfSpace = getAverageFontWidth(); } } catch (Exception e) { Log.e("PdfBox-Android", "Can't determine the width of the space character, assuming 250", e); fontWidthOfSpace = 250f; } } return fontWidthOfSpace; } /** * Returns true if the font uses vertical writing mode. */ public abstract boolean isVertical(); /** * Returns true if this font is one of the "Standard 14" fonts and receives special handling. */ public boolean isStandard14() { // this logic is based on Acrobat's behaviour, see see PDFBOX-2372 // embedded fonts never get special treatment if (isEmbedded()) { return false; } // if the name matches, this is a Standard 14 font return Standard14Fonts.containsName(getName()); } /** * Adds the given Unicode point to the subset. * * @param codePoint Unicode code point */ public abstract void addToSubset(int codePoint); /** * Replaces this font with a subset containing only the given Unicode characters. * * @throws IOException if the subset could not be written */ public abstract void subset() throws IOException; /** * Returns true if this font will be subset when embedded. */ public abstract boolean willBeSubset(); @Override public abstract boolean isDamaged(); @Override public boolean equals(Object other) { return other instanceof PDFont && ((PDFont) other).getCOSObject() == this.getCOSObject(); } @Override public int hashCode() { return this.getCOSObject().hashCode(); } @Override public String toString() { return getClass().getSimpleName() + " " + getName(); } }