gojs
Version:
Interactive diagrams, charts, and graphs, such as trees, flowcharts, orgcharts, UML, BPMN, or business diagrams
282 lines (271 loc) • 11.8 kB
text/typescript
'use strict';
/*
* Copyright (C) 1998-2020 by Northwoods Software Corporation. All Rights Reserved.
*/
import * as go from '../release/go.js';
// A custom layout that shows the two families related to a person's parents
export class GenogramLayout extends go.LayeredDigraphLayout {
public spouseSpacing: number;
public constructor() {
super();
this.initializeOption = go.LayeredDigraphLayout.InitDepthFirstIn;
this.spouseSpacing = 30; // minimum space between spouses
}
public makeNetwork(coll: go.Diagram | go.Group | go.Iterable<go.Part>) {
// generate LayoutEdges for each parent-child Link
const net = this.createNetwork();
if (coll instanceof go.Diagram) {
this.add(net, coll.nodes, true);
this.add(net, coll.links, true);
} else if (coll instanceof go.Group) {
this.add(net, coll.memberParts, false);
} else if (coll.iterator) {
this.add(net, coll.iterator, false);
}
return net;
}
// internal method for creating LayeredDigraphNetwork where husband/wife pairs are represented
// by a single LayeredDigraphVertex corresponding to the label Node on the marriage Link
protected add(net: go.LayeredDigraphNetwork, coll: go.Iterable<go.Part>, nonmemberonly: boolean) {
const multiSpousePeople = new go.Set() as go.Set<go.Node>;
// consider all Nodes in the given collection
const it = coll.iterator;
while (it.next()) {
const node = it.value as go.Node;
if (!(node instanceof go.Node)) continue;
if (!node.isLayoutPositioned || !node.isVisible()) continue;
if (nonmemberonly && node.containingGroup !== null) continue;
// if it's an unmarried Node, or if it's a Link Label Node, create a LayoutVertex for it
if (node.isLinkLabel) {
// get marriage Link
const link = node.labeledLink;
if (link) {
const spouseA = link.fromNode;
const spouseB = link.toNode;
// create vertex representing both husband and wife
const vertex = net.addNode(node);
// now define the vertex size to be big enough to hold both spouses
if (spouseA && spouseB) {
vertex.width = spouseA.actualBounds.width + this.spouseSpacing + spouseB.actualBounds.width;
vertex.height = Math.max(spouseA.actualBounds.height, spouseB.actualBounds.height);
vertex.focus = new go.Point(spouseA.actualBounds.width + this.spouseSpacing / 2, vertex.height / 2);
}
}
} else {
// don't add a vertex for any married person!
// instead, code above adds label node for marriage link
// assume a marriage Link has a label Node
let marriages = 0;
node.linksConnected.each(function(l) { if (l.isLabeledLink) marriages++; });
if (marriages === 0) {
const vertex = net.addNode(node);
} else if (marriages > 1) {
multiSpousePeople.add(node);
}
}
}
// now do all Links
it.reset();
while (it.next()) {
const link = it.value as go.Link;
if (!(link instanceof go.Link)) continue;
if (!link.isLayoutPositioned || !link.isVisible()) continue;
if (nonmemberonly && link.containingGroup !== null) continue;
// if it's a parent-child link, add a LayoutEdge for it
if (!link.isLabeledLink) {
const fromNode = link.fromNode;
const toNode = link.toNode;
if (fromNode !== null && toNode !== null) {
const parent = net.findVertex(fromNode); // should be a label node
const child = net.findVertex(toNode);
if (parent !== null && child !== null) { // an unmarried child
net.linkVertexes(parent, child, link);
} else if (parent !== null) { // a married child
toNode.linksConnected.each(function(l) {
if (!l.isLabeledLink) return; // if it has no label node, it's a parent-child link
// found the Marriage Link, now get its label Node
const mlab = l.labelNodes.first();
// parent-child link should connect with the label node,
// so the LayoutEdge should connect with the LayoutVertex representing the label node
if (mlab !== null) {
const mlabvert = net.findVertex(mlab);
if (mlabvert !== null) {
net.linkVertexes(parent, mlabvert, link);
}
}
});
}
}
}
}
while (multiSpousePeople.count > 0) {
// find all collections of people that are indirectly married to each other
const node = multiSpousePeople.first() as go.Node;
const cohort = new go.Set() as go.Set<go.Node>;
this.extendCohort(cohort, node);
// then encourage them all to be the same generation by connecting them all with a common vertex
const dummyvert = net.createVertex();
net.addVertex(dummyvert);
const marriages = new go.Set() as go.Set<go.Link>;
cohort.each(function(n) {
n.linksConnected.each(function(l) {
marriages.add(l);
});
});
marriages.each(function(link) {
// find the vertex for the marriage link (i.e. for the label node)
const mlab = link.labelNodes.first();
if (mlab !== null) {
const v = net.findVertex(mlab);
if (v !== null) {
net.linkVertexes(dummyvert, v, null);
}
}
});
// done with these people, now see if there are any other multiple-married people
multiSpousePeople.removeAll(cohort);
}
}
// collect all of the people indirectly married with a person
protected extendCohort(coll: go.Set<go.Node>, node: go.Node) {
if (coll.contains(node)) return;
coll.add(node);
const lay = this;
node.linksConnected.each(function(l) {
if (l.isLabeledLink) { // if it's a marriage link, continue with both spouses
if (l.fromNode !== null) lay.extendCohort(coll, l.fromNode);
if (l.toNode !== null) lay.extendCohort(coll, l.toNode);
}
});
}
public assignLayers() {
super.assignLayers();
const horiz = this.direction === 0.0 || this.direction === 180.0;
// for every vertex, record the maximum vertex width or height for the vertex's layer
const maxsizes = [] as Array<number>;
const net = this.network;
if (net !== null) {
const vit = net.vertexes.iterator;
while (vit.next()) {
const v = vit.value as go.LayeredDigraphVertex;
const lay = v.layer;
let max = maxsizes[lay];
if (max === undefined) max = 0;
const sz = (horiz ? v.width : v.height);
if (sz > max) maxsizes[lay] = sz;
}
vit.reset();
// now make sure every vertex has the maximum width or height according to which layer it is in,
// and aligned on the left (if horizontal) or the top (if vertical)
while (vit.next()) {
const v = vit.value as go.LayeredDigraphVertex;
const lay = v.layer;
const max = maxsizes[lay];
if (horiz) {
v.focus = new go.Point(0, v.height / 2);
v.width = max;
} else {
v.focus = new go.Point(v.width / 2, 0);
v.height = max;
}
}
// from now on, the LayeredDigraphLayout will think that the Node is bigger than it really is
// (other than the ones that are the widest or tallest in their respective layer).
}
}
public commitNodes() {
super.commitNodes();
const net = this.network;
// position regular nodes
if (net !== null) {
const vit = net.vertexes.iterator;
while (vit.next()) {
const v = vit.value as go.LayeredDigraphVertex;
if (v.node !== null && !v.node.isLinkLabel) {
v.node.position = new go.Point(v.x, v.y);
}
}
vit.reset();
// position the spouses of each marriage vertex
const layout = this;
while (vit.next()) {
const v = vit.value as go.LayeredDigraphVertex;
if (v.node === null) continue;
if (!v.node.isLinkLabel) continue;
const labnode = v.node;
const lablink = labnode.labeledLink;
if (lablink !== null) {
// In case the spouses are not actually moved, we need to have the marriage link
// position the label node, because LayoutVertex.commit() was called above on these vertexes.
// Alternatively we could override LayoutVetex.commit to be a no-op for label node vertexes.
lablink.invalidateRoute();
let spouseA = lablink.fromNode;
let spouseB = lablink.toNode;
if (spouseA !== null && spouseB != null) {
// prefer fathers on the left, mothers on the right
if (spouseA.data.s === 'F') { // sex is female
const temp = spouseA;
spouseA = spouseB;
spouseB = temp;
}
// see if the parents are on the desired sides, to avoid a link crossing
const aParentsNode = layout.findParentsMarriageLabelNode(spouseA);
const bParentsNode = layout.findParentsMarriageLabelNode(spouseB);
if (aParentsNode !== null && bParentsNode !== null && aParentsNode.position.x > bParentsNode.position.x) {
// swap the spouses
const temp = spouseA;
spouseA = spouseB;
spouseB = temp;
}
spouseA.position = new go.Point(v.x, v.y);
spouseB.position = new go.Point(v.x + spouseA.actualBounds.width + layout.spouseSpacing, v.y);
if (spouseA.opacity === 0) {
const pos = new go.Point(v.centerX - spouseA.actualBounds.width / 2, v.y);
spouseA.position = pos;
spouseB.position = pos;
} else if (spouseB.opacity === 0) {
const pos = new go.Point(v.centerX - spouseB.actualBounds.width / 2, v.y);
spouseA.position = pos;
spouseB.position = pos;
}
}
}
}
vit.reset();
// position only-child nodes to be under the marriage label node
while (vit.next()) {
const v = vit.value as go.LayeredDigraphVertex;
if (v.node === null || v.node.linksConnected.count > 1) continue;
const mnode = layout.findParentsMarriageLabelNode(v.node);
if (mnode !== null && mnode.linksConnected.count === 1) { // if only one child
if (layout.network === null) continue;
const mvert = layout.network.findVertex(mnode);
if (mvert !== null) {
const newbnds = v.node.actualBounds.copy();
newbnds.x = mvert.centerX - v.node.actualBounds.width / 2;
// see if there's any empty space at the horizontal mid-point in that layer
if (layout.diagram !== null) {
const overlaps = layout.diagram.findObjectsIn(newbnds,
(x) => { const p = x.part; return (p instanceof go.Part) ? p : null; },
(p) => p !== v.node,
true
);
if (overlaps.count === 0) {
v.node.move(newbnds.position);
}
}
}
}
}
}
}
public findParentsMarriageLabelNode(node: go.Node) {
const it = node.findNodesInto();
while (it.next()) {
const n = it.value;
if (n.isLinkLabel) return n;
}
return null;
}
}
// end GenogramLayout class