400 lines
7.0 KiB
JavaScript
400 lines
7.0 KiB
JavaScript
|
|
const {
|
|
map,
|
|
values,
|
|
flatten,
|
|
memoizeWith,
|
|
pick,
|
|
} = require('ramda');
|
|
|
|
const React = require('react');
|
|
|
|
const r = require('r-dom');
|
|
|
|
const { connect } = require('react-redux');
|
|
const { bindActionCreators } = require('redux');
|
|
|
|
const {
|
|
GraphView,
|
|
Edge,
|
|
} = require('react-digraph');
|
|
|
|
const math = require('mathjs');
|
|
|
|
const { pulse: pulseActions } = require('../../actions');
|
|
|
|
const { getPaiByTypeAndIndex } = require('../../selectors');
|
|
|
|
Edge.calculateOffset = function (nodeSize, source, target) {
|
|
const arrowVector = math.matrix([ target.x - source.x, target.y - source.y ]);
|
|
const offsetLength = Math.max(0, Math.min((0.85 * size), (math.norm(arrowVector) / 2) - 40));
|
|
const offsetVector = math.dotMultiply(arrowVector, (offsetLength / math.norm(arrowVector)) || 0);
|
|
|
|
return {
|
|
xOff: offsetVector.get([ 0 ]),
|
|
yOff: offsetVector.get([ 1 ]),
|
|
};
|
|
};
|
|
|
|
const weakmapId_ = new WeakMap();
|
|
const weakmapId = o => {
|
|
if (!weakmapId_.has(o)) {
|
|
weakmapId_.set(o, String(Math.random()));
|
|
}
|
|
return weakmapId_.get(o);
|
|
};
|
|
|
|
const dgoToPai = new WeakMap();
|
|
|
|
const memoize = memoizeWith(weakmapId);
|
|
|
|
const key = pao => `${pao.type}-${pao.index}`;
|
|
|
|
const sourceKey = pai => {
|
|
if (pai.clientIndex === -1) {
|
|
return `module-${pai.moduleIndex}`;
|
|
}
|
|
return `client-${pai.clientIndex}`;
|
|
};
|
|
|
|
const targetKey = pai => {
|
|
if (pai.type === 'sinkInput') {
|
|
return `sink-${pai.sinkIndex}`;
|
|
}
|
|
return `source-${pai.sourceIndex}`;
|
|
};
|
|
|
|
const paoToNode = memoize(pao => ({
|
|
id: key(pao),
|
|
index: pao.index,
|
|
type: pao.type,
|
|
}));
|
|
|
|
const paiToEdge = memoize(pai => ({
|
|
source: sourceKey(pai),
|
|
target: targetKey(pai),
|
|
index: pai.index,
|
|
type: pai.type,
|
|
}));
|
|
|
|
const graphConfig = {
|
|
nodeTypes: {},
|
|
|
|
nodeSubtypes: {},
|
|
|
|
edgeTypes: {
|
|
sinkInput: {
|
|
shapeId: '#sinkInput',
|
|
shape: r('symbol', {
|
|
viewBox: '0 0 50 50',
|
|
id: 'sinkInput',
|
|
key: '0',
|
|
}, r.circle({
|
|
cx: '25',
|
|
cy: '25',
|
|
r: '8',
|
|
fill: 'currentColor',
|
|
})),
|
|
},
|
|
sourceOutput: {
|
|
shapeId: '#sourceOutput',
|
|
shape: r('symbol', {
|
|
viewBox: '0 0 50 50',
|
|
id: 'sourceOutput',
|
|
key: '0',
|
|
}, r.circle({
|
|
cx: '25',
|
|
cy: '25',
|
|
r: '8',
|
|
fill: 'currentColor',
|
|
})),
|
|
},
|
|
},
|
|
};
|
|
|
|
class D {
|
|
constructor(s = '') {
|
|
this._s = s;
|
|
}
|
|
|
|
_next(...args) {
|
|
return new this.constructor([ this._s, ...args ].join(' '));
|
|
}
|
|
|
|
moveTo(x, y) {
|
|
return this._next('M', x, y);
|
|
}
|
|
|
|
lineTo(x, y) {
|
|
return this._next('L', x, y);
|
|
}
|
|
|
|
close() {
|
|
return this._next('z');
|
|
}
|
|
|
|
toString() {
|
|
return this._s;
|
|
}
|
|
}
|
|
|
|
const d = () => new D();
|
|
|
|
const size = 120;
|
|
const s2 = size / 2;
|
|
|
|
const Sink = () => r.path({
|
|
d: d()
|
|
.moveTo(-s2, 0)
|
|
.lineTo(-s2 * 1.3, -s2)
|
|
.lineTo(s2, -s2)
|
|
.lineTo(s2, s2)
|
|
.lineTo(-s2 * 1.3, s2)
|
|
.close()
|
|
.toString(),
|
|
});
|
|
|
|
const Source = () => r.path({
|
|
d: d()
|
|
.moveTo(s2 * 1.3, 0)
|
|
.lineTo(s2, s2)
|
|
.lineTo(-s2, s2)
|
|
.lineTo(-s2, -s2)
|
|
.lineTo(s2, -s2)
|
|
.close()
|
|
.toString(),
|
|
});
|
|
|
|
const Client = () => r.path({
|
|
d: d()
|
|
.moveTo(s2 * 1.3, 0)
|
|
.lineTo(s2, s2)
|
|
.lineTo(-s2 * 1.3, s2)
|
|
.lineTo(-s2, 0)
|
|
.lineTo(-s2 * 1.3, -s2)
|
|
.lineTo(s2, -s2)
|
|
.close()
|
|
.toString(),
|
|
});
|
|
|
|
const Module = Client;
|
|
|
|
const gridDotSize = 2;
|
|
const gridSpacing = 36;
|
|
const renderDefs = () => r(React.Fragment, [
|
|
r.pattern({
|
|
id: 'background-pattern',
|
|
key: 'background-pattern',
|
|
width: gridSpacing,
|
|
height: gridSpacing,
|
|
patternUnits: 'userSpaceOnUse',
|
|
}, r.circle({
|
|
className: 'grid-dot',
|
|
cx: (gridSpacing || 0) / 2,
|
|
cy: (gridSpacing || 0) / 2,
|
|
r: gridDotSize,
|
|
})),
|
|
|
|
r('marker', {
|
|
id: 'start-arrow',
|
|
viewBox: '0 -8 16 16',
|
|
refX: '8',
|
|
markerWidth: '16',
|
|
markerHeight: '16',
|
|
orient: 'auto',
|
|
}, r.path({
|
|
className: 'arrow',
|
|
d: 'M 16,-8 L 0,0 L 16,8',
|
|
})),
|
|
]);
|
|
|
|
const renderNode = (nodeRef, data, key, selected, hovered) => r({
|
|
sink: Sink,
|
|
source: Source,
|
|
client: Client,
|
|
module: Module,
|
|
}[data.type], {
|
|
selected,
|
|
hovered,
|
|
});
|
|
|
|
const DebugText = ({ dgo, pai, open = false }) => r.div({
|
|
style: {
|
|
fontSize: '50%',
|
|
},
|
|
}, [
|
|
open && JSON.stringify(dgo, null, 2),
|
|
open && JSON.stringify(pai, null, 2),
|
|
]);
|
|
|
|
const SinkText = ({ dgo, pai }) => r.div([
|
|
r.div({
|
|
title: pai.name,
|
|
}, pai.description),
|
|
r(DebugText, { dgo, pai }),
|
|
]);
|
|
|
|
const SourceText = ({ dgo, pai }) => r.div([
|
|
r.div({
|
|
title: pai.name,
|
|
}, pai.description),
|
|
r(DebugText, { dgo, pai }),
|
|
]);
|
|
|
|
const ClientText = ({ dgo, pai }) => r.div([
|
|
r.div({
|
|
}, pai.name),
|
|
r(DebugText, { dgo, pai }),
|
|
]);
|
|
|
|
const ModuleText = ({ dgo, pai }) => r.div([
|
|
r.div({
|
|
title: pai.properties.module.description,
|
|
}, pai.name),
|
|
r(DebugText, { dgo, pai }),
|
|
]);
|
|
|
|
const renderNodeText = dgo => r('foreignObject', {
|
|
x: -s2,
|
|
y: -s2,
|
|
}, r.div({
|
|
style: {
|
|
width: size,
|
|
height: size,
|
|
|
|
padding: 2,
|
|
|
|
whiteSpace: 'pre',
|
|
},
|
|
}, r({
|
|
sink: SinkText,
|
|
source: SourceText,
|
|
client: ClientText,
|
|
module: ModuleText,
|
|
}[dgo.type] || ModuleText, {
|
|
dgo,
|
|
pai: dgoToPai.get(dgo),
|
|
})));
|
|
|
|
const afterRenderEdge = (id, element, edge, edgeContainer) => {
|
|
if (edge.type) {
|
|
edgeContainer.classList.add(edge.type);
|
|
}
|
|
};
|
|
|
|
class Graph extends React.Component {
|
|
constructor(props) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
selected: null,
|
|
};
|
|
}
|
|
|
|
onSelectNode(selected) {
|
|
this.setState({ selected });
|
|
}
|
|
|
|
onCreateNode() {
|
|
|
|
}
|
|
|
|
onUpdateNode() {
|
|
}
|
|
|
|
onDeleteNode() {
|
|
}
|
|
|
|
onSelectEdge() {
|
|
}
|
|
|
|
onCreateEdge() {
|
|
}
|
|
|
|
onSwapEdge(sourceNode, targetNode, edge) {
|
|
if (edge.type === 'sinkInput') {
|
|
this.props.moveSinkInput(edge.index, targetNode.index);
|
|
} else {
|
|
this.props.moveSourceOutput(edge.index, targetNode.index);
|
|
}
|
|
}
|
|
|
|
onDeleteEdge() {}
|
|
|
|
render() {
|
|
const nodes = map(paoToNode, flatten(map(values, [
|
|
this.props.objects.sinks,
|
|
this.props.objects.sources,
|
|
this.props.objects.clients,
|
|
this.props.objects.modules,
|
|
])));
|
|
const edges = map(paiToEdge, flatten(map(values, [
|
|
this.props.infos.sinkInputs,
|
|
this.props.infos.sourceOutputs,
|
|
])));
|
|
|
|
nodes.forEach(node => {
|
|
if (node.x !== undefined) {
|
|
return;
|
|
}
|
|
|
|
if (node.type === 'source') {
|
|
node.x = 0 * size;
|
|
} else if (node.type === 'sink') {
|
|
node.x = 10 * size;
|
|
} else {
|
|
node.x = (2 * size) + (Math.round(6 * Math.random()) * size);
|
|
}
|
|
|
|
node.y = Math.random() * 1200;
|
|
});
|
|
|
|
nodes.forEach(node => {
|
|
const pai = getPaiByTypeAndIndex(node.type, node.index)({ pulse: this.props });
|
|
dgoToPai.set(node, pai);
|
|
});
|
|
|
|
return r.div({
|
|
id: 'graph',
|
|
style: {},
|
|
}, r(GraphView, {
|
|
nodeKey: 'id',
|
|
|
|
nodes,
|
|
edges,
|
|
|
|
selected: this.state.selected,
|
|
|
|
...graphConfig,
|
|
|
|
onSelectNode: this.onSelectNode.bind(this),
|
|
onCreateNode: this.onCreateNode.bind(this),
|
|
onUpdateNode: this.onUpdateNode.bind(this),
|
|
onDeleteNode: this.onDeleteNode.bind(this),
|
|
onSelectEdge: this.onSelectEdge.bind(this),
|
|
onCreateEdge: this.onCreateEdge.bind(this),
|
|
onSwapEdge: this.onSwapEdge.bind(this),
|
|
onDeleteEdge: this.onDeleteEdge.bind(this),
|
|
|
|
showGraphControls: false,
|
|
|
|
edgeArrowSize: 16,
|
|
|
|
backgroundFillId: '#background-pattern',
|
|
|
|
renderDefs,
|
|
renderNode,
|
|
renderNodeText,
|
|
afterRenderEdge,
|
|
}));
|
|
}
|
|
}
|
|
|
|
module.exports = connect(
|
|
state => state.pulse,
|
|
dispatch => bindActionCreators(pick([
|
|
'moveSinkInput',
|
|
'moveSourceOutput',
|
|
], pulseActions), dispatch),
|
|
)(Graph);
|