Merge pull request #762 from adrianVmariano/master

chamfer tweak
This commit is contained in:
Revar Desmera 2022-01-18 21:26:13 -08:00 committed by GitHub
commit 051f0c98a0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -44,6 +44,8 @@ include <structs.scad>
// For circular rounding you can also use the `radius` parameter, which sets a circular rounding // For circular rounding you can also use the `radius` parameter, which sets a circular rounding
// radius. // radius.
// . // .
// For chamfers you can use the `length` parameter, which sets the length of the chamfer edge.
// .
// The `"smooth"` method rounding also has a parameter that specifies how smooth the curvature match // The `"smooth"` method rounding also has a parameter that specifies how smooth the curvature match
// is. This parameter, `k`, ranges from 0 to 1, with a default of 0.5. Larger values give a more // is. This parameter, `k`, ranges from 0 to 1, with a default of 0.5. Larger values give a more
// abrupt transition and smaller ones a more gradual transition. If you set the value much higher // abrupt transition and smaller ones a more gradual transition. If you set the value much higher
@ -82,7 +84,28 @@ include <structs.scad>
// This guarantees a specific output length. It also means that if // This guarantees a specific output length. It also means that if
// you set `joint` nonzero on a flat "corner", with collinear points, you will get $fn points at that "corner." // you set `joint` nonzero on a flat "corner", with collinear points, you will get $fn points at that "corner."
// //
// Figure(2D,Med,NoAxes): // Figure(2D,Med,NoAxes): Parameters of a "circle" roundover
// h = 18;
// w = 12.6;
// strokewidth = .3;
// example = [[0,0],[w,h],[2*w,0]];
// stroke(example, width=strokewidth*1.5);
// textangle = 90-vector_angle(example)/2;
// theta = vector_angle(example)/2;
// color("green"){ stroke([[w,h], [w,h-18*(1-sin(theta))/cos(theta)]], width=strokewidth, endcaps="arrow2");
// translate([w-1.75,h-7])scale(.1)rotate(textangle)text("cut",size=14); }
// ll=lerp([w,h], [0,0],18/norm([w,h]-[0,0]) );
// color("blue"){ stroke(_shift_segment([[w,h], ll], -.7), width=strokewidth,endcaps="arrow2");
// translate([w/2-1.3,h/2+.6]) scale(.1)rotate(textangle)text("joint",size=14);}
// color("red")stroke(
// select(round_corners(example, joint=18, method="circle",$fn=64,closed=false),1,-2),
// width=strokewidth);
// r=18*tan(theta);
// color("black"){
// stroke([ll, [w,h-r-18*(1-sin(theta))/cos(theta)]], width=strokewidth, endcaps="arrow2");
// translate([w/1.6,0])text("radius", size=1.4);
// }
// Figure(2D,Med,NoAxes): Parameters of a "smooth" roundover
// h = 18; // h = 18;
// w = 12.6; // w = 12.6;
// strokewidth = .3; // strokewidth = .3;
@ -90,7 +113,6 @@ include <structs.scad>
// stroke(example, width=strokewidth*1.5); // stroke(example, width=strokewidth*1.5);
// textangle = 90-vector_angle(example)/2; // textangle = 90-vector_angle(example)/2;
// color("green"){ stroke([[w,h], [w,h-cos(vector_angle(example)/2) *3/8*h]], width=strokewidth, endcaps="arrow2"); // color("green"){ stroke([[w,h], [w,h-cos(vector_angle(example)/2) *3/8*h]], width=strokewidth, endcaps="arrow2");
// //translate([w-.3,h-4.4])scale(.1)rotate(90)text("cut",size=12); }
// translate([w-1.75,h-5.5])scale(.1)rotate(textangle)text("cut",size=14); } // translate([w-1.75,h-5.5])scale(.1)rotate(textangle)text("cut",size=14); }
// ll=lerp([w,h], [0,0],18/norm([w,h]-[0,0]) ); // ll=lerp([w,h], [0,0],18/norm([w,h]-[0,0]) );
// color("blue"){ stroke(_shift_segment([[w,h], ll], -.7), width=strokewidth,endcaps="arrow2"); // color("blue"){ stroke(_shift_segment([[w,h], ll], -.7), width=strokewidth,endcaps="arrow2");
@ -98,6 +120,27 @@ include <structs.scad>
// color("red")stroke( // color("red")stroke(
// select(round_corners(example, joint=18, method="smooth",closed=false),1,-2), // select(round_corners(example, joint=18, method="smooth",closed=false),1,-2),
// width=strokewidth); // width=strokewidth);
// Figure(2D,Med,NoAxes): Parameters of "chamfer"
// h = 18;
// w = 12.6;
// strokewidth = .3;
// example = [[0,0],[w,h],[2*w,0]];
// stroke(example, width=strokewidth*1.5);
// textangle = 90-vector_angle(example)/2;
// color("black"){
// stroke(fwd(1,
// select(round_corners(example, joint=18, method="chamfer",closed=false),1,-2)),
// width=strokewidth,endcaps="arrow2");
// translate([w,.3])text("length", size=1.4,halign="center");
// }
// color("green"){ stroke([[w,h], [w,h-18*cos(vector_angle(example)/2)]], width=strokewidth, endcaps="arrow2");
// translate([w-1.75,h-5.5])scale(.1)rotate(textangle)text("cut",size=14); }
// ll=lerp([w,h], [0,0],18/norm([w,h]-[0,0]) );
// color("blue"){ stroke(_shift_segment([[w,h], ll], -.7), width=strokewidth,endcaps="arrow2");
// translate([w/2-1.3,h/2+.6]) rotate(textangle)text("joint",size=1.4);}
// color("red")stroke(
// select(round_corners(example, joint=18, method="chamfer",closed=false),1,-2),
// width=strokewidth);
// Arguments: // Arguments:
// path = list of 2d or 3d points defining the path to be rounded. // path = list of 2d or 3d points defining the path to be rounded.
// method = rounding method to use. Set to "chamfer" for chamfers, "circle" for circular rounding and "smooth" for continuous curvature 4th order bezier rounding. Default: "circle" // method = rounding method to use. Set to "chamfer" for chamfers, "circle" for circular rounding and "smooth" for continuous curvature 4th order bezier rounding. Default: "circle"
@ -105,6 +148,7 @@ include <structs.scad>
// radius = rounding radius, only compatible with `method="circle"`. Can be a number or vector. // radius = rounding radius, only compatible with `method="circle"`. Can be a number or vector.
// cut = rounding cut distance, compatible with all methods. Can be a number or vector. // cut = rounding cut distance, compatible with all methods. Can be a number or vector.
// joint = rounding joint distance, compatible with `method="chamfer"` and `method="smooth"`. Can be a number or vector. // joint = rounding joint distance, compatible with `method="chamfer"` and `method="smooth"`. Can be a number or vector.
// flat = length of the flat edge created by chamfering, compatible with `method="chamfer"`. Can be a number of vector.
// k = continuous curvature smoothness parameter for `method="smooth"`. Can be a number or vector. Default: 0.5 // k = continuous curvature smoothness parameter for `method="smooth"`. Can be a number or vector. Default: 0.5
// closed = if true treat the path as a closed polygon, otherwise treat it as open. Default: true. // closed = if true treat the path as a closed polygon, otherwise treat it as open. Default: true.
// verbose = if true display rounding scale factors that show how close roundovers are to overlapping. Default: false // verbose = if true display rounding scale factors that show how close roundovers are to overlapping. Default: false
@ -227,9 +271,9 @@ include <structs.scad>
// path_len = path_segment_lengths(path,closed=true); // path_len = path_segment_lengths(path,closed=true);
// halflen = [for(i=idx(path)) min(select(path_len,i-1,i))/2]; // halflen = [for(i=idx(path)) min(select(path_len,i-1,i))/2];
// polygon(round_corners(path,joint = halflen, method="circle",verbose=true)); // polygon(round_corners(path,joint = halflen, method="circle",verbose=true));
// Example(2D): Chamfering, specifying the flats // Example(2D): Chamfering, specifying the chamfer length
// path = star(5, step=2, d=100); // path = star(5, step=2, d=100);
// path2 = round_corners(path, method="chamfer", flat=5); // path2 = round_corners(path, method="chamfer", length=5);
// polygon(path2); // polygon(path2);
// Example(2D): Chamfering, specifying the cut // Example(2D): Chamfering, specifying the cut
// path = star(5, step=2, d=100); // path = star(5, step=2, d=100);
@ -259,19 +303,36 @@ include <structs.scad>
// cut=chamfcut), // cut=chamfcut),
// radius=radii); // radius=radii);
// stroke(path2, closed=true); // stroke(path2, closed=true);
module round_corners(path, method="circle", radius, cut, joint, flat, k, closed=true, verbose=false) {no_module();} // Example(2D): Specifying by corner index. Use {{list_set()}} to construct the full chamfer cut list.
function round_corners(path, method="circle", radius, cut, joint, flat, k, closed=true, verbose=false) = // path = star(47, ir=25, or=50); // long path, lots of corners
// chamfind = [8, 28, 60]; // But only want 3 chamfers
// chamfcut = list_set([],chamfind,[10,13,15],minlen=len(path)-1);
// rpath = round_corners(path, cut=chamfcut, method="chamfer");
// polygon(rpath);
// Example(2D): Two-pass to chamfer and round by index. Use {{repeat_entries()}} to correct for first pass chamfers.
// $fn=32;
// path = star(47, ir=25, or=50); // long path, lots of corners
// chamfind = [8, 28, 60]; // But only want 3 chamfers
// roundind = [7,9,27,29,59,61]; // And 6 roundovers
// chamfcut = list_set([],chamfind,[10,13,15],minlen=len(path)-1);
// roundcut = list_set([],roundind,repeat(8,6),minlen=len(path)-1);
// dups = list_set([], chamfind, repeat(2,len(chamfind)), dflt=1, minlen=len(path)-1);
// rpath1 = round_corners(path, cut=chamfcut, method="chamfer");
// rpath2 = round_corners(rpath1, cut=repeat_entries(roundcut,dups));
// polygon(rpath2);
module round_corners(path, method="circle", radius, cut, joint, length, k, closed=true, verbose=false) {no_module();}
function round_corners(path, method="circle", radius, cut, joint, length, k, closed=true, verbose=false) =
assert(in_list(method,["circle", "smooth", "chamfer"]), "method must be one of \"circle\", \"smooth\" or \"chamfer\"") assert(in_list(method,["circle", "smooth", "chamfer"]), "method must be one of \"circle\", \"smooth\" or \"chamfer\"")
let( let(
default_k = 0.5, default_k = 0.5,
size=one_defined([radius, cut, joint, flat], "radius,cut,joint,flat"), size=one_defined([radius, cut, joint, length], "radius,cut,joint,length"),
path = force_path(path), path = force_path(path),
size_ok = is_num(size) || len(size)==len(path) || (!closed && len(size)==len(path)-2), size_ok = is_num(size) || len(size)==len(path) || (!closed && len(size)==len(path)-2),
k_ok = is_undef(k) || (method=="smooth" && (is_num(k) || len(k)==len(path) || (!closed && len(k)==len(path)-2))), k_ok = is_undef(k) || (method=="smooth" && (is_num(k) || len(k)==len(path) || (!closed && len(k)==len(path)-2))),
measure = is_def(radius) ? "radius" measure = is_def(radius) ? "radius"
: is_def(cut) ? "cut" : is_def(cut) ? "cut"
: is_def(joint) ? "joint" : is_def(joint) ? "joint"
: "flat" : "length"
) )
assert(is_path(path,[2,3]), "input path must be a 2d or 3d path") assert(is_path(path,[2,3]), "input path must be a 2d or 3d path")
assert(len(path)>2,str("Path has length ",len(path),". Length must be 3 or more.")) assert(len(path)>2,str("Path has length ",len(path),". Length must be 3 or more."))
@ -279,7 +340,7 @@ function round_corners(path, method="circle", radius, cut, joint, flat, k, close
assert(k_ok,method=="smooth" ? str("Input k must be a number or list with length ",len(path), closed?"":str(" or ",len(path)-2)) : assert(k_ok,method=="smooth" ? str("Input k must be a number or list with length ",len(path), closed?"":str(" or ",len(path)-2)) :
"Input k is only allowed with method=\"smooth\"") "Input k is only allowed with method=\"smooth\"")
assert(method=="circle" || measure!="radius", "radius parameter allowed only with method=\"circle\"") assert(method=="circle" || measure!="radius", "radius parameter allowed only with method=\"circle\"")
assert(method=="chamfer" || measure!="flat", "flat parameter allowed only with method=\"chamfer\"") assert(method=="chamfer" || measure!="length", "length parameter allowed only with method=\"chamfer\"")
let( let(
parm = is_num(size) ? repeat(size, len(path)) : parm = is_num(size) ? repeat(size, len(path)) :
len(size)<len(path) ? [0, each size, 0] : len(size)<len(path) ? [0, each size, 0] :
@ -313,7 +374,7 @@ function round_corners(path, method="circle", radius, cut, joint, flat, k, close
str("Path turns back on itself at index ",i," with nonzero rounding")) str("Path turns back on itself at index ",i," with nonzero rounding"))
(method=="chamfer" && measure=="joint")? [parm[i]] : (method=="chamfer" && measure=="joint")? [parm[i]] :
(method=="chamfer" && measure=="cut") ? [parm[i]/cos(angle)] : (method=="chamfer" && measure=="cut") ? [parm[i]/cos(angle)] :
(method=="chamfer" && measure=="flat") ? [parm[i]/sin(angle)/2] : (method=="chamfer" && measure=="length") ? [parm[i]/sin(angle)/2] :
(method=="smooth" && measure=="joint") ? [parm[i],k[i]] : (method=="smooth" && measure=="joint") ? [parm[i],k[i]] :
(method=="smooth" && measure=="cut") ? [8*parm[i]/cos(angle)/(1+4*k[i]),k[i]] : (method=="smooth" && measure=="cut") ? [8*parm[i]/cos(angle)/(1+4*k[i]),k[i]] :
(method=="circle" && measure=="radius")? [parm[i]/tan(angle), parm[i]] : (method=="circle" && measure=="radius")? [parm[i]/tan(angle), parm[i]] :