一、引入插件
vue引入:npm install --save pdfjs
下载pdfjs:http://mozilla.github.io/pdf.js/getting_started/
方式一:只引入pdf.js的核心文件pdf.js和pdf.work.js,其他无关的文件全部删除,如图:
在这里插入图片描述
方式二:将插件直接放在static文件夹下,如图:
在这里插入图片描述
二、前端页面代码
方式一和方式二:特点精简

<template>
  <div>
    <canvas v-for="page in pages" :id="'the-canvas'+page" :key="page"></canvas>
  </div>
</template>
 
<script>
// 方式一
import PDFJS from 'pdfjs-dist'
// 方式二
import * as PDFJS from '../../../static/pdf/build/pdf'
 
export default {
  // 返回数据
  data () {
    return {
      pdfDoc: null,
      pages: 0
    }
  },
  created () {
  },
  mounted () {
    this.showPdf()
  },
  methods: {
    showPdf: function () {
      // 请求本地文件
      let url = '/static/pdf/web/compressed.tracemonkey-pldi-09.pdf'
      // 跨域请求文件,需要走后台代理,后台需要将文件流返回前端才可在页面显示
      // let url = '/pdf/showPdf?pdfUrl=http://test.hccb.cc/corporBankWXTest/static/123.pdf'
      this.loadFile(url)
    },
    renderPage: function (num) {
      let _this = this
      this.pdfDoc.getPage(num).then(function (page) {
        let canvas = document.getElementById('the-canvas' + num)
        let ctx = canvas.getContext('2d')
        let dpr = window.devicePixelRatio || 1.0
        let bsr = ctx.webkitBackingStorePixelRatio ||
          ctx.mozBackingStorePixelRatio ||
          ctx.msBackingStorePixelRatio ||
          ctx.oBackingStorePixelRatio ||
          ctx.backingStorePixelRatio || 1.0
        let ratio = dpr / bsr
        let viewport = page.getViewport(window.screen.availWidth / page.getViewport(1).width)
        canvas.width = viewport.width * ratio
        canvas.height = viewport.height * ratio
        canvas.style.width = viewport.width + 'px'
        canvas.style.height = viewport.height + 'px'
        ctx.setTransform(ratio, 0, 0, ratio, 0, 0)
        var renderContext = {
          canvasContext: ctx,
          viewport: viewport
        }
        page.render(renderContext)
        if (_this.pages > num) {
          _this.renderPage(num + 1)
        }
      })
    },
    loadFile: function (url) {
      let _this = this
      PDFJS.getDocument(url).then(function (pdf) {
        _this.pdfDoc = pdf
        _this.pages = _this.pdfDoc.numPages
        _this.$nextTick(() => {
          _this.renderPage(1)
        })
      })
    }
  }
}
</script>
 
<style scoped>
canvas {
  display: block;
  border-bottom: 1px solid black;
}
</style>

方式三:功能强大,但是引入过多无用文件,此种方式的filePath如为本地文件不进行编码也可发送请求,如为跨域文件不进行编码无法发送请求,因此建议统一进行编码。

<template>
  <div >
    <iframe :src="url" id="iframe" style="width: 100%;" @load="sureHeight"></iframe>
  </div>
</template>
 
<script>
export default {
  // 返回数据
  data () {
    return {
      url: ''
    }
  },
  // 模块创建时执行
  created () {
  },
  // 模块渲染时执行
  mounted () {
    // 本地请求文件
    let filePath = encodeURIComponent('/static/pdf/web/compressed.tracemonkey-pldi-09.pdf')
    // 跨域请求文件,需走后台代理
    // let filePath2 = encodeURIComponent('/pdf/showPdf?pdfUrl=http://test.hccb.cc/corporBankWXTest/static/123.pdf')
    // pdf文档展示的页面
    this.url = '/static/pdf/web/viewer.html?file=' + filePath
  },
  // 定义模块测试方法
  methods: {
    // 此方法用于动态确定元素iframe的高度,使展示的pdf文档占满整个屏幕
    sureHeight: function () {
      let element = document.getElementById('iframe')
      element.style.height = window.screen.height + 'px'
    }
  }
}
</script>
 
<style scoped>
 
</style>

三、后台代码实现
后台通过http请求将获取的文档流返回给前端

@Controller
public class ShowPdfController {
    @RequestMapping(name = "/showPdf")
    public String showPdf(HttpServletRequest request, HttpServletResponse response, String pdfUrl) {
        try {
            pdfUrl = pdfUrl.trim();
            URL url = new URL(pdfUrl);
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setConnectTimeout(5*1000);
            InputStream inputStream = conn.getInputStream();
            response.setHeader("Content-Disposition", "attachment;fileName=show.pdf");
            response.setContentType("multipart/form-data");
            OutputStream outputStream = response.getOutputStream();
            IOUtils.write(IOUtils.toByteArray(inputStream), outputStream);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
}

具体采用哪种方式实现pdf文档的在线预览,可根据项目实际情况选择,如业务简单建议使用方式一和方式二(精简),如业务复杂建议使用方式三(功能强大)。

下面列举出现的问题:
1、路径问题:vue-cli3之后的项目没有单独的static文件夹,只有public。
如果自带static文件夹或者你自己创建一个,静态资源访问路径为:/static/pdf/web/XX.pdf
如果使用高版本的public文件夹,则不用加/public,直接从下一个路径写起,例如:
在这里插入图片描述

/static/pdf/web/安徽XX公司能效诊断报告.pdf

2、view.html是pdf浏览插件,如果报错:在这里插入图片描述
则文件路径不对。
3、如果报错意外的服务器响应,我的是IDM这个插件的问题,它拦截了服务器请求,关掉IDM的获取Chrome站点关闭即可。
在这里插入图片描述
还有什么问题可以评论区提问喔。

Logo

前往低代码交流专区

更多推荐