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

org.jpedal.objects.structuredtext.MarkedContentGenerator 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@
 *
 * ---------------
 * MarkedContentGenerator.java
 * ---------------
 */
package org.jpedal.objects.structuredtext;

import java.util.HashMap;
import java.util.Map;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;

import org.jpedal.PdfDecoderInt;
import org.jpedal.io.ObjectDecoder;
import org.jpedal.io.ObjectStore;
import org.jpedal.io.PdfObjectReader;
import org.jpedal.objects.PdfPageData;
import org.jpedal.objects.PdfResources;
import org.jpedal.objects.layers.PdfLayerList;
import org.jpedal.objects.raw.*;
import org.jpedal.parser.PdfStreamDecoder;
import org.jpedal.parser.ValueTypes;
import org.jpedal.render.SwingDisplay;
import org.jpedal.utils.LogWriter;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Text;

/**
 * extract as marked content
 */
public class MarkedContentGenerator {

    private PdfObjectReader currentPdfFile;

    private DocumentBuilder db;

    private Document doc;

    private Element root;

    private final Map pageStreams = new HashMap();

    private PdfResources res;

    private PdfLayerList layers;

    private PdfPageData pdfPageData;

    private boolean isDecoding;

    static boolean debug;

    //used to indent debug output
    static String indent = "";

    final Map reverseLookup = new HashMap();

    final Map rolemapLookup = new HashMap();

    boolean isHTML;

    /**
     * main entry paint
     */
    public Document getMarkedContentTree(final PdfResources res, final PdfPageData pdfPageData, final PdfObjectReader currentPdfFile) {

        final PdfObject structTreeRootObj = res.getPdfObject(PdfResources.StructTreeRootObj);

        //PdfObject markInfoObj=res.getPdfObject(PdfResources.MarkInfoObj);  //not used at present

        this.res = res;
        this.layers = res.getPdfLayerList();

        this.pdfPageData = pdfPageData;

        this.currentPdfFile = currentPdfFile;

        //read values as needed
        this.currentPdfFile.checkResolved(structTreeRootObj);

        /*
         * create the empty XMLtree and root to add data onto
         **/
        if (!isHTML) {
            setupTree();

            final boolean hasTree = structTreeRootObj != null && structTreeRootObj.getDictionary(PdfDictionary.ParentTree) != null;

            if (debug) {
                System.out.println("hastree=" + hasTree);
            }

            //choose appropriate method
            if (hasTree) {
                
                /*
                 * scan PDF and add nodes to XML tree
                 */
                buildTree(structTreeRootObj);

                //flush all objects
                pageStreams.clear();

            } else { //from the page stream

                try {
                    decodePageForMarkedContent(1, null, doc);
                } catch (final Exception e) {
                    LogWriter.writeLog("Exception: " + e.getMessage());
                }
            }
        }

        return doc;

    }

    /**
     * create a blank XML structure and a root. Add comment to say created by JPedal
     */
    private void setupTree() {

        try {
            final DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
            db = dbf.newDocumentBuilder();
        } catch (final ParserConfigurationException e) {
            LogWriter.writeLog("Exception: " + e.getMessage());
        }

        doc = db.newDocument();

        doc.appendChild(doc.createComment(" Created from JPedal "));
        doc.appendChild(doc.createComment(" http://www.idrsolutions.com "));

    }

    /**
     * scan down PDF struct object, creating XML tree
     */
    private void buildTree(final PdfObject structTreeRootObj) {

        final PdfObject RoleMap = structTreeRootObj.getDictionary(PdfDictionary.RoleMap);

        if (RoleMap != null) {
            readRoleMap(RoleMap);
        }

        root = doc.createElement("TaggedPDF-doc");
        doc.appendChild(root);

        traverseContentTree(structTreeRootObj);
    }

    private void readRoleMap(final PdfObject roleMap) {

        String key, value;

        final PdfKeyPairsIterator keyPairs = roleMap.getKeyPairsIterator();

        while (keyPairs.hasMorePairs()) {
            key = keyPairs.getNextKeyAsString();
            value = keyPairs.getNextValueAsString();
            rolemapLookup.put(key, value);
            keyPairs.nextPair();
        }
    }

    public void traverseContentTree(final PdfObject structTreeRootObj) {
        /*
         * read struct K value and decide what type
         * (can be dictionary or Array so we check both options)
         */
        final PdfObject K = structTreeRootObj.getDictionary(PdfDictionary.K);
        if (K == null) {
            final PdfArrayIterator Karray = structTreeRootObj.getMixedArray(PdfDictionary.K);

            if (debug) {
                System.out.println("Karray=");
            }


            readKarray(Karray, root, null, "");

            if (debug) {
                System.out.println("Karray read");
            }

        } else {

            if (debug) {
                System.out.println("read child=" + K.getObjectRefAsString());
            }

            readChildNode(K, root, null, "");
        }
    }

    private void readChildNode(final PdfObject K, final Element root, Map pageStream, String fullS) {

        if (debug) {
            indent += "   ";
            System.out.println(indent + "read child node " + K.getObjectRefAsString() + ' ' + K.getInt(PdfDictionary.K));
        }

        final PdfObject Pg;
        final PdfArrayIterator Karray = K.getMixedArray(PdfDictionary.K);
        final int Kint = K.getInt(PdfDictionary.K);

        final PdfObject Kdict = K.getDictionary(PdfDictionary.K);

        final String lang = K.getTextStreamValue(PdfDictionary.Lang);
        String S = K.getName(PdfDictionary.S);

        if (rolemapLookup.containsKey(S)) {
            S = rolemapLookup.get(S);
        }
        fullS = fullS + '.' + S;

        Element child = null;

        if (debug) {
            System.out.println(indent + "S= " + S + ' ');

            if (S == null) {
                System.out.println("S is null in " + K.getObjectRefAsString());

            }
        }


        //add child but collapse /Span into main Tag
        if (S != null) {
            if (S.equals("Span")) {
                child = root;
            } else {
                if (doc != null) {
                    child = doc.createElement(cleanName(S));
                }
                if (lang != null) {
                    child.setAttribute("xml:lang", lang);
                }

                if (root != null) {
                    root.appendChild(child);
                }
            }
        }

        //get page object
        Pg = K.getDictionary(PdfDictionary.Pg);

        if (Pg != null && pageStream == null && !isHTML) {

            //if not yet decoded, get values from it see if cached and decode if not
            //pageStream=(Map)pageStreams.get(Pg);


            if (debug) {
                System.out.println(indent + "decode page ");
            }


            pageStream = new HashMap();
            try {
                decodePageForMarkedContent(-1, Pg, pageStream); //-1 deliberate bum value as should not be used

                //20130717 - disabled by Mark for memory issues
                //ie Postgres_Plus_Cloud_Database_Getting_Started_Guide_20130219.pdf
                //  pageStreams.put(Pg,pageStream);

            } catch (final Exception e) {
                LogWriter.writeLog("Exception: " + e.getMessage());
            }
        }


        if (debug) {
            System.out.println(indent + "page decoded karray" + Karray + " Kdict=" + Kdict + " kint=" + Kint);
        }


        if (Karray != null) {
            readKarray(Karray, child, pageStream, fullS);
        } else if (Kdict != null) {
            readChildNode(Kdict, child, pageStream, fullS);
        } else if (Kint != -1 && !isHTML) { // actual value

            //reached the bottom so allow recursion to unwind naturally
            addContentToNode(pageStream, String.valueOf(Kint), child);
        } else if (K.getTextStreamValue(PdfDictionary.T) != null) {
            //System.out.println("ANnot");
        } else if (debug) {
            System.out.println("unimplemented " + K.getObjectRefAsString());
        }

        if (debug) {
            System.out.println(indent + "child node read " + K.getObjectRefAsString());

            indent = indent.substring(0, indent.length() - 3);
        }
    }

    private void addContentToNode(final Map pageStream, final String Kint, final Element child) {

        if (!isHTML) {

            String text = (String) pageStream.get(Kint);

            if (text != null) {
                text = handleXMLCharacters(text);

                if (doc != null) {
                    final Text textNode = doc.createTextNode(text);
                    child.appendChild(textNode);
                }
            }

            if (debug) {
                System.out.println(indent + " added " + text);
            }
        }
    }

    private static String handleXMLCharacters(String text) {
        text = text.replaceAll("<", "<");
        text = text.replaceAll(">", ">");

        return text;
    }

    private void readKarray(final PdfArrayIterator Karray, final Element root, final Map pageStream, final String fullS) {

        final int count = Karray.getTokenCount();
        PdfObject kidObj;
        String KValue;

        for (int i = 0; i < count; i++) {
            final byte[] Kbyte = Karray.getNextValueAsByte(true);
            KValue = new String(Kbyte);

            if (debug) {
                System.out.println(indent + "aK value=" + KValue);
            }

            if (KValue.contains("R")) {
                kidObj = getMCObjectFromRefOrDirect(currentPdfFile, Kbyte);
                currentPdfFile.readObject(kidObj);
                readChildNode(kidObj, root, pageStream, fullS);
            } else {
                if (isHTML) {
                    if (!reverseLookup.containsKey(KValue)) {
                        reverseLookup.put(KValue, fullS);
                    }
                }

                addContentToNode(pageStream, KValue, root);
            }
        }
    }

    private static String cleanName(String s) {
        //make sure S is valid XML

        final StringBuilder cleanedS = new StringBuilder(10);

        final int length = s.length();
        for (int i = 0; i < length; i++) {
            char c = s.charAt(i);

            //translate any hex values ( #xx ) into chars
            if (c == '#') { //assume 2 byte
                final StringBuilder num = new StringBuilder(2);
                for (int j = 0; j < 2; j++) { //read number
                    i++;
                    num.append(s.charAt(i));
                }
                //Convert from hex value as string into ASCI char
                c = (char) Integer.parseInt(num.toString(), 16);

                //hard-coded based on Adobe output
                if (!Character.isLetterOrDigit(c)) {
                    c = '-';
                }

            }

            //remap spaces
            if (c == ' ') {
                cleanedS.append('-');
            } else if (c == '-') // hard-coded
            {
                cleanedS.append(c);
            } else if (c == '_') {
                cleanedS.append(c);
            } else if (Character.isLetterOrDigit(c)) //reject non-valid
            {
                cleanedS.append(c);
            }
        }

        s = cleanedS.toString();
        return s;
    }

    /**
     * extract marked content
     */
    private synchronized void decodePageForMarkedContent(int pageNumber, PdfObject pdfObject, final Object pageStream) throws Exception {

        if (isDecoding) {

            LogWriter.writeLog("[PDF]WARNING - this file is being decoded already");

        } else {

            //if no tree use page
            if (pdfObject == null) {
                final String currentPageOffset = currentPdfFile.getReferenceforPage(pageNumber);

                pdfObject = new PageObject(currentPageOffset);
                currentPdfFile.readObject(pdfObject);

            } else {
                pageNumber = currentPdfFile.convertObjectToPageNumber(new String(pdfObject.getUnresolvedData()));
                currentPdfFile.checkResolved(pdfObject);
            }

            try {
                isDecoding = true;

                final ObjectStore objectStoreRef = new ObjectStore();

                final PdfStreamDecoder current = new PdfStreamDecoder(currentPdfFile, layers);
                current.setParameters(true, false, 0, PdfDecoderInt.TEXT + PdfDecoderInt.RAWIMAGES + PdfDecoderInt.FINALIMAGES, false, false);
                current.setXMLExtraction(false);
                current.setObjectValue(ValueTypes.Name, "markedContent");
                current.setObjectValue(ValueTypes.ObjectStore, objectStoreRef);
                current.setObjectValue(ValueTypes.StatusBar, null);
                current.setObjectValue(ValueTypes.PDFPageData, pdfPageData);
                current.setIntValue(ValueTypes.PageNum, pageNumber);
                current.setRenderer(new SwingDisplay(pageNumber, objectStoreRef, false));

                res.setupResources(current, false, pdfObject.getDictionary(PdfDictionary.Resources), pageNumber, currentPdfFile);

                current.setObjectValue(ValueTypes.MarkedContent, pageStream);

                if (debug) {
                    System.out.println(indent + " about to decode page " + pdfObject.getObjectRefAsString());
                }

                current.decodePageContent(pdfObject);

                objectStoreRef.flush();

            } catch (final Exception e) {
                LogWriter.writeLog("Exception: " + e.getMessage());
            } finally {
                isDecoding = false;
            }
        }
    }

    private static MCObject getMCObjectFromRefOrDirect(final PdfObjectReader currentPdfFile, final byte[] data) {

        final MCObject mcObj = new MCObject(new String(data));

        if (data[0] == '<') {
            mcObj.setStatus(PdfObject.UNDECODED_DIRECT);
        } else {
            mcObj.setStatus(PdfObject.UNDECODED_REF);
        }
        mcObj.setUnresolvedData(data, PdfDictionary.MCID);

        final ObjectDecoder objectDecoder = new ObjectDecoder(currentPdfFile.getObjectReader());
        objectDecoder.checkResolved(mcObj);

        return mcObj;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy