vue--ofd/pdf預覽實現

背景

實現預覽ofd/pdf超鏈接功能

業務實現

  1. pdf的預覽

    實現方式:

    1. 直接使用 <iframe :src="${url}#navpanes=0&toolbar=0" /> 實現pdf的預覽。

      • navpanes=0 隱藏側邊欄
      • toolbar=0 隱藏頂部工具欄
    2. 使用pdf.js,代碼先行:

      <template><a-tabsv-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"><canvasv-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) returnconst loadingTask = pdfjsLib.getDocument(pdfUrl)const pdf = await loadingTask.promiseconst page = await pdf.getPage(1)const viewport = page.getViewport({ scale: 1.3 })canvas.height = viewport.heightcanvas.width = viewport.widthconst context = canvas.getContext('2d')if (!context) returnconst renderContext = {canvasContext: context,viewport,}await page.render(renderContext).promise} catch (error) {console.error('PDF 渲染失敗:', error)}
      }//  標簽頁切換時加載 PDF
      async function tabChangeHandler(key: string) {url.value = keyactivateTab.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.pswp.cn/pingmian/82150.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/82150.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/82150.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

【C++20新特性】ranges::sort()使用方法,優勢,注意點

以下是關于 ranges::sort() 的詳細說明&#xff1a; 1. ranges::sort() 的使用方法 ranges::sort() 是 C20 引入的基于范圍&#xff08;Ranges&#xff09;的排序函數&#xff0c;其語法更簡潔&#xff0c;支持直接操作容器或范圍對象。 (1)基本用法 #include <vector&g…

深入理解設計模式之適配器模式

深入理解設計模式之適配器模式 1. 適配器模式概述 適配器模式(Adapter Pattern)是一種結構型設計模式&#xff0c;它允許將一個類的接口轉換為客戶端所期望的另一個接口。適配器模式使得原本由于接口不兼容而不能一起工作的類能夠協同工作&#xff0c;扮演了"轉換器&quo…

【數據結構 · 初階】- 快速排序

目錄 一. Hoare 版本 1. 單趟 2. 整體 3. 時間復雜度 4. 優化&#xff08;搶救一下&#xff09; 4.1 隨機選 key 4.2 三數取中 二. 挖坑法 格式優化 三. 前后指針&#xff08;最好&#xff09; 四. 小區間優化 五. 改非遞歸 快速排序是 Hoare 提出的一種基于二叉樹…

第2周 PINN核心技術揭秘: 如何用神經網絡求解偏微分方程

1. PDEs與傳統數值方法回顧 (Review of PDEs & Traditional Numerical Methods) 1.1 什么是偏微分方程 (Partial Differential Equations, PDEs)? 偏微分方程是描述自然界和工程領域中各種物理現象(如熱量傳播、流體流動、波的振動、電磁場分布等)的基本數學語言。 1.…

Neo4j(二) - 使用Cypher操作Neo4j

文章目錄 前言一、Cypher簡介二、數據庫操作1. 創建數據庫2. 查看數據庫3. 刪除數據庫4. 切換數據庫 三、節點、關系及屬性操作1. 創建節點與關系1.1 語法1.2 示例 2. 查詢數據2.1 語法2.2 示例 3. 更新數據3.1 語法3.2 示例 4. 刪除節點與關系4.1 語法4.2 示例 5. 合并數據5.1…

RabbitMQ的Web管理頁面給我看懵了,這都什么意思啊

文章目錄 OverviewTotalsMessage RatesQueued Messages NodesChurn StatisticsPorts and ContextsExport DefinitionsImport Definitions ConnectionsChannelsExchangesQueuesAdmin他們之間的關聯 在上一篇文章中我們講到了如何在Windows中安裝Rabbitmq&#xff0c; 小白也能搞…

安全基礎與協議分析

5.1 Web安全基礎 5.1.1 Web安全基礎概覽&#xff08;一、二&#xff09; Web安全的核心目標是保護Web應用及其數據免受攻擊&#xff0c;涵蓋以下關鍵領域&#xff1a; 攻擊面&#xff1a; 前端漏洞&#xff08;XSS、CSRF&#xff09;。 后端漏洞&#xff08;SQL注入、RCE&a…

STM32項目實戰:ADC采集

STM32F103C8T6的ADC配置。PB0對應的是ADC1的通道8。在標準庫中&#xff0c;需要初始化ADC&#xff0c;設置通道&#xff0c;時鐘&#xff0c;轉換模式等。需要配置GPIOB的第0腳為模擬輸入模式&#xff0c;然后配置ADC1的通道8&#xff0c;設置轉換周期和觸發方式。 接下來是I2C…

第十四章:數據治理之數據源:數據源的數據接入、業務屬性梳理及監控

本章開始&#xff0c;將進入9大模塊的介紹。第一個模塊我們先介紹&#xff1a;數據源。數據源是整個數據中臺數據的來源&#xff0c;是一個起點。更好的管理好數據源這個起點&#xff0c;是數據治理的一個好的開始。 在【數據&#xff1a;業務生數據&#xff0c;數據生“萬物”…

【C/C++】多線程開發:wait、sleep、yield全解析

文章目錄 多線程開發&#xff1a;wait、sleep、yield全解析1 What簡要介紹詳細介紹wait() — 條件等待&#xff08;用于線程同步&#xff09;sleep() — 睡覺&#xff0c;定時掛起yield() — 自愿讓出 CPU 2 區別以及建議區別應用場景建議 3 三者協作使用示例 多線程開發&#…

阿里云CDN刷新預熱--刷新URL

文章目錄 一、全英文URL刷新預熱二、摻雜中文的URL刷新預熱2.1 對帶中文URL進行編碼2.2 預熱刷新 三、CDN刷新-核心作用與價值3.1 核心作用3.2 核心價值3.3 典型使用場景 *最后我想說&#xff1a;請你不要相信我說的每一句話&#xff0c;這只是我的個人經驗* 一、全英文URL刷新…

Oracle 19c DG備庫報錯ORA-00313、ORA-00312、ORA-27037

Oracle 19c DG備庫報錯ORA-00313、ORA-00312、ORA-27037 錯誤排查問題處理錯誤排查 DG同步完成后,DG Broker show database發現以下告警信息: Database Warning(s):ORA-16826: apply service state is inconsistent with the DelayMins propertyORA-16789: standby redo log…

開源與閉源之爭:AI時代的創新博弈與未來抉擇

在人工智能技術狂飆突進的今天&#xff0c;開源與閉源之爭已不再局限于技術圈的討論&#xff0c;而是演變為一場關乎技術倫理、商業格局乃至人類文明走向的深度博弈。當Meta的Llama 3開源模型下載量突破百萬&#xff0c;當OpenAI的GPT-5繼續加固技術壁壘&#xff0c;這場沒有硝…

NIFI的處理器:JSLTTransformJSON 2.4.0

該處理器使用JSLT轉換FlowFile JSON有效負載的格式。使用轉換后的內容創建新的FlowFile&#xff0c;并將其路由到“成功”關系。如果JSLT轉換失敗&#xff0c;則將原始FlowFile路由到“失敗”關系。 需要注意的是&#xff0c;編譯JSLT轉換可能相當昂貴。理想情況下&#xff0c…

MySQL 索引失效及其解決辦法

一、前言 在數據庫優化中,索引(Index)是一項至關重要的技術手段,可以顯著提升查詢性能。然而,在實際開發過程中,MySQL 索引并不總是如預期生效。本文將從原理出發,系統地介紹索引失效的常見場景及其解決方案,幫助開發者有效規避性能陷阱。 二、索引基礎回顧 MySQL 支…

趨勢觸發策略

趨勢觸發策略(TS版)是一種基于TrendTriggerFactor(TTF)的交易策略,通過柱狀圖顏色變化指示市場趨勢的強度,并根據TTF的穿越信號進行買賣操作。 TTF是通過計算買方力量和賣方力量的差值除以兩者之和的一半再乘以100得到的。 當TTF大于100時,柱狀圖顯示為綠色,表示市場…

DeepSeek-R1 模型現已在亞馬遜云科技上推出

亞馬遜云科技提供眾多免費云產品&#xff0c;可以訪問&#xff1a;亞馬遜云科技 在剛剛過去的 Amazon re&#xff1a;Invent 期間&#xff0c;Amazon 首席執行官 Andy Jassy 分享了從 Amazon 自己在全公司開發近 1000 個生成式 AI 應用程序的經驗中汲取的寶貴經驗。從這種廣泛…

中臺項目-微前端qiankun-umimax

學習視頻&#x1f50a; 基礎&#xff1a; 黑馬前端基于qiankun搭建微前端項目實戰教程_嗶哩嗶哩_bilibili 路由、部署配置注意&#xff1a;qiankunvite微前端上線注意事項&#xff0c;base公共路徑設置_嗶哩嗶哩_bilibili 微前端 什么是微前端&#xff1f; 微前端是將前端應…

【Java學習筆記】代碼塊

代碼塊 介紹&#xff1a;代碼塊又稱為初始化塊&#xff0c;屬于類中的成員&#xff08;即是類的一部分&#xff09;&#xff0c;類似于方法&#xff0c;將邏輯語句封裝在方法體中&#xff0c;通過{}包圍起來 與類方法的不同點 無方法名 無返回類型 無參數 只有方法體&#…

spring boot 2.7集成舊的springfox-boot-starter swagger oas 3.0

舊版本目前已經不維護推薦使用 springdoc-openapi-ui&#xff0c;這里為了演示使用舊的最新依賴 <dependency><groupId>io.springfox</groupId><artifactId>springfox-boot-starter</artifactId><version>3.0.0</version> </dep…