build123d/docs/general_examples.py
2023-04-01 14:38:05 +02:00

856 lines
21 KiB
Python

"""
name: general_examples.py
by: jdegenstein
date: December 29th 2022
desc:
This is the build123d general examples python script. It generates the SVGs
when run as a script, and is pulled into sphinx docs by
tutorial_general.rst.
license:
Copyright 2022 jdegenstein
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
from build123d import *
svg_opts = {
"width": 500,
"height": 220,
# "pixel_scale": 4,
"margin_left": 10,
"margin_top": 10,
"show_axes": False,
"show_hidden": True,
}
def svgout(ex_counter, width=500, height=220):
svg_opts["width"] = width
svg_opts["height"] = height
obj = globals()[f"ex{ex_counter}"]
obj.part.export_svg(
f"assets/general_ex{ex_counter}.svg",
(-100, -100, 70),
(0, 0, 1),
svg_opts=svg_opts,
)
ex_counter = 1
##########################################
# 1. Simple Rectangular Plate
# [Ex. 1]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex1:
Box(length, width, thickness)
# [Ex. 1]
svgout(ex_counter)
ex_counter += 1
# show_object(ex1.part)
##########################################
# 2. Plane with Hole
# [Ex. 2]
length, width, thickness = 80.0, 60.0, 10.0
center_hole_dia = 22.0
with BuildPart() as ex2:
Box(length, width, thickness)
Cylinder(radius=center_hole_dia / 2, height=thickness, mode=Mode.SUBTRACT)
# [Ex. 2]
svgout(ex_counter)
ex_counter += 1
# show_object(ex2.part)
##########################################
# 3. An extruded prismatic solid
# [Ex. 3]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex3:
with BuildSketch() as ex3_sk:
Circle(width)
Rectangle(length / 2, width / 2, mode=Mode.SUBTRACT)
extrude(amount=2 * thickness)
# [Ex. 3]
svgout(ex_counter)
ex_counter += 1
# show_object(ex3.part)
##########################################
# Building Profiles using lines and arcs
# [Ex. 4]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex4:
with BuildSketch() as ex4_sk:
with BuildLine() as ex4_ln:
l1 = Line((0, 0), (length, 0))
l2 = Line((length, 0), (length, width))
l3 = ThreePointArc((length, width), (width, width * 1.5), (0.0, width))
l4 = Line((0.0, width), (0, 0))
make_face()
extrude(amount=thickness)
# [Ex. 4]
svgout(ex_counter)
ex_counter += 1
# show_object(ex4.part)
##########################################
# Moving The Current working point
# [Ex. 5]
a, b, c, d = 90, 45, 15, 7.5
with BuildPart() as ex5:
with BuildSketch() as ex5_sk:
Circle(a)
with Locations((b, 0.0)):
Rectangle(c, c, mode=Mode.SUBTRACT)
with Locations((0, b)):
Circle(d, mode=Mode.SUBTRACT)
extrude(amount=c)
# [Ex. 5]
svgout(ex_counter)
ex_counter += 1
# show_object(ex5.part)
##########################################
# Using Point Lists
# [Ex. 6]
a, b, c = 80, 60, 10
with BuildPart() as ex6:
with BuildSketch() as ex6_sk:
Circle(a)
with Locations((b, 0), (0, b), (-b, 0), (0, -b)):
Circle(c, mode=Mode.SUBTRACT)
extrude(amount=c)
# [Ex. 6]
svgout(ex_counter)
ex_counter += 1
# show_object(ex6.part)
#############################
# Polygons
# [Ex. 7]
a, b, c = 60, 80, 5
with BuildPart() as ex7:
with BuildSketch() as ex7_sk:
Rectangle(a, b)
with Locations((0, 3 * c), (0, -3 * c)):
RegularPolygon(radius=2 * c, side_count=6, mode=Mode.SUBTRACT)
extrude(amount=c)
# [Ex. 7]
svgout(ex_counter)
ex_counter += 1
# show_object(ex7.part)
##########################################
# 8. Polylines
# [Ex. 8]
(L, H, W, t) = (100.0, 20.0, 20.0, 1.0)
pts = [
(0, H / 2.0),
(W / 2.0, H / 2.0),
(W / 2.0, (H / 2.0 - t)),
(t / 2.0, (H / 2.0 - t)),
(t / 2.0, (t - H / 2.0)),
(W / 2.0, (t - H / 2.0)),
(W / 2.0, H / -2.0),
(0, H / -2.0),
]
with BuildPart() as ex8:
with BuildSketch(Plane.YZ) as ex8_sk:
with BuildLine() as ex8_ln:
Polyline(*pts)
mirror(ex8_ln.line, about=Plane.YZ)
make_face()
extrude(amount=L)
# [Ex. 8]
svgout(ex_counter)
ex_counter += 1
# show_object(ex8.part)
##########################################
# 9. Selectors, fillets, and chamfers
# [Ex. 9]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex9:
Box(length, width, thickness)
chamfer(ex9.edges().group_by(Axis.Z)[-1], length=4)
fillet(ex9.edges().filter_by(Axis.Z), radius=5)
# [Ex. 9]
svgout(ex_counter)
ex_counter += 1
# show_object(ex9.part)
##########################################
# 10. Select Last and Hole
# [Ex. 10]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex10:
Box(length, width, thickness)
chamfer(ex10.edges().group_by(Axis.Z)[-1], length=4)
fillet(ex10.edges().filter_by(Axis.Z), radius=5)
Hole(radius=width / 4)
fillet(ex10.edges(Select.LAST).sort_by(Axis.Z)[-1], radius=2)
# [Ex. 10]
svgout(ex_counter)
ex_counter += 1
# show_object(ex10.part)
##########################################
# 11. Use a face as workplane for BuildSketch and introduce GridLocations
# [Ex. 11]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex11:
Box(length, width, thickness)
chamfer(ex11.edges().group_by(Axis.Z)[-1], length=4)
fillet(ex11.edges().filter_by(Axis.Z), radius=5)
Hole(radius=width / 4)
fillet(ex11.edges(Select.LAST).sort_by(Axis.Z)[-1], radius=2)
with BuildSketch(ex11.faces().sort_by(Axis.Z)[-1]) as ex11_sk:
with GridLocations(length / 2, width / 2, 2, 2):
RegularPolygon(radius=5, side_count=5)
extrude(amount=-thickness, mode=Mode.SUBTRACT)
# [Ex. 11]
svgout(ex_counter)
ex_counter += 1
# show_object(ex11)
##########################################
# 12. Defining an Edge with a Spline
# [Ex. 12]
sPnts = [
(55, 30),
(50, 35),
(40, 30),
(30, 20),
(20, 25),
(10, 20),
(0, 20),
]
with BuildPart() as ex12:
with BuildSketch() as ex12_sk:
with BuildLine() as ex12_ln:
l1 = Spline(*sPnts)
l2 = Line((55, 30), (60, 0))
l3 = Line((60, 0), (0, 0))
l4 = Line((0, 0), (0, 20))
make_face()
extrude(amount=10)
# [Ex. 12]
svgout(ex_counter)
ex_counter += 1
# show_object(ex12.part)
##########################################
# 13. CounterBoreHoles, CounterSinkHoles and PolarLocations
# [Ex. 13]
a, b = 40, 4
with BuildPart() as ex13:
Cylinder(radius=50, height=10)
with Locations(ex13.faces().sort_by(Axis.Z)[-1]):
with PolarLocations(radius=a, count=4):
CounterSinkHole(radius=b, counter_sink_radius=2 * b)
with PolarLocations(radius=a, count=4, start_angle=45, angular_range=360):
CounterBoreHole(radius=b, counter_bore_radius=2 * b, counter_bore_depth=b)
# [Ex. 13]
svgout(ex_counter)
ex_counter += 1
# show_object(ex13.part)
##########################################
# 14. Position on a line with '@', '%' and introduce sweep
# [Ex. 14]
a, b = 40, 20
with BuildPart() as ex14:
with BuildLine() as ex14_ln:
l1 = JernArc(start=(0, 0), tangent=(0, 1), radius=a, arc_size=180)
l2 = JernArc(start=l1 @ 1, tangent=l1 % 1, radius=a, arc_size=-90)
l3 = Line(l2 @ 1, l2 @ 1 + Vector(-a, a))
with BuildSketch(Plane.XZ) as ex14_sk:
Rectangle(b, b)
sweep()
# [Ex. 14]
svgout(ex_counter)
ex_counter += 1
# show_object(ex14.part)
##########################################
# 15. Mirroring Symmetric Geometry
# [Ex. 15]
a, b, c = 80, 40, 20
with BuildPart() as ex15:
with BuildSketch() as ex15_sk:
with BuildLine() as ex15_ln:
l1 = Line((0, 0), (a, 0))
l2 = Line(l1 @ 1, l1 @ 1 + Vector(0, b))
l3 = Line(l2 @ 1, l2 @ 1 + Vector(-c, 0))
l4 = Line(l3 @ 1, l3 @ 1 + Vector(0, -c))
l5 = Line(l4 @ 1, Vector(0, (l4 @ 1).Y))
mirror(ex15_ln.line, about=Plane.YZ)
make_face()
extrude(amount=c)
# [Ex. 15]
svgout(ex_counter)
ex_counter += 1
# show_object(ex15.part)
##########################################
# 16. Mirroring 3D Objects
# same concept as CQ docs, but different object
# [Ex. 16]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex16_single:
with BuildSketch(Plane.XZ) as ex16_sk:
Rectangle(length, width)
fillet(ex16_sk.vertices(), radius=length / 10)
with GridLocations(x_spacing=length / 4, y_spacing=0, x_count=3, y_count=1):
Circle(length / 12, mode=Mode.SUBTRACT)
Rectangle(length, width, align=(Align.MIN, Align.MIN), mode=Mode.SUBTRACT)
extrude(amount=length)
with BuildPart() as ex16:
add(ex16_single.part)
mirror(ex16_single.part, about=Plane.XY.offset(width))
mirror(ex16_single.part, about=Plane.YX.offset(width))
mirror(ex16_single.part, about=Plane.YZ.offset(width))
mirror(ex16_single.part, about=Plane.YZ.offset(-width))
# [Ex. 16]
svgout(ex_counter, height=400)
ex_counter += 1
# show_object(ex16.part)
##########################################
# 17. Mirroring From Faces
# [Ex. 17]
a, b = 30, 20
with BuildPart() as ex17:
with BuildSketch() as ex17_sk:
RegularPolygon(radius=a, side_count=5)
extrude(amount=b)
mirror(ex17.part, about=Plane((ex17.faces().group_by(Axis.Y)[0])[0]))
# [Ex. 17]
svgout(ex_counter)
ex_counter += 1
# show_object(ex17.part)
##########################################
# 18. Creating Workplanes on Faces
# based on Ex. 9
# [Ex. 18]
length, width, thickness = 80.0, 60.0, 10.0
a, b = 4, 5
with BuildPart() as ex18:
Box(length, width, thickness)
chamfer(ex18.edges().group_by(Axis.Z)[-1], length=a)
fillet(ex18.edges().filter_by(Axis.Z), radius=b)
with BuildSketch(ex18.faces().sort_by(Axis.Z)[-1]):
Rectangle(2 * b, 2 * b)
extrude(amount=-thickness, mode=Mode.SUBTRACT)
# [Ex. 18]
svgout(ex_counter)
ex_counter += 1
# show_object(ex18.part)
##########################################
# 19. Locating a Workplane on a vertex
# [Ex. 19]
length, thickness = 80.0, 10.0
with BuildPart() as ex19:
with BuildSketch() as ex19_sk2:
RegularPolygon(radius=length / 2, side_count=7)
extrude(amount=thickness)
topf = ex19.faces().sort_by(Axis.Z)[-1]
vtx = topf.vertices().group_by(Axis.X)[-1][0]
vtx2Axis = Axis((0, 0, 0), (-1, -0.5, 0))
vtx2 = topf.vertices().sort_by(vtx2Axis)[-1]
with BuildSketch(topf) as ex19_sk2:
with Locations((vtx.X, vtx.Y), (vtx2.X, vtx2.Y)):
Circle(radius=length / 8)
extrude(amount=-thickness, mode=Mode.SUBTRACT)
# [Ex. 19]
svgout(ex_counter)
ex_counter += 1
# show_object(ex19.part)
##########################################
# 20. Offset Sketch Workplane
# [Ex. 20]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex20:
Box(length, width, thickness)
plane = Plane((ex20.faces().group_by(Axis.X))[0][0])
with BuildSketch(plane.offset(2 * thickness)):
Circle(width / 3)
extrude(amount=width)
# [Ex. 20]
svgout(ex_counter)
ex_counter += 1
# show_object(ex20.part)
##########################################
# 21. Copying Workplanes
# [Ex. 21]
width, length = 10.0, 60.0
with BuildPart() as ex21:
with BuildSketch() as ex21_sk:
Circle(width / 2)
extrude(amount=length)
with BuildSketch(Plane(origin=ex21.part.center(), z_dir=(-1, 0, 0))):
Circle(width / 2)
extrude(amount=length)
# [Ex. 21]
svgout(ex_counter)
ex_counter += 1
# show_object(ex21.part)
##########################################
# 22. Rotated Workplanes
# [Ex. 22]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex22:
Box(length, width, thickness)
pln = Plane((ex22.faces().group_by(Axis.Z)[0])[0]).rotated((0, -50, 0))
with BuildSketch(pln) as ex22_sk:
with GridLocations(length / 4, width / 4, 2, 2):
Circle(thickness / 4)
extrude(amount=-100, both=True, mode=Mode.SUBTRACT)
# [Ex. 22]
svgout(ex_counter)
ex_counter += 1
# show_object(ex22.part)
##########################################
# 23. Revolve
# [Ex. 23]
pts = [
(-25, 35),
(-25, 0),
(-20, 0),
(-20, 5),
(-15, 10),
(-15, 35),
]
with BuildPart() as ex23:
with BuildSketch(Plane.XZ) as ex23_sk:
with BuildLine() as ex23_ln:
l1 = Polyline(*pts)
l2 = Line(l1 @ 1, l1 @ 0)
make_face()
with Locations((0, 35)):
Circle(25)
split(bisect_by=Plane.ZY)
revolve(axis=Axis.Z)
# [Ex. 23]
svgout(ex_counter, height=300)
ex_counter += 1
# show_object(ex23.part)
##########################################
# 24. Lofts
# [Ex. 24]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex24:
Box(length, length, thickness)
with BuildSketch((ex24.faces().group_by(Axis.Z)[0])[0]) as ex24_sk:
Circle(length / 3)
with BuildSketch(ex24_sk.faces()[0].offset(length / 2)) as ex24_sk2:
Rectangle(length / 6, width / 6)
loft()
# [Ex. 24]
svgout(ex_counter)
ex_counter += 1
# show_object(ex24.part)
##########################################
# 25. Offset Sketch
# [Ex. 25]
rad, offs = 50, 10
with BuildPart() as ex25:
with BuildSketch() as ex25_sk1:
RegularPolygon(radius=rad, side_count=5)
with BuildSketch(Plane.XY.offset(15)) as ex25_sk2:
RegularPolygon(radius=rad, side_count=5)
offset(amount=offs)
with BuildSketch(Plane.XY.offset(30)) as ex25_sk3:
RegularPolygon(radius=rad, side_count=5)
offset(amount=offs, kind=Kind.INTERSECTION)
extrude(amount=1)
# [Ex. 25]
svgout(ex_counter)
ex_counter += 1
# show_object(ex25.part)
##########################################
# 26. Offset Part To Create Thin features
# [Ex. 26]
length, width, thickness, wall = 80.0, 60.0, 10.0, 2.0
with BuildPart() as ex26:
Box(length, width, thickness)
topf = ex26.faces().sort_by(Axis.Z)[-1]
offset(amount=-wall, openings=topf)
# [Ex. 26]
svgout(ex_counter)
ex_counter += 1
# show_object(ex26.part)
##########################################
# 27. Splitting an Object
# [Ex. 27]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex27:
Box(length, width, thickness)
with BuildSketch(ex27.faces().sort_by(Axis.Z)[0]) as ex27_sk:
Circle(width / 4)
extrude(amount=-thickness, mode=Mode.SUBTRACT)
split(bisect_by=Plane(ex27.faces().sort_by(Axis.Y)[-1]).offset(-width / 2))
# [Ex. 27]
svgout(ex_counter)
ex_counter += 1
# show_object(ex27.part)
##########################################
# 28. Locating features based on Faces
# [Ex. 28]
width, thickness = 80.0, 10.0
with BuildPart() as ex28:
with BuildSketch() as ex28_sk:
RegularPolygon(radius=width / 4, side_count=3)
ex28_ex = extrude(amount=thickness, mode=Mode.PRIVATE)
midfaces = ex28_ex.faces().group_by(Axis.Z)[1]
Sphere(radius=width / 2)
for face in midfaces:
with Locations(face):
Hole(thickness / 2)
# [Ex. 28]
svgout(ex_counter)
ex_counter += 1
# show_object(ex28.part)
##########################################
# 29. The Classic OCC Bottle
# [Ex. 29]
L, w, t, b, h, n = 60.0, 18.0, 9.0, 0.9, 90.0, 6.0
with BuildPart() as ex29:
with BuildSketch(Plane.XY.offset(-b)) as ex29_ow_sk:
with BuildLine() as ex29_ow_ln:
l1 = Line((0, 0), (0, w / 2))
l2 = ThreePointArc(l1 @ 1, (L / 2.0, w / 2.0 + t), (L, w / 2.0))
l3 = Line(l2 @ 1, Vector((l2 @ 1).X, 0, 0))
mirror(ex29_ow_ln.line)
make_face()
extrude(amount=h + b)
fillet(ex29.edges(), radius=w / 6)
with BuildSketch(ex29.faces().sort_by(Axis.Z)[-1]):
Circle(t)
extrude(amount=n)
necktopf = ex29.faces().sort_by(Axis.Z)[-1]
offset(ex29.solids()[0], amount=-b, openings=necktopf)
# [Ex. 29]
svgout(ex_counter, height=400)
ex_counter += 1
# show_object(ex29.part)
##########################################
# 30. Bezier Curve
# [Ex. 30]
pts = [
(0, 0),
(20, 20),
(40, 0),
(0, -40),
(-60, 0),
(0, 100),
(100, 0),
]
wts = [
1.0,
1.0,
2.0,
3.0,
4.0,
2.0,
1.0,
]
with BuildPart() as ex30:
with BuildSketch() as ex30_sk:
with BuildLine() as ex30_ln:
l0 = Polyline(*pts)
l1 = Bezier(*pts, weights=wts)
make_face()
extrude(amount=10)
# [Ex. 30]
svgout(ex_counter)
ex_counter += 1
# show_object(ex30.part)
##########################################
# 31. Nesting Locations
# [Ex. 31]
a, b, c = 80.0, 5.0, 3.0
with BuildPart() as ex31:
with BuildSketch() as ex31_sk:
with PolarLocations(a / 2, 6):
with GridLocations(3 * b, 3 * b, 2, 2):
RegularPolygon(b, 3)
RegularPolygon(b, 4)
RegularPolygon(3 * b, 6, rotation=30)
extrude(amount=c)
# [Ex. 31]
svgout(ex_counter)
ex_counter += 1
# show_object(ex31.part)
##########################################
# 32. Python for-loop
# [Ex. 32]
a, b, c = 80.0, 10.0, 1.0
with BuildPart() as ex32:
with BuildSketch(mode=Mode.PRIVATE) as ex32_sk:
RegularPolygon(2 * b, 6, rotation=30)
with PolarLocations(a / 2, 6):
RegularPolygon(b, 4)
for idx, obj in enumerate(ex32_sk.sketch.faces()):
add(obj)
extrude(amount=c + 3 * idx)
# [Ex. 32]
svgout(ex_counter)
ex_counter += 1
# show_object(ex32.part)
##########################################
# 33. Python function and for-loop
# [Ex. 33]
a, b, c = 80.0, 5.0, 1.0
def square(rad, loc):
with BuildSketch() as sk:
with Locations(loc):
RegularPolygon(rad, 4)
return sk.sketch
with BuildPart() as ex33:
with BuildSketch(mode=Mode.PRIVATE) as ex33_sk:
locs = PolarLocations(a / 2, 6)
for i, j in enumerate(locs):
add(square(b + 2 * i, j))
for idx, obj in enumerate(ex33_sk.sketch.faces()):
add(obj)
extrude(amount=c + 2 * idx)
# [Ex. 33]
svgout(ex_counter)
ex_counter += 1
# show_object(ex33.part)
##########################################
# 34. Embossed and Debossed Text
# [Ex. 34]
length, width, thickness, fontsz, fontht = 80.0, 60.0, 10.0, 25.0, 4.0
with BuildPart() as ex34:
Box(length, width, thickness)
topf = ex34.faces().sort_by(Axis.Z)[-1]
with BuildSketch(topf) as ex34_sk:
Text("Hello", font_size=fontsz, align=(Align.CENTER, Align.MIN))
extrude(amount=fontht)
with BuildSketch(topf) as ex34_sk2:
Text("World", font_size=fontsz, align=(Align.CENTER, Align.MAX))
extrude(amount=-fontht, mode=Mode.SUBTRACT)
# [Ex. 34]
svgout(ex_counter)
ex_counter += 1
# show_object(ex34.part)
##########################################
# 35. Slots
# [Ex. 35]
length, width, thickness = 80.0, 60.0, 10.0
with BuildPart() as ex35:
Box(length, length, thickness)
topf = ex35.faces().sort_by(Axis.Z)[-1]
with BuildSketch(topf) as ex35_sk:
SlotCenterToCenter(width / 2, 10)
with BuildLine(mode=Mode.PRIVATE) as ex35_ln:
RadiusArc((-width / 2, 0), (0, width / 2), radius=width / 2)
SlotArc(arc=ex35_ln.edges()[0], height=thickness, rotation=0)
with BuildLine(mode=Mode.PRIVATE) as ex35_ln2:
RadiusArc((0, -width / 2), (width / 2, 0), radius=-width / 2)
SlotArc(arc=ex35_ln2.edges()[0], height=thickness, rotation=0)
extrude(amount=-thickness, mode=Mode.SUBTRACT)
# [Ex. 35]
svgout(ex_counter)
ex_counter += 1
# show_object(ex35.part)
##########################################
# 36. Extrude-Until
# [Ex. 36]
rad, rev = 6, 50
with BuildPart() as ex36:
with BuildSketch() as ex36_sk:
with Locations((0, rev)):
Circle(rad)
revolve(axis=Axis.X, revolution_arc=180)
with BuildSketch() as ex36_sk2:
Rectangle(rad, rev)
extrude(until=Until.NEXT)
# [Ex. 36]
svgout(ex_counter)
ex_counter += 1
# show_object(ex36.part)