前端预览和打印PDF的两种方式

最近工作中遇到了一个需求,就是前端选择表格中的某一条数据去请求后端接口,后端返回的是一个PDF文件的下载地址,但是需求不希望用户下载下来再去打印,而是直接预览展示,然后就能打印。

一开始按照网上的方式去操作,但是每一次浏览器直接就下载了下来,后面一看响应头,原来后端的Content-type设置成了如图所示,直接触发了浏览器的下载。

怎么办呢?于是就想到了以下两种方案,两种方案都是再一次去请求返回的地址,只不过前端改成了用二进制Blob(responseType: 'blob'),获取原始二进制数据

方案一如下图所示:

其中的data为要传入的地址,然后将拿到的结果创建一个Blob对象,并重新指定其MIME类型为{ type: 'application/pdf '},这样就不会再触发浏览器的下载功能。接着使用URL.createObjectURL()生成临时访问地址,通过window.open()的方式在新窗口打开预览,等待加载后自动打印。

但是这种方案要确保后端设置了如下的CORS头,这种方案兼容所有现代浏览器(Chrome/Firefox/Edge/Safari)

复制代码
Access-Control-Allow-Origin: *

Access-Control-Allow-Methods: GET

方案二如下图所示:

创建一个隐藏的iframe标签src设置为pdf地址;前端通过获取隐藏的iframe标签的id来实现打印指定内容;

附上具体代码:

复制代码
 async batchPrintReturn(data) {
      axios.get(data,{responseType: 'blob'}).then(res => { // 以二进制Blob格式接收
        console.log('res>', res)
        const blob = new Blob([res.data], { type: 'application/pdf' }); // 创建Blob对象并指定新的MIME类型
        // 方案一
        const objectUrl = URL.createObjectURL(blob); // 生成临时URL
        const previewWindow = window.open(objectUrl, '_blank'); // 打开新窗口预览
        previewWindow.onload = function() { // 等待加载后自动打印
          previewWindow.print();
        };
        // 方案二
        var date = (new Date()).getTime()
        var ifr = document.createElement('iframe')
        ifr.style.frameborder = 'no'
        ifr.style.display = 'none'
        ifr.style.pageBreakBefore = 'always'
        ifr.setAttribute('id', 'printPdf' + date)
        ifr.setAttribute('name', 'printPdf' + date)
        ifr.src = window.URL.createObjectURL(blob)
        document.body.appendChild(ifr)
        this.doPrint('printPdf' + date)
        window.URL.revokeObjectURL(ifr.src) // 释放URL 对象
      })
    },
    doPrint(val) {
      var ordonnance = document.getElementById(val).contentWindow
      setTimeout(() => {
        ordonnance.print()
      }, 100)
    },

获取原始二进制数据

相关推荐
军军君0115 分钟前
Three.js基础功能学习十五:智能黑板实现实例二
开发语言·前端·javascript·vue.js·3d·threejs·三维
四千岁38 分钟前
Ollama+OpenWebUI 最佳组合:本地大模型可视化交互方案
前端·javascript·后端
写不来代码的草莓熊40 分钟前
el-date-picker ,自定义输入数字自动转换显示yyyy-mm-dd HH:mm:ss格式
前端·javascript·vue.js
Wect1 小时前
JS手撕:手写Koa中间件与Promise核心特性
前端·javascript·面试
张元清1 小时前
React 文件处理:上传、拖放区与对象 URL
前端·javascript·面试
煜bart1 小时前
使用 TypeScript 实现算法处理
开发语言·前端·javascript
Cobyte2 小时前
4.响应式系统基础:从发布订阅模式的角度理解 Vue3 的数据响应式原理
前端·javascript·vue.js
晓得迷路了2 小时前
栗子前端技术周刊第 124 期 - ESLint v10.2.0、React Native 0.85、Node.js 25.9.0...
前端·javascript·eslint
星空椰2 小时前
JavaScript基础:运算符和流程控制
开发语言·javascript·ecmascript
窝子面3 小时前
NestJs+MongoDB+Deepseek+Langchain实现ai聊天助手
javascript·数据库·人工智能·mongodb