From 686b79d568bb81185b013d632bd23ea39df815a5 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 09:18:22 -0400 Subject: [PATCH 1/7] Added is_range and one_defined --- common.scad | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/common.scad b/common.scad index 3467d1a..d94208f 100644 --- a/common.scad +++ b/common.scad @@ -81,6 +81,11 @@ function is_integer(n) = is_num(n) && n == round(n); function is_nan(x) = (x!=x); +// Function: is_range() +// Description: +// Returns true if its argument is a range +function is_range(x) = is_num(x[0]) && !is_list(x); + // Function: is_list_of() // Usage: // is_list_of(list, pattern) @@ -157,6 +162,25 @@ function first_defined(v,recursive=false,_i=0) = )? first_defined(v,recursive=recursive,_i=_i+1) : v[_i]; +// Function: one_defined() +// Usage: +// one_defined(vars, names, [required]) +// Description: +// Examines the input list `vars` and returns the entry which is not `undef`. If more +// than one entry is `undef` then issues an assertion specifying "Must define exactly one of" followed +// by the defined items from the `names` parameter. If `required` is set to false then it is OK if all of the +// entries of `vars` are undefined, and in this case, `undef` is returned. +// Example: +// length = one_defined([length,L,l], ["length","L","l"]); +function one_defined(vars, names, required=true) = + assert(len(vars)==len(names)) + let ( + ok = num_defined(vars)==1 || (!required && num_defined(vars)==0) + ) + assert(ok,str("Must define ",required?"exactly":"at most"," one of ",[for(i=[0:len(vars)]) if (is_def(vars[i])) names[i]])) + first_defined(vars); + + // Function: num_defined() // Description: Counts how many items in list `v` are not `undef`. function num_defined(v,_i=0,_cnt=0) = _i>=len(v)? _cnt : num_defined(v,_i+1,_cnt+(is_undef(v[_i])? 0 : 1)); @@ -180,6 +204,8 @@ function any_defined(v,recursive=false) = first_defined(v,recursive=recursive) ! function all_defined(v,recursive=false) = max([for (x=v) is_undef(x)||(recursive&&is_list(x)&&!all_defined(x))? 1 : 0])==0; + + // Section: Argument Helpers From 20c6e0fe72ddb031c2c74a1c036bde73846c5b56 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 09:18:39 -0400 Subject: [PATCH 2/7] Fixed bug: vquant no longer exists! --- polyhedra.scad | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/polyhedra.scad b/polyhedra.scad index 9e10b50..f64dab2 100644 --- a/polyhedra.scad +++ b/polyhedra.scad @@ -766,7 +766,7 @@ function _facenormal(pts, face) = unit(cross(pts[face[2]]-pts[face[0]], pts[face function _full_faces(pts,faces) = let( - normals = [for(face=faces) vquant(_facenormal(pts,face),1e-12)], + normals = [for(face=faces) quant(_facenormal(pts,face),1e-12)], groups = _unique_groups(normals), faces = [for(entry=groups) unique(flatten(select(faces, entry)))], facenormals = [for(entry=groups) normals[entry[0]]] From 79358ddf6f30f45629f64c6d52e2f08b4ee15ce0 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 09:19:02 -0400 Subject: [PATCH 3/7] Added clarifying docs to vnf_centroid and vnf_volume. Changed vsqr to sqr --- vnf.scad | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/vnf.scad b/vnf.scad index e1ea665..b0471c4 100644 --- a/vnf.scad +++ b/vnf.scad @@ -340,7 +340,9 @@ module vnf_polyhedron(vnf, convexity=2) { // Usage: // vol = vnf_volume(vnf); // Description: -// Returns the volume enclosed by the given manifold VNF. May return a negative value if faces are reversed. +// Returns the volume enclosed by the given manifold VNF. The VNF must describe a valid polyhedron with consistent face direction and +// no holes; otherwise the results are undefined. Returns a positive volume if face direction is clockwise and a negative volume +// if face direction is counter-clockwise. function vnf_volume(vnf) = let(vnf = vnf_triangulate(vnf)) sum([ @@ -355,7 +357,8 @@ function vnf_volume(vnf) = // Usage: // vol = vnf_centroid(vnf); // Description: -// Returns the centroid of the given manifold VNF. +// Returns the centroid of the given manifold VNF. The VNF must describe a valid polyhedron with consistent face direction and +// no holes; otherwise the results are undefined. function vnf_centroid(vnf) = let( vnf = vnf_triangulate(vnf), @@ -367,9 +370,9 @@ function vnf_centroid(vnf) = ) [ face[0] * n, vmul(n, - vsqr(face[0] + face[1]) + - vsqr(face[0] + face[2]) + - vsqr(face[1] + face[2]) + sqr(face[0] + face[1]) + + sqr(face[0] + face[2]) + + sqr(face[1] + face[2]) ) ] ]) From 72aae0e42c10d5b30bd5e317cd9942370b279f22 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 09:19:49 -0400 Subject: [PATCH 4/7] Expanded sqr() to take lists and removed vsqr() --- math.scad | 11 +++++++---- vectors.scad | 7 ------- 2 files changed, 7 insertions(+), 11 deletions(-) diff --git a/math.scad b/math.scad index 4dfe845..92af87c 100644 --- a/math.scad +++ b/math.scad @@ -26,11 +26,14 @@ NAN = acos(2); // The value `nan`, useful for comparisons. // Usage: // sqr(x); // Description: -// Returns the square of the given number. +// Returns the square of the given number or entries in list // Examples: -// sqr(3); // Returns: 9 -// sqr(-4); // Returns: 16 -function sqr(x) = x*x; +// sqr(3); // Returns: 9 +// sqr(-4); // Returns: 16 +// sqr([3,4]); // Returns: [9,16] +// sqr([[1,2],[3,4]]); // Returns [[1,4],[9,16]] +// sqr([[1,2],3]); // Returns [[1,4],9] +function sqr(x) = is_list(x) ? [for(val=x) sqr(val)] : x*x; // Function: log2() diff --git a/vectors.scad b/vectors.scad index 9509b6c..ce82094 100644 --- a/vectors.scad +++ b/vectors.scad @@ -95,13 +95,6 @@ function vdiv(v1, v2) = [for (i = [0:1:len(v1)-1]) v1[i]/v2[i]]; function vabs(v) = [for (x=v) abs(x)]; -// Function: vsqr() -// Usage: -// x = vsqr(v); -// Description: -// Returns a vector where each value in the original given vector is squared. -function vsqr(v) = [for(i=v) i*i]; - // Function: unit() // Description: From e67ba766c89936d561763df2fa1efe5a88f409f8 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 10:58:57 -0400 Subject: [PATCH 5/7] added reference --- vnf.scad | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/vnf.scad b/vnf.scad index b0471c4..d8e73bf 100644 --- a/vnf.scad +++ b/vnf.scad @@ -358,7 +358,9 @@ function vnf_volume(vnf) = // vol = vnf_centroid(vnf); // Description: // Returns the centroid of the given manifold VNF. The VNF must describe a valid polyhedron with consistent face direction and -// no holes; otherwise the results are undefined. +// no holes; otherwise the results are undefined. + +// Algorithm from: https://wwwf.imperial.ac.uk/~rn/centroid.pdf function vnf_centroid(vnf) = let( vnf = vnf_triangulate(vnf), From 5994e47cf9bc79c48d3cdcbd440161ba9a65936e Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 18:24:49 -0400 Subject: [PATCH 6/7] Added vnf option to regular_polyhedron_info and removed translate_points. --- coords.scad | 34 ++-------------------------------- polyhedra.scad | 24 ++++++++++++++++-------- 2 files changed, 18 insertions(+), 40 deletions(-) diff --git a/coords.scad b/coords.scad index 2d3edd0..5ef1366 100644 --- a/coords.scad +++ b/coords.scad @@ -100,36 +100,6 @@ function path4d(points, fill=0) = result + repeat(addition, len(result)); -// Function: translate_points() -// Usage: -// translate_points(pts, v); -// Description: -// Moves each point in an array by a given amount. -// Arguments: -// pts = List of points to translate. -// v = Amount to translate points by. -function translate_points(pts, v=[0,0,0]) = - pts==[]? [] : let( - v=point3d(v) - ) [for (pt = pts) pt+v]; - - -// Function: scale_points() -// Usage: -// scale_points(pts, v, [cp]); -// Description: -// Scales each point in an array by a given amount, around a given centerpoint. -// Arguments: -// pts = List of points to scale. -// v = A vector with a scaling factor for each axis. -// cp = Centerpoint to scale around. -function scale_points(pts, v=[1,1,1], cp=[0,0,0]) = - pts==[]? [] : let( - cp = point3d(cp), - v = point3d(v,fill=1) - ) [for (pt = pts) vmul(pt-cp,v)+cp]; - - // Function: rotate_points2d() // Usage: // rotate_points2d(pts, a, [cp]); @@ -289,7 +259,7 @@ function project_plane(point, a, b, c) = v = unit(c-a), n = unit(cross(u,v)), w = unit(cross(n,u)), - relpoint = is_vector(point)? (point-a) : translate_points(point,-a) + relpoint = apply(move(-a),point) ) relpoint * transpose([w,u]); @@ -320,7 +290,7 @@ function lift_plane(point, a, b, c) = n = unit(cross(u,v)), w = unit(cross(n,u)), remapped = point*[w,u] - ) is_vector(remapped)? (a+remapped) : translate_points(remapped,a); + ) apply(move(a),remapped); // Function: cylindrical_to_xyz() diff --git a/polyhedra.scad b/polyhedra.scad index f64dab2..a25858b 100644 --- a/polyhedra.scad +++ b/polyhedra.scad @@ -9,7 +9,7 @@ ////////////////////////////////////////////////////////////////////// -include +include // CommonCode: @@ -320,14 +320,14 @@ module regular_polyhedron( in_radius = entry[5]; if (draw){ if (rounding==0) - polyhedron(translate_points(scaled_points, translation), faces = face_triangles); + polyhedron(move(translation, p=scaled_points), faces = face_triangles); else { fn = segs(rounding); rounding = rounding/cos(180/fn); adjusted_scale = 1 - rounding / in_radius; minkowski(){ sphere(r=rounding, $fn=fn); - polyhedron(translate_points(adjusted_scale*scaled_points,translation), faces = face_triangles); + polyhedron(move(translation,p=adjusted_scale*scaled_points), faces = face_triangles); } } } @@ -335,13 +335,13 @@ module regular_polyhedron( maxrange = repeat ? len(faces)-1 : $children-1; for(i=[0:1:maxrange]) { // Would like to orient so an edge (longest edge?) is parallel to x axis - facepts = translate_points(select(scaled_points, faces[i]), translation); + facepts = move(translation, p=select(scaled_points, faces[i])); center = mean(facepts); - rotatedface = rotate_points3d(translate_points(facepts,-center), from=face_normals[i], to=[0,0,1]); + rotatedface = rotate_points3d(move(-center, p=facepts), from=face_normals[i], to=[0,0,1]); clockwise = sortidx([for(pt=rotatedface) -atan2(pt.y,pt.x)]); $face = rotate_children? path2d(select(rotatedface,clockwise)) : - select(translate_points(facepts,-center), clockwise); + select(move(-center,p=facepts), clockwise); $faceindex = i; $center = -translation-center; translate(center) @@ -549,6 +549,7 @@ _stellated_polyhedra_ = [ // Calculate characteristics of regular polyhedra or the selection set for regular_polyhedron(). // Invoke with the same arguments used by regular_polyhedron() and use the `info` argument to // request the desired return value. Set `info` to: +// * `"vnf"`: vnf for the selected polyhedron // * `"vertices"`: vertex list for the selected polyhedron // * `"faces"`: list of faces for the selected polyhedron, where each entry on the list is a list of point index values to be used with the vertex list // * `"face normals"`: list of normal vectors for each face @@ -688,8 +689,15 @@ function regular_polyhedron_info( face_normals = rotate_points3d(faces_normals_vertices[1], from=down_direction, to=[0,0,-1]), side_length = scalefactor * entry[edgelen] ) - info == "fullentry" ? [scaled_points, translation,stellate ? faces : face_triangles, faces, face_normals, side_length*entry[in_radius]] : - info == "vertices" ? translate_points(scaled_points,translation) : + info == "fullentry" ? [scaled_points, + translation, + stellate ? faces : face_triangles, + faces, + face_normals, + side_length*entry[in_radius]] : + info == "vnf" ? [move(translation,p=scaled_points), + stellate ? faces : face_triangles] : + info == "vertices" ? move(translation,p=scaled_points) : info == "faces" ? faces : info == "face normals" ? face_normals : info == "in_radius" ? side_length * entry[in_radius] : From 187311c48405108f43846d26c6e802bd04c001f2 Mon Sep 17 00:00:00 2001 From: Adrian Mariano Date: Sat, 21 Mar 2020 18:25:31 -0400 Subject: [PATCH 7/7] removed translate_points --- threading.scad | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/threading.scad b/threading.scad index 1d5a1bf..bf3c7e8 100644 --- a/threading.scad +++ b/threading.scad @@ -60,7 +60,7 @@ module thread_helix(base_d, pitch, thread_depth=undef, thread_angle=15, twist=72 [0, -cap/2-dz], ] ); - pline = scale_points(profile, [1,1,1]*pitch); + pline = scale(pitch, p=profile); dir = left_handed? -1 : 1; idir = internal? -1 : 1; attachable(anchor,spin,orient, r=r, l=h) {