canvas實現高階貝塞爾曲線
寫在最前
由於原生的Canvas最高只支援到三階貝塞爾曲線,那麼我想新增多個控制點怎麼辦呢?(即便大部分複雜曲線都可以用3階貝塞爾來模擬)與此同時,關於貝塞爾控制點的位置我們很難非常直觀的清楚到底將控制點設定為多少可以形成我們想要的曲線。本著解決以上兩個痛點同時社群內好像並沒有N階的解決方案(js版)故這次作者非常認真的開源了bezierMaker.js!
bezierMaker.js理論上支援N階貝塞爾曲線的生成,同時提供了試驗場供開發者可以自行新增並拖拽控制點最終生成一組繪製動畫。非常直觀的讓開發者知道不同位置的控制點所對應的不同生成曲線。
如果你喜歡這個作品歡迎Star,畢竟star來之不易。。
專案地址:這裡✨✨✨
歡迎關注我的部落格,不定期更新中——
為什麼需要一個試驗場?
在繪製複雜的高階貝塞爾曲線時無法知道自己需要的曲線的控制點的精確位置。在試驗場中進行模擬,可以實時得到控制點的座標值,將得到的點座標變為物件陣列傳遞進BezierMaker類就可以生成目標曲線
效果圖
功能
- [x] 試驗場可新增任意數量控制點
- [x] 試驗場支援展示曲線繪製的形成動畫
- [x] 控制點可自由拖拽
- [x] 支援顯示貝塞爾曲線形成過程的切線
- [x] 3階及以下貝塞爾曲線的繪製採用原生API
引入
<script src="./bezierMaker.js"></script>
繪製
上面的效果圖為試驗場的使用,當你通過試驗場獲得控制點的準確座標之後,就可以呼叫bezierMaker.js進行曲線的直接繪製。
/**
* canvas canvas的dom物件
* bezierCtrlNodesArr 控制點陣列,包含x,y座標
* color 曲線顏色
*/
var canvas = document.getElementById('canvas')
//3階之前採用原生方法實現
var arr0 = [{x:70,y:25},{x:24,y:51}]
var arr1 = [{x:233,y:225},{x:170,y:279},{x:240,y:51}]
var arr2 = [{x:23,y:225},{x:70,y:79},{x:40,y:51},{x:300, y:44}]
var arr3 = [{x:333,y:15},{x:70,y:79},{x:40,y:551},{x:170,y:279},{x:17,y:239}]
var arr4 = [{x:53,y:85},{x:170,y:279},{x:240,y:551},{x:70,y:79},{x:40,y:551},{x:170,y:279}]
var bezier0 = new BezierMaker(canvas, arr0, 'black')
var bezier1 = new BezierMaker(canvas, arr1, 'red')
var bezier2 = new BezierMaker(canvas, arr2, 'blue')
var bezier3 = new BezierMaker(canvas, arr3, 'yellow')
var bezier4 = new BezierMaker(canvas, arr4, 'green')
bezier0.drawBezier()
bezier1.drawBezier()
bezier2.drawBezier()
bezier3.drawBezier()
bezier4.drawBezier()
繪製結果
當控制點少於3個時,會適配使用原生的API介面。當控制點多於2個後,由我們自己實現的函式進行描點繪製。
核心原理
繪製貝塞爾曲線
繪製貝塞爾曲線的核心點在於貝塞爾公式的運用:
這個公式中的P0-Pn代表了從起點到各個控制點再到終點的各點與佔比t的各種冪運算。
BezierMaker.prototype.bezier = function(t) { //貝塞爾公式呼叫
var x = 0,
y = 0,
bezierCtrlNodesArr = this.bezierCtrlNodesArr,
//控制點陣列
n = bezierCtrlNodesArr.length - 1,
self = this
bezierCtrlNodesArr.forEach(function(item, index) {
if(!index) {
x += item.x * Math.pow(( 1 - t ), n - index) * Math.pow(t, index)
y += item.y * Math.pow(( 1 - t ), n - index) * Math.pow(t, index)
} else {
//factorial為階乘函式
x += self.factorial(n) / self.factorial(index) / self.factorial(n - index) * item.x * Math.pow(( 1 - t ), n - index) * Math.pow(t, index)
y += self.factorial(n) / self.factorial(index) / self.factorial(n - index) * item.y * Math.pow(( 1 - t ), n - index) * Math.pow(t, index)
}
})
return {
x: x,
y: y
}
}
對所有點進行遍歷同時根據當前佔比t的值(0<=t<=1),計算出當前在貝塞爾曲線上的點座標x,y。t的取值作者分成了1000份,即每次運算t+=0.01。此時算出的x,y即所求的貝塞爾曲線分成了1000份之後的某一點。當t值從0~1遍歷1000次後生成1000個x,y對應座標,依次描點畫線即可模擬出高階貝塞爾曲線。
對於貝塞爾公式的推導作者會在之後的文章中專門說明,現在你只需要知道我們通過貝塞爾公式計算出實際貝塞爾曲線被等分成了1000份的各點,用直線連線各點後即可模擬出類曲線。
對於模擬場貝塞爾曲線生成動畫的實現
這個部分相關程式碼可以參考這裡
整體思路是用遞迴的方式來將每個一層控制點當做1階貝塞爾函式來計算下一層控制點並對應連線。具體邏輯作者會留到深入講解貝塞爾曲線公式原理的時候一起梳理一下試驗場的動畫生成原理~
小結
作者一直想開源一些東西(但是菜,也沒啥能寫的),然而平時會用到的都被人寫了,再造輪子也沒別人寫得好。這次也算是發現了一個貌似空白一些的區域。所以非常鄭重的決定開源。貝塞爾的高階運用在gayhub中大多是安卓的實現,前端領域中還有很多地方可以更多的展開,歡迎討論~ 多多批評!
其他canvas相關文章
- 基於canvas使用粒子拼出你想要的文字[2]——粒子的動畫效果
- 基於canvas使用粒子拼出你想要的文字
- 基於canvas使用貝塞爾曲線平滑擬合折線段
- 用canvas實現視訊播放與彈幕功能
- 基於canvas實現波浪式繪製圖片
- 基於 canvas 實現的一個截圖小 demo
最後
專案地址:這裡✨✨
試驗場地址:一定進來玩✨✨✨
慣例po作者的部落格,不定時更新中——
有問題歡迎在issues下交流。
相關文章
- 【Flutter高階玩法】 貝塞爾曲線的表象認知Flutter
- canvas bezierCurveTo() 三次貝塞爾曲線Canvas
- canvas 二次貝塞爾曲線quadraticCurveTo()Canvas
- html5中canvas繪製貝塞爾曲線HTMLCanvas
- 使用二階貝塞爾曲線實現新增購物車動畫動畫
- 貝塞爾曲線(Bezier curve)實現節點連線
- 安卓自定義 View 進階:貝塞爾曲線安卓View
- 用canvas繪製一個曲線動畫——深入理解貝塞爾曲線Canvas動畫
- canvas基礎[二]教你編寫貝塞爾曲線工具Canvas
- html5中canvas貝塞爾曲線繪製菊花HTMLCanvas
- Android 貝塞爾曲線Android
- webGL入門-四階貝塞爾曲線繪製Web
- Android-貝塞爾曲線實現水波紋動畫Android動畫
- 貝塞爾曲線原理、推導及Matlab實現Matlab
- 貝塞爾曲線基礎部分
- UIBezierPath貝塞爾曲線UI
- bezierMaker.js——N階貝塞爾曲線生成器JS
- 簡易製作貝塞爾曲線動畫(JS+css3+canvas)動畫JSCSSS3Canvas
- 貝塞爾曲線理解與應用
- iOS UIBezierPath 貝塞爾曲線iOSUI
- 如何理解並應用貝塞爾曲線
- Android 自定義貝塞爾曲線工具Android
- 貝塞爾曲線開發的藝術
- 【flutter高階玩法】貝塞爾實戰1 - 波浪Flutter
- 貝塞爾曲線的css實現——淘寶加入購物車基礎動畫CSS動畫
- iOS UIBezierPath貝塞爾曲線常用方法iOSUI
- 使用貝塞爾曲線裁圓優化tableView優化View
- Path從懵逼到精通(2)——貝塞爾曲線
- js控制貝塞爾曲線程式碼例項JS線程
- canvas實現的賽貝爾曲線效果程式碼例項Canvas
- 自定義View合輯(6)-波浪(貝塞爾曲線)View
- SVG <path> C 指令 三次貝塞爾曲線SVG
- SVG <path> Q指令 二次貝塞爾曲線SVG
- iOS開發之畫圖板(貝塞爾曲線)iOS
- Android日常學習:OpenGL 實踐之貝塞爾曲線繪製Android
- Flutter 自定義元件之貝塞爾曲線畫波浪球Flutter元件
- SVG <path>元素C指令三次貝塞爾曲線SVG
- SVG <path>元素Q指令二次貝塞爾曲線SVG