Geometry Nodes Modifier
The "Geometry Nodes Modifier", or "Nodes Modifier" for short, is a a modifier to handle more complex behavior. Its logic is built with a node group owned by the modifier. The geometry node group which can be used by multiple modifiers on different objects, or shared for different projects, just like a shader node group. High level settings are exposed in the modifier stack.
Any of the existent modifiers can be ported to nodes, though they may need small adjustments in a per-case basis.
Geometry Node Editor
The modifier's node group is edited in the node editor. The editor is context sensitive, showing the modifier node group of the active modifier of the active object.
High Level Abstraction
Modifiers are black boxes with geometry as the main input and output. External dependencies are possible on ID level. Users should be able to use the system in a high level. More general nodes contain several lower level building block nodes, with certain parameters exposed.
The building blocks of the system are kept inside the main node groups, abstracted away but tweak-able for advanced usage. Properties connected to the node group input are exposed in the modifier stack.
Inputs and Outputs
The modifier input and output are explicit nodes that represent the flow of geometry into and out of the modifier. The modifier must have a geometry input and a geometry output
The high level nodes operate with a clear flow of data. This means that each node finishes its calculation independently, then passes the resulting data to its output nodes.
The geometry socket is used to pass around multiple types of data
- Point cloud
- Instances data
And in the future:
- Volume (already in Blender)
Those data types, as well as the mesh objects can be passed around in the nodetree via the "Geometry" socket. It is possible for the system to handle implicit conversion, but there are explicit settings required in some cases. Often, the simulation effects require a different input and output geometry types. For example, in this image a spray node gets the nozzle as input, and outputs foam particles. (The icons represent the data type output.)
Different Socket Types for Geometry Types?
One question that comes up a lot is why one socket is used for all geometry data types, instead of a separate for each socket type. There are arguments in both directions, but in general a single socket makes much more sense.
Pro Multiple Socket Types
- One knows exactly what geometry type one is working with at any point.
- Transformation of one data type to another is displayed visually.
Pro Single Socket Type
- Requires fewer socket types (otherwise we might need socket types for mesh, volume, curve, text, metaballs, point clouds, hair, grease pencil).
- Some nodes can work on multiple different geometry types (e.g. Transform). Those would have to exist multiple times or have some kind of dynamic input.
- The geometry node group used by a modifier can have exactly one output, no need for one output per type.
- When building a node group that displaces points, one probably does not care whether the points belong to a mesh, curve, hair or whatever. The group could have a point cloud input, but then all the connectivity data would be lost after the displacement.
- Most other node-based software uses a single socket type for this purpose.
- A common workflow is to first build smaller objects of various types, which are then joined into bigger objects and so on. These bigger objects are mostly only transformed and instanced. At that point there is no need to distinguish between different geometry types anymore. You wouldn't want to copy Transform nodes multiple times to transform all components the same way. A single Transform node in the node tree should be enough.
Some of the disadvantages of having multiple socket types can be counteracted by introducing yet another "generic geometry" socket type that can be anything and can be used on nodes that support multiple types. But there is more complexity to deal with: What if a node only supports a subset of all geometry types (e.g. shrinkwrap), do we use a different type for the different combinations? Also, how does instancing fit into the design when there are multiple socket types? Is there a separate "Instances" socket? What happens if you want to make those instances "real"? The type of the instances might not be known statically, and might change over time.
So on balance, it's better to use a single socket type, and make it clear in the documentation and UI which data types each node supports.
An important part of the pipeline is to assign attributes to the geometry to change how the geometry looks and behaves. Attributes can
There are three kinds of attributes:
- Built-in attributes are attributes whose type and domain is fixed and cannot be changed.
- Name convention attributes are attributes that nodes expect to operate on. For example, the
Point Instance nodeexpects its input to have
- User defined attributes are created either in the source data (e.g., mesh), such as UV maps, or during evaluation, in the node tree.
The existing attributes, in order of least complex to most complex:
- Boolean (
- Integer (
- Float (
- 2D Float Vector (
- 3D Float Vector (
- Color (
Implicit conversions between data types are defined in
Built-in and name convention attributes use "snake_case", while user defined attributes are initialized as "PascalCase", but can be renamed later to anything (unicodes, space, lowercase).
For all types of attributes, abbreviations are to be avoided. If they eventually happen, they should be lower case for the built-in attributes. For user defined attributes they can be decided on a per-case basis.
Why use "names" for attribute sockets?
Attributes are stored directly in each geometry component. Because different geometries can have different size, different topology, and potentially different domains, an attribute generally only has has meaning when in combination with the corresponding geometry. This means that they aren't passed around separately from their geometry, they are ("will be" as of writing) passed between geometries with dedicated nodes.
The attribute sockets are string sockets, used to lookup the list of data with that name stored in the geometry.
There are many ways to create instances in Blender:
- Collection instances
- Child instancing
- Instances created by the "Point Instance" node.
- The "Object Info" and "Collection Info" node actually output instances instead of copied geometry.
All of these instances are treated as real geometry in the node tree:
- Instances are real on demand. This is also known as copy-on-write. This improves performance (because copying data can be skipped where possible) while keeping all the functionality of real geometry.
- When we support attribute debugging (spreadsheet), and socket inspections users will be able to see at which moment the instances are no longer treated as instances.
That means that once a geometry is instanced with a Point Instance node, for example, it is treated as a single object with all the instances made real and joined.
Geometry can be "made real" with the function
geometry_set_realize_instances. However, any node that just creates data based on the input should not use this function, and instead acquire a read-only (
const) version of the input data, and output new data based on that.
For events there is a need for a different representation for its callbacks.
Conversion vs Emission
- Conversion tries to convert from different data types close to 1:1
- Emission requires distribution functions + maps
The simulation context is defined by the current view layer. If the simulation needs to be instanced in a different scene it has to be baked to preserve its original context.
Baking vs Caching
- Baking overrides the modifier evaluation (has no dependencies)
- Caching happens dynamically for playback performance
Interaction and physics
- Physics needs its own clock, detached from animation playback
- There is no concept of (absolute) “frames” inside a simulation
The solver node requires new kind of input, the influences.
The geometry is passed to the solver as the initial points (for the particles solver). The solver reaches out to its influences that can:
- Create or delete geometry
- Update settings (e.g., color)
- Execute operations on events such as on collision, on birth, on death.
In this example the callbacks are represented as vertical lines, while the geometry dataflow is horizontal.
The emitter node generates geometry in the simulation. It can receive its own set of influences that will operate locally. For instance, an artist can setup gravity to only affect a sub-set of particles.
Historical Design Documents
- Particle Workshop - September 2020
- Everything Nodes
- Initial Functions System
- Mesh Type Requirements
- Breaking up the Modifier Stack
- Simulation Architecture Proposal
- Simulation Framework - First Steps
- Node Interface Framework
- Representing Particle Systems with Nodes
- Particle System Code Architecture
- Updated Particle Nodes UI
- Updated Particle Nodes UI 2
- Particle Nodes Core Concepts
- Unified Simulation System Proposal
- Function System