init
This commit is contained in:
commit
a0e63e4f94
12
.editorconfig
Normal file
12
.editorconfig
Normal file
|
@ -0,0 +1,12 @@
|
|||
root = true
|
||||
|
||||
[*]
|
||||
indent_style = tab
|
||||
end_of_line = lf
|
||||
charset = utf-8
|
||||
trim_trailing_whitespace = true
|
||||
insert_final_newline = true
|
||||
|
||||
[{package.json,*.yml}]
|
||||
indent_style = space
|
||||
indent_size = 2
|
76
.gitignore
vendored
Normal file
76
.gitignore
vendored
Normal file
|
@ -0,0 +1,76 @@
|
|||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
|
||||
# Runtime data
|
||||
pids
|
||||
*.pid
|
||||
*.seed
|
||||
*.pid.lock
|
||||
|
||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||
lib-cov
|
||||
|
||||
# Coverage directory used by tools like istanbul
|
||||
coverage
|
||||
|
||||
# nyc test coverage
|
||||
.nyc_output
|
||||
|
||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||
.grunt
|
||||
|
||||
# Bower dependency directory (https://bower.io/)
|
||||
bower_components
|
||||
|
||||
# node-waf configuration
|
||||
.lock-wscript
|
||||
|
||||
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||
build/Release
|
||||
|
||||
# Dependency directories
|
||||
node_modules/
|
||||
jspm_packages/
|
||||
|
||||
# TypeScript v1 declaration files
|
||||
typings/
|
||||
|
||||
# Optional npm cache directory
|
||||
.npm
|
||||
|
||||
# Optional eslint cache
|
||||
.eslintcache
|
||||
|
||||
# Optional REPL history
|
||||
.node_repl_history
|
||||
|
||||
# Output of 'npm pack'
|
||||
*.tgz
|
||||
|
||||
# Yarn Integrity file
|
||||
.yarn-integrity
|
||||
|
||||
# dotenv environment variables file
|
||||
.env
|
||||
|
||||
# parcel-bundler cache (https://parceljs.org/)
|
||||
.cache
|
||||
|
||||
# next.js build output
|
||||
.next
|
||||
|
||||
# nuxt.js build output
|
||||
.nuxt
|
||||
|
||||
# vuepress build output
|
||||
.vuepress/dist
|
||||
|
||||
# Serverless directories
|
||||
.serverless
|
||||
|
||||
# FuseBox cache
|
||||
.fusebox/
|
5
actions/index.js
Normal file
5
actions/index.js
Normal file
|
@ -0,0 +1,5 @@
|
|||
|
||||
module.exports = Object.assign(
|
||||
{},
|
||||
require('./pulse'),
|
||||
);
|
18
actions/pulse.js
Normal file
18
actions/pulse.js
Normal file
|
@ -0,0 +1,18 @@
|
|||
|
||||
const { createActions: createActionCreators } = require('redux-actions');
|
||||
|
||||
module.exports = createActionCreators({
|
||||
PULSE: {
|
||||
READY: null,
|
||||
CLOSE: null,
|
||||
|
||||
NEW: null,
|
||||
CHANGE: null,
|
||||
REMOVE: null,
|
||||
|
||||
INFO: null,
|
||||
|
||||
MOVE_SINK_INPUT: (sinkInputIndex, destSinkIndex) => ({ sinkInputIndex, destSinkIndex }),
|
||||
MOVE_SOURCE_OUTPUT: (sourceOutputIndex, destSourceIndex) => ({ sourceOutputIndex, destSourceIndex }),
|
||||
},
|
||||
});
|
149
components/graph.bak.bak/index.js
Normal file
149
components/graph.bak.bak/index.js
Normal file
|
@ -0,0 +1,149 @@
|
|||
|
||||
const {
|
||||
map,
|
||||
values,
|
||||
flatten,
|
||||
merge,
|
||||
indexBy,
|
||||
} = require('ramda');
|
||||
|
||||
const React = require('react');
|
||||
|
||||
const r = require('r-dom');
|
||||
|
||||
const { connect } = require('react-redux');
|
||||
|
||||
const d3 = require('d3');
|
||||
|
||||
const key = pao => `${pao.type}-${pao.index}`;
|
||||
|
||||
class Graph extends React.Component {
|
||||
constructor(props) {
|
||||
super(props);
|
||||
|
||||
this.container = React.createRef();
|
||||
|
||||
this.connectionToPao = new WeakMap();
|
||||
}
|
||||
|
||||
componentDidMount() {
|
||||
this.svg = d3
|
||||
.select(this.container.current)
|
||||
.append('svg');
|
||||
}
|
||||
|
||||
_connectSinkInput(sinkInput) {
|
||||
const connection = this.jsPlumb.connect({
|
||||
source: `client-${sinkInput.info.clientIndex}`,
|
||||
target: `sink-${sinkInput.info.sinkIndex}`,
|
||||
anchor: 'Continuous',
|
||||
overlays: [ [ 'Arrow', { location: -1 } ] ],
|
||||
});
|
||||
this.connectionToPao.set(connection, sinkInput);
|
||||
}
|
||||
|
||||
_connectSourceOutput(sourceOutput) {
|
||||
const connection = this.jsPlumb.connect({
|
||||
source: `source-${sourceOutput.info.sourceIndex}`,
|
||||
target: `client-${sourceOutput.info.clientIndex}`,
|
||||
anchor: 'Continuous',
|
||||
overlays: [ [ 'Arrow', { location: -1 } ] ],
|
||||
});
|
||||
this.connectionToPao.set(connection, sourceOutput);
|
||||
}
|
||||
|
||||
_connectByType(pao) {
|
||||
if (pao.type === 'sinkInput') {
|
||||
this._connectSinkInput(pao);
|
||||
} else {
|
||||
this._connectSourceOutput(pao);
|
||||
}
|
||||
}
|
||||
|
||||
_handleResize() {
|
||||
this.svg
|
||||
.attr('width', this.container.current.clientWidth)
|
||||
.attr('height', this.container.current.clientHeight);
|
||||
}
|
||||
|
||||
componentDidUpdate() {
|
||||
this._handleResizeBound = this._handleResize.bind(this);
|
||||
this.container.current.ownerDocument.defaultView.addEventListener('resize', this._handleResizeBound);
|
||||
|
||||
/*
|
||||
this.jsPlumb.batch(() => {
|
||||
const propsPaos = merge(
|
||||
indexBy(key, values(this.props.sinkInputs)),
|
||||
indexBy(key, values(this.props.sourceOutputs)),
|
||||
);
|
||||
this.jsPlumb.getAllConnections().forEach(connection => {
|
||||
const connectionPao = this.connectionToPao.get(connection);
|
||||
const k = key(connectionPao);
|
||||
const propsPao = propsPaos[k];
|
||||
if (!propsPao) {
|
||||
this.jsPlumb.deleteConnection(connection);
|
||||
} else if (propsPao === connectionPao) {
|
||||
// Noop
|
||||
} else if (propsPao.info) {
|
||||
this.jsPlumb.deleteConnection(connection);
|
||||
this._connectByType(propsPao);
|
||||
}
|
||||
delete propsPaos[k];
|
||||
});
|
||||
values(propsPaos).forEach(propsPao => {
|
||||
if (propsPao.info) {
|
||||
this._connectByType(propsPao);
|
||||
}
|
||||
});
|
||||
});
|
||||
*/
|
||||
|
||||
/*
|
||||
flatten(
|
||||
map(paos => map(pao => r.div({
|
||||
id: key(pao),
|
||||
key: key(pao),
|
||||
className: 'jtk-node',
|
||||
style: {
|
||||
border: '1px solid black',
|
||||
userSelect: 'none',
|
||||
cursor: 'default',
|
||||
},
|
||||
ref: el => {
|
||||
if (el) {
|
||||
this.jsPlumb.draggable(el, {});
|
||||
/// this.jsPlumb.addEndpoint();
|
||||
}
|
||||
},
|
||||
}, [
|
||||
key(pao),
|
||||
]), values(paos)), [
|
||||
this.props.sinks,
|
||||
this.props.sources,
|
||||
this.props.clients,
|
||||
]),
|
||||
)
|
||||
*/
|
||||
}
|
||||
|
||||
componentWillUnmount() {
|
||||
this.container.current.ownerDocument.defaultView.removeEventListener('resize', this._handleResizeBound);
|
||||
}
|
||||
|
||||
render() {
|
||||
return r.div({
|
||||
ref: this.container,
|
||||
style: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
top: 0,
|
||||
bottom: 0,
|
||||
},
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = connect(
|
||||
state => state.pulse,
|
||||
)(Graph);
|
126
components/graph.bak/index.js
Normal file
126
components/graph.bak/index.js
Normal file
|
@ -0,0 +1,126 @@
|
|||
|
||||
const {
|
||||
map,
|
||||
values,
|
||||
flatten,
|
||||
merge,
|
||||
indexBy,
|
||||
} = require('ramda');
|
||||
|
||||
const React = require('react');
|
||||
|
||||
const r = require('r-dom');
|
||||
|
||||
const { connect } = require('react-redux');
|
||||
|
||||
const {
|
||||
jsPlumb,
|
||||
} = require('jsplumb');
|
||||
|
||||
const key = pao => `${pao.type}-${pao.index}`;
|
||||
|
||||
class Graph extends React.Component {
|
||||
constructor(props) {
|
||||
super(props);
|
||||
|
||||
this.container = React.createRef();
|
||||
|
||||
this.connectionToPao = new WeakMap();
|
||||
}
|
||||
|
||||
componentDidMount() {
|
||||
this.jsPlumb = jsPlumb.getInstance({
|
||||
Container: this.container.current,
|
||||
});
|
||||
}
|
||||
|
||||
_connectSinkInput(sinkInput) {
|
||||
const connection = this.jsPlumb.connect({
|
||||
source: `client-${sinkInput.info.clientIndex}`,
|
||||
target: `sink-${sinkInput.info.sinkIndex}`,
|
||||
anchor: 'Continuous',
|
||||
overlays: [ [ 'Arrow', { location: -1 } ] ],
|
||||
});
|
||||
this.connectionToPao.set(connection, sinkInput);
|
||||
}
|
||||
|
||||
_connectSourceOutput(sourceOutput) {
|
||||
const connection = this.jsPlumb.connect({
|
||||
source: `source-${sourceOutput.info.sourceIndex}`,
|
||||
target: `client-${sourceOutput.info.clientIndex}`,
|
||||
anchor: 'Continuous',
|
||||
overlays: [ [ 'Arrow', { location: -1 } ] ],
|
||||
});
|
||||
this.connectionToPao.set(connection, sourceOutput);
|
||||
}
|
||||
|
||||
_connectByType(pao) {
|
||||
if (pao.type === 'sinkInput') {
|
||||
this._connectSinkInput(pao);
|
||||
} else {
|
||||
this._connectSourceOutput(pao);
|
||||
}
|
||||
}
|
||||
|
||||
componentDidUpdate() {
|
||||
this.jsPlumb.batch(() => {
|
||||
const propsPaos = merge(
|
||||
indexBy(key, values(this.props.sinkInputs)),
|
||||
indexBy(key, values(this.props.sourceOutputs)),
|
||||
);
|
||||
this.jsPlumb.getAllConnections().forEach(connection => {
|
||||
const connectionPao = this.connectionToPao.get(connection);
|
||||
const k = key(connectionPao);
|
||||
const propsPao = propsPaos[k];
|
||||
if (!propsPao) {
|
||||
this.jsPlumb.deleteConnection(connection);
|
||||
} else if (propsPao === connectionPao) {
|
||||
// Noop
|
||||
} else if (propsPao.info) {
|
||||
this.jsPlumb.deleteConnection(connection);
|
||||
this._connectByType(propsPao);
|
||||
}
|
||||
delete propsPaos[k];
|
||||
});
|
||||
values(propsPaos).forEach(propsPao => {
|
||||
if (propsPao.info) {
|
||||
this._connectByType(propsPao);
|
||||
}
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
render() {
|
||||
return r.div({
|
||||
ref: this.container,
|
||||
style: { position: 'relative' },
|
||||
}, flatten(
|
||||
map(paos => map(pao => r.div({
|
||||
id: key(pao),
|
||||
key: key(pao),
|
||||
className: 'jtk-node',
|
||||
style: {
|
||||
border: '1px solid black',
|
||||
userSelect: 'none',
|
||||
cursor: 'default',
|
||||
},
|
||||
ref: el => {
|
||||
if (el) {
|
||||
this.jsPlumb.draggable(el, {});
|
||||
/// this.jsPlumb.addEndpoint();
|
||||
}
|
||||
},
|
||||
}, [
|
||||
key(pao),
|
||||
]), values(paos)), [
|
||||
this.props.sinks,
|
||||
this.props.sources,
|
||||
this.props.clients,
|
||||
]),
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
module.exports = connect(
|
||||
state => state.pulse,
|
||||
)(Graph);
|
399
components/graph/index.js
Normal file
399
components/graph/index.js
Normal file
|
@ -0,0 +1,399 @@
|
|||
|
||||
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);
|
34
constants/pulse.js
Normal file
34
constants/pulse.js
Normal file
|
@ -0,0 +1,34 @@
|
|||
|
||||
const things = [ {
|
||||
method: 'getModules',
|
||||
type: 'module',
|
||||
key: 'modules',
|
||||
}, {
|
||||
method: 'getCards',
|
||||
type: 'card',
|
||||
key: 'cards',
|
||||
}, {
|
||||
method: 'getClients',
|
||||
type: 'client',
|
||||
key: 'clients',
|
||||
}, {
|
||||
method: 'getSinks',
|
||||
type: 'sink',
|
||||
key: 'sinks',
|
||||
}, {
|
||||
method: 'getSources',
|
||||
type: 'source',
|
||||
key: 'sources',
|
||||
}, {
|
||||
method: 'getSinkInputs',
|
||||
type: 'sinkInput',
|
||||
key: 'sinkInputs',
|
||||
}, {
|
||||
method: 'getSourceOutputs',
|
||||
type: 'sourceOutput',
|
||||
key: 'sourceOutputs',
|
||||
} ];
|
||||
|
||||
module.exports = {
|
||||
things,
|
||||
};
|
41
index.css
Normal file
41
index.css
Normal file
|
@ -0,0 +1,41 @@
|
|||
body {
|
||||
margin: 0;
|
||||
background: var(--themeBaseColor);
|
||||
color: var(--themeTextColor);
|
||||
font: -webkit-control;
|
||||
}
|
||||
|
||||
.view-wrapper .graph {
|
||||
background: var(--themeBaseColor);
|
||||
}
|
||||
|
||||
.view-wrapper .grid-dot {
|
||||
fill: var(--themeBgColor);
|
||||
}
|
||||
|
||||
.view-wrapper .node {
|
||||
fill: var(--themeBgColor);
|
||||
stroke: var(--borders);
|
||||
}
|
||||
|
||||
.view-wrapper .node.hovered {
|
||||
stroke: var(--themeSelectedBgColor);
|
||||
}
|
||||
.view-wrapper .node.selected {
|
||||
fill: var(--themeSelectedBgColor);
|
||||
color: var(--themeSelectedFgColor);
|
||||
}
|
||||
|
||||
.view-wrapper .sourceOutput .edge {
|
||||
/* marker-end: none; */
|
||||
/* marker-start: url(#start-arrow); */
|
||||
marker-end: url(#start-arrow);
|
||||
}
|
||||
|
||||
.view-wrapper .graph .edge {
|
||||
stroke: var(--successColor);
|
||||
}
|
||||
|
||||
.view-wrapper .graph .arrow {
|
||||
fill: var(--successColor);
|
||||
}
|
3
index.html
Normal file
3
index.html
Normal file
|
@ -0,0 +1,3 @@
|
|||
<link rel="stylesheet" href="./index.css">
|
||||
<div id="root"></div>
|
||||
<script>require('./renderer.js')</script>
|
11
index.js
Normal file
11
index.js
Normal file
|
@ -0,0 +1,11 @@
|
|||
|
||||
const { app, BrowserWindow } = require('electron');
|
||||
|
||||
const theme = require('./utils/theme');
|
||||
|
||||
app.on('ready', () => {
|
||||
const win = new BrowserWindow({
|
||||
backgroundColor: theme.colors.themeBaseColor,
|
||||
});
|
||||
win.loadFile('index.html');
|
||||
});
|
43
package.json
Normal file
43
package.json
Normal file
|
@ -0,0 +1,43 @@
|
|||
{
|
||||
"name": "pagraphcontrol3",
|
||||
"version": "1.0.0",
|
||||
"main": "index.js",
|
||||
"license": "GPL-3.0",
|
||||
"devDependencies": {
|
||||
"ava": "^0.25.0",
|
||||
"electron": "^3.0.8",
|
||||
"eslint-config-xo-overrides": "^1.1.2",
|
||||
"remotedev-server": "^0.2.6",
|
||||
"uws": "^99.0.0",
|
||||
"xo": "^0.23.0"
|
||||
},
|
||||
"scripts": {
|
||||
"start": "electron ."
|
||||
},
|
||||
"xo": {
|
||||
"extends": [
|
||||
"eslint-config-xo-overrides"
|
||||
]
|
||||
},
|
||||
"dependencies": {
|
||||
"@jakejarrett/gtk-theme": "^1.1.2",
|
||||
"camelcase": "^5.0.0",
|
||||
"mathjs": "^5.2.3",
|
||||
"paclient": "^0.0.2",
|
||||
"r-dom": "^2.4.0",
|
||||
"ramda": "^0.25.0",
|
||||
"react": "^16.6.0",
|
||||
"react-digraph": "^5.1.3",
|
||||
"react-dom": "^16.6.0",
|
||||
"react-redux": "^5.1.0",
|
||||
"recompose": "^0.30.0",
|
||||
"redux": "^4.0.1",
|
||||
"redux-actions": "^2.6.4",
|
||||
"redux-logger": "^3.0.6",
|
||||
"redux-persist": "^5.10.0",
|
||||
"redux-promise-middleware": "^5.1.1",
|
||||
"redux-thunk": "^2.3.0",
|
||||
"remote-redux-devtools": "^0.5.13",
|
||||
"reselect": "^4.0.0"
|
||||
}
|
||||
}
|
17
reducers/index.js
Normal file
17
reducers/index.js
Normal file
|
@ -0,0 +1,17 @@
|
|||
|
||||
const { combineReducers } = require('redux');
|
||||
|
||||
const { reducer: pulse, initialState: pulseInitialState } = require('./pulse');
|
||||
|
||||
const initialState = {
|
||||
pulse: pulseInitialState,
|
||||
};
|
||||
|
||||
const reducer = combineReducers({
|
||||
pulse,
|
||||
});
|
||||
|
||||
module.exports = {
|
||||
initialState,
|
||||
reducer,
|
||||
};
|
69
reducers/pulse.js
Normal file
69
reducers/pulse.js
Normal file
|
@ -0,0 +1,69 @@
|
|||
|
||||
const {
|
||||
always,
|
||||
merge,
|
||||
omit,
|
||||
fromPairs,
|
||||
map,
|
||||
} = require('ramda');
|
||||
|
||||
const { combineReducers } = require('redux');
|
||||
|
||||
const { handleActions } = require('redux-actions');
|
||||
|
||||
const { pulse } = require('../actions');
|
||||
|
||||
const { things } = require('../constants/pulse');
|
||||
|
||||
const initialState = {
|
||||
state: 'closed',
|
||||
|
||||
objects: fromPairs(map(({ key }) => [ key, {} ], things)),
|
||||
infos: fromPairs(map(({ key }) => [ key, {} ], things)),
|
||||
};
|
||||
|
||||
const reducer = combineReducers({
|
||||
state: handleActions({
|
||||
[pulse.ready]: always('ready'),
|
||||
[pulse.close]: always('closed'),
|
||||
}, initialState.state),
|
||||
|
||||
objects: combineReducers(fromPairs(map(({ key, type }) => [ key, handleActions({
|
||||
[pulse.new]: (state, { payload }) => {
|
||||
if (payload.type !== type) {
|
||||
return state;
|
||||
}
|
||||
return merge(state, {
|
||||
[payload.index]: payload,
|
||||
});
|
||||
},
|
||||
[pulse.remove]: (state, { payload }) => {
|
||||
if (payload.type !== type) {
|
||||
return state;
|
||||
}
|
||||
return omit([ payload.index ], state);
|
||||
},
|
||||
}, initialState.objects[key]) ], things))),
|
||||
|
||||
infos: combineReducers(fromPairs(map(({ key, type }) => [ key, handleActions({
|
||||
[pulse.remove]: (state, { payload }) => {
|
||||
if (payload.type !== type) {
|
||||
return state;
|
||||
}
|
||||
return omit([ payload.index ], state);
|
||||
},
|
||||
[pulse.info]: (state, { payload }) => {
|
||||
if (payload.type !== type) {
|
||||
return state;
|
||||
}
|
||||
return merge(state, {
|
||||
[payload.index]: payload,
|
||||
});
|
||||
},
|
||||
}, initialState.infos[key]) ], things))),
|
||||
});
|
||||
|
||||
module.exports = {
|
||||
initialState,
|
||||
reducer,
|
||||
};
|
25
renderer.js
Normal file
25
renderer.js
Normal file
|
@ -0,0 +1,25 @@
|
|||
/* global document */
|
||||
|
||||
const r = require('r-dom');
|
||||
|
||||
const { render } = require('react-dom');
|
||||
|
||||
const { Provider } = require('react-redux');
|
||||
|
||||
const createStore = require('./store');
|
||||
|
||||
const Graph = require('./components/graph');
|
||||
|
||||
const theme = require('./utils/theme');
|
||||
|
||||
const Root = () => r(Provider, {
|
||||
store: createStore(),
|
||||
}, [
|
||||
r(Graph),
|
||||
]);
|
||||
|
||||
Object.entries(theme.colors).forEach(([ key, value ]) => {
|
||||
document.body.style.setProperty('--' + key, value);
|
||||
});
|
||||
|
||||
render(r(Root), document.getElementById('root'));
|
16
selectors/index.js
Normal file
16
selectors/index.js
Normal file
|
@ -0,0 +1,16 @@
|
|||
|
||||
const {
|
||||
map,
|
||||
prop,
|
||||
indexBy,
|
||||
} = require('ramda');
|
||||
|
||||
const { things } = require('../constants/pulse');
|
||||
|
||||
const storeKeyByType = map(prop('key'), indexBy(prop('type'), things));
|
||||
|
||||
const getPaiByTypeAndIndex = (type, index) => state => state.pulse.infos[storeKeyByType[type]][index];
|
||||
|
||||
module.exports = {
|
||||
getPaiByTypeAndIndex,
|
||||
};
|
47
store/index.js
Normal file
47
store/index.js
Normal file
|
@ -0,0 +1,47 @@
|
|||
|
||||
const { createStore, applyMiddleware } = require('redux');
|
||||
|
||||
// const { createLogger } = require('redux-logger');
|
||||
const { composeWithDevTools } = require('remote-redux-devtools');
|
||||
|
||||
const { default: thunkMiddleware } = require('redux-thunk');
|
||||
const { default: createPromiseMiddleware } = require('redux-promise-middleware');
|
||||
|
||||
const { persistStore, persistReducer } = require('redux-persist');
|
||||
const { default: storage } = require('redux-persist/lib/storage');
|
||||
|
||||
const { reducer, initialState } = require('../reducers');
|
||||
|
||||
const pulseMiddleware = require('./pulse-middleware');
|
||||
|
||||
const persistConfig = {
|
||||
key: 'redux-persist',
|
||||
whitelist: [ 'localStorage' ],
|
||||
storage,
|
||||
};
|
||||
|
||||
const dev = process.env.NODE_ENV !== 'production';
|
||||
|
||||
module.exports = (state = initialState) => {
|
||||
const middlewares = [
|
||||
thunkMiddleware,
|
||||
createPromiseMiddleware(),
|
||||
pulseMiddleware,
|
||||
// dev && createLogger(),
|
||||
].filter(Boolean);
|
||||
|
||||
const reducer_ = persistReducer(persistConfig, reducer);
|
||||
|
||||
const store = createStore(
|
||||
reducer_,
|
||||
state,
|
||||
composeWithDevTools({
|
||||
realtime: dev,
|
||||
hostname: 'localhost', port: 8000,
|
||||
})(applyMiddleware(...middlewares)),
|
||||
);
|
||||
|
||||
persistStore(store);
|
||||
|
||||
return store;
|
||||
};
|
106
store/pulse-middleware.js
Normal file
106
store/pulse-middleware.js
Normal file
|
@ -0,0 +1,106 @@
|
|||
|
||||
const PAClient = require('paclient');
|
||||
|
||||
const { handleActions } = require('redux-actions');
|
||||
|
||||
const { pulse: pulseActions } = require('../actions');
|
||||
|
||||
const { things } = require('../constants/pulse');
|
||||
|
||||
function getFnFromType(type) {
|
||||
let fn;
|
||||
switch (type) {
|
||||
case 'sink':
|
||||
case 'card':
|
||||
case 'source':
|
||||
fn = type;
|
||||
break;
|
||||
case 'sinkInput':
|
||||
case 'sourceOutput':
|
||||
case 'client':
|
||||
case 'module':
|
||||
fn = `${type}ByIndex`;
|
||||
break;
|
||||
default:
|
||||
throw new Error('Unexpected type: ' + type);
|
||||
}
|
||||
return 'get' + fn[0].toUpperCase() + fn.slice(1);
|
||||
}
|
||||
|
||||
module.exports = store => {
|
||||
const pa = new PAClient();
|
||||
|
||||
const getInfo = (type, index) => pa[getFnFromType(type)](index, (err, info) => {
|
||||
if (err) {
|
||||
if (err.message === 'No such entity') {
|
||||
console.warn(err.message, type, index);
|
||||
return;
|
||||
}
|
||||
throw err;
|
||||
}
|
||||
info.type = info.type || type;
|
||||
store.dispatch(pulseActions.info(info));
|
||||
});
|
||||
|
||||
pa
|
||||
.on('ready', () => {
|
||||
store.dispatch(pulseActions.ready());
|
||||
pa.subscribe('all');
|
||||
things.forEach(({ method, type }) => {
|
||||
pa[method]((err, infos) => {
|
||||
if (err) {
|
||||
throw err;
|
||||
}
|
||||
infos.forEach(info => {
|
||||
const { index } = info;
|
||||
info.type = info.type || type;
|
||||
store.dispatch(pulseActions.new({ type, index }));
|
||||
store.dispatch(pulseActions.info(info));
|
||||
});
|
||||
});
|
||||
});
|
||||
})
|
||||
.on('close', () => {
|
||||
store.dispatch(pulseActions.close());
|
||||
})
|
||||
.on('new', (type, index) => {
|
||||
store.dispatch(pulseActions.new({ type, index }));
|
||||
getInfo(type, index);
|
||||
})
|
||||
.on('change', (type, index) => {
|
||||
store.dispatch(pulseActions.change({ type, index }));
|
||||
getInfo(type, index);
|
||||
})
|
||||
.on('remove', (type, index) => {
|
||||
store.dispatch(pulseActions.remove({ type, index }));
|
||||
});
|
||||
|
||||
pa.connect();
|
||||
|
||||
const handlePulseActions = handleActions({
|
||||
[pulseActions.moveSinkInput]: (state, { payload: { sinkInputIndex, destSinkIndex } }) => {
|
||||
pa.moveSinkInput(sinkInputIndex, destSinkIndex, error => {
|
||||
if (error) {
|
||||
throw error;
|
||||
}
|
||||
});
|
||||
return state;
|
||||
},
|
||||
[pulseActions.moveSourceOutput]: (state, { payload: { sourceOutputIndex, destSourceIndex } }) => {
|
||||
pa.moveSourceOutput(sourceOutputIndex, destSourceIndex, error => {
|
||||
if (error) {
|
||||
throw error;
|
||||
}
|
||||
});
|
||||
return state;
|
||||
},
|
||||
}, null);
|
||||
|
||||
return next => action => {
|
||||
const ret = next(action);
|
||||
|
||||
handlePulseActions(null, action);
|
||||
|
||||
return ret;
|
||||
};
|
||||
};
|
11
utils/theme/index.js
Normal file
11
utils/theme/index.js
Normal file
|
@ -0,0 +1,11 @@
|
|||
|
||||
const { theme } = require('@jakejarrett/gtk-theme');
|
||||
const camelCase = require('camelcase');
|
||||
|
||||
const colors = {};
|
||||
|
||||
theme.css.replace(/@define-color\s+([\w_]+?)\s+(.+?);/g, (_, name, value) => {
|
||||
colors[camelCase(name)] = value;
|
||||
});
|
||||
|
||||
module.exports = { colors };
|
Loading…
Reference in New Issue
Block a user