/SumoNetVis

A library for visualizing a Sumo network and trajectories

Primary LanguagePythonMIT LicenseMIT

SumoNetVis

A Python library for visualizing a Sumo network and trajectories with matplotlib or as an OBJ file.

Network and Trajectory Plotting

Example plot of an intersection with trajectory colored by speed

Highly Customizable

Example plot showing USA and EUR style lane markings

Trajectory Animation

Example animation

Basic trajectory plotting from FCD outputs is built in, but it is also possible to plot custom data and graphics on top of the network with the full flexibility and power of matplotlib and other compatible libraries, such as seaborn.

3D geometry for a network can be generated and saved as a Wavefront-OBJ file. Example_rendering of OBJ export of an intersection

Installation

This package can be installed via pip with the command pip install SumoNetVis. You can then import the library with:

import SumoNetVis

Dependencies

  • shapely (>=1.7.0 for stop lines and OBJ export)
  • triangle (for OBJ terrain export)
  • matplotlib
  • numpy

Usage

To plot a Sumo net file and trajectories, you can use the following code:

import SumoNetVis
import matplotlib.pyplot as plt
# Plot Sumo Network
net = SumoNetVis.Net("path/to/yourfile.net.xml")
net.plot()
# Plot trajectories
trajectories = SumoNetVis.Trajectories("path/to/fcd-output.xml")
trajectories["vehicle_id"].assign_colors_speed()
trajectories["vehicle_id"].plot()
# Show figure
plt.show()

the Net.plot() function takes the following optional parameters:

  • ax: matplotlib Axes object (defaults to currently active Axes)
  • clip_to_limits: if True, only objects visible in the current view extents will be drawn
  • zoom_to_extents: auto-zoom to Net extents (defaults to True)
  • style: lane marking style to use ("USA" or "EUR")
  • stripe_width_scale: scale factor for lane marking widths (defaults to 1)
  • plot_stop_lines: whether to plot stop lines
  • apply_netOffset: whether to translate the network by the inverse of the netOffset value
  • lane_kwargs: dict of kwargs to pass to the lane plotting function (matplotlib.patches.Polygon()), for example alpha
  • lane_marking_kwargs: dict of kwargs to pass to the lane markings plotting function (matplotlib.lines.Line2D())
  • junction_kwargs: dict of kwargs to pass to the junction plotting function (matplotlib.patches.Polygon())

Any kwargs passed directly to Net.plot() will be passed to each of the plotting functions. These will, however, be overridden by any object-type-specific kwargs (lane_kwargs, etc.).

To plot all junctions at 50% opacity and all other objects at 80% opacity, for example, one can use:

net.plot(junction_kwargs={"alpha": 0.5}, alpha=0.8)

The color scheme of junctions and various lane types can be customized by modifying entries in the global variable COLOR_SCHEME. For example, to plot bike lanes as dark green instead of dark red, do the following:

SumoNetVis.COLOR_SCHEME["bicycle"] = "#006600"

Any color specification supported by matplotlib can be given here, such as RGB and RGBA hex strings and float tuples, as well as color names and abbreviations. See the matplotlib documentation for more detailed information.

Trajectory Colorization

When using the various "assign_colors" functions, when applicable, a corresponding matplotlib ScalarMappable will be stored in the Trajectory.mappable property. The Trajectories class also contains a property, mappables, which contains a dict of vehID: ScalarMappable pairs. Therefore, to automatically generate a colorbar, one of the two following (equivalent) methods can be used:

plt.colorbar(trajectories["vehicle_id"].mappable)
# OR
plt.colorbar(trajectories.mappables["vehicle_id"])

Animation

Instead of visualizing Trajectories as lines, an animation can be generated using the matplotlib.animation module.

import matplotlib.animation as animation
trajectories = SumoNetVis.Trajectories("path/to/fcd-output.xml")
fig, ax = plt.subplots()
a = animation.FuncAnimation(fig, trajectories.plot_points, frames=trajectories.timestep_range(), repeat=False,
                            interval=1000*trajectories.timestep, fargs=(ax,), blit=True)
plt.show()

The plot settings for each vehicle can be customized and the color of each point can be animated, as shown in the following example.

for trajectory in trajectories:
        trajectory.assign_colors_speed()
        trajectory.point_plot_kwargs["ms"] = 8  # set marker size. Can set any kwargs taken by matplotlib.pyplot.plot().

In order to animate the color of the points based on the assigned color scheme, an additional farg must be passed when creating the animation.

a = animation.FuncAnimation(fig, trajectories.plot_points, frames=trajectories.timestep_range(), repeat=False,
                            interval=1000*trajectories.timestep, fargs=(ax, True), blit=True)

Additional Files

Currently, polygons, POIs, and bus stops are supported. Sumo additional files can be loaded and plotted in one of two ways:

Load directly with Network

net = SumoNetVis.Net("path/to/yourfile.net.xml", additional_files="path/to/additionals_file.add.xml")

Load and handle additional files separately

addls = SumoNetVis.Additionals("path/to/additionals_file.add.xml", reference_net=net)
addls.plot()

The reference_net argument is optional, and only necessary for bus stops and for POIs whose position is defined relative to a lane in the network.

Bus stop styles

Several styles of bus stop are supported. The style can be changed using the function SumoNetVis.set_bus_stop_style(). See documentation for further details

Generic Parameters

Generic parameter values are loaded for supported objects (edges, lanes, junctions, vehicles, polys) and are stored in an attribute called params in each of the respective object classes. These can be used to aid in implementing custom functionality on top of SumoNetVis. There are also some built-in features which can make use of these parameters. For example:

  • trajectory colorization
  • OBJ export material
  • OBJ export extrude height

See the Sumo documentation on generic parameters as well as the full SumoNetVis documentation for more information.

OBJ Export

The Wavefront-OBJ format is a text-based file format. The Net.generate_obj_text() method generates the contents of this file for the given Sumo network. In order to save an OBJ file, do the following:

# Save a network as an OBJ file
with open("model.obj", "w") as f:
    f.write(net.generate_obj_text())

# Save bus stops and polygons from an additional file as an OBJ file
with open("busstops.obj", "w") as f:
    f.write(addls.generate_bus_stops_obj_text())

The axis configuration in the generated file is Y-Forward, Z-Up. Check these settings if the orientation of the model is incorrect when importing the file into a 3D modelling program.

A flat planar "terrain" mesh can optionally be generated for all areas within a given distance of a network object. See full documentation for further information.

Each type of object is defined with a corresponding material (i.e. all bike lanes have the same material, all sidewalks, and so on), making it easy to quickly set the desired material properties before rendering. These can also be mapped to user-defined material names if desired (see full documentation). Default material names are of the following forms: for markings, "[c]_marking", where [c] is the marking color ("w" for white, "y" for yellow, etc.); for lanes, "[type]_lane", where [type] is the lane type ("pedestrian", "bicycle", "no_passenger", "none", "other"); for junctions, "junction"; for terrain, "terrain".

Extensibility

To aid in extensibility, each plotting function returns an iterable object consisting of all matplotlib Artist objects generated by that function. Trajectory plotting functions return simple lists or tuples. Net and Additionals plotting functions return an ArtistCollection object which is iterable just like a normal list, but also contains a list attribute for each object type. Each artist is also given an attribute sumo_object which contains a reference to the SumoNetVis object that generated it. The artists can then be acted on as desired, for example to create custom animations or to make an interactive plot. An illustrative example of the latter is shown below. In the example, when a user clicks on a lane, that lane's id and speed properties are printed out.

def onpick(event):
    sumo_object = event.artist.sumo_object
    print("Lane ", sumo_object.id, "clicked. Speed limit: ", sumo_object.speed)

artists = net.plot()
for lane_artist in artists.lanes:
    lane_artist.set_picker(True)
fig.canvas.mpl_connect("pick_event", onpick)
plt.show()

Documentation

API documentation can be found here

Contribution

Issues and pull requests are welcome.