Given the end points and control points along a bezier curve, this code shows how to get any point and any angle along the curve.
Last active
May 25, 2022 17:34
-
-
Save pbeshai/72c446033a98f99ce1e1371c6eee9644 to your computer and use it in GitHub Desktop.
Compute points and angles along a bezier curve
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
license: mit | |
height: 320 | |
border: no |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
function quadraticPath(t){return"M"+t.start[0]+","+t.start[1]+" Q"+t.control[0]+","+t.control[1]+" "+t.end[0]+","+t.end[1]}function interpolateQuadraticBezier(t,r,a){return function(n){return[Math.pow(1-n,2)*t[0]+2*(1-n)*n*r[0]+Math.pow(n,2)*a[0],Math.pow(1-n,2)*t[1]+2*(1-n)*n*r[1]+Math.pow(n,2)*a[1]]}}function interpolateQuadraticBezierAngle(t,r,a){return function(n){var e=2*(1-n)*(r[0]-t[0])+2*n*(a[0]-r[0]),o=2*(1-n)*(r[1]-t[1])+2*n*(a[1]-r[1]);return Math.atan2(o,e)*(180/Math.PI)}}function cubicPath(t){return"M"+t.start[0]+","+t.start[1]+" C"+t.control1[0]+","+t.control1[1]+" "+t.control2[0]+","+t.control2[1]+" "+t.end[0]+","+t.end[1]}function interpolateCubicBezier(t,r,a,n){return function(e){return[Math.pow(1-e,3)*t[0]+3*Math.pow(1-e,2)*e*r[0]+3*(1-e)*Math.pow(e,2)*a[0]+Math.pow(e,3)*n[0],Math.pow(1-e,3)*t[1]+3*Math.pow(1-e,2)*e*r[1]+3*(1-e)*Math.pow(e,2)*a[1]+Math.pow(e,3)*n[1]]}}function interpolateCubicBezierAngle(t,r,a,n){return function(e){var o=3*Math.pow(1-e,2)*(r[0]-t[0])+6*(1-e)*e*(a[0]-r[0])+3*Math.pow(e,2)*(n[0]-a[0]),c=3*Math.pow(1-e,2)*(r[1]-t[1])+6*(1-e)*e*(a[1]-r[1])+3*Math.pow(e,2)*(n[1]-a[1]);return Math.atan2(c,o)*(180/Math.PI)}}function drawQuadratic(t){var r=g.append("g").attr("class","quadratic");r.append("text").text("Quadratic Bezier").attr("dy","0.4em"),r.append("circle").attr("r",5).attr("class","start-point").attr("cx",t.start[0]).attr("cy",t.start[1]),r.append("circle").attr("r",5).attr("class","end-point").attr("cx",t.end[0]).attr("cy",t.end[1]),r.append("circle").attr("r",3).attr("class","control-point").attr("cx",t.control[0]).attr("cy",t.control[1]),r.append("path").attr("class","curve").attr("d",quadraticPath(t));var a=interpolateQuadraticBezier(t.start,t.control,t.end),n=d3.range(10).map(function(t,r,n){return a(t/(n.length-1))});r.selectAll(".interpolated-point").data(n).enter().append("circle").attr("class","interpolated-point").attr("r",3).attr("cx",function(t){return t[0]}).attr("cy",function(t){return t[1]});var e=interpolateQuadraticBezierAngle(t.start,t.control,t.end),o=d3.range(3).map(function(t,r,n){var o=t/(n.length-1);return{t:o,position:a(o),angle:e(o)}});return r.selectAll(".rotated-point").data(o).enter().append("path").attr("d","M12,0 L-5,-8 L0,0 L-5,8 Z").attr("class","rotated-point").attr("transform",function(t){return"translate("+t.position[0]+", "+t.position[1]+") rotate("+t.angle+")"}),console.log("QUADRATIC"),console.log(JSON.stringify(t)),console.log(JSON.stringify(o,null,2)),r}function drawCubic(t){var r=g.append("g").attr("class","cubic").attr("transform","translate(250, 0)");r.append("text").text("Cubic Bezier").attr("dy","0.4em"),r.append("circle").attr("r",5).attr("class","start-point").attr("cx",t.start[0]).attr("cy",t.start[1]),r.append("circle").attr("r",5).attr("class","end-point").attr("cx",t.end[0]).attr("cy",t.end[1]),r.append("circle").attr("r",3).attr("class","control-point").attr("cx",t.control1[0]).attr("cy",t.control1[1]),r.append("circle").attr("r",3).attr("class","control-point").attr("cx",t.control2[0]).attr("cy",t.control2[1]),r.append("path").attr("class","curve").attr("d",cubicPath(t));var a=interpolateCubicBezier(t.start,t.control1,t.control2,t.end),n=d3.range(10).map(function(t,r,n){return a(t/(n.length-1))});r.selectAll(".interpolated-point").data(n).enter().append("circle").attr("class","interpolated-point").attr("r",3).attr("cx",function(t){return t[0]}).attr("cy",function(t){return t[1]});var e=interpolateCubicBezierAngle(t.start,t.control1,t.control2,t.end),o=d3.range(3).map(function(t,r,n){var o=t/(n.length-1);return{t:o,position:a(o),angle:e(o)}});return r.selectAll(".rotated-point").data(o).enter().append("path").attr("d","M12,0 L-5,-8 L0,0 L-5,8 Z").attr("class","rotated-point").attr("transform",function(t){return"translate("+t.position[0]+", "+t.position[1]+") rotate("+t.angle+")"}),console.log("CUBIC"),console.log(JSON.stringify(t)),console.log(JSON.stringify(o,null,2)),r}var width=500,height=300,padding={top:10,right:10,bottom:10,left:10},svg=d3.select("body").append("svg").attr("width",width).attr("height",height),g=svg.append("g").attr("transform","translate("+padding.left+", "+padding.top+")");drawQuadratic({start:[0,70],end:[200,100],control:[160,20]}),drawCubic({start:[0,70],end:[200,100],control1:[40,180],control2:[160,20]}); | |
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["script.js"],"names":["quadraticPath","q","start","control","end","interpolateQuadraticBezier","t","Math","pow","interpolateQuadraticBezierAngle","const","tangentX","tangentY","atan2","PI","cubicPath","c","control1","control2","interpolateCubicBezier","interpolateCubicBezierAngle","drawQuadratic","quadratic","gQuadratic","g","append","attr","text","quadraticInterpolator","interpolatedPoints","d3","range","map","d","i","a","length","selectAll","data","enter","quadraticAngleInterpolator","rotatedPoints","position","angle","console","log","JSON","stringify","drawCubic","cubic","gCubic","cubicInterpolator","cubicAngleInterpolator","width","height","padding","top","right","bottom","left","svg","select"],"mappings":"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","file":"script.js","sourcesContent":["\nconst width = 500;\nconst height = 300;\nconst padding = { top: 10, right: 10, bottom: 10, left: 10 };\n\n\nconst svg = d3.select('body').append('svg')\n\t\t.attr('width', width)\n\t\t.attr('height', height);\n\nconst g = svg.append('g')\n\t\t.attr('transform', `translate(${padding.left}, ${padding.top})`)\n\n\nfunction quadraticPath(q) {\n\treturn `M${q.start[0]},${q.start[1]} Q${q.control[0]},${q.control[1]} ${q.end[0]},${q.end[1]}`;\n}\n\n// B(t) = (1 - t)^2P0 + 2(1 - t)tP1 + t^2P2\nfunction interpolateQuadraticBezier(start, control, end) {\n\t// 0 <= t <= 1\n\treturn function interpolator(t) {\n\t\treturn [\n\t\t\t(Math.pow(1 - t, 2) * start[0]) +\n      (2 * (1 - t) * t * control[0]) +\n      (Math.pow(t, 2) * end[0]),\n      (Math.pow(1 - t, 2) * start[1]) +\n      (2 * (1 - t) * t * control[1]) +\n      (Math.pow(t, 2) * end[1]),\n\t\t];\n\t};\n}\n\n// B'(t) = 2(1 - t)(P1 - P0) + 2t(P2 - P1)\nfunction interpolateQuadraticBezierAngle(start, control, end) {\n\t// 0 <= t <= 1\n\treturn function interpolator(t) {\n\t\tconst tangentX = (2 * (1 - t) * (control[0] - start[0])) +\n\t\t\t\t\t\t\t\t\t\t (2 * t * (end[0] - control[0]));\n\t\tconst tangentY = (2 * (1 - t) * (control[1] - start[1])) +\n\t\t\t\t\t\t\t\t\t\t (2 * t * (end[1] - control[1]));\n\n\t\treturn Math.atan2(tangentY, tangentX) * (180 / Math.PI);\n\t}\n}\n\nfunction cubicPath(c) {\n\treturn `M${c.start[0]},${c.start[1]} C${c.control1[0]},${c.control1[1]} ${c.control2[0]},${c.control2[1]} ${c.end[0]},${c.end[1]}`;\n}\n\n// B(t) = (1 - t)^3P0 + 3(1 - t)^2tP1 + 3(1 - t)t^2P2 + t^3P3\nfunction interpolateCubicBezier(start, control1, control2, end) {\n\t// 0 <= t <= 1\n\treturn function interpolator(t) {\n\t\treturn [\n      (Math.pow(1 - t, 3) * start[0]) +\n      (3 * Math.pow(1 - t, 2) * t * control1[0]) +\n      (3 * (1 - t) * Math.pow(t, 2) * control2[0]) +\n\t\t\t(Math.pow(t, 3) * end[0]),\n\t\t\t(Math.pow(1 - t, 3) * start[1]) +\n      (3 * Math.pow(1 - t, 2) * t * control1[1]) +\n      (3 * (1 - t) * Math.pow(t, 2) * control2[1]) +\n\t\t\t(Math.pow(t, 3) * end[1]),\n\t\t];\n\t};\n}\n\n// B'(t) = 3(1- t)^2(P1 - P0) + 6(1 - t)t(P2 - P1) + 3t^2(P3 - P2)\nfunction interpolateCubicBezierAngle(start, control1, control2, end) {\n\t// 0 <= t <= 1\n\treturn function interpolator(t) {\n\t\tconst tangentX =  (3 * Math.pow(1 - t, 2) * (control1[0] - start[0])) +\n\t\t\t\t\t\t\t\t\t\t\t(6 * (1 - t) * t * (control2[0] - control1[0])) +\n\t\t\t\t\t\t\t\t\t\t\t(3 * Math.pow(t, 2) * (end[0] - control2[0]));\n\t\tconst tangentY =  (3 * Math.pow(1 - t, 2) * (control1[1] - start[1])) +\n\t\t\t\t\t\t\t\t\t\t\t(6 * (1 - t) * t * (control2[1] - control1[1])) +\n\t\t\t\t\t\t\t\t\t\t\t(3 * Math.pow(t, 2) * (end[1] - control2[1]));\n\n\t\treturn Math.atan2(tangentY, tangentX) * (180 / Math.PI);\n\t}\n}\n\n\n// draw a quadratic bezier curve\nfunction drawQuadratic(quadratic) {\n\tconst gQuadratic = g.append('g')\n\t\t.attr('class', 'quadratic');\n\n\tgQuadratic.append('text')\n\t\t.text('Quadratic Bezier')\n\t\t.attr('dy', '0.4em')\n\n\t// draw the points\n\tgQuadratic.append('circle')\n\t\t.attr('r', 5)\n\t\t.attr('class', 'start-point')\n\t\t.attr('cx', quadratic.start[0])\n\t\t.attr('cy', quadratic.start[1])\n\n\tgQuadratic.append('circle')\n\t\t.attr('r', 5)\n\t\t.attr('class', 'end-point')\n\t\t.attr('cx', quadratic.end[0])\n\t\t.attr('cy', quadratic.end[1])\n\n\tgQuadratic.append('circle')\n\t\t.attr('r', 3)\n\t\t.attr('class', 'control-point')\n\t\t.attr('cx', quadratic.control[0])\n\t\t.attr('cy', quadratic.control[1])\n\n\t// draw the path\n\tgQuadratic.append('path')\n\t\t.attr('class', 'curve')\n\t\t.attr('d', quadraticPath(quadratic));\n\n\tconst quadraticInterpolator = interpolateQuadraticBezier(quadratic.start, quadratic.control, quadratic.end);\n\tconst interpolatedPoints = d3.range(10).map((d, i, a) => quadraticInterpolator(d / (a.length - 1)));\n\n\tgQuadratic.selectAll('.interpolated-point').data(interpolatedPoints)\n\t\t.enter()\n\t\t\t.append('circle')\n\t\t\t\t.attr('class', 'interpolated-point')\n\t\t\t\t.attr('r', 3)\n\t\t\t\t.attr('cx', d => d[0])\n\t\t\t\t.attr('cy', d => d[1]);\n\n\n\tconst quadraticAngleInterpolator = interpolateQuadraticBezierAngle(quadratic.start, quadratic.control, quadratic.end);\n\n\tconst rotatedPoints = d3.range(3).map((d, i, a) => {\n\t\tconst t = d / (a.length - 1);\n\t\treturn {\n\t\t\tt: t,\n\t\t\tposition: quadraticInterpolator(t),\n\t\t\tangle: quadraticAngleInterpolator(t),\n\t\t};\n\t});\n\n\tgQuadratic.selectAll('.rotated-point').data(rotatedPoints)\n\t\t.enter()\n\t\t\t.append('path')\n\t\t\t\t.attr('d', 'M12,0 L-5,-8 L0,0 L-5,8 Z')\n\t\t\t\t.attr('class', 'rotated-point')\n\t\t\t\t.attr('transform', d => `translate(${d.position[0]}, ${d.position[1]}) rotate(${d.angle})`)\n\n\n\tconsole.log('QUADRATIC');\n\tconsole.log(JSON.stringify(quadratic));\n\tconsole.log(JSON.stringify(rotatedPoints, null, 2))\n\n\treturn gQuadratic;\n}\n\n\nfunction drawCubic(cubic) {\n\tconst gCubic = g.append('g')\n\t\t.attr('class', 'cubic')\n\t\t.attr('transform', 'translate(250, 0)');\n\n\tgCubic.append('text')\n\t\t.text('Cubic Bezier')\n\t\t.attr('dy', '0.4em')\n\n\t// draw the points\n\tgCubic.append('circle')\n\t\t.attr('r', 5)\n\t\t.attr('class', 'start-point')\n\t\t.attr('cx', cubic.start[0])\n\t\t.attr('cy', cubic.start[1])\n\n\tgCubic.append('circle')\n\t\t.attr('r', 5)\n\t\t.attr('class', 'end-point')\n\t\t.attr('cx', cubic.end[0])\n\t\t.attr('cy', cubic.end[1])\n\n\tgCubic.append('circle')\n\t\t.attr('r', 3)\n\t\t.attr('class', 'control-point')\n\t\t.attr('cx', cubic.control1[0])\n\t\t.attr('cy', cubic.control1[1])\n\n\tgCubic.append('circle')\n\t\t.attr('r', 3)\n\t\t.attr('class', 'control-point')\n\t\t.attr('cx', cubic.control2[0])\n\t\t.attr('cy', cubic.control2[1])\n\n\t// draw the path\n\tgCubic.append('path')\n\t\t.attr('class', 'curve')\n\t\t.attr('d', cubicPath(cubic));\n\n\tconst cubicInterpolator = interpolateCubicBezier(cubic.start, cubic.control1, cubic.control2, cubic.end);\n\tconst interpolatedPoints = d3.range(10).map((d, i, a) => cubicInterpolator(d / (a.length - 1)));\n\n\tgCubic.selectAll('.interpolated-point').data(interpolatedPoints)\n\t\t.enter()\n\t\t\t.append('circle')\n\t\t\t\t.attr('class', 'interpolated-point')\n\t\t\t\t.attr('r', 3)\n\t\t\t\t.attr('cx', d => d[0])\n\t\t\t\t.attr('cy', d => d[1]);\n\n\n\tconst cubicAngleInterpolator = interpolateCubicBezierAngle(cubic.start, cubic.control1, cubic.control2, cubic.end);\n\n\tconst rotatedPoints = d3.range(3).map((d, i, a) => {\n\t\tconst t = d / (a.length - 1);\n\t\treturn {\n\t\t\tt: t,\n\t\t\tposition: cubicInterpolator(t),\n\t\t\tangle: cubicAngleInterpolator(t),\n\t\t};\n\t});\n\n\tgCubic.selectAll('.rotated-point').data(rotatedPoints)\n\t\t.enter()\n\t\t\t.append('path')\n\t\t\t\t.attr('d', 'M12,0 L-5,-8 L0,0 L-5,8 Z')\n\t\t\t\t.attr('class', 'rotated-point')\n\t\t\t\t.attr('transform', d => `translate(${d.position[0]}, ${d.position[1]}) rotate(${d.angle})`)\n\n\n\tconsole.log('CUBIC');\n\tconsole.log(JSON.stringify(cubic));\n\tconsole.log(JSON.stringify(rotatedPoints, null, 2))\n\n\treturn gCubic;\n}\n\n\nconst quadratic = { start: [0, 70], end: [200, 100], control: [160, 20] };\ndrawQuadratic(quadratic);\n\nconst cubic = { start: [0, 70], end: [200, 100], control1: [40, 180], control2: [160, 20] };\ndrawCubic(cubic);\n"]} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<title>Compute points and angles along a bezier curve</title> | |
<link href='style.css' rel='stylesheet' /> | |
<body> | |
<script src='https://d3js.org/d3.v4.min.js'></script> | |
<script src='dist.js'></script> | |
</body> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
const width = 500; | |
const height = 300; | |
const padding = { top: 10, right: 10, bottom: 10, left: 10 }; | |
const svg = d3.select('body').append('svg') | |
.attr('width', width) | |
.attr('height', height); | |
const g = svg.append('g') | |
.attr('transform', `translate(${padding.left}, ${padding.top})`) | |
function quadraticPath(q) { | |
return `M${q.start[0]},${q.start[1]} Q${q.control[0]},${q.control[1]} ${q.end[0]},${q.end[1]}`; | |
} | |
// B(t) = (1 - t)^2P0 + 2(1 - t)tP1 + t^2P2 | |
function interpolateQuadraticBezier(start, control, end) { | |
// 0 <= t <= 1 | |
return function interpolator(t) { | |
return [ | |
(Math.pow(1 - t, 2) * start[0]) + | |
(2 * (1 - t) * t * control[0]) + | |
(Math.pow(t, 2) * end[0]), | |
(Math.pow(1 - t, 2) * start[1]) + | |
(2 * (1 - t) * t * control[1]) + | |
(Math.pow(t, 2) * end[1]), | |
]; | |
}; | |
} | |
// B'(t) = 2(1 - t)(P1 - P0) + 2t(P2 - P1) | |
function interpolateQuadraticBezierAngle(start, control, end) { | |
// 0 <= t <= 1 | |
return function interpolator(t) { | |
const tangentX = (2 * (1 - t) * (control[0] - start[0])) + | |
(2 * t * (end[0] - control[0])); | |
const tangentY = (2 * (1 - t) * (control[1] - start[1])) + | |
(2 * t * (end[1] - control[1])); | |
return Math.atan2(tangentY, tangentX) * (180 / Math.PI); | |
} | |
} | |
function cubicPath(c) { | |
return `M${c.start[0]},${c.start[1]} C${c.control1[0]},${c.control1[1]} ${c.control2[0]},${c.control2[1]} ${c.end[0]},${c.end[1]}`; | |
} | |
// B(t) = (1 - t)^3P0 + 3(1 - t)^2tP1 + 3(1 - t)t^2P2 + t^3P3 | |
function interpolateCubicBezier(start, control1, control2, end) { | |
// 0 <= t <= 1 | |
return function interpolator(t) { | |
return [ | |
(Math.pow(1 - t, 3) * start[0]) + | |
(3 * Math.pow(1 - t, 2) * t * control1[0]) + | |
(3 * (1 - t) * Math.pow(t, 2) * control2[0]) + | |
(Math.pow(t, 3) * end[0]), | |
(Math.pow(1 - t, 3) * start[1]) + | |
(3 * Math.pow(1 - t, 2) * t * control1[1]) + | |
(3 * (1 - t) * Math.pow(t, 2) * control2[1]) + | |
(Math.pow(t, 3) * end[1]), | |
]; | |
}; | |
} | |
// B'(t) = 3(1- t)^2(P1 - P0) + 6(1 - t)t(P2 - P1) + 3t^2(P3 - P2) | |
function interpolateCubicBezierAngle(start, control1, control2, end) { | |
// 0 <= t <= 1 | |
return function interpolator(t) { | |
const tangentX = (3 * Math.pow(1 - t, 2) * (control1[0] - start[0])) + | |
(6 * (1 - t) * t * (control2[0] - control1[0])) + | |
(3 * Math.pow(t, 2) * (end[0] - control2[0])); | |
const tangentY = (3 * Math.pow(1 - t, 2) * (control1[1] - start[1])) + | |
(6 * (1 - t) * t * (control2[1] - control1[1])) + | |
(3 * Math.pow(t, 2) * (end[1] - control2[1])); | |
return Math.atan2(tangentY, tangentX) * (180 / Math.PI); | |
} | |
} | |
// draw a quadratic bezier curve | |
function drawQuadratic(quadratic) { | |
const gQuadratic = g.append('g') | |
.attr('class', 'quadratic'); | |
gQuadratic.append('text') | |
.text('Quadratic Bezier') | |
.attr('dy', '0.4em') | |
// draw the points | |
gQuadratic.append('circle') | |
.attr('r', 5) | |
.attr('class', 'start-point') | |
.attr('cx', quadratic.start[0]) | |
.attr('cy', quadratic.start[1]) | |
gQuadratic.append('circle') | |
.attr('r', 5) | |
.attr('class', 'end-point') | |
.attr('cx', quadratic.end[0]) | |
.attr('cy', quadratic.end[1]) | |
gQuadratic.append('circle') | |
.attr('r', 3) | |
.attr('class', 'control-point') | |
.attr('cx', quadratic.control[0]) | |
.attr('cy', quadratic.control[1]) | |
// draw the path | |
gQuadratic.append('path') | |
.attr('class', 'curve') | |
.attr('d', quadraticPath(quadratic)); | |
const quadraticInterpolator = interpolateQuadraticBezier(quadratic.start, quadratic.control, quadratic.end); | |
const interpolatedPoints = d3.range(10).map((d, i, a) => quadraticInterpolator(d / (a.length - 1))); | |
gQuadratic.selectAll('.interpolated-point').data(interpolatedPoints) | |
.enter() | |
.append('circle') | |
.attr('class', 'interpolated-point') | |
.attr('r', 3) | |
.attr('cx', d => d[0]) | |
.attr('cy', d => d[1]); | |
const quadraticAngleInterpolator = interpolateQuadraticBezierAngle(quadratic.start, quadratic.control, quadratic.end); | |
const rotatedPoints = d3.range(3).map((d, i, a) => { | |
const t = d / (a.length - 1); | |
return { | |
t: t, | |
position: quadraticInterpolator(t), | |
angle: quadraticAngleInterpolator(t), | |
}; | |
}); | |
gQuadratic.selectAll('.rotated-point').data(rotatedPoints) | |
.enter() | |
.append('path') | |
.attr('d', 'M12,0 L-5,-8 L0,0 L-5,8 Z') | |
.attr('class', 'rotated-point') | |
.attr('transform', d => `translate(${d.position[0]}, ${d.position[1]}) rotate(${d.angle})`) | |
console.log('QUADRATIC'); | |
console.log(JSON.stringify(quadratic)); | |
console.log(JSON.stringify(rotatedPoints, null, 2)) | |
return gQuadratic; | |
} | |
function drawCubic(cubic) { | |
const gCubic = g.append('g') | |
.attr('class', 'cubic') | |
.attr('transform', 'translate(250, 0)'); | |
gCubic.append('text') | |
.text('Cubic Bezier') | |
.attr('dy', '0.4em') | |
// draw the points | |
gCubic.append('circle') | |
.attr('r', 5) | |
.attr('class', 'start-point') | |
.attr('cx', cubic.start[0]) | |
.attr('cy', cubic.start[1]) | |
gCubic.append('circle') | |
.attr('r', 5) | |
.attr('class', 'end-point') | |
.attr('cx', cubic.end[0]) | |
.attr('cy', cubic.end[1]) | |
gCubic.append('circle') | |
.attr('r', 3) | |
.attr('class', 'control-point') | |
.attr('cx', cubic.control1[0]) | |
.attr('cy', cubic.control1[1]) | |
gCubic.append('circle') | |
.attr('r', 3) | |
.attr('class', 'control-point') | |
.attr('cx', cubic.control2[0]) | |
.attr('cy', cubic.control2[1]) | |
// draw the path | |
gCubic.append('path') | |
.attr('class', 'curve') | |
.attr('d', cubicPath(cubic)); | |
const cubicInterpolator = interpolateCubicBezier(cubic.start, cubic.control1, cubic.control2, cubic.end); | |
const interpolatedPoints = d3.range(10).map((d, i, a) => cubicInterpolator(d / (a.length - 1))); | |
gCubic.selectAll('.interpolated-point').data(interpolatedPoints) | |
.enter() | |
.append('circle') | |
.attr('class', 'interpolated-point') | |
.attr('r', 3) | |
.attr('cx', d => d[0]) | |
.attr('cy', d => d[1]); | |
const cubicAngleInterpolator = interpolateCubicBezierAngle(cubic.start, cubic.control1, cubic.control2, cubic.end); | |
const rotatedPoints = d3.range(3).map((d, i, a) => { | |
const t = d / (a.length - 1); | |
return { | |
t: t, | |
position: cubicInterpolator(t), | |
angle: cubicAngleInterpolator(t), | |
}; | |
}); | |
gCubic.selectAll('.rotated-point').data(rotatedPoints) | |
.enter() | |
.append('path') | |
.attr('d', 'M12,0 L-5,-8 L0,0 L-5,8 Z') | |
.attr('class', 'rotated-point') | |
.attr('transform', d => `translate(${d.position[0]}, ${d.position[1]}) rotate(${d.angle})`) | |
console.log('CUBIC'); | |
console.log(JSON.stringify(cubic)); | |
console.log(JSON.stringify(rotatedPoints, null, 2)) | |
return gCubic; | |
} | |
const quadratic = { start: [0, 70], end: [200, 100], control: [160, 20] }; | |
drawQuadratic(quadratic); | |
const cubic = { start: [0, 70], end: [200, 100], control1: [40, 180], control2: [160, 20] }; | |
drawCubic(cubic); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
.start-point { | |
fill: #fff; | |
stroke: #0bb; | |
} | |
.end-point { | |
fill: #fff; | |
stroke: #0bb; | |
} | |
.control-point { | |
fill: tomato; | |
} | |
.curve { | |
fill: none; | |
stroke: #0bb; | |
stroke-width: 2px; | |
} | |
.interpolated-point { | |
fill: #077; | |
} | |
.rotated-point { | |
fill: #af84e6; | |
stroke: #660cd0; | |
} |
Awesome, thanks for sharing!
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Thank you very much for the good demo. You solution helped to implement proper rotation (orientation) of arrowheads with reactflow.dev library.