ファイルプレビュー機能のフロントエンド実装

ファイルプレビュー機能のフロントエンド実装

要件: 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: このキーワードが使用されない場合、埋め込まれた閲覧コンテキストは別のオリジンからのものとして扱われ、同一オリジン ポリシー チェックに失敗します。この属性が使用される場合、現在のページと 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 [email protected]
<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 [email protected]版本,以下例子使用的是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_ignorant の使用 新人のブログ - CSDN ブログ
pdfjs 公式コード: 例 (mozilla.github.io)
上記のコードは理解できないことを確認できます。ローカルエリアの公式コード。そのほとんどは修正されています。

 

「上記に関する注意事項」

  • PDF 内のファイルは非同期で参照する必要があります。

  • PDF デモ ファイルは public/test2.pdf にあります。

  • 変換: [1, 0, 0, -1, 0, viewport.height]、変換で行列を使用

  • 次のページと前のページの両方の関数を再レンダリングする必要がある

おすすめ

転載: blog.csdn.net/qq_44848480/article/details/132269811