Reference
Changelog
v0.5.0
- (fix) Fix inconsistent axes colors in the editor. Now, +X is red, +Y is green, and +Z is blue across both the navigation widget and the grid lines.
- Respect
.gitignore
when uploading simulation artifacts for monte carlo runs. Also, ignore hidden files and directories.
v0.4.1
- Add local cache for remote GLB assets to improve model load times and reduce network usage.
- Add more built-in templates for creating new simulations (
ball
anddrone
).
v0.4.0
- (fix) Fix cutoff titlebar in editor on browser and windows.
- (fix) Fix bug in the drone example’s rate controller where the derivative term wasn’t filtered correctly (https://github.com/elodin-sys/elodin/issues/18). It effectively caused the integral gain to amplified and the derivative gain to be suppressed.
- (breaking) Replace
el.Time
andel.advance_time
withel.SimulationTick
andel.SimulationTimeStep
.- The simulation tick is automatically advanced by a built-in system.
- The simulation time step is set to the same value as the “sim_time_step” provided in
World.run
.
- Add a built-in
SystemGlobals
archetype that is automatically spawned for every simulation.- This archetype is only associated with a single managed entity and contains global variables that can be accessed by any system.
- Currently, it contains
el.SimulationTick
andel.SimulationTimeStep
which are automatically set to the current tick and simulation time step respectively.
- Add icon for the editor on windows.
- Add support for panning and zooming plots in the editor.
- Enable vsync on Windows and Linux.
- Don’t throttle FPS when cursor is not moving on Windows and Linux.
v0.3.30
- (fix) Fix issue where
edge_fold
would not filter out edges where the “right” entity doesn’t match the “right” query.
v0.3.29
- Support adding
el.SpatialMotion
toel.SpatialTransform
. - Add body-frame quaternion integration to
el.Quaternion
viaintegrate_body
.
v0.3.28
- Various bug fixes.
v0.3.27
- (fix) Set the correct type information for the return value of
el.Quaternion
__matmul__
. Previously, it would always return ajax.BatchTracer
even if a spatial type was being multiplied. - (fix) Fix issue where CLI is unable to verify the license key.
v0.3.26
- Improve simulation startup times.
v0.3.25
- (breaking) Use variable positional arguments for
el.Panel.vsplit
,el.Panel.hsplit
,el.Panel.graph
, andel.GraphEntity
instead of lists. This results in much less verbose panel configuration code. To update your code, either replace the list of arguments with individual arguments separated by commas or unpack the list with*
.# before: world.spawn( el.Panel.hsplit( [ el.Panel.vsplit( [ el.Panel.graph( [ el.GraphEntity( sat, [ el.Component.index(SunRef), ], ) ] ), ] ), ] ), ) world.spawn( el.Panel.graph( [el.GraphEntity(b, el.Component.index(el.WorldPos)[4:]) for b in balls] ), ) # after: world.spawn( el.Panel.hsplit( el.Panel.vsplit( el.Panel.graph(el.GraphEntity(sat, el.Component.index(SunRef))), ), ), ) world.spawn( el.Panel.graph( *[el.GraphEntity(b, *el.Component.index(el.WorldPos)[4:]) for b in balls] ), )
- (fix): Fix editor crash on Windows due to:
Requested alpha mode PostMultiplied is not in the list of supported alpha modes: [Opaque]
GraphEntity
constructor now accepts component classes directly instead ofel.Component.index(ComponentClass)
. However,el.Component.index
is still needed for slicing and indexing into individual elements of a component.# before: el.GraphEntity(sat, el.Component.index(SunRef)) # after: el.GraphEntity(sat, SunRef)
- Add better default names for viewports and graphs. E.g. Track: “<entity name>” for viewports and “<entity name>: <component name>” for graphs.
- Add basic support for 3D plots/traces in the editor.
- Trace an entity’s position by spawning in
elodin.Line3d
assets:# The entity id is required as the first positional argument. world.spawn(el.Line3d(rocket)) # You can specify the color and width of the line. w.spawn(el.Line3d(rocket, line_width=5.0, color=el.Color(1.0, 0.0, 0.0))) # You can enable or disable perspective rendering, which makes the line scale with distance to the camera. This is enabled by default. w.spawn(el.Line3d(rocket, perspective=False))
- Trace an entity’s position by spawning in
v0.3.24
- Decouple simulation and playback running. You can now pause and rewind the editor without pausing the simulation. You can also change the playback speed by using
output_time_step
onWorldBuilder.run
. We are deprecating thetime_step
parameter and replacing it withsim_time_step
. This is to disambiguate it withrun_time_step
, which allows you to control the amount of time elodin waits between ticks of the simulation. Settingrun_time_step
to0.0
effectively lets you simulate maximum speed. - Add
max_ticks
parameter toWorldBuilder.run
. The simulation will run until the specified number of ticks is reached. - Add body frame visualization option.
To try it out, either open the command palette and type
Toggle Body Axes
or add the following code to your simulation file:w.spawn(el.BodyAxes(entity, scale=1.0))
- Add the ability to create graphs and viewports entirely from the command palette. Try spawning a new graph by typing
Create Graph
, to see the new workflow. - (fix) If there were pytest failures in a Monte Carlo sample, the sample would still be reported as a pass. This is now fixed.
- (fix) Fix line pixelation in long-running plots.
- Add
SpatialTransform()
constructor that replacesSpatialTransform.zero()
,SpatialTransform.from_linear(linear)
, andSpatialTransform.from_angular(quaternion)
by making use of optional arguments. The old methods are still available but deprecated.# before world_pos = el.SpatialTransform.from_linear(jnp.array([0.0, 0.0, 1.0])) + el.SpatialTransform.from_angular(euler_to_quat(jnp.array([0.0, 70.0, 0.0]))) # after world_pos = el.SpatialTransform(linear=jnp.array([0.0, 0.0, 1.0]), angular=euler_to_quat(jnp.array([0.0, 70.0, 0.0]))) # before world_pos = el.SpatialTransform.zero() # after world_pos = el.SpatialTransform()
- Add
SpatialForce()
constructor that replacesSpatialForce.zero()
,SpatialForce.from_linear(linear)
, andSpatialForce.from_torque(torque)
by making use of optional arguments. The old methods are still available but deprecated.# before force = el.SpatialForce.from_linear(jnp.array([0.0, 0.0, 1.0])) + el.SpatialForce.from_torque(jnp.array([0.0, 70.0, 0.0])) # after force = el.SpatialForce(linear=jnp.array([0.0, 0.0, 1.0]), torque=jnp.array([0.0, 70.0, 0.0])) # before force = el.SpatialForce.zero() # after force = el.SpatialForce()
- Deprecate
SpatialTransform.from_axis_angle(axis, angle)
in favor ofSpatialTransform(angular=Quaternion.from_axis_angle(axis, angle))
. - Deprecate
SpatialMotion.from_linear(linear)
andSpatialMotion.from_angular(angular)
in favor ofSpatialMotion(linear=linear, angular=angular)
. - Add filtered search for entities and components.
- Always include zero as a tick in the y-axis of plots.
- Replace “Welcome” panel with a new UI for creating viewports and graphs.
- Add “Save Replay” command to command palette (Cmd + P).
- Show progress bar when executing
exec.run(ticks)
unless explicitly disabled withexec.run(ticks, show_progress=False)
. - Add a create command for templates to the Elodin CLI
- Allowing naming of viewports and graphs from code.
el.Panel.graph( [ el.GraphEntity(drone, [ el.Component.index(mekf.AttEstError) ]) ], name="Attitude Estimation Error", )
v0.3.23
- (breaking) Render the Z-axis as up in the editor (instead of the Y-axis). This is a purely visual change and does not affect simulation results, but it’s recommended to update simulations to match the new visual orientation. Typically, this requires swapping the Y and Z axes when operating on spatial positions, velocities, and forces.
- (fix) When a simulation file was changed, the associated pycache files would also be updated, causing the simulation to be re-built multiple times in some cases. This is now fixed.
- Add Status Bar to the editor (currently shows FPS/TPS and basic version of the connection status).
elodin editor <path/to/sim>
now watches the parent directory of the simulation file for changes in addition to the file itself. This is useful for multi-file projects. This is also the case when using the--watch
flag directly. E.g.python <path/to/sim> run --watch
.- Export simulation data to a directory using
exec.write_to_dir(path)
.
v0.3.22
- (fix) Fix errors when using
vmap
withscan
, and non-scalar values- When the arguments to a scan operation were non-scalar values (i.e their rank was above 0), scan would error in various ways when combined with vmap. The core issue is that some of our logic accidentally assumed an empty shape array, and when that array was non-empty, dimensions would be inserted into the wrong place.
v0.3.21
- (fix) Fix missing 1/2 factor in angular velocity integration and
Quaternion::from_axis_angle
.- In
nox
, the constantField::two
returned a1
constant. This constant was only used in the implementation ofAdd
betweenSpatialMotion
andSpatialTransform
and inQuaternion::from_axis_angle
. Unfortunately, this caused angular velocity integration to return incorrect results. This bug caused the applied angular velocity to be multiplied by a factor of 2. - The most significant impact of this bug is on the stability of any attitude control system. This bug has led to an increase in small oscillations, potentially affecting the performance of PID controllers tuned to work with previous versions of Elodin. PID controllers tuned to work with earlier versions of Elodin will likely need to be re-tuned
- We regret not catching this bug earlier. To prevent a bug like this happening ever again, we have taken the following steps:
- We created a set of unit tests for the 6 DOF implementation that compare Elodin’s results with Simulink. They have confirmed that our implementation now matched Simulink’s 6DOF Quaternion block.
- We will expand our set of unit tests to have 100% coverage of our math modules. We will test each module against Simulink and other trusted implementations.
- We will publish documentation on our testing methodology and reports for each module.
- In
- (fix) Fix incorrent angular acceleration due to mismatched coordinate frames.
- Spatial force and motion are defined in the world frame. The inertia tensor, however, is defined in the body frame. To correctly calculate angular acceleration, torque and inertia tensor must be in the same frame. To fix this, we now transform torque from the world frame to the body frame before calculating angular acceleration. Then, we transform the angular acceleration back to the world frame.
- (breaking) Split
el.Pbr
intoel.Mesh
,el.Material
,el.Shape
, andel.Glb
.- Use the
el.shape
andel.glb
helpers instead# before: w.spawn(el.Body(pbr = w.insert_asset( el.Pbr(el.Mesh.sphere(0.2), el.Material.color(0.0, 10.0, 10.0))))) w.spawn(el.Body(w.insert_asset(el.Pbr.from_url("https://storage.googleapis.com/elodin-assets/earth.glb")))) # after w.spawn([ el.Body(), w.shape(el.Mesh.sphere(0.2), el.Material.color(0.0, 10.0, 10.0)) ]) w.spawn([ el.Body(), w.glb("https://storage.googleapis.com/elodin-assets/earth.glb") ])
- Use the
- (breaking) Remove
SpatialInertia.from_mass()
.- Use the
SpatialInertia()
constructor instead, which now accepts inertia tensor as an optional keyword argument.# before: inertia=el.SpatialInertia.from_mass(1.0 / G), # after: inertia=el.SpatialInertia(1.0 / G),
- Use the
- If a
Component
base class providesComponentType
information via__metadata__
, then it can be omitted from theComponent(...)
annotation as it can be inferred from the base class instead.Quaternion
,Handle
,Edge
, and all spatial classes (SpatialTransform
,SpatialForce
,SpatialMotion
,SpatialInertia
) have been updated to provideComponentType
information directly. So, components that annotate these classes can now be defined more simply:# before: ControlForce = Annotated[ el.SpatialForce, el.Component("control_force", el.ComponentType.SpatialMotionF64) ] # after: ControlForce = Annotated[ el.SpatialForce, el.Component("control_force") ]
- Add visualization of the current tick in the graph views of the editor.
- Add
TotalEdge
, an edge type that connects every entity to every other entity. - Fix issue where stepping backwards or forwards in the editor would sometimes not work.
- Add Command Palette to the editor
v0.3.20
- (breaking) Replace entity builder pattern with a more flexible
spawn
API.- To migrate to the new API, replace
name()
with the keyword argument:# before: w.spawn(el.Body()).name("entity_name") # after: w.spawn(el.Body(), name="entity_name")
- If the entity has multiple archetypes, just provide a list of archetypes as the first positional argument instead of using
insert()
:# before: w.spawn(el.Body()).name("entity_name").insert(OtherArchetype()) # after: w.spawn([el.Body(), OtherArchetype()], name="entity_name")
- Also,
spawn
now returnsEntityId
directly instead ofEntity
. So, there’s no need to callid()
to reference the entity in an edge component or viewport.a = w.spawn(...) b = w.spawn(...) # before: w.spawn(Rel(el.Edge(a.id(), b.id()))) # after: w.spawn(Rel(el.Edge(a, b)))
- To migrate to the new API, replace
- Add multi-file support for Monte Carlo runs.
- Add ability to use ranges in viewports for replay.
v0.3.19
- (breaking) Allow querying different components from the left and right entities via the new
edge_fold
API.- To migrate to the new API, move the graph query’s input components to a separate query parameter:
And then reference it in
# before: @el.system def gravity( graph: el.GraphQuery[GravityEdge, el.WorldPos, el.Inertia], ) # after: @el.system def gravity( graph: el.GraphQuery[GravityEdge], query: el.Query[el.WorldPos, el.Inertia], )
edge_fold
:# before: return graph.edge_fold(el.Force, ... # after: return graph.edge_fold(query, query, el.Force, ...
- To query different components from the left and right entities, use multiple queries:
@el.system def rw_effector( rw_force: el.GraphQuery[RWEdge], force_query: el.Query[el.Force], rw_query: el.Query[RWForce] ) -> el.Query[el.Force]: return rw_force.edge_fold(force_query, rw_query, el.Force, ...
- To migrate to the new API, move the graph query’s input components to a separate query parameter:
- Make graph colors deterministic.
- Drop milliseconds from the x-axis in graphs.
- Prevent grid from changing origin.
- Add ranges to graphs to allow zooming + panning.
- Add support for spawning graphs and splits from code.
- Switch to GPU based plotting.
- Add configurable line width for graphs.
v0.3.18
- (breaking) Make RK4 the default integrator (can still override to use semi-implicit euler).
- Add element names to graphs.
- Add component priorities.
- Components in inspector are displayed in descending order of priority.
- Priority can be set via metadata. E.g.:
metadata={"element_names": "q0,q1,q2,q3,x,y,z", "priority": 5}
. - Default priority, if unset, is 10.
- Components with priority of < 0 are hidden from the inspector.
- Priority can be set via metadata. E.g.:
- Increase y-axis margin in graphs to prevent axis labels from getting cut off.
- Use scientific notation for large values.
- Prevent graph cursor modal from being cut off.
- Add
interia_diag()
helper to el.SpatialInertia. - Fix graph data interpolation issues.
v0.3.17
- Reduce memory usage of plots.
- Add rocket example with thrust curve interpolation.
- Clear graphs on sim update.
- Have a stable ordering of components in inspector.
- Make decimal point stable in component inspector.
- Add a built-in Time component + system.
- Remember window size on restart.
- Add configurable labels for component elements.
Was this page helpful?