mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
Rewrote bezier_segment_closest_point() algorithm to search all minima.
This commit is contained in:
parent
d069cf39ce
commit
6dbff74f2b
1 changed files with 28 additions and 12 deletions
40
beziers.scad
40
beziers.scad
|
@ -114,19 +114,35 @@ function bezier_curve(curve,n) = [for(i=[0:1:n-1]) bez_point(curve, i/(n-1))];
|
||||||
// trace_bezier(bez, N=len(bez)-1);
|
// trace_bezier(bez, N=len(bez)-1);
|
||||||
// color("red") translate(pt) sphere(r=1);
|
// color("red") translate(pt) sphere(r=1);
|
||||||
// color("blue") translate(bez_point(bez,u)) sphere(r=1);
|
// color("blue") translate(bez_point(bez,u)) sphere(r=1);
|
||||||
function bezier_segment_closest_point(curve, pt, max_err=0.01, u=0, end_u=1, step_u=undef, min_dist=undef, min_u=undef) =
|
function bezier_segment_closest_point(curve, pt, max_err=0.01, u=0, end_u=1) =
|
||||||
let(
|
let(
|
||||||
step = step_u == undef? (end_u-u)/(len(curve)*2) : step_u,
|
steps = len(curve)*3,
|
||||||
t_u = min(u, end_u),
|
path = [for (i=[0:1:steps]) let(v=(end_u-u)*(i/steps)+u) [v, bez_point(curve, v)]],
|
||||||
dist = norm(bez_point(curve, t_u)-pt),
|
bracketed = concat([path[0]], path, [path[len(path)-1]]),
|
||||||
md = (min_dist==undef || dist<min_dist)? dist : min_dist,
|
minima_ranges = [
|
||||||
mu = (min_dist==undef || dist<min_dist)? t_u : min_u
|
for (pts = triplet(bracketed)) let(
|
||||||
)
|
d1=norm(pts.x.y-pt),
|
||||||
(u>(end_u-step/2))? (
|
d2=norm(pts.y.y-pt),
|
||||||
(step<max_err)? mu : bezier_segment_closest_point(curve, pt, max_err, max(0, mu-step/2), min(1, mu+step/2), step/2)
|
d3=norm(pts.z.y-pt)
|
||||||
) : (
|
) if(d2<=d1 && d2<=d3) [pts.x.x,pts.z.x]
|
||||||
bezier_segment_closest_point(curve, pt, max_err, u+step, end_u, step, md, mu)
|
]
|
||||||
);
|
) len(minima_ranges)>1? (
|
||||||
|
let(
|
||||||
|
min_us = [
|
||||||
|
for (minima = minima_ranges)
|
||||||
|
bezier_segment_closest_point(curve, pt, max_err=max_err, u=minima.x, end_u=minima.y)
|
||||||
|
],
|
||||||
|
dists = [for (v=min_us) norm(bez_point(curve,v)-pt)],
|
||||||
|
min_i = min_index(dists)
|
||||||
|
) min_us[min_i]
|
||||||
|
) : let(
|
||||||
|
minima = minima_ranges[0],
|
||||||
|
p1 = bez_point(curve, minima.x),
|
||||||
|
p2 = bez_point(curve, minima.y),
|
||||||
|
err = norm(p2-p1)
|
||||||
|
) err<max_err? mean(minima) :
|
||||||
|
bezier_segment_closest_point(curve, pt, max_err=max_err, u=minima.x, end_u=minima.y);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue