From 20766f539e81279344d5e1a7646cf7c2815e1835 Mon Sep 17 00:00:00 2001 From: marian2js Date: Thu, 10 Aug 2023 15:34:16 +0100 Subject: [PATCH] fix: default link widget with react strict mode --- .changeset/moody-radios-tease.md | 5 + .../src/link/DefaultLinkWidget.tsx | 188 ++++++++---------- 2 files changed, 84 insertions(+), 109 deletions(-) create mode 100644 .changeset/moody-radios-tease.md diff --git a/.changeset/moody-radios-tease.md b/.changeset/moody-radios-tease.md new file mode 100644 index 00000000..f42626ed --- /dev/null +++ b/.changeset/moody-radios-tease.md @@ -0,0 +1,5 @@ +--- +'@projectstorm/react-diagrams-defaults': patch +--- + +fix default link widget with react strict mode diff --git a/packages/react-diagrams-defaults/src/link/DefaultLinkWidget.tsx b/packages/react-diagrams-defaults/src/link/DefaultLinkWidget.tsx index c3c92636..ec9077ac 100644 --- a/packages/react-diagrams-defaults/src/link/DefaultLinkWidget.tsx +++ b/packages/react-diagrams-defaults/src/link/DefaultLinkWidget.tsx @@ -1,9 +1,9 @@ -import * as React from 'react'; import { DiagramEngine, LinkWidget, PointModel } from '@projectstorm/react-diagrams-core'; +import * as React from 'react'; +import { MouseEvent, useEffect, useRef } from 'react'; import { DefaultLinkModel } from './DefaultLinkModel'; import { DefaultLinkPointWidget } from './DefaultLinkPointWidget'; import { DefaultLinkSegmentWidget } from './DefaultLinkSegmentWidget'; -import { MouseEvent } from 'react'; export interface DefaultLinkProps { link: DefaultLinkModel; @@ -13,150 +13,120 @@ export interface DefaultLinkProps { selected?: (event: MouseEvent) => any; } -export interface DefaultLinkState { - selected: boolean; -} +export const DefaultLinkWidget: React.FC = (props) => { + const [selected, setSelected] = React.useState(false); + const refPaths = useRef[]>([]); -export class DefaultLinkWidget extends React.Component { - refPaths: React.RefObject[]; + const renderPoints = () => { + return props.renderPoints ?? true; + }; - constructor(props: DefaultLinkProps) { - super(props); - this.refPaths = []; - this.state = { - selected: false + useEffect(() => { + props.link.setRenderedPaths(refPaths.current.map((ref) => ref.current).filter(Boolean) as SVGPathElement[]); + return () => { + props.link.setRenderedPaths([]); }; - } - - renderPoints() { - return this.props.renderPoints ?? true; - } - - componentDidUpdate(): void { - this.props.link.setRenderedPaths( - this.refPaths.map((ref) => { - return ref.current; - }) - ); - } + }, [props.link]); - componentDidMount(): void { - this.props.link.setRenderedPaths( - this.refPaths.map((ref) => { - return ref.current; - }) - ); - } - - componentWillUnmount(): void { - this.props.link.setRenderedPaths([]); - } + const generateRef = () => { + const ref = React.createRef(); + refPaths.current.push(ref); + return ref; + }; - addPointToLink(event: MouseEvent, index: number) { + const addPointToLink = (event: MouseEvent, index: number) => { if ( !event.shiftKey && - !this.props.link.isLocked() && - this.props.link.getPoints().length - 1 <= this.props.diagramEngine.getMaxNumberPointsPerLink() + !props.link.isLocked() && + props.link.getPoints().length - 1 <= props.diagramEngine.getMaxNumberPointsPerLink() ) { - const position = this.props.diagramEngine.getRelativeMousePoint(event); - const point = this.props.link.point(position.x, position.y, index); + const position = props.diagramEngine.getRelativeMousePoint(event); + const point = props.link.point(position.x, position.y, index); event.persist(); event.stopPropagation(); - this.forceUpdate(() => { - this.props.diagramEngine.getActionEventBus().fireAction({ - event, - model: point - }); + props.diagramEngine.getActionEventBus().fireAction({ + event, + model: point }); } - } + }; - generatePoint(point: PointModel): JSX.Element { + const generatePoint = (point: PointModel): JSX.Element => { return ( ); - } + }; - generateLink(path: string, extraProps: any, id: string | number): JSX.Element { - const ref = React.createRef(); - this.refPaths.push(ref); + const generateLink = (path: string, extraProps: any, id: string | number): JSX.Element => { return ( { - this.setState({ selected: selected }); - }} + selected={selected} + diagramEngine={props.diagramEngine} + factory={props.diagramEngine.getFactoryForLink(props.link)} + link={props.link} + forwardRef={generateRef()} + onSelection={setSelected} extras={extraProps} /> ); - } - - render() { - //ensure id is present for all points on the path - var points = this.props.link.getPoints(); - var paths = []; - this.refPaths = []; + }; + + const points = props.link.getPoints(); + const paths = []; + refPaths.current = []; // Reset the refPaths for the current render + + if (points.length === 2) { + paths.push( + generateLink( + props.link.getSVGPath(), + { + onMouseDown: (event: MouseEvent) => { + props.selected?.(event); + addPointToLink(event, 1); + } + }, + '0' + ) + ); - if (points.length === 2) { + if (props.link.getTargetPort() == null) { + paths.push(generatePoint(points[1])); + } + } else { + for (let j = 0; j < points.length - 1; j++) { paths.push( - this.generateLink( - this.props.link.getSVGPath(), + generateLink( + LinkWidget.generateLinePath(points[j], points[j + 1]), { - onMouseDown: (event) => { - this.props.selected?.(event); - this.addPointToLink(event, 1); + 'data-linkid': props.link.getID(), + 'data-point': j, + onMouseDown: (event: MouseEvent) => { + props.selected?.(event); + addPointToLink(event, j + 1); } }, - '0' + j ) ); + } - // draw the link as dangeling - if (this.props.link.getTargetPort() == null) { - paths.push(this.generatePoint(points[1])); - } - } else { - //draw the multiple anchors and complex line instead - for (let j = 0; j < points.length - 1; j++) { - paths.push( - this.generateLink( - LinkWidget.generateLinePath(points[j], points[j + 1]), - { - 'data-linkid': this.props.link.getID(), - 'data-point': j, - onMouseDown: (event: MouseEvent) => { - this.props.selected?.(event); - this.addPointToLink(event, j + 1); - } - }, - j - ) - ); + if (renderPoints()) { + for (let i = 1; i < points.length - 1; i++) { + paths.push(generatePoint(points[i])); } - if (this.renderPoints()) { - //render the circles - for (let i = 1; i < points.length - 1; i++) { - paths.push(this.generatePoint(points[i])); - } - - if (this.props.link.getTargetPort() == null) { - paths.push(this.generatePoint(points[points.length - 1])); - } + if (props.link.getTargetPort() == null) { + paths.push(generatePoint(points[points.length - 1])); } } - - return {paths}; } -} + + return {paths}; +};