distributors name normalization & bug fix

This commit is contained in:
Adrian Mariano 2022-10-22 23:15:35 -04:00
parent 01c07dc892
commit a7711edf0b
5 changed files with 295 additions and 249 deletions

View file

@ -835,7 +835,7 @@ module tag_scope(scope){
// right(20) // right(20)
// circle(5); // circle(5);
// } // }
// Example: Here is another example where two children are intersected using the native intersection operator, and then tagged with {{force_tag()}}. Note that because the children are at the save level, you don't need to use a tagged operator for their intersection. // Example: Here is another example where two children are intersected using the native intersection operator, and then tagged with {{force_tag()}}. Note that because the children are at the same level, you don't need to use a tagged operator for their intersection.
// $fn=32; // $fn=32;
// diff() // diff()
// cuboid(10){ // cuboid(10){

View file

@ -46,113 +46,6 @@ module move_copies(a=[[0,0,0]])
} }
// Function&Module: line_of()
//
// Usage: Place `n` copies at a given spacing along the line
// line_of(spacing, [n], [p1=]) CHILDREN;
// Usage: Place as many copies as will fit at a given spacing
// line_of(spacing, [l=], [p1=]) CHILDREN;
// Usage: Place `n` copies along the length of the line
// line_of([n=], [l=], [p1=]) CHILDREN;
// Usage: Place `n` copies along the line from `p1` to `p2`
// line_of([n=], [p1=], [p2=]) CHILDREN;
// Usage: Place copies at the given spacing, centered along the line from `p1` to `p2`
// line_of([spacing], [p1=], [p2=]) CHILDREN;
// Usage: As a function
// pts = line_of([spacing], [n], [p1=]);
// pts = line_of([spacing], [l=], [p1=]);
// pts = line_of([n=], [l=], [p1=]);
// pts = line_of([n=], [p1=], [p2=]);
// pts = line_of([spacing], [p1=], [p2=]);
// Description:
// When called as a function, returns a list of points at evenly spaced positions along a line.
// When called as a module, copies `children()` at one or more evenly spaced positions along a line.
// By default, the line will be centered at the origin, unless the starting point `p1` is given.
// The line will be pointed towards `RIGHT` (X+) unless otherwise given as a vector in `l`,
// `spacing`, or `p1`/`p2`. The psotion of the copies is specified in one of several ways:
// .
// If You Know... | Then Use Something Like...
// -------------------------------- | --------------------------------
// Spacing distance, Count | `line_of(spacing=10, n=5) ...` or `line_of(10, n=5) ...`
// Spacing vector, Count | `line_of(spacing=[10,5], n=5) ...` or `line_of([10,5], n=5) ...`
// Spacing distance, Line length | `line_of(spacing=10, l=50) ...` or `line_of(10, l=50) ...`
// Spacing distance, Line vector | `line_of(spacing=10, l=[50,30]) ...` or `line_of(10, l=[50,30]) ...`
// Spacing vector, Line length | `line_of(spacing=[10,5], l=50) ...` or `line_of([10,5], l=50) ...`
// Line length, Count | `line_of(l=50, n=5) ...`
// Line vector, Count | `line_of(l=[50,40], n=5) ...`
// Line endpoints, Count | `line_of(p1=[10,10], p2=[60,-10], n=5) ...`
// Line endpoints, Spacing distance | `line_of(p1=[10,10], p2=[60,-10], spacing=10) ...`
//
// Arguments:
// spacing = Either the scalar spacing distance along the X+ direction, or the vector giving both the direction and spacing distance between each set of copies.
// n = Number of copies to distribute along the line. (Default: 2)
// ---
// l = Either the scalar length of the line, or a vector giving both the direction and length of the line.
// p1 = If given, specifies the starting point of the line.
// p2 = If given with `p1`, specifies the ending point of line, and indirectly calculates the line length.
//
// Side Effects:
// `$pos` is set to the relative centerpoint of each child copy, and can be used to modify each child individually.
// `$idx` is set to the index number of each child being copied.
//
// Examples:
// line_of(10) sphere(d=1);
// line_of(10, n=5) sphere(d=1);
// line_of([10,5], n=5) sphere(d=1);
// line_of(spacing=10, n=6) sphere(d=1);
// line_of(spacing=[10,5], n=6) sphere(d=1);
// line_of(spacing=10, l=50) sphere(d=1);
// line_of(spacing=10, l=[50,30]) sphere(d=1);
// line_of(spacing=[10,5], l=50) sphere(d=1);
// line_of(l=50, n=4) sphere(d=1);
// line_of(l=[50,-30], n=4) sphere(d=1);
// Example(FlatSpin,VPD=133):
// line_of(p1=[0,0,0], p2=[5,5,20], n=6) cube(size=[3,2,1],center=true);
// Example(FlatSpin,VPD=133):
// line_of(p1=[0,0,0], p2=[5,5,20], spacing=6) cube(size=[3,2,1],center=true);
// Example: All children are copied to each position
// line_of(l=20, n=3) {
// cube(size=[1,3,1],center=true);
// cube(size=[3,1,1],center=true);
// }
// Example(2D): The functional form of line_of() returns a list of points.
// pts = line_of([10,5],n=5);
// move_copies(pts) circle(d=2);
module line_of(spacing, n, l, p1, p2)
{
req_children($children);
pts = line_of(spacing=spacing, n=n, l=l, p1=p1, p2=p2);
for (i=idx(pts)) {
$idx = i;
$pos = pts[i];
translate($pos) children();
}
}
function line_of(spacing, n, l, p1, p2) =
assert(is_undef(spacing) || is_finite(spacing) || is_vector(spacing))
assert(is_undef(n) || is_finite(n))
assert(is_undef(l) || is_finite(l) || is_vector(l))
assert(is_undef(p1) || is_vector(p1))
assert(is_undef(p2) || is_vector(p2))
let(
ll = !is_undef(l)? scalar_vec3(l, 0) :
(!is_undef(spacing) && !is_undef(n))? ((n-1) * scalar_vec3(spacing, 0)) :
(!is_undef(p1) && !is_undef(p2))? point3d(p2-p1) :
undef,
cnt = !is_undef(n)? n :
(!is_undef(spacing) && !is_undef(ll))? floor(norm(ll) / norm(scalar_vec3(spacing, 0)) + 1.000001) :
2,
spc = cnt<=1? [0,0,0] :
is_undef(spacing)? (ll/(cnt-1)) :
is_num(spacing) && !is_undef(ll)? (ll/(cnt-1)) :
scalar_vec3(spacing, 0)
)
assert(!is_undef(cnt), "Need two of `spacing`, 'l', 'n', or `p1`/`p2` arguments in `line_of()`.")
let( spos = !is_undef(p1)? point3d(p1) : -(cnt-1)/2 * spc )
[for (i=[0:1:cnt-1]) i * spc + spos];
// Module: xcopies() // Module: xcopies()
// //
// Description: // Description:
@ -192,12 +85,14 @@ module xcopies(spacing, n, l, sp)
sp = is_finite(sp)? (sp*dir) : sp; sp = is_finite(sp)? (sp*dir) : sp;
if (is_vector(spacing)) { if (is_vector(spacing)) {
translate(default(sp,[0,0,0])) { translate(default(sp,[0,0,0])) {
for (x = spacing) { for (i = idx(spacing)) {
translate(x*dir) children(); $idx = i;
$pos = spacing[i]*dir;
translate($pos) children();
} }
} }
} else { } else {
line_of( line_copies(
l=u_mul(l,dir), l=u_mul(l,dir),
spacing=u_mul(spacing,dir), spacing=u_mul(spacing,dir),
n=n, p1=sp n=n, p1=sp
@ -245,12 +140,14 @@ module ycopies(spacing, n, l, sp)
sp = is_finite(sp)? (sp*dir) : sp; sp = is_finite(sp)? (sp*dir) : sp;
if (is_vector(spacing)) { if (is_vector(spacing)) {
translate(default(sp,[0,0,0])) { translate(default(sp,[0,0,0])) {
for (x = spacing) { for (i = idx(spacing)) {
translate(x*dir) children(); $idx = i;
$pos = spacing[i]*dir;
translate($pos) children();
} }
} }
} else { } else {
line_of( line_copies(
l=u_mul(l,dir), l=u_mul(l,dir),
spacing=u_mul(spacing,dir), spacing=u_mul(spacing,dir),
n=n, p1=sp n=n, p1=sp
@ -312,12 +209,14 @@ module zcopies(spacing, n, l, sp)
sp = is_finite(sp)? (sp*dir) : sp; sp = is_finite(sp)? (sp*dir) : sp;
if (is_vector(spacing)) { if (is_vector(spacing)) {
translate(default(sp,[0,0,0])) { translate(default(sp,[0,0,0])) {
for (x = spacing) { for (i = idx(spacing)) {
translate(x*dir) children(); $idx = i;
$pos = spacing[i]*dir;
translate($pos) children();
} }
} }
} else { } else {
line_of( line_copies(
l=u_mul(l,dir), l=u_mul(l,dir),
spacing=u_mul(spacing,dir), spacing=u_mul(spacing,dir),
n=n, p1=sp n=n, p1=sp
@ -327,6 +226,121 @@ module zcopies(spacing, n, l, sp)
// Function&Module: line_copies()
//
// Usage: Place `n` copies at a given spacing along the line
// line_copies(spacing, [n], [p1=]) CHILDREN;
// Usage: Place as many copies as will fit at a given spacing
// line_copies(spacing, [l=], [p1=]) CHILDREN;
// Usage: Place `n` copies along the length of the line
// line_copies([n=], [l=], [p1=]) CHILDREN;
// Usage: Place `n` copies along the line from `p1` to `p2`
// line_copies([n=], [p1=], [p2=]) CHILDREN;
// Usage: Place copies at the given spacing, centered along the line from `p1` to `p2`
// line_copies([spacing], [p1=], [p2=]) CHILDREN;
// Usage: As a function
// pts = line_copies([spacing], [n], [p1=]);
// pts = line_copies([spacing], [l=], [p1=]);
// pts = line_copies([n=], [l=], [p1=]);
// pts = line_copies([n=], [p1=], [p2=]);
// pts = line_copies([spacing], [p1=], [p2=]);
// Description:
// When called as a function, returns a list of points at evenly spaced positions along a line.
// When called as a module, copies `children()` at one or more evenly spaced positions along a line.
// By default, the line will be centered at the origin, unless the starting point `p1` is given.
// The line will be pointed towards `RIGHT` (X+) unless otherwise given as a vector in `l`,
// `spacing`, or `p1`/`p2`. The psotion of the copies is specified in one of several ways:
// .
// If You Know... | Then Use Something Like...
// -------------------------------- | --------------------------------
// Spacing distance, Count | `line_copies(spacing=10, n=5) ...` or `line_copies(10, n=5) ...`
// Spacing vector, Count | `line_copies(spacing=[10,5], n=5) ...` or `line_copies([10,5], n=5) ...`
// Spacing distance, Line length | `line_copies(spacing=10, l=50) ...` or `line_copies(10, l=50) ...`
// Spacing distance, Line vector | `line_copies(spacing=10, l=[50,30]) ...` or `line_copies(10, l=[50,30]) ...`
// Spacing vector, Line length | `line_copies(spacing=[10,5], l=50) ...` or `line_copies([10,5], l=50) ...`
// Line length, Count | `line_copies(l=50, n=5) ...`
// Line vector, Count | `line_copies(l=[50,40], n=5) ...`
// Line endpoints, Count | `line_copies(p1=[10,10], p2=[60,-10], n=5) ...`
// Line endpoints, Spacing distance | `line_copies(p1=[10,10], p2=[60,-10], spacing=10) ...`
//
// Arguments:
// spacing = Either the scalar spacing distance along the X+ direction, or the vector giving both the direction and spacing distance between each set of copies.
// n = Number of copies to distribute along the line. (Default: 2)
// ---
// l = Either the scalar length of the line, or a vector giving both the direction and length of the line.
// p1 = If given, specifies the starting point of the line.
// p2 = If given with `p1`, specifies the ending point of line, and indirectly calculates the line length.
//
// Side Effects:
// `$pos` is set to the relative centerpoint of each child copy, and can be used to modify each child individually.
// `$idx` is set to the index number of each child being copied.
//
// Examples:
// line_copies(10) sphere(d=1);
// line_copies(10, n=5) sphere(d=1);
// line_copies([10,5], n=5) sphere(d=1);
// line_copies(spacing=10, n=6) sphere(d=1);
// line_copies(spacing=[10,5], n=6) sphere(d=1);
// line_copies(spacing=10, l=50) sphere(d=1);
// line_copies(spacing=10, l=[50,30]) sphere(d=1);
// line_copies(spacing=[10,5], l=50) sphere(d=1);
// line_copies(l=50, n=4) sphere(d=1);
// line_copies(l=[50,-30], n=4) sphere(d=1);
// Example(FlatSpin,VPD=133):
// line_copies(p1=[0,0,0], p2=[5,5,20], n=6) cube(size=[3,2,1],center=true);
// Example(FlatSpin,VPD=133):
// line_copies(p1=[0,0,0], p2=[5,5,20], spacing=6) cube(size=[3,2,1],center=true);
// Example: All children are copied to each position
// line_copies(l=20, n=3) {
// cube(size=[1,3,1],center=true);
// cube(size=[3,1,1],center=true);
// }
// Example(2D): The functional form of line_copies() returns a list of points.
// pts = line_copies([10,5],n=5);
// move_copies(pts) circle(d=2);
module line_of(spacing, n, l, p1, p2) {
deprecate("line_copies");
line_copies(spacing, n, l, p1, p2);
}
module line_copies(spacing, n, l, p1, p2)
{
req_children($children);
pts = line_copies(spacing=spacing, n=n, l=l, p1=p1, p2=p2);
for (i=idx(pts)) {
$idx = i;
$pos = pts[i];
translate($pos) children();
}
}
function line_copies(spacing, n, l, p1, p2) =
assert(is_undef(spacing) || is_finite(spacing) || is_vector(spacing))
assert(is_undef(n) || is_finite(n))
assert(is_undef(l) || is_finite(l) || is_vector(l))
assert(is_undef(p1) || is_vector(p1))
assert(is_undef(p2) || is_vector(p2))
let(
ll = !is_undef(l)? scalar_vec3(l, 0) :
(!is_undef(spacing) && !is_undef(n))? ((n-1) * scalar_vec3(spacing, 0)) :
(!is_undef(p1) && !is_undef(p2))? point3d(p2-p1) :
undef,
cnt = !is_undef(n)? n :
(!is_undef(spacing) && !is_undef(ll))? floor(norm(ll) / norm(scalar_vec3(spacing, 0)) + 1.000001) :
2,
spc = cnt<=1? [0,0,0] :
is_undef(spacing)? (ll/(cnt-1)) :
is_num(spacing) && !is_undef(ll)? (ll/(cnt-1)) :
scalar_vec3(spacing, 0)
)
assert(!is_undef(cnt), "Need two of `spacing`, 'l', 'n', or `p1`/`p2` arguments in `line_copies()`.")
let( spos = !is_undef(p1)? point3d(p1) : -(cnt-1)/2 * spc )
[for (i=[0:1:cnt-1]) i * spc + spos];
// Module: grid_copies() // Module: grid_copies()
@ -743,14 +757,14 @@ module zrot_copies(rots=[], cp=[0,0,0], n, sa=0, r, d, subrot=true)
} }
// Module: arc_of() // Module: arc_copies()
// //
// Description: // Description:
// Evenly distributes n duplicate children around an ovoid arc on the XY plane. // Evenly distributes n duplicate children around an ovoid arc on the XY plane.
// //
// Usage: // Usage:
// arc_of(n, r|d=, [sa=], [ea=], [rot=]) CHILDREN; // arc_copies(n, r|d=, [sa=], [ea=], [rot=]) CHILDREN;
// arc_of(n, rx=|dx=, ry=|dy=, [sa=], [ea=], [rot=]) CHILDREN; // arc_copies(n, rx=|dx=, ry=|dy=, [sa=], [ea=], [rot=]) CHILDREN;
// //
// Arguments: // Arguments:
// n = number of copies to distribute around the circle. (Default: 6) // n = number of copies to distribute around the circle. (Default: 6)
@ -772,24 +786,30 @@ module zrot_copies(rots=[], cp=[0,0,0], n, sa=0, r, d, subrot=true)
// //
// Example: // Example:
// #cube(size=[10,3,3],center=true); // #cube(size=[10,3,3],center=true);
// arc_of(d=40, n=5) cube(size=[10,3,3],center=true); // arc_copies(d=40, n=5) cube(size=[10,3,3],center=true);
// //
// Example: // Example:
// #cube(size=[10,3,3],center=true); // #cube(size=[10,3,3],center=true);
// arc_of(d=40, n=5, sa=45, ea=225) cube(size=[10,3,3],center=true); // arc_copies(d=40, n=5, sa=45, ea=225) cube(size=[10,3,3],center=true);
// //
// Example: // Example:
// #cube(size=[10,3,3],center=true); // #cube(size=[10,3,3],center=true);
// arc_of(r=15, n=8, rot=false) cube(size=[10,3,3],center=true); // arc_copies(r=15, n=8, rot=false) cube(size=[10,3,3],center=true);
// //
// Example: // Example:
// #cube(size=[10,3,3],center=true); // #cube(size=[10,3,3],center=true);
// arc_of(rx=20, ry=10, n=8) cube(size=[10,3,3],center=true); // arc_copies(rx=20, ry=10, n=8) cube(size=[10,3,3],center=true);
// Example(2D): Using `$idx` to alternate shapes // Example(2D): Using `$idx` to alternate shapes
// arc_of(r=50, n=19, sa=0, ea=180) // arc_copies(r=50, n=19, sa=0, ea=180)
// if ($idx % 2 == 0) rect(6); // if ($idx % 2 == 0) rect(6);
// else circle(d=6); // else circle(d=6);
module arc_of(
function arc_copies(n=6,r,rx,ry,d,dx,dy,sa=0,ea=360,rot=true) = no_function("arc_copies");
module arc_of(n=6,r,rx,ry,d,dx,dy,sa=0,ea=360,rot=true){
deprecate("arc_copies");
arc_copies(n,r,rx,ry,d,dx,dy,sa,ea,rot);
}
module arc_copies(
n=6, n=6,
r=undef, r=undef,
rx=undef, ry=undef, rx=undef, ry=undef,
@ -1046,61 +1066,6 @@ module path_copies(path, n, spacing, sp=undef, dist, rotate_children=true, dist,
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
// Module: mirror_copy()
//
// Description:
// Makes a copy of the children, mirrored across the given plane.
//
// Usage:
// mirror_copy(v, [cp], [offset]) CHILDREN;
//
// Arguments:
// v = The normal vector of the plane to mirror across.
// offset = distance to offset away from the plane.
// cp = A point that lies on the mirroring plane.
//
// Side Effects:
// `$orig` is true for the original instance of children. False for the copy.
// `$idx` is set to the index value of each copy.
//
// Example:
// mirror_copy([1,-1,0]) zrot(-45) yrot(90) cylinder(d1=10, d2=0, h=20);
// color("blue",0.25) zrot(-45) cube([0.01,15,15], center=true);
//
// Example:
// mirror_copy([1,1,0], offset=5) rot(a=90,v=[-1,1,0]) cylinder(d1=10, d2=0, h=20);
// color("blue",0.25) zrot(45) cube([0.01,15,15], center=true);
//
// Example:
// mirror_copy(UP+BACK, cp=[0,-5,-5]) rot(from=UP, to=BACK+UP) cylinder(d1=10, d2=0, h=20);
// color("blue",0.25) translate([0,-5,-5]) rot(from=UP, to=BACK+UP) cube([15,15,0.01], center=true);
module mirror_copy(v=[0,0,1], offset=0, cp)
{
req_children($children);
cp = is_vector(v,4)? plane_normal(v) * v[3] :
is_vector(cp)? cp :
is_num(cp)? cp*unit(v) :
[0,0,0];
nv = is_vector(v,4)? plane_normal(v) : unit(v);
off = nv*offset;
if (cp == [0,0,0]) {
translate(off) {
$orig = true;
$idx = 0;
children();
}
mirror(nv) translate(off) {
$orig = false;
$idx = 1;
children();
}
} else {
translate(off) children();
translate(cp) mirror(nv) translate(-cp) translate(off) children();
}
}
// Module: xflip_copy() // Module: xflip_copy()
// //
// Description: // Description:
@ -1202,56 +1167,67 @@ module zflip_copy(offset=0, z=0)
mirror_copy(v=[0,0,1], offset=offset, cp=[0,0,z]) children(); mirror_copy(v=[0,0,1], offset=offset, cp=[0,0,z]) children();
} }
////////////////////
// Section: Distributing children individually along a line
///////////////////
// Module: distribute() // Module: mirror_copy()
// //
// Description: // Description:
// Spreads out the children individually along the direction `dir`. // Makes a copy of the children, mirrored across the given plane.
// Every child is placed at a different position, in order.
// This is useful for laying out groups of disparate objects
// where you only really care about the spacing between them.
// //
// Usage: // Usage:
// distribute(spacing, sizes, dir) CHILDREN; // mirror_copy(v, [cp], [offset]) CHILDREN;
// distribute(l=, [sizes=], [dir=]) CHILDREN;
// //
// Arguments: // Arguments:
// spacing = Spacing to add between each child. (Default: 10.0) // v = The normal vector of the plane to mirror across.
// sizes = Array containing how much space each child will need. // offset = distance to offset away from the plane.
// dir = Vector direction to distribute copies along. Default: RIGHT // cp = A point that lies on the mirroring plane.
// l = Length to distribute copies along.
// //
// Side Effects: // Side Effects:
// `$pos` is set to the relative centerpoint of each child copy, and can be used to modify each child individually. // `$orig` is true for the original instance of children. False for the copy.
// `$idx` is set to the index number of each child being copied. // `$idx` is set to the index value of each copy.
// //
// Example: // Example:
// distribute(sizes=[100, 30, 50], dir=UP) { // mirror_copy([1,-1,0]) zrot(-45) yrot(90) cylinder(d1=10, d2=0, h=20);
// sphere(r=50); // color("blue",0.25) zrot(-45) cube([0.01,15,15], center=true);
// cube([10,20,30], center=true); //
// cylinder(d=30, h=50, center=true); // Example:
// } // mirror_copy([1,1,0], offset=5) rot(a=90,v=[-1,1,0]) cylinder(d1=10, d2=0, h=20);
module distribute(spacing=undef, sizes=undef, dir=RIGHT, l=undef) // color("blue",0.25) zrot(45) cube([0.01,15,15], center=true);
//
// Example:
// mirror_copy(UP+BACK, cp=[0,-5,-5]) rot(from=UP, to=BACK+UP) cylinder(d1=10, d2=0, h=20);
// color("blue",0.25) translate([0,-5,-5]) rot(from=UP, to=BACK+UP) cube([15,15,0.01], center=true);
module mirror_copy(v=[0,0,1], offset=0, cp)
{ {
req_children($children); req_children($children);
gaps = ($children < 2)? [0] : cp = is_vector(v,4)? plane_normal(v) * v[3] :
!is_undef(sizes)? [for (i=[0:1:$children-2]) sizes[i]/2 + sizes[i+1]/2] : is_vector(cp)? cp :
[for (i=[0:1:$children-2]) 0]; is_num(cp)? cp*unit(v) :
spc = !is_undef(l)? ((l - sum(gaps)) / ($children-1)) : default(spacing, 10); [0,0,0];
gaps2 = [for (gap = gaps) gap+spc]; nv = is_vector(v,4)? plane_normal(v) : unit(v);
spos = dir * -sum(gaps2)/2; off = nv*offset;
spacings = cumsum([0, each gaps2]); if (cp == [0,0,0]) {
for (i=[0:1:$children-1]) { translate(off) {
$pos = spos + spacings[i] * dir; $orig = true;
$idx = i; $idx = 0;
translate($pos) children(i); children();
}
mirror(nv) translate(off) {
$orig = false;
$idx = 1;
children();
}
} else {
translate(off) children();
translate(cp) mirror(nv) translate(-cp) translate(off) children();
} }
} }
////////////////////
// Section: Distributing children individually along a line
///////////////////
// Module: xdistribute() // Module: xdistribute()
// //
// Description: // Description:
@ -1391,5 +1367,53 @@ module zdistribute(spacing=10, sizes=undef, l=undef)
// Module: distribute()
//
// Description:
// Spreads out the children individually along the direction `dir`.
// Every child is placed at a different position, in order.
// This is useful for laying out groups of disparate objects
// where you only really care about the spacing between them.
//
// Usage:
// distribute(spacing, sizes, dir) CHILDREN;
// distribute(l=, [sizes=], [dir=]) CHILDREN;
//
// Arguments:
// spacing = Spacing to add between each child. (Default: 10.0)
// sizes = Array containing how much space each child will need.
// dir = Vector direction to distribute copies along. Default: RIGHT
// l = Length to distribute copies along.
//
// Side Effects:
// `$pos` is set to the relative centerpoint of each child copy, and can be used to modify each child individually.
// `$idx` is set to the index number of each child being copied.
//
// Example:
// distribute(sizes=[100, 30, 50], dir=UP) {
// sphere(r=50);
// cube([10,20,30], center=true);
// cylinder(d=30, h=50, center=true);
// }
module distribute(spacing=undef, sizes=undef, dir=RIGHT, l=undef)
{
req_children($children);
gaps = ($children < 2)? [0] :
!is_undef(sizes)? [for (i=[0:1:$children-2]) sizes[i]/2 + sizes[i+1]/2] :
[for (i=[0:1:$children-2]) 0];
spc = !is_undef(l)? ((l - sum(gaps)) / ($children-1)) : default(spacing, 10);
gaps2 = [for (gap = gaps) gap+spc];
spos = dir * -sum(gaps2)/2;
spacings = cumsum([0, each gaps2]);
for (i=[0:1:$children-1]) {
$pos = spos + spacings[i] * dir;
$idx = i;
translate($pos) children(i);
}
}
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap // vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap

View file

@ -1595,9 +1595,9 @@ function reuleaux_polygon(n=3, r, d, anchor=CENTER, spin=0) =
// text("Foobar", size=12, font="Helvetica"); // text("Foobar", size=12, font="Helvetica");
// text("Foobar", anchor=CENTER); // text("Foobar", anchor=CENTER);
// text("Foobar", anchor=str("baseline",CENTER)); // text("Foobar", anchor=str("baseline",CENTER));
// Example: Using line_of() distributor // Example: Using line_copies() distributor
// txt = "This is the string."; // txt = "This is the string.";
// line_of(spacing=[10,-5],n=len(txt)) // line_copies(spacing=[10,-5],n=len(txt))
// text(txt[$idx], size=10, anchor=CENTER); // text(txt[$idx], size=10, anchor=CENTER);
// Example: Using arc_of() distributor // Example: Using arc_of() distributor
// txt = "This is the string"; // txt = "This is the string";

View file

@ -403,10 +403,11 @@ cylinder(h=100, d=100, center=true)
## Tagged Operations ## Tagged Operations
BOSL2 introduces the concept of tags. Tags are names that can be given to attachables, so that BOSL2 introduces the concept of tags. Tags are names that can be given to attachables, so that
you can refer to them when performing `diff()`, `intersect()`, and `conv_hull()` operations. you can refer to them when performing `diff()`, `intersect()`, and `conv_hull()` operations.
Each object can have no more than one tag at a time.
### `diff(remove, <keep>)` ### `diff([remove], [keep])`
The `diff()` operator is used to difference away all shapes marked with the tag(s) given to The `diff()` operator is used to difference away all shapes marked with the tag(s) given to
`remove=`, from the other shapes. `remove`, from the other shapes.
For example, to difference away a child cylinder from the middle of a parent cube, you can For example, to difference away a child cylinder from the middle of a parent cube, you can
do this: do this:
@ -445,7 +446,7 @@ tag("keep")cube(100, center=true)
``` ```
If you need to mark multiple children with a tag, you can use the `tag()` module. You can of course apply `tag()` to several children.
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
@ -458,14 +459,28 @@ cube(100, center=true)
} }
``` ```
Many of the modules that use tags have default values for their tags. For diff the default
remove tag is "remove" and the default keep tag is "keep". In this example we rely on the
default values:
```openscad-3D
include <BOSL2/std.scad>
diff()
sphere(d=100) {
tag("keep")xcyl(d=40, l=120);
tag("remove")cuboid([40,120,100]);
}
```
The parent object can be differenced away from other shapes. Tags are inherited by children, The parent object can be differenced away from other shapes. Tags are inherited by children,
though, so you will need to set the tags of the children as well as the parent. though, so you will need to set the tags of the children as well as the parent.
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
diff("hole") diff("hole")
cube([20,11,45], center=true, $tag="hole") tag("hole")cube([20,11,45], center=true)
cube([40,10,90], center=true, $tag="body"); tag("body")cube([40,10,90], center=true);
``` ```
Tags (and therefore tag-based operations like `diff()`) only work correctly with attachable Tags (and therefore tag-based operations like `diff()`) only work correctly with attachable
@ -488,55 +503,62 @@ cuboid(50)
square(10,center=true); square(10,center=true);
``` ```
### `intersect(a, <b>, <keep>)` ### `intersect([intersect], [keep])`
To perform an intersection of attachables, you can use the `intersect()` module. If given one To perform an intersection of attachables, you can use the `intersect()` module. This is
argument to `a=`, the parent and all children *not* tagged with that will be intersected by specifically intended to address the situation where you want intersections involving a parent
everything that *is* tagged with it. and a child, something that is impossible with the native `intersection()` module. This module
treats the children in three groups: objects matching the `intersect` tags, objects matching
the tags listed in `keep` and the remaining objects that don't match any listed tags. The
intersection is computed between the union of the `intersect` tagged objects and the union of
the objects that don't match any listed tags. Finally the objects lsited in `keep` are union
ed with the result.
In this example the parent is intersected with a conical bounding shape.
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
intersect("bounds") intersect("bounds")
cube(100, center=true) cube(100, center=true)
cylinder(h=100, d1=120, d2=95, center=true, $fn=72, $tag="bounds"); tag("bounds") cylinder(h=100, d1=120, d2=95, center=true, $fn=72);
``` ```
If given both the `a=` and `b=` arguments, then shapes marked with tags given to `a=` will be In this example the child objects are intersected with the bounding box parent.
intersected with shapes marked with tags given to `b=`, then unioned with all other shapes.
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
intersect("pole", "cap") intersect("pole cap")
cube(100, center=true) cube(100, center=true)
attach([TOP,RIGHT]) { attach([TOP,RIGHT]) {
cube([40,40,80],center=true, $tag="pole"); tag("pole")cube([40,40,80],center=true);
sphere(d=40*sqrt(2), $tag="cap"); tag("cap")sphere(d=40*sqrt(2));
} }
``` ```
If the `keep=` argument is given, anything marked with tags passed to it will be unioned with The default `intersect` tag is "intersect" and the default `keep` tag is "keep". Here is an
the result of the union: example where "keep" is used to keep the pole from being removed by the intersection.
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
intersect("bounds", keep="pole") intersect()
cube(100, center=true) { cube(100, center=true) {
cylinder(h=100, d1=120, d2=95, center=true, $fn=72, $tag="bounds"); tag("intersect")cylinder(h=100, d1=120, d2=95, center=true, $fn=72);
zrot(45) xcyl(h=140, d=20, $fn=36, $tag="pole"); tag("keep")zrot(45) xcyl(h=140, d=20, $fn=36);
} }
``` ```
### `conv_hull(keep)` ### `conv_hull([keep])`
You can use the `conv_hull()` module to hull shapes together the You can use the `conv_hull()` module to hull shapes together. Objects
shapes not marked with the keep tags, before unioning the keep shapes marked with the keep tags are excluded from the hull and unioned into the final result.
into the final result. The default keep tag is "keep".
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
conv_hull("pole") conv_hull()
cube(50, center=true, $tag="hull") { cube(50, center=true) {
cyl(h=100, d=20); cyl(h=100, d=20);
xcyl(h=100, d=20, $tag="pole"); tag("keep")xcyl(h=100, d=20);
} }
``` ```

View file

@ -13,11 +13,11 @@ Transforms | Related Distributors
`left()`, `right()` | `xcopies()` `left()`, `right()` | `xcopies()`
`fwd()`, `back()` | `ycopies()` `fwd()`, `back()` | `ycopies()`
`down()`, `up()` | `zcopies()` `down()`, `up()` | `zcopies()`
`move()`, `translate()` | `move_copies()`, `line_of()`, `grid_copies()` `move()`, `translate()` | `move_copies()`, `line_copies()`, `grid_copies()`
`xrot()` | `xrot_copies()` `xrot()` | `xrot_copies()`
`yrot()` | `yrot_copies()` `yrot()` | `yrot_copies()`
`zrot()` | `zrot_copies()` `zrot()` | `zrot_copies()`
`rot()`, `rotate()` | `rot_copies()`, `arc_of()` `rot()`, `rotate()` | `rot_copies()`, `arc_copies()`
`xflip()` | `xflip_copy()` `xflip()` | `xflip_copy()`
`yflip()` | `yflip_copy()` `yflip()` | `yflip_copy()`
`zflip()` | `zflip_copy()` `zflip()` | `zflip_copy()`
@ -106,24 +106,24 @@ zcopies(20, n=5, sp=[0,0,0]) sphere(d=10);
``` ```
If you need to distribute copies along an arbitrary line, you can use the If you need to distribute copies along an arbitrary line, you can use the
`line_of()` command. You can give both the direction vector and the spacing `line_copies()` command. You can give both the direction vector and the spacing
of the line of copies with the `spacing=` argument: of the line of copies with the `spacing=` argument:
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
line_of(spacing=(BACK+RIGHT)*20, n=5) sphere(d=10); line_copies(spacing=(BACK+RIGHT)*20, n=5) sphere(d=10);
``` ```
With the `p1=` argument, you can specify the starting point of the line: With the `p1=` argument, you can specify the starting point of the line:
```openscad-3D ```openscad-3D
include <BOSL2/std.scad> include <BOSL2/std.scad>
line_of(spacing=(BACK+RIGHT)*20, n=5, p1=[0,0,0]) sphere(d=10); line_copies(spacing=(BACK+RIGHT)*20, n=5, p1=[0,0,0]) sphere(d=10);
``` ```
If you give both `p1=` and `p2=`, you can nail down both the start and If you give both `p1=` and `p2=`, you can nail down both the start and
endpoints of the line of copies: endpoints of the line of copies:
```openscad-2D ```openscad-2D
include <BOSL2/std.scad> include <BOSL2/std.scad>
line_of(p1=[0,100,0], p2=[100,0,0], n=4) line_copies(p1=[0,100,0], p2=[100,0,0], n=4)
sphere(d=10); sphere(d=10);
``` ```