Convert HTML to PDF in Angular 6

后端 未结 6 1689
执念已碎
执念已碎 2021-02-05 04:50

I have a component (Angular 6) which is an aggregation of several components. This produces a long HTML (I am using Bootstrap 4). Now I want to convert this HTML to PDF. I have

相关标签:
6条回答
  • 2021-02-05 05:22

    Try to use a backend service for this such as Thymeleaf or itextpdf with Spring Boot.

    0 讨论(0)
  • 2021-02-05 05:36

    Best possible solution I could come up with till now.

    You would have to install the below packages from npm

    html2canvas

    jspdf

    import * as jsPDF from 'jspdf';
    import html2canvas from 'html2canvas';
    
    htmltoPDF()
    {
        // parentdiv is the html element which has to be converted to PDF
        html2canvas(document.querySelector("#parentdiv")).then(canvas => {
    
          var pdf = new jsPDF('p', 'pt', [canvas.width, canvas.height]);
    
          var imgData  = canvas.toDataURL("image/jpeg", 1.0);
          pdf.addImage(imgData,0,0,canvas.width, canvas.height);
          pdf.save('converteddoc.pdf');
    
      });
    
    }
    

    UPDATE:

    Came up with another solution. I wasn't able to break it down into A4 size pages, but I was able to make a single pdf file.

    Packages:

    dom-to-image

    jspdf

    import domtoimage from 'dom-to-image';
    import * as jsPDF from 'jspdf';
    
    
    
                downloadPDF()
                {
    
                  var node = document.getElementById('parentdiv');
    
                  var img;
                  var filename;
                  var newImage;
    
    
                  domtoimage.toPng(node, { bgcolor: '#fff' })
    
                    .then(function(dataUrl) {
    
                      img = new Image();
                      img.src = dataUrl;
                      newImage = img.src;
    
                      img.onload = function(){
    
                      var pdfWidth = img.width;
                      var pdfHeight = img.height;
    
                        // FileSaver.saveAs(dataUrl, 'my-pdfimage.png'); // Save as Image
    
                        var doc;
    
                        if(pdfWidth > pdfHeight)
                        {
                          doc = new jsPDF('l', 'px', [pdfWidth , pdfHeight]);
                        }
                        else
                        {
                          doc = new jsPDF('p', 'px', [pdfWidth , pdfHeight]);
                        }
    
    
                        var width = doc.internal.pageSize.getWidth();
                        var height = doc.internal.pageSize.getHeight();
    
    
                        doc.addImage(newImage, 'PNG',  10, 10, width, height);
                        filename = 'mypdf_' + '.pdf';
                        doc.save(filename);
    
                      };
    
    
                    })
                    .catch(function(error) {
    
                     // Error Handling
    
                    });
    
    
    
                }
    
    0 讨论(0)
  • 2021-02-05 05:36

    Step 1: Install required packages

    npm install jspdf jspdf-autotable
    

    Step 2: Import the installed packages to use

     import jsPDF from 'jspdf';
     import 'jspdf-autotable';
    

    Step 3: If your data has images, convert images to Base64

    tabledata = [{
        img: 'https://i.picsum.photos/id/258/536/354.jpg',
        name: 'Joseph',
        domain: 'Angular 9'
      },
      {
        img: 'https://i.picsum.photos/id/258/536/354.jpg',
        name: 'Vani',
        domain: 'Angular 8'
      }, {
        img: 'https://i.picsum.photos/id/258/536/354.jpg',
        name: 'Raj',
        domain: 'React.js'
      },
    ];
    

    The above is data with images converted to base64. The convert function looks like:

    convertImagetoBase64(url, callback) {
        const xhr = new XMLHttpRequest();
        xhr.onload = () => {
          const reader = new FileReader();
          reader.onloadend = () => {
            callback(reader.result);
          };
          reader.readAsDataURL(xhr.response);
        };
        xhr.open('GET', url);
        xhr.responseType = 'blob';
        xhr.send();
    }
    

    Let's convert the image URL to base64 before we bind to the table in HTML.

    this.tabledata.forEach(elem => {
      this.convertImagetoBase64(elem.img, (b64Img) => {
        elem.img = b64Img;
    });
    

    Now bind the data to the table. If you don't want to display the table in the front end, just style to display none as in the below code:

    <table id="imgTable" style=”display:none”>
    <thead>
        <tr><th>Name</th>
        <th>Domain</th>
        <th>Image</th>
        </tr>
    </thead>
    <tbody>
        <tr *ngFor="let data of tabledata">
            <td>{{data.name}}</td>
            <td>{{data.domain}}</td>
            <td><img src="{{data.img}}" width="100" height="100" /></td>
        </tr>
    
    </tbody>
    </table>
    <button (click)="generatePdf()">Generate Pdf</button>
    

    The Generate Pdf function looks like:

    app.component.ts

    generatePdf(){
     doc.autoTable({
      html: '#imgTable',
      bodyStyles: { minCellHeight: 20 },
      theme: 'grid',
      styles: { valign: 'middle', overflow: 'linebreak', halign: 'center', minCellHeight: 21 },
      pageBreak: 'avoid',
      columnStyles: {
        2: { cellWidth: 22, minCellHeight: 22 },
      },
      headStyles: { fillColor: '#f2f2f2', textColor: '#000', fontStyle: 'bold', lineWidth: 0.5, lineColor: '#ccc' },
      didDrawCell: (data) => {
        if (data.column.index === 2 && data.cell.section === 'body') {
          const td = data.cell.raw;
          const img = td.getElementsByTagName('img')[0];
          // let dim = data.cell.height - data.cell.padding('vertical');
          // let textPos = data.cell.textPos;
          doc.addImage(img.src, data.cell.x + 1, data.cell.y + 1, 19, 19);
    
        }
      }
    });
     doc.save('yourpdfname.pdf');
    }
    

    Example pdf generation looks like:

    For more Pdf format without table visit helperscript.com

    0 讨论(0)
  • 2021-02-05 05:37

    With some effort, the best solution my team and I came up with for our own Angular HTML to PDF problems involved sending the raw HTML to our Java API and then utilizing wkhtmltopdf (https://github.com/wkhtmltopdf/wkhtmltopdf) with the Java wrapper (https://github.com/jhonnymertz/java-wkhtmltopdf-wrapper). Overcoming a few gotchas (like making sure the server has the correctly installed fonts; everything is properly encoded; figuring out page breaks and making them work), we were able to reproduce our page exactly and use our Angular HTML as the universal template (which changes based on different scenarios). It is not an easy road, but it is one that produces great results in an enterprise-level production environment once you get it figured out.

    It should be noted that we also tried jspdf (and other libraries) and came to similar conclusions as yourself: they just didn't do what we needed them to do. Hopefully this helps.

    0 讨论(0)
  • 2021-02-05 05:37

    Firstly install the following: 1) npm install jspdf 2) npm install html2canvas

    Later import and use the following code

     public captureScreen() {
          const data = document.getElementById('invoice');
          html2canvas(data).then(canvas => {
          const imgWidth = 208;
          const pageHeight = 295;
          const imgHeight = canvas.height * imgWidth / canvas.width;
          const heightLeft = imgHeight;
          const contentDataURL = canvas.toDataURL('image/png');
          const pdf = new jspdf('p', 'mm', 'a4'); 
          const position = 0;
          pdf.addImage(contentDataURL, 'PNG', 0, position, imgWidth, imgHeight);
          pdf.save('invoice.pdf'); 
          });
          }
    
    0 讨论(0)
  • 2021-02-05 05:46

    The following code was used and worked for my project

    Step 1 : Run following commands to install npm packages

    > npm install jspdf
    > npm install html2canvas
    

    Step 2: Import installed packages in app.components.ts. I haven't imported those packages in constructor()

    > import * as jspdf from 'jspdf';
    > import html2canvas from 'html2canvas';
    

    Step 3: Give an id for the HTML div that has to be exported as PDF. Add a button that activates the function too.

    <div id="MyDIv" style="margin-left: 45px;" class="main-container">
    
    </div>
    <div class="icon_image " title="Share As PDF" (click)="exportAsPDF('MyDIv');"><img src="assets/img/pdf.png"></div>
    

    Step 4 : Write the code for generating PDF as follows

    exportAsPDF(div_id)
      {
        let data = document.getElementById(div_id);  
        html2canvas(data).then(canvas => {
          const contentDataURL = canvas.toDataURL('image/png')  
          let pdf = new jspdf('l', 'cm', 'a4'); //Generates PDF in landscape mode
          // let pdf = new jspdf('p', 'cm', 'a4'); Generates PDF in portrait mode
          pdf.addImage(contentDataURL, 'PNG', 0, 0, 29.7, 21.0);  
          pdf.save('Filename.pdf');   
        }); 
      }
    
    0 讨论(0)
提交回复
热议问题