欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue項(xiàng)目pdf(base64)轉(zhuǎn)圖片遇到的問題及解決方法

 更新時(shí)間:2018年10月19日 17:20:55   投稿:mrr  
這篇文章主要介紹了Vue項(xiàng)目pdf(base64)轉(zhuǎn)圖片遇到的問題及解決方法,需要的朋友可以參考下

公司有個(gè)業(yè)務(wù)需求,要求后臺(tái)傳pdf的base64編碼給前端,前端顯示到界面上,后來在網(wǎng)上搜索了很多關(guān)于base64轉(zhuǎn)pdf的文章,都寫的不是非常的詳細(xì),在實(shí)現(xiàn)的過程中遇到很多坑,經(jīng)過一天的研究終于實(shí)現(xiàn)了這個(gè)功能,分享一下我在這個(gè)功能中遇到的問題和解決方法

要注明的是這里用到的核心插件是pdf.js,原理是動(dòng)態(tài)生成canvas標(biāo)簽,然后通過pdf.js生成一個(gè)能渲染出pdf的對(duì)象,隨后渲染每個(gè)canvas,并且生成的pdf是畫面的形式,并沒有pdf之類的控件

引入插件

這里很多博客都是使用JavaScript原生的方法引入pdf.js,例如使用script標(biāo)簽引入外部的js腳本,或者直接把pdf.js的源碼復(fù)制到項(xiàng)目中,但是我嘗試這些方法的時(shí)候都不是特別好用,而且引入后導(dǎo)致項(xiàng)目體積過于龐大,

隨后我去github上尋找通過包管理器引入pdf.js的方法,在pdf.js的github上官方說明的是用gulp如何使用pdf.js,但是對(duì)于npm來說并沒有詳細(xì)說明,終于我在字里行間發(fā)現(xiàn)了這么一句話

To use PDF.js in a web application you can choose to use a pre-built version of the library or to build it from source. We supply pre-built versions for usage with NPM and Bower under the pdfjs-dist name. For more information and examples please refer to the wiki page on this subject.

大致的意思就是如果使用npm包管理器或者bower的話,引入的名字為pdfjs-dist,那么我們使用npm的方法引入這個(gè)pdfjs-dist,引入的名字就隨意取名了這里我叫PDFJS

 import PDFJS from 'pdfjs-dist'

使用pdfjs-dist

這里后臺(tái)傳給我的是一個(gè)由pdf文件名字和pdf的base64編碼組成的對(duì)象的數(shù)組,我取名為pdfDataList

可以看到fileName是pdf的名字,fileVale是pdf文件的base64編碼,thumbnail是pdf縮略圖的base64編碼這里用不到先不管,之前說到需要?jiǎng)討B(tài)生成canvas節(jié)點(diǎn)(這里不會(huì)canvas也不要緊,只需要根據(jù)代碼一步步做就能渲染canvas)

1.首先我們創(chuàng)建一個(gè)承載所有canvas節(jié)點(diǎn)的父節(jié)點(diǎn),取名為pdfList

2.然后創(chuàng)建一個(gè)異步函數(shù)showPdf(不懂什么是異步函數(shù)的可以去查一下async/await,這里不用異步函數(shù)也可以使用promise.then的方法,但是async/await作為異步操作的終極方案最好還是學(xué)習(xí)一下)

async showPdf() {   
  }

使用querySelector選擇類名為pdfList的dom節(jié)點(diǎn),隨后遍歷后臺(tái)傳過來的pdfDataList數(shù)組的每一項(xiàng),這里用到一個(gè)瀏覽器自帶的atob()方法解碼base64,MDN上是這么解釋的:

你可以使用 window.btoa() 方法來編碼一個(gè)可能在傳輸過程中出現(xiàn)問題的數(shù)據(jù),并且在接受數(shù)據(jù)之后,使用 atob() 方法再將數(shù)據(jù)解碼。

語法: var decodedData = scope.atob(encodedData);

隨后調(diào)用pdf.js插件的getDocument方法,getDocument是一個(gè)promise,所以使用異步函數(shù)的話前面需要加await關(guān)鍵字(不使用異步函數(shù)的話在方法后面加.then((pdf)=>{.......}),這個(gè)pdf對(duì)象和我這個(gè)pdf對(duì)象是同一個(gè),同時(shí)這里暫時(shí)也沒考慮異步操作出錯(cuò)的情況,有要求的話可以在加個(gè)catch捕獲錯(cuò)誤)

getDocument方法的參數(shù)是一個(gè)對(duì)象,對(duì)象鍵名為data,值為base64解碼后的值,此方法返回一個(gè)pdf對(duì)象,這個(gè)對(duì)象有幾個(gè)屬性,可以打印出來觀察一下

這里我們先用到的是numPages屬性,它指的是當(dāng)前pdf文件有多少頁

async showPdf() {
    let pdfList = document.querySelector('.pdfList') //通過querySelector選擇DOM節(jié)點(diǎn),使用document.getElementById()也一樣
    for(let value of this.pdfDataList){ //遍歷后臺(tái)傳過來的pdfDataList
      let base64 = value.fileValue //獲得bas464編碼
      let decodedBase64 = atob(base64) //使用瀏覽器自帶的方法解碼
      let pdf = await PDFJS.getDocument({data: decodedBase64}) //返回一個(gè)pdf對(duì)象
      let pages = pdf.numPages //聲明一個(gè)pages變量等于當(dāng)前pdf文件的頁數(shù)
    }
  }

獲取當(dāng)前pdf文件的對(duì)象和當(dāng)前pdf文件的所有頁數(shù)后,循環(huán)遍歷每個(gè)頁數(shù),執(zhí)行如下操作:

1)動(dòng)態(tài)創(chuàng)建canvas節(jié)點(diǎn)
2)調(diào)用pdf對(duì)象原型上的getPage()方法和getViewport()方法,依次傳入當(dāng)前循環(huán)的頁數(shù)和canvas的縮放大小(這里不懂的可以直接復(fù)制黏貼)
3)渲染當(dāng)前的canvas節(jié)點(diǎn)
4)調(diào)用page對(duì)象的render()方法渲染當(dāng)前頁,此方法也是一個(gè)promise,需要使用await關(guān)鍵字等到狀態(tài)為resolve后再執(zhí)行之后的代碼
5)給顯示當(dāng)前頁面的canvas節(jié)點(diǎn)一個(gè)className為canvas方便修改樣式,最后把這個(gè)canvas節(jié)點(diǎn)插入到pdfList節(jié)點(diǎn)中

async showPdf() {
    let pdfList = document.querySelector('.pdfList') //通過querySelector選擇DOM節(jié)點(diǎn),使用document.getElementById()也一樣
    for(let value of this.pdfDataList){ //遍歷后臺(tái)傳過來的pdfDataList
      let base64 = value.fileValue //獲得bas464編碼
      let decodedBase64 = atob(base64) //使用瀏覽器自帶的方法解碼
      let pdf = await PDFJS.getDocument({data: decodedBase64}) //返回一個(gè)pdf對(duì)象
      let pages = pdf.numPages //聲明一個(gè)pages變量等于當(dāng)前pdf文件的頁數(shù)
      for (let i = 1; i <= pages; i++) { //循環(huán)頁數(shù)
       let canvas = document.createElement('canvas') 
       let page = await pdf.getPage(i) //調(diào)用getPage方法傳入當(dāng)前循環(huán)的頁數(shù),返回一個(gè)page對(duì)象
       let scale = 1;//縮放倍數(shù),1表示原始大小
       let viewport = page.getViewport(scale); 
       let context = canvas.getContext('2d'); //創(chuàng)建繪制canvas的對(duì)象
       canvas.height = viewport.height; //定義canvas高和寬
       canvas.width = viewport.width;
       let renderContext = {
        canvasContext: context,
        viewport: viewport
       };
       await page.render(renderContext)

       canvas.className = 'canvas' //給canvas節(jié)點(diǎn)定義一個(gè)class名,這里我取名為canvas
       pdfList.appendChild(canvas) //插入到pdfList節(jié)點(diǎn)的最后
      }
    }
  } 

至此頁面上就會(huì)多出一個(gè)canvas節(jié)點(diǎn)并且顯示當(dāng)前pdf文件的第一頁的圖片,如果當(dāng)前pdf文件有多頁就會(huì)渲染出多個(gè)canvas節(jié)點(diǎn),有多個(gè)pdf文件就會(huì)先循環(huán)外層,然后再循環(huán)內(nèi)層,把每個(gè)pdf文件的每一頁都生成一個(gè)canvas節(jié)點(diǎn)

 

修改樣式

渲染出頁面后還有個(gè)要注意的點(diǎn),Vue框架會(huì)給每個(gè)組件的DOM節(jié)點(diǎn)生成一個(gè)自定義屬性,而節(jié)點(diǎn)動(dòng)態(tài)生成的canvas節(jié)點(diǎn),并沒有data-v-xxxxx這樣的自定義屬性

而Vue會(huì)給每個(gè)組件里面的樣式添加這個(gè)自定義屬性,Vue框架這樣做可以防止樣式的相互污染(也就是style旁邊的scoped屬性)

我們這里可以在這個(gè)style下面再創(chuàng)建一個(gè)style寫入樣式來達(dá)到修改canvas樣式的效果,但是記得這樣做你整個(gè)項(xiàng)目里面類名叫canvas的都會(huì)獲得這個(gè)樣式,需要注意

寫在最后

這里使用的是動(dòng)態(tài)生成canvas節(jié)點(diǎn)然后渲染這個(gè)節(jié)點(diǎn)生成的圖片,然而直接使用createElement生成一個(gè)節(jié)點(diǎn)并且頻繁操作DOM會(huì)對(duì)性能有一定的影響,如果有更好的方法歡迎留言交流,感謝觀看。希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

相關(guān)文章

  • 解決父組件將子組件作為彈窗調(diào)用只執(zhí)行一次created的問題

    解決父組件將子組件作為彈窗調(diào)用只執(zhí)行一次created的問題

    這篇文章主要介紹了解決父組件將子組件作為彈窗調(diào)用只執(zhí)行一次created的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-07-07
  • vue2路由基本用法實(shí)例分析

    vue2路由基本用法實(shí)例分析

    這篇文章主要介紹了vue2路由基本用法,結(jié)合實(shí)例形式分析了vue2路由基本功能、原理、用法與操作注意事項(xiàng),需要的朋友可以參考下
    2020-03-03
  • Vue實(shí)現(xiàn)todolist刪除功能

    Vue實(shí)現(xiàn)todolist刪除功能

    這篇文章主要介紹了Vue實(shí)現(xiàn)todolist刪除功能,,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-06-06
  • vue-quill-editor的使用及個(gè)性化定制操作

    vue-quill-editor的使用及個(gè)性化定制操作

    這篇文章主要介紹了vue-quill-editor的使用及個(gè)性化定制操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • vue中swiper?vue-awesome-swiper的使用方法及各種坑解決

    vue中swiper?vue-awesome-swiper的使用方法及各種坑解決

    這篇文章主要介紹了vue中swiper?vue-awesome-swiper的使用方法及各種坑解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • vue項(xiàng)目中使用axios遇到的相對(duì)路徑和絕對(duì)路徑問題

    vue項(xiàng)目中使用axios遇到的相對(duì)路徑和絕對(duì)路徑問題

    這篇文章主要介紹了vue項(xiàng)目中使用axios遇到的相對(duì)路徑和絕對(duì)路徑問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • vue實(shí)現(xiàn)登錄時(shí)圖形驗(yàn)證碼

    vue實(shí)現(xiàn)登錄時(shí)圖形驗(yàn)證碼

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)登錄時(shí)圖形驗(yàn)證碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • VUE頁面中加載外部HTML的示例代碼

    VUE頁面中加載外部HTML的示例代碼

    本篇文章主要介紹了VUE頁面中加載外部HTML的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-09-09
  • Vux+Axios攔截器增加loading的問題及實(shí)現(xiàn)方法

    Vux+Axios攔截器增加loading的問題及實(shí)現(xiàn)方法

    這篇文章主要介紹了Vux+Axios攔截器增加loading的問題及實(shí)現(xiàn)方法,文中通過實(shí)例代碼介紹了vue中使用axios的相關(guān)知識(shí),需要的朋友可以參考下
    2018-11-11
  • 使用vue-video-player實(shí)現(xiàn)直播的方式

    使用vue-video-player實(shí)現(xiàn)直播的方式

    在開發(fā)期間使用過video.js、mui-player等插件,發(fā)現(xiàn)這些video插件對(duì)移動(dòng)端的兼容性都不友好,最后發(fā)現(xiàn)一個(gè)在移動(dòng)端兼容不錯(cuò)的插件vue-video-player,下面通過場(chǎng)景分析給大家介紹使用vue-video-player實(shí)現(xiàn)直播的方法,感興趣的朋友一起看看吧
    2022-01-01

最新評(píng)論