成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

canvas繪制網(wǎng)絡(luò)字體幾種方法

Alfred / 1753人閱讀

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

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

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

webfontloader

webfontloader是一個(gè)由Google和Typekit共同開發(fā)的組件庫,提供了一組標(biāo)準(zhǔn)事件監(jiān)聽字體的加載,雖然已經(jīng)很長時(shí)間沒有更新了,但是對字體加載的監(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,并且動態(tài)插入一個(gè)script標(biāo)簽載入google的字體,然后調(diào)用webfontloader的load方法進(jìn)行配置監(jiān)聽,當(dāng)字體加載完成后就會觸發(fā)active鉤子,開始繪制對應(yīng)字體的內(nèi)容,webfontloader提供了一個(gè)完整的事件系統(tǒng)鉤子給開發(fā)者調(diào)用:


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

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í)候并不會生效:

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

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

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

有解決辦法嗎?答案是有的,使用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存在兼容性問題,來看具體表格:

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

對比繪制

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

總結(jié)

這篇文章介紹了幾種canvas繪制網(wǎng)絡(luò)字體時(shí)的常用方法,每個(gè)方法都各有優(yōu)劣,希望對大家有所幫助,使用時(shí)根據(jù)具體情況選用。
如果有錯(cuò)誤或不嚴(yán)謹(jǐn)?shù)牡胤剑瑲g迎批評指正,如果喜歡,歡迎點(diǎn)贊

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/106945.html

相關(guān)文章

  • H5 canvas生成圖片并上傳文件轉(zhuǎn)成PDF下載canvas文字排版

    摘要:將預(yù)覽的圖片上傳,后端生成,在管理系統(tǒng)中下載。技術(shù)要點(diǎn)文字排版設(shè)置指定背景顏色引入外部字體繪制文字圖片將生成的圖片轉(zhuǎn)成上傳這里根據(jù)后端協(xié)商,此處后端要求將圖片生成,并點(diǎn)擊批量下載實(shí)現(xiàn)步驟文字排版在一般容器中,如果要實(shí)現(xiàn)文字的排版很容易。 最近遇到一個(gè)業(yè)務(wù)需求,在小程序端定制預(yù)覽功能,并在預(yù)覽的圖片中使用指定的外部字體。將預(yù)覽的圖片上傳OSS,后端生成PDF,在管理系統(tǒng)中下載。但是………...

    canopus4u 評論0 收藏0
  • HTML5 中 canvas 的使用總結(jié)

    摘要:上面代碼中,方法指定參數(shù),表示該節(jié)點(diǎn)用于生成圖案即平面圖案。小球不斷反彈后,逐步趨于靜止。假定是像素?cái)?shù)組中一個(gè)象素的紅色值,則為綠色值,為藍(lán)色值,就是通道值。 1. 概述 Canvas API(畫布)用于在網(wǎng)頁實(shí)時(shí)生成圖像,并且可以操作圖像內(nèi)容,基本上它是一個(gè)可以用JavaScript操作的位圖(bitmap)。 使用前,首先需要新建一個(gè)網(wǎng)頁元素。 您的瀏覽器不支持canvas!...

    Batkid 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<