vue3项目中使用pdfjs-dist踩坑记录

有这么一个需求,要求打开pdf文件实现预览,分页跳转、搜索关键字。期望效果如下:
在这里插入图片描述
经过一番调研选择使用pdfjs-dist来完成这一功能。

第一种方案:使用npm在项目中安装pdfjs-dist

npm install pdfjs-dist

执行安装文件报错了,应该是node版本不兼容,到https://www.npmjs.com/上找往期版本,查了一些评论,评价2.16.105版本兼容性好,安装试一下。

npm i pdfjs-dist@2.16.105

在代码中引入

<template>
  <Modal
    v-model="visible"
    fullscreen
    footer-hide
    @on-cancel="cancel"
  >
    <div id="pdf-view">
      <canvas v-for="page in state.pdfPages" :key="page" id="pdfCanvas" />
      <div id="text-view"></div>
    </div>
  </Modal>
</template>
<script>
    import * as pdfjsViewer from 'pdfjs-dist/web/pdf_viewer.js'
    import 'pdfjs-dist/web/pdf_viewer.css'
    import * as PDF from 'pdfjs-dist'
    PDF.GlobalWorkerOptions.workerSrc = '/pdf.worker.js'
    const pdf = 'http://www.nhc.gov.cn/sps/s7887k/202301/0e55a01df50c47d9a4a43db026e3afc3/files/63f752a17cfd4b4781f744477561866f.pdf'
    let pdfDoc = null;
    export default {
        data () {
            return {
                state: {
                    // 文件路径
                    pdfPath: pdf,
                    // 总页数
                    pdfPages: 1, 
                    // 页面缩放
                    pdfScale: 1, 
                },
                visible: false,

            }
        },
        mounted () {
            // this.loadFile(this.state.pdfPath)
        },
        methods: {
            init () {
                this.visible = true
                this.loadFile(this.state.pdfPath)
            },
            cancel () {
                this.visible = false
            },
            loadFile(url) {
                PDF.getDocument({
                    url,
                    cMapUrl: 'https://cdn.jsdelivr.net/npm/pdfjs-dist@2.16.105/cmaps/',
                    cMapPacked: true,
                }).promise.then((pdf) => {
                    pdfDoc = pdf
                    // 获取pdf文件总页数
                    this.state.pdfPages = pdf.numPages
                    this.$nextTick(() => {
                        this.renderPage(1) // 从第一页开始渲染
                    })
                })
            },
            renderPage(num) {
                pdfDoc.getPage(num).then((page) => {
                    const canvas = document.getElementById('pdfCanvas')
                    const ctx = canvas.getContext('2d')
                    const viewport = page.getViewport({ scale: this.state.pdfScale })
                    canvas.width = viewport.width
                    canvas.height = viewport.height
                    const renderContext = {
                        canvasContext: ctx,
                        viewport
                    }
                    page.render(renderContext)
                })
            }
        }
    }
</script>

引入的过程中发生了报错
在这里插入图片描述
原因是需要在引入 pdf is-dist 之后配置 workerSrc,但是引入 pdf is-dist/bui1d/pdf. worker.entry之后浏览器还是有个警告:Warning;Settingup fake worker.,几方查找原来是pdf.workerjs必须位于自己的文件中(而不是与pdfjs捆绑在起)。否则它不能在服务工作线程中运行。
解决方式:将 pdfis-dist/bui1d/pdf.worker.is 复制一份放到项目 public 目录下

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

能够预览但是不是想要的效果,貌似这种只能实现pdf预览功能,而我想要的是官方演示的那种带各种操作的效果。

第二种方法:直接将预构建版本放入public文件夹下使用viewer.html

在这里插入图片描述
下载,解压到public文件夹下。
在这里插入图片描述
使用

// 直接在新窗口中打开
window.open(`/pdfjs-2.5.207-dist/web/viewer.html?file=${pdfurl}`, '_blank')

在这里插入图片描述
找到pdfjs-2.5.207-dist/web/wiewer.js
在这里插入图片描述
效果如下,有个bug,打开pdf文件以后,分页数字会被缓存,下次再打开,会停留在这个页面,如果想回到第1页,需要做一下处理。
在这里插入图片描述
在这里插入图片描述

上面这个方法只能禁用本地对跨域的检查,如果引用的pdf文件地址是外链,依然会报跨域问题,如果对pdf的来源不能确定是不太好解决的。
在这里插入图片描述
上面这种跨域限制,就需要后端介入处理了。
如果产品只是要预览pdf文件,可以用第一种方法或者在新窗口打开pdf文件查看。

window.open(pdfurl, '_blank')

在浏览器的新窗口打开查看pdf文件,没有跨域问题,遗憾的是显示效果不能控制。先记到这儿吧,后期如果有更好的解决跨域方案再来更新。

参考链接

Vue 3和TypeScript可以使用pdfjs-dist库来预览PDF文件。下面是一个简单的示例代码,演示如何在Vue 3和TypeScript使用pdfjs-dist库进行PDF预览: 首先,你需要安装pdfjs-dist库。可以通过npm或者yarn命令来安装: ```shell npm install pdfjs-dist ``` 或者 ```shell yarn add pdfjs-dist ``` 接下来,在你的Vue组件引入pdfjs-dist库并创建一个用于显示PDF内容的容器元素: ```html <template> <div> <canvas ref="pdfCanvas"></canvas> </div> </template> ``` 在组件的`mounted`生命周期钩子,你可以使用pdfjs-dist来加载PDF文件并在Canvas上渲染它: ```typescript import { ref, onMounted } from 'vue'; import * as pdfjs from 'pdfjs-dist'; export default { setup() { const pdfCanvas = ref<HTMLCanvasElement | null>(null); let loadingTask: any; onMounted(async () => { try { const pdf = await pdfjs.getDocument('path/to/your/pdf'); // 替换为你的PDF文件路径 if (pdf) { const page = await pdf.getPage(1); // 获取第一页(页码从0开始) if (page) { const viewport = page.getViewport({ scale: 1 }); const canvas = document.createElement('canvas'); canvas.width = viewport.width; canvas.height = viewport.height; const context = canvas.getContext('2d'); page.render({ canvasContext: context, viewport }).then(() => { pdfCanvas.value = canvas; // 将渲染后的Canvas赋值给组件的容器元素 }); } else { console.error('Failed to render PDF page'); } } else { console.error('Failed to load PDF document'); } } catch (error) { console.error('Error rendering PDF:', error); } finally { if (loadingTask) { loadingTask.cancel(); // 确保在组件销毁时取消加载任务以避免内存泄漏 } } }); return { pdfCanvas }; }, }; ``` 以上代码会在Vue组件创建一个Canvas元素,并在该元素上渲染PDF的第一页。你可以根据需要修改代码以适应不同的PDF文件和页面范围。记得将`'path/to/your/pdf'`替换为你实际的PDF文件路径。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

前端李易安

打赏1元鼓励作者

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值