org.icepdf.ri.common.views.annotations.AbstractAnnotationComponent Maven / Gradle / Ivy
/*
* Copyright 2006-2016 ICEsoft Technologies Inc.
*
* Licensed 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.icepdf.ri.common.views.annotations;
import org.icepdf.core.pobjects.Document;
import org.icepdf.core.pobjects.Name;
import org.icepdf.core.pobjects.Page;
import org.icepdf.core.pobjects.acroform.AdditionalActionsDictionary;
import org.icepdf.core.pobjects.acroform.FieldDictionary;
import org.icepdf.core.pobjects.actions.Action;
import org.icepdf.core.pobjects.annotations.AbstractWidgetAnnotation;
import org.icepdf.core.pobjects.annotations.Annotation;
import org.icepdf.core.pobjects.annotations.Appearance;
import org.icepdf.core.util.ColorUtil;
import org.icepdf.core.util.Defs;
import org.icepdf.core.util.PropertyConstants;
import org.icepdf.ri.common.views.*;
import javax.swing.*;
import javax.swing.border.Border;
import javax.swing.event.MouseInputListener;
import java.awt.*;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.awt.event.MouseEvent;
import java.awt.geom.AffineTransform;
import java.awt.geom.GeneralPath;
import java.awt.geom.NoninvertibleTransformException;
import java.awt.geom.Rectangle2D;
import java.util.ResourceBundle;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* AbstractAnnotationComponent contains base functionality for annotation
* components which are used to display annotation for a given page view. This
* class controls icon state, focus and basic component states: editable,
* movable, resizable, selected and show invisible border.
*
* @since 5.0
*/
public abstract class AbstractAnnotationComponent extends JComponent implements FocusListener,
MouseInputListener, AnnotationComponent {
protected static final Logger logger =
Logger.getLogger(AbstractAnnotationComponent.class.toString());
protected static boolean isInteractiveAnnotationsEnabled;
protected static Color annotationHighlightColor;
protected static float annotationHighlightAlpha;
static {
// enables interactive annotation support.
isInteractiveAnnotationsEnabled =
Defs.sysPropertyBoolean(
"org.icepdf.core.annotations.interactive.enabled", true);
// sets annotation selected highlight colour
try {
String color = Defs.sysProperty(
"org.icepdf.core.views.page.annotation.highlight.color", "#000000");
int colorValue = ColorUtil.convertColor(color);
annotationHighlightColor =
new Color(colorValue >= 0 ? colorValue :
Integer.parseInt("000000", 16));
} catch (NumberFormatException e) {
if (logger.isLoggable(Level.WARNING)) {
logger.warning("Error reading page annotation highlight colour");
}
}
// set the annotation alpha value.
// sets annotation selected highlight colour
try {
String alpha = Defs.sysProperty(
"org.icepdf.core.views.page.annotation.highlight.alpha", "0.4");
annotationHighlightAlpha = Float.parseFloat(alpha);
} catch (NumberFormatException e) {
if (logger.isLoggable(Level.WARNING)) {
logger.warning("Error reading page annotation highlight alpha");
}
annotationHighlightAlpha = 0.4f;
}
}
public static final int resizeBoxSize = 4;
// reusable border
protected static ResizableBorder resizableBorder =
new ResizableBorder(resizeBoxSize);
protected AbstractPageViewComponent pageViewComponent;
protected DocumentViewController documentViewController;
protected DocumentViewModel documentViewModel;
protected float currentZoom;
protected float currentRotation;
protected Annotation annotation;
protected boolean isMousePressed;
protected boolean resized;
protected boolean wasResized;
// border state flags.
protected boolean isEditable;
protected boolean isRollover;
protected boolean isMovable;
protected boolean isResizable;
protected boolean isShowInvisibleBorder;
protected boolean isSelected;
// selection, move and resize handling.
protected int cursor;
protected Point startPos;
protected AnnotationState previousAnnotationState;
// total distance moved on mouse down/up.
protected Point startOfMousePress;
protected Point endOfMousePress;
protected ResourceBundle messageBundle;
public AbstractAnnotationComponent(Annotation annotation,
DocumentViewController documentViewController,
AbstractPageViewComponent pageViewComponent,
DocumentViewModel documentViewModel) {
this.pageViewComponent = pageViewComponent;
this.documentViewModel = documentViewModel;
this.documentViewController = documentViewController;
this.annotation = annotation;
messageBundle = documentViewController.getParentController().getMessageBundle();
// border and behavior default properties.
isEditable = !annotation.getFlagReadOnly();
isRollover = false;
isMovable = !(annotation.getFlagReadOnly() || annotation.getFlagLocked());
isResizable = !(annotation.getFlagReadOnly() || annotation.getFlagLocked());
addMouseListener(this);
addMouseMotionListener(this);
// disabled focus until we are ready to implement our own handler.
setFocusable(true);
addFocusListener(this);
// setup a resizable border.
setLayout(new BorderLayout());
setBorder(resizableBorder);
// set component location and original size.
Page currentPage = pageViewComponent.getPage();
AffineTransform at = currentPage.getPageTransform(
documentViewModel.getPageBoundary(),
documentViewModel.getViewRotation(),
documentViewModel.getViewZoom());
final Rectangle location =
at.createTransformedShape(annotation.getUserSpaceRectangle()).getBounds();
SwingUtilities.invokeLater(new Runnable() {
public void run() {
setBounds(location);
}
});
// update zoom and rotation state
currentRotation = documentViewModel.getViewRotation();
currentZoom = documentViewModel.getViewZoom();
resizableBorder.setZoom(currentZoom);
}
public abstract boolean isActive();
public Document getDocument() {
return documentViewModel.getDocument();
}
public int getPageIndex() {
return pageViewComponent.getPageIndex();
}
public PageViewComponent getParentPageView() {
return pageViewComponent;
}
public AbstractPageViewComponent getPageViewComponent() {
return pageViewComponent;
}
public void removeMouseListeners() {
removeMouseListener(this);
removeMouseMotionListener(this);
}
public Annotation getAnnotation() {
return annotation;
}
public void focusGained(FocusEvent e) {
isSelected = true;
// on mouse enter pass event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_FO_KEY, null);
repaint();
}
public void focusLost(FocusEvent e) {
// if we've lost focus then drop the selected state
isSelected = false;
// on mouse enter pass event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_Bl_KEY, null);
repaint();
}
protected void resize() {
if (getParent() != null) {
getParent().validate();
}
resized = true;
}
/**
* Refreshes the components bounds for the current page transformation.
* Bounds have are already in user space.
*/
public void refreshDirtyBounds() {
Page currentPage = pageViewComponent.getPage();
AffineTransform at = currentPage.getPageTransform(
documentViewModel.getPageBoundary(),
documentViewModel.getViewRotation(),
documentViewModel.getViewZoom());
setBounds(commonBoundsNormalization(new GeneralPath(
annotation.getUserSpaceRectangle()), at));
}
/**
* Refreshes/transforms the page space bounds back to user space. This
* must be done in order refresh the annotation user space rectangle after
* UI manipulation, otherwise the annotation will be incorrectly located
* on the next repaint.
*/
public void refreshAnnotationRect() {
Page currentPage = pageViewComponent.getPage();
AffineTransform at = currentPage.getPageTransform(
documentViewModel.getPageBoundary(),
documentViewModel.getViewRotation(),
documentViewModel.getViewZoom());
try {
at = at.createInverse();
} catch (NoninvertibleTransformException e) {
logger.log(Level.FINE, "Error refreshing annotation rectangle", e);
}
// store the new annotation rectangle in its original user space
Rectangle2D rect = annotation.getUserSpaceRectangle();
rect = new Rectangle2D.Double(rect.getX(), rect.getY(), rect.getWidth(), rect.getHeight());
Rectangle bounds = getBounds();
rect.setRect(commonBoundsNormalization(new GeneralPath(bounds), at));
annotation.syncBBoxToUserSpaceRectangle(rect);
}
/**
* Normalizes and the given path with the specified transform. The method
* also rounds the Rectangle2D bounds values when creating a new rectangle
* instead of truncating the values.
*
* @param shapePath path to apply transform to
* @param at transform to apply to shapePath
* @return bound value of the shape path.
*/
protected Rectangle commonBoundsNormalization(GeneralPath shapePath,
AffineTransform at) {
shapePath.transform(at);
Rectangle2D pageSpaceBound = shapePath.getBounds2D();
return new Rectangle(
(int) Math.round(pageSpaceBound.getX()),
(int) Math.round(pageSpaceBound.getY()),
(int) Math.round(pageSpaceBound.getWidth()),
(int) Math.round(pageSpaceBound.getHeight()));
}
public void validate() {
if (currentZoom != documentViewModel.getViewZoom() ||
currentRotation != documentViewModel.getViewRotation()) {
refreshDirtyBounds();
currentRotation = documentViewModel.getViewRotation();
currentZoom = documentViewModel.getViewZoom();
resizableBorder.setZoom(currentZoom);
}
if (resized) {
refreshAnnotationRect();
if (getParent() != null) {
// getParent().validate();
getParent().repaint();
}
resized = false;
wasResized = true;
}
}
abstract public void paintComponent(Graphics g);
abstract public void resetAppearanceShapes();
public void mouseMoved(MouseEvent me) {
int toolMode = documentViewModel.getViewToolMode();
if (toolMode == DocumentViewModel.DISPLAY_TOOL_SELECTION &&
!(annotation.getFlagLocked() || annotation.getFlagReadOnly())) {
Border border = getBorder();
if (border instanceof ResizableBorder) {
setCursor(Cursor.getPredefinedCursor(((ResizableBorder) border).getCursor(me)));
}
} else {
// set cursor back to the hand cursor.
setCursor(documentViewController.getViewCursor(
DocumentViewController.CURSOR_HAND_ANNOTATION));
}
}
public void dispose() {
removeMouseListener(this);
removeMouseMotionListener(this);
// disabled focus until we are ready to implement our own handler.
removeFocusListener(this);
}
public void mouseExited(MouseEvent mouseEvent) {
// set selected appearance state
annotation.setCurrentAppearance(Annotation.APPEARANCE_STREAM_NORMAL_KEY);
// on exit pass event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_X_KEY, mouseEvent);
setCursor(Cursor.getDefaultCursor());
isRollover = false;
repaint();
}
public void mouseClicked(MouseEvent e) {
// clear the selection.
requestFocus();
}
public void mouseEntered(MouseEvent e) {
// reset the appearance steam
Appearance hover = annotation.getAppearances().get(Annotation.APPEARANCE_STREAM_ROLLOVER_KEY);
if (hover != null && hover.hasAlternativeAppearance()) {
// set selected appearance state
hover.setSelectedName(hover.getOnName());
annotation.setCurrentAppearance(Annotation.APPEARANCE_STREAM_ROLLOVER_KEY);
}
// set border highlight when mouse over.
isRollover = (documentViewModel.getViewToolMode() ==
DocumentViewModel.DISPLAY_TOOL_SELECTION ||
(this instanceof PopupAnnotationComponent));
// on mouse enter pass event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
//additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_E_KEY, e);
repaint();
}
public void mousePressed(MouseEvent e) {
// setup visual effect when the mouse button is pressed or held down
// inside the active area of the annotation.
isMousePressed = true;
int x = 0, y = 0;
Point point = new Point();
if (e != null){
x = e.getX();
y = e.getY();
point = e.getPoint();
}
startOfMousePress = point;
endOfMousePress = new Point(point); // need clone not a copy...
// check if there is a mouse down state
Appearance down = annotation.getAppearances().get(Annotation.APPEARANCE_STREAM_DOWN_KEY);
if (down != null && down.hasAlternativeAppearance()) {
if (down.getSelectedName().equals(down.getOnName())) {
down.setSelectedName(down.getOffName());
} else {
down.setSelectedName(down.getOnName());
}
annotation.setCurrentAppearance(Annotation.APPEARANCE_STREAM_DOWN_KEY);
}
if (documentViewModel.getViewToolMode() ==
DocumentViewModel.DISPLAY_TOOL_SELECTION &&
isInteractiveAnnotationsEnabled &&
!annotation.getFlagReadOnly()) {
initiateMouseMoved(e);
}
// on mouse pressed event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
boolean actionFired = additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_D_KEY, e);
// fire the main action associated with the
if (!actionFired && !(AbstractPageViewComponent.isAnnotationTool(
documentViewModel.getViewToolMode())) &&
isInteractiveAnnotationsEnabled) {
if (documentViewController.getAnnotationCallback() != null) {
// get the A and AA entries.
Action action = annotation.getAction();
documentViewController.getAnnotationCallback()
.processAnnotationAction(annotation, action, x, y);
}
}
repaint();
}
protected boolean additionalActionsHandler(Name additionalActionKey, MouseEvent e) {
if (!(AbstractPageViewComponent.isAnnotationTool(
documentViewModel.getViewToolMode())) &&
isInteractiveAnnotationsEnabled) {
if (documentViewController.getAnnotationCallback() != null) {
int x = -1, y = -1;
if (e != null) {
x = e.getX();
y = e.getY();
}
// get the A and AA entries.
if (annotation instanceof AbstractWidgetAnnotation) {
AbstractWidgetAnnotation widgetAnnotation = (AbstractWidgetAnnotation) annotation;
FieldDictionary fieldDictionary = (FieldDictionary) widgetAnnotation.getFieldDictionary();
if (fieldDictionary != null) {
AdditionalActionsDictionary additionalActionsDictionary =
fieldDictionary.getAdditionalActionsDictionary();
if (additionalActionsDictionary != null &&
additionalActionsDictionary.isAnnotationValue(additionalActionKey)) {
documentViewController.getAnnotationCallback()
.processAnnotationAction(annotation,
additionalActionsDictionary.getAction(additionalActionKey),
x, y);
return true;
}
}
}
}
}
return false;
}
protected void initiateMouseMoved(MouseEvent e) {
Border border = getBorder();
if (border != null && border instanceof ResizableBorder) {
cursor = ((ResizableBorder) border).getCursor(e);
}
startPos = e.getPoint();
previousAnnotationState = new AnnotationState(this);
// mark annotation as selected.
documentViewController.assignSelectedAnnotation(this);
}
public void mouseDragged(MouseEvent me) {
if (startPos != null && isMovable &&
!(annotation.getFlagLocked() || annotation.getFlagReadOnly())) {
int x = getX();
int y = getY();
int w = getWidth();
int h = getHeight();
int dx = me.getX() - startPos.x;
int dy = me.getY() - startPos.y;
if (endOfMousePress != null) {
endOfMousePress.setLocation(endOfMousePress.x + dx, endOfMousePress.y + dy);
}
switch (cursor) {
case Cursor.N_RESIZE_CURSOR:
if (isResizable && !(h - dy < 12)) {
setBounds(x, y + dy, w, h - dy);
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.S_RESIZE_CURSOR:
if (isResizable && !(h + dy < 12)) {
setBounds(x, y, w, h + dy);
startPos = me.getPoint();
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.W_RESIZE_CURSOR:
if (isResizable && !(w - dx < 18)) {
setBounds(x + dx, y, w - dx, h);
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.E_RESIZE_CURSOR:
if (isResizable && !(w + dx < 18)) {
setBounds(x, y, w + dx, h);
startPos = me.getPoint();
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.NW_RESIZE_CURSOR:
if (isResizable && !(w - dx < 18) && !(h - dy < 18)) {
setBounds(x + dx, y + dy, w - dx, h - dy);
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.NE_RESIZE_CURSOR:
if (isResizable && !(w + dx < 18) && !(h - dy < 18)) {
setBounds(x, y + dy, w + dx, h - dy);
startPos = new Point(me.getX(), startPos.y);
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.SW_RESIZE_CURSOR:
if (isResizable && !(w - dx < 18) && !(h + dy < 18)) {
setBounds(x + dx, y, w - dx, h + dy);
startPos = new Point(startPos.x, me.getY());
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.SE_RESIZE_CURSOR:
if (isResizable && !(w + dx < 18) && !(h + dy < 18)) {
setBounds(x, y, w + dx, h + dy);
startPos = me.getPoint();
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
case Cursor.MOVE_CURSOR:
if (isMovable) {
Rectangle bounds = getBounds();
bounds.translate(dx, dy);
setBounds(bounds);
resize();
setCursor(Cursor.getPredefinedCursor(cursor));
}
break;
}
validate();
}
}
public void mouseReleased(MouseEvent mouseEvent) {
startPos = null;
isMousePressed = false;
// reset the appearance steam
Appearance down = annotation.getAppearances().get(Annotation.APPEARANCE_STREAM_DOWN_KEY);
if (down != null && down.hasAlternativeAppearance()) {
if (down.getSelectedName().equals(down.getOnName())) {
down.setSelectedName(down.getOffName());
} else {
down.setSelectedName(down.getOnName());
}
}
// set selected appearance state
annotation.setCurrentAppearance(Annotation.APPEARANCE_STREAM_NORMAL_KEY);
// check to see if a move/resize occurred and if so we add the
// state change to the memento in document view.
if (wasResized) {
wasResized = false;
// update the bounds
refreshAnnotationRect();
double dx = 0;
double dy = 0;
if (startOfMousePress != null &&
endOfMousePress != null) {
dx = endOfMousePress.getX() - startOfMousePress.getX();
dy = endOfMousePress.getY() - startOfMousePress.getY();
}
annotation.resetAppearanceStream(dx, -dy, getPageTransform());
// fire new bounds change event, let the listener handle
// how to deal with the bound change.
documentViewController.firePropertyChange(
PropertyConstants.ANNOTATION_BOUNDS,
previousAnnotationState, new AnnotationState(this));
// notify the annotation callback of the annotation resize.
if (documentViewController.getAnnotationCallback() != null) {
documentViewController.getAnnotationCallback()
.updateAnnotation(this);
}
}
// on mouse released event to annotation callback if we are in normal viewing
// mode. A and AA dictionaries are taken into consideration.
additionalActionsHandler(AdditionalActionsDictionary.ANNOTATION_U_KEY, mouseEvent);
repaint();
}
/**
* Convert the shapes that make up the annotation to page space so that
* they will scale correctly at different zooms.
*
* @return transformed bbox.
*/
protected Rectangle convertToPageSpace(Rectangle rect) {
Page currentPage = pageViewComponent.getPage();
AffineTransform at = currentPage.getPageTransform(
documentViewModel.getPageBoundary(),
documentViewModel.getViewRotation(),
documentViewModel.getViewZoom());
try {
at = at.createInverse();
} catch (NoninvertibleTransformException e) {
logger.log(Level.FINE, "Error converting to page space.", e);
}
// convert the two points as well as the bbox.
Rectangle tBbox = new Rectangle(rect.x, rect.y,
rect.width, rect.height);
tBbox = at.createTransformedShape(tBbox).getBounds();
return tBbox;
}
protected AffineTransform getPageTransform() {
Page currentPage = pageViewComponent.getPage();
AffineTransform at = currentPage.getPageTransform(
documentViewModel.getPageBoundary(),
documentViewModel.getViewRotation(),
documentViewModel.getViewZoom());
try {
at = at.createInverse();
} catch (NoninvertibleTransformException e) {
logger.log(Level.FINE, "Error getting page transform.", e);
}
return at;
}
/**
* Is the annotation editable
*
* @return true if editable, false otherwise.
*/
public boolean isEditable() {
return isEditable;
}
public boolean isRollover() {
return isRollover;
}
public boolean isBorderStyle() {
return annotation.isBorder();
}
public boolean isSelected() {
return isSelected;
}
public void setSelected(boolean selected) {
isSelected = selected;
}
public boolean isMovable() {
return isMovable;
}
public boolean isResizable() {
return isResizable;
}
public boolean isShowInvisibleBorder() {
return isShowInvisibleBorder;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy