All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jpedal.parser.PdfFontFactory Maven / Gradle / Ivy

There is a newer version: 7.15.25
Show newest version
/*
 * ===========================================
 * Java Pdf Extraction Decoding Access Library
 * ===========================================
 *
 * Project Info:  http://www.idrsolutions.com
 * Help section for developers at http://www.idrsolutions.com/support/
 *
 * (C) Copyright 1997-2017 IDRsolutions and Contributors.
 *
 * This file is part of JPedal/JPDF2HTML5
 *
 @LICENSE@
 *
 * ---------------
 * PdfFontFactory.java
 * ---------------
 */
package org.jpedal.parser;

import java.io.RandomAccessFile;
import java.util.HashMap;
import java.util.HashSet;

import org.jpedal.exception.PdfException;
import org.jpedal.exception.PdfFontException;
import org.jpedal.external.ErrorTracker;
import org.jpedal.fonts.FontMappings;
import org.jpedal.fonts.PdfFont;
import org.jpedal.fonts.StandardFonts;
import org.jpedal.io.ObjectStore;
import org.jpedal.io.PdfObjectReader;
import org.jpedal.objects.raw.FontObject;
import org.jpedal.objects.raw.PdfDictionary;
import org.jpedal.objects.raw.PdfObject;
import org.jpedal.utils.LogWriter;
import org.jpedal.utils.StringUtils;

/**
 * Convert font info into one of our supporting classes
 */
public class PdfFontFactory {

    /**
     * flag to show embedded fonts present
     */
    private boolean hasEmbeddedFonts;

    /**
     * flag to show if non-embedded CID fonts
     */
    private boolean hasNonEmbeddedCIDFonts;

    /**
     * list of fonts used for display
     */
    private String fontsInFile;

    /**
     * and the list of CID fonts
     */
    private final StringBuilder nonEmbeddedCIDFonts = new StringBuilder(200);

    private String baseFont = "", rawFontName, subFont;
    private int origfontType;

    //only load 1 instance of any 1 font
    private final HashMap fontsLoaded = new HashMap(50);

    final PdfObjectReader currentPdfFile;

    public PdfFontFactory(final PdfObjectReader currentPdfFile) {
        this.currentPdfFile = currentPdfFile;
    }

    public PdfFont createFont(final boolean fallbackToArial, final PdfObject pdfObject, final String font_id, final ObjectStore objectStoreStreamRef,
                              final boolean renderPage, final ErrorTracker errorTracker,
                              final boolean isHTML) throws PdfException {

        PdfFont currentFontData = null;

        baseFont = "";
        rawFontName = null;
        subFont = null;

        int fontType = PdfDictionary.Unknown;
        origfontType = PdfDictionary.Unknown;

        final PdfObject descendantFont = pdfObject.getDictionary(PdfDictionary.DescendantFonts);

        boolean isEmbedded = isFontEmbedded(pdfObject);

        if (isHTML && !isEmbedded) {
            if (inlineLiberationFontForHTML(pdfObject, font_id, descendantFont)) {
                fontType = StandardFonts.TRUETYPE;
                isEmbedded = true;
            }
        }

        boolean isFontBroken = true; //ensure enters once

        while (isFontBroken) { //will try to sub font if error in embedded
            isFontBroken = false;

            /*
             * handle any font remapping but not on CID fonts or Type3 and gets too messy
             **/
            if (FontMappings.fontSubstitutionTable != null && !isEmbedded &&
                    pdfObject.getParameterConstant(PdfDictionary.Subtype) != StandardFonts.TYPE3) {

                final String rawFont = getFontName(pdfObject, font_id, descendantFont);

                fontType = getFontMapping(pdfObject, rawFont, fontType, descendantFont);
            }

            //get subtype if not set above
            if (fontType == PdfDictionary.Unknown) {
                fontType = pdfObject.getParameterConstant(PdfDictionary.Subtype);

                /*handle CID fonts where /Subtype stored inside sub object*/
                if (fontType == StandardFonts.TYPE0) {

                    //get CID type and use in preference to Type0 on CID fonts
                    final PdfObject desc = pdfObject.getDictionary(PdfDictionary.DescendantFonts);
                    fontType = desc.getParameterConstant(PdfDictionary.Subtype);

                    origfontType = fontType;

                }
            }

            if (fontType == PdfDictionary.Unknown) {

                LogWriter.writeLog("Font type not supported");

                currentFontData = new PdfFont(currentPdfFile);
            }

            /*
             * check for OpenType fonts and reassign type
             */
            final int rawFontType = fontType;
            if (fontType == StandardFonts.TYPE1 || fontType == StandardFonts.CIDTYPE0) {
                fontType = scanForOpenType(pdfObject, currentPdfFile, fontType);
            }

            if (!isEmbedded && subFont == null && fallbackToArial && fontType != StandardFonts.TYPE3) {
                String replacementFont = "arial";

                String testFont = pdfObject.getName(PdfDictionary.BaseFont);

                if (testFont != null) { //try to match

                    testFont = testFont.toLowerCase();

                    if (testFont.contains("bolditalic")) {
                        replacementFont = "arial bold italic";
                    } else if (testFont.contains("italic")) {
                        replacementFont = "arial italic";
                    } else if (testFont.contains("bold")) {
                        replacementFont = "arial bold";
                    }
                }

                subFont = FontMappings.fontSubstitutionLocation.get(replacementFont);

                if (subFont == null) {
                    subFont = useLiberationFont(testFont, "LiberationSans", "liberation-sans");
                }

                fontType = StandardFonts.TRUETYPE;

            }

            try {
                currentFontData = FontFactory.createFont(fontType, currentPdfFile, subFont);

                if (rawFontType == StandardFonts.CIDTYPE0 && fontType == StandardFonts.CIDTYPE2) {
                    currentFontData.getGlyphData().setRemappedCFFFont(true);
                }

                /*set an alternative to Lucida*/
                if (FontMappings.defaultFont != null) {
                    currentFontData.setDefaultDisplayFont(FontMappings.defaultFont);
                }

                currentFontData.createFont(pdfObject, font_id, renderPage, objectStoreStreamRef, fontsLoaded);

                //track non-embedded, non-substituted CID fonts
                if ((fontType == StandardFonts.CIDTYPE0 || fontType == StandardFonts.CIDTYPE2) && !isEmbedded && subFont == null) {

                    //allow for it being substituted
                    subFont = currentFontData.getSubstituteFont();
                    if (subFont == null) {

                        hasNonEmbeddedCIDFonts = true;

                        //track list
                        if (nonEmbeddedCIDFonts.length() > 0) {
                            nonEmbeddedCIDFonts.append(',');
                        }
                        nonEmbeddedCIDFonts.append(baseFont);
                    }
                }

                //save raw version
                currentFontData.setRawFontName(rawFontName);

                //fix for odd file
                if (fontType == StandardFonts.TYPE1 && currentFontData.is1C() && pdfObject.getInt(PdfDictionary.FirstChar) == 32 &&
                        pdfObject.getInt(PdfDictionary.FirstChar) == pdfObject.getInt(PdfDictionary.LastChar)) {

                    if (isEmbedded) {
                        isFontBroken = true;
                        isEmbedded = false;
                    } else {
                        currentFontData.isFontEmbedded = false;
                    }
                }

                //see if we failed and loop round to substitute
                if (!currentFontData.isFontEmbedded && isEmbedded) {
                    isFontBroken = true;
                    isEmbedded = false;
                }

            } catch (final Exception e) {

                LogWriter.writeLog("[PDF] Problem " + e + " reading Font  type " + StandardFonts.getFontypeAsString(fontType));

                errorTracker.addPageFailureMessage("Problem " + e + " reading Font type " + StandardFonts.getFontypeAsString(fontType));
            }
        }

        /*
         * add line giving font info so we can display or user access
         */
        setDetails(font_id, currentFontData, fontType, descendantFont);

        return currentFontData;
    }

    boolean inlineLiberationFontForHTML(final PdfObject pdfObject, final String font_id, final PdfObject descendantFont) {

        final String rawFont = getFontName(pdfObject, font_id, descendantFont);
        final String test = rawFont.toLowerCase();
        String libFont = null;

        if (test.contains("timesnewroman")) {
            libFont = useLiberationFont(test, "LiberationSerif", "liberation-serif");
        } else if (test.contains("arial")) {
            libFont = useLiberationFont(test, "LiberationSans", "liberation-sans");
        }

        if (libFont != null) {

            if (libFont != null) {
                final RandomAccessFile f;
                try {
                    f = new RandomAccessFile(libFont, "r");

                    final byte[] stream = new byte[(int) f.length()];

                    f.readFully(stream);

                    f.close();

                    final FontObject fontObj = new FontObject(pdfObject.getObjectRefAsString());
                    fontObj.setDecodedStream(stream);

                    final PdfObject pdfFontDescriptor = pdfObject.getDictionary(PdfDictionary.FontDescriptor);

                    if (pdfFontDescriptor != null) {
                        pdfFontDescriptor.setDictionary(PdfDictionary.FontFile2, fontObj);
                    }
                } catch (final Exception e) {
                    e.printStackTrace();
                    LogWriter.writeLog("[PDF] Exception replacing Liberation font " + e);
                }
            }
        }

        return libFont != null;
    }

    public String useLiberationFont(final String testFont, final String family, final String path) {

        String subFont = null;
        String replacementFont; //add Liberation Sans for if all else fails
        try {
            FontMappings.addLiberationAsFallBackFont(family, path);

            replacementFont = "-Regular";

            if (testFont.contains("bolditalic")) {
                replacementFont = "-BoldItalic";
            } else if (testFont.contains("italic")) {
                replacementFont = "-Italic";
            } else if (testFont.contains("bold")) {
                replacementFont = "-Bold";
            }
            subFont = FontMappings.fontSubstitutionLocation.get(family + replacementFont);

        } catch (final Exception e) {
            LogWriter.writeLog("Unable to read liberation fonts " + e.getMessage());
        }

        return subFont;
    }

    private void setDetails(final String font_id, final PdfFont currentFontData, final int fontType, final PdfObject descendantFont) {
        String name = currentFontData.getFontName();

        //deal with odd chars
        if (name.indexOf('#') != -1) {
            name = StringUtils.convertHexChars(name);
        }

        final String details;
        if (currentFontData.isFontSubstituted()) {
            details = font_id + "  " + name + "  " + StandardFonts.getFontypeAsString(origfontType) + "  Substituted (" + subFont + ' ' + StandardFonts.getFontypeAsString(fontType) + ')';
        } else if (currentFontData.isFontEmbedded) {
            hasEmbeddedFonts = true;
            if (currentFontData.is1C() && descendantFont == null) {
                details = font_id + "  " + name + " Type1C  Embedded";
            } else {
                details = font_id + "  " + name + "  " + StandardFonts.getFontypeAsString(fontType) + "  Embedded";
            }
        } else {
            details = font_id + "  " + name + "  " + StandardFonts.getFontypeAsString(fontType);
        }

        if (fontsInFile == null) {
            fontsInFile = details;
        } else {
            fontsInFile = details + '\n' + fontsInFile;
        }
    }

    private static int scanForOpenType(final PdfObject pdfObject, final PdfObjectReader currentPdfFile, int fontType) {

        if (fontType == StandardFonts.CIDTYPE0) {
            final PdfObject desc = pdfObject.getDictionary(PdfDictionary.DescendantFonts);

            if (pdfObject != null) {
                final PdfObject FontDescriptor = desc.getDictionary(PdfDictionary.FontDescriptor);

                if (FontDescriptor != null) {
                    PdfObject FontFile2 = FontDescriptor.getDictionary(PdfDictionary.FontFile2);

                    if (FontFile2 == null) { //must be present for OTTF font
                        FontFile2 = FontDescriptor.getDictionary(PdfDictionary.FontFile3);
                    }

                    if (FontFile2 != null) { //must be present for OTTF font

                        //get data
                        final byte[] stream = currentPdfFile.readStream(FontFile2, true, true, false, false, false, FontFile2.getCacheName(currentPdfFile.getObjectReader()));

                        //check first 4 bytes
                        if (stream != null && stream.length > 3 && stream[0] == 79 && stream[1] == 84 && stream[2] == 84 && stream[3] == 79) {
                            fontType = StandardFonts.CIDTYPE2; //put it through our TT handler which also does OT
                        }

                    }
                }
            }
        } else {
            final PdfObject FontDescriptor = pdfObject.getDictionary(PdfDictionary.FontDescriptor);
            if (FontDescriptor != null) {

                final PdfObject FontFile3 = FontDescriptor.getDictionary(PdfDictionary.FontFile3);
                if (FontFile3 != null) { //must be present for OTTF font

                    //get data
                    final byte[] stream = currentPdfFile.readStream(FontFile3, true, true, false, false, false, FontFile3.getCacheName(currentPdfFile.getObjectReader()));

                    //check first 4 bytes
                    if (stream != null && stream.length > 3 && stream[0] == 79 && stream[1] == 84 && stream[2] == 84 && stream[3] == 79) {
                        fontType = StandardFonts.TRUETYPE; //put it through our TT handler which also does OT
                    }

                }
            }
        }

        return fontType;
    }

    private int getFontMapping(final PdfObject pdfObject, final String rawFont, int fontType, final PdfObject descendantFont) throws PdfException {

        String newSubtype = getFontSub(rawFont);

        //Case 24809 ie WingDings-regular needs to use WingDing(s)
        if (newSubtype == null && rawFont.startsWith("Wingdings")) {
            if (DecoderOptions.isRunningOnMac) {
                newSubtype = getFontSub("WingDings");
            } else {
                newSubtype = getFontSub("WingDing");
            }
        }

        if (newSubtype != null && descendantFont == null) {

            //convert String to correct int value
            if (newSubtype.equals("/Type1") || newSubtype.equals("/Type1C") || newSubtype.equals("/MMType1")) {
                fontType = StandardFonts.TYPE1;
            } else if (newSubtype.equals("/TrueType")) {
                fontType = StandardFonts.TRUETYPE;
            } else if (newSubtype.equals("/Type3")) {
                fontType = StandardFonts.TYPE3;
            } else {
                throw new RuntimeException("Unknown font type " + newSubtype + " used for font substitution");
            }

            origfontType = pdfObject.getParameterConstant(PdfDictionary.Subtype);

        } else if (FontMappings.enforceFontSubstitution) {

            LogWriter.writeLog("baseFont=" + baseFont + " fonts added= " + FontMappings.fontSubstitutionTable);

            throw new PdfFontException("No substitute Font found for font=" + baseFont + '<');
        }
        return fontType;
    }

    private String getFontName(final PdfObject pdfObject, final String font_id, final PdfObject descendantFont) {
        String rawFont;

        if (descendantFont == null) {
            rawFont = pdfObject.getName(PdfDictionary.BaseFont);
        } else {
            rawFont = descendantFont.getName(PdfDictionary.BaseFont);
        }

        if (rawFont == null) {
            rawFont = pdfObject.getName(PdfDictionary.Name);
        }

        if (rawFont == null) {
            rawFont = font_id;
        }
        return rawFont;
    }

    public String getFontSub(String rawFont) throws PdfException {
        if (rawFont.indexOf('#') != -1) {
            rawFont = StringUtils.convertHexChars(rawFont);
        }

        //save in case we need later
        rawFontName = rawFont;

        baseFont = (rawFont).toLowerCase();

        //strip any postscript
        final int pointer = baseFont.indexOf('+');
        if (pointer == 6) {
            baseFont = baseFont.substring(7);
        }

        String testFont = baseFont, nextSubType;

        subFont = FontMappings.fontSubstitutionLocation.get(testFont);

        String newSubtype = FontMappings.fontSubstitutionTable.get(testFont);

        //do not replace on MAC as default does not have certain values we need
        if (DecoderOptions.isRunningOnMac && testFont.equals("zapfdingbats")) {
            testFont = "No match found";
        }

        //check aliases
        if (newSubtype == null) {
            //check for mapping
            final HashSet fontsMapped = new HashSet(50);
            String nextFont;
            while (true) {
                nextFont = FontMappings.fontSubstitutionAliasTable.get(testFont);

                if (nextFont == null) {
                    break;
                }

                testFont = nextFont;

                nextSubType = FontMappings.fontSubstitutionTable.get(testFont);

                if (nextSubType != null) {
                    newSubtype = nextSubType;
                    subFont = FontMappings.fontSubstitutionLocation.get(testFont);
                }

                if (fontsMapped.contains(testFont)) {
                    //use string buffer and stringbuilder does not exist in java ME
                    final StringBuilder errorMessage = new StringBuilder("[PDF] Circular font mapping for fonts");
                    for (final Object o : fontsMapped.toArray()) {
                        errorMessage.append(' ');
                        errorMessage.append(o);
                    }
                    throw new PdfException(errorMessage.toString());
                }
                fontsMapped.add(nextFont);
            }
        }
        return newSubtype;
    }

    /**
     * check for embedded font file to see if font embedded
     */
    public static boolean isFontEmbedded(PdfObject pdfObject) {

        //ensure we are looking in DescendantFonts object if CID
        final int fontType = pdfObject.getParameterConstant(PdfDictionary.Subtype);
        if (fontType == StandardFonts.TYPE0) {
            pdfObject = pdfObject.getDictionary(PdfDictionary.DescendantFonts);
        }


        final PdfObject descFontObj = pdfObject.getDictionary(PdfDictionary.FontDescriptor);


        if (descFontObj == null) {
            return false;
        } else {
            return descFontObj.hasStream();
        }
    }


    public String getnonEmbeddedCIDFonts() {
        return nonEmbeddedCIDFonts.toString();
    }

    public String getFontsInFile() {
        return fontsInFile;
    }

    public void resetfontsInFile() {
        fontsInFile = "";
    }

    public boolean hasEmbeddedFonts() {
        return hasEmbeddedFonts;
    }

    public boolean hasNonEmbeddedCIDFonts() {
        return hasNonEmbeddedCIDFonts;
    }

    public String getMapFont() {
        return subFont;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy