I am making a web application using Javascript for the front end and this is how it works:
I start the application and it opens a web page via my browser.
It displays a PDF page obtained from my directory.
I have the option to click on a stamp and drag and move around the pdf and place where ever I want.
When I'm done, I could click Save and it automatically saves the pdf file in my directory.
I can open the pdf file in my folder to view the updated PDF along with the stamp added.
The problem is when I open the PDF file to view, the positioning is not identical to the positioning of the stamp in the web browser.
window.dragMoveListener = dragMoveListener;
interact('.signer-box')
.draggable({
onmove: dragMoveListener,
inertia: true,
autoScroll: true,
restrict: {
elementRect: {top: 0, left: 0, bottom: 1, right: 1}
}
})
.resizable({
onmove: resizeMoveListener,
inertia: true,
edges: {left: true, right: true, bottom: true, top: true}
})
function dragMoveListener(event) {
var target = event.target;
var x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;
var y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;
target.style.webkitTransform = target.style.transform = 'translate(' + x + 'px, ' + y + 'px)';
target.setAttribute('data-x', x);
target.setAttribute('data-y', y);
computeSignerBoxPosition();
}
function resizeMoveListener(event) {
var target = event.target;
var x = (parseFloat(target.getAttribute('data-x')) || 0);
var y = (parseFloat(target.getAttribute('data-y')) || 0);
x += event.deltaRect.left;
y += event.deltaRect.top;
target.style.width = event.rect.width + 'px';
target.style.height = event.rect.height + 'px';
target.style.webkitTransform = target.style.transform = 'translate(' + x + 'px,' + y + 'px)';
target.setAttribute('data-x', x);
target.setAttribute('data-y', y);
computeSignerBoxPosition();
}
function computeSignerBoxPosition() {
var $signatureBox = $('.signer-box');
var sbDataX = parseFloat($signatureBox.attr('data-x'));
var sbDataY = parseFloat($signatureBox.attr('data-y'));
var sbOuterWidth = $signatureBox.outerWidth();
var sbOuterHeight = $signatureBox.outerHeight();
var w = $('#pdf-page').width();
var h = $('#pdf-page').height();
var top = sbDataX / w;
var left = sbDataY / h;
var width = sbOuterWidth / w;
var height = sbOuterHeight / h;
document.getElementById("widthValue").value = width;
document.getElementById("heightValue").value = height;
document.getElementById("coorX").value = top;
document.getElementById("coorY").value = left;
}
@charset "UTF-8";
#content{
text-align: center;
}
#pdf-container {
display: inline-block;
width: 100%;
user-select: none;
}
#pdf-page {
width: 100%;
}
.signer-box {
background: url('../images/pen_icon.png') #29e no-repeat 50% 50%;
background-size: 50%;
color: white;
font-size: 20px;
font-family: sans-serif;
border-radius: 8px;
width: 180px;
height: 150px;
position:absolute;
opacity: .8;
box-sizing: border-box;
box-shadow: rgb(0, 0, 0, 0.7) 0.2em 0.2em 0.5em;
-ms-touch-action: none;
touch-action: none;
}
#signature-pad {
position: relative;
width: 100%;
height: 160px;
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
user-select: none;
}
#signatureImg{
width: 100%;
}
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<div id="content">
<div class="wrap">
<hr style="border:15px;"><hr style="border:2px;">
<div id="wrapper">
<div id="content">
<div id="pdf-container" >
<div id="signers-list">
<div id="signer-1" class="signer-box"></div>
</div>
<img id="pdf-page" src="" />
</div>
</div>
</div>
</table></form>
<hr style="border:15px;"><hr style="border:2px;">
<div class="content">
<table id="customers">
<tr>
<td>
X:
</td>
<td>
<input type="text" name="coorX" id="coorX" value="0" readonly="readonly">
</td>
<td>
h:
</td>
<td>
<input type="text" name="heightValue" id="heightValue" value="150" readonly="readonly">
</td>
</tr>
<tr>
<td>
Y:
</td>
<td>
<input type="text" name="coorY" id="coorY" value="0" readonly="readonly">
</td>
<td>
w:
</td>
<td>
<input type="text" name="widthValue" id="widthValue" value="180" readonly="readonly">
</td>
</tr>
</table>
<hr>
</form>
</div><!-- /.wrap -->
</div><!-- /.content -->
</div>
<script src='https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js'></script>
<script src='https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.5/js/bootstrap.min.js'></script>
<script src='https://cdnjs.cloudflare.com/ajax/libs/pdf.js/2.0.943/pdf.min.js'></script>
<script src='https://cdnjs.cloudflare.com/ajax/libs/interact.js/1.2.9/interact.min.js'></script>
<script src='https://cdnjs.cloudflare.com/ajax/libs/pdf.js/2.0.943/pdf.worker.min.js'></script>
</body>
</html>
Below is the back-end code to handle the stamping part using iText:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException
{
File file = new File("D:\\Documents\\pruebaPdf\\Ejemplo_Uno.pdf");
file.getParentFile().mkdirs();
PdfReader reader = new PdfReader("D:\\Documents\\pruebaPdf\\Ejemplo_Uno.pdf");
PdfStamper stamper = new PdfStamper(reader, new FileOutputStream("D:\\Documents\\pruebaPdf\\Ejemplo_Dos.pdf"));
Document document = new Document();
try {
PdfContentByte cb = stamper.getOverContent(request.getParameter("page"));
float top = Float.valueOf(request.getParameter("top"));
float left = Float.valueOf(request.getParameter("left"));
float width = Float.valueOf(request.getParameter("width"));
float height = Float.valueOf(request.getParameter("height"));
// Just in case, take into account page rotation
Rectangle pdfRectangle = reader.getPageSizeWithRotation(1);
float pdfWidth = pdfRectangle.getWidth();
float pdfHeight = pdfRectangle.getHeight();
float llx = pdfWidth * left;
float lly = pdfHeight * (1 - top - height);
// Until iText 5 this code should work
float urx = llx + (pdfWidth * width);
float ury = lly + (pdfHeight * height);
Rectangle rect = new Rectangle(llx, lly, urx, ury);
rect.setBorder(Rectangle.BOX);
rect.setBorderWidth(1);
rect.setBackgroundColor(BaseColor.GRAY);
rect.setBorderColor(BaseColor.GREEN);
cb.rectangle(rect);
} catch (Exception e) {
System.out.println("ERROR=>>>>>>" + e);
}finally{
stamper.close();
reader.close();
document.close();
}
}
The position where I want the painting to come out
the position that I obtain with the coordinates that are calculated
As indicated in this companion question, the goal is to be able to translate the position and dimension of your image relative to the PDF between its representation in the browser and in the actual PDF.
In this specific use case you already have a well defined structure of elements, in which your PDF preview image is displayed in a predictable way.
Following the advice of the aforementioned question, I think you need to take the relevant points of your signature box, say:
var $signatureBox = $('.signer-box');
var sbDataX = parseFloat($signatureBox.attr('data-x'));
var sbDataY = parseFloat($signatureBox.attr('data-y'));
var sbOuterWidth = $signatureBox.outerWidth();
var sbOuterHeight = $signatureBox.outerHeight();
And convert them to percentages relative to the width and height of your PDF image:
var w = $('#pdf-page').width();
var h = $('#pdf-page').height();
var top = sbDataY / h;
var left = sbDataX / w;
var width = sbOuterWidth / w;
var height = sbOuterHeight / h;
I have used values relative to the unit, please, feel free to multiply them by 100
if you prefer to work with percentages. Just take into consideration in the next step.
These relative values top
, left
, width
and height
will be sent to your backend.
This information can be computed in your different listeners. Consider for example the definition of a common function that will be used to defined the right form values when either a drag and drop or a resize event occur:
window.dragMoveListener = dragMoveListener;
interact('.signer-box')
.draggable({
onmove: dragMoveListener,
inertia: true,
autoScroll: true,
restrict: {
elementRect: {top: 0, left: 0, bottom: 1, right: 1}
}
})
.resizable({
onmove: resizeMoveListener,
inertia: true,
edges: {left: true, right: true, bottom: true, top: true}
})
function dragMoveListener(event) {
var target = event.target;
var x = (parseFloat(target.getAttribute('data-x')) || 0) + event.dx;
var y = (parseFloat(target.getAttribute('data-y')) || 0) + event.dy;
target.style.webkitTransform = target.style.transform = 'translate(' + x + 'px, ' + y + 'px)';
target.setAttribute('data-x', x);
target.setAttribute('data-y', y);
computeSignerBoxPosition();
}
function resizeMoveListener(event) {
var target = event.target;
var x = (parseFloat(target.getAttribute('data-x')) || 0);
var y = (parseFloat(target.getAttribute('data-y')) || 0);
x += event.deltaRect.left;
y += event.deltaRect.top;
target.style.width = event.rect.width + 'px';
target.style.height = event.rect.height + 'px';
target.style.webkitTransform = target.style.transform = 'translate(' + x + 'px,' + y + 'px)';
target.setAttribute('data-x', x);
target.setAttribute('data-y', y);
computeSignerBoxPosition();
}
function computeSignerBoxPosition() {
var $signatureBox = $('.signer-box');
var sbDataX = parseFloat($signatureBox.attr('data-x'));
var sbDataY = parseFloat($signatureBox.attr('data-y'));
var sbOuterWidth = $signatureBox.outerWidth();
var sbOuterHeight = $signatureBox.outerHeight();
var w = $('#pdf-page').width();
var h = $('#pdf-page').height();
var top = sbDataY / h;
var left = sbDataX / w;
var width = sbOuterWidth / w;
var height = sbOuterHeight / h;
document.getElementById("widthValue").value = width;
document.getElementById("heightValue").value = height;
document.getElementById("coorX").value = left;
document.getElementById("coorY").value = top;
}
As mentioned, the idea is to compute the information that should be send to the backend when both listeners change. As an optimization, instead of computing the necessary information in the listener, please, consider attach to the button or visual element that you are using to submit your form a click listener that invoke the mentioned computeSignerBoxPosition
function prior sending the information to the backend as it is only necessary in that moment.
With that information, as indicated in the original answer, you can obtain the right position like this:
float top = Float.valueOf(request.getParameter("top"));
float left = Float.valueOf(request.getParameter("left"));
float width = Float.valueOf(request.getParameter("width"));
float height = Float.valueOf(request.getParameter("height"));
// Just in case, take into account page rotation
Rectangle pdfRectangle = reader.getPageSizeWithRotation(1);
float pdfWidth = pdfRectangle.getWidth();
float pdfHeight = pdfRectangle.getHeight();
float llx = pdfWidth * left;
float lly = pdfHeight * (1 - top - height);
// Until iText 5 this code should work
float urx = llx + (pdfWidth * width);
float ury = lly + (pdfHeight * height);
// It seems that changed in Itext7 to this (they use just width and height)
// If it is your use case, please, comment the block above
// and uncomment the following lines
// float urx = pdfWidth * width;
// float ury = pdfHeight * height;
Rectangle rect = new Rectangle(llx, lly, urx, ury);
PdfStampAnnotation stamp = new PdfStampAnnotation(rect).setStampName(new PdfName("Approved"));
PdfFormXObject xObj = new PdfFormXObject(new Rectangle(width,height));
PdfCanvas canvas = new PdfCanvas(xObj,doc);
canvas.addImage(image,0,0,false);
The algorithm takes into consideration that in PDF the lower-left corner of the page coincides with the origin of the coordinate system (0, 0)
, where positive x
values are to the right of the origin and positive y
values are above the origin (in contrast with the browser):
The credit for the image is for Bruno Lowagie when he describes how to interpret the coordinates of a rectangle in PDF and where is the origin of a PDF page.
It is important to note as well that the actual dimensions of your image and the PDF page in their representation in the browser and the actual PDF file should be constant, they need to preserve the proper aspect ratio; on the contrary, you need to correct your dimensions by the corresponding factor.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With