react-xarrows
introduction
Draw arrows between components in React!
This library is all about customizable and reliable arrows(or lines) between DOM elements in React.
I've needed such a components in one of my projects - and found out(surprisingly enough) that there is no such(good) lib, so I've decided to create one from scratch, and share it.
now (since v1.1.4) i can say, after a lot of tests and improvements- the arrows should work and act naturally under any normal circumstances(but don't try put some negative curveness ha?).
found a problem? not a problem! post a new issue(here) and i will do my best to fix it.
liked my work? please star this repo.
this project developed with the help of using codesandbox. see and fork easily here.
installation
with npm npm install react-xarrows
.
(or yarn add react-xarrows
)
Examples
see here! codebox of few examples(in this repo at /examples).
simple example:
import React, { useRef } from "react";
import Xarrow from "react-xarrows";
const boxStyle = {
border: "grey solid 2px",
borderRadius: "10px",
padding: "5px",
};
function SimpleExample() {
const box1Ref = useRef(null);
return (
<div style={{ display: "flex", justifyContent: "space-evenly", width: "100%" }}>
<div ref={box1Ref} style={boxStyle}>
hey
</div>
<p id="elem2" style={boxStyle}>
hey2
</p>
<Xarrow
start={box1Ref} //can be react ref
end="elem2" //or an id
/>
</div>
);
}
export default SimpleExample;
The API
types definitions
the properties the xarrow component receives is as follow:
export type xarrowPropsType = {
start: refType;
end: refType;
startAnchor?: anchorType | anchorType[];
endAnchor?: anchorType | anchorType[];
label?: labelType | labelsType;
color?: string;
lineColor?: string | null;
headColor?: string | null;
strokeWidth?: number;
headSize?: number;
path: "smooth" | "grid" | "straight";
curveness?: number;
dashness?: boolean | { strokeLen?: number; nonStrokeLen?: number; animation?: boolean | number };
consoleWarning?: boolean;
passProps?: React.SVGProps<SVGPathElement>;
advanced?: {
extendSVGcanvas?: number;
passProps?: {
SVGcanvas?: React.SVGAttributes<SVGSVGElement>;
arrowBody?: React.SVGProps<SVGPathElement>;
arrowHead?: React.SVGProps<SVGPathElement>;
};
};
monitorDOMchanges?: boolean;
registerEvents?: registerEventsType[];
};
export type anchorType = anchorPositionType | anchorCustomPositionType;
export type anchorPositionType = "middle" | "left" | "right" | "top" | "bottom" | "auto";
export type anchorCustomPositionType = {
position: anchorPositionType;
offset: { rightness: number; bottomness: number };
};
export type reactRefType = { current: null | HTMLElement };
export type refType = reactRefType | string;
export type labelsType = { start?: labelType; middle?: labelType; end?: labelType };
export type labelType = JSX.Element;
export type domEventType = keyof GlobalEventHandlersEventMap;
export type registerEventsType = {
ref: refType;
eventName: domEventType;
callback?: CallableFunction;
};
you can keep things simple or provide more detailed props for more custom behavior - the API except both.
for example - you can provide label:"middleLabel"
and the string will appear as middle label or customize the labels as you please: label:{end:{text:"end",extra:{fill:"red",dx:-10}}}
.
see typescript types above for detailed descriptions of what type excepts every prop.
'start' and 'end'
can be a reference to a react ref to html element or string - an id of a DOM element.
'startAnchor' and 'endAnchor'
each anchor can be: "auto" | "middle" | "left" | "right" | "top" | "bottom"
.
auto
will choose automatically the path with the smallest length.
can also be a list of possible anchors. if list is provided - the minimal length anchors will be choose from the list.
you can also offset each anchor passing offset
.
examples:
endAnchor="middle"
will choose anchor or the end of the line to in the middle of the elementendAnchor= { position: "auto", offset: { rightness: 20 } }
will choose automatic anchoring for end anchor but will offset it 20 pixels to the right after normal positioning.
label
changed api since v1.4.0. now each label can be jsx element or string, but not object.
can be a string that will default to be at the middle or an object that describes where to place label and how to customize it. see label
at xarrowPropsType
above.
examples:
label="middleLabel"
label=<div style={{ fontSize: "1.3em", fontFamily: "fantasy", fontStyle: "italic" }}>styled middle label</div>
label={{ start:"I'm start label",middle: "middleLabel",end:<div style={{ fontSize: "1.3em", fontFamily: "fantasy", fontStyle: "italic" }}>big end label</div> }}
color,lineColor and headColor
color defines color for all the arrow include head. if lineColor or headColor is given so it overrides color specifically for line or head.
examples:
color="red"
will change the color of the arrow to red(body and head).headColor="red"
will change the color of the head of the arrow to red.lineColor="red"
will change the color of the body of the arrow to red.
(NOTE - maybe headColor
and lineColor
will be combined to color)
strokeWidth and headSize
strokeWidth defines the thickness of the arrow(line and head).
headSize defines how big will be the head relative to the line(set headSize to 0 to make the arrow like a line).
examples:
strokeWidth={15}
will make the arrow more thick(body and head).headSize={15}
will make the head of the arrow more thick(relative to strokeWidth as well).
path
path
can be one of: "smooth" | "grid" | "straight"
, and it controls the path arrow is drawn, exactly how their name suggest.
examples:
path={"grid"}
will draw the line in sharp curves(90 degrees) like grid.
curveness
defines how much the lines curve.
examples:
curveness={false}
will make the line straight without curves(exactly like path='straight').curveness={true}
will choose default values of curveness.curveness={2}
will make Xarrow extra curved.
dashness
can make the arrow dashed and can even animate.
if true default values(for dashness) are choose. if object is passed then default values are choose except what passed.
examples:
dashness={true}
will make the line of the arrow to be dashed.dashness={{ strokeLen: 10, nonStrokeLen: 15, animation: -2 }}
will make a custom looking dashness.
passProps
new and powerful feature!
you can pass properties to visible parts of the arrow (such event handlers and much more).
examples:
passProps= {{onClick: () => console.log("xarrow clicked!")}}
- now the arrow will console log a message when clicked.passProps= {{cursor: "pointer"}}
- now the cursor will change to pointer style when hovering over Xarrow.passProps= {{pointerEvents: "none"}}
- now the user cannot interact with Xarrow via mouse events.
consoleWarning
we provide some nice warnings (and errors) whenever we detect issues. see 'Example3' at the examples codesandbox.
advanced
here i will provide some flexibility to the API for some cases that i may not thought of.
extendSVGcanvas
will extend the svg canvas at all sides. can be useful if for some reason the arrow(or labels) is cut though to small svg canvas(should be used in advanced custom arrows, for example if you used dx
to move one of the labels and at exceeded the canvas).
example: advanced= {{extendSVGcanvas: 30 }}
- will extended svg canvas in all sides by 30 pixels.
passProps
if you wish you can pass props specifically to either the body of the arrow,or his head,or even the svg canvas which contains both of them.
note that arrowBody
and arrowHead
receives props of svg path element and SVGcanvas
receives props of svg element.
examples:
advanced= {{passProps: {arrowHead:{onClick: () => console.log("head clicked!")}}}}
- now only the head will console log a message when clicked.
monitorDOMchanges
A boolean. set this property to true to add relevant eventListeners to the DOM so the xarrow component will update anchors position whenever needed(scroll and resize and so on). (NOTE - maybe will removed in future updates )
examples:
monitorDOMchanges={false}
will disable any DOM monitoring.
registerEvents
you can register the xarrow to DOM event as you please. each time a event that his registered will fire the xarrow component will update his position and will call callback
(if provided). (NOTE - planned to be removed)
default props
default props is as follows:
Xarrow.defaultProps = {
startAnchor: "auto",
endAnchor: "auto",
label: null,
color: "CornflowerBlue",
lineColor: null,
headColor: null,
strokeWidth: 4,
headSize: 6,
path: "smooth",
curveness: 0.8,
dashness: false,
consoleWarning: false,
passProps: {},
advanced: { extendSVGcanvas: 0, passProps: { arrowBody: {}, arrowHead: {}, SVGcanvas: {} } },
monitorDOMchanges: true,
registerEvents: [],
};
};
Versions
All version notes moved to releases.
1.4.0 - changed label
property API and added path
property.
1.4.1-2 - minor bug fixes, performance improvement.