DEV Community

Cover image for Learn React Flow and how to add real-time sync between participants
Vitor Norton for SuperViz

Posted on • Originally published at superviz.com

Learn React Flow and how to add real-time sync between participants

Today I want to present to you this powerful tool for creating a node-based application: the React Flow. It is a library for building anything from simple static diagrams to data visualizations to complex visual editors. In this article I will show you how to create a simple React Flow application, understand more about nodes and edges, and add real-time synchronization between participants in the same content.

Installation

To begin with React Flow, you first will need to have a React base application and add the package reactflow to it, you can use any package manager

npm install reactflow
Enter fullscreen mode Exit fullscreen mode

Nodes and Edges

Before starting to use it, you should be familiarized with what a node and an edge is.

In simple terms, nodes are the primary elements or objects that you want to represent in your graph, and edges are the connections or relationships between these nodes. For example, in a social network, each person could be a node, and the friendships between them could be the edges.

Technically speaking, React Flow has these two types: the Node, which will contain an ID, a position and its data, and the Edge which will contain an ID for it, a source and target. In the code below I created a list of initial nodes (a couple of blocks) and a list of initial edges: connecting the first node to the second node.

export const initialNodes = [
    {
        id: '1',
        position: { x: 200, y: 200 },
        data: { label: 'First block' },
    },
    { 
        id: '2', 
        position: { x: 250, y: 300 }, 
        data: { label: 'Second block' } 
    },
]

export const initialEdges = [ 
    {
        id: 'some-id', 
        source: '1', 
        target: '2' 
    }
]

Enter fullscreen mode Exit fullscreen mode

Start using React Flow

After creating the initial nodes and edges, you are now able to add the ReactFlow component to your application. Such as shown below:

import React from 'react';
import ReactFlow from 'reactflow';

import 'reactflow/dist/style.css';

export default function App() {
  return (
    <div style={{ width: '100vw', height: '100vh' }}>
      <ReactFlow nodes={initialNodes} edges={initialEdges} />
    </div>
  );
}
Enter fullscreen mode Exit fullscreen mode

In the code above we create a full screen container for our nodes. We also import the styles from reactflow/dist/style.css.

To make manipulation of the data easy, I’m going to use the useNodesState and useEdgesState hooks, which work like the useState hook. With an addiction to a callback function when there are changes on the state.

import ReactFlow, { useNodesState, useEdgesState } from 'reactflow'

export default function App() {
    const [nodes, setNodes, onNodesChange] = useNodesState(initialNodes)
    const [edges, setEdges, onEdgesChange] = useEdgesState(initialEdges)

  return (
    <div style={{ width: '100vw', height: '100vh' }}>
      <ReactFlow nodes={nodes} edges={edges} />
    </div>
  );
}
Enter fullscreen mode Exit fullscreen mode

To create a better experience for our users, let’s add some controls, allowing them to zoom in and out, to see a mini map of their content and other features. We will add it as a child to our ReactFlow component.

<ReactFlow nodes={nodes} edges={edges}>
    <Controls />
    <MiniMap />
</ReactFlow>
Enter fullscreen mode Exit fullscreen mode

To make a great background, you can also add the Background element, in which you will be able to choose variants of background like dots, crosses or lines as well their colors.

<Background variant={BackgroundVariant.Cross} color='#f00' />
Enter fullscreen mode Exit fullscreen mode

Synchronizing between users

If you want to create a collaborative environment where multiple users can interact with the same React Flow application at the same time, you will need to use a Real-time Data Engine. This engine allows any changes made by one user to be instantly visible to all other users in the same room. This involves creating a unique room for each group of users and synchronizing their activities.

Creating a shared room

A room is a virtual space where users can join and collaborate. When creating a room, you can specify a unique ID for it.

To create a room, you need to use the SuperVizRoom that’s available from the @superviz/sdk package, which takes the Developer Key and an object as a parameter. The object should have the following properties:

  • id: The ID of the room, which should be a unique string shared between the participants of that room.
  • participant: An object that contains information about the current user, such as nameid.
  • group: An object that contains information about the group that the user belongs to, such as name and id.

Here’s an example of how to create a room with SuperViz:

// Import the SuperViz SDK
import SuperVizRoom from '@superviz/sdk';

// Create a room object
const room = await SuperVizRoom(DEVELOPER_KEY, {
  roomId: "<ROOM-ID>",
  participant: {
    id: "<USER-ID>",
    name: "<USER-NAME>"
  },
});
Enter fullscreen mode Exit fullscreen mode

Adding the Real-time Data Engine

To add the event broker, which will be responsible for notifying when a change is made and to listen to state changes. Right after creating a room, you can add the engine to it.

const realtime = new Realtime();
room.addComponent(realtime);
Enter fullscreen mode Exit fullscreen mode

Listening to state changes

To make full synchronization we are going to listen to a couple of events: new-edge and node-drag, which happens when we connect one React Flow node to another and when we drag a node across the canvas.

Let’s first create a function to handle when new edges are added, and then subscribe it to the new-edge event.

function onNewEdgesAdded({data, participantId}){
  // Verify if the origin of the event dispatch isn't the same participant on the page
  // preventing a infinity loop
    if (participantId === currentParticipantId) return;

  setEdges((eds) => addEdge(data.edge, eds));
}

realtime.subscribe('new-edge', onNewEdgesAdded)
Enter fullscreen mode Exit fullscreen mode

Now, we will do something similar to listening to node-drag. In this code we will update our nodes with the data, containing its new position, that we receive from the event.

function onNodeDragEventHandler({data, participantId}){
    // Verify if the origin of the event dispatch isn't the same participant on the page
  // preventing a infinity loop
    if (participantId === currentParticipantId) return;

    setNodes((nodes: { id }[]) =>
        nodes.map((node: { id }) => (node.id === data.node.id ? data.node : node))
    );
}

realtime.subscribe('node-drag', onNodeDragEventHandler)
Enter fullscreen mode Exit fullscreen mode

Publishing an event

We are now listening to the events, but to trigger these events, we need to publish them. We can do this by using the publish method from the realtime object. Whenever a user adds a new edge or drags a node, we will call realtime.publish, passing the name of the event and the relevant data. This will then trigger the corresponding event listeners in other users' applications, keeping all participants in sync.

To understand when new changes are made on both scenarios, we need to use React Flow’s callback functions onConnect and onNodeDrag, such as the code below:

<ReactFlow
    ...
    onNodeDrag={onNodeDrag}
    onConnect={onConnect}>
</ReactFlow>
Enter fullscreen mode Exit fullscreen mode

The implementation of the onNodeDrag is straightforward, as shown below:

function onNodeDrag(event, node) {
    realtime.publish('node-drag', { node });
}
Enter fullscreen mode Exit fullscreen mode

To publish the new-edge event on a new connection (onConnect) it's a bit different, we first need to recreate the edge object, add it to our current list using the React Flow’s addEdge method, and then publish to the Real-time Data Engine.

function onConnect(params) => {
    const edge = {
        ...params,
        type: ConnectionLineType.SmoothStep,
        animated: true,
    }

    setEdges((edges) => addEdge(edge, edges))

    realtime.publish('new-edge', { edge })
}
Enter fullscreen mode Exit fullscreen mode

Conclusion

In this tutorial, we've learned the basics of React Flow and how to create a real-time, collaborative environment using SuperViz's Real-time Data Engine. You can see this project in functioning on our demo page dedicated to React Flow, as well the source code on our GitHub repository.

Happy coding!

Top comments (0)