Newer
Older
DH_Apicture / src / utils / exportPdf.js
@zhangqy zhangqy on 29 Nov 4 KB first commit
  1. // 页面导出为pdf格式以及分页隔断处理
  2. //title表示为下载的标题,html表示document.querySelector('#myPrintHtml')
  3. import html2Canvas from 'html2canvas';
  4. import JsPDF from 'jspdf';
  5.  
  6. let noTableHeight = 0; //table外的元素高度
  7. export function pageToPDF(title, html, lableList, type) {
  8. // type传有效值pdf则为横版
  9. if (lableList) {
  10. const pageHeight = Math.floor((277 * html.scrollWidth) / 190) + 20; //计算pdf高度
  11. for (let i = 0; i < lableList.length; i++) {
  12. //循环获取的元素
  13. const multiple = Math.ceil((lableList[i].offsetTop + lableList[i].offsetHeight) / pageHeight); //元素的高度
  14. if (isSplit(lableList, i, multiple * pageHeight)) {
  15. //计算是否超出一页
  16. var _H = ''; //向pdf插入空白块的内容高度
  17. if (lableList[i].localName !== 'tr') {
  18. //判断是不是表格里的内容
  19. _H = multiple * pageHeight - (lableList[i].offsetTop + lableList[i].offsetHeight);
  20. } else {
  21. _H = multiple * pageHeight - (lableList[i].offsetTop + lableList[i].offsetHeight + noTableHeight) + 20;
  22. }
  23. var newNode = getFooterElement(_H); //向pdf插入空白块的内容
  24. const divParent = lableList[i].parentNode; // 获取该div的父节点
  25. const next = lableList[i].nextSibling; // 获取div的下一个兄弟节点
  26. // 判断兄弟节点是否存在
  27. if (next) {
  28. // 存在则将新节点插入到div的下一个兄弟节点之前,即div之后
  29. divParent.insertBefore(newNode, next);
  30. } else {
  31. // 否则向节点添加最后一个子节点
  32. divParent.appendChild(newNode);
  33. }
  34. }
  35. }
  36. }
  37. html2Canvas(html, {
  38. allowTaint: false,
  39. taintTest: false,
  40. logging: false,
  41. useCORS: true,
  42. dpi: window.devicePixelRatio * 1,
  43. scale: 1, // 按比例增加分辨率
  44. }).then(canvas => {
  45. var pdf = new JsPDF('p', 'mm', 'a4'); // A4纸,纵向
  46. var ctx = canvas.getContext('2d');
  47. var a4w = type ? 277 : 190;
  48. var a4h = type ? 190 : 277; // A4大小,210mm x 297mm,四边各保留10mm的边距,显示区域190x277
  49. var imgHeight = Math.floor((a4h * canvas.width) / a4w); // 按A4显示比例换算一页图像的像素高度
  50. var renderedHeight = 0;
  51. while (renderedHeight < canvas.height) {
  52. var page = document.createElement('canvas');
  53. page.width = canvas.width;
  54. page.height = Math.min(imgHeight, canvas.height - renderedHeight); // 可能内容不足一页
  55.  
  56. // 用getImageData剪裁指定区域,并画到前面创建的canvas对象中
  57. page
  58. .getContext('2d')
  59. .putImageData(ctx.getImageData(0, renderedHeight, canvas.width, Math.min(imgHeight, canvas.height - renderedHeight)), 0, 0);
  60. pdf.addImage(page.toDataURL('image/jpeg', 1.0), 'JPEG', 10, 10, a4w, Math.min(a4h, (a4w * page.height) / page.width)); // 添加图像到页面,保留10mm边距
  61.  
  62. renderedHeight += imgHeight;
  63. if (renderedHeight < canvas.height) {
  64. pdf.addPage(); // 如果后面还有内容,添加一个空页
  65. }
  66. // delete page;
  67. }
  68. // 保存文件
  69. pdf.save(title + '.pdf');
  70. });
  71. }
  72. // pdf截断需要一个空白位置来补充
  73. function getFooterElement(remainingHeight, fillingHeight = 0) {
  74. const newNode = document.createElement('div');
  75. newNode.style.background = '#ffffff';
  76. newNode.style.width = 'calc(100% + 8px)';
  77. newNode.style.marginLeft = '-4px';
  78. newNode.style.marginBottom = '0px';
  79. newNode.classList.add('divRemove');
  80. newNode.style.height = remainingHeight + fillingHeight + 'px';
  81. return newNode;
  82. }
  83. function isSplit(nodes, index, pageHeight) {
  84. // 判断是不是tr 如果不是高度存起来
  85. // 表格里的内容要特殊处理
  86. // tr.offsetTop 是tr到table表格的高度
  87. // 所以计算高速时候要把表格外的高度加起来
  88. // 生成的pdf没有表格了这里可以不做处理 直接计算就行
  89. if (nodes[index].localName !== 'tr') {
  90. //判断元素是不是tr
  91. noTableHeight += nodes[index].clientHeight;
  92. }
  93.  
  94. if (nodes[index].localName !== 'tr') {
  95. return (
  96. nodes[index].offsetTop + nodes[index].offsetHeight < pageHeight &&
  97. nodes[index + 1] &&
  98. nodes[index + 1].offsetTop + nodes[index + 1].offsetHeight > pageHeight
  99. );
  100. } else {
  101. return (
  102. nodes[index].offsetTop + nodes[index].offsetHeight + noTableHeight < pageHeight &&
  103. nodes[index + 1] &&
  104. nodes[index + 1].offsetTop + nodes[index + 1].offsetHeight + noTableHeight > pageHeight
  105. );
  106. }
  107. }