mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
tutorial update, doc fixes, bugfixes
This commit is contained in:
parent
3fa4f967d4
commit
5564b4d5e1
2 changed files with 153 additions and 81 deletions
211
attachments.scad
211
attachments.scad
|
@ -602,14 +602,13 @@ module orient(anchor, spin) {
|
||||||
// align = optional alignment direction or directions for aligning the children. Default: CENTER
|
// align = optional alignment direction or directions for aligning the children. Default: CENTER
|
||||||
// ---
|
// ---
|
||||||
// inside = if true, place object inside the parent instead of outside. Default: false
|
// inside = if true, place object inside the parent instead of outside. Default: false
|
||||||
// inset = a value to shift the child inward, away from the alignent location. Default: 0
|
// inset = shift the child away from the alignment edge/corner by this amount. Default: 0
|
||||||
// shiftout = A value to shift an inside object outward so that it overlaps all the aligned faces. Default: 0
|
// shiftout = Shift an inside object outward so that it overlaps all the aligned faces. Default: 0
|
||||||
// overlap = Amount to sink the child into the parent. Defaults to `$overlap` which is zero by default.
|
// overlap = Amount to sink the child into the parent. Defaults to `$overlap` which is zero by default.
|
||||||
// Side Effects:
|
// Side Effects:
|
||||||
// `$anchor` set to the anchor value used for the child.
|
// `$anchor` set to the anchor value used for the child.
|
||||||
// `$align` set to the align value used for the child.
|
// `$align` set to the align value used for the child.
|
||||||
// `$idx` set to a unique index for each child, increasing by alignment first.
|
// `$idx` set to a unique index for each child, increasing by alignment first.
|
||||||
// `$pos` position where child was placed.
|
|
||||||
// `$attach_anchor` for each anchor given, this is set to the `[ANCHOR, POSITION, ORIENT, SPIN]` information for that anchor.
|
// `$attach_anchor` for each anchor given, this is set to the `[ANCHOR, POSITION, ORIENT, SPIN]` information for that anchor.
|
||||||
// if inside is true then set default tag to "remove"
|
// if inside is true then set default tag to "remove"
|
||||||
// Example: Cuboid positioned on the right of its parent. Note that it is in its native orientation.
|
// Example: Cuboid positioned on the right of its parent. Note that it is in its native orientation.
|
||||||
|
@ -687,9 +686,7 @@ module align(anchor,align=CENTER,inside=false,inset=0,shiftout=0,overlap)
|
||||||
anchor = is_vector(anchor) ? [anchor] : anchor;
|
anchor = is_vector(anchor) ? [anchor] : anchor;
|
||||||
align = is_vector(align) ? [align] : align;
|
align = is_vector(align) ? [align] : align;
|
||||||
two_d = _attach_geom_2d($parent_geom);
|
two_d = _attach_geom_2d($parent_geom);
|
||||||
|
|
||||||
factor = inside?-1:1;
|
factor = inside?-1:1;
|
||||||
|
|
||||||
for (i = idx(anchor)) {
|
for (i = idx(anchor)) {
|
||||||
face = anchor[i];
|
face = anchor[i];
|
||||||
$anchor=face;
|
$anchor=face;
|
||||||
|
@ -731,7 +728,7 @@ function _quant_anch(x) = approx(x,0) ? 0 : sign(x);
|
||||||
|
|
||||||
// Make arbitrary anchor legal for a given geometry
|
// Make arbitrary anchor legal for a given geometry
|
||||||
function _make_anchor_legal(anchor,geom) =
|
function _make_anchor_legal(anchor,geom) =
|
||||||
in_list(geom[0], ["prismoid","trapzeoid"]) ? [for(v=anchor) _quant_anch(v)]
|
in_list(geom[0], ["prismoid","trapezoid"]) ? [for(v=anchor) _quant_anch(v)]
|
||||||
: in_list(geom[0], ["conoid", "extrusion_extent"]) ? [anchor.x,anchor.y, _quant_anch(anchor.z)]
|
: in_list(geom[0], ["conoid", "extrusion_extent"]) ? [anchor.x,anchor.y, _quant_anch(anchor.z)]
|
||||||
: anchor;
|
: anchor;
|
||||||
|
|
||||||
|
@ -743,7 +740,7 @@ function _make_anchor_legal(anchor,geom) =
|
||||||
// Topics: Attachments
|
// Topics: Attachments
|
||||||
// See Also: attachable(), position(), align(), face_profile(), edge_profile(), corner_profile()
|
// See Also: attachable(), position(), align(), face_profile(), edge_profile(), corner_profile()
|
||||||
// Usage:
|
// Usage:
|
||||||
// PARENT() attach(parent, child, [align=], [spin=], [overlap=], [inside=]) CHILDREN;
|
// PARENT() attach(parent, child, [align=], [spin=], [overlap=], [inside=], [inset=], [shiftout=]) CHILDREN;
|
||||||
// PARENT() attach(parent, [overlap=], [spin=]) CHILDREN;
|
// PARENT() attach(parent, [overlap=], [spin=]) CHILDREN;
|
||||||
// Description:
|
// Description:
|
||||||
// Attaches children to a parent object at an anchor point or points, oriented in the anchor direction.
|
// Attaches children to a parent object at an anchor point or points, oriented in the anchor direction.
|
||||||
|
@ -762,24 +759,30 @@ function _make_anchor_legal(anchor,geom) =
|
||||||
// When an object is attached to one of the other anchors its FRONT will be pointed DOWN and its
|
// When an object is attached to one of the other anchors its FRONT will be pointed DOWN and its
|
||||||
// BACK pointed UP. You can change this using the `spin=` argument to attach(). Note that this spin
|
// BACK pointed UP. You can change this using the `spin=` argument to attach(). Note that this spin
|
||||||
// rotates around the attachment vector and is not the same as the spin argument to the child, which
|
// rotates around the attachment vector and is not the same as the spin argument to the child, which
|
||||||
// will usually rotate around some other direction that may be hard to predict.
|
// will usually rotate around some other direction that may be hard to predict. For 2D objects you cannot
|
||||||
|
// give spin because it is not possible to spin around the attachment vector; spinning the object around the Z axis
|
||||||
|
// would change the child orientation so that the anchors are no longer parallel. Furthermore, any spin
|
||||||
|
// parameter you give to the child will be ignored so that the attachment condition of parallel anchors is preserved.
|
||||||
// .
|
// .
|
||||||
// As with {{align()}} you can use the `align=` parameter to align the child to an edge or corner of the
|
// As with {{align()}} you can use the `align=` parameter to align the child to an edge or corner of the
|
||||||
// face where that child is attached. For example `attach(TOP,BOT,align=RIGHT)` would stand the child
|
// face where that child is attached. For example `attach(TOP,BOT,align=RIGHT)` would stand the child
|
||||||
// up on the top while aligning it with the right edge of the top, and `attach(RIGHT,BOT,align=TOP)` which
|
// up on the top while aligning it with the right edge of the top face, and `attach(RIGHT,BOT,align=TOP)` which
|
||||||
// stand the object on the right face while aligning with the top edge. If you apply spin using the
|
// stand the object on the right face while aligning with the top edge. If you apply spin using the
|
||||||
// argument to `attach()` then it will be taken into account for the alignment. If you apply spin with
|
// argument to `attach()` then it will be taken into account for the alignment. If you apply spin with
|
||||||
// a parameter to the child it will NOT be taken into account.
|
// a parameter to the child it will NOT be taken into account. Note that spin is not permitted for
|
||||||
// .
|
// 2D objects because it would change the child orientation so that the anchors are no longer parallel.
|
||||||
// Because the attachment process forces an orientation and anchor point for the child, it overrides
|
// When you use `align=` you can also adjust the position using `inset=`, which shifts the child
|
||||||
// any such specifications you give to the child: both `anchor=` and `orient=` given to the child are
|
// away from the edge or corner it is aligned to.
|
||||||
// ignored with the **double argument** version of `attach()`. As noted above, you can give `spin=` to the
|
|
||||||
// child but using the `spin=` parameter to `attach()` is more likely to be useful.
|
|
||||||
// .
|
// .
|
||||||
// If you give `inside=true` then the anchor arrows are lined up so they are pointing the same direction and
|
// If you give `inside=true` then the anchor arrows are lined up so they are pointing the same direction and
|
||||||
// the child object will be located inside the parent. In this case a default "remove" tag is applied to
|
// the child object will be located inside the parent. In this case a default "remove" tag is applied to
|
||||||
// the children.
|
// the children.
|
||||||
// .
|
// .
|
||||||
|
// Because the attachment process forces an orientation and anchor point for the child, it overrides
|
||||||
|
// any such specifications you give to the child: **both `anchor=` and `orient=` given to the child are
|
||||||
|
// ignored** with the **double argument** version of `attach()`. As noted above, you can give `spin=` to the
|
||||||
|
// child but using the `spin=` parameter to `attach()` is more likely to be useful.
|
||||||
|
// .
|
||||||
// For the single parameter version of `attach()` you give only the `parent` anchor. The `align` direction
|
// For the single parameter version of `attach()` you give only the `parent` anchor. The `align` direction
|
||||||
// is not permitted. In this case the child is placed at the specified parent anchor point
|
// is not permitted. In this case the child is placed at the specified parent anchor point
|
||||||
// and rotated to the anchor direction. For example, `attach(TOP) cuboid(2);` will place a small
|
// and rotated to the anchor direction. For example, `attach(TOP) cuboid(2);` will place a small
|
||||||
|
@ -787,7 +790,7 @@ function _make_anchor_legal(anchor,geom) =
|
||||||
// from the parent. If you want the cube sitting on the parent you need to anchor the cube to its bottom:
|
// from the parent. If you want the cube sitting on the parent you need to anchor the cube to its bottom:
|
||||||
// `attach(TOP) cuboid(2,anchor=BOT);`.
|
// `attach(TOP) cuboid(2,anchor=BOT);`.
|
||||||
// .
|
// .
|
||||||
// The **single argument** version of `attach()` respects `anchor=` and `orient=` given to the child.
|
// The **single argument** version of `attach()` **respects `anchor=` and `orient=` given to the child.**
|
||||||
// These options will probably be necessary, in fact, to get the child correctly positioned. Note that
|
// These options will probably be necessary, in fact, to get the child correctly positioned. Note that
|
||||||
// giving `spin=` to `attach()` in this case is the same as applying `zrot()` to the child.
|
// giving `spin=` to `attach()` in this case is the same as applying `zrot()` to the child.
|
||||||
// .
|
// .
|
||||||
|
@ -802,18 +805,68 @@ function _make_anchor_legal(anchor,geom) =
|
||||||
// parent = The parent anchor point to attach to or a list of parent anchor points.
|
// parent = The parent anchor point to attach to or a list of parent anchor points.
|
||||||
// child = Optional child anchor point. If given, orients the child to connect this anchor point to the parent anchor.
|
// child = Optional child anchor point. If given, orients the child to connect this anchor point to the parent anchor.
|
||||||
// ---
|
// ---
|
||||||
// align = If `child` is given you can specify alignment to shift the child to an edge or corner of the parent.
|
// align = If `child` is given you can specify alignment or list of alistnments to shift the child to an edge or corner of the parent.
|
||||||
|
// inset = Shift aligned children away from their alignment edge/corner by this amount. Default: 0
|
||||||
|
// shiftout = Shift an inside object outward so that it overlaps all the aligned faces. Default: 0
|
||||||
|
// inside = If `child` is given you can set `inside=true` to attach the child to the inside of the parent for diff() operations. Default: false
|
||||||
// overlap = Amount to sink child into the parent. Equivalent to `down(X)` after the attach. This defaults to the value in `$overlap`, which is `0` by default.
|
// overlap = Amount to sink child into the parent. Equivalent to `down(X)` after the attach. This defaults to the value in `$overlap`, which is `0` by default.
|
||||||
// spin = Amount to rotate the parent around the axis of the parent anchor.
|
// spin = Amount to rotate the parent around the axis of the parent anchor. (Only permitted in 3D.)
|
||||||
// Side Effects:
|
// Side Effects:
|
||||||
// `$idx` is set to the index number of each anchor if a list of anchors is given. Otherwise is set to `0`.
|
// `$anchor` set to the parent anchor value used for the child.
|
||||||
// `$attach_anchor` for each `from=` anchor given, this is set to the `[ANCHOR, POSITION, ORIENT, SPIN]` information for that anchor.
|
// `$align` set to the align value used for the child.
|
||||||
// `$attach_to` is set to the value of the `to=` argument, if given. Otherwise, `undef`
|
// `$idx` set to a unique index for each child, increasing by alignment first.
|
||||||
// Example:
|
// `$attach_anchor` for each anchor given, this is set to the `[ANCHOR, POSITION, ORIENT, SPIN]` information for that anchor.
|
||||||
// spheroid(d=20) {
|
// if inside is true then set default tag to "remove"
|
||||||
// attach(TOP) down(1.5) cyl(l=11.5, d1=10, d2=5, anchor=BOTTOM);
|
// `$attach_to` is set to the value of the `child` argument, if given. Otherwise, `undef`
|
||||||
// attach(RIGHT, BOTTOM) down(1.5) cyl(l=11.5, d1=10, d2=5);
|
// Example: Cylinder placed on top of cube:
|
||||||
// attach(FRONT, BOTTOM, overlap=1.5) cyl(l=11.5, d1=10, d2=5);
|
// cuboid(50)
|
||||||
|
// attach(TOP,BOT) cylinder(d1=30,d2=15,h=25);
|
||||||
|
// Example: Cylinder on right and front side of cube:
|
||||||
|
// cuboid(50)
|
||||||
|
// attach([RIGHT,FRONT],BOT) cylinder(d1=30,d2=15,h=25);
|
||||||
|
// Example: Using `align` can align child object(s) with edges
|
||||||
|
// prismoid(50,25,25) color("green"){
|
||||||
|
// attach(TOP,BOT,align=[BACK,FWD]) cuboid(4);
|
||||||
|
// attach(RIGHT,BOT,align=[TOP,BOT]) cuboid(4);
|
||||||
|
// }
|
||||||
|
// Example: One aligned to the corner upside down (light blue) and one inset fromt the corner (pink), one aligned on a side (orange) and one rotated and aligned (green).
|
||||||
|
// cuboid(30) {
|
||||||
|
// attach(TOP,TOP,align=FRONT+RIGHT) color("lightblue") prismoid(5,3,3);
|
||||||
|
// attach(TOP,BOT,inset=3,align=FRONT+LEFT) color("pink") prismoid(5,3,3);
|
||||||
|
// attach(FRONT,RIGHT,align=TOP) color("orange") prismoid(5,3,3);
|
||||||
|
// attach(FRONT,RIGHT,align=RIGHT,spin=90) color("lightgreen") prismoid(5,3,3);
|
||||||
|
// }
|
||||||
|
// Example: Rotation not a multiple of 90 degrees with alignment. The children are aligned on a corner.
|
||||||
|
// cuboid(30)
|
||||||
|
// attach(FRONT,BOT,spin=33,align=[RIGHT,LEFT,TOP,BOT,RIGHT+TOP])
|
||||||
|
// color("lightblue")cuboid(4);
|
||||||
|
// Example: Anchoring the cone onto the sphere gives a single point of contact.
|
||||||
|
// spheroid(d=20)
|
||||||
|
// attach([1,1.5,1], BOTTOM) cyl(l=11.5, d1=10, d2=5);
|
||||||
|
// Example: Using the `overlap` option can help:
|
||||||
|
// spheroid(d=20)
|
||||||
|
// attach([1,1.5,1], BOTTOM, overlap=1.5) cyl(l=11.5, d1=10, d2=5);
|
||||||
|
// Example: Alignment works for cylinders but you can only align with either the top or bototm face:
|
||||||
|
// cyl(h=30,d=10)
|
||||||
|
// attach([LEFT,[1,1.3]], BOT,align=TOP) cuboid(6);
|
||||||
|
// Example: Attaching to edges. The light blue and orange objects are attached to edges. The purple object is attached to an edge and aligned.
|
||||||
|
// prismoid([20,10],[10,10],7){
|
||||||
|
// attach(RIGHT+TOP,BOT,align=FRONT) color("pink")cuboid(2);
|
||||||
|
// attach(BACK+TOP, BOT) color("lightblue")cuboid(2);
|
||||||
|
// attach(RIGHT+BOT, RIGHT,spin=90) color("orange")cyl(h=8,d=1);
|
||||||
|
// }
|
||||||
|
// Example: Attaching inside the parent. For inside attachment the anchors are lined up pointing the same direction, so the most natural way to anchor the child is using its TOP anchor.
|
||||||
|
// back_half()
|
||||||
|
// diff()
|
||||||
|
// cuboid(20)
|
||||||
|
// attach(TOP,TOP,inside=true,shiftout=0.01) cyl(d1=10,d2=5,h=10);
|
||||||
|
// Example: Attaching inside the parent with alignment
|
||||||
|
// diff()
|
||||||
|
// cuboid(20){
|
||||||
|
// attach(TOP,TOP,inside=true,align=RIGHT,shiftout=.01) cuboid([8,7,3]);
|
||||||
|
// attach(TOP,TOP,inside=true,align=LEFT+FRONT,shiftout=0.01) cuboid([3,4,5]);
|
||||||
|
// attach(RIGHT+FRONT, TOP, inside=true) cuboid([10,3,5]);
|
||||||
|
// attach(RIGHT+FRONT, TOP, inside=true, align=TOP,shiftout=.01) cuboid([5,1,2]);
|
||||||
// }
|
// }
|
||||||
|
|
||||||
module attach(parent, child, overlap, align, spin=0, norot, inset=0, shiftout=0, inside=false, from, to)
|
module attach(parent, child, overlap, align, spin=0, norot, inset=0, shiftout=0, inside=false, from, to)
|
||||||
|
@ -830,62 +883,72 @@ module attach(parent, child, overlap, align, spin=0, norot, inset=0, shiftout=0,
|
||||||
req_children($children);
|
req_children($children);
|
||||||
|
|
||||||
dummy=assert($parent_geom != undef, "No object to attach to!")
|
dummy=assert($parent_geom != undef, "No object to attach to!")
|
||||||
assert(is_undef(align) || (is_vector(align) && (len(align)==2 || len(align)==3)), "align must be a 2-vector or 3-vector")
|
|
||||||
assert(is_undef(child) || is_string(child) || (is_vector(child) && (len(child)==2 || len(child)==3)), "child must be a named anchor (a string) or a 2-vector or 3-vector")
|
assert(is_undef(child) || is_string(child) || (is_vector(child) && (len(child)==2 || len(child)==3)), "child must be a named anchor (a string) or a 2-vector or 3-vector")
|
||||||
assert(is_undef(align) || !is_string(child), "child is a named anchor. Named anchors are not supported with align=");
|
assert(is_undef(align) || !is_string(child), "child is a named anchor. Named anchors are not supported with align=");
|
||||||
|
|
||||||
overlap = (overlap!=undef)? overlap : $overlap;
|
|
||||||
anchors = (is_vector(parent)||is_string(parent))? [parent] : parent;
|
|
||||||
two_d = _attach_geom_2d($parent_geom);
|
two_d = _attach_geom_2d($parent_geom);
|
||||||
|
overlap = (overlap!=undef)? overlap : $overlap;
|
||||||
parent = first_defined([parent,from]);
|
parent = first_defined([parent,from]);
|
||||||
dummy4 = assert(is_string(parent) || is_list(parent), "Invalid parent anchor or anchor list");
|
anchors = is_vector(parent) || is_string(parent) ? [parent] : parent;
|
||||||
|
align_list = is_undef(align) ? [undef]
|
||||||
|
: is_vector(align) || is_string(align) ? [align] : align;
|
||||||
|
dummy4 = assert(is_string(parent) || is_list(parent), "Invalid parent anchor or anchor list")
|
||||||
|
assert(spin==0 || (!two_d || is_undef(child)), "spin is not allowed for 2d objects when 'child' is given");
|
||||||
child_temp = first_defined([child,to]);
|
child_temp = first_defined([child,to]);
|
||||||
child = two_d ? _force_anchor_2d(child_temp) : child_temp;
|
child = two_d ? _force_anchor_2d(child_temp) : child_temp;
|
||||||
align = is_undef(align) ? undef
|
dummy2=assert(align_list==[undef] || is_def(child), "Cannot use 'align' without 'child'")
|
||||||
: two_d ? _force_anchor_2d(align) : point3d(align);
|
|
||||||
dummy2=assert(is_undef(align) || is_def(child), "Cannot use 'align' without 'child'")
|
|
||||||
assert(!inside || is_def(child), "Cannot use 'inside' without 'child'")
|
assert(!inside || is_def(child), "Cannot use 'inside' without 'child'")
|
||||||
assert(inset==0 || is_def(child), "Cannot specify 'inset' without 'child'")
|
assert(inset==0 || is_def(child), "Cannot specify 'inset' without 'child'")
|
||||||
assert(shiftout==0 || is_def(child), "Cannot specify 'shiftout' without 'child'");
|
assert(shiftout==0 || is_def(child), "Cannot specify 'shiftout' without 'child'");
|
||||||
for ($idx = idx(anchors)) {
|
factor = inside?-1:1;
|
||||||
dummy2=
|
$attach_to = u_mul(factor,child);
|
||||||
assert(is_string(anchors[$idx]) || (is_vector(anchors[$idx]) && (len(anchors[$idx])==2 || len(anchors[$idx])==3)),
|
for (anch_ind = idx(anchors)) {
|
||||||
str("parent[",$idx,"] is ",anchors[$idx]," but it must be a named anchor (string) or a 2-vector or 3-vector"))
|
dummy=assert(is_string(anchors[anch_ind]) || (is_vector(anchors[anch_ind]) && (len(anchors[anch_ind])==2 || len(anchors[anch_ind])==3)),
|
||||||
assert(is_undef(align) || !is_string(anchors[$idx]),
|
str("parent[",anch_ind,"] is ",anchors[anch_ind]," but it must be a named anchor (string) or a 2-vector or 3-vector"))
|
||||||
str("parent[",$idx,"] is a named anchor (",anchors[$idx],"), but named anchors are not wupported with align="));
|
assert(align_list==[undef] || !is_string(anchors[anch_ind]),
|
||||||
anchr = is_string(anchors[$idx])? anchors[$idx]
|
str("parent[",anch_ind,"] is a named anchor (",anchors[anch_ind],"), but named anchors are not supported with align="));
|
||||||
: two_d?_force_anchor_2d(anchors[$idx])
|
anchor = is_string(anchors[anch_ind])? anchors[anch_ind]
|
||||||
:anchors[$idx];
|
: two_d?_force_anchor_2d(anchors[anch_ind])
|
||||||
dummy=assert(is_undef(align) || all_zero(v_mul(anchr,align)),
|
: point3d(anchors[anch_ind]);
|
||||||
str("align (",align,") cannot include component parallel to parent anchor (",anchr,")"));
|
anchor_data = _find_anchor(anchor, $parent_geom);
|
||||||
anch = _find_anchor(anchr, $parent_geom);
|
$anchor=anchor;
|
||||||
pos = is_undef(align) ? anch[1] : _find_anchor(anchr+align, $parent_geom)[1];
|
for(align_ind = idx(align_list)){
|
||||||
factor = inside?-1:1;
|
align = is_undef(align_list[align_ind]) ? undef
|
||||||
$attach_to = is_vector(child) ? factor*child : child;
|
: assert(is_vector(align_list[align_ind],2) || is_vector(align_list[align_ind],3), "align direction must be a 2-vector or 3-vector")
|
||||||
$attach_anchor = list_set(anch, 1, pos); ///
|
two_d ? _force_anchor_2d(align_list[align_ind])
|
||||||
startdir = anchr==UP || anchr==DOWN ? BACK : UP;
|
: point3d(align_list[align_ind]);
|
||||||
enddir = is_undef(child) || child.z==0 ? UP : BACK;
|
$idx = align_ind+len(align_list)*anch_ind;
|
||||||
anchor_adjustment = is_undef(align)? CTR
|
$align=align;
|
||||||
: two_d ? zrot(spin, rot(to=factor*child,from=-anchr,p=align))
|
dummy=assert(is_undef(align) || all_zero(v_mul(anchor,align)),
|
||||||
: apply( frame_map(x=factor*child, z=enddir)
|
str("Invalid alignment: align value (",align,") includes component parallel to parent anchor (",anchor,")"));
|
||||||
*frame_map(x=-anchr, z=startdir, reverse=true)
|
pos = is_undef(align) ? anchor_data[1] : _find_anchor(anchor+align, $parent_geom)[1];
|
||||||
*rot(v=parent,-spin), align);
|
$attach_anchor = list_set(anchor_data, 1, pos); ///
|
||||||
$anchor_override=all_zero(anchor_adjustment)? inside?child:undef
|
startdir = anchor==UP || anchor==DOWN ? BACK : UP - (anchor*UP)*anchor/(anchor*anchor);
|
||||||
:child+anchor_adjustment;
|
enddir = is_undef(child) || child.z==0 ? UP : BACK;
|
||||||
reference = two_d? BACK : UP;
|
child_adjustment = is_undef(align)? CTR
|
||||||
offsetdir = is_undef(align) ? CTR
|
: two_d ? rot(to=factor*child,from=-anchor,p=align)
|
||||||
: apply(zrot(-spin)*frame_map(x=reference, z=BACK)*frame_map(x=anchr, z=startdir, reverse=true),
|
: apply( frame_map(x=factor*child, z=enddir)
|
||||||
align);
|
*frame_map(x=-anchor, z=startdir, reverse=true)
|
||||||
spinaxis = two_d? UP : anch[2];
|
*rot(v=anchor,-spin), align);
|
||||||
olap = - overlap * reference - inset*offsetdir - shiftout * (-offsetdir - reference);
|
$anchor_override = all_zero(child_adjustment)? inside?child:undef
|
||||||
if (norot || (approx(anch[2],reference) && anch[3]==0)) {
|
: two_d ? zrot(-spin, child+child_adjustment)
|
||||||
translate(pos) rot(v=spinaxis,a=spin) translate(olap) default_tag("remove",inside) children();
|
: child+child_adjustment;
|
||||||
} else {
|
reference = two_d? BACK : UP;
|
||||||
translate(pos)
|
inset_dir = is_undef(align) ? CTR
|
||||||
rot(v=spinaxis,a=spin)
|
: two_d ? zrot(-spin, rot(to=reference, from=anchor,p=align))
|
||||||
rot(anch[3],from=reference,to=anch[2]){
|
: apply(zrot(-spin)*frame_map(x=reference, z=BACK)*frame_map(x=anchor, z=startdir, reverse=true),
|
||||||
translate(olap)
|
align);
|
||||||
default_tag("remove",inside) children();}}
|
spinaxis = two_d? UP : anchor_data[2];
|
||||||
|
olap = - overlap * reference - inset*inset_dir - shiftout * (-inset_dir - reference);
|
||||||
|
if (norot || (approx(anchor_data[2],reference) && anchor_data[3]==0)) {
|
||||||
|
translate(pos) rot(v=spinaxis,a=spin) translate(olap) default_tag("remove",inside) children();
|
||||||
|
} else {
|
||||||
|
translate(pos)
|
||||||
|
rot(v=spinaxis,a=spin)
|
||||||
|
rot(anchor_data[3],from=reference,to=anchor_data[2]){
|
||||||
|
translate(olap)
|
||||||
|
default_tag("remove",inside) children();}}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3399,8 +3462,8 @@ function _attach_transform(anchor, spin, orient, geom, p) =
|
||||||
)
|
)
|
||||||
two_d?
|
two_d?
|
||||||
assert(is_num(spin))
|
assert(is_num(spin))
|
||||||
affine3d_zrot(spin)
|
/*affine3d_zrot(spin) * */
|
||||||
* rot(to=FWD, from=point3d(anch[2]))
|
rot(to=FWD, from=point3d(anch[2]))
|
||||||
* affine3d_translate(point3d(-pos))
|
* affine3d_translate(point3d(-pos))
|
||||||
:
|
:
|
||||||
assert(is_num(spin) || is_vector(spin,3))
|
assert(is_num(spin) || is_vector(spin,3))
|
||||||
|
|
|
@ -1172,25 +1172,34 @@ cube(50, center=true)
|
||||||
|
|
||||||
## Attaching 2D Children
|
## Attaching 2D Children
|
||||||
You can use attachments in 2D as well. As usual for the 2D case you
|
You can use attachments in 2D as well. As usual for the 2D case you
|
||||||
can use TOP and BOTTOM as alternative to BACK and FORWARD.
|
can use TOP and BOTTOM as alternative to BACK and FORWARD. With
|
||||||
|
parent-child anchor attachment you cannot use the spin parameter to
|
||||||
|
`attach()` nor can you specify spin to the child. Spinning the child
|
||||||
|
on the Z axis would rotate the anchor arrows out of alignment.
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
include <BOSL2/std.scad>
|
include <BOSL2/std.scad>
|
||||||
square(50,center=true)
|
rect(50){
|
||||||
attach(RIGHT,FRONT)
|
attach(RIGHT,FRONT)
|
||||||
trapezoid(w1=30,w2=0,h=30);
|
color("red")trapezoid(w1=30,w2=0,h=30);
|
||||||
|
attach(LEFT,FRONT,align=[FRONT,BACK],inset=3)
|
||||||
|
color("green") trapezoid(w1=25, w2=0,h=30);
|
||||||
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
include <BOSL2/std.scad>
|
include <BOSL2/std.scad>
|
||||||
circle(d=50)
|
diff()
|
||||||
|
circle(d=50){
|
||||||
attach(TOP,BOT,overlap=5)
|
attach(TOP,BOT,overlap=5)
|
||||||
trapezoid(w1=30,w2=0,h=30);
|
trapezoid(w1=30,w2=0,h=30);
|
||||||
|
attach(BOT,BOT,inside=true)
|
||||||
|
tag("remove")
|
||||||
|
trapezoid(w1=30,w2=0,h=30);
|
||||||
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## 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.
|
||||||
|
|
Loading…
Reference in a new issue