pdf.js實戰(zhàn)优炬,含水印、電子簽章解決方案

項目涉及到移動端查看電子合同的問題厅贪,前前后后試了三種方案蠢护,真是一步一個坑,三種方案各有各的優(yōu)點养涮,不水葵硕,直接上代碼,按照自己的需求選擇贯吓。

一懈凹、pdf-vue

直接使用vue-pdf插件,核心的代碼是pdf.js悄谐,只不過就是自己封裝了一下介评,優(yōu)點是方便快捷,缺點是無法加載電子簽章爬舰。

github地址: https://github.com/FranckFreiburger/vue-pdf#readme

1们陆、npm install pdf-vue --save

2寒瓦、template代碼

<template>
  <div class="pdf" v-show="fileType === 'pdf'">
    <p class="arrow">
    // 上一頁
    <span @click="changePdfPage(0)" class="turn" :class="{grey: currentPage==1}">Preview</span>
    {{currentPage}} / {{pageCount}}
    // 下一頁
    <span @click="changePdfPage(1)" class="turn" :class="{grey: currentPage==pageCount}">Next</span>
    </p>
    // 自己引入就可以使用,這里我的需求是做了分頁功能,如果不需要分頁功能,只要src就可以了
    <pdf
      :src="src" // src需要展示的PDF地址
      :page="currentPage" // 當前展示的PDF頁碼
      @num-pages="pageCount=$event" // PDF文件總頁碼
      @page-loaded="currentPage=$event" // 一開始加載的頁面
      @loaded="loadPdfHandler"> // 加載事件
    </pdf>
  </div>
</template>

3、js代碼

import pdf from 'vue-pdf'
export default {
    components: {pdf},
    data () {
      return {
        currentPage: 0, // pdf文件頁碼
        pageCount: 0, // pdf文件總頁數
        fileType: 'pdf', // 文件類型
        src: '', // pdf文件地址
      }
    },
  created: {
    // 有時PDF文件地址會出現跨域的情況,這里最好處理一下
    this.src = pdf.createLoadingTask(this.src)
  }
    method: {
      // 改變PDF頁碼,val傳過來區(qū)分上一頁下一頁的值,0上一頁,1下一頁
      changePdfPage (val) {
        // console.log(val)
        if (val === 0 && this.currentPage > 1) {
          this.currentPage--
          // console.log(this.currentPage)
        }
        if (val === 1 && this.currentPage < this.pageCount) {
          this.currentPage++
          // console.log(this.currentPage)
        }
      },
    
      // pdf加載時
      loadPdfHandler (e) {
        this.currentPage = 1 // 加載的時候先加載第一頁
      }
    
    }
}

使用非常方便棒掠,尤其是只需要翻頁孵构,或者不需要翻頁的,強烈推薦烟很。

二、pdf-dist

pdf-dist也是基于pdf.js的一個組件蜡镶,只不過沒有封裝雾袱,需要自己配置,優(yōu)點是可配置官还,可實現特殊的需求芹橡,缺點是需要自己封裝,水印可加載望伦,網上說可以加載電子簽章林说,我的加載不出來,所以還是沒采用屯伞。

1腿箩、npm install pdf-dist --save

2、封裝一個pdf.vue

<template>
  <div class="cpdf" id="cpdf">
    <div class="center">
      <canvas class="canvasstyle" id="the-canvas"></canvas>
      <div class="contor">
        <button @click="prev" style="margin-right: 10px">上一頁</button>
        <span>Page: <span v-text="page_num"></span> / <span v-text="page_count"></span></span>
        <button @click="next" style="margin-left: 10px">下一頁</button>
      </div>
    </div>
  </div>
</template>
<script>
import PDFJS from 'pdfjs-dist'

export default {
  name: 'c-pdf',
  // 接收父組件傳來的參數   
  props: ['pdfurl'],
  components: { },
  data () {
    return {
      pdfDoc: null, // pdfjs 生成的對象
      pageNum: 1, //
      pageRendering: false,
      pageNumPending: null,
      scale: 1, // 放大倍數
      page_num: 0, // 當前頁數
      page_count: 0, // 總頁數
      maxscale: 2, // 最大放大倍數
      minscale: 0.8// 最小放大倍數
    }
  },
  methods: {
    renderPage (num) { // 渲染pdf
      let vm = this
      this.pageRendering = true
      let canvas = document.getElementById('the-canvas')
      let ctx = canvas.getContext('2d')
      let bsr =
                ctx.webkitBackingStorePixelRatio ||
                ctx.mozBackingStorePixelRatio ||
                ctx.msBackingStorePixelRatio ||
                ctx.oBackingStorePixelRatio ||
                ctx.backingStorePixelRatio ||
                1
      let dpr = window.devicePixelRatio || 1
      let ratio = dpr / bsr
      // Using promise to fetch the page
      this.pdfDoc.getPage(num).then(function (page) {
        var viewport = page.getViewport(screen.availWidth / page.getViewport(1).width)
        // alert(vm.canvas.height)
        canvas.height = ratio * viewport.width
        canvas.width = ratio * viewport.height
        canvas.style.width = 1.5 * viewport.width + 'px'
        canvas.style.height = 1 * viewport.height + 'px'
        ctx.setTransform(ratio, 0, 0, ratio, 0, 0)
        // Render PDF page into canvas context
        var renderContext = {
          canvasContext: ctx,
          viewport: viewport
        }
        var renderTask = page.render(renderContext)

        // Wait for rendering to finish
        renderTask.promise.then(function () {
          vm.pageRendering = false
          if (vm.pageNumPending !== null) {
            // New page rendering is pending
            vm.renderPage(vm.pageNumPending)
            vm.pageNumPending = null
          }
        })
      })
      vm.page_num = vm.pageNum
    },
    addscale () { // 放大
      if (this.scale >= this.maxscale) {
        return
      }
      this.scale += 0.1
      this.queueRenderPage(this.pageNum)
    },
    minus () { // 縮小
      if (this.scale <= this.minscale) {
        return
      }
      this.scale -= 0.1
      this.queueRenderPage(this.pageNum)
    },
    prev () { // 上一頁
      let vm = this
      if (vm.pageNum <= 1) {
        return
      }
      vm.pageNum--
      vm.queueRenderPage(vm.pageNum)
    },
    next () { // 下一頁
      let vm = this
      if (vm.pageNum >= vm.page_count) {
        return
      }
      vm.pageNum++
      vm.queueRenderPage(vm.pageNum)
    },
    closepdf () { // 關閉PDF
      this.$emit('closepdf')
    },
    queueRenderPage (num) {
      if (this.pageRendering) {
        this.pageNumPending = num
      } else {
        this.renderPage(num)
      }
    }
  },
  computed: {
    ctx () {
      let id = document.getElementById('the-canvas')
      return id.getContext('2d')
    }
  },
  mounted () {
    let vm = this
    PDFJS.getDocument(vm.pdfurl).then(function (pdfDoc_) { // 初始化pdf
      vm.pdfDoc = pdfDoc_
      vm.page_count = vm.pdfDoc.numPages
      vm.renderPage(vm.pageNum)
    })
  }
}

</script>
<style  lang="stylus" scoped>
  .cpdf {
    display: flex;
    justify-content: center;
    align-items: center;
    .center {
      text-align: center;
      height: 100%;
      overflow: hidden;
      padding-top: 20px;
      .contor {
        position: fixed;
        bottom: 30px;
        left: 0;
        width: 100%;
        z-index: 99999;
        font-size 30px
        margin-top 20px
        margin-bottom: 10px;
      }
    }
  }
</style>

3劣摇、直接當成組件珠移,引用就可以了

import cdpdf from '../../../components/pdf.vue'

<cdpdf :pdfurl="pdfurl"></cdpdf>

一開始項目使用的是pdf-dist,因為后來電子簽章顯示不出來:

Warning: Unimplemented widget field type "Sig", falling back to base field type.

從網上搜了很多方法末融,說是需要修改pdf.work.js的源碼钧惧,全局搜索AnnotationFlag.HIDDEN:

if(data.fieldType==='Sig') {
    warn('unimplemented annotation type: Widget signature');
    // 注釋下面這行代碼
    this.setFlags(AnnotationFlag.HIDDEN);
 }

可能是移動端使用微信瀏覽器的原因,注釋掉代碼還是不好使勾习,只能再想其他辦法了

三浓瞪、pdf.js

最后用了最笨的辦法,直接從GitHub拉下來pdf.js的demo巧婶,用iframe標簽包住demo里的HTML文件乾颁,直接套著用,完美解決電子簽章的問題:

1粹舵、從GitHub拉一下源碼钮孵,或者從這個地址直接下載

https://mozilla.github.io/pdf.js/getting_started/#download

下載下來以后放在public文件下(3.x腳手架)

2、iframe標簽直接粗暴的設置src

<iframe :src="pdfUrl" :style="{height: Height}" style="width: 100%"></iframe>

this.pdfUrl = '../pdf/web/viewer.html?file=' + this.pdfurl, +'PDF'

pdfUrl是iframe標簽的URL眼滤,pdfurl是需要查看的PDF文件的url

總結

只要能實現需求的代碼就是好代碼巴席,我的項目是移動端查看PDF文件,因為文件上有電子簽章诅需,所以嘗試了好幾種方案漾唉,個人還是推薦第二種方案荧库,如果沒有電子簽章的情況下。

各位哥哥姐姐點個關注吧

gzh.jpg
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末赵刑,一起剝皮案震驚了整個濱河市分衫,隨后出現的幾起案子,更是在濱河造成了極大的恐慌般此,老刑警劉巖蚪战,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現場離奇詭異铐懊,居然都是意外死亡邀桑,警方通過查閱死者的電腦和手機,發(fā)現死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進店門科乎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來壁畸,“玉大人,你說我怎么就攤上這事茅茂∧笃迹” “怎么了?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵空闲,是天一觀的道長令杈。 經常有香客問我,道長进副,這世上最難降的妖魔是什么这揣? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮影斑,結果婚禮上给赞,老公的妹妹穿的比我還像新娘。我一直安慰自己矫户,他們只是感情好片迅,可當我...
    茶點故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著皆辽,像睡著了一般柑蛇。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上驱闷,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天耻台,我揣著相機與錄音,去河邊找鬼空另。 笑死盆耽,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播摄杂,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼坝咐,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了析恢?” 一聲冷哼從身側響起墨坚,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎映挂,沒想到半個月后泽篮,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡袖肥,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年咪辱,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片椎组。...
    茶點故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖历恐,靈堂內的尸體忽然破棺而出寸癌,到底是詐尸還是另有隱情,我是刑警寧澤弱贼,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布蒸苇,位于F島的核電站,受9級特大地震影響吮旅,放射性物質發(fā)生泄漏溪烤。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一庇勃、第九天 我趴在偏房一處隱蔽的房頂上張望檬嘀。 院中可真熱鬧,春花似錦责嚷、人聲如沸鸳兽。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽揍异。三九已至,卻和暖如春爆班,著一層夾襖步出監(jiān)牢的瞬間衷掷,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工柿菩, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留戚嗅,地道東北人。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像渡处,于是被迫代替她去往敵國和親镜悉。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,925評論 2 344