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

org.apache.pdfbox.pdmodel.fdf.FDFDictionary Maven / Gradle / Ivy

Go to download

The Apache PDFBox library is an open source Java tool for working with PDF documents.

There is a newer version: 3.0.2
Show newest version
/*
 * 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.pdfbox.pdmodel.fdf;

import java.io.IOException;
import java.io.Writer;
import java.util.ArrayList;
import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.pdfbox.cos.COSArray;
import org.apache.pdfbox.cos.COSDictionary;
import org.apache.pdfbox.cos.COSName;
import org.apache.pdfbox.cos.COSStream;
import org.apache.pdfbox.cos.COSString;
import org.apache.pdfbox.pdmodel.common.COSObjectable;
import org.apache.pdfbox.pdmodel.common.COSArrayList;
import org.apache.pdfbox.pdmodel.common.filespecification.PDFileSpecification;
import org.apache.pdfbox.pdmodel.common.filespecification.PDSimpleFileSpecification;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

/**
 * This represents an FDF dictionary that is part of the FDF document.
 *
 * @author Ben Litchfield
 */
public class FDFDictionary implements COSObjectable
{

    private static final Log LOG = LogFactory.getLog(FDFDictionary.class);

    private COSDictionary fdf;

    /**
     * Default constructor.
     */
    public FDFDictionary()
    {
        fdf = new COSDictionary();
    }

    /**
     * Constructor.
     *
     * @param fdfDictionary The FDF documents catalog.
     */
    public FDFDictionary(COSDictionary fdfDictionary)
    {
        fdf = fdfDictionary;
    }

    /**
     * This will create an FDF dictionary from an XFDF XML document.
     *
     * @param fdfXML The XML document that contains the XFDF data.
     * @throws IOException If there is an error reading from the dom.
     */
    public FDFDictionary(Element fdfXML)
    {
        this();
        NodeList nodeList = fdfXML.getChildNodes();
        for (int i = 0; i < nodeList.getLength(); i++)
        {
            Node node = nodeList.item(i);
            if (node instanceof Element)
            {
                Element child = (Element) node;
                if (child.getTagName().equals("f"))
                {
                    PDSimpleFileSpecification fs = new PDSimpleFileSpecification();
                    fs.setFile(child.getAttribute("href"));
                    setFile(fs);
                }
                else if (child.getTagName().equals("ids"))
                {
                    COSArray ids = new COSArray();
                    String original = child.getAttribute("original");
                    String modified = child.getAttribute("modified");
                    try
                    {
                        ids.add(COSString.parseHex(original));
                    }
                    catch (IOException e)
                    {
                        LOG.warn("Error parsing ID entry for attribute 'original' [" + original
                                + "]. ID entry ignored.", e);
                    }
                    try
                    {
                        ids.add(COSString.parseHex(modified));
                    }
                    catch (IOException e)
                    {
                        LOG.warn("Error parsing ID entry for attribute 'modified' [" + modified
                                + "]. ID entry ignored.", e);
                    }
                    setID(ids);
                }
                else if (child.getTagName().equals("fields"))
                {
                    NodeList fields = child.getChildNodes();
                    List fieldList = new ArrayList();
                    for (int f = 0; f < fields.getLength(); f++)
                    {
                        Node currentNode = fields.item(f);
                        if (currentNode instanceof Element
                                && ((Element) currentNode).getTagName().equals("field"))
                        {
                            try
                            {
                                fieldList.add(new FDFField((Element) fields.item(f)));
                            }
                            catch (IOException e)
                            {
                                LOG.warn("Error parsing field entry [" + currentNode.getNodeValue()
                                        + "]. Field ignored.", e);
                            }
                        }
                    }
                    setFields(fieldList);
                }
                else if (child.getTagName().equals("annots"))
                {
                    NodeList annots = child.getChildNodes();
                    List annotList = new ArrayList();
                    for (int j = 0; j < annots.getLength(); j++)
                    {
                        Node annotNode = annots.item(j);
                        if (annotNode instanceof Element)
                        {

                            // the node name defines the annotation type
                            Element annot = (Element) annotNode;
                            String annotationName = annot.getNodeName();
                            try
                            {
                                if (annotationName.equals("text"))
                                {
                                    annotList.add(new FDFAnnotationText(annot));
                                }
                                else if (annotationName.equals("caret"))
                                {
                                    annotList.add(new FDFAnnotationCaret(annot));
                                }
                                else if (annotationName.equals("freetext"))
                                {
                                    annotList.add(new FDFAnnotationFreeText(annot));
                                }
                                else if (annotationName.equals("fileattachment"))
                                {
                                    annotList.add(new FDFAnnotationFileAttachment(annot));
                                }
                                else if (annotationName.equals("highlight"))
                                {
                                    annotList.add(new FDFAnnotationHighlight(annot));
                                }
                                else if (annotationName.equals("ink"))
                                {
                                    annotList.add(new FDFAnnotationInk(annot));
                                }
                                else if (annotationName.equals("line"))
                                {
                                    annotList.add(new FDFAnnotationLine(annot));
                                }
                                else if (annotationName.equals("link"))
                                {
                                    annotList.add(new FDFAnnotationLink(annot));
                                }
                                else if (annotationName.equals("circle"))
                                {
                                    annotList.add(new FDFAnnotationCircle(annot));
                                }
                                else if (annotationName.equals("square"))
                                {
                                    annotList.add(new FDFAnnotationSquare(annot));
                                }
                                else if (annotationName.equals("polygon"))
                                {
                                    annotList.add(new FDFAnnotationPolygon(annot));
                                }
                                else if (annotationName.equals("polyline"))
                                {
                                    annotList.add(new FDFAnnotationPolyline(annot));
                                }
                                else if (annotationName.equals("sound"))
                                {
                                    annotList.add(new FDFAnnotationSound(annot));
                                }
                                else if (annotationName.equals("squiggly"))
                                {
                                    annotList.add(new FDFAnnotationSquiggly(annot));
                                }
                                else if (annotationName.equals("stamp"))
                                {
                                    annotList.add(new FDFAnnotationStamp(annot));
                                }
                                else if (annotationName.equals("strikeout"))
                                {
                                    annotList.add(new FDFAnnotationStrikeOut(annot));
                                }
                                else if (annotationName.equals("underline"))
                                {
                                    annotList.add(new FDFAnnotationUnderline(annot));
                                }
                                else
                                {
                                    LOG.warn("Unknown or unsupported annotation type '"
                                            + annotationName + "'");
                                }
                            }
                            catch (IOException e)
                            {
                                LOG.warn(
                                        "Error parsing annotation information ["
                                                + annot.getNodeValue() + "]. Annotation ignored", e);
                            }
                        }
                    }
                    setAnnotations(annotList);
                }
            }
        }
    }

    /**
     * This will write this element as an XML document.
     *
     * @param output The stream to write the xml to.
     *
     * @throws IOException If there is an error writing the XML.
     */
    public void writeXML(Writer output) throws IOException
    {
        PDFileSpecification fs = this.getFile();
        if (fs != null)
        {
            output.write("\n");
        }
        COSArray ids = this.getID();
        if (ids != null)
        {
            COSString original = (COSString) ids.getObject(0);
            COSString modified = (COSString) ids.getObject(1);
            output.write("\n");
        }
        List fields = getFields();
        if (fields != null && fields.size() > 0)
        {
            output.write("\n");
            for (FDFField field : fields)
            {
                field.writeXML(output);
            }
            output.write("\n");
        }
    }

    /**
     * Convert this standard java object to a COS object.
     *
     * @return The cos object that matches this Java object.
     */
    @Override
    public COSDictionary getCOSObject()
    {
        return fdf;
    }

    /**
     * The source file or target file: the PDF document file that this FDF file was exported from or is intended to be
     * imported into.
     *
     * @return The F entry of the FDF dictionary.
     *
     * @throws IOException If there is an error creating the file spec.
     */
    public PDFileSpecification getFile() throws IOException
    {
        return PDFileSpecification.createFS(fdf.getDictionaryObject(COSName.F));
    }

    /**
     * This will set the file specification.
     *
     * @param fs The file specification.
     */
    public void setFile(PDFileSpecification fs)
    {
        fdf.setItem(COSName.F, fs);
    }

    /**
     * This is the FDF id.
     *
     * @return The FDF ID.
     */
    public COSArray getID()
    {
        return (COSArray) fdf.getDictionaryObject(COSName.ID);
    }

    /**
     * This will set the FDF id.
     *
     * @param id The new id for the FDF.
     */
    public void setID(COSArray id)
    {
        fdf.setItem(COSName.ID, id);
    }

    /**
     * This will get the list of FDF Fields. This will return a list of FDFField objects.
     *
     * @return A list of FDF fields.
     */
    public List getFields()
    {
        List retval = null;
        COSArray fieldArray = (COSArray) fdf.getDictionaryObject(COSName.FIELDS);
        if (fieldArray != null)
        {
            List fields = new ArrayList();
            for (int i = 0; i < fieldArray.size(); i++)
            {
                fields.add(new FDFField((COSDictionary) fieldArray.getObject(i)));
            }
            retval = new COSArrayList(fields, fieldArray);
        }
        return retval;
    }

    /**
     * This will set the list of fields. This should be a list of FDFField objects.
     *
     * @param fields The list of fields.
     */
    public void setFields(List fields)
    {
        fdf.setItem(COSName.FIELDS, COSArrayList.converterToCOSArray(fields));
    }

    /**
     * This will get the status string to be displayed as the result of an action.
     *
     * @return The status.
     */
    public String getStatus()
    {
        return fdf.getString(COSName.STATUS);
    }

    /**
     * This will set the status string.
     *
     * @param status The new status string.
     */
    public void setStatus(String status)
    {
        fdf.setString(COSName.STATUS, status);
    }

    /**
     * This will get the list of FDF Pages. This will return a list of FDFPage objects.
     *
     * @return A list of FDF pages.
     */
    public List getPages()
    {
        List retval = null;
        COSArray pageArray = (COSArray) fdf.getDictionaryObject(COSName.PAGES);
        if (pageArray != null)
        {
            List pages = new ArrayList();
            for (int i = 0; i < pageArray.size(); i++)
            {
                pages.add(new FDFPage((COSDictionary) pageArray.get(i)));
            }
            retval = new COSArrayList(pages, pageArray);
        }
        return retval;
    }

    /**
     * This will set the list of pages. This should be a list of FDFPage objects.
     *
     *
     * @param pages The list of pages.
     */
    public void setPages(List pages)
    {
        fdf.setItem(COSName.PAGES, COSArrayList.converterToCOSArray(pages));
    }

    /**
     * The encoding to be used for a FDF field. The default is PDFDocEncoding and this method will never return null.
     *
     * @return The encoding value.
     */
    public String getEncoding()
    {
        String encoding = fdf.getNameAsString(COSName.ENCODING);
        if (encoding == null)
        {
            encoding = "PDFDocEncoding";
        }
        return encoding;

    }

    /**
     * This will set the encoding.
     *
     * @param encoding The new encoding.
     */
    public void setEncoding(String encoding)
    {
        fdf.setName(COSName.ENCODING, encoding);
    }

    /**
     * This will get the list of FDF Annotations. This will return a list of FDFAnnotation objects or null if the entry
     * is not set.
     *
     * @return A list of FDF annotations.
     *
     * @throws IOException If there is an error creating the annotation list.
     */
    public List getAnnotations() throws IOException
    {
        List retval = null;
        COSArray annotArray = (COSArray) fdf.getDictionaryObject(COSName.ANNOTS);
        if (annotArray != null)
        {
            List annots = new ArrayList();
            for (int i = 0; i < annotArray.size(); i++)
            {
                annots.add(FDFAnnotation.create((COSDictionary) annotArray.getObject(i)));
            }
            retval = new COSArrayList(annots, annotArray);
        }
        return retval;
    }

    /**
     * This will set the list of annotations. This should be a list of FDFAnnotation objects.
     *
     *
     * @param annots The list of annotations.
     */
    public void setAnnotations(List annots)
    {
        fdf.setItem(COSName.ANNOTS, COSArrayList.converterToCOSArray(annots));
    }

    /**
     * This will get the incremental updates since the PDF was last opened.
     *
     * @return The differences entry of the FDF dictionary.
     */
    public COSStream getDifferences()
    {
        return (COSStream) fdf.getDictionaryObject(COSName.DIFFERENCES);
    }

    /**
     * This will set the differences stream.
     *
     * @param diff The new differences stream.
     */
    public void setDifferences(COSStream diff)
    {
        fdf.setItem(COSName.DIFFERENCES, diff);
    }

    /**
     * This will get the target frame in the browser to open this document.
     *
     * @return The target frame.
     */
    public String getTarget()
    {
        return fdf.getString(COSName.TARGET);
    }

    /**
     * This will set the target frame in the browser to open this document.
     *
     * @param target The new target frame.
     */
    public void setTarget(String target)
    {
        fdf.setString(COSName.TARGET, target);
    }

    /**
     * This will get the list of embedded FDF entries, or null if the entry is null. This will return a list of
     * PDFileSpecification objects.
     *
     * @return A list of embedded FDF files.
     *
     * @throws IOException If there is an error creating the file spec.
     */
    public List getEmbeddedFDFs() throws IOException
    {
        List retval = null;
        COSArray embeddedArray = (COSArray) fdf.getDictionaryObject(COSName.EMBEDDED_FDFS);
        if (embeddedArray != null)
        {
            List embedded = new ArrayList();
            for (int i = 0; i < embeddedArray.size(); i++)
            {
                embedded.add(PDFileSpecification.createFS(embeddedArray.get(i)));
            }
            retval = new COSArrayList(embedded, embeddedArray);
        }
        return retval;
    }

    /**
     * This will set the list of embedded FDFs. This should be a list of PDFileSpecification objects.
     *
     *
     * @param embedded The list of embedded FDFs.
     */
    public void setEmbeddedFDFs(List embedded)
    {
        fdf.setItem(COSName.EMBEDDED_FDFS, COSArrayList.converterToCOSArray(embedded));
    }

    /**
     * This will get the java script entry.
     *
     * @return The java script entry describing javascript commands.
     */
    public FDFJavaScript getJavaScript()
    {
        FDFJavaScript fs = null;
        COSDictionary dic = (COSDictionary) fdf.getDictionaryObject(COSName.JAVA_SCRIPT);
        if (dic != null)
        {
            fs = new FDFJavaScript(dic);
        }
        return fs;
    }

    /**
     * This will set the JavaScript entry.
     *
     * @param js The javascript entries.
     */
    public void setJavaScript(FDFJavaScript js)
    {
        fdf.setItem(COSName.JAVA_SCRIPT, js);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy