vue--ofd/pdf预览实现

news2025/5/25 21:01:49

背景

实现预览ofd/pdf超链接功能

业务实现

  1. pdf的预览

    实现方式:

    1. 直接使用 <iframe :src="${url}#navpanes=0&toolbar=0" /> 实现pdf的预览。

      • navpanes=0 隐藏侧边栏
      • toolbar=0 隐藏顶部工具栏
    2. 使用pdf.js,代码先行:

      <template>
        <a-tabs
          v-if="props.urls.length > 0"
          :default-active-key="activateTab"
          type="card"
          class="pdf-tabs"
          @change="tabChangeHandler"
        >
          <a-tab-pane v-for="url in props.urls" :key="url" :tab="fileName(url)">
            <div class="pdf-container">
              <canvas
                v-if="url.endsWith('.pdf')"
                class="canvas"
                :ref="(el) => (canvasRefs[url] = el)"
              ></canvas>
              <a-button class="mb-2" type="link" @click="handleDownload(url)">
                {{ fileName(url) }}
              </a-button>
            </div>
          </a-tab-pane>
        </a-tabs>
      </template>
      
      <script lang="ts" setup>
      import { ref, watch, nextTick } from 'vue'
      import * as pdfjsLib from 'pdfjs-dist'
      import { debounce } from 'lodash-es'
      import { saveAs } from 'file-saver'
      import EasyOFD from 'easyofd'
      
      interface Props {
        urls?: string[]
      }
      
      const props = withDefaults(defineProps<Props>(), {
        urls: () => [],
      })
      const url = ref<string>('')
      const activateTab = ref<string>('')
      const canvasRefs = ref<Record<string, HTMLCanvasElement | null>>({})
      
      // 文件类型判断
      const ext = ref<string>('pdf')
      const isOfd = ref<boolean>(false)
      const isPdf = ref<boolean>(false)
      
      //  设置 PDF.js worker 路径(推荐方式)
      pdfjsLib.GlobalWorkerOptions.workerSrc = '/pdf.worker.min.mjs'
      
      //  从 URL 中提取文件名
      function fileName(url: string): string {
        try {
          const decodeURL = decodeURIComponent(url).split('/')
          const lastSegment = decodeURL[decodeURL.length - 1]
      
          const firstIndex = lastSegment.indexOf('-')
          const lastIndex = lastSegment.lastIndexOf('-')
      
          if (firstIndex === -1 || lastIndex === -1 || lastIndex <= firstIndex) {
            return lastSegment.split('.')[0] // fallback 文件名
          }
      
          const name = lastSegment.substring(firstIndex + 1, lastIndex)
          const ext = name.split('.').pop()
      
          if (['pdf', 'ofd'].includes(ext ?? '')) {
            return name.substring(0, name.lastIndexOf('.'))
          }
      
          return name
        } catch {
          return 'unknown'
        }
      }
      
      // 获取文件类型
      const getFileType = (url: string) => {
        const decodeURL = decodeURIComponent(url)
        ext.value = decodeURL.endsWith('.pdf') ? 'pdf' : 'ofd'
        isPdf.value = ext.value === 'pdf'
        isOfd.value = ext.value === 'ofd'
        isPdf.value ? loadAndRenderPdf(url) : loadAndRenderOfd(url)
      }
      //  下载文件
      const handleDownload = debounce((url: string) => {
        saveAs(url, `${fileName(url)}.${ext.value}`)
      }, 300)
      
      //  加载并渲染 PDF
      async function loadAndRenderPdf(pdfUrl: string) {
        try {
          const canvas = canvasRefs.value[pdfUrl]
          if (!canvas) return
      
          const loadingTask = pdfjsLib.getDocument(pdfUrl)
          const pdf = await loadingTask.promise
          const page = await pdf.getPage(1)
      
          const viewport = page.getViewport({ scale: 1.3 })
      
          canvas.height = viewport.height
          canvas.width = viewport.width
      
          const context = canvas.getContext('2d')
          if (!context) return
      
          const renderContext = {
            canvasContext: context,
            viewport,
          }
      
          await page.render(renderContext).promise
        } catch (error) {
          console.error('PDF 渲染失败:', error)
        }
      }
      
      //  标签页切换时加载 PDF
      async function tabChangeHandler(key: string) {
        url.value = key
        activateTab.value = fileName(key)
      
        await nextTick() // 等待 DOM 更新
        if (key.endsWith('.pdf')) {
          await loadAndRenderPdf(key)
        }
      }
      
      //  页面初始化时自动加载第一个 PDF
      watch(
        () => props.urls,
        async (newUrls) => {
          if (newUrls && newUrls.length > 0) {
            console.log('newUrls:', newUrls)
            url.value = newUrls[0]
            activateTab.value = fileName(newUrls[0])
            await nextTick()
            getFileType(newUrls[0])
          }
        },
        { immediate: true },
      )
      </script>
      
      <style lang="less" scoped>
      .canvas {
        border: 1px solid #000;
        width: 100%; // 响应式宽度
        border: 1px solid #000;
      }
      .pdf-container {
        display: flex;
        flex-direction: column;
        align-items: start;
        gap: 12px;
        max-width: 100%; // 限制最大宽度
        max-height: 400px;
        overflow: auto;
      }
      </style>
      

      说一下重点:

      问题一: 通过命令pnpm install pdf.js安装后,通常出现引用问题;Cannot resolve pdf.worker.entry。代码中使用的版本"pdfjs-dist": "^5.2.133"

      import * as pdfjsLib from 'pdfjs-dist';
      import pdfjsWorker from 'pdfjs-dist/build/pdf.worker.entry';
      
      pdfjsLib.GlobalWorkerOptions.workerSrc = pdfjsWorker;
      

      解决方案:

      1. 将文件从node_modules/pdfjs-dist/build/pdf.worker.min.mjs移动至项目的public/pdf.worker.min.mjs,可以使用命令 cp node_modules/pdfjs-dist/build/pdf.worker.min.mjs public/pdf.worker.min.mjs
      2. 修改引用:
        import * as pdfjsLib from 'pdfjs-dist';
        
        pdfjsLib.GlobalWorkerOptions.workerSrc = '/pdf.worker.min.mjs'
        

      ps: 上面的代码中包含了文件的下载功能,需要安装 "file-saver": "^2.0.5",

  2. ofd的预览

    实现方式:easyofd
    安装的依赖:pnpm -i jszip x2js jb2 opentype.js easyofd
    业务实现:

    <template>
      <div ref="containerRef" style="width: 100%; height: 800px;"></div>
    </template>
    
    <script setup>
    import EasyOFD from "easyofd"
    import { ref, onMounted } from 'vue'
    
    const containerRef = ref(null)
    
    onMounted(async () => {
      if (!containerRef.value) {
        console.error('OFD 容器不存在')
        return
      }
    
      const ofd = new EasyOFD('myOFD', containerRef.value)
    
      try {
        const response = await fetch('/files/sample.ofd')
        const blob = await response.blob()
        ofd.loadFromBlob(blob)
      } catch (e) {
        console.error('OFD 加载失败:', e)
      }
    })
    </script>
    	
    <style lang="less" scoped>
    // 隐藏右侧的ppi模块,减少空白
    :deep(#myOFD-ppi) {
      display: none;
    }
    // 增加边框
    :deep(#myOFD-ofd-canvas) {
      border: 1px solid #000;
    }
    // 隐藏顶部按钮
    :deep(.OfdButton) {
      display: none !important;
    }
    </style>
    

官网效果:(easyOfd官网手册)
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2385607.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python 爬虫之requests 模块的应用

requests 是用 python 语言编写的一个开源的HTTP库&#xff0c;可以通过 requests 库编写 python 代码发送网络请求&#xff0c;其简单易用&#xff0c;是编写爬虫程序时必知必会的一个模块。 requests 模块的作用 发送网络请求&#xff0c;获取响应数据。 中文文档&#xf…

【MySQL】CRUD

CRUD 简介 CRUD是对数据库中的记录进行基本的增删改查操作 Create&#xff08;创建&#xff09;Retrieve&#xff08;读取&#xff09;Update&#xff08;更新&#xff09;Delete&#xff08;删除&#xff09; 一、新增&#xff08;Create&#xff09; 语法&#xff1a; I…

Spring Boot微服务架构(三):Spring Initializr创建CRM项目

使用Spring Initializr创建CRM项目 一、创建项目前的准备 访问Spring Initializr网站&#xff1a; 打开浏览器访问 https://start.spring.io/或者直接使用IDE&#xff08;如IntelliJ IDEA或Eclipse&#xff09;内置的Spring Initializr功能 项目基本信息配置&#xff1a; Proj…

【笔记】PyCharm 中创建Poetry解释器

#工作记录 在使用 PyCharm 进行 Python 项目开发时&#xff0c;为项目配置合适的 Python 解释器至关重要。Poetry 作为一款强大的依赖管理和打包工具&#xff0c;能帮助我们更便捷地管理项目的依赖项与虚拟环境。下面将详细记录在 PyCharm 中创建 Poetry 解释器的步骤。 前提条…

python中的numpy(数组)

&#xff08;0&#xff09;numpy介绍 NumPy是Python中用于科学计算的基础库&#xff0c;提供高效的多维数组对象ndarray&#xff0c;支持向量化运算&#xff0c;能大幅提高数值计算效率。它集成了大量数学函数&#xff08;如线性代数、傅里叶变换等&#xff09;&#xff0c;可…

rce命令执行原理及靶场实战(详细)

2. 原理 在根源上应用系统从设计上要给用户提供一个指定的远程命令操作的接口。漏洞主要出现在常见的路由器、防火墙、入侵检测等设备的web管理界面上。在管理界面提供了一个ping服务。提交后&#xff0c;系统对该IP进行ping&#xff0c;并且返回结果。如果后台服务器并没有对…

Fuzz 模糊测试篇JS 算法口令隐藏参数盲 Payload未知文件目录

1 、 Fuzz 是一种基于黑盒的自动化软件模糊测试技术 , 简单的说一种懒惰且暴力的技术融合了常见 的以及精心构建的数据文本进行网站、软件安全性测试。 2 、 Fuzz 的核心思想 : 口令 Fuzz( 弱口令 ) 目录 Fuzz( 漏洞点 ) 参数 Fuzz( 利用参数 ) PayloadFuzz(Bypass)…

展示了一个三轴(X, Y, Z)坐标系!

等轴测投影”&#xff08;isometric projection&#xff09;风格的手绘风格三维图&#xff0c;即三条坐标轴&#xff08;x₁, x₂, x₃&#xff09;看起来彼此垂直、等角分布&#xff08;通常是 120 夹角&#xff09;&#xff0c;它是常见于教材和数学书籍的 “假三维”表示法。…

【b站计算机拓荒者】【2025】微信小程序开发教程 - chapter1 初识小程序 - 3项目目录结构4快速上手

3 项目目录结构 3.1 项目目录结构 3.1.1 目录介绍 # 1 项目主配置文件&#xff0c;在项目根路径下&#xff0c;控制整个项目的-app.js # 小程序入口文件&#xff0c;小程序启动&#xff0c;会执行此js-app.json # 小程序全局配置文件&#xff0c;配置小程序导航栏颜色等信息…

LLM Tuning

Lora-Tuning 什么是Lora微调&#xff1f; LoRA&#xff08;Low-Rank Adaptation&#xff09; 是一种参数高效微调方法&#xff08;PEFT, Parameter-Efficient Fine-Tuning&#xff09;&#xff0c;它通过引入低秩矩阵到预训练模型的权重变换中&#xff0c;实现无需大规模修改…

云计算与大数据进阶 | 28、存储系统如何突破容量天花板?可扩展架构的核心技术与实践—— 分布式、弹性扩展、高可用的底层逻辑(下)

在上篇中&#xff0c;我们围绕存储系统可扩展架构详细探讨了基础技术原理与典型实践。然而&#xff0c;在实际应用场景中&#xff0c;存储系统面临的挑战远不止于此。随着数据规模呈指数级增长&#xff0c;业务需求日益复杂多变&#xff0c;存储系统还需不断优化升级&#xff0…

水利数据采集MCU水资源的智能守护者

水利数据采集仪MCU&#xff0c;堪称水资源的智能守护者&#xff0c;其重要性不言而喻。在水利工程建设和水资源管理领域&#xff0c;MCU数据采集仪扮演着不可或缺的角色。它通过高精度的传感器和先进的微控制器技术&#xff0c;实时监测和采集水流量、水位、水质等关键数据&…

origin绘图之【如何将横坐标/x设置为文字、字母形式】

在使用 Origin 进行科研绘图或数据可视化的过程中&#xff0c;我们常常会遇到这样一种需求&#xff1a;希望将横坐标&#xff08;X轴&#xff09;由默认的数字形式&#xff0c;改为字母&#xff08;如 A、B、C……&#xff09;或中文文字&#xff08;如 一、二、三……&#xf…

工业智能网关建立烤漆设备故障预警及远程诊断系统

一、项目背景 烤漆房是汽车、机械、家具等工业领域广泛应用的设备&#xff0c;主要用于产品的表面涂装。传统的烤漆房控制柜采用本地控制方式&#xff0c;操作人员需在现场进行参数设置和设备控制&#xff0c;且存在设备智能化程度低、数据孤岛、设备维护成本高以及依靠传统人…

Kafka Streams 和 Apache Flink 的无状态流处理与有状态流处理

Kafka Streams 和 Apache Flink 与数据库和数据湖相比的无状态和有状态流处理的概念和优势。 在数据驱动的应用中&#xff0c;流处理的兴起改变了我们处理和操作数据的方式。虽然传统数据库、数据湖和数据仓库对于许多基于批处理的用例来说非常有效&#xff0c;但在要求低延迟…

LM-BFF——语言模型微调新范式

gpt3&#xff08;GPT3——少样本示例推动下的通用语言模型雏形)结合提示词和少样本示例后&#xff0c;展示出了强大性能。但大语言模型的训练门槛太高&#xff0c;普通研究人员无力&#xff0c;LM-BFF(Making Pre-trained Language Models Better Few-shot Learners)的作者受gp…

NVMe高速传输之摆脱XDMA设计2

NVMe IP放弃XDMA原因 选用XDMA做NVMe IP的关键传输模块&#xff0c;可以加速IP的设计&#xff0c;但是XDMA对于开发者来说&#xff0c;还是不方便&#xff0c;原因是它就象一个黑匣子&#xff0c;调试也非一番周折&#xff0c;尤其是后面PCIe4.0升级。 因此决定直接采用PCIe设…

pycharm无需科学上网工具下载插件的解决方案

以下是两种无需科学上网即可下载 PyCharm 插件的解决思路&#xff1a; 方法 1&#xff1a;设置 PyCharm 代理 打开 PyCharm选择菜单&#xff1a;File → Settings → Appearance & Behavior → System Settings → HTTP Proxy在代理设置中进行如下配置&#xff1a; 代理地…

Halcon计算点到平面的距离没有那么简单

Halcon计算点到平面距离 1. 一些基本概念2. 浅谈有无符号的距离2.1 无符号距离的用武之地2.2 有符号距离的必要性 3. 无符号距离怎么算3.1 创建一个无限延展的基准平面&#xff0c;对距离有什么影响&#xff1f;Halcon代码图示 3.2 创建一个小小小的基准平面&#xff0c;对距离…

数据中台如何设计?中台开发技术方案,数据治理方案,大数据建设方案合集

中台的价值与核心理念 中台的核心在于“企业级能力复用”&#xff0c;其价值体现在四大维度&#xff1a; 能力整合&#xff1a;将分散的数字化能力&#xff08;如营销、供应链&#xff09;集中管理&#xff0c;形成核心竞争力&#xff1b; 业务创新&#xff1a;通过跨领域融合…