gojs
Version:
Interactive diagrams, charts, and graphs, such as trees, flowcharts, orgcharts, UML, BPMN, or business diagrams
345 lines (304 loc) • 12.7 kB
text/typescript
/*
* Copyright (C) 1998-2020 by Northwoods Software Corporation. All Rights Reserved.
*/
/*
* This is an extension and not part of the main GoJS library.
* Note that the API for this class may change with any version, even point releases.
* If you intend to use an extension in production, you should copy the code to your own source directory.
* Extensions can be found in the GoJS kit under the extensions or extensionsTS folders.
* See the Extensions intro page (https://gojs.net/latest/intro/extensions.html) for more information.
*/
import * as go from '../release/go.js';
import './Figures.js';
import { PortShiftingTool } from './PortShiftingTool.js';
const red: string = 'orangered'; // 0 or false
const green: string = 'forestgreen'; // 1 or true
let myDiagram: go.Diagram;
export function init() {
if ((window as any).goSamples) (window as any).goSamples(); // init for these samples -- you don't need to call this
const $ = go.GraphObject.make; // for conciseness in defining templates
myDiagram =
$(go.Diagram, 'myDiagramDiv', // create a new Diagram in the HTML DIV element "myDiagramDiv"
{
'draggingTool.isGridSnapEnabled': true, // dragged nodes will snap to a grid of 10x10 cells
'undoManager.isEnabled': true
});
// install the PortShiftingTool as a "mouse move" tool
myDiagram.toolManager.mouseMoveTools.insertAt(0, new PortShiftingTool());
// when the document is modified, add a "*" to the title and enable the "Save" button
myDiagram.addDiagramListener('Modified', function(e) {
const button = (document.getElementById('saveModel') as any);
if (button) button.disabled = !myDiagram.isModified;
const idx = document.title.indexOf('*');
if (myDiagram.isModified) {
if (idx < 0) document.title += '*';
} else {
if (idx >= 0) document.title = document.title.substr(0, idx);
}
});
const palette = new go.Palette('palette'); // create a new Palette in the HTML DIV element "palette"
// creates relinkable Links that will avoid crossing Nodes when possible and will jump over other Links in their paths
myDiagram.linkTemplate =
$(go.Link,
{
routing: go.Link.AvoidsNodes,
curve: go.Link.JumpOver,
corner: 3,
relinkableFrom: true, relinkableTo: true,
selectionAdorned: false, // Links are not adorned when selected so that their color remains visible.
shadowOffset: new go.Point(0, 0), shadowBlur: 5, shadowColor: 'blue'
},
new go.Binding('isShadowed', 'isSelected').ofObject(),
$(go.Shape,
{ name: 'SHAPE', strokeWidth: 2, stroke: red }));
// node template helpers
const sharedToolTip =
$<go.Adornment>('ToolTip',
{ 'Border.figure': 'RoundedRectangle' },
$(go.TextBlock, { margin: 2 },
new go.Binding('text', '', function(d) { return d.category; })));
// define some common property settings
function nodeStyle() {
return [new go.Binding('location', 'loc', go.Point.parse).makeTwoWay(go.Point.stringify),
new go.Binding('isShadowed', 'isSelected').ofObject(),
{
selectionAdorned: false,
shadowOffset: new go.Point(0, 0),
shadowBlur: 15,
shadowColor: 'blue',
resizable: true,
resizeObjectName: 'NODESHAPE',
toolTip: sharedToolTip
}];
}
function shapeStyle() {
return {
name: 'NODESHAPE',
fill: 'lightgray',
stroke: 'darkslategray',
desiredSize: new go.Size(40, 40),
strokeWidth: 2
};
}
function portStyle(input: boolean) {
return {
desiredSize: new go.Size(6, 6),
fill: 'black',
fromSpot: go.Spot.Right,
fromLinkable: !input,
toSpot: go.Spot.Left,
toLinkable: input,
toMaxLinks: 1,
cursor: 'pointer'
};
}
// define templates for each type of node
const inputTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'Circle', shapeStyle(),
{ fill: red }), // override the default fill (from shapeStyle()) to be red
$(go.Shape, 'Rectangle', portStyle(false), // the only port
{ portId: '', alignment: new go.Spot(1, 0.5) }),
{ // if double-clicked, an input node will change its value, represented by the color.
doubleClick: function(e: go.InputEvent, obj: go.GraphObject) {
if (!(obj instanceof go.Node)) return;
e.diagram.startTransaction('Toggle Input');
const shp = obj.findObject('NODESHAPE') as go.Shape;
shp.fill = (shp.fill === green) ? red : green;
updateStates();
e.diagram.commitTransaction('Toggle Input');
}
}
);
const outputTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'Rectangle', shapeStyle(),
{ fill: green }), // override the default fill (from shapeStyle()) to be green
$(go.Shape, 'Rectangle', portStyle(true), // the only port
{ portId: '', alignment: new go.Spot(0, 0.5) })
);
const andTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'AndGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const orTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'OrGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0.16, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0.16, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const xorTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'XorGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0.26, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0.26, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const norTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'NorGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0.16, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0.16, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const xnorTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'XnorGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0.26, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0.26, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const nandTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'NandGate', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in1', alignment: new go.Spot(0, 0.3) }),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in2', alignment: new go.Spot(0, 0.7) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
const notTemplate =
$(go.Node, 'Spot', nodeStyle(),
$(go.Shape, 'Inverter', shapeStyle()),
$(go.Shape, 'Rectangle', portStyle(true),
{ portId: 'in', alignment: new go.Spot(0, 0.5) }),
$(go.Shape, 'Rectangle', portStyle(false),
{ portId: 'out', alignment: new go.Spot(1, 0.5) })
);
// add the templates created above to myDiagram and palette
myDiagram.nodeTemplateMap.add('input', inputTemplate);
myDiagram.nodeTemplateMap.add('output', outputTemplate);
myDiagram.nodeTemplateMap.add('and', andTemplate);
myDiagram.nodeTemplateMap.add('or', orTemplate);
myDiagram.nodeTemplateMap.add('xor', xorTemplate);
myDiagram.nodeTemplateMap.add('not', notTemplate);
myDiagram.nodeTemplateMap.add('nand', nandTemplate);
myDiagram.nodeTemplateMap.add('nor', norTemplate);
myDiagram.nodeTemplateMap.add('xnor', xnorTemplate);
// share the template map with the Palette
palette.nodeTemplateMap = myDiagram.nodeTemplateMap;
palette.model.nodeDataArray = [
{ category: 'input' },
{ category: 'output' },
{ category: 'and' },
{ category: 'or' },
{ category: 'xor' },
{ category: 'not' },
{ category: 'nand' },
{ category: 'nor' },
{ category: 'xnor' }
];
// load the initial diagram
load();
// continually update the diagram
loop();
}
// update the diagram every 250 milliseconds
function loop() {
setTimeout(function() { updateStates(); loop(); }, 250);
}
// update the value and appearance of each node according to its type and input values
function updateStates() {
const oldskip = myDiagram.skipsUndoManager;
myDiagram.skipsUndoManager = true;
// do all "input" nodes first
myDiagram.nodes.each(function(node) {
if (node.category === 'input') {
doInput(node);
}
});
// now we can do all other kinds of nodes
myDiagram.nodes.each(function(node) {
switch (node.category) {
case 'and': doAnd(node); break;
case 'or': doOr(node); break;
case 'xor': doXor(node); break;
case 'not': doNot(node); break;
case 'nand': doNand(node); break;
case 'nor': doNor(node); break;
case 'xnor': doXnor(node); break;
case 'output': doOutput(node); break;
case 'input': break; // doInput already called, above
}
});
myDiagram.skipsUndoManager = oldskip;
}
// helper predicate
function linkIsTrue(link: go.Link) { // assume the given Link has a Shape named "SHAPE"
return (link.findObject('SHAPE') as go.Shape).stroke === green;
}
// helper function for propagating results
function setOutputLinks(node: go.Node, color: string) {
node.findLinksOutOf().each(function(link) { (link.findObject('SHAPE') as go.Shape).stroke = color; });
}
// update nodes by the specific function for its type
// determine the color of links coming out of this node based on those coming in and node type
function doInput(node: go.Node) {
// the output is just the node's Shape.fill
setOutputLinks(node, (node.findObject('NODESHAPE') as go.Shape).fill as string);
}
function doAnd(node: go.Node) {
const color = node.findLinksInto().all(linkIsTrue) ? green : red;
setOutputLinks(node, color);
}
function doNand(node: go.Node) {
const color = !node.findLinksInto().all(linkIsTrue) ? green : red;
setOutputLinks(node, color);
}
function doNot(node: go.Node) {
const color = !node.findLinksInto().all(linkIsTrue) ? green : red;
setOutputLinks(node, color);
}
function doOr(node: go.Node) {
const color = node.findLinksInto().any(linkIsTrue) ? green : red;
setOutputLinks(node, color);
}
function doNor(node: go.Node) {
const color = !node.findLinksInto().any(linkIsTrue) ? green : red;
setOutputLinks(node, color);
}
function doXor(node: go.Node) {
let truecount = 0;
node.findLinksInto().each(function(link) { if (linkIsTrue(link)) truecount++; });
const color = truecount % 2 !== 0 ? green : red;
setOutputLinks(node, color);
}
function doXnor(node: go.Node) {
let truecount = 0;
node.findLinksInto().each(function(link) { if (linkIsTrue(link)) truecount++; });
const color = truecount % 2 === 0 ? green : red;
setOutputLinks(node, color);
}
function doOutput(node: go.Node) {
// assume there is just one input link
// we just need to update the node's Shape.fill
node.linksConnected.each(function(link) { (node.findObject('NODESHAPE') as go.Shape).fill = (link.findObject('SHAPE') as go.Shape).stroke; });
}
// save a model to and load a model from Json text, displayed below the Diagram
export function save() {
(document.getElementById('mySavedModel') as any).value = myDiagram.model.toJson();
myDiagram.isModified = false;
}
export function load() {
myDiagram.model = go.Model.fromJson((document.getElementById('mySavedModel') as any).value);
}