mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-19 19:09:36 +00:00
1062 lines
43 KiB
OpenSCAD
1062 lines
43 KiB
OpenSCAD
//////////////////////////////////////////////////////////////////////
|
|
// LibFile: beziers.scad
|
|
// Bezier functions and modules.
|
|
// To use, add the following lines to the beginning of your file:
|
|
// ```
|
|
// include <BOSL2/std.scad>
|
|
// include <BOSL2/paths.scad>
|
|
// include <BOSL2/beziers.scad>
|
|
// ```
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
/*
|
|
BSD 2-Clause License
|
|
|
|
Copyright (c) 2017-2019, Revar Desmera
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
* Redistributions of source code must retain the above copyright notice, this
|
|
list of conditions and the following disclaimer.
|
|
|
|
* Redistributions in binary form must reproduce the above copyright notice,
|
|
this list of conditions and the following disclaimer in the documentation
|
|
and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
|
|
|
|
// Section: Terminology
|
|
// **Polyline**: A series of points joined by straight line segements.
|
|
//
|
|
// **Bezier Curve**: A mathematical curve that joins two endpoints, following a curve determined by one or more control points.
|
|
//
|
|
// **Endpoint**: A point that is on the end of a bezier segment. This point lies on the bezier curve.
|
|
//
|
|
// **Control Point**: A point that influences the shape of the curve that connects two endpoints. This is often *NOT* on the bezier curve.
|
|
//
|
|
// **Degree**: The number of control points, plus one endpoint, needed to specify a bezier segment. Most beziers are cubic (degree 3).
|
|
//
|
|
// **Bezier Segment**: A list consisting of an endpoint, one or more control points, and a final endpoint. The number of control points is one less than the degree of the bezier. A cubic (degree 3) bezier segment looks something like:
|
|
// `[endpt1, cp1, cp2, endpt2]`
|
|
//
|
|
// **Bezier Path**: A list of bezier segments flattened out into a list of points, where each segment shares the endpoint of the previous segment as a start point. A cubic Bezier Path looks something like:
|
|
// `[endpt1, cp1, cp2, endpt2, cp3, cp4, endpt3]`
|
|
// **NOTE**: A bezier path is *NOT* a polyline. It is only the points and controls used to define the curve.
|
|
//
|
|
// **Bezier Patch**: A surface defining grid of (N+1) by (N+1) bezier points. If a Bezier Segment defines a curved line, a Bezier Patch defines a curved surface.
|
|
//
|
|
// **Bezier Surface**: A surface defined by a list of one or more bezier patches.
|
|
//
|
|
// **Spline Steps**: The number of straight-line segments to split a bezier segment into, to approximate the bezier curve. The more spline steps, the closer the approximation will be to the curve, but the slower it will be to generate. Usually defaults to 16.
|
|
|
|
|
|
// Section: Segment Functions
|
|
|
|
// Function: bez_point()
|
|
// Usage:
|
|
// bez_point(curve, u)
|
|
// Description:
|
|
// Formula to calculate points on a bezier curve. The degree of
|
|
// the curve, N, is one less than the number of points in `curve`.
|
|
// Arguments:
|
|
// curve = The list of endpoints and control points for this bezier segment.
|
|
// u = The proportion of the way along the curve to find the point of. 0<=`u`<=1
|
|
// Example(2D): Quadratic (Degree 2) Bezier.
|
|
// bez = [[0,0], [30,30], [80,0]];
|
|
// trace_bezier(bez, N=len(bez)-1);
|
|
// translate(bez_point(bez, 0.3)) color("red") sphere(1);
|
|
// Example(2D): Cubic (Degree 3) Bezier
|
|
// bez = [[0,0], [5,35], [60,-25], [80,0]];
|
|
// trace_bezier(bez, N=len(bez)-1);
|
|
// translate(bez_point(bez, 0.4)) color("red") sphere(1);
|
|
// Example(2D): Degree 4 Bezier.
|
|
// bez = [[0,0], [5,15], [40,20], [60,-15], [80,0]];
|
|
// trace_bezier(bez, N=len(bez)-1);
|
|
// translate(bez_point(bez, 0.8)) color("red") sphere(1);
|
|
function bez_point(curve,u)=
|
|
(len(curve) <= 1) ?
|
|
curve[0] :
|
|
bez_point(
|
|
[for(i=[0:len(curve)-2]) curve[i]*(1-u)+curve[i+1]*u],
|
|
u
|
|
);
|
|
|
|
|
|
// Function: bezier_segment_closest_point()
|
|
// Usage:
|
|
// bezier_segment_closest_point(bezier,pt)
|
|
// Description:
|
|
// Finds the closest part of the given bezier segment to point `pt`.
|
|
// The degree of the curve, N, is one less than the number of points in `curve`.
|
|
// Returns `u` for the shortest position on the bezier segment to the given point `pt`.
|
|
// Arguments:
|
|
// curve = The list of endpoints and control points for this bezier segment.
|
|
// pt = The point to find the closest curve point to.
|
|
// max_err = The maximum allowed error when approximating the closest approach.
|
|
// Example(2D):
|
|
// pt = [40,15];
|
|
// bez = [[0,0], [20,40], [60,-25], [80,0]];
|
|
// u = bezier_segment_closest_point(bez, pt);
|
|
// trace_bezier(bez, N=len(bez)-1);
|
|
// color("red") translate(pt) 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) =
|
|
let(
|
|
step = step_u == undef? (end_u-u)/(len(curve)*2) : step_u,
|
|
t_u = min(u, end_u),
|
|
dist = norm(bez_point(curve, t_u)-pt),
|
|
md = (min_dist==undef || dist<min_dist)? dist : min_dist,
|
|
mu = (min_dist==undef || dist<min_dist)? t_u : min_u
|
|
)
|
|
(u>(end_u-step/2))? (
|
|
(step<max_err)? mu : bezier_segment_closest_point(curve, pt, max_err, max(0, mu-step/2), min(1, mu+step/2), step/2)
|
|
) : (
|
|
bezier_segment_closest_point(curve, pt, max_err, u+step, end_u, step, md, mu)
|
|
);
|
|
|
|
|
|
|
|
// Function: bezier_segment_length()
|
|
// Usage:
|
|
// bezier_segment_length(curve, [start_u], [end_u], [max_deflect]);
|
|
// Description:
|
|
// Approximates the length of the bezier segment between start_u and end_u.
|
|
// Arguments:
|
|
// curve = The list of endpoints and control points for this bezier segment.
|
|
// start_u = The proportion of the way along the curve to start measuring from. Between 0 and 1.
|
|
// end_u = The proportion of the way along the curve to end measuring at. Between 0 and 1. Greater than start_u.
|
|
// max_deflect = The largest amount of deflection from the true curve to allow for approximation.
|
|
// Example:
|
|
// bez = [[0,0], [5,35], [60,-25], [80,0]];
|
|
// echo(bezier_segment_length(bez));
|
|
function bezier_segment_length(curve, start_u=0, end_u=1, max_deflect=0.01) =
|
|
let(
|
|
mid_u=lerp(start_u, end_u, 0.5),
|
|
sp = bez_point(curve,start_u),
|
|
bez_mp = bez_point(curve,mid_u),
|
|
ep = bez_point(curve,end_u),
|
|
lin_mp = lerp(sp,ep,0.5),
|
|
defl = norm(bez_mp-lin_mp)
|
|
)
|
|
((end_u-start_u) >= 0.125 || defl > max_deflect)? (
|
|
bezier_segment_length(curve, start_u, mid_u, max_deflect) +
|
|
bezier_segment_length(curve, mid_u, end_u, max_deflect)
|
|
) : norm(ep-sp);
|
|
|
|
|
|
|
|
// Function: fillet3pts()
|
|
// Usage:
|
|
// fillet3pts(p0, p1, p2, r);
|
|
// Description:
|
|
// Takes three points, defining two line segments, and works out the
|
|
// cubic (degree 3) bezier segment (and surrounding control points)
|
|
// needed to approximate a rounding of the corner with radius `r`.
|
|
// If there isn't room for a radius `r` rounding, uses the largest
|
|
// radius that will fit. Returns [cp1, endpt1, cp2, cp3, endpt2, cp4]
|
|
// Arguments:
|
|
// p0 = The starting point.
|
|
// p1 = The middle point.
|
|
// p2 = The ending point.
|
|
// r = The radius of the fillet/rounding.
|
|
// maxerr = Max amount bezier curve should diverge from actual radius curve. Default: 0.1
|
|
// Example(2D):
|
|
// p0 = [40, 0];
|
|
// p1 = [0, 0];
|
|
// p2 = [30, 30];
|
|
// trace_polyline([p0,p1,p2], showpts=true, size=0.5, color="green");
|
|
// fbez = fillet3pts(p0,p1,p2, 10);
|
|
// trace_bezier(slice(fbez, 1, -2), size=1);
|
|
function fillet3pts(p0, p1, p2, r, maxerr=0.1, w=0.5, dw=0.25) = let(
|
|
v0 = normalize(p0-p1),
|
|
v1 = normalize(p2-p1),
|
|
midv = normalize((v0+v1)/2),
|
|
a = vector_angle(v0,v1),
|
|
tanr = min(r/tan(a/2), norm(p0-p1)*0.99, norm(p2-p1)*0.99),
|
|
tp0 = p1+v0*tanr,
|
|
tp1 = p1+v1*tanr,
|
|
cp = p1 + midv * tanr / cos(a/2),
|
|
cp0 = lerp(tp0, p1, w),
|
|
cp1 = lerp(tp1, p1, w),
|
|
cpr = norm(cp-tp0),
|
|
bp = bez_point([tp0, cp0, cp1, tp1], 0.5),
|
|
tdist = norm(cp-bp)
|
|
) (abs(tdist-cpr) <= maxerr)? [tp0, tp0, cp0, cp1, tp1, tp1] :
|
|
(tdist<cpr)? fillet3pts(p0, p1, p2, r, maxerr=maxerr, w=w+dw, dw=dw/2) :
|
|
fillet3pts(p0, p1, p2, r, maxerr=maxerr, w=w-dw, dw=dw/2);
|
|
|
|
|
|
|
|
// Section: Path Functions
|
|
|
|
|
|
// Function: bezier_path_point()
|
|
// Usage:
|
|
// bezier_path_point(path, seg, u, [N])
|
|
// Description: Returns the coordinates of bezier path segment `seg` at position `u`.
|
|
// Arguments:
|
|
// path = A bezier path to approximate.
|
|
// seg = Segment number along the path. Each segment is N points long.
|
|
// u = The proportion of the way along the segment to find the point of. 0<=`u`<=1
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
function bezier_path_point(path, seg, u, N=3) = bez_point(select(path,seg*N,(seg+1)*N), u);
|
|
|
|
|
|
|
|
// Function: bezier_path_closest_point()
|
|
// Usage:
|
|
// bezier_path_closest_point(bezier,pt)
|
|
// Description:
|
|
// Finds the closest part of the given bezier path to point `pt`.
|
|
// Returns [segnum, u] for the closest position on the bezier path to the given point `pt`.
|
|
// Arguments:
|
|
// path = A bezier path to approximate.
|
|
// pt = The point to find the closest curve point to.
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// max_err = The maximum allowed error when approximating the closest approach.
|
|
// Example(2D):
|
|
// pt = [100,0];
|
|
// bez = [[0,0], [20,40], [60,-25], [80,0], [100,25], [140,25], [160,0]];
|
|
// pos = bezier_path_closest_point(bez, pt);
|
|
// xy = bezier_path_point(bez,pos[0],pos[1]);
|
|
// trace_bezier(bez, N=3);
|
|
// color("red") translate(pt) sphere(r=1);
|
|
// color("blue") translate(xy) sphere(r=1);
|
|
function bezier_path_closest_point(path, pt, N=3, max_err=0.01, seg=0, min_seg=undef, min_u=undef, min_dist=undef) =
|
|
let(curve = select(path,seg*N,(seg+1)*N))
|
|
(seg*N+1 >= len(path))? (
|
|
let(curve = select(path, min_seg*N, (min_seg+1)*N))
|
|
[min_seg, bezier_segment_closest_point(curve, pt, max_err=max_err)]
|
|
) : (
|
|
let(
|
|
curve = select(path,seg*N,(seg+1)*N),
|
|
u = bezier_segment_closest_point(curve, pt, max_err=0.05),
|
|
dist = norm(bez_point(curve, u)-pt),
|
|
mseg = (min_dist==undef || dist<min_dist)? seg : min_seg,
|
|
mdist = (min_dist==undef || dist<min_dist)? dist : min_dist,
|
|
mu = (min_dist==undef || dist<min_dist)? u : min_u
|
|
)
|
|
bezier_path_closest_point(path, pt, N, max_err, seg+1, mseg, mu, mdist)
|
|
);
|
|
|
|
|
|
|
|
// Function: bezier_path_length()
|
|
// Usage:
|
|
// bezier_path_length(path, [N], [max_deflect]);
|
|
// Description:
|
|
// Approximates the length of the bezier path.
|
|
// Arguments:
|
|
// path = A bezier path to approximate.
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// max_deflect = The largest amount of deflection from the true curve to allow for approximation.
|
|
function bezier_path_length(path, N=3, max_deflect=0.001) =
|
|
sum([
|
|
for (seg=[0:(len(path)-1)/N-1]) (
|
|
bezier_segment_length(
|
|
select(path, seg*N, (seg+1)*N),
|
|
max_deflect=max_deflect
|
|
)
|
|
)
|
|
]);
|
|
|
|
|
|
|
|
// Function: bezier_polyline()
|
|
// Usage:
|
|
// bezier_polyline(bezier, [splinesteps], [N])
|
|
// Description:
|
|
// Takes a bezier path and converts it into a polyline.
|
|
// Arguments:
|
|
// bezier = A bezier path to approximate.
|
|
// splinesteps = Number of straight lines to split each bezier segment into. default=16
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// Example(2D):
|
|
// bez = [
|
|
// [0,0], [-5,30],
|
|
// [20,60], [50,50], [110,30],
|
|
// [60,25], [70,0], [80,-25],
|
|
// [80,-50], [50,-50]
|
|
// ];
|
|
// trace_polyline(bez, size=1, N=3, showpts=true);
|
|
// trace_polyline(bezier_polyline(bez, N=3), size=3);
|
|
function bezier_polyline(bezier, splinesteps=16, N=3) = let(
|
|
segs = (len(bezier)-1)/N
|
|
) concat(
|
|
[for (seg = [0:segs-1], i = [0:splinesteps-1]) bezier_path_point(bezier, seg, i/splinesteps, N=N)],
|
|
[bezier_path_point(bezier, segs-1, 1, N=N)]
|
|
);
|
|
|
|
|
|
|
|
// Function: fillet_path()
|
|
// Usage:
|
|
// fillet_path(pts, fillet, [maxerr]);
|
|
// Description:
|
|
// Takes a 3D polyline path and fillets the corners, returning a 3d cubic (degree 3) bezier path.
|
|
// Arguments:
|
|
// pts = 3D Polyline path to fillet.
|
|
// fillet = The radius to fillet/round the polyline corners by.
|
|
// maxerr = Max amount bezier curve should diverge from actual radius curve. Default: 0.1
|
|
// Example(2D):
|
|
// pline = [[40,0], [0,0], [35,35], [0,70], [-10,60], [-5,55], [0,60]];
|
|
// bez = fillet_path(pline, 10);
|
|
// trace_polyline(pline, showpts=true, size=0.5, color="green");
|
|
// trace_bezier(bez, size=1);
|
|
function fillet_path(pts, fillet, maxerr=0.1) = concat(
|
|
[pts[0], pts[0]],
|
|
(len(pts) < 3)? [] : [
|
|
for (p = [1 : len(pts)-2]) let(
|
|
p1 = pts[p],
|
|
p0 = (pts[p-1]+p1)/2,
|
|
p2 = (pts[p+1]+p1)/2
|
|
) for (pt = fillet3pts(p0, p1, p2, fillet, maxerr=maxerr)) pt
|
|
],
|
|
[pts[len(pts)-1], pts[len(pts)-1]]
|
|
);
|
|
|
|
|
|
// Function: bezier_close_to_axis()
|
|
// Usage:
|
|
// bezier_close_to_axis(bezier, [N], [axis]);
|
|
// Description:
|
|
// Takes a 2D bezier path and closes it to the specified axis.
|
|
// Arguments:
|
|
// bezier = The 2D bezier path to close to the axis.
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// axis = The axis to close to, "X", or "Y". Default: "X"
|
|
// Example(2D):
|
|
// bez = [[50,30], [40,10], [10,50], [0,30], [-10, 10], [-30,10], [-50,20]];
|
|
// closed = bezier_close_to_axis(bez);
|
|
// trace_bezier(closed, size=1);
|
|
// Example(2D):
|
|
// bez = [[30,50], [10,40], [50,10], [30,0], [10, -10], [10,-30], [20,-50]];
|
|
// closed = bezier_close_to_axis(bez, axis="Y");
|
|
// trace_bezier(closed, size=1);
|
|
function bezier_close_to_axis(bezier, N=3, axis="X") =
|
|
let(
|
|
bezend = len(bezier)-1,
|
|
sp = bezier[0],
|
|
ep = bezier[bezend]
|
|
) (axis=="X")? concat(
|
|
[for (i=[0:N-1]) lerp([sp.x,0], sp, i/N)],
|
|
bezier,
|
|
[for (i=[1:N]) lerp(ep, [ep.x,0], i/N)],
|
|
[for (i=[1:N]) lerp([ep.x,0], [sp.x,0], i/N)]
|
|
) : (axis=="Y")? concat(
|
|
[for (i=[0:N-1]) lerp([0,sp.y], sp, i/N)],
|
|
bezier,
|
|
[for (i=[1:N]) lerp(ep, [0,ep.y], i/N)],
|
|
[for (i=[1:N]) lerp([0,ep.y], [0,sp.y], i/N)]
|
|
) : (
|
|
assert_in_list("axis", axis, ["X","Y"])
|
|
);
|
|
|
|
|
|
// Function: bezier_offset()
|
|
// Usage:
|
|
// bezier_offset(inset, bezier, [N], [axis]);
|
|
// Description:
|
|
// Takes a 2D bezier path and closes it with a matching reversed path that is closer to the given axis by distance `inset`.
|
|
// Arguments:
|
|
// inset = Amount to lower second path by.
|
|
// bezier = The 2D bezier path.
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// axis = The axis to offset towards, "X", or "Y". Default: "X"
|
|
// Example(2D):
|
|
// bez = [[50,30], [40,10], [10,50], [0,30], [-10, 10], [-30,10], [-50,20]];
|
|
// closed = bezier_offset(5, bez);
|
|
// trace_bezier(closed, size=1);
|
|
// Example(2D):
|
|
// bez = [[30,50], [10,40], [50,10], [30,0], [10, -10], [10,-30], [20,-50]];
|
|
// closed = bezier_offset(5, bez, axis="Y");
|
|
// trace_bezier(closed, size=1);
|
|
function bezier_offset(inset, bezier, N=3, axis="X") =
|
|
let(
|
|
backbez = reverse([ for (pt = bezier) pt-(axis=="X"? [0,inset] : [inset,0]) ]),
|
|
bezend = len(bezier)-1
|
|
) concat(
|
|
bezier,
|
|
[for (i=[1:N-1]) lerp(bezier[bezend], backbez[0], i/N)],
|
|
backbez,
|
|
[for (i=[1:N]) lerp(backbez[bezend], bezier[0], i/N)]
|
|
);
|
|
|
|
|
|
|
|
// Section: Path Modules
|
|
|
|
|
|
// Module: bezier_polygon()
|
|
// Usage:
|
|
// bezier_polygon(bezier, [splinesteps], [N]) {
|
|
// Description:
|
|
// Takes a closed 2D bezier path, and creates a 2D polygon from it.
|
|
// Arguments:
|
|
// bezier = The closed bezier path to make into a polygon.
|
|
// splinesteps = Number of straight lines to split each bezier segment into. default=16
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// Example(2D):
|
|
// bez = [
|
|
// [0,0], [-5,30],
|
|
// [20,60], [50,50], [110,30],
|
|
// [60,25], [70,0], [80,-25],
|
|
// [80,-50], [50,-50], [30,-50],
|
|
// [5,-30], [0,0]
|
|
// ];
|
|
// trace_bezier(bez, N=3, size=3);
|
|
// linear_extrude(height=0.1) bezier_polygon(bez, N=3);
|
|
module bezier_polygon(bezier, splinesteps=16, N=3) {
|
|
polypoints=bezier_polyline(bezier, splinesteps, N);
|
|
polygon(points=slice(polypoints, 0, -1));
|
|
}
|
|
|
|
|
|
// Module: linear_extrude_bezier()
|
|
// Usage:
|
|
// linear_extrude_bezier(bezier, height, [splinesteps], [N], [center], [convexity], [twist], [slices], [scale], [orient], [anchor]);
|
|
// Description:
|
|
// Takes a closed 2D bezier path, centered on the XY plane, and
|
|
// extrudes it linearly upwards, forming a solid.
|
|
// Arguments:
|
|
// bezier = Array of 2D points of a bezier path, to be extruded.
|
|
// splinesteps = Number of steps to divide each bezier segment into. default=16
|
|
// N = The degree of the bezier curves. Cubic beziers have N=3. Default: 3
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
// twist = Angle in degrees to twist over the length of extrusion. default=0
|
|
// scale = Relative size of top of extrusion to the bottom. default=1.0
|
|
// slices = Number of vertical slices to use for twisted extrusion. default=20
|
|
// center = If true, the extruded solid is centered vertically at z=0.
|
|
// orient = Orientation of the extrusion. Use the `ORIENT_` constants from `constants.scad`. Default: `ORIENT_Z`.
|
|
// anchor = Alignment of the extrusion. Use the constants from `constants.scad`. Default: `UP`.
|
|
// Example:
|
|
// bez = [
|
|
// [-10, 0], [-15, -5],
|
|
// [ -5, -10], [ 0, -10], [ 5, -10],
|
|
// [ 10, -5], [ 15, 0], [10, 5],
|
|
// [ 5, 10], [ 0, 10], [-5, 10],
|
|
// [ 25, -15], [-10, 0]
|
|
// ];
|
|
// linear_extrude_bezier(bez, height=20, splinesteps=32);
|
|
module linear_extrude_bezier(bezier, height=100, splinesteps=16, N=3, center=undef, convexity=undef, twist=undef, slices=undef, scale=undef, orient=ORIENT_Z, anchor=UP) {
|
|
maxx = max([for (pt = bezier) abs(pt[0])]);
|
|
maxy = max([for (pt = bezier) abs(pt[1])]);
|
|
orient_and_anchor([maxx*2,maxy*2,height], orient, anchor) {
|
|
linear_extrude(height=height, center=true, convexity=convexity, twist=twist, slices=slices, scale=scale) {
|
|
bezier_polygon(bezier, splinesteps=splinesteps, N=N);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Module: revolve_bezier()
|
|
// Usage:
|
|
// revolve_bezier(bezier, [splinesteps], [N], [convexity], [angle], [orient], [anchor])
|
|
// Description:
|
|
// Takes a closed 2D bezier and rotates it around the X axis, forming a solid.
|
|
// Arguments:
|
|
// bezier = array of 2D points for the bezier path to rotate.
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
|
// N = number of points in each bezier segment. default=3 (cubic)
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
// angle = Degrees of sweep to make. Default: 360
|
|
// orient = Orientation of the extrusion. Use the `ORIENT_` constants from `constants.scad`. Default: `ORIENT_X`.
|
|
// anchor = Alignment of the extrusion. Use the constants from `constants.scad`. Default: `CENTER`.
|
|
// Example(FlatSpin):
|
|
// path = [
|
|
// [ 0, 10], [ 50, 0], [ 50, 40],
|
|
// [ 95, 40], [100, 40], [100, 45],
|
|
// [ 95, 45], [ 66, 45], [ 0, 20],
|
|
// [ 0, 12], [ 0, 12], [ 0, 10],
|
|
// [ 0, 10]
|
|
// ];
|
|
// revolve_bezier(path, splinesteps=32, $fn=180);
|
|
module revolve_bezier(bezier, splinesteps=16, N=3, convexity=10, angle=360, orient=ORIENT_X, anchor=CENTER)
|
|
{
|
|
maxx = max([for (pt = bezier) abs(pt[0])]);
|
|
maxy = max([for (pt = bezier) abs(pt[1])]);
|
|
orient_and_anchor([maxx*2,maxx*2,maxy*2], orient, anchor) {
|
|
rotate_extrude(convexity=convexity, angle=angle) {
|
|
xrot(180) zrot(-90) bezier_polygon(bezier, splinesteps, N);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Module: rotate_extrude_bezier()
|
|
// Usage:
|
|
// rotate_extrude_bezier(bezier, splinesteps=16, N=3, convexity=10, angle=360)
|
|
// Description:
|
|
// Takes a closed 2D bezier and rotates it around the Z axis, forming a solid.
|
|
// Behaves like rotate_extrude(), except for beziers instead of shapes.
|
|
// Arguments:
|
|
// bezier = array of 2D points for the bezier path to rotate.
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
|
// N = number of points in each bezier segment. default=3 (cubic)
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
// angle = Degrees of sweep to make. Default: 360
|
|
// orient = Orientation of the extrusion. Use the `ORIENT_` constants from `constants.scad`. Default: `ORIENT_Z`.
|
|
// anchor = Alignment of the extrusion. Use the constants from `constants.scad`. Default: `CENTER`.
|
|
// Example(Spin):
|
|
// path = [
|
|
// [ 0, 10], [ 50, 0], [ 50, 40],
|
|
// [ 95, 40], [100, 40], [100, 45],
|
|
// [ 95, 45], [ 66, 45], [ 0, 20],
|
|
// [ 0, 12], [ 0, 12], [ 0, 10],
|
|
// [ 0, 10]
|
|
// ];
|
|
// rotate_extrude_bezier(path, splinesteps=32, $fn=180);
|
|
module rotate_extrude_bezier(bezier, splinesteps=16, N=3, convexity=10, angle=360, orient=ORIENT_Z, anchor=CENTER)
|
|
{
|
|
maxx = max([for (pt = bezier) abs(pt[0])]);
|
|
maxy = max([for (pt = bezier) abs(pt[1])]);
|
|
orient_and_anchor([maxx*2,maxx*2,0], orient, anchor) {
|
|
rotate_extrude(convexity=convexity, angle=angle) {
|
|
bezier_polygon(bezier, splinesteps, N);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Module: revolve_bezier_solid_to_axis()
|
|
// Usage:
|
|
// revolve_bezier_solid_to_axis(bezier, [splinesteps], [N], [convexity], [angle], [orient], [anchor]);
|
|
// Description:
|
|
// Takes a 2D bezier and rotates it around the X axis, forming a solid.
|
|
// Arguments:
|
|
// bezier = array of points for the bezier path to rotate.
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
|
// N = number of points in each bezier segment. default=3 (cubic)
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
// angle = Degrees of sweep to make. Default: 360
|
|
// orient = Orientation of the extrusion. Use the `ORIENT_` constants from `constants.scad`. Default: `ORIENT_X`.
|
|
// anchor = Alignment of the extrusion. Use the constants from `constants.scad`. Default: `CENTER`.
|
|
// Example(FlatSpin):
|
|
// path = [ [0, 10], [33, 10], [66, 40], [100, 40] ];
|
|
// revolve_bezier_solid_to_axis(path, splinesteps=32, $fn=72);
|
|
module revolve_bezier_solid_to_axis(bezier, splinesteps=16, N=3, convexity=10, angle=360, orient=ORIENT_X, anchor=CENTER) {
|
|
revolve_bezier(bezier=bezier_close_to_axis(bezier), splinesteps=splinesteps, N=N, convexity=convexity, angle=angle, orient=orient, anchor=anchor);
|
|
}
|
|
|
|
|
|
// Module: revolve_bezier_offset_shell()
|
|
// Usage:
|
|
// revolve_bezier_offset_shell(bezier, offset, [splinesteps], [N], [convexity], [angle], [orient], [anchor]);
|
|
// Description:
|
|
// Takes a 2D bezier and rotates it around the X axis, into a hollow shell.
|
|
// Arguments:
|
|
// bezier = array of points for the bezier path to rotate.
|
|
// offset = the thickness of the created shell.
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
|
// N = number of points in each bezier segment. default=3 (cubic)
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
// angle = degrees of sweep to make. Default: 360
|
|
// orient = Orientation of the extrusion. Use the `ORIENT_` constants from `constants.scad`. Default: `ORIENT_X`.
|
|
// anchor = Alignment of the extrusion. Use the constants from `constants.scad`. Default: `CENTER`.
|
|
// Example(FlatSpin):
|
|
// path = [ [0, 10], [33, 10], [66, 40], [100, 40] ];
|
|
// revolve_bezier_offset_shell(path, offset=1, splinesteps=32, $fn=72);
|
|
module revolve_bezier_offset_shell(bezier, offset=1, splinesteps=16, N=3, convexity=10, angle=360, orient=ORIENT_X, anchor=CENTER) {
|
|
revolve_bezier(bezier=bezier_offset(offset, bezier), splinesteps=splinesteps, N=N, orient=orient, anchor=anchor);
|
|
}
|
|
|
|
|
|
// Module: extrude_2d_shapes_along_bezier()
|
|
// Usage:
|
|
// extrude_2d_shapes_along_bezier(bezier, [splinesteps], [N], [convexity], [clipsize]) ...
|
|
// Description:
|
|
// Extrudes 2D children along a bezier path.
|
|
// Arguments:
|
|
// bezier = array of points for the bezier path to extrude along.
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
|
// Example(FR):
|
|
// path = [ [0, 0, 0], [33, 33, 33], [66, -33, -33], [100, 0, 0] ];
|
|
// extrude_2d_shapes_along_bezier(path) difference(){
|
|
// circle(r=10);
|
|
// fwd(10/2) circle(r=8);
|
|
// }
|
|
module extrude_2d_shapes_along_bezier(bezier, splinesteps=16, N=3, convexity=10, clipsize=1000) {
|
|
path = slice(bezier_polyline(bezier, splinesteps, N), 0, -1);
|
|
extrude_2d_shapes_along_3dpath(path, convexity=convexity, clipsize=clipsize) children();
|
|
}
|
|
|
|
|
|
// Module: extrude_bezier_along_bezier()
|
|
// Usage:
|
|
// extrude_bezier_along_bezier(bezier, path, [pathsteps], [bezsteps], [bezN], [pathN]);
|
|
// Description:
|
|
// Takes a closed 2D bezier path, centered on the XY plane, and
|
|
// extrudes it perpendicularly along a 3D bezier path, forming a solid.
|
|
// Arguments:
|
|
// bezier = Array of 2D points of a bezier path, to be extruded.
|
|
// path = Array of 3D points of a bezier path, to extrude along.
|
|
// pathsteps = number of steps to divide each path segment into.
|
|
// bezsteps = number of steps to divide each bezier segment into.
|
|
// bezN = number of points in each extruded bezier segment. default=3 (cubic)
|
|
// pathN = number of points in each path bezier segment. default=3 (cubic)
|
|
// Example(FlatSpin):
|
|
// bez = [
|
|
// [-10, 0], [-15, -5],
|
|
// [ -5, -10], [ 0, -10], [ 5, -10],
|
|
// [ 10, -5], [ 15, 0], [10, 5],
|
|
// [ 5, 10], [ 0, 10], [-5, 10],
|
|
// [ 25, -15], [-10, 0]
|
|
// ];
|
|
// path = [ [0, 0, 0], [33, 33, 33], [90, 33, -33], [100, 0, 0] ];
|
|
// extrude_bezier_along_bezier(bez, path, pathsteps=32, bezsteps=16);
|
|
module extrude_bezier_along_bezier(bezier, path, pathsteps=16, bezsteps=16, bezN=3, pathN=3) {
|
|
bez_points = simplify2d_path(bezier_polyline(bezier, bezsteps, bezN));
|
|
path_points = simplify3d_path(path3d(bezier_polyline(path, pathsteps, pathN)));
|
|
extrude_2dpath_along_3dpath(bez_points, path_points);
|
|
}
|
|
|
|
|
|
|
|
// Module: trace_bezier()
|
|
// Description:
|
|
// Renders 2D or 3D bezier paths and their associated control points.
|
|
// Useful for debugging bezier paths.
|
|
// Arguments:
|
|
// bez = the array of points in the bezier.
|
|
// N = Mark the first and every Nth vertex after in a different color and shape.
|
|
// size = diameter of the lines drawn.
|
|
// Example(2D):
|
|
// bez = [
|
|
// [-10, 0], [-15, -5],
|
|
// [ -5, -10], [ 0, -10], [ 5, -10],
|
|
// [ 14, -5], [ 15, 0], [16, 5],
|
|
// [ 5, 10], [ 0, 10]
|
|
// ];
|
|
// trace_bezier(bez, N=3, size=0.5);
|
|
module trace_bezier(bez, N=3, size=1) {
|
|
trace_polyline(bez, N=N, showpts=true, size=size/2, color="green");
|
|
trace_polyline(bezier_polyline(bez, N=N), size=size, color="cyan");
|
|
}
|
|
|
|
|
|
|
|
// Section: Patch Functions
|
|
|
|
|
|
// Function: bezier_patch_point()
|
|
// Usage:
|
|
// bezier_patch_point(patch, u, v)
|
|
// Description:
|
|
// Given a square 2-dimensional array of (N+1) by (N+1) points size,
|
|
// that represents a Bezier Patch of degree N, returns a point on that
|
|
// surface, at positions `u`, and `v`. A cubic bezier patch will be 4x4
|
|
// points in size. If given a non-square array, each direction will have
|
|
// its own degree.
|
|
// Arguments:
|
|
// patch = The 2D array of endpoints and control points for this bezier patch.
|
|
// u = The proportion of the way along the first dimension of the patch to find the point of. 0<=`u`<=1
|
|
// v = The proportion of the way along the second dimension of the patch to find the point of. 0<=`v`<=1
|
|
// Example(3D):
|
|
// patch = [
|
|
// [[-50, 50, 0], [-16, 50, 20], [ 16, 50, 20], [50, 50, 0]],
|
|
// [[-50, 16, 20], [-16, 16, 40], [ 16, 16, 40], [50, 16, 20]],
|
|
// [[-50,-16, 20], [-16,-16, 40], [ 16,-16, 40], [50,-16, 20]],
|
|
// [[-50,-50, 0], [-16,-50, 20], [ 16,-50, 20], [50,-50, 0]]
|
|
// ];
|
|
// trace_bezier_patches(patches=[patch], size=1, showcps=true);
|
|
// pt = bezier_patch_point(patch, 0.6, 0.75);
|
|
// translate(pt) color("magenta") sphere(d=3, $fn=12);
|
|
function bezier_patch_point(patch, u, v) = bez_point([for (bez = patch) bez_point(bez, u)], v);
|
|
|
|
|
|
// Function: bezier_triangle_point()
|
|
// Usage:
|
|
// bezier_triangle_point(tri, u, v)
|
|
// Description:
|
|
// Given a triangular 2-dimensional array of N+1 by (for the first row) N+1 points,
|
|
// that represents a Bezier triangular patch of degree N, returns a point on
|
|
// that surface, at positions `u`, and `v`. A cubic bezier triangular patch
|
|
// will have a list of 4 points in the first row, 3 in the second, 2 in the
|
|
// third, and 1 in the last row.
|
|
// Arguments:
|
|
// tri = Triangular bezier patch to get point on.
|
|
// u = The proportion of the way along the first dimension of the triangular patch to find the point of. 0<=`u`<=1
|
|
// v = The proportion of the way along the second dimension of the triangular patch to find the point of. 0<=`v`<=(1-`u`)
|
|
// Example(3D):
|
|
// tri = [
|
|
// [[-50,-33,0], [-25,16,40], [20,66,20]],
|
|
// [[0,-33,30], [25,16,30]],
|
|
// [[50,-33,0]]
|
|
// ];
|
|
// trace_bezier_patches(tris=[tri], size=1, showcps=true);
|
|
// pt = bezier_triangle_point(tri, 0.5, 0.2);
|
|
// translate(pt) color("magenta") sphere(d=3, $fn=12);
|
|
function bezier_triangle_point(tri, u, v) =
|
|
len(tri) == 1 ? tri[0][0] :
|
|
let(
|
|
n = len(tri)-1,
|
|
Pu = [for(i=[0:n-1]) [for (j=[1:len(tri[i])-1]) tri[i][j]]],
|
|
Pv = [for(i=[0:n-1]) [for (j=[0:len(tri[i])-2]) tri[i][j]]],
|
|
Pw = [for(i=[1:len(tri)-1]) tri[i]]
|
|
)
|
|
bezier_triangle_point(u*Pu + v*Pv + (1-u-v)*Pw, u, v);
|
|
|
|
|
|
|
|
// Function: bezier_patch()
|
|
// Usage:
|
|
// bezier_patch(patch, [splinesteps], [vertices], [faces]);
|
|
// Description:
|
|
// Calculate vertices and faces for forming a partial polyhedron
|
|
// from the given bezier rectangular patch. Returns a list containing
|
|
// two elements. The first is the list of unique vertices. The
|
|
// second is the list of faces, where each face is a list of indices
|
|
// into the list of vertices. You can chain calls to this, to add
|
|
// more vertices and faces for multiple bezier patches, to stitch
|
|
// them together into a complete polyhedron.
|
|
// Arguments:
|
|
// patch = The rectangular array of endpoints and control points for this bezier patch.
|
|
// splinesteps = Number of steps to divide each bezier segment into. Default: 16
|
|
// vertices = Vertex list to add new points to. Default: []
|
|
// faces = Face list to add new faces to. Default: []
|
|
// Example(3D):
|
|
// patch = [
|
|
// [[-50, 50, 0], [-16, 50, -20], [ 16, 50, 20], [50, 50, 0]],
|
|
// [[-50, 16, 20], [-16, 16, -20], [ 16, 16, 20], [50, 16, 20]],
|
|
// [[-50,-16, 20], [-16,-16, 20], [ 16,-16, -20], [50,-16, 20]],
|
|
// [[-50,-50, 0], [-16,-50, 20], [ 16,-50, -20], [50,-50, 0]]
|
|
// ];
|
|
// vnf = bezier_patch(patch, splinesteps=16);
|
|
// polyhedron(points=vnf[0], faces=vnf[1]);
|
|
function bezier_patch(patch, splinesteps=16, vertices=[], faces=[]) =
|
|
let(
|
|
base = len(vertices),
|
|
pts = [for (v=[0:splinesteps], u=[0:splinesteps]) bezier_patch_point(patch, u/splinesteps, v/splinesteps)],
|
|
new_vertices = concat(vertices, pts),
|
|
new_faces = [
|
|
for (
|
|
v=[0:splinesteps-1],
|
|
u=[0:splinesteps-1],
|
|
i=[0,1]
|
|
) let (
|
|
v1 = u+v*(splinesteps+1) + base,
|
|
v2 = v1 + 1,
|
|
v3 = v1 + splinesteps + 1,
|
|
v4 = v3 + 1,
|
|
face = i? [v1,v3,v2] : [v2,v3,v4]
|
|
) face
|
|
]
|
|
) [new_vertices, concat(faces, new_faces)];
|
|
|
|
|
|
function _tri_count(n) = (n*(1+n))/2;
|
|
|
|
|
|
// Function: bezier_triangle()
|
|
// Usage:
|
|
// bezier_triangle(tri, [splinesteps], [vertices], [faces]);
|
|
// Description:
|
|
// Calculate vertices and faces for forming a partial polyhedron
|
|
// from the given bezier triangular patch. Returns a list containing
|
|
// two elements. The first is the list of unique vertices. The
|
|
// second is the list of faces, where each face is a list of indices
|
|
// into the list of vertices. You can chain calls to this, to add
|
|
// more vertices and faces for multiple bezier patches, to stitch
|
|
// them together into a complete polyhedron.
|
|
// Arguments:
|
|
// tri = The triangular array of endpoints and control points for this bezier patch.
|
|
// splinesteps = Number of steps to divide each bezier segment into. Default: 16
|
|
// vertices = Vertex list to add new points to. Default: []
|
|
// faces = Face list to add new faces to. Default: []
|
|
// Example(3D):
|
|
// tri = [
|
|
// [[-50,-33,0], [-25,16,50], [0,66,0]],
|
|
// [[0,-33,50], [25,16,50]],
|
|
// [[50,-33,0]]
|
|
// ];
|
|
// vnf = bezier_triangle(tri, splinesteps=16);
|
|
// polyhedron(points=vnf[0], faces=vnf[1]);
|
|
function bezier_triangle(tri, splinesteps=16, vertices=[], faces=[]) =
|
|
let(
|
|
base = len(vertices),
|
|
pts = [
|
|
for (
|
|
u=[0:splinesteps],
|
|
v=[0:splinesteps-u]
|
|
) bezier_triangle_point(tri, u/splinesteps, v/splinesteps)
|
|
],
|
|
new_vertices = concat(vertices, pts),
|
|
patchlen = len(tri),
|
|
tricnt = _tri_count(splinesteps+1),
|
|
new_faces = [
|
|
for (
|
|
u=[0:splinesteps-1],
|
|
v=[0:splinesteps-u-1]
|
|
) let (
|
|
v1 = v + (tricnt - _tri_count(splinesteps+1-u)) + base,
|
|
v2 = v1 + 1,
|
|
v3 = v + (tricnt - _tri_count(splinesteps-u)) + base,
|
|
v4 = v3 + 1,
|
|
allfaces = concat(
|
|
[[v1,v2,v3]],
|
|
((u<splinesteps-1 && v<splinesteps-u-1)? [[v2,v4,v3]] : [])
|
|
)
|
|
) for (face=allfaces) face
|
|
]
|
|
) [new_vertices, concat(faces, new_faces)];
|
|
|
|
|
|
|
|
// Function: bezier_patch_flat()
|
|
// Usage:
|
|
// bezier_patch_flat(size, [N], [orient], [trans]);
|
|
// Description:
|
|
// Returns a flat rectangular bezier patch of degree `N`, centered on the XY plane.
|
|
// Arguments:
|
|
// size = 2D XY size of the patch.
|
|
// N = Degree of the patch to generate. Since this is flat, a degree of 1 should usually be sufficient.
|
|
// orient = The orientation to rotate the edge patch into. Use the `ORIENT` constants in `constants.scad`.
|
|
// trans = Amount to translate patch, after rotating to `orient`.
|
|
// Example(3D):
|
|
// patch = bezier_patch_flat(size=[100,100], N=3);
|
|
// trace_bezier_patches([patch], size=1, showcps=true);
|
|
function bezier_patch_flat(size=[100,100], N=4, orient=ORIENT_Z, trans=[0,0,0]) =
|
|
let(
|
|
patch = [for (x=[0:N]) [for (y=[0:N]) vmul(point3d(size),[x/N-0.5, 0.5-y/N, 0])]]
|
|
) [for (row=patch)
|
|
translate_points(v=trans,
|
|
rotate_points3d(v=orient,row)
|
|
)
|
|
];
|
|
|
|
|
|
|
|
// Function: patch_reverse()
|
|
// Usage:
|
|
// patch_reverse(patch)
|
|
// Description:
|
|
// Reverses the patch, so that the faces generated from it are flipped back to front.
|
|
// Arguments:
|
|
// patch = The patch to reverse.
|
|
function patch_reverse(patch) = [for (row=patch) reverse(row)];
|
|
|
|
|
|
|
|
// Function: patch_translate()
|
|
// Usage:
|
|
// patch_translate(patch, v)
|
|
// Description: Translates all coordinates in a rectangular or triangular patch by a given amount.
|
|
// Arguments:
|
|
// patch = The patch to translate.
|
|
// v = Vector to translate by.
|
|
function patch_translate(patch, v=[0,0,0]) = [for(row=patch) translate_points(row, v)];
|
|
|
|
|
|
// Function: patch_scale()
|
|
// Usage:
|
|
// patch_scale(patch, v, [cp])
|
|
// Description: Scales all coordinates in a rectangular or triangular patch by a given amount.
|
|
// Arguments:
|
|
// patch = The patch to scale.
|
|
// v = [X,Y,Z] scaling factors.
|
|
// cp = Centerpoint to scale around.
|
|
function patch_scale(patch, v=[1,1,1], cp=[0,0,0]) = [for(row=patch) scale_points(row, v, cp)];
|
|
|
|
|
|
// Function: patch_rotate()
|
|
// Usage:
|
|
// patch_rotate(patch, a, [cp])
|
|
// patch_rotate(patch, a, v, [cp])
|
|
// Description: Rotates all coordinates in a rectangular or triangular patch by a given amount.
|
|
// Arguments:
|
|
// patch = The patch to rotate.
|
|
// a = Rotation angle(s) in degrees.
|
|
// v = Vector axis to rotate round.
|
|
// cp = Centerpoint to rotate around.
|
|
function patch_rotate(patch, a=undef, v=undef, cp=[0,0,0]) =
|
|
[for(row=patch) rotate_points3d(row, a=a, v=v, cp=cp)];
|
|
|
|
|
|
// Function: patches_translate()
|
|
// Usage:
|
|
// patches_translate(patch, v, [cp])
|
|
// Description: Translates all coordinates in each of a list of rectangular or triangular patches.
|
|
// Arguments:
|
|
// patches = List of patches to translate.
|
|
// v = Vector to translate by.
|
|
function patches_translate(patches, v=[0,0,0]) = [for (patch=patches) patch_translate(patch,v)];
|
|
|
|
|
|
// Function: patches_scale()
|
|
// Usage:
|
|
// patches_scale(patch, v, [cp])
|
|
// Description: Scales all coordinates in each of a list of rectangular or triangular patches.
|
|
// Arguments:
|
|
// patches = List of patches to scale.
|
|
// v = [X,Y,Z] scaling factors.
|
|
// cp = Centerpoint to scale around.
|
|
function patches_scale(patches, v=[1,1,1], cp=[0,0,0]) = [for (patch=patches) patch_scale(patch,v,cp)];
|
|
|
|
|
|
// Function: patches_rotate()
|
|
// Usage:
|
|
// patches_rotate(patch, a, [cp])
|
|
// patches_rotate(patch, a, v, [cp])
|
|
// Description: Rotates all coordinates in each of a list of rectangular or triangular patches.
|
|
// Arguments:
|
|
// patches = List of patches to rotate.
|
|
// a = Rotation angle(s) in degrees.
|
|
// v = Vector axis to rotate round.
|
|
// cp = Centerpoint to rotate around.
|
|
function patches_rotate(patches, a=undef, v=undef, cp=[0,0,0]) = [for (patch=patches) patch_rotate(patch, a=a, v=v, cp=cp)];
|
|
|
|
|
|
// Function: bezier_surface()
|
|
// Usage:
|
|
// bezier_surface(patches, [splinesteps], [vertices], [faces]);
|
|
// Description:
|
|
// Calculate vertices and faces for forming a (possibly partial)
|
|
// polyhedron from the given rectangular and triangular bezier
|
|
// patches. Returns a list containing two elements. The first is
|
|
// the list of unique vertices. The second is the list of faces,
|
|
// where each face is a list of indices into the list of vertices.
|
|
// You can chain calls to this, to add more vertices and faces for
|
|
// multiple bezier patches, to stitch them together into a complete
|
|
// polyhedron.
|
|
// Arguments:
|
|
// patches = A list of rectangular bezier patches.
|
|
// tris = A list of triangular bezier patches.
|
|
// splinesteps = Number of steps to divide each bezier segment into. Default: 16
|
|
// vertices = Vertex list to add new points to. Default: []
|
|
// faces = Face list to add new faces to. Default: []
|
|
// Example(3D):
|
|
// patch1 = [
|
|
// [[18,18,0], [33, 0, 0], [ 67, 0, 0], [ 82, 18,0]],
|
|
// [[ 0,40,0], [ 0, 0,100], [100, 0, 20], [100, 40,0]],
|
|
// [[ 0,60,0], [ 0,100,100], [100,100, 20], [100, 60,0]],
|
|
// [[18,82,0], [33,100, 0], [ 67,100, 0], [ 82, 82,0]],
|
|
// ];
|
|
// patch2 = [
|
|
// [[18,18,0], [33, 0, 0], [ 67, 0, 0], [ 82, 18,0]],
|
|
// [[ 0,40,0], [ 0, 0,-50], [100, 0,-50], [100, 40,0]],
|
|
// [[ 0,60,0], [ 0,100,-50], [100,100,-50], [100, 60,0]],
|
|
// [[18,82,0], [33,100, 0], [ 67,100, 0], [ 82, 82,0]],
|
|
// ];
|
|
// vnf = bezier_surface(patches=[patch1, patch2], splinesteps=16);
|
|
// polyhedron(points=vnf[0], faces=vnf[1]);
|
|
function bezier_surface(patches=[], tris=[], splinesteps=16, i=0, vertices=[], faces=[]) =
|
|
let(
|
|
vnf = (i >= len(patches))? [vertices, faces] :
|
|
bezier_patch(patches[i], splinesteps=splinesteps, vertices=vertices, faces=faces),
|
|
vnf2 = (i >= len(tris))? vnf :
|
|
bezier_triangle(tris[i], splinesteps=splinesteps, vertices=vnf[0], faces=vnf[1])
|
|
) (i >= len(patches) && i >= len(tris))? vnf2 :
|
|
bezier_surface(patches=patches, tris=tris, splinesteps=splinesteps, i=i+1, vertices=vnf2[0], faces=vnf2[1]);
|
|
|
|
|
|
|
|
// Section: Bezier Surface Modules
|
|
|
|
|
|
// Module: bezier_polyhedron()
|
|
// Useage:
|
|
// bezier_polyhedron(patches)
|
|
// Description:
|
|
// Takes a list of two or more bezier patches and attempts to make a complete polyhedron from them.
|
|
// Arguments:
|
|
// patches = A list of rectangular bezier patches.
|
|
// tris = A list of triangular bezier patches.
|
|
// vertices = Vertex list for additional non-bezier faces. Default: []
|
|
// faces = Additional non-bezier faces. Default: []
|
|
// splinesteps = Number of steps to divide each bezier segment into. Default: 16
|
|
// Example:
|
|
// patch1 = [
|
|
// [[18,18,0], [33, 0, 0], [ 67, 0, 0], [ 82, 18,0]],
|
|
// [[ 0,40,0], [ 0, 0, 20], [100, 0, 20], [100, 40,0]],
|
|
// [[ 0,60,0], [ 0,100, 20], [100,100,100], [100, 60,0]],
|
|
// [[18,82,0], [33,100, 0], [ 67,100, 0], [ 82, 82,0]],
|
|
// ];
|
|
// patch2 = [
|
|
// [[18,18,0], [33, 0, 0], [ 67, 0, 0], [ 82, 18,0]],
|
|
// [[ 0,40,0], [ 0, 0,-50], [100, 0,-50], [100, 40,0]],
|
|
// [[ 0,60,0], [ 0,100,-50], [100,100,-50], [100, 60,0]],
|
|
// [[18,82,0], [33,100, 0], [ 67,100, 0], [ 82, 82,0]],
|
|
// ];
|
|
// bezier_polyhedron([patch1, patch2], splinesteps=8);
|
|
module bezier_polyhedron(patches=[], tris=[], splinesteps=16, vertices=[], faces=[])
|
|
{
|
|
sfc = bezier_surface(patches=patches, tris=tris, splinesteps=splinesteps, vertices=vertices, faces=faces);
|
|
polyhedron(points=sfc[0], faces=sfc[1]);
|
|
}
|
|
|
|
|
|
|
|
// Module: trace_bezier_patches()
|
|
// Usage:
|
|
// trace_bezier_patches(patches, [size], [showcps], [splinesteps]);
|
|
// trace_bezier_patches(tris, [size], [showcps], [splinesteps]);
|
|
// trace_bezier_patches(patches, tris, [size], [showcps], [splinesteps]);
|
|
// Description:
|
|
// Shows the surface, and optionally, control points of a list of bezier patches.
|
|
// Arguments:
|
|
// patches = A list of rectangular bezier patches.
|
|
// tris = A list of triangular bezier patches.
|
|
// splinesteps = Number of steps to divide each bezier segment into. default=16
|
|
// showcps = If true, show the controlpoints as well as the surface.
|
|
// size = Size to show control points and lines.
|
|
// Example:
|
|
// patch1 = [
|
|
// [[15,15,0], [33, 0, 0], [ 67, 0, 0], [ 85, 15,0]],
|
|
// [[ 0,33,0], [33, 33, 50], [ 67, 33, 50], [100, 33,0]],
|
|
// [[ 0,67,0], [33, 67, 50], [ 67, 67, 50], [100, 67,0]],
|
|
// [[15,85,0], [33,100, 0], [ 67,100, 0], [ 85, 85,0]],
|
|
// ];
|
|
// patch2 = [
|
|
// [[15,15,0], [33, 0, 0], [ 67, 0, 0], [ 85, 15,0]],
|
|
// [[ 0,33,0], [33, 33,-50], [ 67, 33,-50], [100, 33,0]],
|
|
// [[ 0,67,0], [33, 67,-50], [ 67, 67,-50], [100, 67,0]],
|
|
// [[15,85,0], [33,100, 0], [ 67,100, 0], [ 85, 85,0]],
|
|
// ];
|
|
// trace_bezier_patches(patches=[patch1, patch2], splinesteps=8, showcps=true);
|
|
module trace_bezier_patches(patches=[], tris=[], size=1, showcps=false, splinesteps=16)
|
|
{
|
|
if (showcps) {
|
|
for (patch = patches) {
|
|
place_copies(flatten(patch)) color("red") sphere(d=size*2);
|
|
color("cyan")
|
|
for (i=[0:len(patch)-1], j=[0:len(patch[i])-1]) {
|
|
if (i<len(patch)-1) extrude_from_to(patch[i][j], patch[i+1][j]) circle(d=size);
|
|
if (j<len(patch[i])-1) extrude_from_to(patch[i][j], patch[i][j+1]) circle(d=size);
|
|
}
|
|
vnf = bezier_patch(patch, splinesteps=splinesteps);
|
|
color("blue") place_copies(vnf[0]) sphere(d=size);
|
|
}
|
|
for (patch = tris) {
|
|
place_copies(flatten(patch)) color("red") sphere(d=size*2);
|
|
color("cyan")
|
|
for (i=[0:len(patch)-2], j=[0:len(patch[i])-2]) {
|
|
extrude_from_to(patch[i][j], patch[i+1][j]) circle(d=size);
|
|
extrude_from_to(patch[i][j], patch[i][j+1]) circle(d=size);
|
|
extrude_from_to(patch[i+1][j], patch[i][j+1]) circle(d=size);
|
|
}
|
|
vnf = bezier_triangle(patch, splinesteps=splinesteps);
|
|
color("blue") place_copies(vnf[0]) sphere(d=size);
|
|
}
|
|
}
|
|
bezier_polyhedron(patches=patches, tris=tris, splinesteps=splinesteps);
|
|
}
|
|
|
|
|
|
|
|
// vim: noexpandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|