MuSCAD lets you write simple, readable, Pythonic code, that automatically translates into valid OpenSCAD code. It is loosely inspired by SolidPython, which is itself based on Phillip Tiefenbacher's openscad module, found on Thingiverse. It is the continuation of the work I initiated a few years ago on GitLab: https://gitlab.com/guillp/muscad.
Here's a simple example. This simple Python code:
from muscad import Cube, Sphere, Cylinder print( Cube(10, 10, 10).down(10) - Sphere(d=15) + Cylinder(h=15, d=12).leftward(16) )
Will output this OpenSCAD code:
union() { difference() { translate(v=[0, 0, -10]) { cube(size=[10, 10, 10]); } sphere(d=15); } translate(v=[-16, 0, 0]) { cylinder(h=15, d=12, $fn=94); } }
Obviously this simple example is not enough to show what MuSCAD allows you to do with ease. Here are a few real-life 3D printer parts designed with MuSCAD:
A stepper motor attachment. See the OpenSCAD code here and check the MuSCAD code here
A feet for 3030 extrusion based printers. Check the OpenSCAD code here and the MuSCAD code here
A heating bed attachment for HyperCube-like printers. Here is OpenSCAD code and here is the MuSCAD code
The main benefit of using MuSCAD is that you actually write Python code, so you get all the advantages of that language, which include:
- you can use all Python features, including all syntactic sugar like built-in dict/list comprehensions, external modules such as text parsing, image recognition, etc. which enables very original ways of creating objects.
- you can use a real IDE and development tools, possibly providing code completion, code formatting, validity/typos checks, etc. which makes it way easier and faster to write code compared to most generally poor tools used to write OpenSCAD code.
The second advantage, compared to SolidPython and other similar modules, is that MuSCAD lets you write Pythonic, easy to read code, and does not try to emulate the way you would write the same code in OpenSCAD. It enables features that are not part of OpenSCAD, like automatic calculation of object sizes, relative alignment between objects, and more. In that sense, MuSCAD syntax is fundamentally different compared to OpenSCAD/SolidPython. Those differences are explained in the next sections.
MuSCAD requires Python 3.6 or more and has no other dependency. You obviously need to install OpenSCAD to view the generated files.
Install via PyPI:
pip install muscad
Usage with Jupyter Notebook or Jupyter Lab is recommended.
MuSCAD includes classes for all OpenSCAD primitives. You can instantiate those
classes, then apply transformations using methods on those instances, and
finally apply boolean operations using the +
, -
and &
operators.
To generate the equivalent OpenSCAD code, simply print()
the resulting object.
Example:
from muscad import Sphere hollow_sphere = Sphere(10) - Sphere(8) print(hollow_sphere)
will output:
difference() { sphere(d=10, $fn=78); sphere(d=8, $fn=62); }
Simply copy/paste that output into OpenSCAD, and it will render the resulting object.
To make things easier, you can programmatically save the output code to a file directly, and have that file opened
in OpenSCAD (if the openscad executable is in your PATH
):
hollow_sphere.render_to_file('hollow_sphere.scad', openscad=True)
If you only want to save the file, without (re)opening OpenSCAD, omit the openscad
parameter or set it to False.
You may also set the environment variable MUSCAD_NO_OPENSCAD
to inhibit MuSCAD from opening OpenSCAD automatically.
If you use Jupyter Notebook, it will automatically display the OpenSCAD code for any object in a cell.
Now, to understand how MuSCAD generates the OpenSCAD code, let's go over the sample Python code above, line by line:
from muscad import Sphere
This import the primitive Sphere from the top level module muscad
. All other primitives can be imported from there.
hollow_sphere = Sphere(10) - Sphere(8)
This generate a MuSCAD object called hollow_sphere
, made from the difference of 2 spheres (a smaller one removed from a bigger one).
What is a Sphere
? It is obviously the equivalent of OpenSCAD's sphere
. Like all other primitives, a Sphere
is an instance of an Object
:
from muscad import Object assert isinstance(Sphere(10), Object)
Why is that important ? Because there is a lot you can do with a MuSCAD Object
, like rendering it, applying transformations to it, using it
in boolean operations, aligning it to absolute coordinates, etc. But we'll see about that later.
Now, what is this hollow_box
object? It is a Difference
:
from muscad import Difference assert type(hollow_sphere) == Difference
That Difference
is itself a subclass of Object
:
from muscad import Object assert isinstance(hollow_sphere) == Object
So you can do to that difference everything you can do to a primitive. Obviously, you can render the OpenSCAD code, that's what happens when we call:
print(hollow_sphere)
results in:
difference() { sphere(d=10, $fn=78); sphere(d=8, $fn=62); }
Note that when rendering a Sphere
, the $fn
parameter, which indicates how many segments OpenSCAD must use to render that sphere) is produced automatically by MuSCAD to create a good-looking round shape.
In the usual where you want your round shapes to actually appear round, just let MuSCAD handle this for you. Otherwise, you can override that number with the segments
parameter to Sphere:
Sphere(20, segments=6)
All available primitives from OpenSCAD are available in MuSCAD:
# 3D Primitives Cube(width, depth, height) # cube Cylinder(h, d, d2=None, segments="auto") # cylinder Sphere(d, segments="auto") # sphere Polyhedron(points, faces, convexity=1) # polyhedron # 2D Primitives Circle(d, segments="auto") # circle Square(width, depth) # square Text(text, size=10, font=None, halign=None, valign=None, spacing=None, direction=None, language=None, script=None, segments=None) # text Polygon(*points, path=None, convexity=None) # polygon
Note that MuSCAD includes a high level Volume
class that is basically a Cube
but with a lot of added features, see below.
Boolean operations union(), difference() and intersection() are applied using the operators +, - and & respectively:
Sphere(d=5) + Cube(10, 2, 1) + Cube(3, 3, 3).leftward(4) # union Sphere(d=5) - Cube(10, 2, 1) - Cube(1, 1, 1) # difference Sphere(d=5) & Cube(10, 2, 1) & Cube(1, 4, 2) # intersection
Beware that standard Python operator precedence applies: +
and -
applies before &
.
So the following codes produce different results:
Sphere(d=5) & Cube(10, 2, 1) & Cube(1, 1, 1) + Sphere(50)
and:
(Sphere(d=5) & Cube(10, 2, 1) & Cube(1, 1, 1)) + Sphere(50)
You can also use a more traditional paradigm:
from muscad import Union, Difference, Intersection union = Union( Sphere(d=5), Cube(10, 2, 1), Cube(3, 3, 3).leftward(4) ) difference = Difference( Sphere(d=5), Cube(10, 2, 1), Cube(1, 1, 1) ) intersection = Intersection( Sphere(d=5), Cube(10, 2, 1), Cube(1, 4, 2) )
As already mentioned above, the result of a boolean operation is itself a MuSCAD Object, so you can keep applying new boolean operations or transformations to it.
You can apply transformations to any MuSCAD Object
by calling the transformation methods .translate(), .rotate(), etc.
Here we translate a Sphere 10 mm upwards:
print(Sphere(10).translate(z=10))
This will give the following OpenSCAD code:
translate(v=[0, 0, 10]) sphere(d=10, $fn=78);
Any Object
, including results of boolean operations, or transformed objects, can be applied a transformation.
So you can obviously chain multiple methods like this:
Cube(10, 10, 10).translate(15, 15, 0).rotate(0, 45, 0)
MuSCAD includes helpers methods for single axis translations and rotations. Using those helpers, the code just above is equivalent to:
Cube(10, 10, 10).rightward(15).up(15).y_rotate(45)
Note that MuSCAD will automatically merge multiple chained translations or rotations on the same object (this however has no effect on the rendered part). So both the lines above will result in the same generated OpenSCAD code (notice that there is a single translation on the cube combining both translations rightward and upward, instead of several):
rotate(a=[0, 45, 0]) { translate(v=[15, 0, 15]) { cube(size=[10, 10, 10]); } }
All available transformation methods are as follow:
.translate(x=0, y=0, z=0) # applies a Translation .rightward(dist) # applies a Translation to the right .leftward(dist) # applies a Translation to the left .forward(dist) # applies a Translation to the front .backward(dist) # applies a Translation to the back .up(dist) # applies a Translation upwards .down(dist) # applies a Translation downwards .rotate(x=0, y=0, z=0) # applies a Rotation .x_rotate(angle) # applies a Rotation on the X axis .y_rotate(angle) # applies a Rotation on the Y axis .z_rotate(angle) # applies a Rotation on the Z axis .left_to_bottom() # turn left face to bottom, alias for .y_rotate(-90) .left_to_top() # turn left face to top, alias for .y_rotate(90) .left_to_front() # alias for .z_rotate(-90) .left_to_back() # alias for .z_rotate(90) .upside_down(y_axis=False) # alias for .x_rotate(180) if y_axis==False else .y_rotate(180) .scale(x=0, y=0, z=0) # applies a Scaling transformation .resize(x=0, y=0, z=0) # applies a Resizing transformation .mirror(x=0, y=0, z=0) # applies a Mirroring transformation .x_mirror(center=0) # applies a mirroring on X axis, offset by `center` .y_mirror(center=0) # applies a mirroring on Y axis, offset by `center` .z_mirror(center=0) # applies a mirroring on Z axis, offset by `center` .linear_extrude(height, center=False, convexity=10, twist=0.0, slices=None, scale=1.0, segments="auto") # applies a LinearExtrusion .z_linear_extrude(distance=None, bottom=None, center_z=None, top=None, convexity=10, twist=0.0, slices=None, scale=1.0, segments="auto") # helper to do a LinearExtrusion on the Z axis .y_linear_extrude(distance=None, back=None, center_y=None, front=None, convexity=10, twist=0.0, slices=None, scale=1.0, segments="auto") # helper to do a LinearExtrusion on the Y axis .x_linear_extrude(distance=None, left=None, center_x=None, right=None, convexity=10, twist=0.0, slices=None, scale=1.0, segments="auto") # helper to do a LinearExtrusion on the X axis .rotational_extrude(angle=360, convexity=None, segments="auto") # applies a RotationalExtrusion .color(colorname) # change the object color .slide(x=0, y=0, z=0) # applies a Slide transformation
So far we have seen the equivalent of what you can do with OpenSCAD, with a different syntax but without much added value.
Here is one of the best added value of MuSCAD: it knows the size and position of any Object
, which make it very easy to position
or reposition the object at absolute coordinates, or even at a relative position compared to another Object
. This is called alignment.
First, let's understand the dimension of an Object
by creating a Cube
(which is badly named in OpenSCAD since it can have different dimensions on each axis):
from muscad import Cube box = Cube(20, 30, 50)
MuSCAD primitives are always created centered, so our box with a width of 20 extends from -10 to +10 on the X axis. MuSCAD knows about that:
assert box.left == -10 assert box.right == 10
Same on the Y and Z axis:
assert box.back == -15 assert box.front == 15 assert box.bottom == -25 assert box.top == 25
The box formed by the leftmost, rightmost, back, front, bottom and top coordinates of an object is called the bounding box.
Since MuSCAD knows the lower and upper bounds on all axis, it can compute the center as well. Since our box
was created centered, the center is 0 on all axis:
assert box.center_x == 0 assert box.center_y == 0 assert box.center_z == 0
And obviously it can compute the object width, depth and height (which is easy enough in the case of a Cube
, since they are directly declared when the Cube
is created):
assert box.width == 20 assert box.depth == 30 assert box.height == 50
MuSCAD is able to calculate the bounding box of all primitives, as well as the results of boolean operations or transformed objects (with some limitations):
assert Cube(10, 20, 30).up(15).bottom == 0 assert ( Cube(10, 20, 30).leftward(5) + Cube(5, 40, 10).up(40) ).height == 60
- The limitations are as follow:
- For rotated objects, MuSCAD is only able to compute the bounding box on an axis rotated by a multiple of 90°.
- For Differences, the bounding box is that of the first Object (the base object from which all other objects are substracted from), even if a substracted object actually reduces that box.
- For Intersections, the bounding box is the intersection of the bounding box of all intersected objects. This won't work accurately from most object shapes, but should be enough for most cases.
Since MuSCAD knows the bounding boxes of all objects it creates, it can also reposition them at absolute coordinates.
That is done using .align()
. Here we create a box and align its left, back and bottom sides to 0:
aligned_box = Cube(10, 10, 10).align(left=0, back=0, bottom=0) assert aligned_box.left == aligned_box.back == aligned_box.bottom == 0 assert aligned_box.right == aligned_box.front == aligned_box.top == 10
As you can expect, alignment is done using a translation in the resulting OpenSCAD code:
translate(v=[5.0, 5.0, 5.0]) box(size=[10, 10, 10], center=true);
You can align an Object
on the left
, right
and center_x
on the X axis, back
, front
and center_y
on the Y axis, and bottom
, top
and center_z
on the Z axis.
Since we can align objects to arbitrary coordinates, and we can get the bounding box coordinates for all objects, we can
also align objects relatively to each other. Here we create a tower of 3 colored cubes:
big_cube = Cube(40, 40, 40).color('blue') medium_cube = ( Cube(30, 30, 30) .color('red') .align(center_x=big_cube.center_x, center_y=big_cube.center_y, bottom=big_cube.top) ) small_cube = ( Cube(20, 20, 20) .color('yellow') .align(center_x=medium_cube.center_x, center_y=medium_cube.center_y, bottom=medium_cube.top) ) print(big_cube + medium_cube + small_cube)
This gives the resulting OpenSCAD code:
union() { color("blue") cube(size=[40, 40, 40], center=true); translate(v=[0.0, 0.0, 35.0]) color("red") cube(size=[30, 30, 30], center=true); translate(v=[0.0, 0.0, 60.0]) color("yellow") cube(size=[20, 20, 20], center=true); }
If you decide later to change the position of the big cube, you only have to change its alignment in the first line of Python code, and the medium and small cube will automatically stay on top of it in the generated OpenSCAD code. That is relative positioning, something that was very hard to do with OpenSCAD, because you had to track the position of objects yourself with variables. MuSCAD does that position tracking for you:
big_cube = Cube(40, 40, 40).color('blue').align(left=10, back=10, bottom=10) # added some alignment for the first cube, the rest of the code is untouched medium_cube = Cube(30, 30, 30).align(center_x=big_cube.center_x, center_y=big_cube.center_y, bottom=big_cube.top).color('red') small_cube = Cube(20, 20, 20).align(center_x=medium_cube.center_x, center_y=medium_cube.center_y, bottom=medium_cube.top).color('yellow') print(big_cube + medium_cube + small_cube) union() { color("blue") translate(v=[30.0, 30.0, 30.0]) box(size=[40, 40, 40], center=true); color("red") translate(v=[30.0, 30.0, 65.0]) box(size=[30, 30, 30], center=true); color("yellow") translate(v=[30.0, 30.0, 90.0]) box(size=[20, 20, 20], center=true); }
Often when aligning parts, you want to take into account the tolerance margin for your printer. An appropriate way to do that is to define your tolerance as a constant named T, and add it or substract it whenever needed. MuSCAD includes a default T which is 0.1 mm. If you need a bigger tolerance somewhere, there is a default TT and TTT values that are 0.2 mm and 0.3 mm respectively.
Also, in order to avoid the OpenSCAD "bug" when 2 surfaces are exactly on the same plane, you might want to offset one by a very small value, called an "Epsilon" (something like 0.01 mm). Define this value as a constant named E and add or substract wherever needed. MuSCAD includes a default E value of 0.02 mm, and a "double epsilon" value EE of 0.04 mm.:
from muscad import E, EE, T, TT, TTT
You can apply OpenSCAD modifiers #, %, * and !, by calling the methods .debug()
, .background()
, .disable()
, .root()
respectively:
debugged_object = Cube(10, 10, 10).debug() background_object = Sphere(10).background() disabled_object = Sphere(20).disable() root_object = Sphere(5).root()
While MuSCAD's Cube
class mimics the OpenSCAD cube
primitive, it is too simple to create useful objects.
MuSCAD introduces a high-level class called Volume
that offers extended possibilities over Cube
.
For a start, you don't have to define the size of a Volume
, you can specify its lower and upper limits on each axis instead:
from muscad import Volume my_volume = Volume(left=2, right=10, back=10, front=20, bottom=-4, top=6) print(my_volume)
translate(v=[6.0, 15.0, 1.0]) cube(size=[8, 10, 10], center=true);
You can also specify the size and one limit on a axis, or the center and the size, or the center and a limit. MuSCAD will extrapolate the rest:
my_volume = Volume( left=2, width=8, # this defines the x axis center_y=15, depth=10, # Y axis center_z=1, top=6) # Z axis print(my_volume)
This gives the same render as before:
translate(v=[6.0, 15.0, 1.0]) cube(size=[8, 10.0, 10], center=true);
Note that if you specify only one limit or a size for an axis, that axis will be centered on 0 by default.
You can fillet or chamfer all edges of a Volume
:
chamfered_cube = ( Volume(width=10, depth=10, height=10) .chamfer_all(1) # apply a chamfer of radius 1 to all edges ) # warning: this produces more than a hundred lines of OpenSCAD code print(chamfered_cube)
You can fillet or chamfer specific edges of a Volume:
filleted_side_cube = ( Volume(width=10, depth=10, height=10) .fillet_height(1, left=True) # this fillets the edges along the height of the Volume, restricted to edges on the left side .fillet_depth(1, left=True) # this fillets the edges along the depth of the Volume, again restricted to edges on the left side ) print(filleted_side_cube) # notice that the rendered cube is filleted only on its left side
As the name suggests, using .fillet_width()
, .fillet_depth()
and .fillet_height()
(and the
matching chamfer_*()
methods) will cut your Volume
along its width, depth or height respectively. All 4 fours edges will be cut, unless you select specific edges using
the boolean parameters left
, right
, back
, front
, bottom
and top
. By combining 2 of those, you
can select the specific edge to cut.
In many situations, instead of cutting the edges of your Volumes, you want to add a fillet to soften an inner edge of a Volume with another part. That's a reverse fillet or reverse chamfer.
Use the available reverse_fillet_<face>()
methods to select the face where reverse fillets will be added,
and if needed, select the sides with the boolean parameters left
, right
, back
, front
, bottom
and top
. Here is an example where a reverse fillet will be added at the back of the red part, to join it better
with the blue part:
blue_part = Volume(width=40, depth=12, height=6).fillet_height().color("blue") red_part = ( Volume( width=15, back=blue_part.front, depth=25, center_z=blue_part.center_z, height=blue_part.height, ) .reverse_fillet_back(4, left=True, right=True) .fillet_height(front=True) .color("red") ) print(blue_part + red_part)
This will create the following object:
Now that you are familiar with the basic syntax of primitives, boolean operations, transformations, and alignment, you might want to create complex objects.
While this is possible with the basic functional syntax, it will soon create some hard-to-read code, and you might want some ways to better structure
you code. MuSCAD offers a Part
class that you can inherit to define Objects in an actual Object-Oriented way :)
Here is a simple example, with another tower of cubes, this time with 3 cubes:
from muscad import Part, Cube class CubeTower(Part): def init(self): self.add_child( Cube(40, 40, 40).color('blue') ) self.add_child( Cube(30, 30, 30).align(bottom=self.top).color('red') ) self.add_child( Cube(20, 20, 20).align(bottom=self.top).color('yellow') ) print(CubeTower())
This gives the same OpenSCAD code as before:
union() { color("blue") cube(size=[40, 40, 40], center=true); color("red") translate(v=[0, 0, 35.0]) cube(size=[30, 30, 30], center=true); color("yellow") translate(v=[0, 0, 65.0]) cube(size=[20, 20, 20], center=true); }
But this Python code doesn't look like much of an improvement, right? More code to do the same thing?
That's because it doesn't use the new possibilities offered by the Part
class. Before introducing those features, let's understand what's going on:
class CubeTower(Part):
This creates a CubeTower class, which inherits MuSCAD Part
class. Next:
def init(self):
This special method (not to be confused with Python constructor __init__()
) is the Part
constructor. It will be executed whenever an object of this
class is instantiated. In this constructor:
self.add_child( Cube(40, 40, 40).color('blue') )
Here we create a blue Cube
of size 40, and we add it as a child of this Part
. The object rendered by a part is made of the sum of its children.
Notice that a Part
is a MuSCAD object, so MuSCAD can always calculate its bounding box. That's what we use when we add the second and third Cubes:
self.add_child( Cube(30, 30, 30).align(bottom=self.top).color('red') ) self.add_child( Cube(20, 20, 20).align(bottom=self.top).color('yellow') )
We align that second Cube
to the current Part top, which is, before the second Cube is added, the top of the first Cube. Once that second Cube is added,
that top "moves up" to include the second Cube, so the third Cube is added on top of it.
Since Cubes are created centered by default, the 3 cubes center_x and center_y are aligned to 0.
Finally, once we are done with the CubeTower class definition, we instantiate it and print the resulting OpenSCAD code:
print(CubeTower())
Note that a instantiated Part is a MuSCAD Object
, so the bounding box can still be calculated:
tower = CubeTower() assert tower.top == 70 assert tower.center_x = 0
And the Part can be applied transformations or be used in boolean operations:
assert tower.align(top=0).top == 0 print(tower - Cylinder(d=2, h=tower.height))
The init()
method is executed when you instantiate a Part. You can add parameters to the init()
method,
which will then be required when instantiating the Part
:
class ParametricCubeTower(Part): def init(self, biggest_size, narrowing=10, colors=('blue', 'red', 'yellow', 'green')): size = biggest_size for i, size in enumerate(range(biggest_size, 0, -narrowing)): color = colors[i%len(colors)] self.add_child( Cube(size, size, size) .color(color) .align(bottom=self.bottom) )
Now we start to benefit from Python. The tower of cube will be made from a variable number of cubes depending on the arguments passed at instantiation.
So rendering the following code (using the ParametricCube
class defined above):
print(ParametricCubeTower(60))
will give:
union() { color("blue") cube(size=[60, 60, 60], center=true); color("red") cube(size=[50, 50, 50], center=true); color("yellow") cube(size=[40, 40, 40], center=true); color("green") cube(size=[30, 30, 30], center=true); color("blue") cube(size=[20, 20, 20], center=true); color("red") cube(size=[10, 10, 10], center=true); }
But wait! There is more benefits from the Part
class.
Defining children in the init()
method works well for parametrable children, but it is cumbersome when you have to deal with a lot of static,
non-parametrable children. Also, because they don't use any variable parameters, and because you have to call .add_child()
everytime, you end with a lot
of code noise. To avoid that, you can specify those children as class-level attributes instead.
Let's rewrite our Cube Tower this way:
class StaticCubeTower(Part): big_cube = Cube(40, 40, 40).color('blue') medium_cube = Cube(30, 30, 30).color('red').align(bottom=big_cube.top) small_cube = Cube(20, 20, 20).color('yellow').align(bottom=medium_cube.top)
Since you define class attributes, you have to give each a name. This has 2 additional benefits:
- you can use previously defined child attributes to align new ones. That's what we do when we align the
medium_cube
andsmall_cube
on top of thebig_cube
andmedium_cube
respectively. That is more explicit than relying on the implicit current bounding box like we did before withself
.- the attribute name will be part of the render, as comment, making it easier to find which MuSCAD code produces which OpenSCAD code.
Let's check those comments:
print(StaticCubeTower())
gives:
union() { // big_cube color("blue") cube(size=[40, 40, 40], center=true); // medium_cube translate(v=[0, 0, 35.0]) color("red") cube(size=[30, 30, 30], center=true); // small_cube translate(v=[0, 0, 60.0]) color("yellow") cube(size=[20, 20, 20], center=true); }
So you should model Parts by defining first the most constrained children, then designing other child around them.
All class-level attributes that are MuSCAD objects will be automatically added as children once an object of this class is instantiated,
and before the init()
method is executed. To avoid that, you can prefix the attribute name with an underscore. This will create a non-rendered child.
You can add some objects to a Part as a class attribute that will not be rendered, but that you can use them to align with some other children. To do that, simply prefix its name by a _ like this
class MyPart(Part): _nonrendered = Cube(10, 10, 10) rendered = Sphere(10).align(bottom=_nonrendered.top) print(MyPart())
Note that any class-level attributes that are not Object
, Holes
or Misc
will not be part of the render. You can use them as class-level
variables.
Often, when defining an object, its "holes" are more important than its actual filled matter. You might needs those holes for screws, for extrusions, belts, etc.
When you create such an hole, you don't want it to be accidentally filled again when adding more children later. To make things easier, a Part
also has a list of holes, that are guaranteed to never be filled
by another child. To add such an hole to a Part, simply call add_hole()
with the object to remove as parameter.
Here is an example, creating 2 crossed pipes:
class CrossedPipes(Part): def init(self): self.add_child(Cylinder(d=20, h=30)) self.add_hole(Cylinder(d=15, h=31)) self.add_child(Cylinder(d=20, h=30).x_rotate(90)) self.add_hole(Cylinder(d=15, h=31).x_rotate(90))
For this simple non-parametric example, you can also define holes as class-level attributes, by prefixing their definition by the operator ~
:
class CrossedPipes(Part): vertical_outer_pipe = Cylinder(d=20, h=30) vertical_inner_pipe = ~Cylinder(d=15, h=31) horizontal_outer_pipe = Cylinder(d=20, h=30).x_rotate(90) horizontal_inner_pipe = ~Cylinder(d=15, h=31).x_rotate(90)
Having MuSCAD calculate automatically the dimensions of complex parts is great, but sometimes, there are some Parts that have some miscellaneous features,
that you don't want to be taken into consideration when aligning against other parts.
For example, a stepper motor has a shaft protuding on one of its side, but ignoring that shaft makes it easier to align the motor where you want it to be.
To create such a misc child, call add_misc()
. Here is very simplified Nema17 stepper motor, with the shaft on the top:
class StepperMotor(Part): def init(self): self.add_child(Cube(42, 42, 50)) # this is the body self.add_misc(Cylinder(d=5, h=25).align(bottom=self.top)) # this is the shaft
It is also possible to declare misc children as class-attributes, by calling .misc()
on the attribute value:
class StepperMotor(Part): body = Cube(42, 42, 50) shaft = Cylinder(d=5, h=25).align(bottom=body.top).misc() # this is a misc item
Let's instantiate it and check its sizes on all axis:
stepper = StepperMotor() assert stepper.width == 42 assert stepper.depth == 42 assert stepper.height == 50
As you can see, only the body (which is the only "real", as in "non-misc" child of that Part) determines the size of our simple stepper motor. All misc objects are ignored.
This simple stepper is pretty much useless since it does not include details such as screws. Obviously, you will never want to
print a StepperMotor, you will use a real stepper that actually works instead. But you might want a stepper model anyway, that you can use
when designing some parts, to align other parts against, or to use as a Hole in another part, such as a stepper attachment. If that stepper model would include all details
such as screws, bulge, etc. , the fixation holes would be automatically created in whatever object you align on the fixing holes.
Luckily, MuSCAD includes a much more detailed StepperMotor
Part in its vitamins (see below). Let's use it to create a very simple
stepper attachment:
from muscad.vitamins.steppers import StepperMotor class StepperAttachment(Part): stepper = ~StepperMotor() # let's create the stepper as a hole attachment = Cube(80, stepper.width, 10).align( center_x=stepper.center_x, center_y=stepper.center_y, bottom=stepper.top ) # that's the simplest attachment you can imagine
Notice that the StepperMotor is a hole since it is prefixed with the operator ~
. Note that the order in which normal
children and holes are declared does not matter for the rendering, it only matters to allow you
to properly align those components together.
To ease the creation of surfaces that you can extrude, you can use the various methods provided by Surface
.
With OpenSCAD, you can only do a linear extrusion on the Z axis (the part "growing" upwards by a given distance). You then have to rotate and align the resulting part manually, which is quite cumbersome.
With MuSCAD, you can easily do linear extrusions on all 3 axis by using the methods .x_linear_extrude()
, .y_linear_extrude()
, and .z_linear_extrude()
. They will take care of rotating and aligning your part automatically.
Those methods accept 4 parameters: an extrusion distance
, a start (left
, back
or bottom
), and end (right
, front
or top
) and a center (center_x
, center_y
or center_z
).
You only have to provide 2 of those, and MuSCAD will extrapolate the rest:
spade = ( Union( Surface.free( Circle(d=20).align(right=0, back=5), Circle(d=2).align(center_x=0, front=35), ), Surface.free( Circle(d=2).align(right=5, back=0), Square(1, 15).align(left=0, back=0), ), ) .x_mirror(keep=True) .y_linear_extrude(10, center_y=3) )
MuSCAD includes a library of models for parts commonly used in 3D printing: bolts and nuts, stepper motors, bearings, fans, pulleys, etc.
You can have a look at muscad.vitamins.*
to use and import them.
Most of those models are parametric, so you can have nuts and bolts of any size you like. When there are standardised sizes for some parts,
you can instantiate them via class level factories.
Check out the examples to see how they are used.
Check out the examples here. Examples include a 3D Printer model that I designed from scratch, inspired by the popular HyperCube Evolution, but with a Dual-Wire XY Gantry. There are a few parts I designed to fix things around the house as well. Feel free to check it out, test it, improve it and share your findings!
Here is a moderately complex example, demonstrating all the features exposed above. This is a feet for 3030 extrusion based printers such as the HyperCube Evolution:
from muscad import Part, Sphere, Volume from muscad.vitamins.bolts import Bolt from muscad.vitamins.brackets import CastBracket from muscad.vitamins.extrusions import Extrusion class Feet(Part): # the next 3 parts are 3030 extrusion models, positionned on each axis, that represents # the actual extrusions from the printer # the feet parts will be aligned relatively to those extrusions z_extrusion = ~Extrusion.e3030(60).background() y_extrusion = ( ~Extrusion.e3030(60) .bottom_to_front() .align( center_x=z_extrusion.center_x, back=z_extrusion.front, bottom=z_extrusion.bottom, ) .background() ) x_extrusion = ( ~Extrusion.e3030(60) .bottom_to_right() .align( left=z_extrusion.right, center_y=z_extrusion.center_y, bottom=z_extrusion.bottom, ) .background() ) # a cast bracket to maintain the X and Y extrusion together cast_bracket = ( ~CastBracket.bracket3030() .align( left=y_extrusion.right, back=x_extrusion.front, center_z=x_extrusion.center_z, ) .debug() ) # the base of the feet, that touches the X and Y extrusions base = ( Volume( left=z_extrusion.left, right=cast_bracket.right, back=z_extrusion.back, front=cast_bracket.front, top=z_extrusion.bottom, height=6, ) .fillet_height(10) .fillet_height(30, right=True, front=True) ) # bolts to attach the base to the extrusions right_bolt = ~Bolt.M6(10).align( center_x=base.right - 10, center_y=x_extrusion.center_y, center_z=x_extrusion.bottom - 2, ) front_bolt = ~Bolt.M6(10).align( center_x=y_extrusion.center_x, center_y=base.front - 10, center_z=y_extrusion.bottom - 2, ) center_bolt = ~Bolt.M8(20).align( center_x=z_extrusion.center_x, center_y=z_extrusion.center_y, center_z=z_extrusion.bottom - 2, ) # let's make the feet height parametrable def init(self, height=33): self.ball_holder = ( Volume( left=self.z_extrusion.left, width=44, back=self.z_extrusion.back, depth=44, top=self.base.bottom, height=height, ) .fillet_height(10) .fillet_height(20, right=True, front=True) ) # the actual squash ball, making a hole into the feet self.squash_ball = ( ~Sphere(40) .align( center_x=self.ball_holder.center_x, center_y=self.ball_holder.center_y, center_z=self.ball_holder.bottom + 1, ) .debug() ) if __name__ == "__main__": Feet().render_to_file()
Those less than 100 easy-to-read, easy-to-modify lines do render to about 200 hardly-readable, next-to-impossible to modify OpenSCAD lines. Don't try to modify them! MuSCAD is not designed to create OpenSCAD code that is easy to use for humans. It is intended to use only by OpenSCAD.
The rendered feet was parts of the sample images above.
Feel free to play with MuSCAD. For any issue, please open a ticket on GitLab. PR are welcome. Guillaume
MIT
Some class docstrings are derived from the OpenSCAD User Manual, so are available under the Creative Commons Attribution-ShareAlike License.