整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:

前端页面如何原样快速导出PDF?附示例

何保持页面样式基本不变的前提下将HTML页面导出为PDF,下面提供一些示例代码,纯属个人原创,如对你有帮助请记得加关注、加收藏、点赞、转发、分享~谢谢~~

  • 基本思路:保持页面样式基本不变,使用 `html2canvas` 将页面转换为图片,然后再通过 `jspdf` 将图片分页导出为PDF文件(中间会遇到图片或文字等内容在分页处被切割开的问题,如何解决了?详见末尾干货)
  • 上基础代码:下面为项目中实际代码截取
<div>
    <!-- 要打印的内容区 -->
    <div ref="contentRef">
        <div class="print-item print-out-flow">这是脱离文档流的内容区域</div>
        <div class="print-item">这是一行内容,也是最小叶子元素内容</div>
    </div>
    <!-- 打印内容容器 -->
    <div ref="printContainerRef" class="print-container"></div>
</div>
/**
  * 1.使用一个隐藏div装载有滚动条的div.innerHTML
  * 2.隐藏div使用position: absolute, z-index: -999, left: -9999px, width: 900px 控制让用户无感知
  * 3.根据需要覆写隐藏div内html样式(例如textarea多行显示有问题, 可以新增一个隐藏的div
  *    包裹textarea的绑定值, 然后在打印样式中覆写样式, 隐藏textarea并显示对应div)
  */
handleExport() {
   // 下面是VUE组件内获取DOM元素代码,将内容放置到打印区(定义的隐藏DIV)中
    const contentRef = this.$refs.contentRef as HTMLElement;
    const printContainerRef = this.$refs.printContainerRef as HTMLElement;
    // 打印区的需额外处理绝对定位值, 调整使得第一个元素的.top值为0, 以便于页面计算
    printContainerRef.innerHTML = contentRef.innerHTML;	
    
    // 所有叶子div元素加上 print-item 样式名, 脱离文档流的额外添加 print-out-flow
    handlePrintItem(printContainerRef);  // 解决多页内容可能被切割问题
    
    html2canvas(printContainerRef, {allowTaint: false, useCORS: true}).then((canvas: any) => {
      const contentHeight = canvas.height;
      const contentWidth = canvas.width;
      // pdf每页显示的内容高度
      const pageHeight = contentWidth / 595.28 * 841.89;
      // 未生成pdf的页面高度
      let offsetHeight = contentHeight;
      // 页面偏移值
      let position = 0;
      // a4纸的尺寸[595.28, 841.89], canvas图片按a4纸大小缩放后的宽高
      const imgWidth = 595.28;
      const imgHeight = 595.28 / contentWidth * contentHeight;

      const dataURL = canvas.toDataURL('image/jpeg', 1.0);
      const doc = new jsPDF('p', 'pt', 'a4');

      if (offsetHeight < pageHeight) {
        doc.addImage(dataURL, 'JPEG', 0, 0, imgWidth, imgHeight);
      } else {
        while (offsetHeight > 0) {
          doc.addImage(dataURL, 'JPEG', 0, position, imgWidth, imgHeight);
          offsetHeight -= pageHeight;
          position -= 841.89;

          if (offsetHeight > 0) {
            doc.addPage();
          }
        }
      }

      doc.save(this.generateReportFileName());
      printContainerRef.innerHTML = '';
    });
}

上干货代码:上面分页导出PDF可能网上能看到类型代码,但绝对找不到下面的代码,纯手搓解决分页元素被切开问题(思路:获取自身定位,如自己刚好在被分页处,则加上一定的margin-top值将内容向下移)

/** 
 * 处理打印元素项, 修复分页后被切割的元素
 * @param printContainerRef 打印内容div容器
 * @param itemClassName 打印最小元素标识类名
 * @param outFlowClassName 脱离文档流的元素标识类名
 */
export function handlePrintItem(
  printContainerRef: HTMLElement,
  itemClassName: string = 'print-item',
  outFlowClassName: string = 'print-out-flow'
): void {
  const rootClientRect = printContainerRef.getBoundingClientRect();
  // 初始化页面相关数据
  const totalHeight = rootClientRect.height;  // 内容总高度
  const a4PageHeight = (printContainerRef.clientWidth / 595.28) * 841.89; // a4纸高度
  let pageNum = Math.ceil(totalHeight / a4PageHeight);  // 总页数
  let addPageHeight = 0;  // 修正被分割元素而增加的页面高度总和
  let currentPage = 1;  // 当前正在处理切割的页面
  const splitItemObj: { [key: number]: HTMLElement[] } = {};  // 内容中各页被切割元素存储对象

  const printItemNodes: NodeListOf<HTMLElement> = printContainerRef.querySelectorAll(`.${itemClassName}`);
  for (let item of printItemNodes) {
    // 如果当前页已经是最后一页, 则中断判断
    if (currentPage >= pageNum) {
      break;
    }

    // 获取元素绝对定位数据
    const clientRect = item.getBoundingClientRect();
    let top = clientRect.top;
    const selfHeight = clientRect.height;
    // 如果当前元素距离顶部高度大于当前页面页脚高度, 则开始判断下一页页脚被切割元素
    if (top > currentPage * a4PageHeight) {
      // 换页前修正上一页被切割元素
      addPageHeight += fixSplitItems(currentPage, a4PageHeight, splitItemObj[currentPage], outFlowClassName);
      pageNum = Math.ceil((totalHeight + addPageHeight) / a4PageHeight);
      top = item.getBoundingClientRect().top;
      currentPage++;
    }
    // 如果元素刚好处于两页之间, 则记录该元素
    if (top > (currentPage - 1) * a4PageHeight && top < currentPage * a4PageHeight && top + selfHeight > currentPage * a4PageHeight) {
      if (!splitItemObj[currentPage]) {
        splitItemObj[currentPage] = [];
      }
      splitItemObj[currentPage].unshift(item);
      // 如果当前元素是最后一个元素, 则直接处理切割元素, 否则交由处理下一页元素时再处理切割
      if (item === printItemNodes[printItemNodes.length - 1]) {
        fixSplitItems(currentPage, a4PageHeight, splitItemObj[currentPage], outFlowClassName);
      }
    }
  }
}

/**
  * 修复当前页所有被切割元素
  * @param currentPage 当前页
  * @param pageHeight 每页高度
  * @param splitElementItems 当前被切割元素数组
  * @param outFlowClassName 脱离文档流的样式类名
  */
function fixSplitItems(
  currentPage: number,
  pageHeight: number,
  splitElementItems: HTMLElement[],
  outFlowClassName: string
): number {
  if (!splitElementItems || !splitElementItems.length) {
    return 0;
  }

  const yMargin = 5;  // y方向距离页眉的距离
  const splitItemsMinTop = getSplitItemsMinTop(splitElementItems);
  if (!splitItemsMinTop) {
    return 0;
  }

  let fixHeight = currentPage * pageHeight - splitItemsMinTop + yMargin;
  const outFlowElement = splitElementItems.find((item) => item.classList.contains(outFlowClassName));
  if (outFlowElement && outFlowElement.parentElement) {
    const parentPreviousElement = outFlowElement.parentElement.previousElementSibling as HTMLElement;
    fixHeight += getMarinTopNum(parentPreviousElement, outFlowElement.parentElement);
    outFlowElement.parentElement.style.marginTop = `${fixHeight}px`;
    return fixHeight;
  }

  splitElementItems.forEach((splitElement) => {
    splitElement.style.marginTop = `${fixHeight}px`;
  });
  return fixHeight;
}

/**
  * 获取被切割元素数组中最小高度值(如一行有多个元素被切割,则选出距离顶部最小的高度值)
  * @param splitElementItems 当前被切割元素数组
  */
function getSplitItemsMinTop(
  splitElementItems: HTMLElement[]
): number | undefined {
  // 获取元素中最小top值作为基准进行修正
  let minTop: number | undefined;
  let minElement: HTMLElement | undefined;
  splitElementItems.forEach((splitElement) => {
    let top = splitElement.getBoundingClientRect().top;
    if (minTop) {
      minTop = top < minTop ? top : minTop;
      minElement = top < minTop ? splitElement : minElement;
    } else {
      minTop = top;
      minElement = splitElement;
    }
  });

  // 修正当前节点及其前面同层级节点的margin值
  if (minTop && minElement) {
    const previousElement = splitElementItems[splitElementItems.length - 1].previousElementSibling as HTMLElement;
    minTop -= getMarinTopNum(previousElement, minElement);
  }
  return minTop;
}

/**
  * 通过前一个兄弟元素和元素自身的位置确认一个距离顶部高度修正值
  * @param previousElement 前一个兄弟元素
  * @param curElement 当前元素
  */
function getMarinTopNum(previousElement: HTMLElement, curElement: HTMLElement): number {
  let preMarginNum = 0;
  let curMarginNum = 0;
  if (previousElement) {
    // 获取外联样式需要getComputedStyle(), 直接.style时对象的值都为空
    const previousMarginBottom = window.getComputedStyle(previousElement).marginBottom;
    preMarginNum = previousMarginBottom ? Number(previousMarginBottom.replace('px', '')) : 0;
  }
  const marginTop = window.getComputedStyle(curElement).marginTop;
  curMarginNum = marginTop ? Number(marginTop.replace('px', '')) : 0;
  return preMarginNum > curMarginNum ? preMarginNum : curMarginNum;
}

以上纯原创!欢迎加关注、加收藏、点赞、转发、分享(代码闲聊站)~

位:1. 静态定位 2.相对定位 3.绝对定位

1.值:static 默认的标准流;

2.值:relative 相对自身进行定位,没有脱离标准流;

3.值:absolute 父元素

a.没有父元素的时候,根据body进行;

b.有父元素但是父元素没有定位,根据body定位;

c.有父元素并且父元素有定位(非静态),根据父元素进行定位;

d.脱离标准流的一种技术;

子绝父相

浮动也是脱离标准流

思考:什么时候用定位?什么使用用浮动?

首选浮动进行模块分区

一些出现叠压效果的时候,或者特殊效果的时候才用定位!

(一)定位

如何让小盒子在大盒子中既定位并且水平居中?

1.先left: 50%,将小盒子在大盒子平移大盒子的一半,

2. 设置margin-left:-(小盒子宽度的一半),注意一定是负数。

2.4 fixed:固定定位

background-attachment:fixed

position:fixed

使用的时候也要配合TRBL属性来使用

特点:

1.不管页面有多大,trbl永远是相对于浏览器的边框来的。

2.固定定位的元素也脱离了标准流(不在页面上占据位置)

(二)图文混排

3.1 图文对齐

文本与图片在同一行中,文字与图片的默认对齐方式是文字的基线对齐图片的底线。

vertical-align:设置文本与图片的对齐方式 top middle bottom

middle:将中线对齐

图片:margin-top的时候,图片和文字会同时下移

pà display:inline-block

图文绕拍

图片 float : left

第一种拆分形式:

Border: border-width | border-style | border-color

Border-width: 1px

Border-top-width:1px

第二种拆分形式

Border-top: border-width | border-style | border-color

路径查找方式:

1.确定HTML页为基准;

2.向下查找 images/images/ 01.jpg

3.向上查找

background-postion: x y;

1.当直接加载的时候,图片的左上角与容器的左上角是(0,0);

2.X为正向右偏移 Y为正向下偏移 ,负数反之;

3.英文

4.数值 + 英文的方式

*行高是具有继承性的*

(一)盒子模型

概念: 在html页面,其实就相当于在这个页面中摆盒子。

margin(外边距)--盒子与盒子之间的间隙

作用:设置盒子与盒子之间的距离。

margin: 10px;

上下左右

margin: 10px 10px;

上下,左右

margin: 10 px 20px 10px;

上 左右 下

margin:10px 10px 20px 10px;

上右下左

浮动的特点

标准流与浮动:

标准流就是浏览器按顺序摆放的一种方式

主流网页设计的布局方式:

a.框架式网站 两栏式 三栏式

b.通栏式布局

c.响应式网页 (国内并不多)bootstrap

d.流式布局

更多内容关注大牛哥教育

  • tml的主体结构

  • 标签的分类

  • 标签的关系

  • sublime快捷键

  • 单标签

  • 双标签

  • 路径

  • 超链接额外知识运用

    • 锚点

    • 空连接

    • 超链接的优化写法 写在head标签中

  • 特殊标记符

  • 列表


Html的主体结构

<!doctype html><html><head>

标签的分类

单标签

<!doctype html>

双标签

<head></head>

标签的关系

包含(嵌套关系、父子关系)

<head>

并列

<head></head><body></body>

sublime快捷键

快捷键作用
html:xt + tabhtml4.01模板【新版本可能失效】
html + tabhtml5模板
tab补全标签
ctrl + shift + d快速复制一行
ctrl + shift + k快速删除一行
ctrl + 鼠标左键单击集体输入
ctrl + h查找替换
ctrl + f查找
ctrl + /注释
ctrl + L快速选中当前行
ctrl + shift + ↑(↓)代码的快速上移和下移

单标签

<!-- --> 注释标签<br /> 换行标签<hr /> 水平线标签<img src="logo.gif" alt="logo" title="这是淘宝的logo" width="200" height="100" />图片标签

双标签

<p></p> 段落标签<h1></h1> 标题标签 h1 - h6<font></font> 文本标签<strong></strong> 文本加粗标签,有语音加强<b></b> 文本加粗标签<em></em> 文字倾斜,有语音加强<i></i> 文字倾斜<del></del> 删除线,有语音加强<s></s> 删除线<ins></ins> 下划线,有语音加强<u></u> 下划线<a href="http://www.baidu.com" title="百度" target="_blank">百度</a> 超链接

路径

相对路径

同一个目录下直接写文件的名称就可以

文件和图片在下一级目录中,需要文件夹名称 + \ + 文件名称

图片在上一级目录中 ../ + 文件名称

图片在上一级的其它文件夹中 ../ + 文件夹名称 + 文件名称

总结:找到下级目录使用/,找到上一级目录使用../

绝对路径

从盘符中开始的,这种路径不能使用!因为项目最后都要移动,文件的路径都是会变的。如:c:\baidu\logo.jpg

超链接额外知识运用

锚点

1.设置一个锚点 设置一个id

<div id="top"></div>

2.超链接到锚点

<a href="#top"></a>

空连接

<a href="#"></a>

超链接的优化写法 写在head标签中

<base targer="_blank">

让所有的超链接都是从新窗口中打开

特殊标记符

空格&nbsp;
<&lt;
>&gt;
©&copy;

更多信息请查阅

列表

无序列表

<ul type="disc">

类型:disc 默认小黑点 circle 空心小圆点 square 小方块

有序列表

<ol type="A" start="C">

类型:a A 字母顺序 i I罗马顺序 1数字 start 表示开始的位置

自定义列表

<dl>