org.sejda.impl.sambox.component.PdfScaler Maven / Gradle / Ivy
/*
* Created on 15 nov 2016
* Copyright 2015 by Andrea Vacondio ([email protected]).
* Copyright 2017 by Edi Weissmann ([email protected]).
*
* This file is part of Sejda.
*
* Sejda 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.
*
* Sejda 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 Sejda. If not, see .
*/
package org.sejda.impl.sambox.component;
import static java.util.Objects.nonNull;
import static java.util.Optional.ofNullable;
import static org.sejda.commons.util.RequireUtils.requireNotNullArg;
import static org.sejda.model.scale.Margins.inchesToPoints;
import java.awt.geom.AffineTransform;
import java.awt.geom.Point2D.Float;
import java.awt.geom.Rectangle2D;
import java.io.IOException;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import org.sejda.model.exception.TaskIOException;
import org.sejda.model.scale.Margins;
import org.sejda.model.scale.ScaleType;
import org.sejda.sambox.cos.COSArray;
import org.sejda.sambox.cos.COSDictionary;
import org.sejda.sambox.cos.COSFloat;
import org.sejda.sambox.cos.COSName;
import org.sejda.sambox.pdmodel.PDDocument;
import org.sejda.sambox.pdmodel.PDPage;
import org.sejda.sambox.pdmodel.PDPageContentStream;
import org.sejda.sambox.pdmodel.PDPageContentStream.AppendMode;
import org.sejda.sambox.pdmodel.common.PDRectangle;
import org.sejda.sambox.pdmodel.interactive.annotation.PDAnnotationLine;
import org.sejda.sambox.util.Matrix;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Component capable of scaling pages or pages content
*
* @author Andrea Vacondio
* @author Eduard Weissmann
*
*/
public class PdfScaler {
private static final Logger LOG = LoggerFactory.getLogger(PdfScaler.class);
private ScaleType type;
public PdfScaler(ScaleType type) {
requireNotNullArg(type, "Scale type cannot be null");
this.type = type;
}
/**
* Resizes all pages in the doc to match the size of the first page Eg: a doc with first 2 pages A4 and next ones A5 will be changed to all pages are A4
*/
public void scalePages(PDDocument doc) throws TaskIOException {
PDPage firstPage = doc.getPage(0);
PDRectangle targetBox = firstPage.getCropBox().rotate(firstPage.getRotation());
scalePages(doc, doc.getPages(), targetBox);
}
/**
* Changes the size of the given pages so they all match the target width The pages are scaled, so the aspect ratio is preserved.
*/
public void scalePages(PDDocument doc, Iterable pages, PDRectangle targetBox) throws TaskIOException {
for (PDPage page : pages) {
PDRectangle cropBox = page.getCropBox().rotate(page.getRotation());
double scale = getScalingFactorMatchWidth(targetBox, cropBox);
LOG.debug("Scaling page from {} to {}, factor of {}", cropBox, targetBox, scale);
scale(doc, page, scale);
}
}
public void changePageSize(PDDocument doc, Iterable pages, PDRectangle desiredPageSize)
throws TaskIOException {
for (PDPage page : pages) {
changePageSize(doc, page, desiredPageSize);
}
}
public void scale(PDDocument doc, double scale) throws TaskIOException {
scale(doc, doc.getPages(), scale);
}
public void scale(PDDocument doc, PDPage page, double scale) throws TaskIOException {
scale(doc, Collections.singletonList(page), scale);
}
public void scale(PDDocument doc, Iterable pages, double scale) throws TaskIOException {
if (scale != 1) {
doScale(doc, pages, scale);
}
}
public void changePageSize(PDDocument doc, PDPage page, PDRectangle desiredPageSize) throws TaskIOException {
PDRectangle currentPageSize = page.getCropBox().rotate(page.getRotation());
LOG.debug("Current page size: {}", currentPageSize);
LOG.debug("Desired page size: {}", desiredPageSize);
// first we scale the page to fit the new desired size
double scale = getScalingFactorMatchWidthOrHeight(desiredPageSize, currentPageSize);
doScale(doc, Collections.singletonList(page), scale);
// if the aspect ratio for the current size and new desired size are the same -> we are done
// otherwise, we need to add margins to reach the desired size
PDRectangle scaledPageSize = page.getCropBox().rotate(page.getRotation());
// LOG.debug("Scaled by {}", scale);
PDRectangle normalizedScaledPageSize = scaledPageSize;
boolean mismatchingOrientation = isLandscape(scaledPageSize) != isLandscape(desiredPageSize);
if (mismatchingOrientation) {
normalizedScaledPageSize = scaledPageSize.rotate();
}
// LOG.debug("Scaled page size: {}", normalizedScaledPageSize);
// LOG.debug("Desired page size: {}", desiredPageSize);
double widthDiff = desiredPageSize.getWidth() - normalizedScaledPageSize.getWidth();
double heightDiff = desiredPageSize.getHeight() - normalizedScaledPageSize.getHeight();
// LOG.debug("Differences are widthDiff: {} heightDiff: {}", widthDiff, heightDiff);
if (widthDiff < 1)
widthDiff = 0;
if (heightDiff < 1)
heightDiff = 0;
if (widthDiff > 0 || heightDiff > 0) {
// Margins are in inches, not points
double top = Margins.pointsToInches(heightDiff) / 2;
double left = Margins.pointsToInches(widthDiff) / 2;
Margins margins = new Margins(top, left, top, left);
if (mismatchingOrientation) {
margins = margins.rotate();
}
margin(doc, Collections.singleton(page), margins);
}
// PDRectangle finalPageSize = page.getCropBox().rotate(page.getRotation());
// LOG.debug("Final page size: {}", finalPageSize);
// LOG.debug("Desired page size: {}", desiredPageSize);
}
private void doScale(PDDocument doc, Iterable pages, double scale) throws TaskIOException {
Set processedAnnots = new HashSet<>();
for (PDPage page : pages) {
try (PDPageContentStream contentStream = new PDPageContentStream(doc, page, AppendMode.PREPEND, true)) {
Matrix matrix = getMatrix(scale, page.getCropBox(), page.getCropBox());
contentStream.transform(matrix);
if (ScaleType.PAGE == type) {
scalePageBoxes(scale, page);
} else {
scaleContentBoxes(scale, page);
}
transformAnnotations(page, matrix, processedAnnots);
} catch (IOException e) {
throw new TaskIOException("An error occurred writing scaling the page.", e);
}
}
}
private static void transformAnnotations(PDPage page, Matrix transform, Set processedAnnots) {
page.getAnnotations().stream().filter(a -> !processedAnnots.contains(a.getCOSObject())).forEach(a -> {
// set the new rectangle
ofNullable(a.getRectangle()).map(r -> r.transform(transform).getBounds2D()).map(PDRectangle::new)
.ifPresent(a::setRectangle);
// Text Markup, Link and Redaction annotations can have quadpoints
ofNullable(a.getCOSObject().getDictionaryObject(COSName.QUADPOINTS, COSArray.class))
.filter(p -> p.size() == 8).map(COSArray::toFloatArray).ifPresent(f -> {
a.getCOSObject().setItem(COSName.QUADPOINTS, transformPoints(f, transform));
});
// adjust line length
if (a instanceof PDAnnotationLine) {
ofNullable(((PDAnnotationLine) a).getLine()).filter(p -> p.length == 4).ifPresent(f -> {
a.getCOSObject().setItem(COSName.L, transformPoints(f, transform));
});
}
// adjust Free Text CL
ofNullable(a.getCOSObject().getDictionaryObject(COSName.CL, COSArray.class)).filter(p -> p.size() % 2 == 0)
.map(COSArray::toFloatArray).ifPresent(f -> {
a.getCOSObject().setItem(COSName.CL, transformPoints(f, transform));
});
// Polygon and Polyline vertices
ofNullable(a.getCOSObject().getDictionaryObject(COSName.VERTICES, COSArray.class))
.filter(p -> p.size() % 2 == 0).map(COSArray::toFloatArray).ifPresent(f -> {
a.getCOSObject().setItem(COSName.VERTICES, transformPoints(f, transform));
});
processedAnnots.add(a.getCOSObject());
});
}
private static COSArray transformPoints(float[] points, Matrix transform) {
COSArray newPoints = new COSArray();
for (int i = 0; i < points.length; i++) {
Float newPoint = transform.transformPoint(points[i], points[++i]);
newPoints.add(new COSFloat(newPoint.x));
newPoints.add(new COSFloat(newPoint.y));
}
return newPoints;
}
/**
* Adds the given margin all around the pages
*/
public static void margin(PDDocument doc, Iterable pages, Margins margins) throws TaskIOException {
if (nonNull(margins)) {
Set processedAnnots = new HashSet<>();
for (PDPage page : pages) {
try (PDPageContentStream contentStream = new PDPageContentStream(doc, page, AppendMode.PREPEND, true)) {
page.setCropBox(addMargins(page.getCropBox().rotate(page.getRotation()), margins)
.rotate(-page.getRotation()));
page.setMediaBox(addMargins(page.getMediaBox().rotate(page.getRotation()), margins)
.rotate(-page.getRotation()));
ofNullable(page.getBleedBoxRaw()).ifPresent(r -> page.setBleedBox(
addMargins(r.rotate(page.getRotation()), margins).rotate(-page.getRotation())));
ofNullable(page.getTrimBoxRaw()).ifPresent(r -> page
.setTrimBox(addMargins(r.rotate(page.getRotation()), margins).rotate(-page.getRotation())));
ofNullable(page.getArtBoxRaw()).ifPresent(r -> page
.setArtBox(addMargins(r.rotate(page.getRotation()), margins).rotate(-page.getRotation())));
Matrix matrix = new Matrix(AffineTransform.getTranslateInstance(inchesToPoints(margins.left),
inchesToPoints(margins.bottom)));
// realign the content
contentStream.transform(matrix);
transformAnnotations(page, matrix, processedAnnots);
} catch (IOException e) {
throw new TaskIOException("An error occurred adding margins to the page.", e);
}
}
}
}
private static PDRectangle addMargins(PDRectangle rect, Margins margins) {
return new PDRectangle(rect.getLowerLeftX(), rect.getLowerLeftY(),
(float) (rect.getWidth() + inchesToPoints(margins.left) + inchesToPoints(margins.right)),
(float) (rect.getHeight() + inchesToPoints(margins.top) + inchesToPoints(margins.bottom)));
}
private Matrix getMatrix(double scale, PDRectangle crop, PDRectangle toScale) {
if (ScaleType.CONTENT == type) {
AffineTransform transform = AffineTransform.getTranslateInstance(
(crop.getWidth() - (toScale.getWidth() * scale)) / 2,
(crop.getHeight() - (toScale.getHeight() * scale)) / 2);
transform.scale(scale, scale);
return new Matrix(transform);
}
return new Matrix(AffineTransform.getScaleInstance(scale, scale));
}
private void scaleContentBoxes(double scale, PDPage page) {
PDRectangle cropBox = page.getCropBox();
// we adjust art and bleed same as Acrobat does
if (scale > 1) {
page.setBleedBox(cropBox);
page.setTrimBox(cropBox);
} else {
page.setBleedBox(new PDRectangle(page.getBleedBox()
.transform(getMatrix(scale, page.getCropBox(), page.getBleedBox())).getBounds2D()));
page.setTrimBox(new PDRectangle(
page.getTrimBox().transform(getMatrix(scale, page.getCropBox(), page.getTrimBox())).getBounds2D()));
}
Rectangle2D newArt = page.getArtBox().transform(getMatrix(scale, page.getCropBox(), page.getArtBox()))
.getBounds2D();
if (newArt.getX() < cropBox.getLowerLeftX() || newArt.getY() < cropBox.getLowerLeftX()) {
// we overlow the cropbox
page.setArtBox(page.getCropBox());
} else {
page.setArtBox(new PDRectangle(newArt));
}
}
private void scalePageBoxes(double scale, PDPage page) {
page.setArtBox(new PDRectangle(
page.getArtBox().transform(getMatrix(scale, page.getCropBox(), page.getArtBox())).getBounds2D()));
page.setBleedBox(new PDRectangle(
page.getBleedBox().transform(getMatrix(scale, page.getCropBox(), page.getBleedBox())).getBounds2D()));
page.setTrimBox(new PDRectangle(
page.getTrimBox().transform(getMatrix(scale, page.getCropBox(), page.getTrimBox())).getBounds2D()));
page.setCropBox(new PDRectangle(
page.getCropBox().transform(getMatrix(scale, page.getCropBox(), page.getCropBox())).getBounds2D()));
page.setMediaBox(new PDRectangle(
page.getMediaBox().transform(getMatrix(scale, page.getMediaBox(), page.getMediaBox())).getBounds2D()));
}
/**
* Calculates the factor that pageBox can be scaled with to fit targetBox's **width** without overflow. For some pages this could mean that pageBox scaled will be taller than
* the targetBox.
*/
private double getScalingFactorMatchWidth(PDRectangle targetBox, PDRectangle pageBox) {
// if both target and page boxes have same orientation (landscape, portrait)
// the scaling factor is targetWidth / pageWidth
if (isLandscape(targetBox) == isLandscape(pageBox)) {
return targetBox.getWidth() / pageBox.getWidth();
}
// the boxes have different orientations
// the page should be scaled to match the target box height
return targetBox.getHeight() / pageBox.getWidth();
}
/**
* Calculates the factor that pageBox can be scaled with to fit targetBox without overflow. Ensures both height and width will not overflow targetBox.
*/
private double getScalingFactorMatchWidthOrHeight(PDRectangle targetBox, PDRectangle pageBox) {
// if the boxes have different orientations
// we want to normalize first rotating the target 90 and then calculate scaling factors
// so landscape is scaled to a landscape
PDRectangle normalizedOrientationTargetBox = targetBox;
if (isLandscape(targetBox) != isLandscape(pageBox)) {
normalizedOrientationTargetBox = targetBox.rotate();
}
float widthFactor = normalizedOrientationTargetBox.getWidth() / pageBox.getWidth();
float heightFactor = normalizedOrientationTargetBox.getHeight() / pageBox.getHeight();
float factor = Math.min(widthFactor, heightFactor);
// LOG.debug("Factor: {} for normalizedOrientationTargetBox: {} pageBox: {}", factor, normalizedOrientationTargetBox, pageBox);
return factor;
}
private boolean isLandscape(PDRectangle box) {
return box.getWidth() > box.getHeight();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy