mirror of
https://github.com/BelfrySCAD/BOSL2.git
synced 2025-01-22 12:29:36 +00:00
617 lines
27 KiB
OpenSCAD
617 lines
27 KiB
OpenSCAD
//////////////////////////////////////////////////////////////////////
|
|
// LibFile: masks2d.scad
|
|
// This file provides 2D masking shapes that you can use with {{edge_profile()}} to mask edges.
|
|
// The shapes include the simple roundover and chamfer as well as more elaborate shapes
|
|
// like the cove and ogee found in furniture and architecture. You can make the masks
|
|
// as geometry or as 2D paths.
|
|
// Includes:
|
|
// include <BOSL2/std.scad>
|
|
// FileGroup: Basic Modeling
|
|
// FileSummary: 2D masking shapes for edge profiling: including roundover, cove, teardrop, ogee.
|
|
// FileFootnotes: STD=Included in std.scad
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
// Section: 2D Masking Shapes
|
|
|
|
// Function&Module: mask2d_roundover()
|
|
// Synopsis: Creates a 2D beading mask shape useful for rounding edges.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D)
|
|
// See Also: corner_profile(), edge_profile(), face_profile(), fillet()
|
|
// Usage: As module
|
|
// mask2d_roundover(r|d=, [inset], [mask_angle], [excess]) [ATTACHMENTS];
|
|
// Usage: As function
|
|
// path = mask2d_roundover(r|d=, [inset], [mask_angle], [excess]);
|
|
// Description:
|
|
// Creates a 2D roundover/bead mask shape that is useful for extruding into a 3D mask for an edge.
|
|
// Conversely, you can use that same extruded shape to make an interior fillet between two walls.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// Arguments:
|
|
// r = Radius of the roundover.
|
|
// inset = Optional bead inset size. Default: 0
|
|
// mask_angle = Number of degrees in the corner angle to mask. Default: 90
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// d = Diameter of the roundover.
|
|
// 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`
|
|
// Example(2D): 2D Roundover Mask
|
|
// mask2d_roundover(r=10);
|
|
// Example(2D): 2D Bead Mask
|
|
// mask2d_roundover(r=10,inset=2);
|
|
// Example(2D): 2D Bead Mask for a Non-Right Edge.
|
|
// mask2d_roundover(r=10, inset=2, mask_angle=75);
|
|
// Example(2D): Increasing the Excess
|
|
// mask2d_roundover(r=10, inset=2, mask_angle=75, excess=2);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile([TOP,"Z"],except=[BACK,TOP+LEFT])
|
|
// mask2d_roundover(r=10, inset=2);
|
|
// Example: Making an interior fillet
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_roundover(r=10);
|
|
module mask2d_roundover(r, inset=0, mask_angle=90, excess=0.01, d, anchor=CENTER,spin=0) {
|
|
path = mask2d_roundover(r=r, d=d, inset=inset, mask_angle=mask_angle, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_roundover(r, inset=0, mask_angle=90, excess=0.01, d, anchor=CENTER, spin=0) =
|
|
assert(is_finite(r)||is_finite(d))
|
|
assert(is_finite(excess))
|
|
assert(is_finite(mask_angle) && mask_angle>0 && mask_angle<180)
|
|
assert(is_finite(inset)||(is_vector(inset)&&len(inset)==2))
|
|
let(
|
|
inset = is_list(inset)? inset : [inset,inset],
|
|
r = get_radius(r=r,d=d,dflt=1),
|
|
avec = polar_to_xy(inset.x,mask_angle-90),
|
|
line1 = [[0,inset.y], [100,inset.y]],
|
|
line2 = [avec, polar_to_xy(100,mask_angle)+avec],
|
|
corner = line_intersection(line1,line2),
|
|
arcpts = arc(r=r, corner=[line2.y, corner, line1.y]),
|
|
ipath = [
|
|
arcpts[0] + polar_to_xy(inset.x+excess, mask_angle+90),
|
|
each arcpts,
|
|
last(arcpts) + polar_to_xy(inset.y+excess, -90),
|
|
[0,-excess],
|
|
[-excess,-excess],
|
|
[-excess,0]
|
|
],
|
|
path = deduplicate(ipath)
|
|
) reorient(anchor,spin, two_d=true, path=path, extent=false, p=path);
|
|
|
|
|
|
// Function&Module: mask2d_cove()
|
|
// Synopsis: Creates a 2D cove (quarter-round) mask shape.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D)
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As module
|
|
// mask2d_cove(r|d=, [inset], [mask_angle], [excess]) [ATTACHMENTS];
|
|
// Usage: As function
|
|
// path = mask2d_cove(r|d=, [inset], [mask_angle], [excess]);
|
|
// Description:
|
|
// Creates a 2D cove mask shape that is useful for extruding into a 3D mask for an edge.
|
|
// Conversely, you can use that same extruded shape to make an interior rounded shelf decoration between two walls.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// Arguments:
|
|
// r = Radius of the cove.
|
|
// inset = Optional amount to inset code from corner. Default: 0
|
|
// mask_angle = Number of degrees in the corner angle to mask. Default: 90
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// d = Diameter of the cove.
|
|
// 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`
|
|
// Example(2D): 2D Cove Mask
|
|
// mask2d_cove(r=10);
|
|
// Example(2D): 2D Inset Cove Mask
|
|
// mask2d_cove(r=10,inset=3);
|
|
// Example(2D): 2D Inset Cove Mask for a Non-Right Edge
|
|
// mask2d_cove(r=10,inset=3,mask_angle=75);
|
|
// Example(2D): Increasing the Excess
|
|
// mask2d_cove(r=10,inset=3,mask_angle=75, excess=2);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile([TOP,"Z"],except=[BACK,TOP+LEFT])
|
|
// mask2d_cove(r=10, inset=2);
|
|
// Example: Making an interior rounded shelf
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_cove(r=5, inset=5);
|
|
module mask2d_cove(r, inset=0, mask_angle=90, excess=0.01, d, anchor=CENTER, spin=0) {
|
|
path = mask2d_cove(r=r, d=d, inset=inset, mask_angle=mask_angle, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_cove(r, inset=0, mask_angle=90, excess=0.01, d, anchor=CENTER, spin=0) =
|
|
assert(is_finite(r)||is_finite(d))
|
|
assert(is_finite(mask_angle) && mask_angle>0 && mask_angle<180)
|
|
assert(is_finite(excess))
|
|
assert(is_finite(inset)||(is_vector(inset)&&len(inset)==2))
|
|
let(
|
|
inset = is_list(inset)? inset : [inset,inset],
|
|
r = get_radius(r=r,d=d,dflt=1),
|
|
avec = polar_to_xy(inset.x,mask_angle-90),
|
|
line1 = [[0,inset.y], [100,inset.y]],
|
|
line2 = [avec, polar_to_xy(100,mask_angle)+avec],
|
|
corner = line_intersection(line1,line2),
|
|
arcpts = arc(r=r, cp=corner, start=mask_angle, angle=-mask_angle),
|
|
ipath = [
|
|
arcpts[0] + polar_to_xy(inset.x+excess, mask_angle+90),
|
|
each arcpts,
|
|
last(arcpts) + polar_to_xy(inset.y+excess, -90),
|
|
[0,-excess],
|
|
[-excess,-excess],
|
|
[-excess,0]
|
|
],
|
|
path = deduplicate(ipath)
|
|
) reorient(anchor,spin, two_d=true, path=path, p=path);
|
|
|
|
|
|
// Function&Module: mask2d_chamfer()
|
|
// Synopsis: Produces a 2D chamfer mask shape.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D)
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As Module
|
|
// mask2d_chamfer(edge, [angle], [inset], [excess]) [ATTACHMENTS];
|
|
// mask2d_chamfer(y=, [angle=], [inset=], [excess=]) [ATTACHMENTS];
|
|
// mask2d_chamfer(x=, [angle=], [inset=], [excess=]) [ATTACHMENTS];
|
|
// Usage: As Function
|
|
// path = mask2d_chamfer(edge, [angle], [inset], [excess]);
|
|
// path = mask2d_chamfer(y=, [angle=], [inset=], [excess=]);
|
|
// path = mask2d_chamfer(x=, [angle=], [inset=], [excess=]);
|
|
// Description:
|
|
// Creates a 2D chamfer mask shape that is useful for extruding into a 3D mask for a 90° edge.
|
|
// Conversely, you can use that same extruded shape to make an interior chamfer between two walls at a 90º angle.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// The edge parameter specifies the length of the chamfer's slanted edge. Alternatively you can give x or y to
|
|
// specify the width or height. Only one of x, y, or width is permitted.
|
|
// Arguments:
|
|
// edge = The length of the edge of the chamfer.
|
|
// angle = The angle of the chamfer edge, away from vertical. Default: 45.
|
|
// inset = Optional amount to inset code from corner. Default: 0
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// x = The width of the chamfer.
|
|
// y = The height of the chamfer.
|
|
// 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`
|
|
// Example(2D): 2D Chamfer Mask
|
|
// mask2d_chamfer(x=10);
|
|
// Example(2D): 2D Chamfer Mask by Width.
|
|
// mask2d_chamfer(x=10, angle=30);
|
|
// Example(2D): 2D Chamfer Mask by Height.
|
|
// mask2d_chamfer(y=10, angle=30);
|
|
// Example(2D): 2D Inset Chamfer Mask
|
|
// mask2d_chamfer(x=10, inset=2);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile([TOP,"Z"],except=[BACK,TOP+LEFT])
|
|
// mask2d_chamfer(x=10, inset=2);
|
|
// Example: Making an interior chamfer
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_chamfer(edge=10);
|
|
module mask2d_chamfer(edge, angle=45, inset=0, excess=0.01, x, y, anchor=CENTER,spin=0) {
|
|
path = mask2d_chamfer(x=x, y=y, edge=edge, angle=angle, excess=excess, inset=inset);
|
|
attachable(anchor,spin, two_d=true, path=path, extent=true) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_chamfer(edge, angle=45, inset=0, excess=0.01, x, y, anchor=CENTER,spin=0) =
|
|
let(dummy=one_defined([x,y,edge],["x","y","edge"]))
|
|
assert(is_finite(angle))
|
|
assert(is_finite(excess))
|
|
assert(is_finite(inset)||(is_vector(inset)&&len(inset)==2))
|
|
let(
|
|
inset = is_list(inset)? inset : [inset,inset],
|
|
x = is_def(x)? x :
|
|
is_def(y)? adj_ang_to_opp(adj=y,ang=angle) :
|
|
hyp_ang_to_opp(hyp=edge,ang=angle),
|
|
y = opp_ang_to_adj(opp=x,ang=angle),
|
|
path = [
|
|
[x+inset.x, -excess],
|
|
[-excess, -excess],
|
|
[-excess, y+inset.y],
|
|
[inset.x, y+inset.y],
|
|
[x+inset.x, inset.y]
|
|
]
|
|
) reorient(anchor,spin, two_d=true, path=path, extent=true, p=path);
|
|
|
|
|
|
// Function&Module: mask2d_rabbet()
|
|
// Synopsis: Creates a rabbet mask shape.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D)
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As Module
|
|
// mask2d_rabbet(size, [mask_angle], [excess]) [ATTACHMENTS];
|
|
// Usage: As Function
|
|
// path = mask2d_rabbet(size, [mask_angle], [excess]);
|
|
// Description:
|
|
// Creates a 2D rabbet mask shape that is useful for extruding into a 3D mask for an edge.
|
|
// Conversely, you can use that same extruded shape to make an interior shelf decoration between two walls.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// Arguments:
|
|
// size = The size of the rabbet, either as a scalar or an [X,Y] list.
|
|
// mask_angle = Number of degrees in the corner angle to mask. Default: 90
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// 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`
|
|
// Example(2D): 2D Rabbet Mask
|
|
// mask2d_rabbet(size=10);
|
|
// Example(2D): 2D Asymmetrical Rabbet Mask
|
|
// mask2d_rabbet(size=[5,10]);
|
|
// Example(2D): 2D Mask for a Non-Right Edge
|
|
// mask2d_rabbet(size=10,mask_angle=75);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile([TOP,"Z"],except=[BACK,TOP+LEFT])
|
|
// mask2d_rabbet(size=10);
|
|
// Example: Making an interior shelf
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_rabbet(size=[5,10]);
|
|
module mask2d_rabbet(size, mask_angle=90, excess=0.01, anchor=CTR, spin=0) {
|
|
path = mask2d_rabbet(size=size, mask_angle=mask_angle, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path, extent=false) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_rabbet(size, mask_angle=90, excess=0.01, anchor=CTR, spin=0) =
|
|
assert(is_finite(size)||(is_vector(size)&&len(size)==2))
|
|
assert(is_finite(mask_angle) && mask_angle>0 && mask_angle<180)
|
|
assert(is_finite(excess))
|
|
let(
|
|
size = is_list(size)? size : [size,size],
|
|
avec = polar_to_xy(size.x,mask_angle-90),
|
|
line1 = [[0,size.y], [100,size.y]],
|
|
line2 = [avec, polar_to_xy(100,mask_angle)+avec],
|
|
cp = line_intersection(line1,line2),
|
|
path = [
|
|
cp + polar_to_xy(size.x+excess, mask_angle+90),
|
|
cp,
|
|
cp + polar_to_xy(size.y+excess, -90),
|
|
[0,-excess],
|
|
[-excess,-excess],
|
|
[-excess,0]
|
|
]
|
|
) reorient(anchor,spin, two_d=true, path=path, extent=false, p=path);
|
|
|
|
|
|
// Function&Module: mask2d_dovetail()
|
|
// Synopsis: Creates a 2D dovetail mask shape.
|
|
// SynTags: Geom, Path
|
|
// Topics: Masks (2D), Shapes (2D), Paths (2D), Path Generators, Attachable
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As Module
|
|
// mask2d_dovetail(edge, [angle], [inset], [shelf], [excess], ...) [ATTACHMENTS];
|
|
// mask2d_dovetail(x=, [angle=], [inset=], [shelf=], [excess=], ...) [ATTACHMENTS];
|
|
// mask2d_dovetail(y=, [angle=], [inset=], [shelf=], [excess=], ...) [ATTACHMENTS];
|
|
// Usage: As Function
|
|
// path = mask2d_dovetail(edge, [angle], [inset], [shelf], [excess]);
|
|
// Description:
|
|
// Creates a 2D dovetail mask shape that is useful for extruding into a 3D mask for a 90° edge.
|
|
// Conversely, you can use that same extruded shape to make an interior dovetail between two walls at a 90º angle.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// Arguments:
|
|
// edge = The length of the edge of the dovetail.
|
|
// angle = The angle of the chamfer edge, away from vertical. Default: 30.
|
|
// inset = Optional amount to inset code from corner. Default: 0
|
|
// shelf = The extra height to add to the inside corner of the dovetail. Default: 0
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// x = The width of the dovetail.
|
|
// y = The height of the dovetail.
|
|
// 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`
|
|
// Example(2D): 2D Dovetail Mask
|
|
// mask2d_dovetail(x=10);
|
|
// Example(2D): 2D Dovetail Mask by Width.
|
|
// mask2d_dovetail(x=10, angle=30);
|
|
// Example(2D): 2D Dovetail Mask by Height.
|
|
// mask2d_dovetail(y=10, angle=30);
|
|
// Example(2D): 2D Inset Dovetail Mask
|
|
// mask2d_dovetail(x=10, inset=2);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile([TOP,"Z"],except=[BACK,TOP+LEFT])
|
|
// mask2d_dovetail(x=10, inset=2);
|
|
// Example: Making an interior dovetail
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_dovetail(x=10);
|
|
module mask2d_dovetail(edge, angle=30, inset=0, shelf=0, excess=0.01, x, y, anchor=CENTER, spin=0) {
|
|
path = mask2d_dovetail(x=x, y=y, edge=edge, angle=angle, inset=inset, shelf=shelf, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_dovetail(edge, angle=30, inset=0, shelf=0, excess=0.01, x, y, anchor=CENTER, spin=0) =
|
|
assert(num_defined([x,y,edge])==1)
|
|
assert(is_finite(first_defined([x,y,edge])))
|
|
assert(is_finite(angle))
|
|
assert(is_finite(excess))
|
|
assert(is_finite(inset)||(is_vector(inset)&&len(inset)==2))
|
|
let(
|
|
inset = is_list(inset)? inset : [inset,inset],
|
|
x = !is_undef(x)? x :
|
|
!is_undef(y)? adj_ang_to_opp(adj=y,ang=angle) :
|
|
hyp_ang_to_opp(hyp=edge,ang=angle),
|
|
y = opp_ang_to_adj(opp=x,ang=angle),
|
|
path = [
|
|
[inset.x,0],
|
|
[-excess, 0],
|
|
[-excess, y+inset.y+shelf],
|
|
inset+[x,y+shelf],
|
|
inset+[x,y],
|
|
inset
|
|
]
|
|
) reorient(anchor,spin, two_d=true, path=path, p=path);
|
|
|
|
|
|
// Function&Module: mask2d_teardrop()
|
|
// Synopsis: Creates a 2D teardrop mask shape with a controllable maximum angle from vertical.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D), FDM Optimized
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As Module
|
|
// mask2d_teardrop(r|d=, [angle], [mask_angle], [excess]) [ATTACHMENTS];
|
|
// Usage: As Function
|
|
// path = mask2d_teardrop(r|d=, [angle], [mask_angle], [excess]);
|
|
// Description:
|
|
// Creates a 2D teardrop mask shape that is useful for extruding into a 3D mask for an edge.
|
|
// Conversely, you can use that same extruded shape to make an interior teardrop fillet between two walls.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// This is particularly useful to make partially rounded bottoms, that don't need support to print.
|
|
// Arguments:
|
|
// r = Radius of the rounding.
|
|
// angle = The maximum angle from vertical.
|
|
// mask_angle = Number of degrees in the corner angle to mask. Default: 90
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// d = Diameter of the rounding.
|
|
// 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`
|
|
// Example(2D): 2D Teardrop Mask
|
|
// mask2d_teardrop(r=10);
|
|
// Example(2D): 2D Teardrop Mask for a Non-Right Edge
|
|
// mask2d_teardrop(r=10, mask_angle=75);
|
|
// Example(2D): Increasing Excess
|
|
// mask2d_teardrop(r=10, mask_angle=75, excess=2);
|
|
// Example(2D): Using a Custom Angle
|
|
// mask2d_teardrop(r=10,angle=30);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile(BOT)
|
|
// mask2d_teardrop(r=10, angle=40);
|
|
// Example: Making an interior teardrop fillet
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_teardrop(r=10);
|
|
function mask2d_teardrop(r, angle=45, mask_angle=90, excess=0.01, d, anchor=CENTER, spin=0) =
|
|
assert(is_finite(angle))
|
|
assert(angle>0 && angle<90)
|
|
assert(is_finite(mask_angle) && mask_angle>0 && mask_angle<180)
|
|
assert(is_finite(excess))
|
|
let(
|
|
r = get_radius(r=r, d=d, dflt=1),
|
|
avec = polar_to_xy(r,mask_angle-90),
|
|
line1 = [[0,r], [100,r]],
|
|
line2 = [avec, polar_to_xy(100,mask_angle)+avec],
|
|
cp = line_intersection(line1,line2),
|
|
tp = cp + polar_to_xy(r,180+angle),
|
|
bp = [tp.x+adj_ang_to_opp(tp.y,angle), 0],
|
|
arcpts = arc(r=r, cp=cp, angle=[mask_angle+90,180+angle]),
|
|
ipath = [
|
|
arcpts[0] + polar_to_xy(excess, mask_angle+90),
|
|
each arcpts,
|
|
bp,
|
|
bp + [0,-excess],
|
|
[0,-excess],
|
|
[-excess,-excess],
|
|
[-excess,0]
|
|
],
|
|
path = deduplicate(ipath)
|
|
) reorient(anchor,spin, two_d=true, path=path, p=path);
|
|
|
|
module mask2d_teardrop(r, angle=45, mask_angle=90, excess=0.01, d, anchor=CENTER, spin=0) {
|
|
path = mask2d_teardrop(r=r, d=d, angle=angle, mask_angle=mask_angle, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
// Function&Module: mask2d_ogee()
|
|
// Synopsis: Creates a 2D ogee mask shape.
|
|
// SynTags: Geom, Path
|
|
// Topics: Shapes (2D), Paths (2D), Path Generators, Attachable, Masks (2D)
|
|
// See Also: corner_profile(), edge_profile(), face_profile()
|
|
// Usage: As Module
|
|
// mask2d_ogee(pattern, [excess], ...) [ATTAHCMENTS];
|
|
// Usage: As Function
|
|
// path = mask2d_ogee(pattern, [excess], ...);
|
|
//
|
|
// Description:
|
|
// Creates a 2D Ogee mask shape that is useful for extruding into a 3D mask for a 90° edge.
|
|
// Conversely, you can use that same extruded shape to make an interior ogee decoration between two walls at a 90º angle.
|
|
// As a 2D mask, this is designed to be differenced away from the edge of a shape that is in the first (X+Y+) quadrant.
|
|
// Since there are a number of shapes that fall under the name ogee, the shape of this mask is given as a pattern.
|
|
// Patterns are given as TYPE, VALUE pairs. ie: `["fillet",10, "xstep",2, "step",[5,5], ...]`. See Patterns below.
|
|
// If called as a function, this just returns a 2D path of the outline of the mask shape.
|
|
// .
|
|
// ### Patterns
|
|
// .
|
|
// Type | Argument | Description
|
|
// -------- | --------- | ----------------
|
|
// "step" | [x,y] | Makes a line to a point `x` right and `y` down.
|
|
// "xstep" | dist | Makes a `dist` length line towards X+.
|
|
// "ystep" | dist | Makes a `dist` length line towards Y-.
|
|
// "round" | radius | Makes an arc that will mask a roundover.
|
|
// "fillet" | radius | Makes an arc that will mask a fillet.
|
|
//
|
|
// Arguments:
|
|
// pattern = A list of pattern pieces to describe the Ogee.
|
|
// excess = Extra amount of mask shape to creates on the X- and Y- sides of the shape. Default: 0.01
|
|
// ---
|
|
// 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`
|
|
//
|
|
// Example(2D): 2D Ogee Mask
|
|
// mask2d_ogee([
|
|
// "xstep",1, "ystep",1, // Starting shoulder.
|
|
// "fillet",5, "round",5, // S-curve.
|
|
// "ystep",1, "xstep",1 // Ending shoulder.
|
|
// ]);
|
|
// Example: Masking by Edge Attachment
|
|
// diff()
|
|
// cube([50,60,70],center=true)
|
|
// edge_profile(TOP)
|
|
// mask2d_ogee([
|
|
// "xstep",1, "ystep",1, // Starting shoulder.
|
|
// "fillet",5, "round",5, // S-curve.
|
|
// "ystep",1, "xstep",1 // Ending shoulder.
|
|
// ]);
|
|
// Example: Making an interior ogee
|
|
// %render() difference() {
|
|
// move(-[5,0,5]) cube(30, anchor=BOT+LEFT);
|
|
// cube(310, anchor=BOT+LEFT);
|
|
// }
|
|
// xrot(90)
|
|
// linear_extrude(height=30, center=true)
|
|
// mask2d_ogee([
|
|
// "xstep", 1, "round",5,
|
|
// "ystep",1, "fillet",5,
|
|
// "xstep", 1, "ystep", 1,
|
|
// ]);
|
|
module mask2d_ogee(pattern, excess=0.01, anchor=CENTER,spin=0) {
|
|
path = mask2d_ogee(pattern, excess=excess);
|
|
attachable(anchor,spin, two_d=true, path=path) {
|
|
polygon(path);
|
|
children();
|
|
}
|
|
}
|
|
|
|
function mask2d_ogee(pattern, excess=0.01, anchor=CENTER, spin=0) =
|
|
assert(is_list(pattern))
|
|
assert(len(pattern)>0)
|
|
assert(len(pattern)%2==0,"pattern must be a list of TYPE, VAL pairs.")
|
|
assert(all([for (i = idx(pattern,step=2)) in_list(pattern[i],["step","xstep","ystep","round","fillet"])]))
|
|
let(
|
|
x = concat([0], cumsum([
|
|
for (i=idx(pattern,step=2)) let(
|
|
type = pattern[i],
|
|
val = pattern[i+1]
|
|
) (
|
|
type=="step"? val.x :
|
|
type=="xstep"? val :
|
|
type=="round"? val :
|
|
type=="fillet"? val :
|
|
0
|
|
)
|
|
])),
|
|
y = concat([0], cumsum([
|
|
for (i=idx(pattern,step=2)) let(
|
|
type = pattern[i],
|
|
val = pattern[i+1]
|
|
) (
|
|
type=="step"? val.y :
|
|
type=="ystep"? val :
|
|
type=="round"? val :
|
|
type=="fillet"? val :
|
|
0
|
|
)
|
|
])),
|
|
tot_x = last(x),
|
|
tot_y = last(y),
|
|
data = [
|
|
for (i=idx(pattern,step=2)) let(
|
|
type = pattern[i],
|
|
val = pattern[i+1],
|
|
pt = [x[i/2], tot_y-y[i/2]] + (
|
|
type=="step"? [val.x,-val.y] :
|
|
type=="xstep"? [val,0] :
|
|
type=="ystep"? [0,-val] :
|
|
type=="round"? [val,0] :
|
|
type=="fillet"? [0,-val] :
|
|
[0,0]
|
|
)
|
|
) [type, val, pt]
|
|
],
|
|
path = [
|
|
[tot_x,-excess],
|
|
[-excess,-excess],
|
|
[-excess,tot_y],
|
|
for (pat = data) each
|
|
pat[0]=="step"? [pat[2]] :
|
|
pat[0]=="xstep"? [pat[2]] :
|
|
pat[0]=="ystep"? [pat[2]] :
|
|
let(
|
|
r = pat[1],
|
|
steps = segs(abs(r)),
|
|
step = 90/steps
|
|
) [
|
|
for (i=[0:1:steps]) let(
|
|
a = pat[0]=="round"? (180+i*step) : (90-i*step)
|
|
) pat[2] + abs(r)*[cos(a),sin(a)]
|
|
]
|
|
],
|
|
path2 = deduplicate(path)
|
|
) reorient(anchor,spin, two_d=true, path=path2, p=path2);
|
|
|
|
|
|
|
|
// vim: expandtab tabstop=4 shiftwidth=4 softtabstop=4 nowrap
|