麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 開發 > HTML5 > 正文

詳解canvas繪制網絡字體幾種方法

2024-09-05 07:23:00
字體:
來源:轉載
供稿:網友

最近在用canvas繪圖時遇到了一個令人頭痛的問題:canvas繪制網絡字體時沒效果,遂開始了一番解決方案查找測試,中間也碰到了不少坑,于是寫下此篇文章做個總結,如果大家在用canvas時遇到了同樣的問題,希望對大家有一定的幫助,接下來就來看看有哪幾種解決辦法

服務端轉換

服務端轉換是什么意思呢?直接把內容和需要的字體傳遞給服務端,服務端提供一個文字轉圖片的接口,將字體轉換成圖片,然后在canvas中直接繪制圖片,這樣就能保證繪制網絡字體不會有問題,不會有任何的兼容性問題,但是這樣做也就意味著服務端的工作會變多,同時如果文字內容是可以被用戶編輯修改的,那就意味著用戶每操作一次,都要請求一次接口,然后重新繪制一次圖片,這樣會導致網絡開銷增加,如果不想要服務端的介入,那就看看下面的解決方案

webfontloader

webfontloader是一個由Google和Typekit共同開發的組件庫,提供了一組標準事件監聽字體的加載,雖然已經很長時間沒有更新了,但是對字體加載的監聽確實有效,下面來看一個具體的例子怎么使用:

var WebFont = require('webfontloader')var canvas = document.getElementById('canvas')var ctx = canvas.getContext('2d')var link = document.createElement('link')link.rel = 'stylesheet'link.type = 'text/css'link.)[0].appendChild(link)WebFont.load({  custom: {    families: ['Vast Shadow']  },  active: function () {    ctx.font = '50px "Vast Shadow"'    ctx.textBaseline = 'top'    ctx.fillText('123', 20, 10)  }})

首先通過require引入webfontloader,并且動態插入一個script標簽載入google的字體,然后調用webfontloader的load方法進行配置監聽,當字體加載完成后就會觸發active鉤子,開始繪制對應字體的內容,webfontloader提供了一個完整的事件系統鉤子給開發者調用:

如果想要了解webfontloader的更多用法可以前往github查看學習,如果你覺得為了繪制網絡字體需要引入一個js庫有點得不償失,沒關系,接下來向你接受不用庫的方法

document.fonts.load

如果你在Google上搜索canvas加載網絡字體,你一定能搜到下面這個方案:

var canvas = document.getElementById('canvas')var ctx = canvas.getContext('2d')var link = document.createElement('link')link.rel = 'stylesheet'link.type = 'text/css'link.)[0].appendChild(link)var image = document.createElement('img')image.src = link.hrefimage.onerror = () => {  ctx.font = '50px "Vast Shadow"'  ctx.textBaseline = 'top'  ctx.fillText('123', 20, 10)}

這個方案存在一點問題,當image onerror事件觸發的時候,并不能保證字體已經加載完成,只能保證css文件已經加載完成,因此,在第一次訪問的時候并不會生效:

但是你再刷新一下瀏覽器之后字體就生效了:


這是什么原因呢?我們來看一下刷新瀏覽器的網絡請求:

可以看到后面的字體走的是緩存,因此可以字體可以繪制出來,但是如果將chrome調試的Disable cache勾選上,將緩存禁用掉,那么無論怎么刷新,字體都不會繪制出來。

有解決辦法嗎?答案是有的,使用Font Load API進行加載,來看具體代碼:

var canvas = document.getElementById('canvas')var ctx = canvas.getContext('2d')var link = document.createElement('link')link.rel = 'stylesheet'link.type = 'text/css'link.)[0].appendChild(link)var image = document.createElement('img')image.src = link.hrefimage.onerror = () => {  document.fonts.load('50px Vast Shadow', '123').then(() => {    ctx.font = '50px "Vast Shadow"'    ctx.textBaseline = 'top'    ctx.fillText('123', 20, 10)  })}

先用image的onerror事件trick css文件的加載,然后調用document.fonts.load看字體是否加載完成,這樣就可以準確監聽到字體加載完成,但是這個api存在兼容性問題,來看具體表格:

想要對這個api了解更多,可以前往mdn查看

對比繪制

對比繪制是什么意思呢?就是先設置一個沒有的字體,然后在設置我們需要的字體進行對比,來看具體代碼:

var canvas = document.getElementById('canvas')var ctx = canvas.getContext('2d')var link = document.createElement('link')link.rel = 'stylesheet'link.type = 'text/css'link.)[0].appendChild(link)ctx.font = '50px UNKNOW'ctx.textBaseline = 'top'ctx.fillText('123', 20, 10)var dataDefault = ctx.getImageData(20, 10, 50, 50).datactx.clearRect(20, 10, 100, 100)var detect = () => {  ctx.font = '50px "Vast Shadow"'  ctx.textBaseline = 'top'  ctx.fillText('123', 20, 10)  var dataNow = ctx.getImageData(20, 10, 50, 50).data  if ([].slice.call(dataNow).join('') === [].slice.call(dataDefault).join('')) {    ctx.clearRect(20, 10, 100, 100)    requestAnimationFrame(detect)  }}detect()

首先設置一個沒有的字體,繪制上去,然后拿到對應區域的渲染數據,然后再將渲染區域清除然后,然后再設置我們需要的字體,拿到對應區域的渲染數據,然后實時對比,當渲染數據一樣時,表示繪制的都是系統默認字體,我們需要的字體沒有渲染出來,然后執行requestAnimationFrame再執行detect檢測方法,直到渲染數據不一樣,就表示我們需要的字體已經渲染完成

總結

這篇文章介紹了幾種canvas繪制網絡字體時的常用方法,每個方法都各有優劣,希望對大家有所幫助,使用時根據具體情況選用。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 福利在线国产 | 欧美a级大胆视频 | 请播放一级毛片 | 日韩黄在线观看 | 香蕉成人在线观看 | 亚洲综合无码一区二区 | av在线免费观看不卡 | 成年人福利视频 | 91网站链接 | 国产在线欧美日韩 | 精品二区在线观看 | 成人在线视频国产 | 免费毛片播放 | 一级做a爱片毛片免费 | 国产日韩大片 | 一区二区久久精品66国产精品 | 久久福利小视频 | 欧美亚洲另类在线 | 亚洲综合91 | 激情久久免费视频 | 国产一级一片免费播放 | 国产妞干网| 青青草免费观看 | 欧美成a人片在线观看久 | 久久久久一本一区二区青青蜜月 | 国产午夜探花 | 免费看操片| 午夜视频你懂的 | 国产精品视频免费网站 | 亚洲欧美国产精品va在线观看 | 欧美18一19sex性护士农村 | 欧美福利视频一区二区三区 | 免费在线观看毛片视频 | 日本成人在线免费 | 日韩av片网站 | 黄色av免费 | 激情网站免费观看 | 龙的两根好大拔不出去h | 欧美性生交大片 | 狠狠婷婷综合久久久久久妖精 | 欧美性激情视频 |