react-sigma-conglei
Version:
Lightweight but powerful library for drawing network graphs built on top of dunnock/react-sigma
200 lines (185 loc) • 6.7 kB
JavaScript
import React from 'react';
import { embedProps } from './tools';
import '../sigma/main';
/**
*
* Sigma - React.JS flow-typed interface for Sigma js library - fastest opensource rendering engine for network graphs.
* Sigma makes it easy to publish networks on Web pages, and allows developers to integrate network exploration in
* rich Web applications.
*
* Parameter types
* ```
* type Sigma$Graph$Data = {
* nodes: [Sigma$Node],
* edges: [Sigma$Edge]
* };
*
* type Sigma$Node = {
* id: string,
* label?: string,
* x?: number,
* y?: number,
* size?: number,
* color?: color
* };
*
* type Sigma$Edge = {
* id: string,
* source: string,
* target: string,
* label?: string,
* color?: color
* };
* ```
*
*
* @signature `<Sigma graph={graph} settings={settings} onClickNode={func}.../>`
*
* @param {CSS} style CSS style description for main div holding graph, should be specified in React format
* @param {Sigma$Settings} settings js object with sigma initialization options, for full list see [sigma settings page](https://github.com/jacomyal/sigma.js/wiki/Settings)
* @param {string} renderer can be "webgl" or "canvas"
* @param {Sigma$Graph$Data} graph js object with array of nodes and edges used to initialize sigma
* @param {Sigma$ErrorHandler} onSigmaException set sigma callback for sigma exceptions / errors
* @param {Sigma$EventHandler} onClickNode set sigma callback for "clickNode" event (see below)
* @param {Sigma$EventHandler} onOverNode set sigma callback for "overNode" event
* @param {Sigma$EventHandler} onOutNode set sigma callback for "outNode" event
* @param {Sigma$EventHandler} onClickEdge set sigma callback for "clickEdge" event
* @param {Sigma$EventHandler} onOverEdge set sigma callback for "overEdge" event
* @param {Sigma$EventHandler} onOutEdge set sigma callback for "outEdge" event
*
* @example
* Can be composed with sigma sub-components using JSX syntax
* <Sigma renderer="webgl" style={{maxWidth:"inherit", height:"400px"}}
* settings={{drawEdges:false}}
* onOverNode={e => console.log("Mouse over node: " + e.data.node.label)}>
* graph={{nodes:["id0", "id1"], edges:[{id:"e0",source:"id0",target:"id1"}]}}>
* <RelativeSize initialSize={8}/>
* </Sigma>
*
*/
class Sigma extends React.PureComponent {
constructor(props) {
super(props);
_initialiseProps.call(this);
this.state = { renderer: false };
let settings = props.settings ? props.settings : {};
this.initRenderer = this.initRenderer.bind(this);
this.createSigmaInstance(props.graph, settings, props);
}
componentWillReceiveProps(nextProps) {
if (nextProps.nodeColorMapping !== this.props.nodeColorMapping) {
const nodes = this.sigma.graph.nodes();
nodes.forEach(node => {
node.color = node[nextProps.nodeColorMapping];
});
this.sigma.refresh();
}
if (nextProps.graph !== this.props.graph) {
let settings = nextProps.settings ? nextProps.settings : {};
this.createSigmaInstance(nextProps.graph, settings, nextProps);
this.setState({ renderer: false });
this.initRenderer(this.container);
// this.sigma.ff = 'ddd';
}
}
initRenderer(container) {
this.container = container;
if (container) {
let options = { container };
if (this.props.renderer) options.type = this.props.renderer;
this.sigmaRenderer = this.sigma.addRenderer(options);
this.sigma.refresh();
this.setState({ renderer: true });
} else if (this.sigmaRenderer) {
this.sigma.killRenderer(this.sigmaRenderer);
this.sigmaRenderer = null;
this.setState({ renderer: false });
}
}
componentWillUnmount() {
this.sigma.kill();
this.sigmaRenderer = null;
}
render() {
let children = this.state.renderer ? embedProps(this.props.children, { sigma: this.sigma }) : null;
return React.createElement(
'div',
{ ref: this.initRenderer, style: this.props.style },
children
);
}
/**
Initialize event handlers with sigma.
Event handler function receives [Sigma Event](https://github.com/jacomyal/sigma.js/wiki/Events-API)
with the structure of following type:
```
type Sigma$Event = {
data: {
node?: Neo4j$Node, //for node events is sigma node data
edge?: Neo4j$Edge, //for edge events is sigma edge data
captor: { // information about event handler, for instance position on the page {clientX, clientY}
clientX: number,
clientY: number
}}}
type Sigma$EventHandler = (node:Sigma$Event) => void
```
**/
static bindHandlers(handlers, sigma) {
['clickNode', 'overNode', 'outNode', 'clickEdge', 'overEdge', 'outEdge', 'clickStage'].forEach(event => {
let handler = 'on' + event[0].toUpperCase() + event.substr(1);
if (handlers[handler]) {
sigma.bind(event, handlers[handler]);
}
});
}
}
Sigma.defaultProps = {
settings: {
defaultNodeColor: '#3388AA',
defaultLabelSize: 8,
defaultLabelColor: '#777',
labelThreshold: 12,
hoverFontStyle: 'text-size: 11',
batchEdgesDrawing: true,
drawEdges: true,
drawEdgeLabels: false
},
style: {
maxWidth: 'inherit',
height: '400px'
}
};
var _initialiseProps = function () {
this.createSigmaInstance = (graph, settings, props) => {
if (this.sigma) {
this.sigma.kill();
this.sigmaRenderer = null;
}
this.sigma = new sigma({ settings });
Sigma.bindHandlers(props, this.sigma);
if (graph) {
try {
this.sigma.graph.read(graph);
} catch (e) {
if (this.props.onSigmaException) this.props.onSigmaException(e);
}
}
};
};
Sigma.propTypes = {
settings: typeof Sigma$Settings === 'function' ? require('prop-types').instanceOf(Sigma$Settings).isRequired : require('prop-types').any.isRequired,
renderer: require('prop-types').oneOf(['webgl', 'canvas', 'svg']),
style: require('prop-types').object,
children: require('prop-types').any,
graph: typeof Sigma$Graph$Data === 'function' ? require('prop-types').instanceOf(Sigma$Graph$Data) : require('prop-types').any,
onSigmaException: require('prop-types').func,
onClickNode: require('prop-types').func,
onClickEdge: require('prop-types').func,
onOverNode: require('prop-types').func,
onOutNode: require('prop-types').func,
onOverEdge: require('prop-types').func,
onOutEdge: require('prop-types').func,
onClickStage: require('prop-types').func,
nodeColorMapping: typeof String === 'function' ? require('prop-types').instanceOf(String) : require('prop-types').any
};
export default Sigma;