org.jpedal.objects.acroforms.AcroRenderer Maven / Gradle / Ivy
Show all versions of OpenViewerFX Show documentation
/*
* ===========================================
* 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@
*
* ---------------
* AcroRenderer.java
* ---------------
*/
package org.jpedal.objects.acroforms;
import java.awt.image.BufferedImage;
import java.util.*;
import org.jpedal.exception.PdfException;
import org.jpedal.external.ExternalHandlers;
import org.jpedal.io.PdfObjectReader;
import org.jpedal.objects.Javascript;
import org.jpedal.objects.PdfPageData;
import org.jpedal.objects.acroforms.actions.ActionHandler;
import org.jpedal.objects.acroforms.creation.FormFactory;
import org.jpedal.objects.acroforms.creation.SwingFormCreator;
import org.jpedal.objects.acroforms.utils.FormUtils;
import org.jpedal.objects.layers.PdfLayerList;
import org.jpedal.objects.raw.*;
import org.jpedal.parser.FormFlattener;
import org.jpedal.parser.PdfStreamDecoder;
import org.jpedal.parser.PdfStreamDecoderForPrinting;
import org.jpedal.parser.PrintStreamDecoder;
import org.jpedal.utils.LogWriter;
import org.jpedal.utils.StringUtils;
/**
* Provides top level to forms handling, assisted by separate classes to
* decode widgets (FormDecoder - default implements Swing set)
* create Form widgets (implementation of FormFactory),
* store and render widgets (GUIData),
* handle Javascript and Actions (Javascript and ActionHandler)
* and support for Signature object
*/
public class AcroRenderer {
private static final boolean flattenForms, ignoreAllForms;
static {
String s = System.getProperty("org.jpedal.flattenForm");
flattenForms = s != null && s.equalsIgnoreCase("true");
s = System.getProperty("org.jpedal.ignoreAllForms");
ignoreAllForms = s != null && s.equalsIgnoreCase("true");
}
public static boolean FROM_JPDF2FORMS;
FormObject[] Fforms, Aforms;
private PdfObject AcroRes;
private float dpi = 72f;
private Object[] CO;
PdfArrayIterator fieldList;
private PdfArrayIterator[] annotList;
/**
* flag to show we ignore forms
*/
private boolean ignoreForms;
private boolean alwaysUseXFA;
/**
* creates all GUI components from raw data in PDF and stores in GUIData instance
*/
public FormFactory formFactory;
/**
* holder for all data (implementations to support Swing and ULC)
*/
public GUIData compData;
/**
* holds sig object so we can easily retrieve
*/
private Set sigObject;
private Map sigKeys; //and allow us to trap multiple if in both Annot and Form
/**
* holds copy of object to access the mediaBox and cropBox values
*/
private PdfPageData pageData;
/**
* number of entries in acroFormDataList, each entry can have a button group of more that one button
* 'A' annot and 'F' form - A is per page, F is total hence 3 variables
*/
private int[] AfieldCount;
private int ATotalCount, FfieldCount;
private int additionalOBjectRef;
private boolean addedMissingPopup;
/**
* number of pages in current PDF document
*/
int pageCount;
/**
* handle on object reader for decoding objects
*/
PdfObjectReader currentPdfFile;
/**
* parses and decodes PDF data into generic data for converting to widgets
*/
FormStream fDecoder;
/**
* handles events like URLS, EMAILS
*/
private ActionHandler formsActionHandler;
/**
* handles Javascript events
*/
private Javascript javascript;
/*flag to show if XFA or FDF*/
boolean hasXFA;
private boolean isContainXFAStream;
/**
* flag to show if we use XFA
*/
private boolean useXFA;
/**
* allow us to differentiate underlying PDF form type
*/
Enum PDFformType;
private SwingFormCreator formCreator;
/**
* used to create version without XFA support in XFA version.
* Should not be used otherwise.
*/
public AcroRenderer() {
}
public void useXFAIfAvailable(final boolean useXFA) {
this.useXFA = useXFA;
}
/**
* reset handler (must be called Before page opened)
* - null Object resets to default
*/
public void resetHandler(final ActionHandler formsActionHandler, final float dpi, final Javascript javascript) {
this.formsActionHandler = formsActionHandler;
this.dpi = dpi;
this.javascript = javascript;
//pass values down
if (formFactory != null) {
formFactory.reset(this.getFormResources(), formsActionHandler, pageData, currentPdfFile);
}
}
/**
* make all components invisible on all pages by removing from Display
*/
public void removeDisplayComponentsFromScreen() {
if (compData != null) {
compData.removeAllComponentsFromScreen();
}
}
/**
* initialise holders and variables, data structures and get a handle on data object
*/
public int openFile(final int pageCount, final int insetW, final int insetH, final PdfPageData pageData, final PdfObjectReader currentPdfFile, final PdfObject acroObj) {
this.pageCount = pageCount;
this.currentPdfFile = currentPdfFile;
this.pageData = pageData;
compData.flushFormData();
//explicitly flush
sigObject = null;
sigKeys = null;
//track inset on page
compData.setPageData(pageData, insetW, insetH);
if (acroObj == null) {
FfieldCount = 0;
fieldList = null;
} else {
//handle XFA
final PdfObject XFAasStream;
PdfArrayIterator XFAasArray = null;
XFAasStream = acroObj.getDictionary(PdfDictionary.XFA);
if (XFAasStream == null) {
XFAasArray = acroObj.getMixedArray(PdfDictionary.XFA);
//empty array
if (XFAasArray != null && XFAasArray.getTokenCount() == 0) {
XFAasArray = null;
}
}
hasXFA = XFAasStream != null || XFAasArray != null;
isContainXFAStream = hasXFA;
/*
* now read the fields
*/
fieldList = acroObj.getMixedArray(PdfDictionary.Fields);
CO = acroObj.getObjectArray(PdfDictionary.CO);
if (fieldList != null) {
FfieldCount = fieldList.getTokenCount();
AcroRes = acroObj.getDictionary(PdfDictionary.DR);
if (AcroRes != null) {
currentPdfFile.checkResolved(AcroRes);
}
} else {
FfieldCount = 0;
AcroRes = null;
}
/*
* choose correct decoder for form data
*/
if (hasXFA && useXFA) {
processXFAFields(acroObj, currentPdfFile, pageData);
}
//we need to read list if FDF
//or redo list if Legacy XFA
if (!hasXFA) {
resolveIndirectFieldList(false);
}
}
resetContainers(true);
return pageCount;
}
/**
* empty implementation in non-XFA AcroRenderer
*
* @param acroObj1
* @param currentPdfFile1
* @param pageData1
*/
void processXFAFields(final PdfObject acroObj1, final PdfObjectReader currentPdfFile1, final PdfPageData pageData1) {
throw new RuntimeException("This code (processXFAFields) should never be called");
}
void resolveIndirectFieldList(final boolean resolveParents) {
//allow for indirect
while (FfieldCount == 1) {
//may have been read before so reset
fieldList.resetToStart();
final String key = fieldList.getNextValueAsString(false);
final FormObject kidObject = new FormObject(key);
currentPdfFile.readObject(kidObject);
final byte[][] childList = getKid(kidObject, resolveParents);
if (childList == null) {
break;
}
fieldList = new PdfArrayIterator(childList);
FfieldCount = fieldList.getTokenCount();
}
}
/**
* initialise holders and variables and get a handle on data object
*
* Complicated as Annotations stored on a PAGE basis whereas FORMS stored on
* a file basis
*/
public void resetAnnotData(final int insetW, final int insetH, final PdfPageData pageData, final int page,
final PdfObjectReader currentPdfFile, final byte[][] currentAnnotList) {
this.currentPdfFile = currentPdfFile;
this.pageData = pageData;
boolean resetToEmpty = true;
addedMissingPopup = false;
//track inset on page
compData.setPageData(pageData, insetW, insetH);
if (currentAnnotList == null) {
AfieldCount = null;
ATotalCount = 0;
if (annotList != null) {
annotList[page] = null;
}
annotList = null;
} else {
int pageCount = pageData.getPageCount() + 1;
if (pageCount <= page) {
pageCount = page + 1;
}
if (annotList == null) {
annotList = new PdfArrayIterator[pageCount];
AfieldCount = new int[pageCount];
} else if (page >= annotList.length) {
final PdfArrayIterator[] tempList = annotList;
final int[] tempCount = AfieldCount;
AfieldCount = new int[pageCount];
annotList = new PdfArrayIterator[pageCount];
for (int ii = 0; ii < tempList.length; ii++) {
AfieldCount[ii] = tempCount[ii];
annotList[ii] = tempList[ii];
}
} else if (AfieldCount == null) {
AfieldCount = new int[pageCount];
}
annotList[page] = new PdfArrayIterator(currentAnnotList);
final int size = annotList[page].getTokenCount();
AfieldCount[page] = size;
ATotalCount += size;
resetToEmpty = false;
/*
* choose correct decoder for form data
*/
if (fDecoder == null) {
PDFformType = FormTypes.NON_XFA;
fDecoder = new FormStream();
}
}
resetContainers(resetToEmpty);
}
/**
* flush or resize data containers
*/
protected void resetContainers(final boolean resetToEmpty) {
/*form or reset Annots*/
if (resetToEmpty) {
compData.resetComponents(ATotalCount + FfieldCount, pageCount, false);
} else {
compData.resetComponents(ATotalCount + FfieldCount, pageCount, true);
}
if (formFactory == null) {
formFactory = formCreator.createFormFactory();
formFactory.reset(this.getFormResources(), formsActionHandler, pageData, currentPdfFile);
} else {
//to keep customers formfactory usable
formFactory.reset(this.getFormResources(), formsActionHandler, pageData, currentPdfFile);
}
}
/**
* build forms display using standard swing components
*/
public void createDisplayComponentsForPage(final int page, final PdfStreamDecoder current) {
//check if we want to flatten forms
if (flattenForms) {
compData.setRasterizeForms(true);
}
/*see if already done*/
if (!ignoreAllForms && (!compData.hasformsOnPageDecoded(page) || (formsRasterizedForDisplay() && current != null))) {
/* ensure space for all values*/
compData.initParametersForPage(pageData, page, formFactory, dpi);
/*
* think this needs to be revised, and different approach maybe storing, and reuse if respecified in file,
* need to look at other files to work out solution.
* files :-
* lettreenvoi.pdf page 2+ no next page field
* costena.pdf checkboxes not changable
*
* maybe if its just reset on multipage files
*/
//list of forms done
final Map formsProcessed = new HashMap();
int Acount = 0;
if (AfieldCount != null && AfieldCount.length > page) {
Acount = AfieldCount[page];
}
Fforms = new FormObject[FfieldCount];
Aforms = new FormObject[Acount];
FormObject formObject;
String objRef;
int i, count;
{
//scan list for all relevant values and add to array if valid
//0 = forms, 1 = annots
final int decodeToForm = 2;
for (int forms = 0; forms < decodeToForm; forms++) {
i = 0;
if (forms == 0) {
count = 0;
if (fieldList != null) {
fieldList.resetToStart();
count = fieldList.getTokenCount() - 1;
}
} else {
if (annotList != null && annotList.length > page && annotList[page] != null) {
if (!isContainXFAStream && formFactory.getType() == FormFactory.HTML) {
annotList[page].resetToStart();
final Map annotOrder = new HashMap();
final int count2 = annotList[page].getTokenCount();
String val;
for (int ii = 0; ii < count2; ii++) {
val = annotList[page].getNextValueAsString(true);
annotOrder.put(val, String.valueOf(ii + 1));
}
formFactory.setAnnotOrder(annotOrder);
}
//We need to do this regardless
annotList[page].resetToStart();
}
count = Acount - 1;
}
for (int fieldNum = count; fieldNum > -1; fieldNum--) {
objRef = null;
if (forms == 0) {
if (fieldList != null) {
objRef = fieldList.getNextValueAsString(true);
}
} else {
if (addedMissingPopup && !annotList[page].hasMoreTokens()) {
//Ignore this as we have added our own object that does not need reading
//with the PdfArrayIterator. Code positioned like this to explain.
//Test File : baseline_screens\cid2\SampleError.pdf
} else {
if (annotList.length > page && annotList[page] != null) {
objRef = annotList[page].getNextValueAsString(true);
}
}
}
if (objRef == null || ((formsProcessed.get(objRef) != null || objRef.isEmpty()))) {
continue;
}
try {
formObject = convertRefToFormObject(objRef, page);
if (!isAnnotation(formObject)) {
continue;
}
} catch (final Exception e) {
LogWriter.writeLog("Exception " + e + " with " + objRef);
continue;
}
/*
* Only allows Annotations if in Annots page stream
*/
if (forms == 0 && formObject != null && formObject.getFormType() == -1) {
continue;
}
if (allowsPopup(formObject) && formObject.getDictionary(PdfDictionary.Popup) == null) {
final FormObject po = new FormObject(PdfDictionary.Annot);
additionalOBjectRef++;
po.setRef((-additionalOBjectRef) + " 0 X");
po.setIntNumber(PdfDictionary.F, 24); //Bit Flag for bits 4+5 (No Zoom, No Rotate)
po.setBoolean(PdfDictionary.Open, formObject.getBoolean(PdfDictionary.Open));
po.setConstant(PdfDictionary.Subtype, PdfDictionary.Popup);
final float[] rect = formObject.getFloatArray(PdfDictionary.Rect);
if (pageData.getRotation(page) % 180 != 0) {
po.setFloatArray(PdfDictionary.Rect, new float[]{rect[2], pageData.getCropBoxHeight(page) - 100, rect[2] + 160, pageData.getCropBoxHeight(page)});
} else {
po.setFloatArray(PdfDictionary.Rect, new float[]{pageData.getCropBoxWidth(page), rect[3] - 100, pageData.getCropBoxWidth(page) + 160, rect[3]});
}
po.setStringKey(PdfDictionary.Parent, formObject.getObjectRefAsString().getBytes());
po.setParentPdfObj(formObject);
po.setPageNumber(page);
formObject.setDictionary(PdfDictionary.Popup, po);
final FormObject[] newForms = new FormObject[Aforms.length + 1];
for (int ii = 0; ii != Aforms.length; ii++) {
newForms[ii] = Aforms[ii];
}
newForms[Aforms.length] = po;
Aforms = newForms;
addedMissingPopup = true;
}
final byte[][] kids = formObject.getKeyArray(PdfDictionary.Kids);
if (kids != null) //not 'proper' kids so process here
{
i = flattenKids(page, formsProcessed, formObject, i, forms);
} else {
i = processFormObject(page, formsProcessed, formObject, objRef, i, forms);
}
}
}
}
final List unsortedForms = new ArrayList();
final List sortedForms = new ArrayList();
compData.setListForPage(page, unsortedForms, false);
compData.setListForPage(page, sortedForms, true);
final Map formsCreated = createAndStoreFormComponents(current, unsortedForms, sortedForms, page);
if (!formsRasterizedForDisplay()) {
// Go through all forms created and run through javascript for initiation
try {
//get current page object
final String ref = currentPdfFile.getReferenceforPage(page);
final PageObject pageObj = new PageObject(ref);
currentPdfFile.readObject(pageObj);
//call Page Level open commands
if (javascript != null && formsActionHandler != null) {
//NOTE: moved here as the Runnable can be executed after forms have been created which is too late.
formsActionHandler.O(pageObj, PdfDictionary.AA);
formsActionHandler.O(pageObj, PdfDictionary.A); //just in case
formsActionHandler.PO(pageObj, PdfDictionary.AA);
formsActionHandler.PO(pageObj, PdfDictionary.A); //just in case
}
if (formFactory.getType() != FormFactory.HTML && formFactory.getType() != FormFactory.SVG) {
//then initilise each field
initJSonFields(formsCreated);
}
} catch (final Exception e) {
LogWriter.writeLog("Exception: " + e.getMessage());
}
}
}
}
private Map createAndStoreFormComponents(final PdfStreamDecoder current, final List unsortedForms, final List sortedForms, final int page) {
final Map formsCreated = new HashMap();
FormObject[] xfaFormList = null;
FormObject formObject;
int count;
if (hasXFA && useXFA) {
xfaFormList = createXFADisplayComponentsForPage(xfaFormList, page);
}
//XFA, FDF FORMS then ANNOTS
final int readToForm = 3;
for (int forms = 0; forms < readToForm; forms++) {
count = 0;
if (forms == 0) {
if (xfaFormList != null) {
count = xfaFormList.length;
}
} else if (forms == 1) {
if (Fforms == null) {
count = 0;
} else {
//store current order of forms for printing
for (final FormObject Fform : Fforms) {
if (Fform != null) {
unsortedForms.add(Fform);
}
}
//sort forms into size order for display
Fforms = FormUtils.sortGroupLargestFirst(Fforms);
count = Fforms.length;
}
} else {
//store current order of forms for printing
for (final FormObject Aform : Aforms) {
if (Aform != null) {
unsortedForms.add(Aform);
}
}
//sort forms into size order for display
if (!formsRasterizedForDisplay()) {
Aforms = FormUtils.sortGroupLargestFirst(Aforms);
}
if (isContainXFAStream) {
final HashMap> tabMap = new HashMap>();
double maxY = 0;
for (final FormObject obj : Aforms) {
if (obj != null) {
final int x = obj.getBoundingRectangle().x;
final Double y = obj.getBounding2DRectangleForTabbing().getY();
maxY = Math.max(y, maxY);
if (tabMap.containsKey(y)) {
final ArrayList fList = tabMap.get(y);
int insertion = -1;
for (int z = 0; z < fList.size(); z++) {
final int nextX = fList.get(z).getBoundingRectangle().x;
if (nextX < x) {
insertion = z;
}
}
if (insertion == -1) {
fList.add(0, obj);
} else {
fList.add(insertion + 1, obj);
}
} else {
final ArrayList list = new ArrayList();
list.add(obj);
tabMap.put(y, list);
}
}
}
final FormObject[] finalList = new FormObject[Aforms.length];
int objCount = 0;
final Object[] keys = new Object[tabMap.size()];
int cc = 0;
for (final Object k : tabMap.keySet().toArray()) {
keys[cc] = k;
cc++;
}
Arrays.sort(keys);
for (int k = keys.length; k > 0; k--) {
final ArrayList objList = tabMap.get(keys[k - 1]);
if (objList != null) {
for (final FormObject f : objList) {
finalList[objCount] = f;
objCount++;
}
}
}
Aforms = finalList;
}
count = Aforms.length;
}
boolean firstPopup = true;
for (int k = 0; k < count; k++) {
if (forms == 0) {
formObject = xfaFormList[k];
} else if (forms == 1) {
formObject = Fforms[k];
} else {
formObject = Aforms[k];
}
if (formObject != null && (formsCreated.get(formObject.getObjectRefAsString()) == null) && page == formObject.getPageNumber()) { // && !formObject.getObjectRefAsString().equals("216 0 R")){
final int type = formFactory.getType();
//NOTE: if this custom form needs redrawing more changediff ReadOnlyTextIcon.MAXSCALEFACTOR to 1;
if ((formsRasterizedForDisplay() && current != null)
|| (type == FormFactory.SVG && formObject.getParameterConstant(PdfDictionary.Subtype) != PdfDictionary.Link)
|| (formFactory.flattenForms() && !AcroRenderer.isAnnotation(formObject))) {
//rasterize any flattened PDF forms here
try {
getFormFlattener().drawFlattenedForm(current, formObject, type == FormFactory.HTML || type == FormFactory.SVG, (PdfObject) this.getFormResources()[0]);
} catch (final PdfException e) {
LogWriter.writeLog("Exception: " + e.getMessage());
}
} else {
createField(formObject); //now we turn the data into a Swing component
//set the raw data here so that the field names are the fully qualified names
compData.storeRawData(formObject); //store data so user can access
formsCreated.put(formObject.getObjectRefAsString(), "x");
//original method we still use for HTML/SVG
if (type == FormFactory.HTML || type == FormFactory.SVG || type == FormFactory.JAVAFX) {
sortedForms.add(formObject);
//neede in display to fix position issues
} else if (formObject.getParameterConstant(PdfDictionary.Subtype) == PdfDictionary.Popup) {
/*
* To match examples the first popup is drawn over all
* then we draw all other popups in the pdf order from
* the bottom to the one before the top
*/
if (firstPopup) {
sortedForms.add(formObject);
firstPopup = false;
} else {
sortedForms.add(sortedForms.size() - 1, formObject);
}
} else {
sortedForms.add(0, formObject);
}
}
}
}
}
return formsCreated;
}
/**
* Utility method to check if formObject should have a popup
*
* @return True if popup should exist
*/
static boolean allowsPopup(final FormObject formObject) {
switch (formObject.getParameterConstant(PdfDictionary.Subtype)) {
case PdfDictionary.Text:
// case PdfDictionary.Line :
case PdfDictionary.Square:
// case PdfDictionary.Circle :
// case PdfDictionary.Polygon :
// case PdfDictionary.PolyLine :
case PdfDictionary.Highlight:
case PdfDictionary.Underline:
// case PdfDictionary.Squiggly :
case PdfDictionary.StrickOut:
case PdfDictionary.Stamp:
// case PdfDictionary.Caret :
// case PdfDictionary.Ink :
// case PdfDictionary.FileAttachment :
// case PdfDictionary.Sound :
// case PdfDictionary.Movie :
// case PdfDictionary.Widget :
// case PdfDictionary.Screen :
// case PdfDictionary.RichMedia:
// case PdfDictionary.PrinterMark :
// case PdfDictionary.TrapNet :
// case PdfDictionary.Watermark :
// case PdfDictionary.3D :
return true;
default:
return false;
}
}
/**
* Utility method to ensure formObject is actually an annotation before we continue
*
* @return True if annotation
*/
public static boolean isAnnotation(final FormObject formObject) {
if (formObject.getParameterConstant(PdfDictionary.Type) == PdfDictionary.Annot) {
return true;
}
switch (formObject.getParameterConstant(PdfDictionary.Subtype)) {
case PdfDictionary.Text:
case PdfDictionary.Link:
case PdfDictionary.FreeText:
case PdfDictionary.Line:
case PdfDictionary.Square:
case PdfDictionary.Circle:
case PdfDictionary.Polygon:
case PdfDictionary.PolyLine:
case PdfDictionary.Highlight:
case PdfDictionary.Underline:
case PdfDictionary.Squiggly:
case PdfDictionary.StrickOut:
case PdfDictionary.Stamp:
case PdfDictionary.Caret:
case PdfDictionary.Ink:
case PdfDictionary.Popup:
case PdfDictionary.FileAttachment:
case PdfDictionary.Sound:
case PdfDictionary.Movie:
case PdfDictionary.Widget:
case PdfDictionary.Screen:
case PdfDictionary.PrinterMark:
case PdfDictionary.TrapNet:
case PdfDictionary.Watermark:
case PdfDictionary.THREE_D:
// From the Supplement to the PDF spec
case PdfDictionary.RichMedia:
case PdfDictionary.Projection:
return true;
default:
return false;
}
}
private void storeSignatures(final FormObject formObject, final int formType) {
//if sig object set global sig object so we can access later
if (formType == PdfDictionary.Sig) {
if (sigObject == null) { //ensure initialised
sigObject = new HashSet();
sigKeys = new HashMap();
}
if (!sigKeys.containsKey(formObject.getObjectRefAsString())) { //avoid duplicates
sigObject.add(formObject);
sigKeys.put(formObject.getObjectRefAsString(), "x");
}
}
}
private int flattenKids(final int page, final Map formsProcessed, final FormObject formObject, int i, final int forms) {
final byte[][] kidList = formObject.getKeyArray(PdfDictionary.Kids);
final int kidCount = kidList.length;
//resize to fit
if (forms == 0) {
final int oldCount = Fforms.length;
final FormObject[] temp = Fforms;
Fforms = new FormObject[oldCount + kidCount - 1];
System.arraycopy(temp, 0, Fforms, 0, oldCount);
} else {
final int oldCount = Aforms.length;
final FormObject[] temp = Aforms;
Aforms = new FormObject[oldCount + kidCount - 1];
System.arraycopy(temp, 0, Aforms, 0, oldCount);
}
for (final byte[] aKidList : kidList) { //iterate through all parts
final String key = new String(aKidList);
//now we have inherited values, read
final FormObject childObj = new FormObject(key);
//inherit values
childObj.copyInheritedValuesFromParent(formObject);
currentPdfFile.readObject(childObj);
childObj.setRef(key);
if (childObj.getKeyArray(PdfDictionary.Kids) == null) {
if (!childObj.isAppearanceUsed()) {
new FormStream().createAppearanceString(childObj, currentPdfFile);
}
i = processFormObject(page, formsProcessed, childObj, key, i, forms);
} else {
i = flattenKids(page, formsProcessed, childObj, i, forms);
}
}
return i;
}
private int processFormObject(final int page, final Map formsProcessed, final FormObject formObject, final String objRef, int i, final int forms) {
boolean isOnPage = false;
if (forms == 0) { //check page
isOnPage = isFormObjectOnPage(formObject, page);
}
if (forms == 1 || isOnPage) {
formObject.setPageNumber(page);
final String parent = formObject.getStringKey(PdfDictionary.Parent);
//clone parent to allow inheritance of values or create new
if (parent != null) {
final FormObject parentObj = getParent(parent, formObject);
/*
* Getting the parent will also make any associated popup objects.
* If this is a popup the parent will instead make a copy so that
* we have a copy in annot list and one in form object. This is causing
* issue in annotation saving and deletion.
*/
//Overwrite created popup here with the one we already have.
if (formObject.getParameterConstant(PdfDictionary.Subtype) == PdfDictionary.Popup) {
parentObj.setDictionary(PdfDictionary.Popup, formObject);
}
//inherited values
if (parentObj != null) {
//all values are copied from the parent inside this call
formObject.setParent(parent, parentObj, true);
}
}
if (!formObject.isAppearanceUsed()) {
fDecoder.createAppearanceString(formObject, currentPdfFile);
}
//Check that type returns as a valid value to lock out broken objects
//Added for case 22215
if (formObject.getParameterConstant(PdfDictionary.Subtype) != -1) {
if (parent != null) {
formObject.setParent(parent); //parent object was added earlier
}
if (forms == 0) {
Fforms[i++] = formObject;
} else {
Aforms[i++] = formObject;
}
//also flag
if (objRef != null) {
formsProcessed.put(objRef, "x");
}
}
}
return i;
}
private boolean isFormObjectOnPage(final FormObject formObject, final int page) {
PdfObject pageObj = formObject.getDictionary(PdfDictionary.P);
byte[] pageRef = null;
if (pageObj != null) {
pageRef = pageObj.getUnresolvedData();
}
if (pageRef == null || pageObj == null) {
final String parent = formObject.getStringKey(PdfDictionary.Parent);
if (parent != null) {
final PdfObject parentObj = getParent(parent, null);
pageObj = parentObj.getDictionary(PdfDictionary.P);
if (pageObj != null) {
pageRef = pageObj.getUnresolvedData();
}
}
}
if (pageRef == null) {
final byte[][] kidList = getKid(formObject, false);
if (kidList != null && kidList.length > 0) {
final int kidCount = kidList.length;
FormObject kidObject;
for (int jj = 0; jj < kidCount; jj++) {
final String key = new String(kidList[jj]);
kidObject = compData.getRawFormData().get(key);
if (kidObject == null) {
kidObject = new FormObject(key);
currentPdfFile.readObject(kidObject);
compData.storeRawData(kidObject);
}
pageObj = kidObject.getDictionary(PdfDictionary.P);
if (pageObj != null) {
pageRef = pageObj.getUnresolvedData();
}
if (pageRef != null) {
jj = kidCount;
}
}
}
}
int objPage = -1;
if (pageRef != null) {
objPage = currentPdfFile.convertObjectToPageNumber(new String(pageRef));
}
return objPage == page;
}
private FormObject getParent(final String parent, final FormObject formObj) {
FormObject parentObj = compData.getRawFormData().get(parent);
if (parentObj == null && parent != null) { //not yet read so read and cache
parentObj = new FormObject(parent);
currentPdfFile.readObject(parentObj);
//remove kids in Parent
parentObj.setKeyArray(PdfDictionary.Kids, null);
compData.storeRawData(parentObj);
}
if (parentObj != null && formObj != null) {
final byte[][] kidsInParent = parentObj.getRawKids();
formObj.setRawKids(kidsInParent);
}
return parentObj;
}
private byte[][] getKid(final FormObject formObject, final boolean ignoreParent) {
final int formType = formObject.getNameAsConstant(PdfDictionary.FT);
if (formType == PdfDictionary.Tx || formType == PdfDictionary.Btn) {
return null;
}
byte[][] kidList = formObject.getKeyArray(PdfDictionary.Kids);
if (kidList != null && !ignoreParent) {
final String parentRef = formObject.getStringKey(PdfDictionary.Parent);
final PdfObject parentObj = this.getFormObject(parentRef);
if (parentObj != null && parentObj.getKeyArray(PdfDictionary.Kids) != null) {
kidList = null;
}
}
return kidList;
}
/**
* display widgets on screen for range (inclusive)
*/
public void displayComponentsOnscreen(final int startPage, int endPage) {
//On rare occurances compData can be null as dispose called during paint.
//As dispose only called on viewer close we can lock issue out here
if (compData != null) {
//make sure this page is inclusive in loop
endPage++;
compData.displayComponents(startPage, endPage);
}
}
private void initJSonFields(final Map formsCreated) {
//scan all fields for javascript actions
for (final String ref : formsCreated.keySet()) {
final FormObject formObject = getFormObject(ref);
javascript.execute(formObject, PdfDictionary.K, ActionHandler.FOCUS_EVENT, ' ');
}
}
/**
* create a widget to handle fields
*/
private void createField(final FormObject formObject) {
//avoid creation of Swing widgets if we have 2 copies in play on ULC
//noinspection PointlessBooleanExpression
if (ExternalHandlers.isULCPresent() &&
formFactory.getType() == FormFactory.SWING) {
return;
}
final Integer widgetType; //no value set
final Object retComponent = null;
final int formType = formObject.getNameAsConstant(PdfDictionary.FT); //FT
final int formFactoryType = formFactory.getType();
//if sig object set global sig object so we can access later
storeSignatures(formObject, formType);
//check if a popup is associated
if (formObject.getDictionary(PdfDictionary.Popup) != null) {
formObject.setActionFlag(FormObject.POPUP);
}
//flags used to alter interactivity of all fields
final boolean readOnly;
final boolean required;
final boolean noexport;
final boolean[] flags = formObject.getFieldFlags(); //Ff
if (flags != null) {
//noinspection UnusedAssignment
readOnly = flags[FormObject.READONLY_ID];
//noinspection UnusedAssignment
required = flags[FormObject.REQUIRED_ID];
//noinspection UnusedAssignment
noexport = flags[FormObject.NOEXPORT_ID];
/*
* boolean comb=flags[FormObject.COMB_ID];
* boolean comminOnSelChange=flags[FormObject.COMMITONSELCHANGE_ID];
* boolean donotScrole=flags[FormObject.DONOTSCROLL_ID];
* boolean doNotSpellCheck=flags[FormObject.DONOTSPELLCHECK_ID];
* boolean fileSelect=flags[FormObject.FILESELECT_ID];
* boolean isCombo=flags[FormObject.COMBO_ID];
* boolean isEditable=flags[FormObject.EDIT_ID];
* boolean isMultiline=flags[FormObject.MULTILINE_ID];
* boolean isPushButton=flags[FormObject.PUSHBUTTON_ID];
* boolean isRadio=flags[FormObject.RADIO_ID];
* boolean hasNoToggleToOff=flags[FormObject.NOTOGGLETOOFF_ID];
* boolean hasPassword=flags[FormObject.PASSWORD_ID];
* boolean multiSelect=flags[FormObject.MULTISELECT_ID];
* boolean radioinUnison=flags[FormObject.RADIOINUNISON_ID];
* boolean richtext=flags[FormObject.RICHTEXT_ID];
* boolean sort=flags[FormObject.SORT_ID];
*/
}
//hard-coded for HTML non-forms
if (!ExternalHandlers.isXFAPresent() && (formFactoryType == FormFactory.HTML || formFactoryType == FormFactory.SVG)) {
widgetType = FormFactory.ANNOTATION;
} else if (formType == PdfDictionary.Btn) { //----------------------------------- BUTTON ----------------------------------------
widgetType = createButtonField(flags);
} else {
widgetType = createInteractiveField(flags, formType);
}
formObject.setFormType(widgetType);
if (formFactory.getType() == FormFactory.HTML || formFactory.getType() == FormFactory.SVG) {
compData.checkGUIObjectResolved(formObject);
} else if (retComponent != null && formFactory.getType() != FormFactory.SWING) {
formObject.setGUIComponent(retComponent, formFactory.getType());
compData.setGUIComp(formObject, retComponent);
}
}
private int createInteractiveField(final boolean[] flags, final int formType) {
final int widgetType;
switch (formType) {
case PdfDictionary.Tx:
boolean isMultiline = false,
hasPassword = false; // doNotScroll = false, richtext = false, fileSelect = false, doNotSpellCheck = false;
if (flags != null) {
isMultiline = flags[FormObject.MULTILINE_ID];
hasPassword = flags[FormObject.PASSWORD_ID];
//doNotScroll = flags[FormObject.DONOTSCROLL_ID];
//richtext = flags[FormObject.RICHTEXT_ID];
//fileSelect = flags[FormObject.FILESELECT_ID];
//doNotSpellCheck = flags[FormObject.DONOTSPELLCHECK_ID];
}
if (isMultiline) {
if (hasPassword) {
widgetType = FormFactory.MULTILINEPASSWORD;
} else {
widgetType = FormFactory.MULTILINETEXT;
}
} else { //singleLine
if (hasPassword) {
widgetType = FormFactory.SINGLELINEPASSWORD;
} else {
widgetType = FormFactory.SINGLELINETEXT;
}
}
break;
case PdfDictionary.Ch:
//----------------------------------------- CHOICE ----------------------------------------------
//flags used for choice types
//20100212 (ms) Unused ones commented out
boolean isCombo = false; // multiSelect = false, sort = false, isEditable = false, doNotSpellCheck = false, comminOnSelChange = false;
if (flags != null) {
isCombo = flags[FormObject.COMBO_ID];
//multiSelect = flags[FormObject.MULTISELECT_ID];
//sort = flags[FormObject.SORT_ID];
//isEditable = flags[FormObject.EDIT_ID];
//doNotSpellCheck = flags[FormObject.DONOTSPELLCHECK_ID];
//comminOnSelChange = flags[FormObject.COMMITONSELCHANGE_ID];
}
if (isCombo) { // || (type==XFAFORM && ((XFAFormObject)formObject).choiceShown!=XFAFormObject.CHOICE_ALWAYS)){
widgetType = FormFactory.COMBOBOX;
} else { //it is a list
widgetType = FormFactory.LIST;
}
break;
case PdfDictionary.Sig:
widgetType = FormFactory.SIGNATURE;
break;
default:
//assume annotation if (formType == ANNOTATION) {
widgetType = FormFactory.ANNOTATION;
break;
}
return widgetType;
}
private int createButtonField(final boolean[] flags) {
final int widgetType;
//flags used for button types
//20100212 (ms) Unused ones commented out
boolean isPushButton = false, isRadio = false; // hasNoToggleToOff = false, radioinUnison = false;
if (flags != null) {
isPushButton = flags[FormObject.PUSHBUTTON_ID];
isRadio = flags[FormObject.RADIO_ID];
//hasNoToggleToOff = flags[FormObject.NOTOGGLETOOFF_ID];
//radioinUnison = flags[FormObject.RADIOINUNISON_ID];
}
if (isPushButton) {
widgetType = FormFactory.PUSHBUTTON;
} else if (isRadio) {
widgetType = FormFactory.RADIOBUTTON;
} else {
widgetType = FormFactory.CHECKBOXBUTTON;
}
return widgetType;
}
/**
* If possible we recommend you work with the Form Objects rather than
* resolve GUI components
*
* null key will return all values
*
* if pageNumber is -1 it will process whole document, otherwise just that
* page
*
* For a full example of usage please see http://files.idrsolutions.com/samplecode/org/jpedal/examples/acroform/ExtractFormDataAsObject.java.html
*
* Object[] will vary depending on what ReturnValues enum is passed in and could contain String (Names), FormObject or Component
*/
public Object[] getFormComponents(final String objectName, final ReturnValues value, final int pageNumber) {
switch (value) {
case EMBEDDED_FILES:
if (currentPdfFile.getNamesLookup() == null) {
return new Object[]{};
}
return currentPdfFile.getNamesLookup().getEmbeddedFiles();
default:
/*make sure all forms decoded*/
if (pageNumber == -1) {
for (int p = 1; p < this.pageCount + 1; p++) //add init method and move scaling/rotation to it
{
createDisplayComponentsForPage(p, null);
}
} else {
createDisplayComponentsForPage(pageNumber, null);
}
return compData.getFormComponents(objectName, value, pageNumber).toArray();
}
}
/**
* setup object which creates all GUI objects
*/
public void setFormFactory(final FormFactory newFormFactory) {
formFactory = newFormFactory;
/*
* allow user to create custom structure to hold data
*/
compData = formFactory.getCustomCompData();
}
/**
* get GUIData object with all widgets
*/
public GUIData getCompData() {
return compData;
}
/**
* return Signature as iterator with one or more objects or null
*/
public Iterator getSignatureObjects() {
if (sigObject == null) {
return null;
} else {
return sigObject.iterator();
}
}
public ActionHandler getActionHandler() {
return formsActionHandler;
}
public FormFactory getFormFactory() {
return formFactory;
}
public void setIgnoreForms(final boolean ignoreForms) {
this.ignoreForms = ignoreForms;
}
public boolean ignoreForms() {
return ignoreForms || ignoreAllForms;
}
public void dispose() {
AfieldCount = null;
fDecoder = null;
formsActionHandler = null;
if (javascript != null) {
javascript.dispose();
}
javascript = null;
Fforms = null;
Aforms = null;
fieldList = null;
annotList = null;
formFactory = null;
if (compData != null) {
compData.dispose();
}
compData = null;
if (sigObject != null) {
sigObject.clear();
}
sigObject = null;
if (sigKeys != null) {
sigKeys.clear();
}
sigKeys = null;
pageData = null;
if (currentPdfFile != null) {
currentPdfFile.dispose();
}
currentPdfFile = null;
fDecoder = null;
formCreator = null;
}
/**
* get Iterator with list of all Annots on page or
* return null if no Annots - no longer needs
* call to decodePage beforehand as checks itself
*
* @deprecated - getFormComponents(String objectName, ReturnValues value,int pageNumber) recommended
* as much more flexible
*/
@Deprecated
public PdfArrayIterator getAnnotsOnPage(final int page) {
//check annots decoded - will just return if done
createDisplayComponentsForPage(page, null);
if (annotList != null && annotList.length > page && annotList[page] != null) {
annotList[page].resetToStart();
return annotList[page];
} else {
return null;
}
}
/**
* returns false if not XFA (or XFA in Legacy mode)
* and true if XFA using XFA data
*
* @return
*/
public boolean isXFA() {
return hasXFA;
}
public boolean useXFA() {
return useXFA;
}
public boolean hasFormsOnPage(final int page) {
final boolean hasAnnots = (annotList != null && annotList.length > page && annotList[page] != null);
final boolean hasForm = (hasXFA && useXFA && fDecoder.hasXFADataSet()) || fieldList != null;
return hasAnnots || hasForm;
}
public Object[] getFormResources() {
return new Object[]{AcroRes, CO};
}
public boolean formsRasterizedForDisplay() {
return compData.formsRasterizedForDisplay();
}
/**
* get FormObject
*
* @param ref
* @return In all modes except HTML,SVG,JavaFX will decode other forms on pages if not
* found
*/
public FormObject getFormObject(final String ref) {
FormObject obj = compData.getRawFormData().get(ref);
//if not found now decode all page and retry
if (obj == null && formFactory.getType() != FormFactory.HTML && formFactory.getType() != FormFactory.SVG) {
for (int ii = 1; ii < this.pageCount; ii++) {
createDisplayComponentsForPage(ii, null);
obj = compData.getRawFormData().get(ref);
if (obj != null) {
break;
}
}
}
return obj;
}
public void setInsets(final int width, final int height) {
compData.setPageData(compData.pageData, width, height);
}
FormObject convertRefToFormObject(final String objRef, final int page) {
FormObject formObject = compData.getRawFormData().get(objRef);
if (formObject == null) {
formObject = new FormObject(objRef);
if (page != -1) {
formObject.setPageRotation(pageData.getRotation(page));
}
//formObject.setPDFRef((String)objRef);
if (objRef.charAt(objRef.length() - 1) == 'R') {
currentPdfFile.readObject(formObject);
} else {
//changed by Mark as cover <<>> as well as 1 0 R
formObject.setStatus(PdfObject.UNDECODED_REF);
formObject.setUnresolvedData(StringUtils.toBytes(objRef), -1);
currentPdfFile.checkResolved(formObject);
}
compData.storeRawData(formObject);
}
return formObject;
}
/**
* Allow user to get ENUM to show type of form
* FormTypes (XFA_LEGACY, XFA_DYNAMIC, NON_XFA)
*
* @return
*/
public Enum getPDFformType() {
return PDFformType;
}
public void alwaysuseXFA(final boolean alwaysUseXFA) {
this.alwaysUseXFA = alwaysUseXFA;
}
public boolean alwaysuseXFA() {
return alwaysUseXFA;
}
public void init(final SwingFormCreator formCreator) {
this.formCreator = formCreator;
compData = formCreator.getData();
}
public PdfStreamDecoder getStreamDecoder(final PdfObjectReader currentPdfFile, final PdfLayerList layer, final boolean isFirst) {
if (isFirst) {
return new PdfStreamDecoder(currentPdfFile);
} else {
return new PdfStreamDecoder(currentPdfFile, layer);
}
}
public boolean showFormWarningMessage(final int page) {
boolean warnOnceOnForms = false;
if (hasXFA) {
warnOnceOnForms = true;
System.out.println("[WARNING] This file contains XFA forms that are not supported by this version of JPDF2HTML5. To convert into functional HTML forms and display non-legacy mode page content, JPDF2HTML5 Forms Edition must be used.");
} else if (hasFormsOnPage(page)) {
warnOnceOnForms = true;
System.out.println("[WARNING] This file contains form components that have been rasterized. To convert into functional HTML forms, JPDF2HTML5 Forms Edition must be used.");
}
return warnOnceOnForms;
}
FormObject[] createXFADisplayComponentsForPage(final FormObject[] xfaFormList, final int page) {
throw new UnsupportedOperationException("createXFADisplayComponentsForPage should never be called");
}
public HashMap getPageMapXFA() {
throw new UnsupportedOperationException("getPageMapXFA should never be called");
}
public byte[] getXMLContentAsBytes(final int dataType) {
return null;
}
public void outputJavascriptXFA(final String path, final String name) {
throw new UnsupportedOperationException("outputJavascriptXFA should never be called");
}
public PrintStreamDecoder getStreamDecoderForPrinting(final PdfObjectReader currentPdfFile, final PdfLayerList pdfLayerList) {
return new PdfStreamDecoderForPrinting(currentPdfFile, pdfLayerList);
}
public BufferedImage decode(final PdfObject pdfObject, final PdfObjectReader currentPdfFile, final PdfObject XObject, final int subtype, final int width, final int height, final int offsetImage, final float pageScaling) {
LogWriter.writeLog("called decode(" + pdfObject + ", " + currentPdfFile + ", " + XObject + ", " + subtype + ", " + width + ", " + height + ", " + offsetImage + ", " + pageScaling);
return null;
}
public FormFlattener getFormFlattener() {
return new FormFlattener();
}
}