com.itextpdf.kernel.pdf.layer.PdfLayer Maven / Gradle / Ivy
The newest version!
/*
This file is part of the iText (R) project.
Copyright (c) 1998-2024 Apryse Group NV
Authors: Apryse Software.
This program is offered under a commercial and under the AGPL license.
For commercial licensing, contact us at https://itextpdf.com/sales. For AGPL licensing, see below.
AGPL licensing:
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
package com.itextpdf.kernel.pdf.layer;
import com.itextpdf.commons.utils.MessageFormatUtil;
import com.itextpdf.io.font.PdfEncodings;
import com.itextpdf.kernel.exceptions.KernelExceptionMessageConstant;
import com.itextpdf.kernel.exceptions.PdfException;
import com.itextpdf.kernel.pdf.PdfArray;
import com.itextpdf.kernel.pdf.PdfDictionary;
import com.itextpdf.kernel.pdf.PdfDocument;
import com.itextpdf.kernel.pdf.PdfIndirectReference;
import com.itextpdf.kernel.pdf.PdfName;
import com.itextpdf.kernel.pdf.PdfNumber;
import com.itextpdf.kernel.pdf.PdfObject;
import com.itextpdf.kernel.pdf.PdfObjectWrapper;
import com.itextpdf.kernel.pdf.PdfString;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
/**
* An optional content group is a dictionary representing a collection of graphics
* that can be made visible or invisible dynamically by users of viewer applications.
* In iText they are referenced as layers.
*
*
* To be able to be wrapped with this {@link PdfObjectWrapper} the {@link PdfObject}
* must be indirect.
*/
public class PdfLayer extends PdfObjectWrapper implements IPdfOCG {
/**
* Used for titling group of objects but not actually grouping them.
*/
protected String title;
protected boolean on = true;
protected boolean onPanel = true;
protected boolean locked = false;
protected PdfLayer parent;
protected List children;
/**
* Creates a new layer by existing dictionary, which must be an indirect object.
*
* @param layerDictionary the layer dictionary, must have an indirect reference.
*/
public PdfLayer(PdfDictionary layerDictionary) {
super(layerDictionary);
setForbidRelease();
ensureObjectIsAddedToDocument(layerDictionary);
}
/**
* Creates a new layer by its name and document.
* @param name the layer name
* @param document the PdfDocument which the layer belongs to
*/
public PdfLayer(String name, PdfDocument document) {
this(document);
setName(name);
document.getCatalog().getOCProperties(true).registerLayer(this);
}
private PdfLayer(PdfDocument document) {
super(new PdfDictionary());
makeIndirect(document);
getPdfObject().put(PdfName.Type, PdfName.OCG);
}
/**
* Creates a title layer. A title layer is not really a layer but a collection of layers
* under the same title heading.
* @param title the title text
* @param document the PdfDocument
* @return the title layer
*/
public static PdfLayer createTitle(String title, PdfDocument document) {
PdfLayer layer = createTitleSilent(title, document);
document.getCatalog().getOCProperties(true).registerLayer(layer);
return layer;
}
/**
* Use this method to set a collection of optional content groups
* whose states are intended to follow a "radio button" paradigm.
* That is, the state of at most one optional content group
* in the array should be ON at a time: if one group is turned
* ON, all others must be turned OFF.
* @param document the PdfDocument
* @param group the radio group
*/
public static void addOCGRadioGroup(PdfDocument document, List group) {
document.getCatalog().getOCProperties(true).addOCGRadioGroup(group);
}
/**
* Adds a child layer. Nested layers can only have one parent.
* @param childLayer the child layer
*/
public void addChild(PdfLayer childLayer) {
//TODO DEVSIX-8490 implement multiple parent support
if (childLayer.parent != null) {
PdfIndirectReference ref = childLayer.getIndirectReference();
throw new PdfException(MessageFormatUtil.format(
KernelExceptionMessageConstant.UNABLE_TO_ADD_SECOND_PARENT_LAYER, ref.toString()));
}
childLayer.parent = this;
if (children == null) {
children = new ArrayList<>();
}
children.add(childLayer);
}
/**
* Gets the parent of this layer, be it a title layer, or a usual one.
* @return the parent of the layer, or null if it has no parent
*/
public PdfLayer getParent() {
return parent;
}
/**
* Sets the name of the layer to be displayed in the Layers panel.
* @param name the name of the layer.
*/
public void setName(String name) {
getPdfObject().put(PdfName.Name, new PdfString(name, PdfEncodings.UNICODE_BIG));
getPdfObject().setModified();
}
/**
* Gets the initial visibility of the layer when the document is opened.
* @return the initial visibility of the layer
*/
public boolean isOn() {
return on;
}
/**
* Sets the initial visibility of the layer when the document is opened.
* @param on the initial visibility of the layer
*/
public void setOn(boolean on) {
if (this.on != on)
fetchOCProperties().setModified();
this.on = on;
}
/**
* Gets whether the layer is currently locked or not. If the layer is locked,
* it will not be possible to change its state (on/off) in a viewer.
* @return true if the layer is currently locked, false otherwise.
*/
public boolean isLocked() {
return locked;
}
/**
* Use this method to lock an optional content group.
* The state of a locked group cannot be changed through the user interface
* of a viewer application. Producers can use this entry to prevent the visibility
* of content that depends on these groups from being changed by users.
*
* @param locked sets whether the layer is currently locked or not
*/
public void setLocked(boolean locked) {
if (this.isLocked() != locked)
fetchOCProperties().setModified();
this.locked = locked;
}
/**
* Gets the layer visibility in Acrobat's layer panel
* @return the layer visibility in Acrobat's layer panel
*/
public boolean isOnPanel() {
return onPanel;
}
/**
* Sets the visibility of the layer in Acrobat's layer panel. If false
* the layer cannot be directly manipulated by the user. Note that any children layers will
* also be absent from the panel.
* @param onPanel the visibility of the layer in Acrobat's layer panel
*/
public void setOnPanel(boolean onPanel) {
if (this.on != onPanel)
fetchOCProperties().setModified();
this.onPanel = onPanel;
}
/**
* Gets a collection of current intents specified for this layer.
* The default value is {@link PdfName#View}, so it will be the only element of the
* resultant collection if no intents are currently specified.
* @return the collection of intents.
*/
public Collection getIntents() {
final PdfObject intent = getPdfObject().get(PdfName.Intent);
if (intent instanceof PdfName)
return Collections.singletonList((PdfName) intent);
else if (intent instanceof PdfArray) {
PdfArray intentArr = (PdfArray) intent;
Collection intentsCollection = new ArrayList<>(intentArr.size());
for (PdfObject i : intentArr) {
if (i instanceof PdfName) {
intentsCollection.add((PdfName) i);
}
}
return intentsCollection;
}
return Collections.singletonList(PdfName.View);
}
/**
* Sets the intents of the layer.
* @param intents the list of intents.
*/
public void setIntents(List intents) {
if (intents == null || intents.size() == 0) {
getPdfObject().remove(PdfName.Intent);
} else if (intents.size() == 1) {
getPdfObject().put(PdfName.Intent, intents.get(0));
} else {
// intents.size() > 1
PdfArray array = new PdfArray();
for (PdfName intent : intents) {
array.add(intent);
}
getPdfObject().put(PdfName.Intent, array);
}
getPdfObject().setModified();
}
/**
* Used by the creating application to store application-specific
* data associated with this optional content group.
* @param creator a text string specifying the application that created the group
* @param subtype a string defining the type of content controlled by the group. Suggested
* values include but are not limited to Artwork, for graphic-design or publishing
* applications, and Technical, for technical designs such as building plans or
* schematics
*/
public void setCreatorInfo(String creator, String subtype) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.Creator, new PdfString(creator, PdfEncodings.UNICODE_BIG));
dic.put(PdfName.Subtype, new PdfName(subtype));
usage.put(PdfName.CreatorInfo, dic);
usage.setModified();
}
/**
* Specifies the language of the content controlled by this
* optional content group
* @param lang a language string which specifies a language and possibly a locale
* (for example, es-MX represents Mexican Spanish)
* @param preferred used by viewer applications when there is a partial match but no exact
* match between the system language and the language strings in all usage dictionaries
*/
public void setLanguage(String lang, boolean preferred) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.Lang, new PdfString(lang, PdfEncodings.UNICODE_BIG));
if (preferred)
dic.put(PdfName.Preferred, PdfName.ON);
usage.put(PdfName.Language, dic);
usage.setModified();
}
/**
* Specifies the recommended state for content in this
* group when the document (or part of it) is saved by a viewer application to a format
* that does not support optional content (for example, an earlier version of
* PDF or a raster image format).
* @param export the export state
*/
public void setExport(boolean export) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.ExportState, export ? PdfName.ON : PdfName.OFF);
usage.put(PdfName.Export, dic);
usage.setModified();
}
/**
* Specifies a range of magnifications at which the content
* in this optional content group is best viewed.
* @param min the minimum recommended magnification factors at which the group
* should be ON. A negative value will set the default to 0
* @param max the maximum recommended magnification factor at which the group
* should be ON. A negative value will set the largest possible magnification supported by the
* viewer application
*/
public void setZoom(float min, float max) {
if (min <= 0 && max < 0)
return;
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
if (min > 0)
dic.put(PdfName.min, new PdfNumber(min));
if (max >= 0)
dic.put(PdfName.max, new PdfNumber(max));
usage.put(PdfName.Zoom, dic);
usage.setModified();
}
/**
* Specifies that the content in this group is intended for
* use in printing
* @param subtype a name specifying the kind of content controlled by the group;
* for example, Trapping, PrintersMarks and Watermark
* @param printState indicates that the group should be
* set to that state when the document is printed from a viewer application
*/
public void setPrint(String subtype, boolean printState) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.Subtype, new PdfName(subtype));
dic.put(PdfName.PrintState, printState ? PdfName.ON : PdfName.OFF);
usage.put(PdfName.Print, dic);
usage.setModified();
}
/**
* Indicates that the group should be set to that state when the
* document is opened in a viewer application.
* @param view the view state
*/
public void setView(boolean view) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.ViewState, view ? PdfName.ON : PdfName.OFF);
usage.put(PdfName.View, dic);
usage.setModified();
}
/**
* Specifies one or more users for whom this optional content group
* is primarily intended.
* @param type a name that can be Ind (individual), Ttl (title), or Org (organization).
* @param names one or more text strings representing
* the name(s) of the individual, position or organization
*/
public void setUser(String type, String... names) {
if (type == null || !"Ind".equals(type) && !"Ttl".equals(type) && !"Org".equals(type))
throw new IllegalArgumentException("Illegal type argument");
if (names == null || names.length == 0)
throw new IllegalArgumentException("Illegal names argument");
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.Type, new PdfName(type));
if (names.length == 1) {
dic.put(PdfName.Name, new PdfString(names[0], PdfEncodings.UNICODE_BIG));
} else {
PdfArray namesArray = new PdfArray();
for (String name : names) {
namesArray.add(new PdfString(name, PdfEncodings.UNICODE_BIG));
}
dic.put(PdfName.Name, namesArray);
}
usage.put(PdfName.User, dic);
usage.setModified();
}
/**
* Indicates that the group contains a pagination artifact.
* @param pe one of the following names: "HF" (Header Footer),
* "FG" (Foreground), "BG" (Background), or "L" (Logo).
*/
public void setPageElement(String pe) {
PdfDictionary usage = getUsage();
PdfDictionary dic = new PdfDictionary();
dic.put(PdfName.Subtype, new PdfName(pe));
usage.put(PdfName.PageElement, dic);
usage.setModified();
}
/**
* Gets the indirect reference to the current layer object.
* @return the indirect reference to the object representing the layer
*/
public PdfIndirectReference getIndirectReference() {
return getPdfObject().getIndirectReference();
}
/**
* Gets the title of the layer if it is a title layer, or null if it is a usual layer.
*
* @return the title of the layer if it is a title layer, or null if it is a usual layer
*/
public String getTitle() {
return title;
}
/**
* Gets the list of the current child layers of the layer.
* BE CAREFUL! Do not try to add a child layer using the resultant child list,
* use #addChild method instead.
* @return the list of the current child layers, null if the layer has no children.
*/
public List getChildren() {
return children == null ? null : new ArrayList<>(children);
}
@Override
protected boolean isWrappedObjectMustBeIndirect() {
return true;
}
/**
* Gets the {@link PdfDocument} that owns that layer.
*
* @return the {@link PdfDocument} that owns that layer
*/
protected PdfDocument getDocument() {
return getPdfObject().getIndirectReference().getDocument();
}
/**
* Creates a title layer without registering it in PdfOCProperties.
* @param title the title of the layer
* @param document the document this title layer belongs to
* @return the created layer
*/
protected static PdfLayer createTitleSilent(String title, PdfDocument document) {
if (title == null)
throw new IllegalArgumentException("Invalid title argument");
PdfLayer layer = new PdfLayer(document);
layer.title = title;
return layer;
}
/**
* Gets the /Usage dictionary, creating a new one if necessary.
* @return the /Usage dictionary
*/
protected PdfDictionary getUsage() {
PdfDictionary usage = getPdfObject().getAsDictionary(PdfName.Usage);
if (usage == null) {
usage = new PdfDictionary();
getPdfObject().put(PdfName.Usage, usage);
}
return usage;
}
private PdfOCProperties fetchOCProperties() {
return getDocument().getCatalog().getOCProperties(true);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy