package com.tom_roush.pdfbox.pdmodel.fdf; 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.COSInteger; 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.cos.COSString; import com.tom_roush.pdfbox.pdmodel.common.COSArrayList; import com.tom_roush.pdfbox.pdmodel.common.COSObjectable; import com.tom_roush.pdfbox.pdmodel.interactive.action.PDAction; import com.tom_roush.pdfbox.pdmodel.interactive.action.PDActionFactory; import com.tom_roush.pdfbox.pdmodel.interactive.action.PDAdditionalActions; import com.tom_roush.pdfbox.pdmodel.interactive.annotation.PDAppearanceDictionary; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import java.io.IOException; import java.io.Writer; import java.util.ArrayList; import java.util.List; /** * This represents an FDF field that is part of the FDF document. * * @author Ben Litchfield */ public class FDFField implements COSObjectable { private COSDictionary field; /** * Default constructor. */ public FDFField() { field = new COSDictionary(); } /** * Constructor. * * @param f The FDF field. */ public FDFField( COSDictionary f ) { field = f; } /** * This will create an FDF field from an XFDF XML document. * * @param fieldXML The XML document that contains the XFDF data. * @throws IOException If there is an error reading from the dom. */ public FDFField( Element fieldXML ) throws IOException { this(); this.setPartialFieldName( fieldXML.getAttribute( "name" ) ); NodeList nodeList = fieldXML.getChildNodes(); List<FDFField> kids = new ArrayList<FDFField>(); 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( "value" ) ) { setValue( XMLUtil.getNodeValue( child ) ); } else if( child.getTagName().equals( "value-richtext" ) ) { setRichText(new COSString(XMLUtil.getNodeValue(child))); } else if( child.getTagName().equals( "field" ) ) { kids.add( new FDFField( child ) ); } } } if( kids.size() > 0 ) { setKids( kids ); } } /** * 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 { output.write("<field name=\"" + getPartialFieldName() + "\">\n"); Object value = getValue(); if( value != null ) { if(value instanceof COSString) { output.write("<value>" + escapeXML(((COSString) value).getString()) + "</value>\n"); } else if(value instanceof COSStream) { output.write("<value>" + escapeXML(((COSStream) value).getString()) + "</value>\n"); } } String rt = getRichText(); if( rt != null ) { output.write("<value-richtext>" + escapeXML(rt) + "</value-richtext>\n"); } List<FDFField> kids = getKids(); if( kids != null ) { for (FDFField kid : kids) { kid.writeXML(output); } } output.write( "</field>\n"); } /** * Convert this standard java object to a COS object. * * @return The cos object that matches this Java object. */ @Override public COSDictionary getCOSObject() { return field; } /** * This will get the list of kids. This will return a list of FDFField objects. * This will return null if the underlying list is null. * * @return The list of kids. */ public List<FDFField> getKids() { COSArray kids = (COSArray)field.getDictionaryObject( COSName.KIDS ); List<FDFField> retval = null; if( kids != null ) { List<FDFField> actuals = new ArrayList<FDFField>(); for( int i=0; i<kids.size(); i++ ) { actuals.add( new FDFField( (COSDictionary)kids.getObject( i ) ) ); } retval = new COSArrayList<FDFField>( actuals, kids ); } return retval; } /** * This will set the list of kids. * * @param kids A list of FDFField objects. */ public void setKids( List<FDFField> kids ) { field.setItem( COSName.KIDS, COSArrayList.converterToCOSArray( kids ) ); } /** * This will get the "T" entry in the field dictionary. A partial field * name. Where the fully qualified field name is a concatenation of * the parent's fully qualified field name and "." as a separator. For example<br/> * Address.State<br /> * Address.City<br /> * * @return The partial field name. */ public String getPartialFieldName() { return field.getString( COSName.T ); } /** * This will set the partial field name. * * @param partial The partial field name. */ public void setPartialFieldName( String partial ) { field.setString( COSName.T, partial ); } /** * This will get the value for the field. This will return type will either be <br /> * String : Checkboxes, Radio Button <br /> * java.util.List of strings: Choice Field * PDTextStream: Textfields * * @return The value of the field. * @throws IOException If there is an error getting the value. */ public Object getValue() throws IOException { COSBase value = field.getDictionaryObject( COSName.V ); if( value instanceof COSName ) { return ((COSName) value).getName(); } else if( value instanceof COSArray ) { return COSArrayList.convertCOSStringCOSArrayToList((COSArray) value); } else if( value instanceof COSString || value instanceof COSStream ) { return value; } else if( value != null ) { throw new IOException( "Error:Unknown type for field import" + value ); } else { return null; } } /** * Returns the COS value of this field. * * @return The COS value of the field. * @throws IOException If there is an error getting the value. */ public COSBase getCOSValue() throws IOException { COSBase value = field.getDictionaryObject(COSName.V); if (value instanceof COSName) { return value; } else if (value instanceof COSArray) { return value; } else if (value instanceof COSString || value instanceof COSStream) { return value; } else if (value != null) { throw new IOException("Error:Unknown type for field import" + value); } else { return null; } } /** * You should pass in a string, or a java.util.List of strings to set the * value. * * @param value The value that should populate when imported. * * @throws IOException If there is an error setting the value. */ public void setValue( Object value ) throws IOException { COSBase cos = null; if( value instanceof List ) { cos = COSArrayList.convertStringListToCOSStringCOSArray( (List<String>)value ); } else if( value instanceof String ) { cos = COSName.getPDFName( (String)value ); } else if( value instanceof COSObjectable ) { cos = ((COSObjectable)value).getCOSObject(); } else if( value != null ) { throw new IOException( "Error:Unknown type for field import" + value ); } field.setItem(COSName.V, cos); } /** * Sets the COS value of this field. * * @param value COS value. */ public void setValue(COSBase value) { field.setItem(COSName.V, value); } /** * This will get the Ff entry of the cos dictionary. If it it not present then * this method will return null. * * @return The field flags. */ public Integer getFieldFlags() { Integer retval = null; COSNumber ff = (COSNumber)field.getDictionaryObject( COSName.FF ); if( ff != null ) { retval = ff.intValue(); } return retval; } /** * This will get the field flags that are associated with this field. The Ff entry * in the FDF field dictionary. * * @param ff The new value for the field flags. */ public void setFieldFlags( Integer ff ) { COSInteger value = null; if( ff != null ) { value = COSInteger.get( ff ); } field.setItem( COSName.FF, value ); } /** * This will get the field flags that are associated with this field. The Ff entry * in the FDF field dictionary. * * @param ff The new value for the field flags. */ public void setFieldFlags( int ff ) { field.setInt( COSName.FF, ff ); } /** * This will get the SetFf entry of the cos dictionary. If it it not present then * this method will return null. * * @return The field flags. */ public Integer getSetFieldFlags() { Integer retval = null; COSNumber ff = (COSNumber)field.getDictionaryObject( COSName.SET_FF ); if( ff != null ) { retval = ff.intValue(); } return retval; } /** * This will get the field flags that are associated with this field. The SetFf entry * in the FDF field dictionary. * * @param ff The new value for the "set field flags". */ public void setSetFieldFlags( Integer ff ) { COSInteger value = null; if( ff != null ) { value = COSInteger.get( ff ); } field.setItem( COSName.SET_FF, value ); } /** * This will get the field flags that are associated with this field. The SetFf entry * in the FDF field dictionary. * * @param ff The new value for the "set field flags". */ public void setSetFieldFlags( int ff ) { field.setInt( COSName.SET_FF, ff ); } /** * This will get the ClrFf entry of the cos dictionary. If it it not present then * this method will return null. * * @return The field flags. */ public Integer getClearFieldFlags() { Integer retval = null; COSNumber ff = (COSNumber)field.getDictionaryObject( COSName.CLR_FF ); if( ff != null ) { retval = ff.intValue(); } return retval; } /** * This will get the field flags that are associated with this field. The ClrFf entry * in the FDF field dictionary. * * @param ff The new value for the "clear field flags". */ public void setClearFieldFlags( Integer ff ) { COSInteger value = null; if( ff != null ) { value = COSInteger.get( ff ); } field.setItem( COSName.CLR_FF, value ); } /** * This will get the field flags that are associated with this field. The ClrFf entry * in the FDF field dictionary. * * @param ff The new value for the "clear field flags". */ public void setClearFieldFlags( int ff ) { field.setInt( COSName.CLR_FF, ff ); } /** * This will get the F entry of the cos dictionary. If it it not present then * this method will return null. * * @return The widget field flags. */ public Integer getWidgetFieldFlags() { Integer retval = null; COSNumber f = (COSNumber)field.getDictionaryObject( "F" ); if( f != null ) { retval = f.intValue(); } return retval; } /** * This will get the widget field flags that are associated with this field. The F entry * in the FDF field dictionary. * * @param f The new value for the field flags. */ public void setWidgetFieldFlags( Integer f ) { COSInteger value = null; if( f != null ) { value = COSInteger.get( f ); } field.setItem( COSName.F, value ); } /** * This will get the field flags that are associated with this field. The F entry * in the FDF field dictionary. * * @param f The new value for the field flags. */ public void setWidgetFieldFlags( int f ) { field.setInt( COSName.F, f ); } /** * This will get the SetF entry of the cos dictionary. If it it not present then * this method will return null. * * @return The field flags. */ public Integer getSetWidgetFieldFlags() { Integer retval = null; COSNumber ff = (COSNumber)field.getDictionaryObject( COSName.SET_F ); if( ff != null ) { retval = ff.intValue(); } return retval; } /** * This will get the widget field flags that are associated with this field. The SetF entry * in the FDF field dictionary. * * @param ff The new value for the "set widget field flags". */ public void setSetWidgetFieldFlags( Integer ff ) { COSInteger value = null; if( ff != null ) { value = COSInteger.get( ff ); } field.setItem( COSName.SET_F, value ); } /** * This will get the widget field flags that are associated with this field. The SetF entry * in the FDF field dictionary. * * @param ff The new value for the "set widget field flags". */ public void setSetWidgetFieldFlags( int ff ) { field.setInt( COSName.SET_F, ff ); } /** * This will get the ClrF entry of the cos dictionary. If it it not present then * this method will return null. * * @return The widget field flags. */ public Integer getClearWidgetFieldFlags() { Integer retval = null; COSNumber ff = (COSNumber)field.getDictionaryObject( COSName.CLR_F ); if( ff != null ) { retval = ff.intValue(); } return retval; } /** * This will get the field flags that are associated with this field. The ClrF entry * in the FDF field dictionary. * * @param ff The new value for the "clear widget field flags". */ public void setClearWidgetFieldFlags( Integer ff ) { COSInteger value = null; if( ff != null ) { value = COSInteger.get( ff ); } field.setItem( COSName.CLR_F, value ); } /** * This will get the field flags that are associated with this field. The ClrF entry * in the FDF field dictionary. * * @param ff The new value for the "clear field flags". */ public void setClearWidgetFieldFlags( int ff ) { field.setInt( COSName.CLR_F, ff ); } /** * This will get the appearance dictionary that specifies the appearance of * a pushbutton field. * * @return The AP entry of this dictionary. */ public PDAppearanceDictionary getAppearanceDictionary() { PDAppearanceDictionary retval = null; COSDictionary dict = (COSDictionary)field.getDictionaryObject( COSName.AP ); if( dict != null ) { retval = new PDAppearanceDictionary( dict ); } return retval; } /** * This will set the appearance dictionary. * * @param ap The apperance dictionary. */ public void setAppearanceDictionary( PDAppearanceDictionary ap ) { field.setItem( COSName.AP, ap ); } /** * This will get named page references.. * * @return The named page references. */ public FDFNamedPageReference getAppearanceStreamReference() { FDFNamedPageReference retval = null; COSDictionary ref = (COSDictionary)field.getDictionaryObject( COSName.AP_REF ); if( ref != null ) { retval = new FDFNamedPageReference( ref ); } return retval; } /** * This will set the named page references. * * @param ref The named page references. */ public void setAppearanceStreamReference( FDFNamedPageReference ref ) { field.setItem( COSName.AP_REF, ref ); } /** * This will get the icon fit that is associated with this field. * * @return The IF entry. */ public FDFIconFit getIconFit() { FDFIconFit retval = null; COSDictionary dic = (COSDictionary)field.getDictionaryObject( "IF" ); if( dic != null ) { retval = new FDFIconFit( dic ); } return retval; } /** * This will set the icon fit entry. * * @param fit The icon fit object. */ public void setIconFit( FDFIconFit fit ) { field.setItem( COSName.IF, fit ); } /** * This will return a list of options for a choice field. The value in the * list will be 1 of 2 types. java.lang.String or FDFOptionElement. * * @return A list of all options. */ public List<Object> getOptions() { List<Object> retval = null; COSArray array = (COSArray)field.getDictionaryObject( COSName.OPT ); if( array != null ) { List<Object> objects = new ArrayList<Object>(); for( int i=0; i<array.size(); i++ ) { COSBase next = array.getObject( i ); if( next instanceof COSString ) { objects.add( ((COSString)next).getString() ); } else { COSArray value = (COSArray)next; objects.add( new FDFOptionElement( value ) ); } } retval = new COSArrayList<Object>( objects, array ); } return retval; } /** * This will set the options for the choice field. The objects in the list * should either be java.lang.String or FDFOptionElement. * * @param options The options to set. */ public void setOptions( List options ) { COSArray value = COSArrayList.converterToCOSArray( options ); field.setItem( COSName.OPT, value ); } /** * This will get the action that is associated with this field. * * @return The A entry in the field dictionary. */ public PDAction getAction() { return PDActionFactory.createAction( (COSDictionary)field.getDictionaryObject( COSName.A ) ); } /** * This will set the action that is associated with this field. * * @param a The new action. */ public void setAction( PDAction a ) { field.setItem( COSName.A, a ); } /** * This will get a list of additional actions that will get executed based * on events. * * @return The AA entry in this field dictionary. */ public PDAdditionalActions getAdditionalActions() { PDAdditionalActions retval = null; COSDictionary dict = (COSDictionary)field.getDictionaryObject( COSName.AA ); if( dict != null ) { retval = new PDAdditionalActions( dict ); } return retval; } /** * This will set the additional actions that are associated with this field. * * @param aa The additional actions. */ public void setAdditionalActions( PDAdditionalActions aa ) { field.setItem( COSName.AA, aa ); } /** * This will set the rich text that is associated with this field. * * @return The rich text XHTML stream. */ public String getRichText() { COSBase rv = field.getDictionaryObject( COSName.RV ); if (rv == null) { return null; } else if (rv instanceof COSString) { return ((COSString) rv).getString(); } else { return ((COSStream) rv).getString(); } } /** * This will set the rich text value. * * @param rv The rich text value for the stream. */ public void setRichText(COSString rv) { field.setItem(COSName.RV, rv); } /** * This will set the rich text value. * * @param rv The rich text value for the stream. */ public void setRichText( COSStream rv ) { field.setItem( COSName.RV, rv ); } /** * Escape special characters. * * @param input the string to be escaped * * @return the resulting string */ private String escapeXML(String input) { StringBuilder escapedXML = new StringBuilder(); for (int i = 0; i < input.length(); i++) { char c = input.charAt(i); switch (c) { case '<': escapedXML.append("<"); break; case '>': escapedXML.append(">"); break; case '\"': escapedXML.append("""); break; case '&': escapedXML.append("&"); break; case '\'': escapedXML.append("'"); break; default: if (c > 0x7e) { escapedXML.append("&#").append((int) c).append(";"); } else { escapedXML.append(c); } } } return escapedXML.toString(); } }