2019-05-12 03:32:56 -07:00
//////////////////////////////////////////////////////////////////////
// LibFile: shapes2d.scad
2021-09-16 21:50:12 -04:00
// This file includes redefinitions of the core modules to
2021-11-11 19:34:16 -05:00
// work with attachment, and functional forms of those modules
// that produce paths. You can create regular polygons
2021-09-06 19:46:18 -04:00
// with optional rounded corners and alignment features not
// available with circle(). The file also provides teardrop2d,
2021-11-11 19:34:16 -05:00
// which is useful for 3D printable holes.
2021-09-06 19:46:18 -04:00
// Many of the commands have module forms that produce geometry and
2021-11-11 19:34:16 -05:00
// function forms that produce a path.
2021-01-05 01:20:01 -08:00
// Includes:
2019-05-12 03:32:56 -07:00
// include <BOSL2/std.scad>
2021-12-13 15:48:30 -08:00
// FileGroup: Basic Modeling
// FileSummary: Attachable circles, squares, polygons, teardrop. Can make geometry or paths.
// FileFootnotes: STD=Included in std.scad
2019-05-12 03:32:56 -07:00
//////////////////////////////////////////////////////////////////////
2021-10-02 09:35:06 -04:00
use < builtins.scad >
2021-09-30 01:22:43 -07:00
2020-03-06 15:32:53 -08:00
2020-04-25 04:00:16 -07:00
// Section: 2D Primitives
2021-09-16 21:50:12 -04:00
// Function&Module: square()
// Topics: Shapes (2D), Path Generators (2D)
2021-09-30 01:22:43 -07:00
// Usage: As a Module
// square(size, [center], ...);
// Usage: With Attachments
// square(size, [center], ...) { attachables }
2021-09-16 21:50:12 -04:00
// Usage: As a Function
2021-09-30 01:22:43 -07:00
// path = square(size, [center], ...);
2021-09-16 21:50:12 -04:00
// See Also: rect()
// Description:
// When called as the builtin module, creates a 2D square or rectangle of the given size.
// When called as a function, returns a 2D path/list of points for a square/rectangle of the given size.
// Arguments:
// size = The size of the square to create. If given as a scalar, both X and Y will be the same size.
// center = If given and true, overrides `anchor` to be `CENTER`. If given and false, overrides `anchor` to be `FRONT+LEFT`.
// ---
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2021-09-16 21:50:12 -04:00
// Example(2D):
// square(40);
// Example(2D): Centered
// square([40,30], center=true);
// Example(2D): Called as Function
// path = square([40,30], anchor=FRONT, spin=30);
// stroke(path, closed=true);
// move_copies(path) color("blue") circle(d=2,$fn=8);
function square ( size = 1 , center , anchor , spin = 0 ) =
let (
anchor = get_anchor ( anchor , center , [ - 1 , - 1 ] , [ - 1 , - 1 ] ) ,
size = is_num ( size ) ? [ size , size ] : point2d ( size ) ,
path = [
[ size . x , - size . y ] ,
[ - size . x , - size . y ] ,
[ - size . x , size . y ] ,
[ size . x , size . y ]
] / 2
) reorient ( anchor , spin , two_d = true , size = size , p = path ) ;
2021-09-30 01:22:43 -07:00
module square ( size = 1 , center , anchor , spin ) {
anchor = get_anchor ( anchor , center , [ - 1 , - 1 ] , [ - 1 , - 1 ] ) ;
size = is_num ( size ) ? [ size , size ] : point2d ( size ) ;
2021-09-30 02:04:31 -07:00
attachable ( anchor , spin , two_d = true , size = size ) {
2021-09-30 01:22:43 -07:00
_square ( size , center = true ) ;
children ( ) ;
}
}
2021-09-16 21:50:12 -04:00
2020-04-25 04:00:16 -07:00
// Function&Module: rect()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-11-20 23:33:04 -05:00
// rect(size, [rounding], [chamfer], ...);
2021-02-25 02:16:55 -08:00
// Usage: With Attachments
2021-11-20 23:33:04 -05:00
// rect(size, ...) { attachables }
2021-02-25 02:16:55 -08:00
// Usage: As Function
2021-11-20 23:33:04 -05:00
// path = rect(size, [rounding], [chamfer], ...);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
// See Also: square()
2020-04-25 04:00:16 -07:00
// Description:
// When called as a module, creates a 2D rectangle of the given size, with optional rounding or chamfering.
// When called as a function, returns a 2D path/list of points for a square/rectangle of the given size.
// Arguments:
// size = The size of the rectangle to create. If given as a scalar, both X and Y will be the same size.
// rounding = The rounding radius for the corners. If given as a list of four numbers, gives individual radii for each corner, in the order [X+Y+,X-Y+,X-Y-,X+Y-]. Default: 0 (no rounding)
// chamfer = The chamfer size for the corners. If given as a list of four numbers, gives individual chamfers for each corner, in the order [X+Y+,X-Y+,X-Y-,X+Y-]. Default: 0 (no chamfer)
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-04-25 04:00:16 -07:00
// Example(2D):
// rect(40);
// Example(2D): Anchored
// rect([40,30], anchor=FRONT);
// Example(2D): Spun
// rect([40,30], anchor=FRONT, spin=30);
// Example(2D): Chamferred Rect
2021-11-20 23:33:04 -05:00
// rect([40,30], chamfer=5);
2020-04-25 04:00:16 -07:00
// Example(2D): Rounded Rect
2021-11-20 23:33:04 -05:00
// rect([40,30], rounding=5);
2020-04-25 04:00:16 -07:00
// Example(2D): Mixed Chamferring and Rounding
2021-11-20 23:33:04 -05:00
// rect([40,30],rounding=[5,0,10,0],chamfer=[0,8,0,15],$fa=1,$fs=1);
2020-04-25 04:00:16 -07:00
// Example(2D): Called as Function
// path = rect([40,30], chamfer=5, anchor=FRONT, spin=30);
// stroke(path, closed=true);
// move_copies(path) color("blue") circle(d=2,$fn=8);
2021-11-20 23:33:04 -05:00
module rect ( size = 1 , rounding = 0 , chamfer = 0 , anchor = CENTER , spin = 0 ) {
2020-05-29 19:04:34 -07:00
size = is_num ( size ) ? [ size , size ] : point2d ( size ) ;
if ( rounding = = 0 && chamfer = = 0 ) {
2021-11-20 23:33:04 -05:00
attachable ( anchor , spin , two_d = true , size = size ) {
2020-05-29 19:04:34 -07:00
square ( size , center = true ) ;
children ( ) ;
}
} else {
2021-12-13 16:31:14 -08:00
pts = rect ( size = size , rounding = rounding , chamfer = chamfer ) ;
2021-11-20 23:33:04 -05:00
attachable ( anchor , spin , two_d = true , path = pts ) {
2020-05-29 19:04:34 -07:00
polygon ( pts ) ;
children ( ) ;
}
}
2020-04-25 04:00:16 -07:00
}
2021-11-20 23:33:04 -05:00
function rect ( size = 1 , rounding = 0 , chamfer = 0 , anchor = CENTER , spin = 0 ) =
2020-05-29 19:04:34 -07:00
assert ( is_num ( size ) || is_vector ( size ) )
assert ( is_num ( chamfer ) || len ( chamfer ) = = 4 )
assert ( is_num ( rounding ) || len ( rounding ) = = 4 )
let (
2021-11-20 23:33:04 -05:00
anchor = point2d ( anchor ) ,
2020-05-29 19:04:34 -07:00
size = is_num ( size ) ? [ size , size ] : point2d ( size ) ,
complex = rounding ! = 0 || chamfer ! = 0
)
( rounding = = 0 && chamfer = = 0 ) ? let (
path = [
[ size . x / 2 , - size . y / 2 ] ,
[ - size . x / 2 , - size . y / 2 ] ,
[ - size . x / 2 , size . y / 2 ] ,
[ size . x / 2 , size . y / 2 ]
]
2021-11-20 23:33:04 -05:00
)
rot ( spin , p = move ( - v_mul ( anchor , size / 2 ) , p = path ) ) :
2020-05-29 19:04:34 -07:00
let (
chamfer = is_list ( chamfer ) ? chamfer : [ for ( i = [ 0 : 3 ] ) chamfer ] ,
rounding = is_list ( rounding ) ? rounding : [ for ( i = [ 0 : 3 ] ) rounding ] ,
quadorder = [ 3 , 2 , 1 , 0 ] ,
quadpos = [ [ 1 , 1 ] , [ - 1 , 1 ] , [ - 1 , - 1 ] , [ 1 , - 1 ] ] ,
insets = [ for ( i = [ 0 : 3 ] ) chamfer [ i ] > 0 ? chamfer [ i ] : rounding [ i ] > 0 ? rounding [ i ] : 0 ] ,
insets_x = max ( insets [ 0 ] + insets [ 1 ] , insets [ 2 ] + insets [ 3 ] ) ,
insets_y = max ( insets [ 0 ] + insets [ 3 ] , insets [ 1 ] + insets [ 2 ] )
)
assert ( insets_x < = size . x , "Requested roundings and/or chamfers exceed the rect width." )
assert ( insets_y < = size . y , "Requested roundings and/or chamfers exceed the rect height." )
let (
path = [
for ( i = [ 0 : 3 ] )
let (
quad = quadorder [ i ] ,
inset = insets [ quad ] ,
cverts = quant ( segs ( inset ) , 4 ) / 4 ,
2021-06-14 20:28:49 -07:00
cp = v_mul ( size / 2 - [ inset , inset ] , quadpos [ quad ] ) ,
2020-05-29 19:04:34 -07:00
step = 90 / cverts ,
angs =
chamfer [ quad ] > 0 ? [ 0 , - 90 ] - 90 * [ i , i ] :
rounding [ quad ] > 0 ? [ for ( j = [ 0 : 1 : cverts ] ) 360 - j * step - i * 90 ] :
[ 0 ]
)
each [ for ( a = angs ) cp + inset * [ cos ( a ) , sin ( a ) ] ]
]
) complex ?
reorient ( anchor , spin , two_d = true , path = path , p = path ) :
reorient ( anchor , spin , two_d = true , size = size , p = path ) ;
2020-04-25 04:00:16 -07:00
2021-09-16 21:50:12 -04:00
// Function&Module: circle()
// Topics: Shapes (2D), Path Generators (2D)
2021-09-30 01:22:43 -07:00
// Usage: As a Module
2021-09-16 21:50:12 -04:00
// circle(r|d=, ...);
2021-09-30 01:22:43 -07:00
// Usage: With Attachments
// circle(r|d=, ...) { attachables }
2021-09-16 21:50:12 -04:00
// Usage: As a Function
// path = circle(r|d=, ...);
2021-11-18 19:02:50 -05:00
// See Also: ellipse(), circle_2tangents(), circle_3points()
2021-09-16 21:50:12 -04:00
// Description:
// When called as the builtin module, creates a 2D polygon that approximates a circle of the given size.
// When called as a function, returns a 2D list of points (path) for a polygon that approximates a circle of the given size.
// Arguments:
// r = The radius of the circle to create.
// d = The diameter of the circle to create.
// ---
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2021-09-16 21:50:12 -04:00
// Example(2D): By Radius
// circle(r=25);
// Example(2D): By Diameter
// circle(d=50);
// Example(NORENDER): Called as Function
// path = circle(d=50, anchor=FRONT, spin=45);
function circle ( r , d , anchor = CENTER , spin = 0 ) =
let (
r = get_radius ( r = r , d = d , dflt = 1 ) ,
sides = segs ( r ) ,
path = [ for ( i = [ 0 : 1 : sides - 1 ] ) let ( a = 360 - i * 360 / sides ) r * [ cos ( a ) , sin ( a ) ] ]
) reorient ( anchor , spin , two_d = true , r = r , p = path ) ;
2021-09-30 01:22:43 -07:00
module circle ( r , d , anchor = CENTER , spin = 0 ) {
r = get_radius ( r = r , d = d , dflt = 1 ) ;
2021-09-30 02:04:31 -07:00
attachable ( anchor , spin , two_d = true , r = r ) {
2021-09-30 01:22:43 -07:00
_circle ( r = r ) ;
children ( ) ;
}
}
2021-09-16 21:50:12 -04:00
2021-11-13 19:45:58 -05:00
// Function&Module: ellipse()
2021-09-30 01:22:43 -07:00
// Usage: As a Module
2021-11-13 19:45:58 -05:00
// ellipse(r|d=, [realign=], [circum=], ...);
2021-09-30 01:22:43 -07:00
// Usage: With Attachments
2021-11-13 19:45:58 -05:00
// ellipse(r|d=, [realign=], [circum=], ...) { attachables }
2021-09-30 01:22:43 -07:00
// Usage: As a Function
2021-11-13 19:45:58 -05:00
// path = ellipse(r|d=, [realign=], [circum=], ...);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2022-01-05 16:20:42 -05:00
// See Also: circle(), circle_2tangents(), circle_3points()
2020-04-25 04:00:16 -07:00
// Description:
2021-11-09 22:27:55 -05:00
// When called as a module, creates a 2D polygon that approximates a circle or ellipse of the given size.
// When called as a function, returns a 2D list of points (path) for a polygon that approximates a circle or ellipse of the given size.
2021-11-18 19:02:50 -05:00
// By default the point list or shape is the same as the one you would get by scaling the output of {{circle()}}, but with this module your
// attachments to the ellipse will retain their dimensions, whereas scaling a circle with attachments will also scale the attachments.
// If you set unifom to true then you will get a polygon with congruent sides whose vertices lie on the ellipse.
2020-04-25 04:00:16 -07:00
// Arguments:
2021-11-13 19:45:58 -05:00
// r = Radius of the circle or pair of semiaxes of ellipse
2021-02-25 02:16:55 -08:00
// ---
2021-11-09 22:27:55 -05:00
// d = Diameter of the circle or a pair giving the full X and Y axis lengths.
2021-11-18 19:02:50 -05:00
// realign = If false starts the approximate ellipse with a point on the X+ axis. If true the midpoint of a side is on the X+ axis and the first point of the polygon is below the X+ axis. This can result in a very different polygon when $fn is small. Default: false
2020-04-25 04:00:16 -07:00
// circum = If true, the polygon that approximates the circle will be upsized slightly to circumscribe the theoretical circle. If false, it inscribes the theoretical circle. Default: false
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-04-25 04:00:16 -07:00
// Example(2D): By Radius
2021-11-13 19:45:58 -05:00
// ellipse(r=25);
2020-04-25 04:00:16 -07:00
// Example(2D): By Diameter
2021-11-13 19:45:58 -05:00
// ellipse(d=50);
2020-04-25 04:00:16 -07:00
// Example(2D): Anchoring
2021-11-13 19:45:58 -05:00
// ellipse(d=50, anchor=FRONT);
2020-04-25 04:00:16 -07:00
// Example(2D): Spin
2021-11-13 19:45:58 -05:00
// ellipse(d=50, anchor=FRONT, spin=45);
2020-04-25 04:00:16 -07:00
// Example(NORENDER): Called as Function
2021-11-13 19:45:58 -05:00
// path = ellipse(d=50, anchor=FRONT, spin=45);
2021-11-18 19:02:50 -05:00
// Example(2D,NoAxes): Uniformly sampled hexagon at the top, regular non-uniform one at the bottom
// r=[10,3];
// ydistribute(7){
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=6)],width=0.1,color="red");
// }
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=6,uniform=true)],width=0.1,color="red");
// }
// }
// Example(2D): The realigned hexagons are even more different
// r=[10,3];
// ydistribute(7){
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=6,realign=true)],width=0.1,color="red");
// }
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=6,realign=true,uniform=true)],width=0.1,color="red");
// }
// }
// Example(2D): For odd $fn the result may not look very elliptical:
// r=[10,3];
// ydistribute(7){
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=5,realign=false)],width=0.1,color="red");
// }
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.05,color="blue");
// stroke([ellipse(r=r, $fn=5,realign=false,uniform=true)],width=0.1,color="red");
// }
// }
// Example(2D): The same ellipse, turned 90 deg, gives a very different result:
// r=[3,10];
// xdistribute(7){
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.1,color="blue");
// stroke([ellipse(r=r, $fn=5,realign=false)],width=0.2,color="red");
// }
// union(){
// stroke([ellipse(r=r, $fn=100)],width=0.1,color="blue");
// stroke([ellipse(r=r, $fn=5,realign=false,uniform=true)],width=0.2,color="red");
// }
// }
2021-11-16 18:49:37 -05:00
module ellipse ( r , d , realign = false , circum = false , uniform = false , anchor = CENTER , spin = 0 )
{
r = force_list ( get_radius ( r = r , d = d , dflt = 1 ) , 2 ) ;
dummy = assert ( is_vector ( r , 2 ) && all_positive ( r ) , "Invalid radius or diameter for ellipse" ) ;
2020-05-29 19:04:34 -07:00
sides = segs ( max ( r ) ) ;
sc = circum ? ( 1 / cos ( 180 / sides ) ) : 1 ;
2021-11-16 18:49:37 -05:00
rx = r . x * sc ;
ry = r . y * sc ;
2020-05-29 19:04:34 -07:00
attachable ( anchor , spin , two_d = true , r = [ rx , ry ] ) {
2021-11-16 18:49:37 -05:00
if ( uniform ) {
assert ( ! circum , "Circum option not allowed when \"uniform\" is true" ) ;
polygon ( ellipse ( r , realign = realign , circum = circum , uniform = true ) ) ;
}
else if ( rx < ry ) {
2020-05-29 19:04:34 -07:00
xscale ( rx / ry ) {
zrot ( realign ? 180 / sides : 0 ) {
circle ( r = ry , $fn = sides ) ;
}
}
} else {
yscale ( ry / rx ) {
zrot ( realign ? 180 / sides : 0 ) {
circle ( r = rx , $fn = sides ) ;
}
}
}
children ( ) ;
}
2020-04-25 04:00:16 -07:00
}
2021-11-16 18:49:37 -05:00
// Iterative refinement to produce an inscribed polygon
// in an ellipse whose side lengths are all equal
function _ellipse_refine ( a , b , N , _theta = [ ] ) =
len ( _theta ) = = 0 ? _ellipse_refine ( a , b , N , lerpn ( 0 , 360 , N , endpoint = false ) )
:
let (
pts = [ for ( t = _theta ) [ a * cos ( t ) , b * sin ( t ) ] ] ,
lenlist = path_segment_lengths ( pts , closed = true ) ,
meanlen = mean ( lenlist ) ,
error = lenlist / meanlen
)
all_equal ( error , EPSILON ) ? pts
:
let (
dtheta = [ each deltas ( _theta ) ,
360 - last ( _theta ) ] ,
newdtheta = [ for ( i = idx ( dtheta ) ) dtheta [ i ] / error [ i ] ] ,
adjusted = [ 0 , each cumsum ( list_head ( newdtheta / sum ( newdtheta ) * 360 ) ) ]
)
_ellipse_refine ( a , b , N , adjusted ) ;
2021-11-18 19:02:50 -05:00
function _ellipse_refine_realign ( a , b , N , _theta = [ ] , i = 0 ) =
len ( _theta ) = = 0 ?
_ellipse_refine_realign ( a , b , N , count ( N - 1 , 180 / N , 360 / N ) )
:
let (
pts = [ for ( t = _theta ) [ a * cos ( t ) , b * sin ( t ) ] ,
[ a * cos ( _theta [ 0 ] ) , - b * sin ( _theta [ 0 ] ) ] ] ,
lenlist = path_segment_lengths ( pts , closed = true ) ,
meanlen = mean ( lenlist ) ,
error = lenlist / meanlen
)
all_equal ( error , EPSILON ) ? pts
:
let (
dtheta = [ each deltas ( _theta ) ,
360 - last ( _theta ) - _theta [ 0 ] ,
2 * _theta [ 0 ] ] ,
newdtheta = [ for ( i = idx ( dtheta ) ) dtheta [ i ] / error [ i ] ] ,
normdtheta = newdtheta / sum ( newdtheta ) * 360 ,
adjusted = cumsum ( [ last ( normdtheta ) / 2 , each list_head ( normdtheta , - 3 ) ] )
)
_ellipse_refine_realign ( a , b , N , adjusted , i + 1 ) ;
2021-11-16 18:49:37 -05:00
function ellipse ( r , d , realign = false , circum = false , uniform = false , anchor = CENTER , spin = 0 ) =
let (
r = force_list ( get_radius ( r = r , d = d , dflt = 1 ) , 2 ) ,
sides = segs ( max ( r ) )
)
uniform ? assert ( ! circum , "Circum option not allowed when \"uniform\" is true" )
2021-11-18 19:02:50 -05:00
reorient ( anchor , spin , two_d = true , r = [ r . x , r . y ] ,
p = realign ? reverse ( _ellipse_refine_realign ( r . x , r . y , sides ) )
: reverse_polygon ( _ellipse_refine ( r . x , r . y , sides ) ) )
2021-11-16 18:49:37 -05:00
:
2020-05-29 19:04:34 -07:00
let (
offset = realign ? 180 / sides : 0 ,
sc = circum ? ( 1 / cos ( 180 / sides ) ) : 1 ,
2021-11-16 18:49:37 -05:00
rx = r . x * sc ,
ry = r . y * sc ,
2020-05-29 19:04:34 -07:00
pts = [ for ( i = [ 0 : 1 : sides - 1 ] ) let ( a = 360 - offset - i * 360 / sides ) [ rx * cos ( a ) , ry * sin ( a ) ] ]
) reorient ( anchor , spin , two_d = true , r = [ rx , ry ] , p = pts ) ;
2020-04-25 04:00:16 -07:00
2021-09-06 19:07:18 -04:00
// Section: Polygons
2019-05-12 03:32:56 -07:00
2019-05-12 18:10:15 -07:00
// Function&Module: regular_ngon()
2019-05-12 03:32:56 -07:00
// Usage:
2021-06-26 20:59:33 -07:00
// regular_ngon(n, r/d=/or=/od=, [realign=]);
// regular_ngon(n, ir=/id=, [realign=]);
// regular_ngon(n, side=, [realign=]);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), pentagon(), hexagon(), octagon(), ellipse(), star()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path for a regular N-sided polygon.
// When called as a module, creates a 2D regular N-sided polygon.
2019-05-12 03:32:56 -07:00
// Arguments:
// n = The number of sides.
2021-02-25 02:16:55 -08:00
// r/or = Outside radius, at points.
// ---
// d/od = Outside diameter, at points.
2019-05-12 03:32:56 -07:00
// ir = Inside radius, at center of sides.
// id = Inside diameter, at center of sides.
// side = Length of each side.
2020-01-14 16:23:35 -08:00
// rounding = Radius of rounding for the tips of the polygon. Default: 0 (no rounding)
2021-11-18 19:02:50 -05:00
// realign = If false, vertex 0 will lie on the X+ axis. If true then the midpoint of the last edge will lie on the X+ axis, and vertex 0 will be below the X axis. Default: false
2020-10-07 16:53:52 -07:00
// align_tip = If given as a 2D vector, rotates the whole shape so that the first vertex points in that direction. This occurs before spin.
// align_side = If given as a 2D vector, rotates the whole shape so that the normal of side0 points in that direction. This occurs before spin.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-05-06 17:17:45 -07:00
// Extra Anchors:
2020-05-07 17:11:35 -07:00
// "tip0", "tip1", etc. = Each tip has an anchor, pointing outwards.
2020-05-06 17:17:45 -07:00
// "side0", "side1", etc. = The center of each side has an anchor, pointing outwards.
2019-05-12 18:10:15 -07:00
// Example(2D): by Outer Size
// regular_ngon(n=5, or=30);
// regular_ngon(n=5, od=60);
// Example(2D): by Inner Size
// regular_ngon(n=5, ir=30);
// regular_ngon(n=5, id=60);
// Example(2D): by Side Length
// regular_ngon(n=8, side=20);
// Example(2D): Realigned
// regular_ngon(n=8, side=20, realign=true);
2020-10-07 16:53:52 -07:00
// Example(2D): Alignment by Tip
// regular_ngon(n=5, r=30, align_tip=BACK+RIGHT)
// attach("tip0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Alignment by Side
// regular_ngon(n=5, r=30, align_side=BACK+RIGHT)
// attach("side0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
2020-01-14 16:48:20 -08:00
// Example(2D): Rounded
// regular_ngon(n=5, od=100, rounding=20, $fn=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, regular_ngon(n=6, or=30));
2020-10-07 16:53:52 -07:00
function regular_ngon ( n = 6 , r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 , _mat , _anchs ) =
assert ( is_undef ( align_tip ) || is_vector ( align_tip ) )
assert ( is_undef ( align_side ) || is_vector ( align_side ) )
assert ( is_undef ( align_tip ) || is_undef ( align_side ) , "Can only specify one of align_tip and align-side" )
2020-05-29 19:04:34 -07:00
let (
sc = 1 / cos ( 180 / n ) ,
2020-10-03 19:50:29 -07:00
ir = is_finite ( ir ) ? ir * sc : undef ,
id = is_finite ( id ) ? id * sc : undef ,
side = is_finite ( side ) ? side / 2 / sin ( 180 / n ) : undef ,
r = get_radius ( r1 = ir , r2 = or , r = r , d1 = id , d2 = od , d = d , dflt = side )
2020-05-29 19:04:34 -07:00
)
assert ( ! is_undef ( r ) , "regular_ngon(): need to specify one of r, d, or, od, ir, id, side." )
let (
inset = opp_ang_to_hyp ( rounding , ( 180 - 360 / n ) / 2 ) ,
2020-10-07 16:53:52 -07:00
mat = ! is_undef ( _mat ) ? _mat :
2021-12-28 17:05:37 -05:00
( realign ? zrot ( - 180 / n ) : ident ( 4 ) ) * (
! is_undef ( align_tip ) ? rot ( from = RIGHT , to = point2d ( align_tip ) ) :
! is_undef ( align_side ) ? rot ( from = RIGHT , to = point2d ( align_side ) ) * zrot ( 180 / n ) :
1
2020-10-07 16:53:52 -07:00
) ,
2021-11-13 19:45:58 -05:00
path4 = rounding = = 0 ? ellipse ( r = r , $fn = n ) : (
2020-05-29 19:04:34 -07:00
let (
steps = floor ( segs ( r ) / n ) ,
step = 360 / n / steps ,
path2 = [
for ( i = [ 0 : 1 : n - 1 ] ) let (
2020-10-07 16:53:52 -07:00
a = 360 - i * 360 / n ,
2020-05-29 19:04:34 -07:00
p = polar_to_xy ( r - inset , a )
)
each arc ( N = steps , cp = p , r = rounding , start = a + 180 / n , angle = - 360 / n )
] ,
2021-10-26 16:45:14 -04:00
maxx_idx = max_index ( column ( path2 , 0 ) ) ,
2021-11-11 18:50:26 -05:00
path3 = list_rotate ( path2 , maxx_idx )
2020-05-29 19:04:34 -07:00
) path3
) ,
2020-10-07 16:53:52 -07:00
path = apply ( mat , path4 ) ,
anchors = ! is_undef ( _anchs ) ? _anchs :
! is_string ( anchor ) ? [ ] : [
2020-05-29 19:04:34 -07:00
for ( i = [ 0 : 1 : n - 1 ] ) let (
2020-10-07 16:53:52 -07:00
a1 = 360 - i * 360 / n ,
2020-05-29 19:04:34 -07:00
a2 = a1 - 360 / n ,
2020-10-07 16:53:52 -07:00
p1 = apply ( mat , polar_to_xy ( r , a1 ) ) ,
p2 = apply ( mat , polar_to_xy ( r , a2 ) ) ,
tipp = apply ( mat , polar_to_xy ( r - inset + rounding , a1 ) ) ,
2020-05-29 19:04:34 -07:00
pos = ( p1 + p2 ) / 2
) each [
2021-09-16 18:32:12 -04:00
named_anchor ( str ( "tip" , i ) , tipp , unit ( tipp , BACK ) , 0 ) ,
named_anchor ( str ( "side" , i ) , pos , unit ( pos , BACK ) , 0 ) ,
2020-05-29 19:04:34 -07:00
]
]
) reorient ( anchor , spin , two_d = true , path = path , extent = false , p = path , anchors = anchors ) ;
2019-05-12 03:32:56 -07:00
2020-10-07 16:53:52 -07:00
module regular_ngon ( n = 6 , r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 ) {
2020-05-29 19:04:34 -07:00
sc = 1 / cos ( 180 / n ) ;
2020-10-03 19:50:29 -07:00
ir = is_finite ( ir ) ? ir * sc : undef ;
id = is_finite ( id ) ? id * sc : undef ;
side = is_finite ( side ) ? side / 2 / sin ( 180 / n ) : undef ;
r = get_radius ( r1 = ir , r2 = or , r = r , d1 = id , d2 = od , d = d , dflt = side ) ;
2020-05-29 19:04:34 -07:00
assert ( ! is_undef ( r ) , "regular_ngon(): need to specify one of r, d, or, od, ir, id, side." ) ;
2021-12-28 17:05:37 -05:00
mat = ( realign ? zrot ( - 180 / n ) : ident ( 4 ) ) * (
! is_undef ( align_tip ) ? rot ( from = RIGHT , to = point2d ( align_tip ) ) :
! is_undef ( align_side ) ? rot ( from = RIGHT , to = point2d ( align_side ) ) * zrot ( 180 / n ) :
1
2020-10-07 16:53:52 -07:00
) ;
2020-05-29 19:04:34 -07:00
inset = opp_ang_to_hyp ( rounding , ( 180 - 360 / n ) / 2 ) ;
anchors = [
for ( i = [ 0 : 1 : n - 1 ] ) let (
2020-10-07 16:53:52 -07:00
a1 = 360 - i * 360 / n ,
2020-05-29 19:04:34 -07:00
a2 = a1 - 360 / n ,
2020-10-07 16:53:52 -07:00
p1 = apply ( mat , polar_to_xy ( r , a1 ) ) ,
p2 = apply ( mat , polar_to_xy ( r , a2 ) ) ,
tipp = apply ( mat , polar_to_xy ( r - inset + rounding , a1 ) ) ,
2020-05-29 19:04:34 -07:00
pos = ( p1 + p2 ) / 2
) each [
2021-09-16 18:32:12 -04:00
named_anchor ( str ( "tip" , i ) , tipp , unit ( tipp , BACK ) , 0 ) ,
named_anchor ( str ( "side" , i ) , pos , unit ( pos , BACK ) , 0 ) ,
2020-05-29 19:04:34 -07:00
]
] ;
2020-10-07 16:53:52 -07:00
path = regular_ngon ( n = n , r = r , rounding = rounding , _mat = mat , _anchs = anchors ) ;
2020-05-29 19:04:34 -07:00
attachable ( anchor , spin , two_d = true , path = path , extent = false , anchors = anchors ) {
polygon ( path ) ;
children ( ) ;
}
2019-11-03 19:01:14 -08:00
}
2019-05-12 03:32:56 -07:00
2019-05-12 18:10:15 -07:00
// Function&Module: pentagon()
2019-05-12 03:32:56 -07:00
// Usage:
2021-06-26 20:59:33 -07:00
// pentagon(or|od=, [realign=]);
// pentagon(ir=|id=, [realign=]);
// pentagon(side=, [realign=]);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), regular_ngon(), hexagon(), octagon(), ellipse(), star()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path for a regular pentagon.
// When called as a module, creates a 2D regular pentagon.
2019-05-12 03:32:56 -07:00
// Arguments:
2021-02-25 02:16:55 -08:00
// r/or = Outside radius, at points.
// ---
// d/od = Outside diameter, at points.
2019-05-12 03:32:56 -07:00
// ir = Inside radius, at center of sides.
// id = Inside diameter, at center of sides.
// side = Length of each side.
2020-01-14 16:23:35 -08:00
// rounding = Radius of rounding for the tips of the polygon. Default: 0 (no rounding)
2021-11-18 19:02:50 -05:00
// realign = If false, vertex 0 will lie on the X+ axis. If true then the midpoint of the last edge will lie on the X+ axis, and vertex 0 will be below the X axis. Default: false
2020-10-07 16:53:52 -07:00
// align_tip = If given as a 2D vector, rotates the whole shape so that the first vertex points in that direction. This occurs before spin.
// align_side = If given as a 2D vector, rotates the whole shape so that the normal of side0 points in that direction. This occurs before spin.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-05-06 17:17:45 -07:00
// Extra Anchors:
2020-05-07 17:11:35 -07:00
// "tip0" ... "tip4" = Each tip has an anchor, pointing outwards.
2020-05-06 17:17:45 -07:00
// "side0" ... "side4" = The center of each side has an anchor, pointing outwards.
2019-05-12 18:10:15 -07:00
// Example(2D): by Outer Size
2019-05-12 03:32:56 -07:00
// pentagon(or=30);
// pentagon(od=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Inner Size
2019-05-12 03:32:56 -07:00
// pentagon(ir=30);
// pentagon(id=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Side Length
2019-05-12 03:32:56 -07:00
// pentagon(side=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Realigned
// pentagon(side=20, realign=true);
2020-10-07 16:53:52 -07:00
// Example(2D): Alignment by Tip
// pentagon(r=30, align_tip=BACK+RIGHT)
// attach("tip0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Alignment by Side
// pentagon(r=30, align_side=BACK+RIGHT)
// attach("side0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
2020-01-14 16:48:20 -08:00
// Example(2D): Rounded
// pentagon(od=100, rounding=20, $fn=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, pentagon(or=30));
2020-10-07 16:53:52 -07:00
function pentagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 ) =
regular_ngon ( n = 5 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) ;
2019-05-12 03:32:56 -07:00
2020-10-07 16:53:52 -07:00
module pentagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 )
regular_ngon ( n = 5 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) children ( ) ;
2019-05-12 03:32:56 -07:00
2019-05-12 18:10:15 -07:00
// Function&Module: hexagon()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// hexagon(r/or, [realign=], <align_tip=|align_side=>, [rounding=], ...);
2021-02-25 02:16:55 -08:00
// hexagon(d=/od=, ...);
// hexagon(ir=/id=, ...);
// hexagon(side=, ...);
// Usage: With Attachments
// hexagon(r/or, ...) { attachments }
// Usage: As Function
// path = hexagon(r/or, ...);
// path = hexagon(d=/od=, ...);
// path = hexagon(ir=/id=, ...);
// path = hexagon(side=, ...);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), regular_ngon(), pentagon(), octagon(), ellipse(), star()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path for a regular hexagon.
// When called as a module, creates a 2D regular hexagon.
2019-05-12 03:32:56 -07:00
// Arguments:
2021-02-25 02:16:55 -08:00
// r/or = Outside radius, at points.
// ---
// d/od = Outside diameter, at points.
2019-05-12 03:32:56 -07:00
// ir = Inside radius, at center of sides.
// id = Inside diameter, at center of sides.
// side = Length of each side.
2020-01-14 16:23:35 -08:00
// rounding = Radius of rounding for the tips of the polygon. Default: 0 (no rounding)
2021-11-18 19:02:50 -05:00
// realign = If false, vertex 0 will lie on the X+ axis. If true then the midpoint of the last edge will lie on the X+ axis, and vertex 0 will be below the X axis. Default: false
2020-10-07 16:53:52 -07:00
// align_tip = If given as a 2D vector, rotates the whole shape so that the first vertex points in that direction. This occurs before spin.
// align_side = If given as a 2D vector, rotates the whole shape so that the normal of side0 points in that direction. This occurs before spin.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-05-06 17:17:45 -07:00
// Extra Anchors:
2020-05-07 17:11:35 -07:00
// "tip0" ... "tip5" = Each tip has an anchor, pointing outwards.
2020-05-06 17:17:45 -07:00
// "side0" ... "side5" = The center of each side has an anchor, pointing outwards.
2019-05-12 18:10:15 -07:00
// Example(2D): by Outer Size
2019-05-12 03:32:56 -07:00
// hexagon(or=30);
// hexagon(od=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Inner Size
2019-05-12 03:32:56 -07:00
// hexagon(ir=30);
// hexagon(id=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Side Length
2019-05-12 03:32:56 -07:00
// hexagon(side=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Realigned
// hexagon(side=20, realign=true);
2020-10-07 16:53:52 -07:00
// Example(2D): Alignment by Tip
// hexagon(r=30, align_tip=BACK+RIGHT)
// attach("tip0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Alignment by Side
// hexagon(r=30, align_side=BACK+RIGHT)
// attach("side0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
2020-01-14 16:48:20 -08:00
// Example(2D): Rounded
// hexagon(od=100, rounding=20, $fn=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, hexagon(or=30));
2020-10-07 16:53:52 -07:00
function hexagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 ) =
regular_ngon ( n = 6 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) ;
2019-05-12 03:32:56 -07:00
2020-10-07 16:53:52 -07:00
module hexagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 )
regular_ngon ( n = 6 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) children ( ) ;
2019-05-12 03:32:56 -07:00
2019-05-12 18:10:15 -07:00
// Function&Module: octagon()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// octagon(r/or, [realign=], <align_tip=|align_side=>, [rounding=], ...);
2021-02-25 02:16:55 -08:00
// octagon(d=/od=, ...);
// octagon(ir=/id=, ...);
// octagon(side=, ...);
// Usage: With Attachments
// octagon(r/or, ...) { attachments }
// Usage: As Function
// path = octagon(r/or, ...);
// path = octagon(d=/od=, ...);
// path = octagon(ir=/id=, ...);
// path = octagon(side=, ...);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), regular_ngon(), pentagon(), hexagon(), ellipse(), star()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path for a regular octagon.
// When called as a module, creates a 2D regular octagon.
2019-05-12 03:32:56 -07:00
// Arguments:
2021-02-25 02:16:55 -08:00
// r/or = Outside radius, at points.
// d/od = Outside diameter, at points.
2019-05-12 03:32:56 -07:00
// ir = Inside radius, at center of sides.
// id = Inside diameter, at center of sides.
// side = Length of each side.
2020-01-14 16:23:35 -08:00
// rounding = Radius of rounding for the tips of the polygon. Default: 0 (no rounding)
2021-11-18 19:02:50 -05:00
// realign = If false, vertex 0 will lie on the X+ axis. If true then the midpoint of the last edge will lie on the X+ axis, and vertex 0 will be below the X axis. Default: false
2020-10-07 16:53:52 -07:00
// align_tip = If given as a 2D vector, rotates the whole shape so that the first vertex points in that direction. This occurs before spin.
// align_side = If given as a 2D vector, rotates the whole shape so that the normal of side0 points in that direction. This occurs before spin.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-05-06 17:17:45 -07:00
// Extra Anchors:
2020-05-07 17:11:35 -07:00
// "tip0" ... "tip7" = Each tip has an anchor, pointing outwards.
2020-05-06 17:17:45 -07:00
// "side0" ... "side7" = The center of each side has an anchor, pointing outwards.
2019-05-12 18:10:15 -07:00
// Example(2D): by Outer Size
2019-05-12 03:32:56 -07:00
// octagon(or=30);
// octagon(od=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Inner Size
2019-05-12 03:32:56 -07:00
// octagon(ir=30);
// octagon(id=60);
2019-05-12 18:10:15 -07:00
// Example(2D): by Side Length
2019-05-12 03:32:56 -07:00
// octagon(side=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Realigned
// octagon(side=20, realign=true);
2020-10-07 16:53:52 -07:00
// Example(2D): Alignment by Tip
// octagon(r=30, align_tip=BACK+RIGHT)
// attach("tip0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Alignment by Side
// octagon(r=30, align_side=BACK+RIGHT)
// attach("side0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
2020-01-14 16:48:20 -08:00
// Example(2D): Rounded
// octagon(od=100, rounding=20, $fn=20);
2019-05-12 18:10:15 -07:00
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, octagon(or=30));
2020-10-07 16:53:52 -07:00
function octagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 ) =
regular_ngon ( n = 8 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) ;
2019-05-12 18:10:15 -07:00
2020-10-07 16:53:52 -07:00
module octagon ( r , d , or , od , ir , id , side , rounding = 0 , realign = false , align_tip , align_side , anchor = CENTER , spin = 0 )
regular_ngon ( n = 8 , r = r , d = d , or = or , od = od , ir = ir , id = id , side = side , rounding = rounding , realign = realign , align_tip = align_tip , align_side = align_side , anchor = anchor , spin = spin ) children ( ) ;
2019-05-12 03:32:56 -07:00
2021-10-30 18:07:43 -04:00
// Function&Module: right_triangle()
2021-10-26 00:42:27 -07:00
// Usage: As Module
// right_triangle(size, [center], ...);
// Usage: With Attachments
// right_triangle(size, [center], ...) { attachments }
// Usage: As Function
// path = right_triangle(size, [center], ...);
// Description:
// Creates a right triangle with the Hypotenuse in the X+Y+ quadrant.
// Arguments:
// size = The width and length of the right triangle, given as a scalar or an XY vector.
// center = If true, forces `anchor=CENTER`. If false, forces `anchor=[-1,-1]`. Default: undef (use `anchor=`)
// ---
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2022-01-07 19:31:58 -08:00
// Example(2D):
2021-10-26 00:42:27 -07:00
// right_triangle([40,30]);
2022-01-07 19:31:58 -08:00
// Example(2D): With `center=true`
2021-10-26 00:42:27 -07:00
// right_triangle([40,30], center=true);
2022-01-07 19:31:58 -08:00
// Example(2D): Anchors
2021-10-26 00:42:27 -07:00
// right_triangle([40,30])
// show_anchors();
function right_triangle ( size = [ 1 , 1 ] , center , anchor , spin = 0 ) =
let (
size = is_num ( size ) ? [ size , size ] : size ,
anchor = get_anchor ( anchor , center , [ - 1 , - 1 ] , [ - 1 , - 1 ] )
)
assert ( is_vector ( size , 2 ) )
let (
path = [ [ size . x / 2 , - size . y / 2 ] , [ - size . x / 2 , - size . y / 2 ] , [ - size . x / 2 , size . y / 2 ] ]
) reorient ( anchor , spin , two_d = true , size = [ size . x , size . y ] , size2 = 0 , shift = - size . x / 2 , p = path ) ;
module right_triangle ( size = [ 1 , 1 ] , center , anchor , spin = 0 ) {
size = is_num ( size ) ? [ size , size ] : size ;
anchor = get_anchor ( anchor , center , [ - 1 , - 1 ] , [ - 1 , - 1 ] ) ;
assert ( is_vector ( size , 2 ) ) ;
path = right_triangle ( size , center = true ) ;
attachable ( anchor , spin , two_d = true , size = [ size . x , size . y ] , size2 = 0 , shift = - size . x / 2 ) {
polygon ( path ) ;
children ( ) ;
}
}
2020-01-05 21:36:27 -08:00
// Function&Module: trapezoid()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// trapezoid(h, w1, w2, [shift=], [rounding=], [chamfer=], ...);
2021-02-25 02:16:55 -08:00
// trapezoid(h, w1, angle=, ...);
// trapezoid(h, w2, angle=, ...);
// trapezoid(w1, w2, angle=, ...);
// Usage: With Attachments
// trapezoid(h, w1, w2, ...) { attachments }
// Usage: As Function
// path = trapezoid(h, w1, w2, ...);
// path = trapezoid(h, w1, angle=, ...);
// path = trapezoid(h, w2=, angle=, ...);
// path = trapezoid(w1=, w2=, angle=, ...);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
// See Also: rect(), square()
2020-01-05 21:36:27 -08:00
// Description:
// When called as a function, returns a 2D path for a trapezoid with parallel front and back sides.
// When called as a module, creates a 2D trapezoid with parallel front and back sides.
// Arguments:
// h = The Y axis height of the trapezoid.
// w1 = The X axis width of the front end of the trapezoid.
// w2 = The X axis width of the back end of the trapezoid.
2021-02-25 02:16:55 -08:00
// ---
2020-10-07 22:50:05 -07:00
// angle = If given in place of `h`, `w1`, or `w2`, then the missing value is calculated such that the right side has that angle away from the Y axis.
// shift = Scalar value to shift the back of the trapezoid along the X axis by. Default: 0
2020-12-30 00:34:25 -08:00
// rounding = The rounding radius for the corners. If given as a list of four numbers, gives individual radii for each corner, in the order [X+Y+,X-Y+,X-Y-,X+Y-]. Default: 0 (no rounding)
// chamfer = The Length of the chamfer faces at the corners. If given as a list of four numbers, gives individual chamfers for each corner, in the order [X+Y+,X-Y+,X-Y-,X+Y-]. Default: 0 (no chamfer)
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-01-05 21:36:27 -08:00
// Examples(2D):
// trapezoid(h=30, w1=40, w2=20);
// trapezoid(h=25, w1=20, w2=35);
// trapezoid(h=20, w1=40, w2=0);
2020-10-07 23:03:09 -07:00
// trapezoid(h=20, w1=30, angle=30);
2020-10-07 22:50:05 -07:00
// trapezoid(h=20, w1=20, angle=-30);
// trapezoid(h=20, w2=10, angle=30);
// trapezoid(h=20, w2=30, angle=-30);
// trapezoid(w1=30, w2=10, angle=30);
2020-12-30 00:34:25 -08:00
// Example(2D): Chamferred Trapezoid
// trapezoid(h=30, w1=60, w2=40, chamfer=5);
// Example(2D): Rounded Trapezoid
// trapezoid(h=30, w1=60, w2=40, rounding=5);
// Example(2D): Mixed Chamfering and Rounding
// trapezoid(h=30, w1=60, w2=40, rounding=[5,0,10,0],chamfer=[0,8,0,15],$fa=1,$fs=1);
2020-01-05 21:36:27 -08:00
// Example(2D): Called as Function
// stroke(closed=true, trapezoid(h=30, w1=40, w2=20));
2020-12-30 00:34:25 -08:00
function trapezoid ( h , w1 , w2 , angle , shift = 0 , chamfer = 0 , rounding = 0 , anchor = CENTER , spin = 0 ) =
2020-10-07 22:50:05 -07:00
assert ( is_undef ( h ) || is_finite ( h ) )
assert ( is_undef ( w1 ) || is_finite ( w1 ) )
assert ( is_undef ( w2 ) || is_finite ( w2 ) )
assert ( is_undef ( angle ) || is_finite ( angle ) )
assert ( num_defined ( [ h , w1 , w2 , angle ] ) = = 3 , "Must give exactly 3 of the arguments h, w1, w2, and angle." )
assert ( is_finite ( shift ) )
2021-06-22 17:20:08 -07:00
assert ( is_finite ( chamfer ) || is_vector ( chamfer , 4 ) )
assert ( is_finite ( rounding ) || is_vector ( rounding , 4 ) )
2020-05-29 19:04:34 -07:00
let (
2021-06-22 17:20:08 -07:00
simple = chamfer = = 0 && rounding = = 0 ,
2020-10-07 22:50:05 -07:00
h = ! is_undef ( h ) ? h : opp_ang_to_adj ( abs ( w2 - w1 ) / 2 , abs ( angle ) ) ,
w1 = ! is_undef ( w1 ) ? w1 : w2 + 2 * ( adj_ang_to_opp ( h , angle ) + shift ) ,
2020-12-30 00:34:25 -08:00
w2 = ! is_undef ( w2 ) ? w2 : w1 - 2 * ( adj_ang_to_opp ( h , angle ) + shift )
2020-10-07 22:50:05 -07:00
)
assert ( w1 >= 0 && w2 >= 0 && h > 0 , "Degenerate trapezoid geometry." )
2020-12-30 00:34:25 -08:00
assert ( w1 + w2 > 0 , "Degenerate trapezoid geometry." )
let (
base_path = [
[ w2 / 2 + shift , h / 2 ] ,
[ - w2 / 2 + shift , h / 2 ] ,
[ - w1 / 2 , - h / 2 ] ,
[ w1 / 2 , - h / 2 ] ,
] ,
2021-06-22 17:20:08 -07:00
cpath = simple ? base_path :
path_chamfer_and_rounding (
base_path , closed = true ,
chamfer = chamfer ,
rounding = rounding
) ,
path = reverse ( cpath )
) simple
? reorient ( anchor , spin , two_d = true , size = [ w1 , h ] , size2 = w2 , shift = shift , p = path )
: reorient ( anchor , spin , two_d = true , path = path , p = path ) ;
2020-12-30 00:34:25 -08:00
module trapezoid ( h , w1 , w2 , angle , shift = 0 , chamfer = 0 , rounding = 0 , anchor = CENTER , spin = 0 ) {
2021-06-22 17:20:08 -07:00
path = trapezoid ( h = h , w1 = w1 , w2 = w2 , angle = angle , shift = shift , chamfer = chamfer , rounding = rounding ) ;
2020-10-07 22:50:05 -07:00
union ( ) {
2021-06-22 17:20:08 -07:00
simple = chamfer = = 0 && rounding = = 0 ;
2020-10-07 22:50:05 -07:00
h = ! is_undef ( h ) ? h : opp_ang_to_adj ( abs ( w2 - w1 ) / 2 , abs ( angle ) ) ;
w1 = ! is_undef ( w1 ) ? w1 : w2 + 2 * ( adj_ang_to_opp ( h , angle ) + shift ) ;
w2 = ! is_undef ( w2 ) ? w2 : w1 - 2 * ( adj_ang_to_opp ( h , angle ) + shift ) ;
2020-12-30 00:34:25 -08:00
if ( simple ) {
attachable ( anchor , spin , two_d = true , size = [ w1 , h ] , size2 = w2 , shift = shift ) {
polygon ( path ) ;
children ( ) ;
}
} else {
attachable ( anchor , spin , two_d = true , path = path ) {
polygon ( path ) ;
children ( ) ;
}
2020-10-07 22:50:05 -07:00
}
2020-05-29 19:04:34 -07:00
}
2020-01-15 14:18:11 -08:00
}
2020-01-05 21:36:27 -08:00
2021-09-06 19:07:18 -04:00
// Function&Module: star()
// Usage: As Module
// star(n, r/or, ir, [realign=], [align_tip=], [align_pit=], ...);
// star(n, r/or, step=, ...);
// Usage: With Attachments
// star(n, r/or, ir, ...) { attachments }
// Usage: As Function
// path = star(n, r/or, ir, [realign=], [align_tip=], [align_pit=], ...);
// path = star(n, r/or, step=, ...);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), ellipse()
2021-09-06 19:07:18 -04:00
// Description:
// When called as a function, returns the path needed to create a star polygon with N points.
// When called as a module, creates a star polygon with N points.
// Arguments:
// n = The number of stellate tips on the star.
// r/or = The radius to the tips of the star.
// ir = The radius to the inner corners of the star.
// ---
// d/od = The diameter to the tips of the star.
// id = The diameter to the inner corners of the star.
// step = Calculates the radius of the inner star corners by virtually drawing a straight line `step` tips around the star. 2 <= step < n/2
2021-11-18 19:02:50 -05:00
// realign = If false, vertex 0 will lie on the X+ axis. If true then the midpoint of the last edge will lie on the X+ axis, and vertex 0 will be below the X axis. Default: false
2021-09-06 19:07:18 -04:00
// align_tip = If given as a 2D vector, rotates the whole shape so that the first star tip points in that direction. This occurs before spin.
// align_pit = If given as a 2D vector, rotates the whole shape so that the first inner corner is pointed towards that direction. This occurs before spin.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2021-11-16 18:49:37 -05:00
// atype = Choose "hull" or "intersect" anchor methods. Default: "hull"
2021-09-06 19:07:18 -04:00
// Extra Anchors:
// "tip0" ... "tip4" = Each tip has an anchor, pointing outwards.
// "pit0" ... "pit4" = The inside corner between each tip has an anchor, pointing outwards.
// "midpt0" ... "midpt4" = The center-point between each pair of tips has an anchor, pointing outwards.
// Examples(2D):
// star(n=5, r=50, ir=25);
// star(n=5, r=50, step=2);
// star(n=7, r=50, step=2);
// star(n=7, r=50, step=3);
// Example(2D): Realigned
// star(n=7, r=50, step=3, realign=true);
// Example(2D): Alignment by Tip
// star(n=5, ir=15, or=30, align_tip=BACK+RIGHT)
// attach("tip0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Alignment by Pit
// star(n=5, ir=15, or=30, align_pit=BACK+RIGHT)
// attach("pit0", FWD) color("blue")
// stroke([[0,0],[0,7]], endcap2="arrow2");
// Example(2D): Called as Function
// stroke(closed=true, star(n=5, r=50, ir=25));
2021-11-16 18:49:37 -05:00
function star ( n , r , ir , d , or , od , id , step , realign = false , align_tip , align_pit , anchor = CENTER , spin = 0 , atype = "hull" , _mat , _anchs ) =
assert ( in_list ( atype , _ANCHOR_TYPES ) , "Anchor type must be \"hull\" or \"intersect\"" )
2021-09-06 19:07:18 -04:00
assert ( is_undef ( align_tip ) || is_vector ( align_tip ) )
assert ( is_undef ( align_pit ) || is_vector ( align_pit ) )
assert ( is_undef ( align_tip ) || is_undef ( align_pit ) , "Can only specify one of align_tip and align_pit" )
2021-09-15 16:30:04 -04:00
assert ( is_def ( n ) , "Must specify number of points, n" )
2021-09-06 19:07:18 -04:00
let (
r = get_radius ( r1 = or , d1 = od , r = r , d = d ) ,
count = num_defined ( [ ir , id , step ] ) ,
stepOK = is_undef ( step ) || ( step > 1 && step < n / 2 )
)
assert ( count = = 1 , "Must specify exactly one of ir, id, step" )
2021-09-14 20:10:55 -04:00
assert ( stepOK , n = = 4 ? "Parameter 'step' not allowed for 4 point stars"
: n = = 5 || n = = 6 ? str ( "Parameter 'step' must be 2 for " , n , " point stars" )
: str ( "Parameter 'step' must be between 2 and " , floor ( n / 2 - 1 / 2 ) , " for " , n , " point stars" ) )
2021-09-06 19:07:18 -04:00
let (
mat = ! is_undef ( _mat ) ? _mat :
2021-12-28 17:05:37 -05:00
( realign ? zrot ( - 180 / n ) : ident ( 4 ) ) * (
! is_undef ( align_tip ) ? rot ( from = RIGHT , to = point2d ( align_tip ) ) :
! is_undef ( align_pit ) ? rot ( from = RIGHT , to = point2d ( align_pit ) ) * zrot ( 180 / n ) :
1
2021-09-06 19:07:18 -04:00
) ,
stepr = is_undef ( step ) ? r : r * cos ( 180 * step / n ) / cos ( 180 * ( step - 1 ) / n ) ,
ir = get_radius ( r = ir , d = id , dflt = stepr ) ,
offset = realign ? 180 / n : 0 ,
path1 = [ for ( i = [ 2 * n : - 1 : 1 ] ) let ( theta = 180 * i / n , radius = ( i % 2 ) ? ir : r ) radius * [ cos ( theta ) , sin ( theta ) ] ] ,
path = apply ( mat , path1 ) ,
anchors = ! is_undef ( _anchs ) ? _anchs :
! is_string ( anchor ) ? [ ] : [
for ( i = [ 0 : 1 : n - 1 ] ) let (
a1 = 360 - i * 360 / n ,
a2 = a1 - 180 / n ,
a3 = a1 - 360 / n ,
p1 = apply ( mat , polar_to_xy ( r , a1 ) ) ,
p2 = apply ( mat , polar_to_xy ( ir , a2 ) ) ,
p3 = apply ( mat , polar_to_xy ( r , a3 ) ) ,
pos = ( p1 + p3 ) / 2
) each [
2021-09-16 18:32:12 -04:00
named_anchor ( str ( "tip" , i ) , p1 , unit ( p1 , BACK ) , 0 ) ,
named_anchor ( str ( "pit" , i ) , p2 , unit ( p2 , BACK ) , 0 ) ,
named_anchor ( str ( "midpt" , i ) , pos , unit ( pos , BACK ) , 0 ) ,
2021-09-06 19:07:18 -04:00
]
]
2021-11-16 18:49:37 -05:00
) reorient ( anchor , spin , two_d = true , path = path , p = path , extent = atype = = "hull" , anchors = anchors ) ;
2021-09-06 19:07:18 -04:00
2021-11-16 18:49:37 -05:00
module star ( n , r , ir , d , or , od , id , step , realign = false , align_tip , align_pit , anchor = CENTER , spin = 0 , atype = "hull" ) {
assert ( in_list ( atype , _ANCHOR_TYPES ) , "Anchor type must be \"hull\" or \"intersect\"" ) ;
2021-09-06 19:07:18 -04:00
assert ( is_undef ( align_tip ) || is_vector ( align_tip ) ) ;
assert ( is_undef ( align_pit ) || is_vector ( align_pit ) ) ;
assert ( is_undef ( align_tip ) || is_undef ( align_pit ) , "Can only specify one of align_tip and align_pit" ) ;
r = get_radius ( r1 = or , d1 = od , r = r , d = d , dflt = undef ) ;
stepr = is_undef ( step ) ? r : r * cos ( 180 * step / n ) / cos ( 180 * ( step - 1 ) / n ) ;
ir = get_radius ( r = ir , d = id , dflt = stepr ) ;
2021-12-28 17:05:37 -05:00
mat = ( realign ? zrot ( - 180 / n ) : ident ( 4 ) ) * (
! is_undef ( align_tip ) ? rot ( from = RIGHT , to = point2d ( align_tip ) ) :
! is_undef ( align_pit ) ? rot ( from = RIGHT , to = point2d ( align_pit ) ) * zrot ( 180 / n ) :
1
2021-09-06 19:07:18 -04:00
) ;
anchors = [
for ( i = [ 0 : 1 : n - 1 ] ) let (
a1 = 360 - i * 360 / n - ( realign ? 180 / n : 0 ) ,
a2 = a1 - 180 / n ,
a3 = a1 - 360 / n ,
p1 = apply ( mat , polar_to_xy ( r , a1 ) ) ,
p2 = apply ( mat , polar_to_xy ( ir , a2 ) ) ,
p3 = apply ( mat , polar_to_xy ( r , a3 ) ) ,
pos = ( p1 + p3 ) / 2
) each [
2021-09-16 18:32:12 -04:00
named_anchor ( str ( "tip" , i ) , p1 , unit ( p1 , BACK ) , 0 ) ,
named_anchor ( str ( "pit" , i ) , p2 , unit ( p2 , BACK ) , 0 ) ,
named_anchor ( str ( "midpt" , i ) , pos , unit ( pos , BACK ) , 0 ) ,
2021-09-06 19:07:18 -04:00
]
] ;
path = star ( n = n , r = r , ir = ir , realign = realign , _mat = mat , _anchs = anchors ) ;
2021-11-16 18:49:37 -05:00
attachable ( anchor , spin , two_d = true , path = path , extent = atype = = "hull" , anchors = anchors ) {
2021-09-06 19:07:18 -04:00
polygon ( path ) ;
children ( ) ;
}
}
2021-09-07 18:47:08 -04:00
2021-09-22 14:59:18 -04:00
/// Internal Function: _path_add_jitter()
/// Topics: Paths
/// See Also: jittered_poly(), subdivide_long_segments()
/// Usage:
/// jpath = _path_add_jitter(path, [dist], [closed=]);
/// Description:
/// Adds tiny jitter offsets to collinear points in the given path so that they
/// are no longer collinear. This is useful for preserving subdivision on long
/// straight segments, when making geometry with `polygon()`, for use with
/// `linear_exrtrude()` with a `twist()`.
/// Arguments:
/// path = The path to add jitter to.
/// dist = The amount to jitter points by. Default: 1/512 (0.00195)
/// ---
/// closed = If true, treat path like a closed polygon. Default: true
/// Example(3D):
/// d = 100; h = 75; quadsize = 5;
/// path = pentagon(d=d);
/// spath = subdivide_long_segments(path, quadsize, closed=true);
/// jpath = _path_add_jitter(spath, closed=true);
/// linear_extrude(height=h, twist=72, slices=h/quadsize)
/// polygon(jpath);
function _path_add_jitter ( path , dist = 1 / 512 , closed = true ) =
assert ( is_path ( path ) )
assert ( is_finite ( dist ) )
assert ( is_bool ( closed ) )
[
path [ 0 ] ,
for ( i = idx ( path , s = 1 , e = closed ? - 1 : - 2 ) ) let (
n = line_normal ( [ path [ i - 1 ] , path [ i ] ] )
) path [ i ] + n * ( is_collinear ( select ( path , i - 1 , i + 1 ) ) ? ( dist * ( ( i % 2 ) * 2 - 1 ) ) : 0 ) ,
if ( ! closed ) last ( path )
] ;
2021-09-07 18:47:08 -04:00
// Module: jittered_poly()
// Topics: Extrusions
2021-09-22 22:41:00 -07:00
// See Also: subdivide_long_segments()
2021-09-07 18:47:08 -04:00
// Usage:
// jittered_poly(path, [dist]);
// Description:
// Creates a 2D polygon shape from the given path in such a way that any extra
// collinear points are not stripped out in the way that `polygon()` normally does.
// This is useful for refining the mesh of a `linear_extrude()` with twist.
// Arguments:
// path = The path to add jitter to.
// dist = The amount to jitter points by. Default: 1/512 (0.00195)
// Example:
// d = 100; h = 75; quadsize = 5;
// path = pentagon(d=d);
// spath = subdivide_long_segments(path, quadsize, closed=true);
// linear_extrude(height=h, twist=72, slices=h/quadsize)
// jittered_poly(spath);
module jittered_poly ( path , dist = 1 / 512 ) {
2021-09-22 14:59:18 -04:00
polygon ( _path_add_jitter ( path , dist , closed = true ) ) ;
2021-09-07 18:47:08 -04:00
}
2021-09-06 19:07:18 -04:00
// Section: Curved 2D Shapes
2020-01-05 21:36:27 -08:00
// Function&Module: teardrop2d()
//
// Description:
2021-11-16 18:49:37 -05:00
// Makes a 2D teardrop shape. Useful for extruding into 3D printable holes. Uses "intersect" style anchoring.
2020-01-05 21:36:27 -08:00
//
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// teardrop2d(r/d=, [ang], [cap_h]);
2021-02-25 02:16:55 -08:00
// Usage: With Attachments
2021-06-26 20:59:33 -07:00
// teardrop2d(r/d=, [ang], [cap_h], ...) { attachments }
2021-02-25 02:16:55 -08:00
// Usage: As Function
2021-06-26 20:59:33 -07:00
// path = teardrop2d(r/d=, [ang], [cap_h]);
2021-02-25 02:16:55 -08:00
//
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
//
// See Also: teardrop(), onion()
2020-01-05 21:36:27 -08:00
//
// Arguments:
// r = radius of circular part of teardrop. (Default: 1)
// ang = angle of hat walls from the Y axis. (Default: 45 degrees)
// cap_h = if given, height above center where the shape will be truncated.
2021-02-25 02:16:55 -08:00
// ---
// d = diameter of spherical portion of bottom. (Use instead of r)
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2020-01-05 21:36:27 -08:00
//
// Example(2D): Typical Shape
// teardrop2d(r=30, ang=30);
// Example(2D): Crop Cap
// teardrop2d(r=30, ang=30, cap_h=40);
// Example(2D): Close Crop
// teardrop2d(r=30, ang=30, cap_h=20);
2021-02-25 02:16:55 -08:00
module teardrop2d ( r , ang = 45 , cap_h , d , anchor = CENTER , spin = 0 )
2020-01-05 21:36:27 -08:00
{
2020-05-29 19:04:34 -07:00
path = teardrop2d ( r = r , d = d , ang = ang , cap_h = cap_h ) ;
2021-11-16 18:49:37 -05:00
attachable ( anchor , spin , two_d = true , path = path , extent = false ) {
2020-05-29 19:04:34 -07:00
polygon ( path ) ;
children ( ) ;
}
2020-01-05 21:36:27 -08:00
}
2021-02-25 02:16:55 -08:00
function teardrop2d ( r , ang = 45 , cap_h , d , anchor = CENTER , spin = 0 ) =
2020-05-29 19:04:34 -07:00
let (
r = get_radius ( r = r , d = d , dflt = 1 ) ,
2020-12-31 16:33:09 -08:00
tanpt = polar_to_xy ( r , ang ) ,
tip_y = adj_ang_to_hyp ( r , 90 - ang ) ,
cap_h = min ( default ( cap_h , tip_y ) , tip_y ) ,
cap_w = tanpt . y >= cap_h
? hyp_opp_to_adj ( r , cap_h )
: adj_ang_to_opp ( tip_y - cap_h , ang ) ,
ang2 = min ( ang , atan2 ( cap_h , cap_w ) ) ,
sa = 180 - ang2 ,
ea = 360 + ang2 ,
2021-08-31 18:13:15 -04:00
steps = ceil ( segs ( r ) * ( ea - sa ) / 360 ) ,
2020-05-29 19:04:34 -07:00
path = deduplicate (
[
2020-12-31 16:33:09 -08:00
[ cap_w , cap_h ] ,
2021-08-31 18:13:15 -04:00
for ( a = lerpn ( ea , sa , steps + 1 ) ) r * [ cos ( a ) , sin ( a ) ] ,
2020-12-31 16:33:09 -08:00
[ - cap_w , cap_h ]
2020-05-29 19:04:34 -07:00
] , closed = true
) ,
2021-10-26 16:45:14 -04:00
maxx_idx = max_index ( column ( path , 0 ) ) ,
2021-11-11 18:50:26 -05:00
path2 = list_rotate ( path , maxx_idx )
2021-11-16 18:49:37 -05:00
) reorient ( anchor , spin , two_d = true , path = path2 , p = path2 , extent = false ) ;
2020-01-05 21:36:27 -08:00
2022-01-07 12:37:47 -05:00
// Function&Module: egg()
// Usage: As Module
// egg(length, r1, r2, R);
// Usage: As Function
// path = egg(length, r1|d2, r2|d2, R|D);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
// See Also: circle(), ellipse(), glued_circles()
// Description:
// Constructs an egg-shaped object by connecting two circles with convex arcs that are tangent to the circles.
// You specify the length of the egg, the radii of the two circles, and the desired arc radius.
// Note that because the side radius, R, is often much larger than the end radii, you may get better
// results using `$fs` and `$fa` to control the number of semgments rather than using `$fn`.
// This shape may be useful for creating a cam.
// Arguments:
// length = length of the egg
// r1 = radius of the left-hand circle
// r2 = radius of the right-hand circle
// R = radius of the joining arcs
// ---
// d1 = diameter of the left-hand circle
// d2 = diameter of the right-hand circle
// D = diameter of the joining arcs
// Extra Anchors:
// "left" = center of the left circle
// "right" = center of the right circle
// Example(2D,NoAxes): This first example shows how the egg is constructed from two circles and two joining arcs.
// $fn=100;
// color("red")stroke(egg(78,25,12, 60),closed=true);
// stroke([left(14,circle(25)),
// right(27,circle(12))]);
2022-01-09 00:53:54 -08:00
// Example(2D,Anim,VPD=250,VPR=[0,0,0]): Varying length between circles
// r1 = 25; r2 = 12; R = 65;
// length = floor(lookup($t, [[0,55], [0.5,90], [1,55]]));
// egg(length,r1,r2,R,$fn=180);
// color("black") text(str("length=",length), size=8, halign="center", valign="center");
// Example(2D,Anim,VPD=250,VPR=[0,0,0]): Varying tangent arc radius R
// length = 78; r1 = 25; r2 = 12;
// R = floor(lookup($t, [[0,45], [0.5,150], [1,45]]));
// egg(length,r1,r2,R,$fn=180);
// color("black") text(str("R=",R), size=8, halign="center", valign="center");
// Example(2D,Anim,VPD=250,VPR=[0,0,0]): Varying circle radius r2
// length = 78; r1 = 25; R = 65;
// r2 = floor(lookup($t, [[0,5], [0.5,30], [1,5]]));
// egg(length,r1,r2,R,$fn=180);
// color("black") text(str("r2=",r2), size=8, halign="center", valign="center");
2022-01-07 12:37:47 -05:00
function egg ( length , r1 , r2 , R , d1 , d2 , D , anchor = CENTER , spin = 0 ) =
let (
r1 = get_radius ( r1 = r1 , d1 = d1 ) ,
r2 = get_radius ( r1 = r2 , d1 = d2 ) ,
D = get_radius ( r1 = R , d1 = D )
)
assert ( length > 0 )
assert ( R > length / 2 , "Side radius R must be larger than length/2" )
assert ( length > r1 + r2 , "Length must be longer than 2*(r1+r2)" )
assert ( length > 2 * r2 , "Length must be longer than 2*r2" )
assert ( length > 2 * r1 , "Length must be longer than 2*r1" )
let (
c1 = [ - length / 2 + r1 , 0 ] ,
c2 = [ length / 2 - r2 , 0 ] ,
Rmin = ( r1 + r2 + norm ( c1 - c2 ) ) / 2 ,
Mlist = circle_circle_intersection ( c1 , R - r1 , c2 , R - r2 ) ,
arcparms = reverse ( [ for ( M = Mlist ) [ M , c1 + r1 * unit ( c1 - M ) , c2 + r2 * unit ( c2 - M ) ] ] ) ,
path = concat (
arc ( r = r2 , cp = c2 , points = [ [ length / 2 , 0 ] , arcparms [ 0 ] [ 2 ] ] , endpoint = false ) ,
arc ( r = R , cp = arcparms [ 0 ] [ 0 ] , points = select ( arcparms [ 0 ] , [ 2 , 1 ] ) , endpoint = false ) ,
arc ( r = r1 , points = [ arcparms [ 0 ] [ 1 ] , [ - length / 2 , 0 ] , arcparms [ 1 ] [ 1 ] ] , endpoint = false ) ,
arc ( r = R , cp = arcparms [ 1 ] [ 0 ] , points = select ( arcparms [ 1 ] , [ 1 , 2 ] ) , endpoint = false ) ,
arc ( r = r2 , cp = c2 , points = [ arcparms [ 1 ] [ 2 ] , [ length / 2 , 0 ] ] , endpoint = false )
) ,
anchors = [ named_anchor ( "left" , c1 , BACK , 0 ) ,
named_anchor ( "right" , c2 , BACK , 0 ) ]
)
reorient ( anchor , spin , two_d = true , path = path , extent = true , p = path , anchors = anchors ) ;
module egg ( length , r1 , r2 , R , d1 , d2 , D , anchor = CENTER , spin = 0 )
{
path = egg ( length , r1 , r2 , R , d1 , d2 , D ) ;
anchors = [ named_anchor ( "left" , [ - length / 2 + r1 , 0 ] , BACK , 0 ) ,
named_anchor ( "right" , [ length / 2 - r2 , 0 ] , BACK , 0 ) ] ;
attachable ( anchor , spin , two_d = true , path = path , extent = true , anchors = anchors ) {
polygon ( path ) ;
children ( ) ;
}
}
2019-05-12 18:10:15 -07:00
// Function&Module: glued_circles()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// glued_circles(r/d=, [spread=], [tangent=], ...);
2021-02-25 02:16:55 -08:00
// Usage: With Attachments
2021-06-26 20:59:33 -07:00
// glued_circles(r/d=, [spread=], [tangent=], ...) { attachments }
2021-02-25 02:16:55 -08:00
// Usage: As Function
2021-06-26 20:59:33 -07:00
// path = glued_circles(r/d=, [spread=], [tangent=], ...);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2022-01-07 12:37:47 -05:00
// See Also: circle(), ellipse(), egg()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path forming a shape of two circles joined by curved waist.
2021-11-16 18:49:37 -05:00
// When called as a module, creates a 2D shape of two circles joined by curved waist. Uses "hull" style anchoring.
2019-05-12 03:32:56 -07:00
// Arguments:
// r = The radius of the end circles.
2020-12-27 21:55:36 -08:00
// spread = The distance between the centers of the end circles. Default: 10
// tangent = The angle in degrees of the tangent point for the joining arcs, measured away from the Y axis. Default: 30
2021-02-25 02:16:55 -08:00
// ---
// d = The diameter of the end circles.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2019-05-12 03:32:56 -07:00
// Examples(2D):
2019-05-12 18:10:15 -07:00
// glued_circles(r=15, spread=40, tangent=45);
// glued_circles(d=30, spread=30, tangent=30);
// glued_circles(d=30, spread=30, tangent=15);
// glued_circles(d=30, spread=30, tangent=-30);
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, glued_circles(r=15, spread=40, tangent=45));
2021-02-25 02:16:55 -08:00
function glued_circles ( r , spread = 10 , tangent = 30 , d , anchor = CENTER , spin = 0 ) =
2020-05-29 19:04:34 -07:00
let (
r = get_radius ( r = r , d = d , dflt = 10 ) ,
r2 = ( spread / 2 / sin ( tangent ) ) - r ,
cp1 = [ spread / 2 , 0 ] ,
cp2 = [ 0 , ( r + r2 ) * cos ( tangent ) ] ,
sa1 = 90 - tangent ,
ea1 = 270 + tangent ,
lobearc = ea1 - sa1 ,
2021-08-31 18:13:15 -04:00
lobesegs = ceil ( segs ( r ) * lobearc / 360 ) ,
2020-05-29 19:04:34 -07:00
sa2 = 270 - tangent ,
ea2 = 270 + tangent ,
subarc = ea2 - sa2 ,
arcsegs = ceil ( segs ( r2 ) * abs ( subarc ) / 360 ) ,
2021-11-11 18:50:26 -05:00
// In the tangent zero case the inner curves are missing so we need to complete the two
// outer curves. In the other case the inner curves are present and endpoint=false
// prevents point duplication.
path = tangent = = 0 ?
concat ( arc ( N = lobesegs + 1 , r = r , cp = - cp1 , angle = [ sa1 , ea1 ] ) ,
arc ( N = lobesegs + 1 , r = r , cp = cp1 , angle = [ sa1 + 180 , ea1 + 180 ] ) )
:
concat ( arc ( N = lobesegs , r = r , cp = - cp1 , angle = [ sa1 , ea1 ] , endpoint = false ) ,
[ for ( theta = lerpn ( ea2 + 180 , ea2 - subarc + 180 , arcsegs , endpoint = false ) ) r2 * [ cos ( theta ) , sin ( theta ) ] - cp2 ] ,
arc ( N = lobesegs , r = r , cp = cp1 , angle = [ sa1 + 180 , ea1 + 180 ] , endpoint = false ) ,
[ for ( theta = lerpn ( ea2 , ea2 - subarc , arcsegs , endpoint = false ) ) r2 * [ cos ( theta ) , sin ( theta ) ] + cp2 ] ) ,
2021-10-26 16:45:14 -04:00
maxx_idx = max_index ( column ( path , 0 ) ) ,
2021-11-11 18:50:26 -05:00
path2 = reverse_polygon ( list_rotate ( path , maxx_idx ) )
2020-05-29 19:04:34 -07:00
) reorient ( anchor , spin , two_d = true , path = path2 , extent = true , p = path2 ) ;
2019-05-12 03:32:56 -07:00
2021-02-25 02:16:55 -08:00
module glued_circles ( r , spread = 10 , tangent = 30 , d , anchor = CENTER , spin = 0 ) {
2020-05-29 19:04:34 -07:00
path = glued_circles ( r = r , d = d , spread = spread , tangent = tangent ) ;
attachable ( anchor , spin , two_d = true , path = path , extent = true ) {
polygon ( path ) ;
children ( ) ;
}
2020-02-29 13:16:15 -08:00
}
2019-05-12 03:32:56 -07:00
function _superformula ( theta , m1 , m2 , n1 , n2 = 1 , n3 = 1 , a = 1 , b = 1 ) =
2020-05-29 19:04:34 -07:00
pow ( pow ( abs ( cos ( m1 * theta / 4 ) / a ) , n2 ) + pow ( abs ( sin ( m2 * theta / 4 ) / b ) , n3 ) , - 1 / n1 ) ;
2019-05-12 03:32:56 -07:00
2019-06-11 21:56:10 -04:00
// Function&Module: supershape()
2021-02-25 02:16:55 -08:00
// Usage: As Module
2021-06-26 20:59:33 -07:00
// supershape(step, [m1=], [m2=], [n1=], [n2=], [n3=], [a=], [b=], <r=/d=>);
2021-02-25 02:16:55 -08:00
// Usage: With Attachments
2021-06-26 20:59:33 -07:00
// supershape(step, [m1=], [m2=], [n1=], [n2=], [n3=], [a=], [b=], <r=/d=>) { attachments }
2021-02-25 02:16:55 -08:00
// Usage: As Function
2021-06-26 20:59:33 -07:00
// path = supershape(step, [m1=], [m2=], [n1=], [n2=], [n3=], [a=], [b=], <r=/d=>);
2021-02-25 02:16:55 -08:00
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
2021-11-13 19:45:58 -05:00
// See Also: circle(), ellipse()
2019-05-12 03:32:56 -07:00
// Description:
2019-05-12 18:10:15 -07:00
// When called as a function, returns a 2D path for the outline of the [Superformula](https://en.wikipedia.org/wiki/Superformula) shape.
// When called as a module, creates a 2D [Superformula](https://en.wikipedia.org/wiki/Superformula) shape.
2021-11-16 18:49:37 -05:00
// Note that the "hull" type anchoring (the default) is more intuitive for concave star-like shapes, but the anchor points do not
// necesarily lie on the line of the anchor vector, which can be confusing, especially for simpler, ellipse-like shapes.
2019-05-12 03:32:56 -07:00
// Arguments:
// step = The angle step size for sampling the superformula shape. Smaller steps are slower but more accurate.
2019-07-14 12:30:46 -07:00
// m1 = The m1 argument for the superformula. Default: 4.
2019-06-11 21:56:10 -04:00
// m2 = The m2 argument for the superformula. Default: m1.
// n1 = The n1 argument for the superformula. Default: 1.
// n2 = The n2 argument for the superformula. Default: n1.
// n3 = The n3 argument for the superformula. Default: n2.
// a = The a argument for the superformula. Default: 1.
// b = The b argument for the superformula. Default: a.
// r = Radius of the shape. Scale shape to fit in a circle of radius r.
2021-02-25 02:16:55 -08:00
// ---
2019-06-11 21:56:10 -04:00
// d = Diameter of the shape. Scale shape to fit in a circle of diameter d.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2021-11-16 18:49:37 -05:00
// atype = Select "hull" or "intersect" style anchoring. Default: "hull".
2019-05-12 03:32:56 -07:00
// Example(2D):
2019-06-11 21:56:10 -04:00
// supershape(step=0.5,m1=16,m2=16,n1=0.5,n2=0.5,n3=16,r=50);
2019-05-12 18:10:15 -07:00
// Example(2D): Called as Function
2019-07-12 13:11:13 -07:00
// stroke(closed=true, supershape(step=0.5,m1=16,m2=16,n1=0.5,n2=0.5,n3=16,d=100));
2019-06-18 01:08:24 -07:00
// Examples(2D,Med):
2019-06-11 21:56:10 -04:00
// for(n=[2:5]) right(2.5*(n-2)) supershape(m1=4,m2=4,n1=n,a=1,b=2); // Superellipses
// m=[2,3,5,7]; for(i=[0:3]) right(2.5*i) supershape(.5,m1=m[i],n1=1);
// m=[6,8,10,12]; for(i=[0:3]) right(2.7*i) supershape(.5,m1=m[i],n1=1,b=1.5); // m should be even
// m=[1,2,3,5]; for(i=[0:3]) fwd(1.5*i) supershape(m1=m[i],n1=0.4);
// supershape(m1=5, n1=4, n2=1); right(2.5) supershape(m1=5, n1=40, n2=10);
// m=[2,3,5,7]; for(i=[0:3]) right(2.5*i) supershape(m1=m[i], n1=60, n2=55, n3=30);
// n=[0.5,0.2,0.1,0.02]; for(i=[0:3]) right(2.5*i) supershape(m1=5,n1=n[i], n2=1.7);
// supershape(m1=2, n1=1, n2=4, n3=8);
// supershape(m1=7, n1=2, n2=8, n3=4);
// supershape(m1=7, n1=3, n2=4, n3=17);
// supershape(m1=4, n1=1/2, n2=1/2, n3=4);
// supershape(m1=4, n1=4.0,n2=16, n3=1.5, a=0.9, b=9);
2019-07-14 12:30:46 -07:00
// for(i=[1:4]) right(3*i) supershape(m1=i, m2=3*i, n1=2);
2019-06-11 21:56:10 -04:00
// m=[4,6,10]; for(i=[0:2]) right(i*5) supershape(m1=m[i], n1=12, n2=8, n3=5, a=2.7);
2019-06-23 17:21:00 -04:00
// for(i=[-1.5:3:1.5]) right(i*1.5) supershape(m1=2,m2=10,n1=i,n2=1);
// for(i=[1:3],j=[-1,1]) translate([3.5*i,1.5*j])supershape(m1=4,m2=6,n1=i*j,n2=1);
// for(i=[1:3]) right(2.5*i)supershape(step=.5,m1=88, m2=64, n1=-i*i,n2=1,r=1);
2019-12-06 19:27:49 -05:00
// Examples:
// linear_extrude(height=0.3, scale=0) supershape(step=1, m1=6, n1=0.4, n2=0, n3=6);
// linear_extrude(height=5, scale=0) supershape(step=1, b=3, m1=6, n1=3.8, n2=16, n3=10);
2021-11-16 18:49:37 -05:00
function supershape ( step = 0.5 , m1 = 4 , m2 , n1 = 1 , n2 , n3 , a = 1 , b , r , d , anchor = CENTER , spin = 0 , atype = "hull" ) =
2021-11-16 20:33:51 -05:00
assert ( in_list ( atype , _ANCHOR_TYPES ) , "Anchor type must be \"hull\" or \"intersect\"" )
2020-05-29 19:04:34 -07:00
let (
2021-11-16 20:33:51 -05:00
2020-05-29 19:04:34 -07:00
r = get_radius ( r = r , d = d , dflt = undef ) ,
m2 = is_def ( m2 ) ? m2 : m1 ,
n2 = is_def ( n2 ) ? n2 : n1 ,
n3 = is_def ( n3 ) ? n3 : n2 ,
b = is_def ( b ) ? b : a ,
steps = ceil ( 360 / step ) ,
step = 360 / steps ,
angs = [ for ( i = [ 0 : steps ] ) step * i ] ,
rads = [ for ( theta = angs ) _superformula ( theta = theta , m1 = m1 , m2 = m2 , n1 = n1 , n2 = n2 , n3 = n3 , a = a , b = b ) ] ,
scale = is_def ( r ) ? r / max ( rads ) : 1 ,
path = [ for ( i = [ steps : - 1 : 1 ] ) let ( a = angs [ i ] ) scale * rads [ i ] * [ cos ( a ) , sin ( a ) ] ]
2021-11-16 18:49:37 -05:00
) reorient ( anchor , spin , two_d = true , path = path , p = path , extent = atype = = "hull" ) ;
2019-05-12 03:32:56 -07:00
2021-11-16 22:13:00 -05:00
module supershape ( step = 0.5 , m1 = 4 , m2 = undef , n1 , n2 = undef , n3 = undef , a = 1 , b = undef , r = undef , d = undef , anchor = CENTER , spin = 0 , atype = "hull" ) {
assert ( in_list ( atype , _ANCHOR_TYPES ) , "Anchor type must be \"hull\" or \"intersect\"" ) ;
2020-05-29 19:04:34 -07:00
path = supershape ( step = step , m1 = m1 , m2 = m2 , n1 = n1 , n2 = n2 , n3 = n3 , a = a , b = b , r = r , d = d ) ;
2021-11-16 18:49:37 -05:00
attachable ( anchor , spin , extent = atype = = "hull" , two_d = true , path = path ) {
2020-05-29 19:04:34 -07:00
polygon ( path ) ;
children ( ) ;
}
2020-02-29 13:16:15 -08:00
}
2019-05-12 03:32:56 -07:00
2019-06-18 01:08:24 -07:00
2021-01-11 01:57:51 -08:00
// Function&Module: reuleaux_polygon()
// Usage: As Module
2021-02-25 02:16:55 -08:00
// reuleaux_polygon(N, r|d, ...);
2021-01-12 00:09:39 -08:00
// Usage: As Function
2021-02-25 02:16:55 -08:00
// path = reuleaux_polygon(N, r|d, ...);
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable
// See Also: regular_ngon(), pentagon(), hexagon(), octagon()
2021-01-11 01:57:51 -08:00
// Description:
2021-11-16 18:49:37 -05:00
// Creates a 2D Reuleaux Polygon; a constant width shape that is not circular. Uses "intersect" type anchoring.
2021-01-11 01:57:51 -08:00
// Arguments:
// N = Number of "sides" to the Reuleaux Polygon. Must be an odd positive number. Default: 3
// r = Radius of the shape. Scale shape to fit in a circle of radius r.
2021-02-25 02:16:55 -08:00
// ---
2021-01-11 01:57:51 -08:00
// d = Diameter of the shape. Scale shape to fit in a circle of diameter d.
2021-11-19 22:33:16 -05:00
// anchor = Translate so anchor point is at origin (0,0,0). See [anchor](attachments.scad#subsection-anchor). Default: `CENTER`
// spin = Rotate this many degrees around the Z axis after anchor. See [spin](attachments.scad#subsection-spin). Default: `0`
2021-01-11 01:57:51 -08:00
// Extra Anchors:
// "tip0", "tip1", etc. = Each tip has an anchor, pointing outwards.
// Examples(2D):
// reuleaux_polygon(N=3, r=50);
// reuleaux_polygon(N=5, d=100);
2021-01-12 00:01:03 -08:00
// Examples(2D): Standard vector anchors are based on extents
// reuleaux_polygon(N=3, d=50) show_anchors(custom=false);
// Examples(2D): Named anchors exist for the tips
// reuleaux_polygon(N=3, d=50) show_anchors(std=false);
2021-01-11 01:57:51 -08:00
module reuleaux_polygon ( N = 3 , r , d , anchor = CENTER , spin = 0 ) {
assert ( N >= 3 && ( N % 2 ) = = 1 ) ;
r = get_radius ( r = r , d = d , dflt = 1 ) ;
path = reuleaux_polygon ( N = N , r = r ) ;
anchors = [
for ( i = [ 0 : 1 : N - 1 ] ) let (
2021-01-12 00:01:03 -08:00
ca = 360 - i * 360 / N ,
2021-01-11 01:57:51 -08:00
cp = polar_to_xy ( r , ca )
2021-09-16 18:32:12 -04:00
) named_anchor ( str ( "tip" , i ) , cp , unit ( cp , BACK ) , 0 ) ,
2021-01-11 01:57:51 -08:00
] ;
2021-11-16 18:49:37 -05:00
attachable ( anchor , spin , two_d = true , path = path , extent = false , anchors = anchors ) {
2021-01-11 01:57:51 -08:00
polygon ( path ) ;
children ( ) ;
}
}
function reuleaux_polygon ( N = 3 , r , d , anchor = CENTER , spin = 0 ) =
assert ( N >= 3 && ( N % 2 ) = = 1 )
let (
r = get_radius ( r = r , d = d , dflt = 1 ) ,
ssegs = max ( 3 , ceil ( segs ( r ) / N ) ) ,
slen = norm ( polar_to_xy ( r , 0 ) - polar_to_xy ( r , 180 - 180 / N ) ) ,
path = [
for ( i = [ 0 : 1 : N - 1 ] ) let (
ca = 180 - ( i + 0.5 ) * 360 / N ,
sa = ca + 180 + ( 90 / N ) ,
ea = ca + 180 - ( 90 / N ) ,
cp = polar_to_xy ( r , ca )
2021-04-02 16:59:29 -04:00
) each arc ( N = ssegs - 1 , r = slen , cp = cp , angle = [ sa , ea ] , endpoint = false )
2021-01-11 01:57:51 -08:00
] ,
anchors = [
for ( i = [ 0 : 1 : N - 1 ] ) let (
2021-01-12 00:01:03 -08:00
ca = 360 - i * 360 / N ,
2021-01-11 01:57:51 -08:00
cp = polar_to_xy ( r , ca )
2021-09-16 18:32:12 -04:00
) named_anchor ( str ( "tip" , i ) , cp , unit ( cp , BACK ) , 0 ) ,
2021-01-11 01:57:51 -08:00
]
2021-11-16 18:49:37 -05:00
) reorient ( anchor , spin , two_d = true , path = path , extent = false , anchors = anchors , p = path ) ;
2021-01-11 01:57:51 -08:00
2020-05-29 19:04:34 -07:00
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap