mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-23 04:49:39 +00:00
Compare commits
2 commits
606b6acde2
...
ab2d381034
Author | SHA1 | Date | |
---|---|---|---|
|
ab2d381034 | ||
|
7cefef9977 |
2 changed files with 34 additions and 4 deletions
34
joiners.scad
34
joiners.scad
|
@ -1263,6 +1263,13 @@ module rabbit_clip(type, length, width, snap, thickness, depth, compression=0.1
|
||||||
// For two hirth joints to mate they must have the same tooth count, opposite cone angles, and the chamfer/rounding values
|
// For two hirth joints to mate they must have the same tooth count, opposite cone angles, and the chamfer/rounding values
|
||||||
// must be equal. (One can be chamfered and one rounded, but with the same value.) The rotation required to mate the parts
|
// must be equal. (One can be chamfered and one rounded, but with the same value.) The rotation required to mate the parts
|
||||||
// depends on the skew and whether the tooth count is odd or even. To apply this rotation automatically, set `rot=true`.
|
// depends on the skew and whether the tooth count is odd or even. To apply this rotation automatically, set `rot=true`.
|
||||||
|
// .
|
||||||
|
// When you pick extreme parameters such as very large cone angle, or very small tooth count (e.g. 2 or 3), the joint may
|
||||||
|
// develop a weird shape, and the shape may be unexpectedly sensitive to things like whether chamfering is enabled. It is difficult
|
||||||
|
// to identify the point where the shapes become odd, or even perhaps invalid. For example, with 2 teeth a skew of 0.95 works fine, but
|
||||||
|
// a skew of 0.97 produces a truncated shape and 0.99 produces a 2-part shape. A skew of 1 produces a degenerate, invalid shape.
|
||||||
|
// Since it's hard to determine which parameters, exactly, produce "bad" outcomes, we have chosen not to limit the production
|
||||||
|
// of the extreme shapes, so take care if using extreme parameter values.
|
||||||
// Named Anchors:
|
// Named Anchors:
|
||||||
// "teeth_bot" = center of the joint, aligned with the bottom of the (unchamfered/unrounded) teeth, pointing DOWN.
|
// "teeth_bot" = center of the joint, aligned with the bottom of the (unchamfered/unrounded) teeth, pointing DOWN.
|
||||||
// Arguments:
|
// Arguments:
|
||||||
|
@ -1351,7 +1358,7 @@ module hirth(n, ir, or, id, od, tooth_angle=60, cone_angle=0, chamfer, rounding,
|
||||||
];
|
];
|
||||||
full = deduplicate(concat(basicprof, reverse(xflip(basicprof))));
|
full = deduplicate(concat(basicprof, reverse(xflip(basicprof))));
|
||||||
skewed = back(valley_angle, skew(sxy=skew*angle/(ridge_angle-valley_angle),fwd(valley_angle,full)));
|
skewed = back(valley_angle, skew(sxy=skew*angle/(ridge_angle-valley_angle),fwd(valley_angle,full)));
|
||||||
profile = is_undef(rounding) ? skewed
|
pprofile = is_undef(rounding) ? skewed
|
||||||
:
|
:
|
||||||
let(
|
let(
|
||||||
segs = max(16,segs(or*rounding)),
|
segs = max(16,segs(or*rounding)),
|
||||||
|
@ -1360,20 +1367,43 @@ module hirth(n, ir, or, id, od, tooth_angle=60, cone_angle=0, chamfer, rounding,
|
||||||
roundpts = round_corners(skewed, joint=joints, closed=false,$fn=segs)
|
roundpts = round_corners(skewed, joint=joints, closed=false,$fn=segs)
|
||||||
)
|
)
|
||||||
roundpts;
|
roundpts;
|
||||||
|
profile = [
|
||||||
|
for(i=[0:1:len(pprofile)-2]) each [pprofile[i],
|
||||||
|
if (pprofile[i+1].x-pprofile[i].x > 90) // Interpolate an extra point if angle > 90 deg
|
||||||
|
let(
|
||||||
|
edge = cyl_proj(or, select(pprofile,i,i+1)),
|
||||||
|
cutpt = xyz_to_spherical(lerp(edge[0],edge[1],.48)) // Exactly .5 is too close to or crosses the origin
|
||||||
|
)
|
||||||
|
[cutpt.y,90-cutpt.z]
|
||||||
|
],
|
||||||
|
last(pprofile)
|
||||||
|
];
|
||||||
|
|
||||||
|
// This code computes the realized tooth angle
|
||||||
|
// out = cyl_proj(or, pprofile);
|
||||||
|
// in = cyl_proj(ir,pprofile);
|
||||||
|
// p1 = plane3pt(out[0], out[1], in[1]);
|
||||||
|
// p2 = plane3pt(out[2], out[1], in[1]);
|
||||||
|
// echo(toothang=vector_angle(plane_normal(p1), plane_normal(p2)));
|
||||||
|
|
||||||
bottom = min([tan(valley_angle)*ir,tan(valley_angle)*or])-base-cone_height*ir;
|
bottom = min([tan(valley_angle)*ir,tan(valley_angle)*or])-base-cone_height*ir;
|
||||||
safebottom = min([tan(valley_angle)*ir/factor,tan(valley_angle)*or*factor])-base-(crop?1:0)-cone_height*ir;
|
|
||||||
ang_ofs = !rot ? -skew*angle
|
ang_ofs = !rot ? -skew*angle
|
||||||
: n%2==0 ? -(angle-skew*angle) - skew*angle
|
: n%2==0 ? -(angle-skew*angle) - skew*angle
|
||||||
: -angle*(2-skew)-skew*angle;
|
: -angle*(2-skew)-skew*angle;
|
||||||
|
|
||||||
topinner = down(cone_height*ir,[for(ang=lerpn(0,360,n,endpoint=false))
|
topinner = down(cone_height*ir,[for(ang=lerpn(0,360,n,endpoint=false))
|
||||||
each zrot(ang+ang_ofs,cyl_proj(ir/factor,profile))]);
|
each zrot(ang+ang_ofs,cyl_proj(ir/factor,profile))]);
|
||||||
topouter = down(cone_height*ir,[for(ang=lerpn(0,360,n,endpoint=false))
|
topouter = down(cone_height*ir,[for(ang=lerpn(0,360,n,endpoint=false))
|
||||||
each zrot(ang+ang_ofs,cyl_proj(factor*or,profile))]);
|
each zrot(ang+ang_ofs,cyl_proj(factor*or,profile))]);
|
||||||
|
|
||||||
|
safebottom = min(min(column(topinner,2)), min(column(topouter,2))) - base - (crop?1:0);
|
||||||
|
|
||||||
botinner = [for(val=topinner) [val.x,val.y,safebottom]];
|
botinner = [for(val=topinner) [val.x,val.y,safebottom]];
|
||||||
botouter = [for(val=topouter) [val.x,val.y,safebottom]];
|
botouter = [for(val=topouter) [val.x,val.y,safebottom]];
|
||||||
vert = [topouter, topinner, botinner, botouter];
|
vert = [topouter, topinner, botinner, botouter];
|
||||||
|
|
||||||
|
datamin = min(min(column(topinner,2)), min(column(topouter,2)));
|
||||||
|
|
||||||
anchors = [
|
anchors = [
|
||||||
named_anchor("teeth_bot", [0,0,bottom], DOWN)
|
named_anchor("teeth_bot", [0,0,bottom], DOWN)
|
||||||
];
|
];
|
||||||
|
|
|
@ -47,7 +47,7 @@
|
||||||
// is_vector([],zero=false); // Returns false
|
// is_vector([],zero=false); // Returns false
|
||||||
function is_vector(v, length, zero, all_nonzero=false, eps=EPSILON) =
|
function is_vector(v, length, zero, all_nonzero=false, eps=EPSILON) =
|
||||||
is_list(v) && len(v)>0 && []==[for(vi=v) if(!is_finite(vi)) 0]
|
is_list(v) && len(v)>0 && []==[for(vi=v) if(!is_finite(vi)) 0]
|
||||||
&& (is_undef(length) || len(v)==length)
|
&& (is_undef(length) || (assert(is_num(length))len(v)==length))
|
||||||
&& (is_undef(zero) || ((norm(v) >= eps) == !zero))
|
&& (is_undef(zero) || ((norm(v) >= eps) == !zero))
|
||||||
&& (!all_nonzero || all_nonzero(v)) ;
|
&& (!all_nonzero || all_nonzero(v)) ;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue