Adding HorizontalLayout

This commit is contained in:
Jeff Avallone 2019-01-28 18:27:40 -05:00
parent 4f1ad26635
commit 4e27c4ef87
5 changed files with 146 additions and 11 deletions

View File

@ -0,0 +1,90 @@
import React from 'react';
import PropTypes from 'prop-types';
import * as style from 'rendering/style';
const HorizontalLayout = ({ connectorPath, transforms, children }) => {
return <>
<path style={ style.connectors } d={ connectorPath }/>
{ React.Children.map(children, (child, i) => (
<g transform={ transforms[i] }>{ child }</g>
)) }
</>;
};
HorizontalLayout.defaultProps = {
withConnectors: false,
spacing: 10
};
HorizontalLayout.propTypes = {
spacing: PropTypes.number,
withConnectors: PropTypes.bool,
connectorPath: PropTypes.string,
transforms: PropTypes.arrayOf(PropTypes.string),
children: PropTypes.oneOfType([
PropTypes.arrayOf(PropTypes.node),
PropTypes.node
]).isRequired
};
const generateOffsetBoxes = (boxes, axisY, spacing) => {
let offset = 0;
return boxes.map(box => {
try {
return {
...box,
offsetX: offset,
offsetY: axisY - box.axisY
};
}
finally {
offset += box.width + spacing;
}
});
};
const calculateChildTransforms = boxes =>
boxes.map(box => `translate(${ box.offsetX } ${ box.offsetY })`);
const calculateConnectorPath = (boxes, axisY) => {
let last = boxes[0];
return boxes.slice(1).map(box => {
try {
return `
M${ last.offsetX + last.axisX2 },${ axisY }
H${ box.offsetX + box.axisX1 }
`;
}
finally {
last = box;
}
}).join('');
};
const layout = data => {
const { withConnectors, spacing } = {
...HorizontalLayout.defaultProps,
...data.props
};
const childBoxes = data.children.map(child => child.box);
data.box = {
width: childBoxes.reduce((width, box) => width + box.width, 0) +
(childBoxes.length - 1) * spacing,
height: Math.max(...(childBoxes.map(box => box.axisY))) +
Math.max(...(childBoxes.map(box => box.height - box.axisY))),
axisY: Math.max(...(childBoxes.map(box => box.axisY)))
};
const offsetBoxes = generateOffsetBoxes(childBoxes, data.box.axisY, spacing);
data.props = {
...data.props,
transforms: calculateChildTransforms(offsetBoxes),
connectorPath: withConnectors
? calculateConnectorPath(offsetBoxes, data.box.axisY)
: undefined
};
return data;
};
export default HorizontalLayout;
export { layout };

View File

@ -18,3 +18,8 @@ export const strokeBase = {
strokeWidth: '2px', strokeWidth: '2px',
stroke: black stroke: black
}; };
export const connectors = {
fillOpacity: 0,
...strokeBase
};

View File

@ -1,9 +1,11 @@
import * as SVG from 'rendering/SVG'; import * as SVG from 'rendering/SVG';
import * as Text from 'rendering/Text'; import * as Text from 'rendering/Text';
import * as Box from 'rendering/Box'; import * as Box from 'rendering/Box';
import * as HorizontalLayout from 'rendering/HorizontalLayout';
export default { export default {
SVG, SVG,
Text, Text,
Box Box,
HorizontalLayout
}; };

View File

@ -4,6 +4,12 @@ import layout from 'layout';
const parse = expr => { const parse = expr => {
return { return {
type: 'SVG', type: 'SVG',
children: [
{
type: 'HorizontalLayout',
props: {
withConnectors: true
},
children: [ children: [
{ {
type: 'Box', type: 'Box',
@ -13,11 +19,24 @@ const parse = expr => {
children: [ children: [
{ {
type: 'Text', type: 'Text',
children: [
'JS'
]
}
]
},
{
type: 'Box',
props: { props: {
quoted: true theme: 'literal'
}, },
children: [ children: [
`JS => ${ expr }` {
type: 'Text',
children: [
expr
]
}
] ]
} }
] ]

View File

@ -4,6 +4,12 @@ import layout from 'layout';
const parse = expr => { const parse = expr => {
return { return {
type: 'SVG', type: 'SVG',
children: [
{
type: 'HorizontalLayout',
props: {
withConnectors: true
},
children: [ children: [
{ {
type: 'Box', type: 'Box',
@ -13,11 +19,24 @@ const parse = expr => {
children: [ children: [
{ {
type: 'Text', type: 'Text',
children: [
'PCRE'
]
}
]
},
{
type: 'Box',
props: { props: {
quoted: true theme: 'literal'
}, },
children: [ children: [
`PCRE => ${ expr }` {
type: 'Text',
children: [
expr
]
}
] ]
} }
] ]