国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

CSS、SVG、Canvas對特殊字體的繪制與導出

開發 前端
簡單解釋一下:所謂繪制,就是視覺上可以看到就行(預覽狀態),導出呢,就是將看到的轉換成圖片(或者Canvas),以便于后續處理。

最近在項目中需要對特殊字體進行繪制與導出,如下:

圖片

簡單解釋一下:所謂繪制,就是視覺上可以看到就行(預覽狀態),導出呢,就是將看到的轉換成圖片(或者Canvas),以便于后續處理。

這里總結了 3 種方式,分別是 CSS 、 SVG、Canvas,來看看各自有什么差異和優缺點吧

一、CSS 的繪制與導出

首先來看 CSS ,這是最簡單的繪制方式了。

假設 HTML是這樣的。

<div class="text">前端偵探</div>

加點樣式。

.text{
display: flex;
width: 200px;
height: 200px;
justify-content: center;
align-items: center;
background-color: rebeccapurple;
color: #fff;
font-size: 36px;
font-family: MFMengYuan-Regular;
}

這里給了一個特殊的字體MFMengYuan-Regular(造字工坊夢緣體),當然現在肯定是沒有效果,因為系統并沒有這樣的字體。

圖片

為了使這個特殊字體生效,需要手動通過@font-face去定義。

@font-face {
font-family: "MFMengYuan-Regular";
src: url("https://webfontsource.yuewen.com/api/v1/yfont/font.eot?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2"); /* IE9 */
src: local('?'),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.woff2?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2") format("woff2"),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.woff?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2") format("woff"),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.ttf?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2");
}

這里引用的是一個在線生成的字體,對于 CSS 來說也是小菜一碟,效果如下:

圖片

是不是非常輕松?

CSS 繪制非常容易,但現在僅僅是視覺上的,那如何將這個樣式轉換成圖片導出呢?

在這里,需要借助 SVG? 中的foreignObject[1]元素,通過這個元素,可以將 HTML?嵌入到SVG中,例如:

<svg xmlns="http://www.w3.org/2000/svg">
<foreignObject width="200" height="200">
<body xmlns="http://www.w3.org/1999/xhtml">
<div>前端偵探</div>
</body>
</foreignObject>
</svg>

一些截圖工具庫,比如 html2canvas都依賴 foreignObject 這個特性。

而SVG本質上就是圖片,然后就可以將這個圖片繪制到 Canvas 上,進一步進行圖片合成和處理了,整體思路如下:

圖片

不過需要注意的是,SVG?是一個獨立的圖片,必須包含繪制內容的全部信息,比如這里需要手動將style?樣式內嵌到div中,就像這樣(代碼結構可能不是很好看)。

<div class="text">
<style>
@font-face {
font-family: "MFMengYuan-Regular";
src: url("https://webfontsource.yuewen.com/api/v1/yfont/font.eot?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2"); /* IE9 */
src: local('?'),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.woff2?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2") format("woff2"),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.woff?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2") format("woff"),
url("https://webfontsource.yuewen.com/api/v1/yfont/font.ttf?base64=0&fnotallow=MFMengYuan-Regular&text=%E5%89%8D%E7%AB%AF%E4%BE%A6%E6%8E%A2");
}
.text{
display: flex;
width: 200px;
height: 200px;
justify-content: center;
align-items: center;
background-color: rebeccapurple;
color: #fff;
font-size: 36px;
font-family: MFMengYuan-Regular;
}
</style>
前端偵探
</div>

接下來通過JS將其包裹上foreignObject元素,注意一下特殊字符的轉義。

const htmlSvg = `data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" width="${width/pixelRatio}" height="${height/pixelRatio}">
<foreignObject x="0" y="0" width="100%" height="100%">
<body xmlns="http://www.w3.org/1999/xhtml" style="height:100%;margin:0">
${dom.outerHTML}
</body>
</foreignObject></svg>`.replace(/"/g,"'").replace(/%/g,"%25").replace(/#/g,"%23").replace(/{/g,"%7B").replace(/}/g,"%7D").replace(/</g,"%3C").replace(/>/g,"%3E");
img.src = htmlSvg

這樣得到的一個??SVG??字符串就是一個完整的圖片了。

圖片

等等...圖片是出來了,不過字體好像丟失了?

為什么會這樣呢?原因在于,上面字體使用的是在線字體,在線字體在轉成字符后就是普通的字符了,不會發出請求,自然也不會包含字體的真實信息了,所以要解決這個問題就必須提前將字體轉成本地??base64??格式,如下

<div class="text">
<style>
@font-face {
font-family: "MFMengYuan-Regular";
src: local('?'),
url(data:font/woff;charset=utf-8;base64,d09GRgABAAAAAAwcAA0AAAAAEPgAAAAAAAAAAAAAAAAAA...==) format('woff');
}
.text{
display: flex;
width: 200px;
height: 200px;
justify-content: center;
align-items: center;
background-color: rebeccapurple;
color: #fff;
font-size: 36px;
font-family: MFMengYuan-Regular;
}
</style>
前端偵探
</div>

這樣就正常了(SVG字符可能會比較長)。

圖片

同樣也能將這個圖片繪制到Canvas上。

const context = canvas.getContext('2d');
context.drawImage(htmlSvg, 0, 0, width, height);

效果如下:

圖片

除此之外,通過Canvas還能將圖片轉成blob地址,相比完整 SVG地址而言,地址更加簡潔,有時候圖片過大,在賦值給圖片src會造成瀏覽器卡頓,盡量用blob方式。

canvas.toBlob(function(blob){
img.src = URL.createObjectURL(blob)
})

效果如下:

圖片

完整轉換過程可以查看以下鏈接:

  • CSS font - 碼上掘金 (juejin.cn)[2]
  • CSS font (codepen.io)[3]
  • CSS font (runjs.work)[4]

二、SVG 的繪制與導出

下面來看SVG方式,相比CSS而言,可能稍微麻煩一點,主要是文本排版方面,同樣需要注意字體base64處理。

<svg id="svg" class="text" xmlns='http://www.w3.org/2000/svg' viewBox="0 0 200 200" width="200" height="200">
<style>
@font-face {
font-family: "MFMengYuan-Regular";
src: local('?'),
url(data:font/woff;charset=utf-8;base64,d09GRgABAAAAAAwcAA0AAAAAEPgAAAAAAAAAAAAAAAAAAAAAAAAAAAA...==) format('woff');
}
.text{
background-color: rebeccapurple;
}
</style>
<text x="50%" y="50%" text-anchor="middle" dominant-baseline="middle" fill="#fff" font-size="36" font-family="MFMengYuan-Regular">前端偵探</text>
</svg>

這里需要注意一下 SVG 中的文本居中方式,用到了dominant-baseline(基線對齊)和text-anchor(錨點對齊),如下:

圖片

兩者結合,再配合x=50%和y=50%就實現了水平垂直居中效果了,如下:

圖片

由于已經是SVG了,所以導出圖片或者繪制到Canvas畫布上就更方便,只需要將整個 dom 結構轉義一下就可以了,無需額外包裹。

const htmlSvg = `data:image/svg+xml,${dom.outerHTML}`.replace(/"/g,"'").replace(/%/g,"%25").replace(/#/g,"%23").replace(/{/g,"%7B").replace(/}/g,"%7D").replace(/</g,"%3C").replace(/>/g,"%3E");
img.src = htmlSvg

效果如下:

圖片

繪制到Canvas上也是同樣的方式。

const context = canvas.getContext('2d');
context.drawImage(htmlSvg, 0, 0, width, height);

效果如下:

圖片

完整轉換過程可以查看以下鏈接:

  • SVG font - 碼上掘金 (juejin.cn)[5]
  • SVG font (codepen.io)[6]
  • SVG font (runjs.work)[7]

三、Canvas 的繪制與導出

最后是 Canvas方式。

這里要繪制的很簡單,就是一個矩形和一行文字,主要步驟如下:

const context = canvas.getContext('2d');
context.fillStyle = 'rebeccapurple'// 填充顏色
context.fillRect(0,0,width,height) // 繪制矩形
context.fillStyle = '#fff' // 填充顏色
context.font = `36px MFMengYuan-Regular`; // 設置字體屬性
context.textAlign = 'center'; // 設置文本對齊
context.textBaseline = 'middle' // 設置基線對齊
context.fillText('前端偵探', width/2, height/2); // 繪制文本

效果如下:

? 圖片 ?

不出意料,字體果然沒有繪制,因為系統并沒有這種字體,那如何主動添加字體呢?

這里有一個策略,Canvas讀取的是頁面上已經渲染過的字體,也就是說頁面上如果提前渲染過該字體,那么在繪制的時候就可以直接繪制出來,如果字體是動態的,可以通過動態創建。

const fontStyle = `
@font-face {
font-family: "MFMengYuan-Regular";
src: local('?'),
url(data:font/woff;charset=utf-8;base64,d09GRgABAAAAAAwcAA0AAAAAEPgAAAAAAAAAAA...==) format('woff');
}
`
const style = document.createElement('style')
style.textContent = fontStyle
document.head.appendChild(style)

現在重新繪制,如下:

圖片

可以看到,起初是沒有字體的,刷新后才繪制新的字體。

原因是,前面這段代碼僅僅是表示頁面有這個字體,但是并沒有渲染過,通過Canvas繪制后,這個字體才真正被渲染,所以到了第二次字體才生效。

知道原因后,解決就很簡單了。如果不是實時繪制,比如說預覽狀態通過 CSS 繪制,那么等到 Canvas 繪制的時候(比如通過按鈕點擊生成預覽圖),字體當然已經渲染過,自然也不會有這個問題。如果一定要實時繪制,可以采用逐幀比對的方式,一旦圖像發生變化,就表示字體渲染成功,實現如下

該方法參考自張鑫旭老師這篇文檔:canvas API中文網 - 中文文檔 - CanvasRenderingContext2D.font[8]。

// 先隨便繪制一個字體
context.font = `36px UNKNOW`;
context.fillText('前端偵探', width/2, height/2);
const dataDefault = context.getImageData(0, 0, width/4, height/2).data;
const detect = function () {
// 然后繪制實際字體
context.font = `36px MFMengYuan-Regular`;
context.fillText('前端偵探', width/2, height/2);
// 如果前后數據一致,說明字體還沒加載成功,繼續檢測
var dataNow = context.getImageData(0, 0, width/4, height/2).data;
if ([].slice.call(dataNow).join('') == [].slice.call(dataDefault).join('')) {
console.log('沒有變化,重新渲染')
requestAnimationFrame(detect);
}
};

這樣就可以實時繪制特殊字體了。

圖片

Canvas本身就是圖片了,直接可以轉換成圖片或者導出,這里就不多介紹了。

完整實現過程可以查看以下鏈接:

  • Canvas font - 碼上掘金 (juejin.cn)[9]
  • Canvas font (codepen.io)[10]
  • Canvas font (runjs.work)[11]

四、總結一下各自優缺點

下面簡單整理了一下各自實現的難易程度。

CSS繪制最簡單,尤其是在文本排版方面,要遠遠領先其他兩種方式。

SVG繪制相對比較簡單,在矢量圖形處理,比如描邊特效要比 CSS 更有優勢,這兩種方式導出的難點在于一些外鏈資源的額外處理。

而Canvas繪制稍微復雜一些,在特殊字體需要逐幀去檢測是否渲染,優點是繪制出來就是圖片,無需額外導出。


繪制

導出

CSS

??????(簡單)

????(一般)

SVG

????(一般)

????(一般)

Canvas

??(復雜)

????????(超級簡單)

關于 CSS 和 SVG 的選擇可以看實際文本排版需求,比如文本需要換行,字號大小也不一致,像這種情況 CSS 就比較有優勢了,無需去精確計算文本坐標。

另外,在實際工作中,根據需求可能需要多種方式結合使用,也就是預覽狀態和導出狀態分別用不同的方式實現,比如圖片混合,在預覽狀態完全可以通過 CSS 實現,在導出時才通過 Canvas 去繪制合成。

參考資料

[1]foreignObject: https://developer.mozilla.org/en-US/docs/Web/SVG/Element/foreignObject?。

[2]CSS font - 碼上掘金 (juejin.cn): https://code.juejin.cn/pen/7170205919391776801?。

[3]CSS font (codepen.io): https://codepen.io/xboxyan/pen/oNydKrv?。

[4]CSS font (runjs.work): https://runjs.work/projects/62abc5942d9042b5?。

[5]SVG font - 碼上掘金 (juejin.cn): https://code.juejin.cn/pen/7170214063337635853?。

[6]SVG font (codepen.io): https://codepen.io/xboxyan/pen/zYaaOvb?。

[7]SVG font (runjs.work): https://runjs.work/projects/e2ff8774f0d0463b?。

[8]canvas API中文網 - 中文文檔 - CanvasRenderingContext2D.font: https://www.canvasapi.cn/CanvasRenderingContext2D/font#&others?。

[9]Canvas font - 碼上掘金 (juejin.cn): https://code.juejin.cn/pen/7170227469218218014?。

[10]Canvas font (codepen.io): https://codepen.io/xboxyan/pen/ExRRYMV?。

[11]Canvas font (runjs.work): https://runjs.work/projects/fc8d17a0c2494531?。

責任編輯:姜華 來源: 前端偵探
相關推薦

2023-01-09 11:33:25

CSS寫作網格線

2024-04-08 09:11:13

SVG前端CSS

2021-12-13 20:08:28

理論渲染技術

2010-08-20 14:47:02

IEFirefoxCSS

2010-09-02 13:53:58

CSS Sprites

2024-05-16 10:35:09

SVGCSS背景平鋪

2022-11-14 11:41:13

SVG開發組件

2017-04-05 16:30:09

Node.jsFFmpeg Canvas

2012-02-22 15:41:50

HTML 5

2022-02-28 15:52:07

canvasHarmonyOS鴻蒙

2010-09-06 10:16:21

CSS字體

2012-05-07 14:25:16

HTML5

2024-01-29 09:33:48

CSS前端SVG

2022-12-18 22:11:46

2012-02-24 15:28:36

ibmdw

2022-07-05 16:21:44

繪制表盤鴻蒙

2022-05-18 20:02:47

SVGcanvasChrome 插件

2010-09-02 09:59:52

CSS SpritesCSS

2023-05-08 09:08:33

CSS前端

2010-09-30 14:55:23

Javascriptfloat
點贊
收藏

51CTO技術棧公眾號

色欧美88888久久久久久影院| 97在线观看免费高清| 日韩激情免费视频| 三级精品在线观看| 亚洲自拍中文字幕| 九九热hot精品视频在线播放 | 欧美1234区| 在线观看一区日韩| 曰韩少妇与小伙激情| 国产精品亲子乱子伦xxxx裸| 国产欧美日韩小视频| 国产综合色视频| 色一情一乱一伦一区二区三区丨 | 亚洲视频精品在线| 男人久久天堂| 精品无人区太爽高潮在线播放 | 欧美另类69精品久久久久9999| 激情av在线| 精品久久久久久久久久久下田 | 亚洲视频在线视频| 羞羞视频在线观看| 成人av资源网址| 黄色污污在线观看| 欧美丰满少妇xxxbbb| 日韩在线观看中文字幕| 久草一本av| 久久久久久久电影一区| 欧美巨大另类极品videosbest| 欧美成人高清| 精品一区二区三区亚洲| 毛片在线能看| 中文在线播放| 久久国产一区| 久久精品91久久香蕉加勒比| 国产精品热视频| 九九99九九精彩| 欧美三级视频在线| 67194成人在线观看| 中文乱码字幕高清在线观看| 99久久夜色精品国产亚洲狼| 日本中文字幕一级片| av成人男女| 欧美有码在线视频| 激情成人综合| 国产91白丝在线播放| 神马久久久久久| 日本中文字幕一区| 欧美日韩精品在线观看| 欧美日韩精品一区二区三区视频| 3d动漫精品啪啪一区二区三区免费| 伊伊综合在线| 国产成人aaaa| 日本一区二区三级电影在线观看| 国产片侵犯亲女视频播放| 国产91精品对白在线播放| 91精品国模一区二区三区| 色777狠狠狠综合伊人| 中国黄色片免费看| 久久91亚洲人成电影网站| 国产98色在线|日韩| 国产三级日本三级在线播放| 好了av在线| 午夜av成人| 在线观看免费亚洲| 亚洲三级网页| av电影在线观看| 亚洲视频在线二区| 日韩一级黄色av| 都市激情亚洲综合| 久久久天堂国产精品| 蜜臀va亚洲va欧美va天堂| 在线观看亚洲a| 国内精品国产三级国产a久久| 97久久精品人人做人人爽50路| 国产玖玖精品视频| 国产夜色精品一区二区av| 欧美一区二区三区婷婷| 凹凸国产熟女精品视频| 色综合色综合网色综合 | 欧美成人一区二区三区在线观看| 欧美1级片网站| 亚洲一区二区三区国产| 亚洲国产高清在线观看视频| 国产乱淫av一区二区三区| 米奇在线777| 国产精品久久久久久婷婷天堂| 性欧美大战久久久久久久久| 亚洲精品乱码| 亚洲性夜色噜噜噜7777| 国产91富婆露脸刺激对白| 本网站久久精品| 久久久亚洲国产| 性人久久久久| 亚洲精品国产品国语在线app| 国产1区在线| 1024av视频| 91久久国产综合久久91精品网站| 欧美一区二区黄色| 成人欧美一区二区三区视频网页 | 国产一区二区三区四区| 欧美一区二区综合| 亚洲第五色综合网| 亚洲一区二区三区国产| 国产一区视频导航| 欧美久久九九| 四虎国产精品免费久久| 午夜视频在线观看网站| 亚洲四虎av| 国产深夜男女无套内射| 调教+趴+乳夹+国产+精品| 中文字幕日韩精品久久| 久久久一区二区三区| 欧美日韩在线资源| 亚洲一区二区三区精品动漫| 国产精品自产拍在线观看中文| 国产香蕉精品视频一区二区三区| 欧美视频中文字幕| 亚洲综合999| 伊人夜夜躁av伊人久久| 久久久亚洲欧洲日产国码αv| 日本成人片在线| 在线综合视频播放| 国产一区二区日韩| 国产精品日韩精品在线播放| 粉嫩tv在线播放| 亚洲jizzjizz妇女| 日本xxxx黄色| 777久久久精品一区二区三区| 无码人妻aⅴ一区二区三区日本| 蜜桃精品久久久久久久免费影院| 亚洲a成v人在线观看| 亚洲自拍偷拍欧美| 免费看国产一级片| 三级在线播放| 欧美亚洲成人免费| 精品免费一区二区三区| 欧美性大战久久久| 4438x亚洲最大成人网| 精品少妇一区二区| 亚洲视频在线看| 久久久爽爽爽美女图片| 国产精品久久久久久av福利| 亚洲一区二区中文字幕| 国内一区在线| 日韩av中文字幕在线免费观看| 中文在线观看免费| 国产乱淫av片杨贵妃| 国产精品久久久久久亚洲毛片| 日韩电影网1区2区| 精品一区二区三区免费观看| av中文字幕一区| 欧美性猛交xxxx乱大交极品| 亚洲成av人片在线观看香蕉| 欧美黑人巨大xxx极品| 99久久精品免费看国产一区二区三区| 深夜福利成人| 性网站在线免费观看| 最近的2019中文字幕免费一页| 国产乱子伦精品视频| 日本va欧美va瓶| 中文字幕午夜精品一区二区三区| 免费精品99久久国产综合精品| 99久久综合精品| 911国产精品| 午夜精品美女自拍福到在线| 国产一区视频观看| 男人插女人下面免费视频| 黑人精品视频| 青梅竹马是消防员在线| 一区二区在线免费视频| 日本欧洲国产一区二区| 亚洲久久久久久久久久| 日韩精品久久久久| 久久中文字幕一区| 麻豆精品视频| 日本ー区在线视频| 全国精品免费看| 中文字幕制服丝袜一区二区三区| 亚洲区免费影片| 日产精品一线二线三线芒果| 青青青手机在线视频观看| 日本韩国欧美超级黄在线观看| a级毛片免费观看在线| 97中文在线| 日韩伦理在线| 中文字幕日韩精品无码内射| 国产一区视频免费观看| 欧美人与牲禽动交com| 日韩一区二区免费看| 欧美人伦禁忌dvd放荡欲情| 中文字幕日韩精品一区| 日韩在线观看免费| 和岳每晚弄的高潮嗷嗷叫视频| 欧美狂野激情性xxxx在线观| 久久精品嫩草影院| av在线播放国产| 综合操久久久| 国内精品视频666| 国产一区毛片| 亚洲欧美自拍偷拍|