Skip to main content

A flexible, customizable, Streamlit component for interactive graph visualization using Cytoscape.js

Project description

streamlit-cytoscape

A flexible, customizable Streamlit component for interactive graph visualization using Cytoscape.js. This project is a fork of st-link-analysis by @AlrasheedA, who developed the bulk of the original component. This fork extends the original with additional customization options and bug fixes.

Overview

This project provides a Streamlit custom component for visualizing and interacting with graph data using Cytoscape.js. It supports customizable edge and node styles, labels, colors, captions, and icons.

Features

  • Customizable Node and Edge Styles: Easily define the appearance of nodes and edges using a variety of style options.
  • Custom Styles Pass-through: Use the custom_styles parameter on NodeStyle and EdgeStyle to pass any valid Cytoscape.js style property directly, enabling fine-grained control over shapes, borders, opacity, fonts, and more.
  • Material Icons Support: Supports a subset of Material icons for styling nodes which can be passed by name (e.g., icon='person'). Custom icons can still be used by passing a URL (e.g., icon='url(...)').
  • Customizable Layouts: Choose from different layout algorithms to arrange the graph elements.
  • Interactive Features:
    • Toolbar with fullscreen, layout refresh, and export dialog (named file download with optional visual styles).
    • View control bar for zooming, fitting, and centering the view.
    • View all properties of the selected elements in a side panel.
    • Highlights neighboring nodes or edges when an element is selected.
  • Node Actions (Expand / Remove): Enable node removal and expansion using the node_actions parameter. Removal can be triggered by a delete keydown or a remove button click, while expansion occurs on a double-click or expand button click.
  • Edge Actions (Collapse / Expand): Collapse parallel edges (multiple edges between the same nodes) into a single meta-edge showing a priority label and count. Double-click to expand back to individual edges.
  • Infopanel Actions: Add custom action buttons to the infopanel using InfopanelAction. When a selected element's action button is clicked, the action name and element data are returned to the Streamlit app.

Installation

pip install streamlit-cytoscape

Usage

import streamlit as st
from streamlit_cytoscape import streamlit_cytoscape, NodeStyle, EdgeStyle

st.set_page_config(layout="wide")

# Sample Data
elements = {
    "nodes": [
        {"data": {"id": 1, "label": "PERSON", "name": "Streamlit"}},
        {"data": {"id": 2, "label": "PERSON", "name": "Hello"}},
        {"data": {"id": 3, "label": "PERSON", "name": "World"}},
        {"data": {"id": 4, "label": "POST", "content": "x"}},
        {"data": {"id": 5, "label": "POST", "content": "y"}},
    ],
    "edges": [
        {"data": {"id": 6, "label": "FOLLOWS", "source": 1, "target": 2}},
        {"data": {"id": 7, "label": "FOLLOWS", "source": 2, "target": 3}},
        {"data": {"id": 8, "label": "POSTED", "source": 3, "target": 4}},
        {"data": {"id": 9, "label": "POSTED", "source": 1, "target": 5}},
        {"data": {"id": 10, "label": "QUOTES", "source": 5, "target": 4}},
    ],
}

# Style node & edge groups
node_styles = [
    NodeStyle("PERSON", "#FF7F3E", "name", "person"),
    NodeStyle("POST", "#2A629A", "content", "description"),
]

edge_styles = [
    EdgeStyle("FOLLOWS", caption='label', directed=True),
    EdgeStyle("POSTED", caption='label', directed=True),
    EdgeStyle("QUOTES", caption='label', directed=True),
]

# Render the component
st.markdown("### streamlit-cytoscape: Example")
streamlit_cytoscape(elements, "cose", node_styles, edge_styles)

Custom Styles

Use the custom_styles parameter to apply any valid Cytoscape.js style property:

from streamlit_cytoscape import NodeStyle, EdgeStyle

# Custom node styling with borders, shapes, and fonts
node_styles = [
    NodeStyle(
        label="PERSON",
        color="#FF7F3E",
        caption="name",
        icon="person",
        custom_styles={
            "shape": "diamond",
            "width": 60,
            "height": 60,
            "border-width": 3,
            "border-color": "#000000",
            "border-style": "dashed",
            "opacity": 0.9,
            "font-size": 14,
            "color": "#FFFFFF",
        },
    ),
]

# Custom edge styling with line styles and arrows
edge_styles = [
    EdgeStyle(
        label="FOLLOWS",
        caption="label",
        directed=True,
        custom_styles={
            "width": 3,
            "line-style": "dashed",
            "opacity": 0.8,
            "target-arrow-shape": "vee",
            "arrow-scale": 1.5,
            "font-size": 10,
        },
    ),
]

See the Cytoscape.js style documentation for all available properties.

Edge Actions (Collapse / Expand Parallel Edges)

When your graph has multiple edges between the same pair of nodes, you can collapse them into a single "meta-edge" that shows a priority label and count:

from streamlit_cytoscape import streamlit_cytoscape, NodeStyle, EdgeStyle

elements = {
    "nodes": [
        {"data": {"id": "alice", "label": "PERSON", "name": "Alice"}},
        {"data": {"id": "bob", "label": "PERSON", "name": "Bob"}},
    ],
    "edges": [
        # Multiple edges between Alice and Bob
        {"data": {"id": "e1", "label": "FOLLOWS", "source": "alice", "target": "bob"}},
        {"data": {"id": "e2", "label": "LIKES", "source": "alice", "target": "bob"}},
        {"data": {"id": "e3", "label": "WORKS_WITH", "source": "alice", "target": "bob"}},
    ],
}

# Collapse parallel edges on load, with FOLLOWS as priority label
streamlit_cytoscape(
    elements,
    layout="cose",
    edge_actions=["collapse", "expand"],
    collapse_parallel_edges=True,
    priority_edge_label="FOLLOWS",  # Shows "FOLLOWS (3)" on the meta-edge
)

You can also customize the appearance of collapsed meta-edges:

streamlit_cytoscape(
    elements,
    layout="cose",
    edge_actions=["collapse", "expand"],
    collapse_parallel_edges=True,
    meta_edge_style={
        "line-color": "#FF0000",
        "width": 5,
        "font-weight": "bold",
    },
)

Infopanel Actions

Add custom action buttons to the infopanel that appear when an element is selected. Clicking a button returns the action name and selected element data to your Streamlit app:

from streamlit_cytoscape import (
    streamlit_cytoscape, NodeStyle, InfopanelAction,
)

actions = [
    InfopanelAction("ai_summary", "AI Summary", icon="science"),
    InfopanelAction("fetch_details", "Fetch Details", icon="analytics"),
    InfopanelAction("flag_review", "Flag for Review", icon="flag"),
]

def on_action():
    val = st.session_state["my_graph"]
    action = val["action"]       # e.g. "ai_summary"
    eid = val["data"]["element_id"]  # selected element ID
    # Update element data, call an API, etc.

streamlit_cytoscape(
    elements,
    layout="fcose",
    node_styles=node_styles,
    infopanel_actions=actions,
    on_change=on_action,
    key="my_graph",
)

API Reference

Element Description
streamlit_cytoscape Main component for creating and displaying the graph, including layout and height settings.
NodeStyle Defines styles for nodes, including labels, colors, captions, icons, and custom_styles for Cytoscape.js pass-through.
EdgeStyle Defines styles for edges, including curve styles, labels, colors, directionality, and custom_styles for Cytoscape.js pass-through.
InfopanelAction Defines a custom action button for the infopanel with a name, label, and optional icon.
Event Define an event to pass to component function and listen to.

Development

Ensure you have Python 3.10+, Node.js, and npm installed.

Setup

# Create conda environment
conda create -n streamlit_cytoscape python=3.10
conda activate streamlit_cytoscape

# Install Python package
poetry install

# Install frontend dependencies
cd src/streamlit_cytoscape/frontend
npm install

Running the App

Change _RELEASE flag in src/streamlit_cytoscape/component.py to False.

In one terminal start the frontend dev server:

cd src/streamlit_cytoscape/frontend
npm run start

In another terminal run the streamlit server:

cd examples
poetry run streamlit run app.py

Testing

poetry run black .
poetry run flake8 src/streamlit_cytoscape tests examples
poetry run mypy src/streamlit_cytoscape
poetry run pytest

Contributing

Interested in contributing? Check out the contributing guidelines. Please note that this project is released with a Code of Conduct. By contributing to this project, you agree to abide by its terms.

License

streamlit_cytoscape was created by Natalie Gill. It is licensed under the terms of the MIT license.

AI Disclosure Statement

Generative AI tools (Claude Code, Anthropic) were used as coding assistants during the development of this project. The authors maintain full responsibility for the accuracy of all code. AI-assisted outputs were reviewed and validated against expected behavior before integration.

Acknowledgments

This project is a fork of st-link-analysis by @AlrasheedA, who developed the bulk of the original component. This fork extends the original with additional customization options.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

streamlit_cytoscape-0.1.5.tar.gz (716.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

streamlit_cytoscape-0.1.5-py3-none-any.whl (756.3 kB view details)

Uploaded Python 3

File details

Details for the file streamlit_cytoscape-0.1.5.tar.gz.

File metadata

  • Download URL: streamlit_cytoscape-0.1.5.tar.gz
  • Upload date:
  • Size: 716.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for streamlit_cytoscape-0.1.5.tar.gz
Algorithm Hash digest
SHA256 2233e92c6bdf65d41916c89bf1c323266efcc5406f835428bf9c3ccd0c0623b1
MD5 067e88dd646d8ebff4d4d1135d76e0c0
BLAKE2b-256 033c2cc4d915d18204ecf79c71c7affd48c26b2e3ca1927ffdbd3325f7d1a469

See more details on using hashes here.

Provenance

The following attestation bundles were made for streamlit_cytoscape-0.1.5.tar.gz:

Publisher: ci.yml on natalie-23-gill/streamlit-cytoscape

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file streamlit_cytoscape-0.1.5-py3-none-any.whl.

File metadata

File hashes

Hashes for streamlit_cytoscape-0.1.5-py3-none-any.whl
Algorithm Hash digest
SHA256 d70d0e7318e0b5ae1ca05fd9727247e60c1c88800644a166fcdabac3ab34dceb
MD5 af95e0221e6a531a6387a39e270549aa
BLAKE2b-256 37a88a0cc493d8a71180d648fb1d864c851442d084bbc9a5fa41a5d0c2e3a4bd

See more details on using hashes here.

Provenance

The following attestation bundles were made for streamlit_cytoscape-0.1.5-py3-none-any.whl:

Publisher: ci.yml on natalie-23-gill/streamlit-cytoscape

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page