5

I am making a web application using Javascript for the front end and this is how it works:

  1. I start the application and it opens a web page via my browser.

  2. It displays a PDF page obtained from my directory.

  3. I have the option to click on a stamp and drag and move around the pdf and place where ever I want.

  4. When I'm done, I could click Save and it automatically saves the pdf file in my directory.

  5. 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 enter image description here

the position that I obtain with the coordinates that are calculated enter image description here

mkl
  • 90,588
  • 15
  • 125
  • 265
Sebastian Ruiz
  • 194
  • 2
  • 15
  • You mention itext and imply that the pdf manipulated using itext has the stamp at the wrong position. So the code using itext is important here. – mkl Jul 29 '21 at 20:46
  • @mkl Hello mkl you are right, I have edited the question – Sebastian Ruiz Jul 29 '21 at 21:06
  • As far as I can see the request parameters *mouseTop* and *mouseLeft* are never set. But they are used in the servlet to position the stamp. – mkl Jul 29 '21 at 21:14
  • 1
    But even if you merely forgot to mention the code responsible for setting those parameters, just say one thing: You are aware that you work in different coordinate systems in the browser and in the pdf and transform the coordinates accordingly, don't you? – mkl Jul 29 '21 at 21:26
  • @mkl and @K J It is exactly what I have in detail, I know it is not the same and that the coordinates will not coincide, so I asked for help to see if someone could help me – Sebastian Ruiz Jul 30 '21 at 00:19
  • @LouisCloete concerning your edit commented as *"This has absolutely nothing to do with Java. Java !== Javascript and Java != Javascript is even false as well."* - The question is about a JavaScript front end interacting with a Java back end. Thus, both tags _do_ apply. – mkl Aug 13 '21 at 08:38
  • Yes, I saw that later, but SO doesn't have the option to undo edit suggestions. I hoped the edit won't be accepted. If I could I'd just remove that edit altogether. Thanks for adding Java back as tag. – Louis Cloete Aug 13 '21 at 08:42

1 Answers1

4

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):

PDF coordinate system

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.

jccampanero
  • 50,989
  • 3
  • 20
  • 49
  • @SebastianRuiz Were you able to try the proposed solution? – jccampanero Aug 05 '21 at 10:18
  • According to your new question, it I very important to note that the `Rectangle` definition changed in iText7. As explained in the answer, until [iText 5](https://api.itextpdf.com/iText5/java/5.5.10/com/itextpdf/text/Rectangle.html#Rectangle-float-float-float-float-) you need to pass the rectangle's lower left and upper right x and y coordinates, in that order. Since [iText7](https://api.itextpdf.com/iText7/java/7.1.0/com/itextpdf/kernel/geom/Rectangle.html#Rectangle-float-float-float-float-), they changed the meaning of the last two arguments to be the width and height of the rectangle. – jccampanero Aug 05 '21 at 10:30
  • Depending on the version of the library you are using you need to take into consideration this fact. – jccampanero Aug 05 '21 at 10:30
  • Hello @jccampanero if it helped me however for the positions of x and and they still do not come out correct, I have updated the question with the code that I mention – Sebastian Ruiz Aug 06 '21 at 04:24
  • Hi @SebastianRuiz. Thank you very much for the feedback. Sorry, but I do not understand: you mean that the x and y position are still incorrect, is it right? Perhaps it has to do with the moment you are computing then in your code. Please, see the updated answer: the idea is to compute the information either in both listeners or ideally prior form submission. Please, can you try and see? Thank you. – jccampanero Aug 06 '21 at 10:07
  • Hi @jccampanero I have updated the question to show what I get – Sebastian Ruiz Aug 06 '21 at 15:21
  • Hi @SebastianRuiz. Of course... Sorry... All is fine, except that the definition of `top` and `left` are interchanged, they should be `var top = sbDataY / h; var left = sbDataX / w;`, as well as the values for `coorX` and `coorY`, thy should be `document.getElementById("coorX").value = left; document.getElementById("coorY").value = top;`. Please, see the updated answer. Please, can you try? – jccampanero Aug 06 '21 at 17:50
  • I already worked, brother thank you very much, do you have any place where I can follow you? – Sebastian Ruiz Aug 06 '21 at 21:58
  • You are welcome!! That is great @SebastianRuiz!! I am very very happy to hear that it worked properly. Thank you mate, although I don't think I deserve it... no Sebastian, unfortunately not: I will update my profile here if someday I wrote a blog or something similar. Thank you very much, I appreciate a lot that you want to follow me. Please, do not hesitate to contact me again if you need further help, I will be glad to help you if I can. – jccampanero Aug 06 '21 at 22:15
  • @SebastianRuiz I do not wanna say that you must grant me the bounty, but if you wanna give me the bounty it should be explicitly awarded before its time run out because you started it after I posted the answer. I am sorry to bother you, but it is quite frustrating when the bounty is lost for both the OP and the people who answered the question. Thank you very much – jccampanero Aug 12 '21 at 21:30
  • @jccampero Excuse me friend, this is the first time I do it, has it already worked? – Sebastian Ruiz Aug 13 '21 at 01:15
  • Hi @SebastianRuiz. Please, there is no need to apologize, on the contrary, excuse me and thank you very much. Yes, it worked. Usually, if you start a bounty and approve an answer, that answer will be the awarded, ... unless it was posted before the bounty starts. In that case, it is necessary to explicitly award it to that answer. Thank you very much again Sebastian, and sorry for the inconveniences. See you my friend – jccampanero Aug 13 '21 at 13:48