Fixed arcTo implementation. Previously it only worked in the orientation in the test, but not for lines curving in other directions.
This commit is contained in:
parent
7fe0aeaaea
commit
9d7c7bae3e
13
README.md
13
README.md
@ -80,6 +80,19 @@ const mySerializedSVG = ctx.getSerializedSvg();
|
|||||||
|
|
||||||
https://zenozeng.github.io/p5.js-svg/test/
|
https://zenozeng.github.io/p5.js-svg/test/
|
||||||
|
|
||||||
|
To run the testsuite:
|
||||||
|
|
||||||
|
```
|
||||||
|
npm run test
|
||||||
|
```
|
||||||
|
|
||||||
|
To debug tests in a browser:
|
||||||
|
|
||||||
|
```
|
||||||
|
open test/index.html
|
||||||
|
npm run watch
|
||||||
|
```
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
This library is licensed under the MIT license.
|
This library is licensed under the MIT license.
|
||||||
|
2
package-lock.json
generated
2
package-lock.json
generated
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@aha-app/svgcanvas",
|
"name": "@aha-app/svgcanvas",
|
||||||
"version": "2.5.0-a11",
|
"version": "2.5.0-a12",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@aha-app/svgcanvas",
|
"name": "@aha-app/svgcanvas",
|
||||||
"version": "2.5.0-a12",
|
"version": "2.5.0-a13",
|
||||||
"description": "svgcanvas",
|
"description": "svgcanvas",
|
||||||
"main": "dist/svgcanvas.js",
|
"main": "dist/svgcanvas.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
106
path2d.js
106
path2d.js
@ -201,7 +201,8 @@ export default (function () {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds the arcTo to the current path
|
* Adds the arcTo to the current path. Based on Webkit implementation from
|
||||||
|
* https://github.com/WebKit/webkit/blob/main/Source/WebCore/platform/graphics/cairo/PathCairo.cpp
|
||||||
*
|
*
|
||||||
* @see http://www.w3.org/TR/2015/WD-2dcontext-20150514/#dom-context-2d-arcto
|
* @see http://www.w3.org/TR/2015/WD-2dcontext-20150514/#dom-context-2d-arcto
|
||||||
*/
|
*/
|
||||||
@ -232,67 +233,66 @@ export default (function () {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, if the points (x0, y0), (x1, y1), and (x2, y2) all lie on a single straight line,
|
const p1p0 = [x0 - x1, y0 - y1];
|
||||||
// then the method must add the point (x1, y1) to the subpath,
|
const p1p2 = [x2 - x1, y2 - y1];
|
||||||
// and connect that point to the previous point (x0, y0) by a straight line.
|
const p1p0_length = Math.hypot(p1p0[0], p1p0[1]);
|
||||||
var unit_vec_p1_p0 = normalize([x0 - x1, y0 - y1]);
|
const p1p2_length = Math.hypot(p1p2[0], p1p2[1]);
|
||||||
var unit_vec_p1_p2 = normalize([x2 - x1, y2 - y1]);
|
const cos_phi = (p1p0[0] * p1p2[0] + p1p0[1] * p1p2[1]) / (p1p0_length * p1p2_length);
|
||||||
if (
|
// all points on a line logic
|
||||||
unit_vec_p1_p0[0] * unit_vec_p1_p2[1] ===
|
if (cos_phi == -1) {
|
||||||
unit_vec_p1_p0[1] * unit_vec_p1_p2[0]
|
|
||||||
) {
|
|
||||||
this.lineTo(x1, y1);
|
this.lineTo(x1, y1);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
if (cos_phi == 1) {
|
||||||
|
// add infinite far away point
|
||||||
|
const max_length = 65535;
|
||||||
|
const factor_max = max_length / p1p0_length;
|
||||||
|
const ep = [xp0 + factor_max * p1p0[0], y0 + factor_max * p1p0[1]];
|
||||||
|
this.lineTo(ep[0], ep[1]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Otherwise, let The Arc be the shortest arc given by circumference of the circle that has radius radius,
|
const tangent = radius / Math.tan(Math.acos(cos_phi) / 2);
|
||||||
// and that has one point tangent to the half-infinite line that crosses the point (x0, y0) and ends at the point (x1, y1),
|
const factor_p1p0 = tangent / p1p0_length;
|
||||||
// and that has a different point tangent to the half-infinite line that ends at the point (x1, y1), and crosses the point (x2, y2).
|
const t_p1p0 = [x1 + factor_p1p0 * p1p0[0], y1 + factor_p1p0 * p1p0[1]];
|
||||||
// The points at which this circle touches these two lines are called the start and end tangent points respectively.
|
|
||||||
|
|
||||||
// note that both vectors are unit vectors, so the length is 1
|
let orth_p1p0 = [p1p0[1], -p1p0[0]];
|
||||||
var cos =
|
const orth_p1p0_length = Math.hypot(orth_p1p0[0], orth_p1p0[1]);
|
||||||
unit_vec_p1_p0[0] * unit_vec_p1_p2[0] +
|
const factor_ra = radius / orth_p1p0_length;
|
||||||
unit_vec_p1_p0[1] * unit_vec_p1_p2[1];
|
|
||||||
var theta = Math.acos(Math.abs(cos));
|
|
||||||
|
|
||||||
// Calculate origin
|
// angle between orth_p1p0 and p1p2 to get the right vector orthographic to p1p0
|
||||||
var unit_vec_p1_origin = normalize([
|
const cos_alpha = (orth_p1p0[0] * p1p2[0] + orth_p1p0[1] * p1p2[1]) / (orth_p1p0_length * p1p2_length);
|
||||||
unit_vec_p1_p0[0] + unit_vec_p1_p2[0],
|
if (cos_alpha < 0) {
|
||||||
unit_vec_p1_p0[1] + unit_vec_p1_p2[1],
|
orth_p1p0 = [-orth_p1p0[0], -orth_p1p0[1]];
|
||||||
]);
|
}
|
||||||
var len_p1_origin = radius / Math.sin(theta / 2);
|
|
||||||
var x = x1 + len_p1_origin * unit_vec_p1_origin[0];
|
|
||||||
var y = y1 + len_p1_origin * unit_vec_p1_origin[1];
|
|
||||||
|
|
||||||
// Calculate start angle and end angle
|
const p = [t_p1p0[0] + factor_ra * orth_p1p0[0], t_p1p0[1] + factor_ra * orth_p1p0[1]];
|
||||||
// rotate 90deg clockwise (note that y axis points to its down)
|
|
||||||
var unit_vec_origin_start_tangent = [-unit_vec_p1_p0[1], unit_vec_p1_p0[0]];
|
|
||||||
// rotate 90deg counter clockwise (note that y axis points to its down)
|
|
||||||
var unit_vec_origin_end_tangent = [unit_vec_p1_p2[1], -unit_vec_p1_p2[0]];
|
|
||||||
var getAngle = function (vector) {
|
|
||||||
// get angle (clockwise) between vector and (1, 0)
|
|
||||||
var x = vector[0];
|
|
||||||
var y = vector[1];
|
|
||||||
if (y >= 0) {
|
|
||||||
// note that y axis points to its down
|
|
||||||
return Math.acos(x);
|
|
||||||
} else {
|
|
||||||
return -Math.acos(x);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
var startAngle = getAngle(unit_vec_origin_start_tangent);
|
|
||||||
var endAngle = getAngle(unit_vec_origin_end_tangent);
|
|
||||||
|
|
||||||
// Connect the point (x0, y0) to the start tangent point by a straight line
|
// calculate angles for addArc
|
||||||
this.lineTo(
|
orth_p1p0 = [-orth_p1p0[0], -orth_p1p0[1]];
|
||||||
x + unit_vec_origin_start_tangent[0] * radius,
|
let sa = Math.acos(orth_p1p0[0] / orth_p1p0_length);
|
||||||
y + unit_vec_origin_start_tangent[1] * radius
|
if (orth_p1p0[1] < 0) {
|
||||||
);
|
sa = 2 * Math.PI - sa;
|
||||||
|
}
|
||||||
|
|
||||||
// Connect the start tangent point to the end tangent point by arc
|
// anticlockwise logic
|
||||||
// and adding the end tangent point to the subpath.
|
let anticlockwise = false;
|
||||||
this.arc(x, y, radius, startAngle, endAngle);
|
|
||||||
|
const factor_p1p2 = tangent / p1p2_length;
|
||||||
|
const t_p1p2 = [x1 + factor_p1p2 * p1p2[0], y1 + factor_p1p2 * p1p2[1]];
|
||||||
|
const orth_p1p2 = [t_p1p2[0] - p[0], t_p1p2[1] - p[1]];
|
||||||
|
const orth_p1p2_length = Math.hypot(orth_p1p2[0], orth_p1p2[1]);
|
||||||
|
let ea = Math.acos(orth_p1p2[0] / orth_p1p2_length);
|
||||||
|
if (orth_p1p2[1] < 0) {
|
||||||
|
ea = 2 * Math.PI - ea;
|
||||||
|
}
|
||||||
|
if (sa > ea && sa - ea < Math.PI)
|
||||||
|
anticlockwise = true;
|
||||||
|
if (sa < ea && ea - sa > Math.PI)
|
||||||
|
anticlockwise = true;
|
||||||
|
|
||||||
|
this.lineTo(t_p1p0[0], t_p1p0[1])
|
||||||
|
this.arc(p[0], p[1], radius, sa, ea, anticlockwise)
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1,16 +1,30 @@
|
|||||||
export default function arcTo(ctx) {
|
export default function arcTo(ctx) {
|
||||||
ctx.beginPath();
|
ctx.beginPath();
|
||||||
ctx.moveTo(150, 20);
|
ctx.moveTo(150, 20);
|
||||||
ctx.arcTo(150, 100, 50, 20, 30);
|
ctx.arcTo(150, 100, 250, 20, 20);
|
||||||
|
ctx.stroke();
|
||||||
|
|
||||||
|
ctx.beginPath();
|
||||||
|
ctx.arc(450, 100, 20, 180/180*Math.PI, 45/180*Math.PI, true);
|
||||||
ctx.stroke();
|
ctx.stroke();
|
||||||
|
|
||||||
ctx.fillStyle = 'blue';
|
ctx.fillStyle = 'blue';
|
||||||
// base point
|
// base point
|
||||||
ctx.fillRect(150, 20, 10, 10);
|
ctx.fillRect(150, 20, 2, 2);
|
||||||
|
|
||||||
ctx.fillStyle = 'red';
|
ctx.fillStyle = 'red';
|
||||||
// control point one
|
// control point one
|
||||||
ctx.fillRect(150, 100, 10, 10);
|
ctx.fillRect(150, 100, 2, 2);
|
||||||
// control point two
|
// control point two
|
||||||
ctx.fillRect(50, 20, 10, 10);
|
ctx.fillRect(250, 20, 2, 2);
|
||||||
|
|
||||||
|
ctx.beginPath();
|
||||||
|
ctx.moveTo(150, 200);
|
||||||
|
ctx.arcTo(250, 200, 250, 250, 20);
|
||||||
|
ctx.stroke();
|
||||||
|
|
||||||
|
ctx.beginPath();
|
||||||
|
ctx.moveTo(150, 400);
|
||||||
|
ctx.arcTo(50, 400, 20, 450, 20);
|
||||||
|
ctx.stroke();
|
||||||
};
|
};
|
@ -4,4 +4,17 @@ export default function arcTo(ctx) {
|
|||||||
ctx.arcTo(300, 25, 500, 225, 75); // P1, P2 and the radius
|
ctx.arcTo(300, 25, 500, 225, 75); // P1, P2 and the radius
|
||||||
ctx.lineTo(500, 225); // P2
|
ctx.lineTo(500, 225); // P2
|
||||||
ctx.stroke();
|
ctx.stroke();
|
||||||
|
|
||||||
|
|
||||||
|
const path = [[50, 50], [50, 150], [100, 150], [100, 150], [200, 150], [200, 50], [300, 50], [300, 150]];
|
||||||
|
ctx.beginPath();
|
||||||
|
let fromPoint = path[0];
|
||||||
|
ctx.moveTo(fromPoint[0], fromPoint[1]);
|
||||||
|
for (let i = 1; i < path.length; i++) {
|
||||||
|
const point = path[i];
|
||||||
|
ctx.arcTo(fromPoint[0], fromPoint[1], point[0], point[1], 20); // P1, P2 and the radius
|
||||||
|
fromPoint = point;
|
||||||
|
}
|
||||||
|
ctx.lineTo(300, 100)
|
||||||
|
ctx.stroke();
|
||||||
};
|
};
|
Loading…
x
Reference in New Issue
Block a user