A React-based Flexbox layout container for SVG elements.
SVG lacks the convenient layout features of CSS; all elements must be positioned absolutely. There are ways around this (with foreignObject for instance), but there are cases where you just end up having to write cumbersome layout calculations for absolute positions. This component attempts to free us from having to do that.
Swizec Teller did a nice write-up that explains why he likes it and provides numerous examples: Build responsive SVG layouts with react-svg-flexbox.
No. It computes positions using Facebook's css-layout, which implements a subset of the Flexbox algorithm. This table shows what is supported. These settings are not supported:
On the container:
align-content: space-between
align-items: baseline
align-items: stretch
flex-wrap: wrap-reverse
justify-content: space-evenly
On the children:
align-self
flex-basis
flex-grow
flex-shrink
order
Despite these omissions, css-layout implements enough of Flexbox to be useful.
Live demo: zoopoetics.github.io/react-svg-flexbox
Here is the source for the examples. To run them locally: yarn install && yarn start
.
react-svg-flexbox is intended to be installed from npm and bundled into your React app.
yarn add react-svg-flexbox
Add an instance of the Flexbox
component to your app, and style it to taste. CSS-Tricks has a great overview of Flexbox if you need help on that score.
import React from 'react';
import Flexbox from 'react-svg-flexbox';
export default class App extends React.Component {
render() {
return (
<svg height={600} width={800}>
<Flexbox
style={{
flexDirection: 'row',
justifyContent: 'flex-start',
}}>
<rect fill={'#f0c'} height={10} width={10} />
<rect fill={'#f0c'} height={10} width={20} />
</Flexbox>
</svg>
);
}
}
For how to achieve specific layouts using this component, see the examples.
Arbitrary string to set as className
on the root element of the Flexbox
instance. By default, the component has no class name.
Callback that will receive the entire computed layout when the layout updates. This is useful if you need to inspect layout values from a Flexbox
instance in order to make decisions elsewhere in your application.
Object containing Flexbox settings, as well as optional width and/or height. You have to pass something in order for layout to occur.
The horizontal position, in pixels, at which the Flexbox
instance should appear within its parent element.
The vertical position, in pixels, at which the Flexbox
instance should appear within its parent element.
When the Flexbox
component lays out a native SVG element, it looks at the element's type and sets its positional attributes accordingly. When Flexbox
encounters a React component, however, it has no element type. As such, Flexbox
passes x
and y
props to the React component instance and expects it to know how to deal with them. Take this usage:
import React from 'react';
import Flexbox from 'react-svg-flexbox';
import CustomComponent from 'my-app/components';
export default class App extends React.Component {
render() {
return (
<svg height={600} width={800}>
<Flexbox>
<CustomComponent />
</Flexbox>
</svg>
);
}
}
In this case, CustomComponent
must expect x
and y
props and use them to position its sub-elements, like so:
import React from 'react';
import PropTypes from 'prop-types';
export default class CustomComponent extends React.Component {
static propTypes = {
x: PropTypes.number,
y: PropTypes.number,
};
render() {
return (
<g
className={'CustomComponent'}
transform={`translate(${this.props.x} ${this.props.y})`}>
{/* stuff */}
</g>
);
}
}
If you don't do this, your layout values will disappear meaninglessly into the void.
You might have to fiddle with baseline style in order to get text elements into the correct vertical alignment.