In this tutorial, we will guide you through adding real-time synchronization to a React Flow application using SuperViz. Real-time synchronization is a key feature for collaborative applications, allowing multiple users to interact with shared content simultaneously and see each other's changes as they happen. With React Flow and SuperViz, you can build interactive flowcharts that update live, providing a seamless collaborative experience.
We'll demonstrate how to integrate contextual comments and real-time mouse pointers into a React Flow application, enabling users to collaborate on flowcharts with real-time updates. This setup allows multiple participants to add nodes, create connections, and drag elements within the flowchart, with changes visible instantly to all users in the session. Let's get started!
Prerequisite
To follow this tutorial, you will need a SuperViz account and a developer token. If you already have an account and a developer token, you can move on to the next step.
Create an account
To create an account, go to https://dashboard.superviz.com/register and create an account using either Google or an email/password. It's important to note that when using an email/password, you will receive a confirmation link that you'll need to click to verify your account.
Retrieving a Developer Token
To use the SDK, you’ll need to provide a developer token, as this token is essential for associating SDK requests with your account. You can retrieve both development and production SuperViz tokens from the dashboard..
Copy and save the developer token, as you will need it in the next steps of this tutorial.
Step 1: Set Up Your React Application
To begin, you'll need to set up a new React project where we will integrate the React Flow and SuperViz SDK for real-time collaboration.
1. Create a New React Project
First, create a new React application using Create React App with TypeScript.
1npx create-react-app realtime-react-flow --template typescript2cd realtime-react-flow
2. Install Required Libraries
Next, install the necessary libraries for our project:
1npm install @superviz/react-sdk reactflow uuid
- @superviz/react-sdk: SuperViz SDK for integrating real-time collaboration features.
- reactflow: A library for building interactive flowcharts and diagrams.
- uuid: A library for generating unique identifiers, useful for creating unique participant IDs.
3. Configure tailwind
In this tutorial, we'll use the Tailwind css framework. First, install the tailwind package.
1npm install -D tailwindcss postcss autoprefixer2npx tailwindcss init -p
We then need to configure the template path. Open tailwind.config.js
in the root of the project and insert the following code.
1/** @type {import('tailwindcss').Config} */2export default {3content: [4"./index.html",5"./src/**/*.{js,ts,jsx,tsx}",6],7theme: {8extend: {},9},10plugins: [],11}
Then we need to add the tailwind directives to the global CSS file. (src/index.css)
1@tailwind base;2@tailwind components;3@tailwind utilities;
4. Set Up Environment Variables
Create a .env
file in your project root and add your SuperViz developer key. This key will be used to authenticate your application with SuperViz services.
1VITE_SUPERVIZ_API_KEY=YOUR_SUPERVIZ_DEVELOPER_KEY
Step 2: Implement the Main Application
In this step, we'll implement the main application logic to initialize SuperViz and handle real-time synchronization in a React Flow application.
1. Implement the App Component
Open src/App.tsx
and set up the main application component using the SuperVizRoomProvider
to manage the collaborative environment.
1import { SuperVizRoomProvider } from '@superviz/react-sdk';2import { v4 as generateId } from 'uuid';3import Room from './Room';4import { ReactFlowProvider } from 'reactflow';56const developerKey = import.meta.env.VITE_SUPERVIZ_API_KEY;7const participantId = generateId();89export default function App() {10return (11<SuperVizRoomProvider12developerKey={developerKey}13group={{14id: 'react-flow-tutorial',15name: 'react-flow-tutorial',16}}17participant={{18id: participantId,19name: 'Participant',20}}21roomId='react-flow-tutorial'22>23<ReactFlowProvider>24<Room participantId={participantId} />25</ReactFlowProvider>26</SuperVizRoomProvider>27);28}
Explanation:
- SuperVizRoomProvider: This component wraps the application to enable real-time features and provides configuration for group and participant details.
- developerKey: Retrieves the developer key from environment variables to authenticate with SuperViz.
- ReactFlowProvider: Wraps the Room component to provide React Flow's context, which manages the state of the flowchart.
- Room Component: Contains the logic for rendering the flowchart and handling real-time interactions.
Step 3: Create the Room Component
The Room component will be responsible for integrating React Flow with SuperViz, allowing users to collaborate on the flowchart in real-time.
Step-by-Step Breakdown of the Room Component
Let's break down the Room component step-by-step to understand how it enables real-time collaboration using React Flow and SuperViz.
1. Import Necessary Modules
First, import the required modules and components from both reactflow
and @superviz/react-sdk
.
1import { useCallback, useEffect, MouseEvent, useRef } from "react";2import ReactFlow, {3useNodesState,4Controls,5Background,6ConnectionLineType,7addEdge,8useEdgesState,9ConnectionMode,10Connection,11useViewport,12Node,13} from "reactflow";14import "reactflow/dist/style.css";15import {16Comments,17MousePointers,18Realtime,19useComments,20useHTMLPin,21useMouse,22useRealtime,23WhoIsOnline,24} from "@superviz/react-sdk";
Explanation:
- ReactFlow Imports: Provide flowchart components and utilities to create interactive diagrams.
- SuperViz SDK Imports: Includes tools for real-time collaboration, such as comments, mouse pointers, and synchronization.
2. Define Initial Nodes and Edges
Define the initial state of nodes and edges for the flowchart.
1const initialNodes = [2{ id: "1", position: { x: 381, y: 265 }, data: { label: "Start" } },3{ id: "2", position: { x: 556, y: 335 }, data: { label: "Action" } },4{ id: "3", position: { x: 701, y: 220 }, data: { label: "Process" } },5{ id: "4", position: { x: 823, y: 333 }, data: { label: "End" } },6];78const initialEdges = [9{10id: "e1-2",11source: "1",12target: "2",13type: ConnectionLineType.SmoothStep,14animated: true,15},16{17id: "e2-3",18source: "2",19target: "3",20type: ConnectionLineType.SmoothStep,21animated: true,22},23{24id: "e3-4",25source: "3",26target: "4",27type: ConnectionLineType.SmoothStep,28animated: true,29},30];
Explanation:
- initialNodes: An array of objects defining each node's position and label in the flowchart.
- initialEdges: An array of objects defining connections between nodes, using a smooth step connection line with animation.
3. Define the Room Component
Create the Room
component with properties to manage the participant ID.
1type Props = {2participantId: string;3};45export default function Room({ participantId }: Props) {6const subscribed = useRef(false);
Explanation:
- Props Type: Defines the expected properties for the
Room
component, including theparticipantId
. - subscribed Ref: A ref to track whether the component has subscribed to real-time events, ensuring it only subscribes once.
4. Set Up Real-Time Hooks and Utilities
Initialize SuperViz hooks and utilities to manage comments, real-time updates, and mouse transformations.
1// Managing comments2const { openThreads, closeThreads } = useComments();34// Managing real-time updates5const { isReady, subscribe, unsubscribe, publish } = useRealtime("default");67// Managing mouse pointers8const { transform } = useMouse();910// Pinning functionality for comments11const { pin } = useHTMLPin({12containerId: "react-flow-container",13dataAttributeName: "data-id",14dataAttributeValueFilters: [/.*null-(target|source)$/],15});
Explanation:
- useComments: Provides functions to open and close comment threads.
- useRealtime: Offers real-time event handling methods like
subscribe
,unsubscribe
, andpublish
. - useMouse: Allows transformations based on mouse movements.
- useHTMLPin: Enables the pinning of comments to specific HTML elements within the application.
5. Initialize State for Nodes and Edges
Manage the state of nodes and edges using React Flow's hooks.
1const { x, y, zoom } = useViewport();2const [nodes, setNodes, onNodesChange] = useNodesState(initialNodes);3const [edges, setEdges, onEdgesChange] = useEdgesState(initialEdges);
Explanation:
- useViewport: Provides the current viewport state, including the
x
andy
translation andzoom
level. - useNodesState: Manages the state of nodes with initial values and handles changes.
- useEdgesState: Manages the state of edges with initial values and handles changes.
6. Handle Edge Connections
Define a callback function for handling new edge connections in the flowchart.
1const onConnect = useCallback(2(connection: Connection) => {3const edge = {4...connection,5type: ConnectionLineType.SmoothStep,6animated: true,7};89setEdges((eds) => addEdge(edge, eds));1011publish("new-edge", {12edge,13});14},15[setEdges, publish]16);
Explanation:
- onConnect: Handles the creation of new edges, updating the state and publishing the changes to all participants.
- addEdge: Adds a new edge to the current state.
7. Handle Node Dragging
Create a callback for handling node dragging events and publish the changes.
1const onNodeDrag = useCallback(2(_: MouseEvent, node: Node) => {3publish("node-drag", { node });4},5[publish]6);
Explanation:
- onNodeDrag: Publishes node position updates as they are dragged, allowing other participants to see the changes.
8. Handle Drag Over Events
Prevent the default behavior for drag-over events to enable custom dragging interactions.
1const onDragOver = useCallback(2(event: React.DragEvent<HTMLButtonElement | HTMLDivElement>) => {3event.preventDefault();4event.dataTransfer.dropEffect = "move";5},6[]7);
Explanation:
- onDragOver: Sets the drag effect to "move" to provide visual feedback during drag operations.
9. Update Viewport on Mouse Movement
Adjust the viewport based on mouse movements to keep the flowchart aligned.
1useEffect(() => {2transform({3translate: {4x: x,5y: y,6},7scale: zoom,8});9}, [x, y, zoom, transform]);
Explanation:
- useEffect: Transforms the viewport position and scale when the mouse moves, ensuring all participants have a synchronized view.
10. Set Data Attribute for SuperViz Pinning
Assign a data attribute to the React Flow container for SuperViz pinning.
1useEffect(() => {2const element = document.querySelector(".react-flow__pane");34if (!element) return;56element.setAttribute("data-superviz-id", "plane");7}, []);
Explanation:
- data-superviz-id: Allows SuperViz to identify elements that can have pins attached, facilitating comment features.
11. Subscribe to Real-Time Events
Set up subscriptions to listen for real-time events and synchronize state changes.
1useEffect(() => {2if (subscribed.current) return;34const centerNodes = () => {5const centerButton = document.querySelector(6".react-flow__controls-fitview"7) as HTMLButtonElement;8centerButton?.click();9};1011centerNodes();1213// Subscribe to new-edge events14subscribe("new-edge", ({ data, participantId: senderId }) => {15if (senderId === participantId) return;1617setEdges((eds) => addEdge(data.edge, eds));18});1920// Subscribe to node-drag events21subscribe(22"node-drag",23({24data,25participantId: senderId,26}: {27data: { node: Node };28participantId: string;29}) => {30if (senderId === participantId) return;3132setNodes((nds) =>33nds.map((node) =>34node.id === data.node.id ? { ...node, ...data.node } : node35)36);37}38);3940subscribed.current = true;41}, [isReady, setEdges, setNodes, unsubscribe, subscribe, participantId]);
Explanation:
- subscribe: Listens for specific events (
new-edge
,node-drag
) and updates the local state based on incoming data. - unsubscribe: Clean up subscriptions when no longer needed.
12. Render the Room Component
Finally, render the Room component with React Flow and SuperViz features.
1return (2<div className="w-full h-full bg-gray-200 flex items-center justify-center flex-col">3<header className="w-full p-5 bg-purple-400 flex items-center justify-between">4<h1 className="text-white text-2xl font-bold">React Flow + SuperViz</h1>5<div id="comments" className="flex gap-2"></div>6</header>7<main className="flex-1 w-full h-full">8<div id="react-flow-container" className="w-full h-full">9<ReactFlow10nodes={nodes}11onNodeDrag={onNodeDrag}12edges={edges}13onConnect={onConnect}14onNodesChange={onNodesChange}15onEdgesChange={onEdgesChange}16onDragOver={onDragOver}17connectionMode={ConnectionMode.Loose}18>19<Controls showFitView={false} />20<Background />21</ReactFlow>22</div>2324{/* SuperViz Components */}25<Realtime />26<WhoIsOnline position="comments" />27<Comments28pin={pin}29position="left"30buttonLocation="comments"31onPinActive={openThreads}32onPinInactive={closeThreads}33/>34<MousePointers elementId="react-flow-container" />35</main>36</div>37);
Explanation:
- ReactFlow: Displays the flowchart with nodes, edges, and interaction handlers.
- Realtime: Manages real-time synchronization of state across participants.
- WhoIsOnline: Shows a list of online participants in the session.
- Comments: Provides the ability to add and view contextual comments.
- MousePointers: Displays real-time mouse pointers for all participants.
Step 4: Running the Application
1. Start the React Application
To run your application, use the following command in your project directory:
1npm run dev
This command will start the development server and open your application in the default web browser. You can interact with the flowchart and see updates in real-time across multiple participants.
2. Test the Application
- Collaborative Flowchart: Open the application in multiple browser windows or tabs to simulate multiple participants and verify that changes made by one participant are reflected in real-time for others.
- Real-Time Updates: Test the responsiveness of the application to see if it syncs correctly with actions performed by other users.
Summary
In this tutorial, we built a collaborative flowchart application using React Flow and SuperViz for real-time synchronization. We configured a React application to handle node and edge interactions, enabling multiple users to collaborate seamlessly on a shared diagram. This setup can be extended and customized to fit various scenarios where real-time collaboration and workflow visualization are required.
Feel free to explore the full code and further examples in the GitHub repository for more details.