Skip to content

Commit

Permalink
feat: lines tangent to arc and ellipse arc b132b7d
Browse files Browse the repository at this point in the history
  • Loading branch information
plantain-00 committed Oct 31, 2024
1 parent 46f7a21 commit 77b1eba
Show file tree
Hide file tree
Showing 2 changed files with 96 additions and 2 deletions.
2 changes: 1 addition & 1 deletion main.bundle.js

Large diffs are not rendered by default.

96 changes: 95 additions & 1 deletion src/utils/tangent-line.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { angleInRange } from "./angle";
import { BezierCurve, getBezierCurveDerivatives, getBezierCurvePointAtPercent, getQuadraticCurveDerivatives, getQuadraticCurvePointAtPercent, QuadraticCurve } from "./bezier";
import { Arc, getCircleDerivatives } from "./circle";
import { EllipseArc, getEllipseDerivatives } from "./ellipse";
import { calculateEquation4, calculateEquation5, newtonIterate2 } from "./equation-calculater";
import { GeometryLine } from "./geometry-line";
Expand All @@ -20,6 +21,9 @@ export function getLinesTangentTo2GeometryLines(line1: GeometryLine, line2: Geom
if (line2.type === 'arc') {
return getLinesTangentTo2Circles(line1.curve, line2.curve)
}
if (line2.type === 'ellipse arc') {
return getLinesTangentToArcAndEllipseArc(line1.curve, line2.curve)
}
return []
}
if (line2.type === 'arc') return getLinesTangentTo2GeometryLines(line2, line1)
Expand All @@ -34,7 +38,7 @@ export function getLinesTangentTo2GeometryLines(line1: GeometryLine, line2: Geom
return getLinesTangentToEllipseArcAndBezierCurve(line1.curve, line2.curve)
}
if (line2.type === 'hyperbola curve') {
return []
return getLinesTangentToEllipseArcAndHyperbola(line1.curve, line2.curve)
}
return getLinesTangentToEllipseArcAndNurbsCurve(line1.curve, line2.curve)
}
Expand Down Expand Up @@ -72,6 +76,52 @@ export function getLinesTangentTo2GeometryLines(line1: GeometryLine, line2: Geom
return getLinesTangentToTwoNurbsCurves(line1.curve, line2.curve)
}

export function getLinesTangentToArcAndEllipseArc(curve1: Arc, curve2: EllipseArc): Tuple2<Position>[] {
const [p1, d1, d2] = getCircleDerivatives(curve1)
const [p2, e1, e2] = getEllipseDerivatives(curve2)
const f1 = (t: Vec2): Vec2 => {
// (y1 - y2)/(x1 - x2) = y1'/x1' = y2'/x2'
// z1 = (y1 - y2)x1' - (x1 - x2)y1'
// z2 = y1'x2' - x1'y2'
const { x: x1, y: y1 } = p1(t[0])
const { x: x11, y: y11 } = d1(t[0])
const { x: x2, y: y2 } = p2(t[1])
const { x: x21, y: y21 } = e1(t[1])
return [(y1 - y2) * x11 - (x1 - x2) * y11, y11 * x21 - x11 * y21]
}
const f2 = (t: Vec2): Matrix2 => {
const { x: x1, y: y1 } = p1(t[0])
const { x: x11, y: y11 } = d1(t[0])
const { x: x12, y: y12 } = d2(t[0])
const { x: x2, y: y2 } = p2(t[1])
const { x: x21, y: y21 } = e1(t[1])
const { x: x22, y: y22 } = e2(t[1])
// dz1/dt1 = y1'x1' + (y1 - y2)x1'' - (x1'y1' + (x1 - x2)y1'')
// dz1/dt2 = -y2'x1' + x2'y1'
// dz2/dt1 = y1''x2' - x1''y2'
// dz2/dt2 = y1'x2'' - x1'y2''
return [
y11 * x11 + (y1 - y2) * x12 - (x11 * y11 + (x1 - x2) * y12),
-y21 * x11 + x21 * y11,
y12 * x21 - x12 * y21,
y11 * x22 - x11 * y22,
]
}
let ts: Vec2[] = []
for (const t1 of [-Math.PI / 2, Math.PI / 2]) {
for (const t2 of [-Math.PI / 2, Math.PI / 2]) {
const t = newtonIterate2([t1, t2], f1, f2, delta2)
if (t !== undefined) {
ts.push(t)
}
}
}
ts = deduplicate(ts, deepEquals)
return ts.filter(v => angleInRange(v[0], curve1) && angleInRange(v[1], curve2)).map(t => {
return [p1(t[0]), p2(t[1])]
})
}

export function getLinesTangentTo2EllipseArcs(curve1: EllipseArc, curve2: EllipseArc): Tuple2<Position>[] {
const [p1, d1, d2] = getEllipseDerivatives(curve1)
const [p2, e1, e2] = getEllipseDerivatives(curve2)
Expand Down Expand Up @@ -210,6 +260,50 @@ export function getLinesTangentToEllipseArcAndBezierCurve(curve1: EllipseArc, cu
})
}

export function getLinesTangentToEllipseArcAndHyperbola(curve1: EllipseArc, curve2: HyperbolaSegment): Tuple2<Position>[] {
const [p1, d1, d2] = getEllipseDerivatives(curve1)
const [p2, e1, e2] = getHyperbolaDerivatives(curve2)
const f1 = (t: Vec2): Vec2 => {
// (y1 - y2)/(x1 - x2) = y1'/x1' = y2'/x2'
// z1 = (y1 - y2)x1' - (x1 - x2)y1'
// z2 = y1'x2' - x1'y2'
const { x: x1, y: y1 } = p1(t[0])
const { x: x11, y: y11 } = d1(t[0])
const { x: x2, y: y2 } = p2(t[1])
const { x: x21, y: y21 } = e1(t[1])
return [(y1 - y2) * x11 - (x1 - x2) * y11, y11 * x21 - x11 * y21]
}
const f2 = (t: Vec2): Matrix2 => {
const { x: x1, y: y1 } = p1(t[0])
const { x: x11, y: y11 } = d1(t[0])
const { x: x12, y: y12 } = d2(t[0])
const { x: x2, y: y2 } = p2(t[1])
const { x: x21, y: y21 } = e1(t[1])
const { x: x22, y: y22 } = e2(t[1])
// dz1/dt1 = y1'x1' + (y1 - y2)x1'' - (x1'y1' + (x1 - x2)y1'')
// dz1/dt2 = -y2'x1' + x2'y1'
// dz2/dt1 = y1''x2' - x1''y2'
// dz2/dt2 = y1'x2'' - x1'y2''
return [
y11 * x11 + (y1 - y2) * x12 - (x11 * y11 + (x1 - x2) * y12),
-y21 * x11 + x21 * y11,
y12 * x21 - x12 * y21,
y11 * x22 - x11 * y22,
]
}
let ts: Vec2[] = []
for (const t1 of [-Math.PI / 2, Math.PI / 2]) {
const t = newtonIterate2([t1, 0], f1, f2, delta2)
if (t !== undefined) {
ts.push(t)
}
}
ts = deduplicate(ts, deepEquals)
return ts.filter(v => angleInRange(v[0], curve1) && isBetween(v[1], curve2.t1, curve2.t2)).map(t => {
return [p1(t[0]), p2(t[1])]
})
}

export function getLinesTangentToEllipseArcAndNurbsCurve(curve1: EllipseArc, curve2: NurbsCurve): Tuple2<Position>[] {
const [p1, d1, d2] = getEllipseDerivatives(curve1)
const nurbs2 = toVerbNurbsCurve(curve2)
Expand Down

0 comments on commit 77b1eba

Please sign in to comment.