Skip to main content

Python Pipeline API

For simple visualization, use the convenience wrappers megane.view() and megane.view_traj():

import megane

viewer = megane.view("protein.pdb") # structure
viewer = megane.view_traj("protein.pdb", xtc="trajectory.xtc") # with trajectory

When you need more control — filtering atoms, multi-layer rendering, labels, polyhedra, or custom styling — build a pipeline manually with the Pipeline class.

For real-world examples, see the Gallery.

Overview

from megane import Pipeline, LoadStructure, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb")) # returns node with .out/.inp
v = pipe.add_node(Viewport())
pipe.add_edge(s.out.particle, v.inp.particle) # connect explicit ports

viewer = MolecularViewer()
viewer.set_pipeline(pipe) # apply to viewer
viewer # display in notebook

After add_node(), each node exposes .out and .inp namespaces for its ports. Pass these port objects to add_edge() to wire nodes together explicitly. The pipeline serializes to the SerializedPipeline v3 JSON format. A Viewport node must be explicitly added and connected for data to be rendered.

Pipeline class

MethodDescription
add_node(node)Add a node to the pipeline. Returns the node (with .out/.inp ports) for use in add_edge()
add_edge(source_port, target_port)Connect source.out.<name>target.inp.<name>
to_dict()Serialize to v3 JSON dict
to_json(indent=2)Serialize to a JSON string
save(path)Save the pipeline to a JSON file
Pipeline.from_dict(d)Reconstruct a Pipeline from a v3 dict
Pipeline.from_json(s)Reconstruct a Pipeline from a JSON string
Pipeline.load(path)Load a Pipeline from a JSON file

Node classes

All node classes are importable from megane:

from megane import (
LoadStructure,
LoadTrajectory,
Streaming,
LoadVector,
LoadVolumetric,
Filter,
Modify,
Color,
Representation,
AddBonds,
AddLabels,
AddPolyhedra,
Isosurface,
VectorOverlay,
Viewport,
Pipeline,
)
note

The Surface Mesh node (alpha-shape envelope) is available in the visual pipeline editor but does not have a Python class. Use the standalone app, JupyterLab, or VSCode to add it interactively, or author the pipeline JSON directly.

LoadStructure

Load a molecular structure file.

LoadStructure(path: str)
ParameterTypeDescription
pathstrFile path. Auto-detected by extension. Supported by the Python-side parser: .pdb, .gro, .xyz, .mol, .sdf (routed through the MOL parser), .mol2, .cif, .data, .lammps, .traj (ASE binary). The browser-side (WASM) parser additionally accepts .mmcif and .prmtop (AMBER topology) when the pipeline runs inside MolecularViewer.

Ports: out.particle, out.traj, out.cell

LoadTrajectory

Load an external trajectory file. Requires connection from a LoadStructure node.

LoadTrajectory(
*,
xtc: str | None = None,
dcd: str | None = None,
nc: str | None = None,
traj: str | None = None,
xyz: str | None = None,
lammpstrj: str | None = None,
)
ParameterTypeDefaultDescription
xtcstr | NoneNonePath to XTC trajectory file
dcdstr | NoneNonePath to DCD trajectory file (CHARMM/NAMD/X-PLOR)
ncstr | NoneNonePath to AMBER NetCDF trajectory file
trajstr | NoneNonePath to ASE .traj trajectory file
xyzstr | NoneNonePath to a multi-frame XYZ trajectory file
lammpstrjstr | NoneNonePath to LAMMPS dump trajectory (.lammpstrj / .dump)

Pass exactly one. Ports: inp.particle, out.traj

Streaming

WebSocket-based real-time data delivery.

Streaming()

No parameters. Ports: out.particle, out.bond, out.traj, out.cell

LoadVector

Load per-atom vector data from a file.

LoadVector(path: str)
ParameterTypeDescription
pathstrPath to vector data file

Ports: out.vector

LoadVolumetric

Load a Gaussian CUBE file and output volumetric data for isosurface rendering.

LoadVolumetric(path: str = "")
ParameterTypeDescription
pathstrFile path to a Gaussian CUBE file (.cube). Parsed in the browser; the Python object only tracks the filename.

Ports: out.volumetric

Isosurface

Extract an isosurface from volumetric data using marching cubes.

Isosurface(
*,
iso_level: float = 0.05,
color: str = "#4488ff",
opacity: float = 0.7,
show_negative: bool = False,
negative_color: str = "#ff4444",
)
ParameterTypeDefaultDescription
iso_levelfloat0.05Contour value for the positive isosurface
colorstr"#4488ff"Hex color for the positive isosurface
opacityfloat0.7Surface transparency (0–1)
show_negativeboolFalseShow a second isosurface at −iso_level (dual-contour for ESP maps)
negative_colorstr"#ff4444"Hex color for the negative isosurface

Ports: inp.volumetric, out.mesh

Filter

Select atoms by a query expression.

Filter(*, query: str = "all", bond_query: str = "")
ParameterTypeDefaultDescription
querystr"all"Atom selection expression (see Filter DSL)
bond_querystr""Bond selection expression (see Bond Selection DSL). Empty string means no bond filtering.

Ports: inp.particle, out.particle

Modify

Override per-atom visual properties.

Modify(*, scale: float = 1.0, opacity: float = 1.0)
ParameterTypeDefaultDescription
scalefloat1.0Atom sphere radius multiplier (0.1–2.0)
opacityfloat1.0Transparency (0 = invisible, 1 = opaque)

Ports: inp.particle, out.particle

Color

Recolor the upstream particle stream by a chosen scheme. Color was split out of Modify so each modifier owns a single visual property (Ovito-style modifier stack).

Color(
*,
mode: Literal["uniform", "byElement", "byResidue", "byChain", "byBFactor", "byProperty"] = "uniform",
uniform_color: str = "#ff8800",
range: tuple[float, float] | None = None,
)
ParameterTypeDefaultDescription
modestr"uniform"Coloring scheme
uniform_colorstr"#ff8800"Hex color used when mode == "uniform"
rangetuple[float, float] | NoneNoneOptional explicit range for byBFactor / byProperty

Ports: inp.particle, out.particle

Representation

Tag the particle stream with the visual representation the Viewport should display. Stacks Ovito-style: the Viewport reads the override from the first particle stream that carries one, so a downstream Representation node wins over an upstream one on the same chain. When no chain has an override, the Viewport falls back to "atoms".

Representation(*, mode: Literal["atoms", "licorice", "cartoon", "both", "surface", "line"] = "atoms")
ParameterTypeDefaultDescription
modestr"atoms"Visual representation: "atoms" (ball-and-stick), "licorice" (equal-radius continuous sticks), "cartoon", "both", "surface", or "line" (thin wireframe)

Ports: inp.particle, out.particle

AddBonds

Compute and display bonds.

AddBonds(*, source: Literal["distance", "structure", "file"] = "distance", top: str | None = None)
ParameterTypeDefaultDescription
sourcestr"distance""distance" for VDW-based inference; "structure" (alias "file") to read bonds from the structure file (e.g. CONECT records in PDB)
topstr | NoneNonePath to a topology file (GROMACS .top or CHARMM/NAMD .psf). When provided, source is ignored and bonds are read from the topology file.

Ports: inp.particle, out.bond

AddLabels

Generate text labels at atom positions.

AddLabels(*, source: Literal["element", "resname", "index"] = "element")
ParameterTypeDefaultDescription
sourcestr"element"Label source: "element", "resname", or "index"

Ports: inp.particle, out.label

AddPolyhedra

Generate coordination polyhedra mesh (VESTA-style auto-detection).

By default, a polyhedron is drawn for every metal/metalloid center coordinated to every typical anion-forming ligand present in the structure. Use excluded_centers / excluded_ligands to opt out specific elements, mirroring VESTA's checkbox UI.

AddPolyhedra(
*,
excluded_centers: list[int] | None = None,
excluded_ligands: list[int] | None = None,
cutoff_tolerance: float = 1.15,
opacity: float = 0.5,
show_edges: bool = False,
edge_color: str = "#dddddd",
edge_width: float = 3.0,
)
ParameterTypeDefaultDescription
excluded_centerslist[int] | None[]Atomic numbers to exclude from center detection (e.g., [38] to skip Sr)
excluded_ligandslist[int] | None[]Atomic numbers to exclude from ligand detection
cutoff_tolerancefloat1.15Bond-length tolerance multiplier applied to VDW radii sum for center–ligand detection
opacityfloat0.5Face transparency (0–1)
show_edgesboolFalseDisplay wireframe edges
edge_colorstr"#dddddd"Wireframe edge color (hex)
edge_widthfloat3.0Wireframe edge width (px)

Ports: inp.particle, out.mesh

VectorOverlay

Configure per-atom vector visualization (e.g. forces, velocities).

VectorOverlay(*, scale: float = 1.0)
ParameterTypeDefaultDescription
scalefloat1.0Vector arrow length multiplier

Ports: inp.vector, out.vector

Viewport

3D rendering output node. All data to be rendered must be explicitly connected to this node.

Viewport(*, perspective: bool = False, cell_axes_visible: bool = True, pivot_marker_visible: bool = True)
ParameterTypeDefaultDescription
perspectiveboolFalseToggle perspective / orthographic projection
cell_axes_visibleboolTrueShow simulation cell axes with labels
pivot_marker_visibleboolTrueShow the rotation pivot marker at the camera target

Ports: inp.particle, inp.bond, inp.cell, inp.traj, inp.label, inp.mesh, inp.vector

Ports

After add_node(), each node exposes two port namespaces:

  • node.out.<name> — output port (pass as first arg to add_edge)
  • node.inp.<name> — input port (pass as second arg to add_edge)

The port name .traj maps to the "trajectory" wire handle internally. Accessing an undefined port raises AttributeError with a helpful message listing available ports.

Example: Basic Structure with Bonds

from megane import Pipeline, LoadStructure, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb"))
bonds = pipe.add_node(AddBonds(source="distance"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(s.out.particle, v.inp.particle)
pipe.add_edge(bonds.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: Filter and Modify

from megane import Pipeline, LoadStructure, Filter, Modify, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb"))
carbons = pipe.add_node(Filter(query="element == 'C'"))
big = pipe.add_node(Modify(scale=1.5, opacity=0.8))
bonds = pipe.add_node(AddBonds(source="distance"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, carbons.inp.particle)
pipe.add_edge(carbons.out.particle, big.inp.particle)
pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(big.out.particle, v.inp.particle)
pipe.add_edge(bonds.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: Trajectory Playback

from megane import Pipeline, LoadStructure, LoadTrajectory, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb"))
t = pipe.add_node(LoadTrajectory(xtc="trajectory.xtc"))
bonds = pipe.add_node(AddBonds(source="structure"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, t.inp.particle)
pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(s.out.particle, v.inp.particle)
pipe.add_edge(t.out.traj, v.inp.traj)
pipe.add_edge(bonds.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer.frame_index = 50 # jump to frame 50

Example: Make Solvent Translucent

from megane import Pipeline, LoadStructure, Filter, Modify, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb"))

# Filter water molecules and make them translucent
water = pipe.add_node(Filter(query='resname == "HOH"'))
transparent = pipe.add_node(Modify(scale=0.5, opacity=0.2))

bonds = pipe.add_node(AddBonds(source="distance"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, water.inp.particle)
pipe.add_edge(water.out.particle, transparent.inp.particle)
pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(s.out.particle, v.inp.particle) # protein particle + cell
pipe.add_edge(transparent.out.particle, v.inp.particle) # translucent water
pipe.add_edge(bonds.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: TiO₆ Coordination Polyhedra

from megane import Pipeline, LoadStructure, AddBonds, AddPolyhedra, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("SrTiO3_supercell.pdb"))
bonds = pipe.add_node(AddBonds(source="distance"))
# All metal centers and anion-forming ligands are included by default (VESTA-style).
# Exclude Sr (38) so only TiO6 polyhedra are drawn.
polyhedra = pipe.add_node(AddPolyhedra(
excluded_centers=[38], # exclude Sr; Ti (22) is kept
opacity=0.5,
show_edges=True,
))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(s.out.particle, polyhedra.inp.particle)
pipe.add_edge(s.out.particle, v.inp.particle)
pipe.add_edge(bonds.out.bond, v.inp.bond)
pipe.add_edge(polyhedra.out.mesh, v.inp.mesh)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: DAG Branching (Multiple Filters)

from megane import Pipeline, LoadStructure, Filter, AddLabels, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("protein.pdb"))

# Two independent filters from the same source
carbon = pipe.add_node(Filter(query="element == 'C'"))
nitrogen = pipe.add_node(Filter(query="element == 'N'"))
labels = pipe.add_node(AddLabels(source="element"))
bonds = pipe.add_node(AddBonds(source="distance"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, carbon.inp.particle)
pipe.add_edge(s.out.particle, nitrogen.inp.particle)
pipe.add_edge(s.out.particle, labels.inp.particle)
pipe.add_edge(s.out.particle, bonds.inp.particle)
pipe.add_edge(carbon.out.particle, v.inp.particle)
pipe.add_edge(nitrogen.out.particle, v.inp.particle)
pipe.add_edge(labels.out.label, v.inp.label)
pipe.add_edge(bonds.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: Multiple Structure Layers

from megane import Pipeline, LoadStructure, AddBonds, Viewport, MolecularViewer

pipe = Pipeline()
protein = pipe.add_node(LoadStructure("protein.pdb"))
ligand = pipe.add_node(LoadStructure("ligand.mol"))
bonds_p = pipe.add_node(AddBonds(source="distance"))
bonds_l = pipe.add_node(AddBonds(source="structure"))
v = pipe.add_node(Viewport())

pipe.add_edge(protein.out.particle, bonds_p.inp.particle)
pipe.add_edge(ligand.out.particle, bonds_l.inp.particle)
pipe.add_edge(protein.out.particle, v.inp.particle)
pipe.add_edge(bonds_p.out.bond, v.inp.bond)
pipe.add_edge(ligand.out.particle, v.inp.particle)
pipe.add_edge(bonds_l.out.bond, v.inp.bond)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer

Example: ASE .traj Trajectory

from megane import Pipeline, LoadStructure, LoadTrajectory, Viewport, MolecularViewer

pipe = Pipeline()
s = pipe.add_node(LoadStructure("structure.pdb"))
t = pipe.add_node(LoadTrajectory(traj="simulation.traj"))
v = pipe.add_node(Viewport())

pipe.add_edge(s.out.particle, t.inp.particle)
pipe.add_edge(s.out.particle, v.inp.particle)
pipe.add_edge(t.out.traj, v.inp.traj)

viewer = MolecularViewer()
viewer.set_pipeline(pipe)
viewer