mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-04 03:09:45 +00:00
Fix manifold issues with trapezoidal_threaded_rod().
This commit is contained in:
parent
99072122ce
commit
323a42177d
3 changed files with 101 additions and 109 deletions
|
@ -103,7 +103,7 @@ module thread_helix(
|
||||||
// internal = If true, make this a mask for making internal threads.
|
// internal = If true, make this a mask for making internal threads.
|
||||||
// d1 = Bottom outside diameter of threads.
|
// d1 = Bottom outside diameter of threads.
|
||||||
// d2 = Top outside diameter of threads.
|
// d2 = Top outside diameter of threads.
|
||||||
// higbee = Length to taper thread ends over. Default: 0
|
// higbee = Length to taper thread ends over. Default: 0 (No higbee thread tapering)
|
||||||
// higbee1 = Length to taper bottom thread end over.
|
// higbee1 = Length to taper bottom thread end over.
|
||||||
// higbee2 = Length to taper top thread end over.
|
// higbee2 = Length to taper top thread end over.
|
||||||
// center = If given, overrides `anchor`. A true value sets `anchor=CENTER`, false sets `anchor=UP`.
|
// center = If given, overrides `anchor`. A true value sets `anchor=CENTER`, false sets `anchor=UP`.
|
||||||
|
@ -148,90 +148,92 @@ module trapezoidal_threaded_rod(
|
||||||
profile,
|
profile,
|
||||||
internal=false,
|
internal=false,
|
||||||
d1, d2,
|
d1, d2,
|
||||||
higbee=0, higbee1, higbee2,
|
higbee, higbee1, higbee2,
|
||||||
center, anchor, spin, orient
|
center, anchor, spin, orient
|
||||||
) {
|
) {
|
||||||
_r1 = get_radius(d1=d1, d=d, dflt=10);
|
_r1 = get_radius(d1=d1, d=d, dflt=10);
|
||||||
_r2 = get_radius(d1=d2, d=d, dflt=10);
|
_r2 = get_radius(d1=d2, d=d, dflt=10);
|
||||||
sides = quantup(segs(max(_r1,_r2)), starts);
|
sides = quantup(segs(max(_r1,_r2)), starts);
|
||||||
rsc = internal? (1/cos(180/sides) + $slop*3) : 1;
|
rsc = internal? (1/cos(180/sides) + $slop*3) : 1;
|
||||||
threads = ceil(l/pitch/starts)+(starts<4?4-starts:1);
|
threads = ceil(l/pitch/starts) + 2;
|
||||||
|
ll = threads * pitch * starts;
|
||||||
depth = min((thread_depth==undef? pitch/2 : thread_depth), pitch/2/tan(thread_angle));
|
depth = min((thread_depth==undef? pitch/2 : thread_depth), pitch/2/tan(thread_angle));
|
||||||
pa_delta = min(pitch/4-0.01,depth*tan(thread_angle)/2)/pitch;
|
pa_delta = min(pitch/4-0.01,depth*tan(thread_angle)/2)/pitch;
|
||||||
dir = left_handed? -1 : 1;
|
dir = left_handed? -1 : 1;
|
||||||
twist = 360 * l / pitch / starts;
|
twist = 360 * l / pitch / starts;
|
||||||
higbee1 = first_defined([higbee1, higbee, 0]);
|
_higbee1 = first_defined([higbee1, higbee, 0]);
|
||||||
higbee2 = first_defined([higbee2, higbee, 0]);
|
_higbee2 = first_defined([higbee2, higbee, 0]);
|
||||||
higang1 = 360 * higbee1 / (2 * _r1 * PI);
|
higang1 = 360 * _higbee1 / (2 * PI * _r1);
|
||||||
higang2 = 360 * higbee2 / (2 * _r2 * PI);
|
higang2 = 360 * _higbee2 / (2 * PI * _r2);
|
||||||
higsteps1 = ceil(higang1/360*sides);
|
|
||||||
higsteps2 = ceil(higang2/360*sides);
|
|
||||||
assert(higang1 < twist/2);
|
assert(higang1 < twist/2);
|
||||||
assert(higang2 < twist/2);
|
assert(higang2 < twist/2);
|
||||||
|
|
||||||
|
higstart = twist/2 + 360/starts/4;
|
||||||
higbee_table = [
|
higbee_table = [
|
||||||
[-twist, 0.01],
|
[-higstart*2, 0.01],
|
||||||
[-twist/2, 0.01],
|
[-higstart-0.001, 0.01],
|
||||||
[-twist/2+higang1, 1],
|
[-higstart+higang1, 1 ],
|
||||||
[ twist/2-higang2, 1],
|
[+higstart-higang2, 1 ],
|
||||||
[ twist/2, 0.01],
|
[+higstart+0.001, 0.01],
|
||||||
[ twist, 0.01]
|
[+higstart*2, 0.01]
|
||||||
];
|
];
|
||||||
|
echo(higbee_table);
|
||||||
|
|
||||||
r1 = -depth/pitch;
|
r1 = -depth/pitch;
|
||||||
z1 = 1/4-pa_delta;
|
z1 = 1/4-pa_delta;
|
||||||
z2 = 1/4+pa_delta;
|
z2 = 1/4+pa_delta;
|
||||||
profile = profile!=undef? profile : [
|
profile = pitch * (
|
||||||
|
profile!=undef? profile : [
|
||||||
[-z2, r1],
|
[-z2, r1],
|
||||||
[-z1, 0],
|
[-z1, 0],
|
||||||
[ z1, 0],
|
[ z1, 0],
|
||||||
[ z2, r1],
|
[ z2, r1],
|
||||||
];
|
]
|
||||||
|
);
|
||||||
pdepth = -min(subindex(profile,1));
|
pdepth = -min(subindex(profile,1));
|
||||||
eprofile = [
|
eprofile = [
|
||||||
[-0.5, 0],
|
|
||||||
each move([0,pdepth], p=profile),
|
each move([0,pdepth], p=profile),
|
||||||
[ 0.5, 0],
|
move([pitch,pdepth], p=profile[0]),
|
||||||
] * pitch;
|
];
|
||||||
angstep = 360 / sides;
|
angstep = 360 / sides;
|
||||||
angsteps = ceil(twist / (360 / sides)) + sides;
|
angsteps = ceil(sides * (twist / 360 + 2));
|
||||||
zang = atan2(_r2-_r1,l);
|
zang = atan2(_r2-_r1,l);
|
||||||
thread_verts = [
|
thread_verts = [
|
||||||
[for (x = eprofile) [0,0,-l/2]],
|
[for (i = idx(eprofile)) [0,0,-ll/2]],
|
||||||
for (a = [0:1:angsteps]) let (
|
for (thread = [0:1:threads-1], side=[0:1:sides-1]) let(
|
||||||
u = (a-angsteps/2) / (angsteps-sides),
|
ang = ((thread - threads/2) + (side / sides)) * 360,
|
||||||
ang = u * twist,
|
u = ang / twist,
|
||||||
r = lerp(_r1, _r2, u) * rsc,
|
r = lerp(_r1, _r2, u) * rsc,
|
||||||
hsc = higbee1==0 && higbee2==0? 1 : lookup(ang, higbee_table),
|
hsc = higbee1==0 && higbee2==0? 1 : lookup(ang, higbee_table),
|
||||||
mat = affine3d_zrot(ang*dir) *
|
mat = affine3d_zrot(ang*dir) *
|
||||||
affine3d_translate([r-pdepth*pitch, 0, l*u-0*pitch]) *
|
affine3d_translate([r-pdepth*pitch, 0, l*u]) *
|
||||||
affine3d_xrot(90) *
|
affine3d_xrot(90) *
|
||||||
affine3d_skew_xz(xa=zang) *
|
affine3d_skew_xz(xa=zang) *
|
||||||
affine3d_mirror([-1,1]) *
|
affine3d_mirror([-1,1]) *
|
||||||
affine3d_scale([1,hsc,1]),
|
affine3d_scale([1,hsc,1]),
|
||||||
pts = apply(mat, path3d(eprofile))
|
pts = apply(mat, path3d(eprofile))
|
||||||
) pts,
|
) pts,
|
||||||
[for (x = eprofile) [0,0, l/2]],
|
[for (x = eprofile) [0,0,+ll/2]],
|
||||||
];
|
];
|
||||||
thread_vnf = vnf_vertex_array(thread_verts, reverse=left_handed);
|
thread_vnf = vnf_vertex_array(thread_verts, reverse=left_handed);
|
||||||
eplen = len(eprofile);
|
eplen = len(eprofile);
|
||||||
vlen = len(thread_vnf[0]);
|
vlen = len(thread_vnf[0]);
|
||||||
thread_vnf2 = [
|
thread_vnf2 = [
|
||||||
concat(thread_vnf[0], [[0,0,-l/2], [0,0,l/2]]),
|
concat(thread_vnf[0], [[0,0,-ll/2], [0,0,+ll/2]]),
|
||||||
concat(thread_vnf[1], [
|
concat(thread_vnf[1], [
|
||||||
for (i = [0:1:sides/starts]) each
|
for (i = [0:1:sides/starts]) each
|
||||||
left_handed? [
|
left_handed? [
|
||||||
[eplen*(i+1), eplen*i, vlen],
|
[eplen*(i+1), eplen*i, vlen],
|
||||||
[vlen-eplen*(i+1)-1, vlen-eplen*i-1, vlen+1]
|
[vlen-eplen*(i+1)-1, vlen-eplen*(i+0)-1, vlen+1]
|
||||||
] : [
|
] : [
|
||||||
[eplen*i, eplen*(i+1), vlen],
|
[eplen*i, eplen*(i+1), vlen],
|
||||||
[vlen-eplen*i-1, vlen-eplen*(i+1)-1, vlen+1]
|
[vlen-eplen*(i+0)-1, vlen-eplen*(i+1)-1, vlen+1]
|
||||||
]
|
]
|
||||||
])
|
])
|
||||||
];
|
];
|
||||||
thread_vnfs = vnf_merge([
|
thread_vnfs = vnf_merge([
|
||||||
for (start = [0:1:starts-1]) zrot(start*360/starts, p=thread_vnf2)
|
for (start = [0:1:starts-1]) zrot(start*360/starts, p=thread_vnf2)
|
||||||
]);
|
], cleanup=true);
|
||||||
anchor = get_anchor(anchor, center, BOT, CENTER);
|
anchor = get_anchor(anchor, center, BOT, CENTER);
|
||||||
attachable(anchor,spin,orient, r1=_r1, r2=_r2, l=l) {
|
attachable(anchor,spin,orient, r1=_r1, r2=_r2, l=l) {
|
||||||
difference() {
|
difference() {
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
BOSL_VERSION = [2,0,555];
|
BOSL_VERSION = [2,0,556];
|
||||||
|
|
||||||
|
|
||||||
// Section: BOSL Library Version Functions
|
// Section: BOSL Library Version Functions
|
||||||
|
|
134
vnf.scad
134
vnf.scad
|
@ -145,18 +145,31 @@ function vnf_add_faces(vnf=EMPTY_VNF, faces) =
|
||||||
|
|
||||||
// Function: vnf_merge()
|
// Function: vnf_merge()
|
||||||
// Usage:
|
// Usage:
|
||||||
// vnf = vnf_merge([VNF, VNF, VNF, ...]);
|
// vnf = vnf_merge([VNF, VNF, VNF, ...], <cleanup>);
|
||||||
// Description:
|
// Description:
|
||||||
// Given a list of VNF structures, merges them all into a single VNF structure.
|
// Given a list of VNF structures, merges them all into a single VNF structure.
|
||||||
function vnf_merge(vnfs=[],_i=0,_acc=EMPTY_VNF) =
|
function vnf_merge(vnfs, cleanup=false) =
|
||||||
(assert(is_vnf_list(vnfs)) _i>=len(vnfs))? _acc :
|
let (
|
||||||
vnf_merge(
|
offs = cumsum([
|
||||||
vnfs, _i=_i+1,
|
0, for (vnf = vnfs) len(vnf[0])
|
||||||
_acc = let(base=len(_acc[0])) [
|
])
|
||||||
concat(_acc[0], vnfs[_i][0]),
|
) [
|
||||||
concat(_acc[1], [for (f=vnfs[_i][1]) [for (i=f) i+base]]),
|
[for (vnf=vnfs) each vnf[0]],
|
||||||
|
[
|
||||||
|
for (i = idx(vnfs)) let(
|
||||||
|
vnf = vnfs[i],
|
||||||
|
verts = vnf[0],
|
||||||
|
faces = vnf[1]
|
||||||
|
)
|
||||||
|
for (face = faces) let(
|
||||||
|
dface = !cleanup ? face :
|
||||||
|
deduplicate_indexed(verts, face, closed=true)
|
||||||
|
)
|
||||||
|
if (len(dface) >= 3)
|
||||||
|
[ for (j = dface) offs[i] + j ]
|
||||||
]
|
]
|
||||||
);
|
];
|
||||||
|
|
||||||
|
|
||||||
// Function: vnf_compact()
|
// Function: vnf_compact()
|
||||||
// Usage:
|
// Usage:
|
||||||
|
@ -281,7 +294,7 @@ function vnf_vertex_array(
|
||||||
rowcnt = rows - (row_wrap?0:1)
|
rowcnt = rows - (row_wrap?0:1)
|
||||||
)
|
)
|
||||||
rows<=1 || cols<=1 ? vnf :
|
rows<=1 || cols<=1 ? vnf :
|
||||||
vnf_merge([
|
vnf_merge(cleanup=true, [
|
||||||
vnf, [
|
vnf, [
|
||||||
concat(
|
concat(
|
||||||
pts,
|
pts,
|
||||||
|
@ -681,13 +694,8 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
uniq_edges = edgecnts[0],
|
uniq_edges = edgecnts[0],
|
||||||
big_faces = !show_warns? [] : [
|
big_faces = !show_warns? [] : [
|
||||||
for (face = faces)
|
for (face = faces)
|
||||||
if (len(face) > 3) [
|
if (len(face) > 3)
|
||||||
"WARNING",
|
_vnf_validate_err("BIG_FACE", [for (i=face) varr[i]])
|
||||||
"BIG_FACE",
|
|
||||||
"Face has more than 3 vertices, and may confuse CGAL",
|
|
||||||
[for (i=face) varr[i]],
|
|
||||||
"yellow"
|
|
||||||
]
|
|
||||||
],
|
],
|
||||||
null_faces = !show_warns? [] : [
|
null_faces = !show_warns? [] : [
|
||||||
for (face = faces) let(
|
for (face = faces) let(
|
||||||
|
@ -696,13 +704,9 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
if (len(face)>=3) let(
|
if (len(face)>=3) let(
|
||||||
faceverts = [for (k=face) varr[k]],
|
faceverts = [for (k=face) varr[k]],
|
||||||
area = polygon_area(faceverts)
|
area = polygon_area(faceverts)
|
||||||
) if (is_num(area) && abs(area) < EPSILON) [
|
)
|
||||||
"WARNING",
|
if (is_num(area) && abs(area) < EPSILON)
|
||||||
"NULL_FACE",
|
_vnf_validate_err("NULL_FACE", faceverts)
|
||||||
str("Face has zero area: ",fmt_float(abs(area),15)),
|
|
||||||
faceverts,
|
|
||||||
"brown"
|
|
||||||
]
|
|
||||||
],
|
],
|
||||||
nonplanars = unique([
|
nonplanars = unique([
|
||||||
for (face = faces) let(
|
for (face = faces) let(
|
||||||
|
@ -710,23 +714,13 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
area = polygon_area(faceverts)
|
area = polygon_area(faceverts)
|
||||||
)
|
)
|
||||||
if (is_num(area) && abs(area) > EPSILON)
|
if (is_num(area) && abs(area) > EPSILON)
|
||||||
if (!coplanar(faceverts)) [
|
if (!coplanar(faceverts))
|
||||||
"ERROR",
|
_vnf_validate_err("NONPLANAR", faceverts)
|
||||||
"NONPLANAR",
|
|
||||||
"Face vertices are not coplanar",
|
|
||||||
faceverts,
|
|
||||||
"cyan"
|
|
||||||
]
|
|
||||||
]),
|
]),
|
||||||
overpop_edges = unique([
|
overpop_edges = unique([
|
||||||
for (i=idx(uniq_edges))
|
for (i=idx(uniq_edges))
|
||||||
if (edgecnts[1][i]>2) [
|
if (edgecnts[1][i]>2)
|
||||||
"ERROR",
|
_vnf_validate_err("OVRPOP_EDGE", [for (i=uniq_edges[i]) varr[i]])
|
||||||
"OVRPOP_EDGE",
|
|
||||||
"Too many faces attached at Edge",
|
|
||||||
[for (i=uniq_edges[i]) varr[i]],
|
|
||||||
"#f70"
|
|
||||||
]
|
|
||||||
]),
|
]),
|
||||||
reversals = unique([
|
reversals = unique([
|
||||||
for(i = idx(faces), j = idx(faces)) if(i != j)
|
for(i = idx(faces), j = idx(faces)) if(i != j)
|
||||||
|
@ -736,13 +730,7 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
for(edge2 = pair(faces[j],true))
|
for(edge2 = pair(faces[j],true))
|
||||||
if(edge1 == edge2) // Valid adjacent faces will never have the same vertex ordering.
|
if(edge1 == edge2) // Valid adjacent faces will never have the same vertex ordering.
|
||||||
if(_edge_not_reported(edge1, varr, overpop_edges))
|
if(_edge_not_reported(edge1, varr, overpop_edges))
|
||||||
[
|
_vnf_validate_err("REVERSAL", [for (i=edge1) varr[i]])
|
||||||
"ERROR",
|
|
||||||
"REVERSAL",
|
|
||||||
"Faces Reverse Across Edge",
|
|
||||||
[for (i=edge1) varr[i]],
|
|
||||||
"violet"
|
|
||||||
]
|
|
||||||
]),
|
]),
|
||||||
t_juncts = unique([
|
t_juncts = unique([
|
||||||
for (v=idx(varr), edge=uniq_edges)
|
for (v=idx(varr), edge=uniq_edges)
|
||||||
|
@ -754,13 +742,8 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
if (a != b && b != c && a != c) let(
|
if (a != b && b != c && a != c) let(
|
||||||
pt = segment_closest_point([a,c],b)
|
pt = segment_closest_point([a,c],b)
|
||||||
)
|
)
|
||||||
if (pt == b) [
|
if (pt == b)
|
||||||
"ERROR",
|
_vnf_validate_err("T_JUNCTION", [b])
|
||||||
"T_JUNCTION",
|
|
||||||
"Vertex is mid-edge on another Face",
|
|
||||||
[b],
|
|
||||||
"red"
|
|
||||||
]
|
|
||||||
]),
|
]),
|
||||||
isect_faces = !check_isects? [] : unique([
|
isect_faces = !check_isects? [] : unique([
|
||||||
for (i = [0:1:len(faces)-2])
|
for (i = [0:1:len(faces)-2])
|
||||||
|
@ -791,26 +774,15 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
)
|
)
|
||||||
if (!is_undef(isects2))
|
if (!is_undef(isects2))
|
||||||
for (seg=isects2)
|
for (seg=isects2)
|
||||||
if (seg[0] != seg[1]) [
|
if (seg[0] != seg[1])
|
||||||
"ERROR",
|
_vnf_validate_err("FACE_ISECT", seg)
|
||||||
"FACE_ISECT",
|
|
||||||
"Faces intersect",
|
|
||||||
seg,
|
|
||||||
"blue"
|
|
||||||
]
|
|
||||||
]),
|
]),
|
||||||
hole_edges = unique([
|
hole_edges = unique([
|
||||||
for (i=idx(uniq_edges))
|
for (i=idx(uniq_edges))
|
||||||
if (edgecnts[1][i]<2)
|
if (edgecnts[1][i]<2)
|
||||||
if (_pts_not_reported(uniq_edges[i], varr, t_juncts))
|
if (_pts_not_reported(uniq_edges[i], varr, t_juncts))
|
||||||
if (_pts_not_reported(uniq_edges[i], varr, isect_faces))
|
if (_pts_not_reported(uniq_edges[i], varr, isect_faces))
|
||||||
[
|
_vnf_validate_err("HOLE_EDGE", [for (i=uniq_edges[i]) varr[i]])
|
||||||
"ERROR",
|
|
||||||
"HOLE_EDGE",
|
|
||||||
"Edge bounds Hole",
|
|
||||||
[for (i=uniq_edges[i]) varr[i]],
|
|
||||||
"magenta"
|
|
||||||
]
|
|
||||||
])
|
])
|
||||||
) concat(
|
) concat(
|
||||||
big_faces,
|
big_faces,
|
||||||
|
@ -824,6 +796,24 @@ function vnf_validate(vnf, show_warns=true, check_isects=false) =
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
|
_vnf_validate_errs = [
|
||||||
|
["BIG_FACE", "WARNING", "cyan", "Face has more than 3 vertices, and may confuse CGAL"],
|
||||||
|
["NULL_FACE", "WARNING", "blue", "Face has zero area."],
|
||||||
|
["NONPLANAR", "ERROR", "yellow", "Face vertices are not coplanar"],
|
||||||
|
["OVRPOP_EDGE", "ERROR", "orange", "Too many faces attached at Edge"],
|
||||||
|
["REVERSAL", "ERROR", "violet", "Faces Reverse Across Edge"],
|
||||||
|
["T_JUNCTION", "ERROR", "magenta", "Vertex is mid-edge on another Face"],
|
||||||
|
["FACE_ISECT", "ERROR", "brown", "Faces intersect"],
|
||||||
|
["HOLE_EDGE", "ERROR", "red", "Edge bounds Hole"]
|
||||||
|
];
|
||||||
|
|
||||||
|
|
||||||
|
function _vnf_validate_err(name, extra) =
|
||||||
|
let(
|
||||||
|
info = [for (x = _vnf_validate_errs) if (x[0] == name) x][0]
|
||||||
|
) concat(info, [extra]);
|
||||||
|
|
||||||
|
|
||||||
function _pts_not_reported(pts, varr, reports) =
|
function _pts_not_reported(pts, varr, reports) =
|
||||||
[
|
[
|
||||||
for (i = pts, report = reports, pt = report[3])
|
for (i = pts, report = reports, pt = report[3])
|
||||||
|
@ -847,12 +837,12 @@ module vnf_validate(vnf, size=1, show_warns=true, check_isects=false) {
|
||||||
check_isects=check_isects
|
check_isects=check_isects
|
||||||
);
|
);
|
||||||
for (fault = faults) {
|
for (fault = faults) {
|
||||||
typ = fault[0];
|
err = fault[0];
|
||||||
err = fault[1];
|
typ = fault[1];
|
||||||
msg = fault[2];
|
clr = fault[2];
|
||||||
pts = fault[3];
|
msg = fault[3];
|
||||||
clr = fault[4];
|
pts = fault[4];
|
||||||
echo(str(typ, " ", err, ": ", msg, " at ", pts));
|
echo(str(typ, " ", err, " (", clr ,"): ", msg, " at ", pts));
|
||||||
color(clr) {
|
color(clr) {
|
||||||
if (len(pts)==2) {
|
if (len(pts)==2) {
|
||||||
stroke(pts, width=size);
|
stroke(pts, width=size);
|
||||||
|
|
Loading…
Reference in a new issue