基于js Canvas實(shí)現(xiàn)二次貝塞爾曲線
本文實(shí)例為大家分享了js Canvas實(shí)現(xiàn)二次貝塞爾曲線的具體代碼,供大家參考,具體內(nèi)容如下
先上效果圖:

實(shí)現(xiàn)代碼:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>二次貝塞爾曲線</title>
<meta name="Keywords" content="">
<meta name="Description" content="">
<style type="text/css">
body, h1{margin:0;}
canvas{margin: 20px; }
</style>
</head>
<body>
<h1>二次貝塞爾曲線</h1>
<canvas id="canvas" width=600 height=600 style="border: 1px solid #ccc;"></canvas>
<script>
/**
* @param sx 起始點(diǎn)x坐標(biāo)
* @param sy 起始點(diǎn)y坐標(biāo)
* @param ex 結(jié)束點(diǎn)x坐標(biāo)
* @param ey 結(jié)束點(diǎn)y坐標(biāo)
* @param cx 控制點(diǎn)x坐標(biāo)
* @param cy 控制點(diǎn)y坐標(biāo)
* @param part 將起始點(diǎn)到控制點(diǎn)的線段分成的份數(shù),數(shù)值越高,計(jì)算出的曲線越精確
*/
function draw(sx, sy, ex, ey, cx, cy, part) {
var canvas = document.getElementById('canvas');
var ctx = canvas.getContext('2d');
//繪制起始點(diǎn)、控制點(diǎn)、終點(diǎn)
ctx.beginPath();
ctx.moveTo(sx, sy);
ctx.lineTo(cx, cy);
ctx.lineTo(ex, ey);
ctx.stroke();
// 繪制二次貝塞爾曲線
ctx.beginPath();
ctx.moveTo(sx, sy);
// 起始點(diǎn)到控制點(diǎn)的x和y每次的增量
var changeX1 = (cx - sx) / part;
var changeY1 = (cy - sy) / part;
// 控制點(diǎn)到結(jié)束點(diǎn)的x和y每次的增量
var changeX2 = (ex - cx) / part;
var changeY2 = (ey - cy) / part;
for(var i = 0; i < part; i++) {
// 計(jì)算兩個(gè)動(dòng)點(diǎn)的坐標(biāo)
var qx1 = sx + changeX1 * i;
var qy1 = sy + changeY1 * i;
var qx2 = cx + changeX2 * i;
var qy2 = cy + changeY2 * i;
// 計(jì)算得到此時(shí)的一個(gè)貝塞爾曲線上的點(diǎn)坐標(biāo)
var bx = qx1 + (qx2 - qx1) * i / part;
var by = qy1 + (qy2 - qy1) * i / part;
ctx.lineTo(bx, by);
}
ctx.stroke();
}
window.onload = function () {
draw(0, 0, 600, 0, 150, 450, 100);
};
</script>
</body>
</html>
上面的是靜態(tài)的,來個(gè)動(dòng)態(tài)的看一看:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>二次貝塞爾曲線</title>
<meta name="Keywords" content="">
<meta name="Description" content="">
<style type="text/css">
body, h1{margin:0;}
canvas{margin: 20px; }
</style>
</head>
<body>
<h1>二次貝塞爾曲線</h1>
<canvas id="canvas" width=600 height=600 style="border: 1px solid #ccc;"></canvas>
<script>
/**
* @param sx 起始點(diǎn)x坐標(biāo)
* @param sy 起始點(diǎn)y坐標(biāo)
* @param ex 結(jié)束點(diǎn)x坐標(biāo)
* @param ey 結(jié)束點(diǎn)y坐標(biāo)
* @param cx 控制點(diǎn)x坐標(biāo)
* @param cy 控制點(diǎn)y坐標(biāo)
* @param part 將起始點(diǎn)到控制點(diǎn)的線段分成的份數(shù),數(shù)值越高,計(jì)算出的曲線越精確
* @param interval 畫圖的間隔
* @return function 調(diào)用一次就向后畫一段曲線
*/
function draw(sx, sy, ex, ey, cx, cy, part, interval) {
var canvas = document.getElementById('canvas');
var ctx = canvas.getContext('2d');
//繪制起始點(diǎn)、控制點(diǎn)、終點(diǎn)
ctx.beginPath();
ctx.moveTo(sx, sy);
ctx.lineTo(cx, cy);
ctx.lineTo(ex, ey);
ctx.stroke();
// 繪制二次貝塞爾曲線
ctx.beginPath();
ctx.moveTo(sx, sy);
// 起始點(diǎn)到控制點(diǎn)的x和y每次的增量
var changeX1 = (cx - sx) / part;
var changeY1 = (cy - sy) / part;
// 控制點(diǎn)到結(jié)束點(diǎn)的x和y每次的增量
var changeX2 = (ex - cx) / part;
var changeY2 = (ey - cy) / part;
// 上次的點(diǎn)坐標(biāo)
var lastX = sx;
var lastY = sy;
var i = 0;
return function () {
// 計(jì)算兩個(gè)動(dòng)點(diǎn)的坐標(biāo)
var qx1 = sx + changeX1 * i;
var qy1 = sy + changeY1 * i;
var qx2 = cx + changeX2 * i;
var qy2 = cy + changeY2 * i;
// 計(jì)算得到此時(shí)的一個(gè)貝塞爾曲線上的點(diǎn)
var bx = qx1 + (qx2 - qx1) * i / part;
var by = qy1 + (qy2 - qy1) * i / part;
// 從上次的點(diǎn)繼續(xù)畫
ctx.beginPath();
ctx.moveTo(lastX, lastY);
ctx.lineTo(bx, by);
ctx.stroke();
// 保存點(diǎn)坐標(biāo)
lastX = bx;
lastY = by;
i += 1;
if (i < part) {
setTimeout(arguments.callee, interval);
}
}
}
window.onload = function () {
var display = draw(0, 0, 600, 0, 150, 450, 200, 50);
display();
};
</script>
</body>
</html>
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
基于javascript實(shí)現(xiàn)表格的簡(jiǎn)單操作
這篇文章主要為大家詳細(xì)介紹了基于javascript實(shí)現(xiàn)表格的簡(jiǎn)單操作,具有一定的參考價(jià)值,感興趣的朋友可以參考一下2016-05-05
JS實(shí)現(xiàn)圖片輪播效果實(shí)例詳解【可自動(dòng)和手動(dòng)】
這篇文章主要介紹了JS實(shí)現(xiàn)圖片輪播效果,結(jié)合完整實(shí)例形式分析了javascript可自動(dòng)和手動(dòng)輪播圖的原理、布局與輪播功能相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-04-04
JavaScript實(shí)現(xiàn)左右下拉框動(dòng)態(tài)增刪示例
本篇文章主要介紹了JavaScript實(shí)現(xiàn)左右下拉框動(dòng)態(tài)增刪示例,可以對(duì)下拉框進(jìn)行刪除和增加,非常具有實(shí)用價(jià)值,需要的朋友可以參考下。2017-03-03
js中異步函數(shù)async function變同步函數(shù)的簡(jiǎn)單入門
這篇文章主要介紹了js中異步函數(shù)async function變同步函數(shù)的簡(jiǎn)單入門,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-04-04
p5.js碼繪“跳動(dòng)的小正方形”的實(shí)現(xiàn)代碼
這篇文章主要介紹了p5.js碼繪“跳動(dòng)的小正方形”,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-10-10

