Volume Lattice and Booleans
Intro
In this tutorial, we’ll use the Genysis python library to modify an existing mechanical part, replacing some of its solid material with a strong, yet light weight, uniform lattice structure. Specifically, we’ll be using the boolean function and the volume lattice function.
The full source code for the python library is available on GitHub at:
Prerequisites
In order to follow along with this tutorial, you will need a copy of python/pip installed (either 2.7+ or 3.6+).
You will also need a Genysis API token, which you can request here:
https://studiobitonti.appspot.com/subscription.html
To use the tutorial code, you’ll need to install the Genysis package via pip. At the command line, type:
Before we dive into the code, you’ll need to upload the 3D files needed for the tutorial to your Genysis account. We’ll be using three .obj files, one is the solid connecting rod and the other two are helper shapes that we’ll use for our boolean operations. First download these:
connecting-rod.obj
connecting-rod_difference-tool-1.obj
connecting-rod_intersection-tool-1.obj
Then upload them to your private Genysis storage using our web interface provided here:
https://studiobitonti.appspot.com/
We’ll be working with the volume colored red here.
The full code for this tutorial is included in an embedded Gist at the bottom of the page and also at the following link.
connecting-rod_volume-lattice.py
Ok, ready to make some shapes!
Now I’ll walk through the script explaining each step along the way.
First, we will import the Genysis library and set a variable for our private token, which will be passed to each function for authentication purposes. Also, we’ll define some variables that hold the filenames of the .obj files we uploaded earlier.
We’ll be repeating the same basic process each time we use one of the Genysis functions:
define a filename where the function will save its output in your private Genysis cloud storage
run the Genysis function, passing in:
the filename(s) from some previously uploaded or computed file(s)
the output filename
some other parameters for the function
Now, we are going to use the boolean function with the “difference” operation to cut the middle section out of the connecting rod using our pre-prepared difference volume.
Next, we will use the boolean function again, this time with the “intersection” operation, to generate a conformal volume for our lattice structure.
Next, we need to define a lattice unit, which will be replicated over and over along a grid to create our Volume Lattice. While it is possible to provide a custom mesh to use for this unit’s shape, Genysis provides a handy tool for generating parametric lattice units. Read more about this function in the documentation here.
Genysis refers to these units as components in many of its functions.
Now we are ready to create our Volume Lattice, which produces a lattice on a uniform cubic grid. We’ll create a lattice instance and then call some of its methods to set various parameters. Then, we will “run” the lattice to generate an output file. Lattice functions create a wireframe representation of the lattice, which has no volume and thus is not directly manufacturable. We’ll deal with that in the next step.
Ok, now we have a wireframe representation of the lattice. In order to transform it into a manufacturable part, we need to “mesh” this skeleton, giving it form. To do this we’ll use the marching cubes function, which uses a voxel representation to create a guaranteed manifold mesh.
We have a few choices now. First, we’ll have to choose a resolution for our voxel grid, which will affect both the surface quality (and polygon count!) of our output as well as the time it takes to compute that output. We also need to decide how thick the edges of our lattice will be. These two variables affect each other, so it can take some experimentation to get the right balance of detail, polygon count, and computation time. There are no hard and fast rules here, it really depends on how large each lattice component is, how large your overall lattice volume is, and how densely filled you want your lattice to be.
Previously, we set our component size (aka. the length of one side of a cube unit of our lattice) to 4.0 . I found that for this project a resolution of 600 gives a nice smooth mesh (a higher resolution will take more time and produce a smoother result) using an edge thickness (memberThickness) of 0.4 . A good place to start when experimenting is 300 for resolution and somewhere around 1/8 of your component size for your member thickness. Here’s how that looks in the code:
The marching cubes function is designed to split this computationally heavy process apart, so instead of returning a single .obj file it instead returns a list of .stl files (each no more than 90MB) which can be recombined into a huge, hi-definition lattice. This allows Genysis to compute truely gargantuan lattice structures, but does mean we have to do a little extra work if we want to use the meshed lattice output as an input for other operations. The result from the meshing operation is a list of values that looks something like this:
I’ve written some code to download these .stl files, then clean and combine them into a single .obj file, which we can then upload back to Genysys file storage for future use.
Finally, we’ll use a special secret endpoint that is not yet exposed in the Genysis python library to upload this large file back up to the cloud file storage.
As our last step, we can perform the final boolean union operation, to combine our solid connecting rod minus the cutout section (remember when we did that!) with the finalized lattice mesh.
We’re done making shapes! Now we can use the visualize function to have a look at our file using a browser (large files may have a hard time being displayed this way). We can also use the Genysis download function to easily pull our final file down to our local folder.
Thanks for following along! If you have any questions or get stuck anywhere, reach out to us using our contact page: https://www.genysis.cloud/contact-2/