Custom Nodes
A powerful feature of Svelte Flow is the ability to add custom nodes. You can render everything you want in your custom nodes. You can define multiple source and target handles and render form inputs or even interactive charts. In this guide we will implement two nodes with color input fields that update the background color of the flow.
Creating a custom node component
A custom node is just a Svelte component. Internally it gets wrapped to provide basic functionality like dragging and selecting. From the wrapper component we are passing props like the position or the data besides other props.
Let's start to implement the ColorPickerNode
. We are using the Handle
component to connect our custom node with other nodes and add an input field with type="color" to the node:
<script lang="ts">
import { Handle, Position, type NodeProps } from '@xyflow/svelte';
import type { Writable } from 'svelte/store';
type $$Props = NodeProps;
export let data: { color: Writable<string> };
const { color } = data;
</script>
<div class="colorpicker">
<Handle type="target" position={Position.Left} />
<div>
color: <strong>{$color}</strong>
</div>
<input
class="nodrag"
type="color"
on:input={(evt) => data.color.set(evt.target?.value)}
value={$color}
/>
<Handle type="source" position={Position.Right} on:connect on:connectstart on:connectend />
</div>
As you can see we've added the class name "nodrag" to the input. This prevents dragging within the input field, which in this case will allow us to select text inside the node.
Heads up: You have to forward on:connect, on:connectstart and on:connectend on the handles for the events to be fired on the wrapper component. This will likely change in the future.
Adding a new node type
You can add a new node type to Svelte Flow by adding it to the nodeTypes
prop. The nodeTypes
prop is an object where the key is the name of the node type and the value is the component that should be rendered for this node type. Let's add the TextUpdaterNode
to the nodeTypes
prop:
<script>
import { SvelteFlow } from '@xyflow/svelte';
import ColorPickerNode from './ColorPickerNode.svelte';
const nodeTypes = {
'color-picker': ColorPickerNode
};
</script>
<SvelteFlow {nodeTypes} />
After defining the "colorPicker" node type, you can apply it to a node by setting the type
node option:
const nodes = writable([
{
id: 'node-1',
// this type needs to match the newly defined node type
type: 'color-picker',
position: { x: 0, y: 0 },
// data is used to store the current color value
data: { color: writable('#ff4000') },
},
]);
The data
field of a node can be used to store whatever data you like. In this case we are storing the current color value in a writable store (opens in a new tab). This makes it easy to update it from within the custom node later. After putting it all together and adding some basic styles, we get a nice looking custom node with a color picker.
Read-onlyAdjust the flow background color
Now that we have a custom node with a color picker, we want to update the background color of the flow. To make this a bit more fun, we add another node and then mix both colors by using the CSS color-mix (opens in a new tab) notation. For this we add another node to the nodes array and then subscribe to the color stores of both nodes. Whenever one of the colors changes, we update the background color of the flow:
Read-onlySuppress unknown prop warnings
Following this guide you will probably wonder why your browser console is flooded with warnings like this:
<CustomNode /> was created with unknown prop 'height'
This is due to the wrapper component always passing every prop to the custom node component, regardless wether you have implemented them in your svelte component or not. Sadly, the only way to surpress the warnings is by actually exporting all props and using them in an empty statement, like this:
<script>
export let id; id;
export let data; data;
export let dragHandle; dragHandle;
export let type; type;
export let selected; selected;
export let isConnectable; isConnectable;
export let zIndex; zIndex;
export let positionAbsolute; positionAbsolute;
export let width; width;
export let height; height;
export let dragging; dragging;
export let targetPosition; targetPosition;
export let sourcePosition; sourcePosition;
</script>
Or if you use Typescript:
<script lang=ts>
type $$Props = NodeProps;
export let id: $$Props['id']; id;
export let data: $$Props['data']; data;
export let dragHandle: $$Props['dragHandle']; dragHandle;
export let type: $$Props['type']; type;
export let selected: $$Props['selected']; selected;
export let isConnectable: $$Props['isConnectable']; isConnectable;
export let zIndex: $$Props['zIndex']; zIndex;
export let positionAbsolute: $$Props['positionAbsolute']; positionAbsolute;
export let width: $$Props['width']; width;
export let height: $$Props['height']; height;
export let dragging: $$Props['dragging']; dragging;
export let targetPosition: $$Props['targetPosition']; targetPosition;
export let sourcePosition: $$Props['sourcePosition']; sourcePosition;
</script>
Help appreciated: If you have a better solution, please let us know. This is quite an annoyance when working with custom nodes and edges. You can find more information in the related Svelte issue (opens in a new tab).