In this guide, we will show you how to import external (static) assets into Infinigen. This is useful if you want to create scenes with categories of assets that are not covered by Infinigen (say sculptures, etc), or if you just want to add more variety to your scenes with custom assets.
We will get started by downloading the objects we want to import. For this example, we will use a shelf, a leather sofa, and a table from Objaverse. You can use any objects you like, as long as it is one of the following formats:
- .dae
- .abc
- .usd
- .obj
- .ply
- .stl
- .fbx
- .glb
- .gltf
- .blend
I would recommend using .glb or .gltf files since they are the most extensively tested.
-
Download the following assets in .glb format from Sketchfab:
-
Iron Shelf (Attributed to Thunder and is licensed under CC BY-SA 4.0. We use this asset with modification, i.e. rendered image).
-
Office Couch (Attributed to Virenimation and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Table (Attributed to DeCloud and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
- Create a folder for each category you want to import in
infinigen/assets/static_assets/source
. That is, create the foldersinfinigen/assets/static_assets/Shelf
,infinigen/assets/static_assets/Sofa
, andinfinigen/assets/static_assets/Table
.
mkdir infinigen/assets/static_assets/source
mkdir infinigen/assets/static_assets/source/Shelf
mkdir infinigen/assets/static_assets/source/Sofa
mkdir infinigen/assets/static_assets/source/Table
- Place the downloaded .glb files in the corresponding folders. For this example, you should have three folders with one .glb file each.
Now, whenever we tell Infinigen to import a static Shelf asset in the scene, it will look for the .glb file in the Shelf folder we created. If you have multiple objects within the same category, Infinigen will randomly choose one of them each time that it wants to place in the scene. You can also put objects of different formats in the same folder: shelf1.glb, shelf2.obj, shelf3.fbx
, etc.
NOTE: Objaverse supports downloads using python API if you want to download a large number of objects.
Now, we just need to define Static{CategoryName}Factory = static_category_factory(infinigen/assets/static_assets/source/{CategoryName})
. Add the following lines to infinigen/assets/static_assets/static_category.py
(This step is already done for this tutorial):
Add Static{CategoryName}Factory
to infinigen/assets/static_assets/__init__.py
so that we can import it later. Again, this step is already done for our example:
IMPORTANT: You must make sure that
-
The objects are a reasonable size. This is because there is no way for Infinigen to infer what the sizes of the objects should be, so it will use the sizes of external objects by default. If the objects are too big or too small, they will look out of place in the scene or Infinigen will fail to place them. To prevent this, you should specify what you want the dimensions to be in meters as we did with
z_dim = 2
above. You should specify only one dimension, and Infinigen will scale the object to satisfy that. -
The front of the object is facing +x, the back is facing -x, the bottom is facing -z, and the top is facing +z. You can do this by passing
rotation_euler = (x,y,z)
in Euler angles tostatic_category_factory
. This information is important, for instance, when placing a sofa against the wall. -
Make sure that
CategoryName
matches the name of the folder you created in thestatic_assets/source
folder.
Here's an example of bad dimensions and orientation (left image); and good dimensions and orientation (right image):
If you want to add more categories, just add more lines with {CategoryName}
as the name of the category you want to import.
Infinigen allows the user to specify high-level semantics for the objects in the scene. These semantics are then used to define high-level constraints. For example, we want to say that our static shelf factory is a type of storage unit, which will be placed against the wall, and there will be a bunch of objects on top of it. In general, if you want your static object factory to be treated like an existing asset factory, you can just imitate the semantics of the existing asset factory. Let's demonstrate this idea by defining semantics for our static shelf. We go to infinigen_examples/constraints/semantics.py
and search for LargeShelfFactory
. We see that it is used as Semantics.Storage
and Semantics.AssetPlaceholderForChildren
. We want our static shelf to be used as a storage unit as well, so we add a line for our new static factory:
Similarly, we add StaticShelfFactory
to Semantics.AssetPlaceholderForChildren
. This will replace the placeholder bounding box for the shelf before placing the small objects.
The semantics for the sofa and the table are analogous. We just define the same semantics as the existing asset factories that are similar to our static ones:
If your category is not similar to any existing category, you would need to think about it a little bit more and define your own semantics. For example, I found that the sofa semantics work well for vending machine as they are both placed against the wall, etc.
The last step is to add constraints for our static assets. We want to make sure that the shelf is placed against the wall, the sofa is placed in the living room, and the table is placed in the dining room, etc. Luckily our new static assets are similar to existing assets, so we can just replace the existing constraints with our new static assets. We go to infinigen_examples/indoor_constraint_examples.py
and search for LargeShelfFactory
, SofaFactory
, and TableDiningFactory
. We just replace these constraints with our new static assets:
If you have some new asset that does not behave like any of the existing assets, you would need to define new constraints.
And, that's it! You can now generate a scene with your new static assets. We just use the regular Infinigen commands:
python -m infinigen_examples.generate_indoors --seed 0 --task coarse --output_folder outputs/indoors/coarse0 -g fast_solve.gin singleroom.gin -p compose_indoors.terrain_enabled=False restrict_solving.restrict_parent_rooms=\[\"LivingRoom\"\]
python -m infinigen_examples.generate_indoors --seed 1 --task coarse --output_folder outputs/indoors/coarse1 -g fast_solve.gin singleroom.gin -p compose_indoors.terrain_enabled=False restrict_solving.restrict_parent_rooms=\[\"LivingRoom\"\]
python -m infinigen_examples.generate_indoors --seed 11 --task coarse --output_folder outputs/indoors/coarse0dining -g fast_solve.gin singleroom.gin -p compose_indoors.terrain_enabled=False restrict_solving.restrict_parent_rooms=\[\"DiningRoom\"\]
You can see that our new static assets are placed in the scene. What's more is that they interact with the existing Infinigen assets. For example, we have a bowl of fruit and a glass on top of the table.
If you want to post-process the imported static objects or customize the object creation, you can define your own static category class similar to StaticCategoryFactory
in infinigen/assets/static_assets/static_category.py
. For instance, you can add a finalize_assets(self, assets)
function to your class to post-process the imported assets.
A shelf should have small objects placed on it. But, how can Infinigen know where to place these objects? The default is that the top surface of the object is used as the surface for placing objects. For instance, in the above example, we saw that some objects were automatically placed on the table even though we did not specify where to place them. A shelf has multiple surfaces, so we need to tag these surfaces as so-called "support surfaces". This is very easy to do in the case where the support surfaces are the distinct planes along the z direction, as is the case with shelves. We just enable the option StaticShelfFactory = static_category_factory("Shelf", tag_support=True)
in infinigen/assets/static_assets/static_category.py
:
Now when we generate the scene, we see that the objects are placed on all surfaces of the shelf:
Let us summarize all the steps you would need to follow to import arbitrary objects into Infinigen:
-
Download the objects in one of the supported formats.
-
Create a folder for this object category in
infinigen/assets/static_assets/source
. E.g.
mkdir infinigen/assets/static_assets/source/MyCategory
-
Place the downloaded objects in the folder you created.
-
Add a line in
infinigen/assets/static_assets/static_category.py
to define the factory for this category. E.g.
StaticMyCategoryFactory = static_category_factory("infinigen/assets/static_assets/source/MyCategory")
-
Add a line in
infinigen/assets/static_assets/__init__.py
to import the factory from other files. -
Define the semantics for the objects in
infinigen_examples/constraints/semantics.py
. E.g.
used_as[Semantics.Furniture] = {...
static_assets.StaticMyCategoryFactory}
- Define the constraints for the objects in
infinigen_examples/indoor_constraint_examples.py
. E.g.
my_cat_against_wall = wallfurn[static_assets.StaticMyCategoryFactory]
...
- Generate the scene using the regular Infinigen commands.
Now that you know how to import static assets, you can import all kinds of different objects. Here are some creative examples:
The external assets used here are:
-
St Olaf (Attributed to Historiska and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Dusty Bookshelves (Attributed to Meanphrog and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Minotaur Statue (Attributed to plasmaernst and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Dusty Piano (Attributed to Vincent074 and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Divergence Meter (Steins;Gate) (Attributed to Amatsukast and is licensed under CC BY-NC-SA 4.0. We use this asset with modification, i.e. rendered image).
-
Stone Griffin (Attributed to Thomas Flynn and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
Hologram Console (Attributed to TooManyDemons and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).
-
vending machine (Attributed to wilsz95 and is licensed under CC BY 4.0. We use this asset with modification, i.e. rendered image).