mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-01 09:49:45 +00:00
commit
e049e2e2c2
3 changed files with 264 additions and 130 deletions
|
@ -1,7 +1,7 @@
|
||||||
DocsDirectory: BOSL2.wiki/
|
DocsDirectory: BOSL2.wiki/
|
||||||
TargetProfile: githubwiki
|
TargetProfile: githubwiki
|
||||||
ProjectName: The Belfry OpenScad Library, v2. (BOSL2)
|
ProjectName: The Belfry OpenScad Library, v2. (BOSL2)
|
||||||
GenerateDocs: Files, TOC, Index, Topics, CheatSheet
|
GenerateDocs: Files, TOC, Index, Topics, CheatSheet, Sidebar
|
||||||
IgnoreFiles:
|
IgnoreFiles:
|
||||||
affine.scad
|
affine.scad
|
||||||
foo.scad
|
foo.scad
|
||||||
|
|
|
@ -429,7 +429,9 @@ module orient(dir, anchor, spin) {
|
||||||
assert(is_vector(dir));
|
assert(is_vector(dir));
|
||||||
spin = default(spin, 0);
|
spin = default(spin, 0);
|
||||||
assert(is_finite(spin));
|
assert(is_finite(spin));
|
||||||
rot(spin, from=UP, to=dir) children();
|
two_d = _attach_geom_2d($parent_geom);
|
||||||
|
fromvec = two_d? BACK : UP;
|
||||||
|
rot(spin, from=fromvec, to=dir) children();
|
||||||
} else {
|
} else {
|
||||||
assert(dir==undef, "Only one of dir= or anchor= may be given to orient()");
|
assert(dir==undef, "Only one of dir= or anchor= may be given to orient()");
|
||||||
assert($parent_geom != undef, "No parent to orient from!");
|
assert($parent_geom != undef, "No parent to orient from!");
|
||||||
|
|
|
@ -4,27 +4,251 @@
|
||||||
|
|
||||||
## Primitives
|
## Primitives
|
||||||
There are two built-in 2D primitive shapes that OpenSCAD provides: `square()`, and `circle()`.
|
There are two built-in 2D primitive shapes that OpenSCAD provides: `square()`, and `circle()`.
|
||||||
The BOSL2 library provides alternative to these shapes so that they support more features,
|
You can still use them in the familiar ways that OpenSCAD provides:
|
||||||
and more ways to simply reorient them.
|
|
||||||
|
|
||||||
|
|
||||||
### 2D Squares
|
|
||||||
You can still use the built-in `square()` in the familiar ways that OpenSCAD provides:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
square(100, center=false);
|
|
||||||
```
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
square(100, center=true);
|
|
||||||
```
|
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
square([60,40], center=true);
|
square([60,40], center=true);
|
||||||
```
|
```
|
||||||
|
|
||||||
The BOSL2 library provides an enhanced equivalent to `square()` called `rect()`.
|
```openscad-2D
|
||||||
You can use it in the same way you use `square()`, but it also provides
|
circle(r=50);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=100, $fn=8);
|
||||||
|
```
|
||||||
|
|
||||||
|
These modules have also been enhanced in the BOSL2 library in three ways: Anchoring, spin, and
|
||||||
|
attachability.
|
||||||
|
|
||||||
|
#### Anchoring:
|
||||||
|
When you create a `square()`, you can specify what corner or side will be anchored at the
|
||||||
|
origin. This is used in place of the `center=` argument, and is more flexible. The `anchor=`
|
||||||
|
argument takes a vector as a value, pointing roughly towards the side or corner you
|
||||||
|
want to align to the origin. For example, to align the center of the back edge to the
|
||||||
|
origin, set the anchor to `[0,1]`:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=[0,1]);
|
||||||
|
```
|
||||||
|
|
||||||
|
To align the front right corner to the origin:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=[1,-1]);
|
||||||
|
```
|
||||||
|
|
||||||
|
To center:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=[0,0]);
|
||||||
|
```
|
||||||
|
|
||||||
|
To make it clearer when giving vectors, there are several standard vector constants defined:
|
||||||
|
|
||||||
|
Constant | Direction | Value
|
||||||
|
-------- | --------- | -----------
|
||||||
|
`LEFT` | X- | `[-1, 0, 0]`
|
||||||
|
`RIGHT` | X+ | `[ 1, 0, 0]`
|
||||||
|
`FRONT`/`FORWARD`/`FWD` | Y- | `[ 0,-1, 0]`
|
||||||
|
`BACK` | Y+ | `[ 0, 1, 0]`
|
||||||
|
`BOTTOM`/`BOT`/`BTM`/`DOWN` | Z- | `[ 0, 0,-1]` (3D only.)
|
||||||
|
`TOP`/`UP` | Z+ | `[ 0, 0, 1]` (3D only.)
|
||||||
|
`CENTER`/`CTR` | Centered | `[ 0, 0, 0]`
|
||||||
|
|
||||||
|
Note that even though these are 3D vectors, you can use most of them,
|
||||||
|
(except `UP`/`DOWN`, of course) for anchors in 2D shapes:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=BACK);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=CENTER);
|
||||||
|
```
|
||||||
|
|
||||||
|
You can add vectors together to point to corners:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=FRONT+RIGHT);
|
||||||
|
```
|
||||||
|
|
||||||
|
For `circle()`, the anchor vector can point at any part of the circle perimeter:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50, anchor=polar_to_xy(1,150));
|
||||||
|
```
|
||||||
|
|
||||||
|
You can see the typical anchor points by making `show_anchors()` the child of the shape:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], center=true)
|
||||||
|
show_anchors();
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
show_anchors();
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Spin:
|
||||||
|
The second way that `square()` and `circle()` have been enhanced is with spin. When you create
|
||||||
|
the shape, you can spin it in place with the `spin=` argument. You just pass it a number of
|
||||||
|
degrees to rotate clockwise:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=CENTER, spin=30);
|
||||||
|
```
|
||||||
|
|
||||||
|
Anchoring or centering is performed before the spin:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([60,40], anchor=BACK, spin=30);
|
||||||
|
```
|
||||||
|
|
||||||
|
For circles, spin can be useful when `$fn=` is also given:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50, $fn=6, spin=15);
|
||||||
|
```
|
||||||
|
|
||||||
|
Since anchoring is performed before spin, you can use them together to spin around the anchor:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50, $fn=6, anchor=LEFT, spin=15);
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
#### Attachability:
|
||||||
|
|
||||||
|
The third way `square()` and `circle()` have been enhanced is that you can attach them together
|
||||||
|
at anchoring points in various ways. This is done by making one shape a child of the shape
|
||||||
|
you want to attach to. By default, just making one shape a child of the other will position
|
||||||
|
the child shape at the center of the parent shape.
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
#square(50, spin=45, center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
#square([30,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
By adding the `position()` module, you can position the child at any anchorpoint on the parent:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
position(BACK)
|
||||||
|
#square(25, spin=45, center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
position(FWD+RIGHT)
|
||||||
|
#square(25, spin=45, center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
position(polar_to_xy(1,60))
|
||||||
|
#square(15, center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
Anchorpoints aren't just positions on the parent, though. They also have an orientation. In most
|
||||||
|
cases, the orientation of an anchorpoint is outward away from the face of the wall, generally away
|
||||||
|
from the center of the shape. You can see this with the `show_anchors()` module:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
show_anchors();
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
show_anchors();
|
||||||
|
```
|
||||||
|
|
||||||
|
If you want to orient the child to match the orientation of an anchorpoint, you can use the `orient()`
|
||||||
|
module. It does not position the child. It only rotates it:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
orient(anchor=LEFT)
|
||||||
|
#square([10,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
orient(anchor=FWD)
|
||||||
|
#square([10,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
orient(anchor=RIGHT)
|
||||||
|
#square([10,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
orient(polar_to_xy(1,30))
|
||||||
|
#square([10,50], center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
You can use `position()` and `orient()` together to both position and orient to an anchorpoint:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
position(RIGHT+BACK)
|
||||||
|
orient(anchor=RIGHT+BACK)
|
||||||
|
#square([10,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
position(polar_to_xy(1,30))
|
||||||
|
orient(polar_to_xy(1,30))
|
||||||
|
#square([10,50], center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
But it's simpler to just use the `attach()` module to do both at once:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square(50, center=true)
|
||||||
|
attach(LEFT+BACK)
|
||||||
|
#square([10,50], anchor=FWD);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
attach(polar_to_xy(1,30))
|
||||||
|
#square([10,50], center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
Instead of specifying the `anchor=` in the child, you can pass a second argument to `attach()`
|
||||||
|
that tells it which side of the child to attach to the parent:
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
square([10,50], center=true)
|
||||||
|
attach(BACK, LEFT)
|
||||||
|
#square([10,50], center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
```openscad-2D
|
||||||
|
circle(d=50)
|
||||||
|
attach(polar_to_xy(1,30), LEFT)
|
||||||
|
#square([10,50], center=true);
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#### `rect()`
|
||||||
|
|
||||||
|
The BOSL2 library provides an alternative to `square()`, that support more features. It is
|
||||||
|
called `rect()`. You can use it in the same way you use `square()`, but it also provides
|
||||||
extended functionality. For example, it allows you to round the corners:
|
extended functionality. For example, it allows you to round the corners:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
|
@ -74,108 +298,22 @@ a corner, specify a 0 chamfer for that corner, and vice versa:
|
||||||
rect([60,40], rounding=[5,0,10,0], chamfer=[0,5,0,15]);
|
rect([60,40], rounding=[5,0,10,0], chamfer=[0,5,0,15]);
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Anchors and Spin
|
### `ellipse()`
|
||||||
Another way that `rect()` is enhanced over `square()`, is that you can anchor,
|
|
||||||
spin and attach it.
|
|
||||||
|
|
||||||
The `anchor=` argument takes a vector as a value, pointing roughly towards
|
|
||||||
the side or corner you want to align to the origin. For example, to align
|
|
||||||
the center of the back edge to the origin, set the anchor to `[0,1]`:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=[0,1]);
|
|
||||||
```
|
|
||||||
|
|
||||||
To align the front right corner to the origin:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=[1,-1]);
|
|
||||||
```
|
|
||||||
|
|
||||||
To center:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=[0,0]);
|
|
||||||
```
|
|
||||||
|
|
||||||
To make it clearer when giving vectors, there are several standard vector
|
|
||||||
constants defined:
|
|
||||||
|
|
||||||
Constant | Direction | Value
|
|
||||||
-------- | --------- | -----------
|
|
||||||
`LEFT` | X- | `[-1, 0, 0]`
|
|
||||||
`RIGHT` | X+ | `[ 1, 0, 0]`
|
|
||||||
`FRONT`/`FORWARD`/`FWD` | Y- | `[ 0,-1, 0]`
|
|
||||||
`BACK` | Y+ | `[ 0, 1, 0]`
|
|
||||||
`BOTTOM`/`BOT`/`BTM`/`DOWN` | Z- | `[ 0, 0,-1]` (3D only.)
|
|
||||||
`TOP`/`UP` | Z+ | `[ 0, 0, 1]` (3D only.)
|
|
||||||
`CENTER`/`CTR` | Centered | `[ 0, 0, 0]`
|
|
||||||
|
|
||||||
Note that even though these are 3D vectors, you can use most of them,
|
|
||||||
(except `UP`/`DOWN`, of course) for anchors in 2D shapes:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=BACK);
|
|
||||||
```
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=CENTER);
|
|
||||||
```
|
|
||||||
|
|
||||||
You can add vectors together to point to corners:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=FRONT+RIGHT);
|
|
||||||
```
|
|
||||||
|
|
||||||
Finally, the `spin` argument can rotate the shape by a given number of degrees
|
|
||||||
clockwise:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=CENTER, spin=30);
|
|
||||||
```
|
|
||||||
|
|
||||||
Anchoring or centering is performed before the spin:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40], anchor=BACK, spin=30);
|
|
||||||
```
|
|
||||||
|
|
||||||
Anchor points double as attachment points, so that you can attach other shapes:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
rect([60,40])
|
|
||||||
show_anchors();
|
|
||||||
```
|
|
||||||
|
|
||||||
### 2D Circles and Ovals
|
|
||||||
The built-in `circle()` primitive can be used as expected:
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
circle(r=50);
|
|
||||||
```
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
circle(d=100);
|
|
||||||
```
|
|
||||||
|
|
||||||
```openscad-2D
|
|
||||||
circle(d=100, $fn=8);
|
|
||||||
```
|
|
||||||
|
|
||||||
The BOSL2 library also provides an enhanced equivalent of `circle()` called `ellipse()`.
|
The BOSL2 library also provides an enhanced equivalent of `circle()` called `ellipse()`.
|
||||||
You can use it in the same way you use `circle()`, but it also provides extended
|
You can use it in the same way you use `circle()`, but it also provides extended
|
||||||
functionality. For example, it allows more control over its size and orientation.
|
functionality. For example, it allows more control over its size.
|
||||||
|
|
||||||
Since a circle in OpenSCAD can only be approximated by a regular polygon with
|
Since a circle in OpenSCAD can only be approximated by a regular polygon with a number
|
||||||
a number of straight sides, this can lead to size and shape inaccuracies.
|
of straight sides, this can lead to size and shape inaccuracies. To counter this, the
|
||||||
To counter this, the `realign=` and `circum=` arguments are also provided.
|
`realign=` and `circum=` arguments are also provided.
|
||||||
|
|
||||||
The `realign=` argument, if set `true`, rotates the `ellipse()` by half the angle
|
The `realign=` argument, if set `true`, rotates the `ellipse()` by half the angle
|
||||||
between the sides:
|
between the polygon sides:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
ellipse(d=100, $fn=8, realign=true);
|
ellipse(d=100, $fn=8);
|
||||||
|
#ellipse(d=100, $fn=8, realign=true);
|
||||||
```
|
```
|
||||||
|
|
||||||
The `circum=` argument, if true, makes it so that the polygon forming the
|
The `circum=` argument, if true, makes it so that the polygon forming the
|
||||||
|
@ -184,19 +322,15 @@ The `circum=` argument, if true, makes it so that the polygon forming the
|
||||||
Inscribing the ideal circle:
|
Inscribing the ideal circle:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
difference() {
|
color("green") ellipse(d=100, $fn=360);
|
||||||
circle(d=100, $fn=360);
|
ellipse(d=100, $fn=6);
|
||||||
ellipse(d=100, $fn=8);
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Circumscribing the ideal circle:
|
Circumscribing the ideal circle:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
difference() {
|
ellipse(d=100, $fn=6, circum=true);
|
||||||
ellipse(d=100, $fn=8, circum=true);
|
color("green") ellipse(d=100, $fn=360);
|
||||||
circle(d=100, $fn=360);
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
The `ellipse()` module, as its name suggests, can be given separate X and Y radii
|
The `ellipse()` module, as its name suggests, can be given separate X and Y radii
|
||||||
|
@ -211,22 +345,20 @@ ellipse(r=[30,20]);
|
||||||
ellipse(d=[60,40]);
|
ellipse(d=[60,40]);
|
||||||
```
|
```
|
||||||
|
|
||||||
Another way that `ellipse()` is enhanced over `circle()`, is that you can anchor,
|
Like `circle()`, you can anchor, spin and attach `ellipse()` shapes:
|
||||||
spin and attach it.
|
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
ellipse(r=50, anchor=BACK);
|
ellipse(d=50, anchor=BACK);
|
||||||
```
|
```
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
ellipse(r=50, anchor=FRONT+RIGHT);
|
ellipse(d=50, anchor=FRONT+RIGHT);
|
||||||
```
|
```
|
||||||
|
|
||||||
Using spin on a circle may not make initial sense, until you remember that
|
|
||||||
anchoring is performed before spin:
|
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
ellipse(r=50, anchor=FRONT, spin=-30);
|
ellipse(d=50)
|
||||||
|
attach(BACK+RIGHT, FRONT+LEFT)
|
||||||
|
ellipse(d=40);
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
@ -278,7 +410,7 @@ trapezoid(w1=50, w2=30, shift=20, h=50);
|
||||||
```
|
```
|
||||||
|
|
||||||
You can use `anchor=` and `spin=`, just like with other attachable shapes. However, the anchor
|
You can use `anchor=` and `spin=`, just like with other attachable shapes. However, the anchor
|
||||||
points are based on the side angles of the faces, and may not be where you expect them:
|
point orientations are based on the side angles of the faces, and may not be what you expect:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
trapezoid(w1=30, w2=50, h=50)
|
trapezoid(w1=30, w2=50, h=50)
|
||||||
|
@ -327,7 +459,7 @@ pentagon(ir=25);
|
||||||
pentagon(id=50);
|
pentagon(id=50);
|
||||||
```
|
```
|
||||||
|
|
||||||
They can be realigned by half a side's angle:
|
They can be rotated by half a side:
|
||||||
|
|
||||||
```openscad-2D
|
```openscad-2D
|
||||||
left(30) pentagon(d=50, realign=true);
|
left(30) pentagon(d=50, realign=true);
|
||||||
|
|
Loading…
Reference in a new issue