Skip to Content

开发工具和调试

¥Devtools and Debugging

这是一项正在进行的实验,旨在实现我们自己的 React Flow devtools。在我们开发实际软件包的同时,我们很乐意听到你对 Discord 或通过 info@xyflow.com 邮件的反馈和想法。

¥This is an ongoing experiment on implementing our own React Flow devtools. While we are working on the actual package, we’d love to hear about your feedback and ideas on Discord or via mail at info@xyflow.com.

React Flow 通常看起来像一个神奇的黑匣子,但实际上,如果你知道在哪里查看,你可以揭示很多有关其内部状态的信息。在本指南中,我们将向你展示三种不同的方式来揭示流程的内部状态:

¥React Flow can often seem like a magic black box, but in reality you can reveal quite a lot about its internal state if you know where to look. In this guide we will show you three different ways to reveal the internal state of your flow:

  • <ViewportLogger /> 组件显示视口的当前位置和缩放级别。

    ¥A <ViewportLogger /> component that shows the current position and zoom level of the viewport.

  • 显示每个节点状态的 <NodeInspector /> 组件。

    ¥A <NodeInspector /> component that reveals the state of each node.

  • <ChangeLogger /> 封装流的 onNodesChange 处理程序并在分派时记录每个更改。

    ¥A <ChangeLogger /> that wraps your flow’s onNodesChange handler and logs each change as it is dispatched.

虽然我们发现这些工具对于确保 React Flow 正常工作很有用,但你可能还会发现它们对于调试你的应用很有用,因为你的流程及其交互变得更加复杂。

¥While we find these tools useful for making sure React Flow is working properly, you might also find them useful for debugging your applications as your flows and their interactions become more complex.

import { useCallback } from 'react'; import { ReactFlow, addEdge, useEdgesState, useNodesState, type Edge, type OnConnect, type Node, } from '@xyflow/react'; import '@xyflow/react/dist/style.css'; import DevTools from './Devtools'; const initNodes: Node[] = [ { id: '1a', type: 'input', data: { label: 'Node 1' }, position: { x: 250, y: 5 }, }, { id: '2a', data: { label: 'Node 2' }, position: { x: 100, y: 120 }, }, { id: '3a', data: { label: 'Node 3' }, position: { x: 400, y: 120 }, }, ]; const initEdges: Edge[] = [ { id: 'e1-2', source: '1a', target: '2a' }, { id: 'e1-3', source: '1a', target: '3a' }, ]; const fitViewOptions = { padding: 0.5 }; function Flow() { const [nodes, , onNodesChange] = useNodesState(initNodes); const [edges, setEdges, onEdgesChange] = useEdgesState(initEdges); const onConnect: OnConnect = useCallback( (params) => setEdges((eds) => addEdge(params, eds)), [setEdges], ); return ( <ReactFlow nodes={nodes} edges={edges} onNodesChange={onNodesChange} onEdgesChange={onEdgesChange} onConnect={onConnect} fitView fitViewOptions={fitViewOptions} > <DevTools /> </ReactFlow> ); } export default Flow;

我们鼓励你将此示例中的任何或所有组件复制到你自己的项目中并根据你的需要进行修改:每个组件独立工作!

¥We encourage you to copy any or all of the components from this example into your own projects and modify them to suit your needs: each component works independently!

节点检查器

¥Node Inspector

<NodeInspector /> 组件利用我们的 useNodes 钩子访问流中的所有节点。通常我们不鼓励使用此钩子,因为它会在你的任何节点发生更改时触发重新渲染,但这正是它在调试中如此有用的原因!

¥The <NodeInspector /> component makes use of our useNodes hook to access all the nodes in the flow. Typically we discourage using this hook because it will trigger a re-render any time any of your nodes change, but that’s exactly what makes it so useful for debugging!

React Flow 在测量节点的尺寸后,将 widthheight 属性添加到每个节点。我们将这些维度以及其他信息(如节点的 id 和类型)传递给自定义 <NodeInfo /> 组件。

¥The width and height properties are added to each node by React Flow after it has measured the node’s dimensions. We pass those dimensions, as well as other information like the node’s id and type, to a custom <NodeInfo /> component.

我们使用 <ViewportPortal /> 组件让我们将检查器渲染到 React Flow 的视口中。这意味着它的内容将在用户平移和缩放时与其余流一起定位和转换。

¥We make use of the <ViewportPortal /> component to let us render the inspector into React Flow’s viewport. That means it’s content will be positioned and transformed along with the rest of the flow as the user pans and zooms.

变更日志器

¥Change Logger

源自 React Flow 本身的任何对节点和边缘的更改都会通过 onNodesChangeonEdgesChange 回调传达给你。如果你使用的是受控流(这意味着你自己管理节点和边缘),则需要将这些更改应用于你的状态以使所有内容保持同步。

¥Any change to your nodes and edges that originates from React Flow itself is communicated to you through the onNodesChange and onEdgesChange callbacks. If you are working with a controlled flow (that means you’re managing the nodes and edges yourself), you need to apply those changes to your state in order to keep everything in sync.

<ChangeLogger /> 组件使用自定义函数封装用户提供的 onNodesChange 处理程序,该函数在分派时拦截并记录每个更改。我们可以通过使用 useStoreuseStoreApi 钩子来访问存储,然后相应地更新 React Flow 的内部状态来实现这一点。这两个钩子为你提供了对 React Flow 内部状态和方法的强大访问。

¥The <ChangeLogger /> component wraps your user-provided onNodesChange handler with a custom function that intercepts and logs each change as it is dispatched. We can do this by using the useStore and useStoreApi hooks to access the store and and then update React Flow’s internal state accordingly. These two hooks give you powerful access to React Flow’s internal state and methods.

除了调试之外,使用 <ChangeLogger /> 可以成为了解 React Flow 工作原理的绝佳方式,让你思考可以在每个更改的基础上构建的不同功能。

¥Beyond debugging, using the <ChangeLogger /> can be a great way to learn more about how React Flow works and get you thinking about the different functionality you can build on top of each change.

你可以在 API 参考中找到有关 NodeChangeEdgeChange 类型的文档。

¥You can find documentation on the NodeChange and EdgeChange types in the API reference.

视口日志器

¥Viewport Logger

如果你知道要查找什么,<ViewportLogger /> 就是你可以从 React Flow 的存储中提取什么状态的最简单的示例。视口的状态内部存储在 transform 键下(我们从 d3-zoom 继承的名称)。此组件提取转换的 xyzoom 组件并将它们渲染为 <Panel /> 组件。

¥The <ViewportLogger /> is the simplest example of what state you can pull out of React Flow’s store if you know what to look for. The state of the viewport is stored internally under the transform key (a name we inherited from d3-zoom). This component extracts the x, y, and zoom components of the transform and renders them into a <Panel /> component.

让我们知道你的想法

¥Let us know what you think

如上所述,如果你对如何改进 devtools 有任何反馈或想法,请在 Discord 或通过 info@xyflow.com 邮件告知我们。如果你使用这些想法构建自己的 devtools,我们很乐意听到它!

¥As mentioned above, if you have any feedback or ideas on how to improve the devtools, please let us know on Discord or via mail at info@xyflow.com. If you build your own devtools using these ideas, we’d love to hear about it!

Last updated on