整合营销服务商

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

免费咨询热线:

PDF.JS与FlexPaper在线显示文档

PDF.JS与FlexPaper在线显示文档

、PDF.JS

用于在浏览器中直接显示 PDF 文档,它是免费开源的 JavaScript,由Mozilla维护。显示方式:
> 方法1:直接获到pdfviewer,复制到文件夹中,在要显示pdf文件网页内用iframe嵌入即可。

pdfviewer

嵌入iframe:“pdfviewer/web/viewer.html?file=pdf文件名.pdf”。

iframe嵌入pdf文件名

> 方法2:也是iframe嵌入pdfview,与方法1区别是,方法1直接暴露pdf文件名,可以间接下载pdf文件,方法2用Blob(Binary Large Object)嵌入。

步骤如下:后台程序把pdf文件内容转换成base64编码传到前端网页;

转base64编码

前端界面读取base64编码内容;

读取base64

base64转二进制;

转二进制

iframe嵌入,src为blob地址;

iframe嵌入blob地址


> 方法3:引用pdf.js和pdf.worker.js,操作pdf对象var loadingTask=PDFJS.getDocument(path).promise;

pdf对象

二、FlexPaper

FlexPaper基于flash对象,有些浏览器不再支持flash对象,故FlexPaper对有些浏览器在线显示文档不工作。FlexPaper是一个开源轻量级的文档浏览组件,目的是在网页上,基于Flash的应用程序能在浏览器中显示PDF文件。FlexPaper项目同时提供了Flex库和独立的Flash版本。

步骤如下:

1、将需要展现的文件如word、ppt转换成pdf;

2、SWFTools工具包中的pdf2swf将pdf格式文件转换成swf格式文件;

pdf转swf

3、在前端网页中引用flexpaper_flash.js;

flexpaper_flash.js

4、前端网页用FlexPaperViewer显示swf文件,其中SwfFile就是生成的flash文件格式swf文件;

方法1,直接写JS

方法2:用swfobject嵌入

了在你的HMTL5 网站上查看PDF 文件,你要将它们嵌入到HTML5 PDF 查看器中。像Chrome这样的浏览器已经有一个插件可以做到这一点,因此,你的PDF 文件将正确显示给使用Chrome浏览器访问你网站的人。不过,还有使用插件的替代方法。例如,AdobeInDesign 将帮助你构建交互式pdf。还有其他提示和技巧可用于呈现你的PDF文件,以便每个人都可以查看它们。想了解更多的查看方法,可以考虑报个HMTL5培训班,有专业导师面授教学,课程紧随企业需求,培养全栈开发人才。

获取HTML5 PDF 查看器

网络上有许多HTML5 PDF 查看器。例如,这些网站都可以提供在你的网站上使用的PDF查看器,并且所有浏览器都应该能够查看你的文件:IDR解决方案、PDF项目、软纸项目、非常PDF。

你上传文件并选择HTML5,然后输出你需要嵌入网页的代码。当有人访问你的网页时,他们将能够在查看器中查看你的PDF 文件,而不是弹出链接。这些大多数PDF查看器网站也有出售软件,这样你就可以在计算机上拥有它的副本,而不必去他们的网站在线使用它。拥有该软件的副本可以保护你的PDF副本(如果它受版权保护),并且通常还会为你提供更好的软件版本。如果你想从事HTML5开发,又不知道从何学起,HMTL5培训学习就是一个不错的选择,不只是学习理论知识,还有实践项目的操作训练,让学生学以致用,提升学员全局性思维和全栈技术能力。

将PDF 文件转换为HTML5

你可以将PDF 文件转换为HTML5,以便每个人都能很好地查看你的文件。这样,你就不必担心别人看不到文件,因为他们的浏览器没有插件来查看它,或者你的插件不适用于他们的计算机或浏览器。所以有人会看不到你的文件。

HTML5– 新方式

除了大多数设计师、Apple和Adobe 使用HTML5来创建他们的东西,使用HTML5 的另一个好处是新改进的页面标题。你可以使用一些有意义的东西,如

移动设备是在网站上使用HTML5的一个重要原因。现在每个人都用手机,会产生巨大的访问量,如果你的网站要接触到这些移动用户,那么你就要使用HTML5编写内容。

有很多方法可以在你的网站上完成HTML5 PDF 查看器。你可以嵌入代码,也可以使用插件。还有一些网站提供免费的在线代码生成器,为你编写代码,这样你的PDF文件就会像杂志一样显示,用户可以直接在你的网站上阅读。如果你想了解有关HTML5的更多信息,可以参加HTML5培训班学习,能够快速掌握更系统全面的HTML5知识,让你在短时间内学有所成。

了解更多

何保持页面样式基本不变的前提下将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;
}

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