濮阳杆衣贸易有限公司

主頁(yè) > 知識(shí)庫(kù) > html5 canvas繪制網(wǎng)絡(luò)字體的常用方法

html5 canvas繪制網(wǎng)絡(luò)字體的常用方法

熱門標(biāo)簽:ok電銷機(jī)器人 高德地圖標(biāo)注商戶怎么標(biāo) 智能電銷機(jī)器人被禁用了么 欣鼎電銷機(jī)器人 效果 地圖標(biāo)注軟件打印出來 電話機(jī)器人技術(shù) 黃石ai電銷機(jī)器人呼叫中心 惡搞電話機(jī)器人 如何查看地圖標(biāo)注

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

服務(wù)端轉(zhuǎn)換
 

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

webfontloader
 

webfontloader是一個(gè)由Google和Typekit共同開發(fā)的組件庫(kù),提供了一組標(biāo)準(zhǔn)事件監(jiān)聽字體的加載,雖然已經(jīng)很長(zhǎng)時(shí)間沒有更新了,但是對(duì)字體加載的監(jiān)聽確實(shí)有效,下面來看一個(gè)具體的例子怎么使用:

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.
document.getElementsByTagName('head')[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,并且動(dòng)態(tài)插入一個(gè)script標(biāo)簽載入google的字體,然后調(diào)用webfontloader的load方法進(jìn)行配置監(jiān)聽,當(dāng)字體加載完成后就會(huì)觸發(fā)active鉤子,開始繪制對(duì)應(yīng)字體的內(nèi)容,webfontloader提供了一個(gè)完整的事件系統(tǒng)鉤子給開發(fā)者調(diào)用:

 

如果想要了解webfontloader的更多用法可以前往 github 查看學(xué)習(xí),如果你覺得為了繪制網(wǎng)絡(luò)字體需要引入一個(gè)js庫(kù)有點(diǎn)得不償失,沒關(guān)系,接下來向你接受不用庫(kù)的方法

document.fonts.load
 

如果你在Google上搜索canvas加載網(wǎng)絡(luò)字體,你一定能搜到下面這個(gè)方案:

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

這個(gè)方案存在一點(diǎn)問題,當(dāng)image onerror事件觸發(fā)的時(shí)候,并不能保證字體已經(jīng)加載完成,只能保證css文件已經(jīng)加載完成,因此,在第一次訪問的時(shí)候并不會(huì)生效:

 

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

 

這是什么原因呢?我們來看一下刷新瀏覽器的網(wǎng)絡(luò)請(qǐng)求:

 
 

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

 

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

var canvas = document.getElementById('canvas')
var ctx = canvas.getContext('2d')
var link = document.createElement('link')
link.rel = 'stylesheet'
link.type = 'text/css'
link.
document.getElementsByTagName('head')[0].appendChild(link)
var image = document.createElement('img')
image.src = link.href
image.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文件的加載,然后調(diào)用document.fonts.load看字體是否加載完成,這樣就可以準(zhǔn)確監(jiān)聽到字體加載完成,但是這個(gè)api存在兼容性問題,來看具體表格:

 

想要對(duì)這個(gè)api了解更多,可以前往 mdn 查看

對(duì)比繪制

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

var canvas = document.getElementById('canvas')
var ctx = canvas.getContext('2d')
var link = document.createElement('link')
link.rel = 'stylesheet'
link.type = 'text/css'
link.
document.getElementsByTagName('head')[0].appendChild(link)
ctx.font = '50px UNKNOW'
ctx.textBaseline = 'top'
ctx.fillText('123', 20, 10)
var dataDefault = ctx.getImageData(20, 10, 50, 50).data
ctx.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()

首先設(shè)置一個(gè)沒有的字體,繪制上去,然后拿到對(duì)應(yīng)區(qū)域的渲染數(shù)據(jù),然后再將渲染區(qū)域清除然后,然后再設(shè)置我們需要的字體,拿到對(duì)應(yīng)區(qū)域的渲染數(shù)據(jù),然后實(shí)時(shí)對(duì)比,當(dāng)渲染數(shù)據(jù)一樣時(shí),表示繪制的都是系統(tǒng)默認(rèn)字體,我們需要的字體沒有渲染出來,然后執(zhí)行requestAnimationFrame再執(zhí)行detect檢測(cè)方法,直到渲染數(shù)據(jù)不一樣,就表示我們需要的字體已經(jīng)渲染完成

總結(jié)

以上所述是小編給大家介紹的html5 canvas繪制網(wǎng)絡(luò)字體的常用方法,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!

標(biāo)簽:盤錦 萍鄉(xiāng) 金昌 聊城 綏化 中山 赤峰 阿壩

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《html5 canvas繪制網(wǎng)絡(luò)字體的常用方法》,本文關(guān)鍵詞  html5,canvas,繪制,網(wǎng)絡(luò),字體,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《html5 canvas繪制網(wǎng)絡(luò)字體的常用方法》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于html5 canvas繪制網(wǎng)絡(luò)字體的常用方法的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    呼伦贝尔市| 子洲县| 全州县| 陈巴尔虎旗| 铅山县| 涟源市| 淮阳县| 四平市| 锦屏县| 洪雅县| 东乡县| 湟中县| 古蔺县| 博白县| 峨边| 余江县| 荔波县| 河北省| 乳山市| 客服| 大安市| 德保县| 石门县| 年辖:市辖区| 泾源县| 香港| 色达县| 吉木萨尔县| 勃利县| 静安区| 富裕县| 凉山| 鹤峰县| 海盐县| 视频| 崇左市| 太仓市| 乌恰县| 葫芦岛市| 大足县| 玛曲县|