前端實現文件預覽(pdf、excel、word、圖片)
前端實現文件預覽功能
需求:實現一個在線預覽 pdf、excel、word、圖片等文件的功能。
介紹:支持 pdf、xlsx、docx、jpg、png、jpeg。
以下使用 Vue3 代碼實現所有功能,建議以下的預覽文件標籤可以在外層包裹一層彈窗。
圖片預覽
iframe 標籤能夠將另一個 HTML 頁面嵌入到當前頁面中,我們的圖片也能夠使用 iframe 標籤來進行展示。
<iframe :src="圖片地址"
style="z-index: 1000; height:650px; width: 100%; margin: 0 auto"
sandbox="allow-scripts allow-top-navigation allow-same-origin allow-popups"
>
**「sandbox」**這個屬性如果是單純預覽圖片可以不使用,該屬性對呈現在 iframe 框架中的內容啓用一些額外的限制條件。屬性值可以爲空字符串(這種情況下會啓用所有限制),也可以是用空格分隔的一系列指定的字符串。
-
allow-scripts
: 允許嵌入的瀏覽上下文運行腳本(但不能創建彈窗)。如果沒有使用該關鍵字,就無法運行腳本。 -
allow-top-navigation
: 允許將框架內所加載頁面中的超鏈接導航到父級窗口 -
allow-same-popups
: 允許彈窗 (例如 window.open, target="_blank")。如果沒有使用該關鍵字,相應的功能將自動被禁用。 -
allow-same-origin
: 如果沒有使用該關鍵字,嵌入的瀏覽上下文將被視爲來自一個獨立的源,這將使 same-origin policy 同源檢查失敗。使用了這個屬性,那麼當前頁面和 iframe 打開的頁面視爲同源。
word 文檔預覽(docx)
先下載npm包
npm i docx-preview --save
<div class="docxRef"></div>
<script>
import { renderAsync } from 'docx-preview';
function fn() {
// 這裏的res.data是 blob文件流,如果自己的不是blob文件流
// 可以通過URL.createObjectURL(參數) 參數爲File格式,轉換爲blob文件流
let blob = res.data
let childRef = document.getElementsByClassName('docxRef');
renderAsync(blob, childRef[0]) //渲染
}
fn()
</script>
「blob 文件流」
預覽 excel 文件 (xlsx)
下載包
npm install xlsx@0.16.0
<div class="xlsxClass"></div>
const reader = new FileReader();
//通過readAsArrayBuffer將blob轉換爲ArrayBuffer對
reader.readAsArrayBuffer(res.data) // 這裏的res.data是blob文件流
reader.onload = (event) => {
// 讀取ArrayBuffer數據變成Uint8Array
var data = new Uint8Array(event.target.result);
// 這裏的data裏面的類型和後面的type類型要對應
var workbook = XLSX.read(data, { type: "array" });
var sheetNames = workbook.SheetNames; // 工作表名稱
var worksheet = workbook.Sheets[sheetNames[0]];
// var excelData = XLSX.utils.sheet_to_json(worksheet); //JSON
let html = XLSX.utils.sheet_to_html(worksheet);
document.getElementsByClassName('xlsxClass')[0].innerHTML = html
};
pdf 預覽
下載包 npm install pdfjs-dist
我使用的是npm install pdfjs-dist@2.0.943版本,以下例子使用的是vue3+vite創建的項目
以下例子通過canvas來渲染pdf
<template>
<div class="box">
<div class="tool-bar">
<div>{{ pdfParams.pageNumber }} / {{ pdfParams.total }}</div>
<button type="primary" :disabled="pdfParams.pageNumber == pdfParams.total" @click="nextPage">下一頁
</button>
<button type="primary" :disabled="pdfParams.pageNumber == 1" @click="prevPage">上一頁</button>
</div>
<canvas id="pdf-render"></canvas>
</div>
</template>
<script setup>
import { onMounted, ref, reactive } from 'vue'
const pdfParams = reactive({
pageNumber: 1, // 當前頁
total: 0, // 總頁數
});
// 不要定義爲ref或reactive格式,就定義爲普通的變量
let pdfDoc = null;
// 這裏必須使用異步去引用pdf文件,直接去import會報錯,也不知道爲什麼
onMounted(async ()=> {
let pdfjs = await import('pdfjs-dist/build/pdf')
let pdfjsWorker = await import('pdfjs-dist/build/pdf.worker.entry')
pdfjs.GlobalWorkerOptions.workerSrc = pdfjsWorker
// 此文件位於public/test2.pdf
let url = ref('/test2.pdf')
pdfjs.getDocument(url.value).promise.then(doc => {
pdfDoc = doc
pdfParams.total = doc.numPages
getPdfPage(1)
})
})
// 加載pdf的某一頁
const getPdfPage = (number) => {
pdfDoc.getPage(number).then(page => {
const viewport = page.getViewport()
const canvas = document.getElementById('pdf-render')
const context = canvas.getContext('2d')
canvas.width = viewport.viewBox[2]
canvas.height = viewport.viewBox[3]
viewport.width = viewport.viewBox[2]
viewport.height = viewport.viewBox[3]
canvas.style.width = Math.floor(viewport.width) + 'px'
canvas.style.height = Math.floor(viewport.height) + 'px'
let renderContext = {
canvasContext: context,
viewport: viewport,
// 這裏transform的六個參數,使用的是transform中的Matrix(矩陣)
transform: [1, 0, 0, -1, 0, viewport.height]
}
// 進行渲染
page.render(renderContext)
})
}
// 下一頁功能
const prevPage = () => {
if(pdfParams.pageNumber > 1) {
pdfParams.pageNumber -= 1
} else {
pdfParams.pageNumber = 1
}
getPdfPage(pdfParams.pageNumber)
}
// 上一頁功能
const nextPage = () => {
if(pdfParams.pageNumber < pdfParams.total) {
pdfParams.pageNumber += 1
} else {
pdfParams.pageNumber = pdfParams.total
}
getPdfPage(pdfParams.pageNumber)
}
</script>
以上 pdf 代碼引用文章:(54 條消息) 前端 pdf 預覽、pdfjs 的使用_pdf.js_無知的小菜雞的博客 - CSDN 博客
pdfjs 官方代碼:例子 (mozilla.github.io)
以上代碼看不懂的地方可以查閱官方代碼,大部分都是固定的寫法。
-
必須異步引用 pdf 的文件!!!
-
pdf 演示文件位於 public/test2.pdf
-
transform: [1, 0, 0, -1, 0, viewport.height], 使用了 transform 中的 Matrix(矩陣)
-
下一頁和上一頁功能都需要重新渲染
本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源:https://mp.weixin.qq.com/s/MFfhkuNy0vbPlZ_mzx3qEg