2017-08-29 17:00:16 -07:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// Bezier functions and modules.
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/*
|
|
|
|
BSD 2-Clause License
|
|
|
|
|
|
|
|
Copyright (c) 2017, 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
|
2018-02-16 14:49:32 -08:00
|
|
|
list of conditions and the following disclaimer.
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
* Redistributions in binary form must reproduce the above copyright notice,
|
2018-02-16 14:49:32 -08:00
|
|
|
this list of conditions and the following disclaimer in the documentation
|
|
|
|
and/or other materials provided with the distribution.
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
include <math.scad>
|
|
|
|
include <paths.scad>
|
2018-02-16 14:49:32 -08:00
|
|
|
include <transforms.scad>
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
|
2018-02-16 14:49:32 -08:00
|
|
|
// Formulae to calculate points on an N-point bezier curve.
|
|
|
|
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
|
|
|
|
);
|
2017-08-29 17:00:16 -07:00
|
|
|
|
2017-08-31 00:38:36 -07:00
|
|
|
// Takes an array of bezier points and converts it into a 3D polyline.
|
2018-02-16 14:49:32 -08:00
|
|
|
function bezier_polyline(bezier, splinesteps=16, N=3) = concat(
|
2017-08-29 17:00:16 -07:00
|
|
|
[
|
|
|
|
for (
|
2018-02-16 14:49:32 -08:00
|
|
|
b = [0 : N : len(bezier)-N-1],
|
2017-08-29 17:00:16 -07:00
|
|
|
l = [0 : splinesteps-1]
|
|
|
|
) let (
|
2018-02-16 14:49:32 -08:00
|
|
|
crv = [for (i=[0 : N]) bezier[b+i]],
|
2017-08-29 17:00:16 -07:00
|
|
|
u = l / splinesteps
|
|
|
|
) bez_point(crv, u)
|
|
|
|
],
|
2018-02-16 14:49:32 -08:00
|
|
|
[bez_point([for (i=[-(N+1) : -1]) bezier[len(bezier)+i]], 1.0)]
|
2017-08-29 17:00:16 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
// Takes a closed 2D bezier path, and creates a 2D polygon from it.
|
2018-02-16 14:49:32 -08:00
|
|
|
// bezier = array of 2D bezier path points
|
2018-10-03 18:24:07 -07:00
|
|
|
// splinesteps = number of straight lines to split each bezier segment into. default=16
|
|
|
|
// N = number of points in each bezier segment. default=3 (cubic)
|
2018-02-16 14:49:32 -08:00
|
|
|
module bezier_polygon(bezier, splinesteps=16, N=3) {
|
|
|
|
polypoints=bezier_polyline(bezier, splinesteps, N);
|
2017-08-29 17:00:16 -07:00
|
|
|
polygon(points=slice(polypoints, 0, -1));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Generate bezier curve to fillet 2 line segments between 3 points.
|
2018-02-16 14:49:32 -08:00
|
|
|
// Returns two path points with surrounding cubic (N=3) bezier control points.
|
2017-08-29 17:00:16 -07:00
|
|
|
function fillet3pts(p0, p1, p2, r) = let(
|
|
|
|
v0 = normalize(p0-p1),
|
|
|
|
v1 = normalize(p2-p1),
|
|
|
|
a = vector3d_angle(v0,v1),
|
|
|
|
mr = min(distance(p0,p1), distance(p2,p1))*0.9,
|
|
|
|
tr = min(r/tan(a/2), mr),
|
|
|
|
tp0 = p1+v0*tr,
|
|
|
|
tp1 = p1+v1*tr,
|
|
|
|
w=-2.7e-5*a*a + 8.5e-3*a - 3e-3,
|
|
|
|
nw=max(0, w),
|
|
|
|
cp0 = tp0+nw*(p1-tp0),
|
|
|
|
cp1 = tp1+nw*(p1-tp1)
|
|
|
|
) [tp0, tp0, cp0, cp1, tp1, tp1];
|
|
|
|
|
|
|
|
|
2018-11-25 15:23:30 -08:00
|
|
|
// Takes a 3D polyline path and fillets the corners, returning a 3d cubic (N=3) bezier path.
|
2017-08-29 17:00:16 -07:00
|
|
|
function fillet_path(pts, fillet) = concat(
|
|
|
|
[pts[0], pts[0]],
|
|
|
|
(len(pts) < 3)? [] : [
|
|
|
|
for (
|
|
|
|
p = [1 : len(pts)-2],
|
|
|
|
pt = fillet3pts(pts[p-1], pts[p], pts[p+1], fillet)
|
|
|
|
) pt
|
|
|
|
],
|
|
|
|
[pts[len(pts)-1], pts[len(pts)-1]]
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
// Takes a closed 2D bezier and rotates it around the X axis, forming a solid.
|
2018-10-03 18:24:07 -07:00
|
|
|
// 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
|
2017-08-29 17:00:16 -07:00
|
|
|
// Example:
|
|
|
|
// 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);
|
2018-10-03 18:24:07 -07:00
|
|
|
module revolve_bezier(bezier, splinesteps=16, N=3, convexity=10, angle=360) {
|
|
|
|
yrot(90) rotate_extrude(convexity=convexity, angle=angle) {
|
2018-02-16 14:49:32 -08:00
|
|
|
xrot(180) zrot(-90) bezier_polygon(bezier, splinesteps, N);
|
2017-08-29 17:00:16 -07:00
|
|
|
}
|
|
|
|
}
|
2018-11-24 01:37:56 -08:00
|
|
|
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
// 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
|
|
|
|
// Example:
|
|
|
|
// 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);
|
2018-10-03 18:24:07 -07:00
|
|
|
module rotate_extrude_bezier(bezier, splinesteps=16, N=3, convexity=10, angle=360) {
|
2018-11-24 01:37:56 -08:00
|
|
|
rotate_extrude(convexity=convexity, angle=angle) {
|
|
|
|
bezier_polygon(bezier, splinesteps, N);
|
|
|
|
}
|
2018-10-03 18:24:07 -07:00
|
|
|
}
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
|
2018-11-24 01:37:56 -08:00
|
|
|
|
2018-02-16 14:49:32 -08:00
|
|
|
// Takes a 2D bezier path and closes it to the X axis.
|
|
|
|
function bezier_close_to_axis(bezier, N=3) =
|
|
|
|
let(
|
|
|
|
bezend = len(bezier)-1
|
|
|
|
) concat(
|
|
|
|
[for (i=[0:N-1]) lerp([bezier[0][0], 0], bezier[0], i/N)],
|
|
|
|
bezier,
|
|
|
|
[for (i=[1:N]) lerp(bezier[bezend], [bezier[bezend][0], 0], i/N)],
|
|
|
|
[for (i=[1:N]) lerp([bezier[bezend][0], 0], [bezier[0][0], 0], i/N)]
|
|
|
|
);
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
|
|
|
|
// Takes a bezier curve and closes it with a matching path that is
|
|
|
|
// lowered by a given amount towards the X axis.
|
2018-02-16 14:49:32 -08:00
|
|
|
function bezier_offset(inset, bezier, N=3) =
|
|
|
|
let(
|
|
|
|
backbez = reverse([ for (pt = bezier) [pt[0], pt[1]-inset] ]),
|
|
|
|
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)]
|
|
|
|
);
|
2017-08-29 17:00:16 -07:00
|
|
|
|
|
|
|
|
|
|
|
// Takes a 2D bezier and rotates it around the X axis, forming a solid.
|
|
|
|
// bezier = array of points for the bezier path to rotate.
|
2018-10-03 18:24:07 -07:00
|
|
|
// 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
|
2017-08-29 17:00:16 -07:00
|
|
|
// Example:
|
|
|
|
// path = [ [0, 10], [33, 10], [66, 40], [100, 40] ];
|
|
|
|
// revolve_bezier_solid_to_axis(path, splinesteps=32, $fn=72);
|
2018-10-03 18:24:07 -07:00
|
|
|
module revolve_bezier_solid_to_axis(bezier, splinesteps=16, N=3, convexity=10, angle=360) {
|
|
|
|
revolve_bezier(bezier=bezier_close_to_axis(bezier), splinesteps=splinesteps, N=N, convexity=convexity, angle=angle);
|
2017-08-29 17:00:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Takes a 2D bezier and rotates it around the X axis, into a hollow shell.
|
|
|
|
// bezier = array of points for the bezier path to rotate.
|
|
|
|
// offset = the thickness of the created shell.
|
2018-10-03 18:24:07 -07:00
|
|
|
// 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
|
2017-08-29 17:00:16 -07:00
|
|
|
// Example:
|
|
|
|
// path = [ [0, 10], [33, 10], [66, 40], [100, 40] ];
|
|
|
|
// revolve_bezier_offset_shell(path, offset=1, splinesteps=32, $fn=72);
|
2018-10-03 18:24:07 -07:00
|
|
|
module revolve_bezier_offset_shell(bezier, offset=1, splinesteps=16, N=3, convexity=10, angle=360) {
|
2018-02-16 14:49:32 -08:00
|
|
|
revolve_bezier(bezier=bezier_offset(offset, bezier), splinesteps=splinesteps, N=N);
|
2017-08-29 17:00:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Extrudes 2D children along a bezier path.
|
|
|
|
// bezier = array of points for the bezier path to extrude along.
|
2018-10-03 18:24:07 -07:00
|
|
|
// splinesteps = number of segments to divide each bezier segment into. default=16
|
2017-08-29 17:00:16 -07:00
|
|
|
// Example:
|
|
|
|
// path = [ [0, 0, 0], [33, 33, 33], [66, -33, -33], [100, 0, 0] ];
|
2018-11-24 01:37:56 -08:00
|
|
|
// extrude_2d_shapes_along_bezier(path) circle(r=10, center=true);
|
|
|
|
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();
|
2017-08-29 17:00:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Takes a closed 2D bezier path, centered on the XY plane, and
|
|
|
|
// extrudes it perpendicularly along a 3D bezier path, forming a solid.
|
2018-10-03 18:24:07 -07:00
|
|
|
// bezier = Array of 2D points of a bezier path, to be extruded.
|
|
|
|
// path = Array of 3D points of a bezier path, to extrude along.
|
2017-08-29 17:00:16 -07:00
|
|
|
// pathsteps = number of steps to divide each path segment into.
|
|
|
|
// bezsteps = number of steps to divide each bezier segment into.
|
2018-10-03 18:24:07 -07:00
|
|
|
// bezN = number of points in each extruded bezier segment. default=3 (cubic)
|
|
|
|
// pathN = number of points in each path bezier segment. default=3 (cubic)
|
2017-08-29 17:00:16 -07:00
|
|
|
// Example:
|
2018-11-24 01:37:56 -08:00
|
|
|
// 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]
|
|
|
|
// ];
|
2017-08-29 17:00:16 -07:00
|
|
|
// path = [ [0, 0, 0], [33, 33, 33], [66, -33, -33], [100, 0, 0] ];
|
2018-11-25 15:23:30 -08:00
|
|
|
// extrude_bezier_along_bezier(bez, path, pathsteps=32, bezsteps=16);
|
2018-02-16 14:49:32 -08:00
|
|
|
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)));
|
2017-08-29 17:00:16 -07:00
|
|
|
extrude_2dpath_along_3dpath(bez_points, path_points);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-10-03 18:24:07 -07:00
|
|
|
|
|
|
|
// Takes a closed 2D bezier path, centered on the XY plane, and
|
|
|
|
// extrudes it linearly upwards, forming a solid.
|
|
|
|
// 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 = number of points in each extruded bezier segment. default = 3 (cubic)
|
|
|
|
// center = if true, the extruded solid is centered vertically at z=0.
|
|
|
|
// convexity = max number of walls a line could pass through, for preview. default=10
|
|
|
|
// twist = degrees to twist over 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
|
|
|
|
// Example:
|
2018-11-24 01:37:56 -08:00
|
|
|
// 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]
|
|
|
|
// ];
|
2018-11-25 15:23:30 -08:00
|
|
|
// linear_extrude_bezier(bez, height=20, splinesteps=32);
|
2018-10-03 18:24:07 -07:00
|
|
|
module linear_extrude_bezier(bezier, height=100, splinesteps=16, N=3, center=true, convexity=10, twist=0, slices=20, scale=1.0) {
|
|
|
|
linear_extrude(height=height, center=center, convexity=convexity, twist=twist, slices=slices, scale=scale) {
|
|
|
|
bezier_polygon(bezier, splinesteps=splinesteps, N=N);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-29 17:00:16 -07:00
|
|
|
// vim: noexpandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|