import { Vertex } from './vertex.js'; import { Edge } from './edge.js'; import { Document } from '../display/document.js'; const allGraphs = []; const makeWDGHandler = (graphIndex) => (vertexId) => { const graph = allGraphs[graphIndex]; // We want a document for editing this node, which may be a vertex or an edge const { editorDoc } = graph; editorDoc.clear(); if (vertexId.startsWith('edge:')) { const [, from, to] = vertexId.split(':'); Edge.prepareEditorDocument(graph, editorDoc, from, to); } else { Vertex.prepareEditorDocument(graph, editorDoc, vertexId); } }; export class WeightedDirectedGraph { constructor(scene, options = {}) { this.scene = scene; this.vertices = new Map(); this.edgeTypes = new Map(); this.nextVertexId = 0; this.flowchart = scene?.flowchart; this.editable = options.editable; // Mermaid supports a click callback, but we can't customize arguments; we just get the vertex ID. // In order to provide the appropriate graph context for each callback, we create a separate callback // function for each graph. this.index = allGraphs.length; allGraphs.push(this); window[`WDGHandler${this.index}`] = makeWDGHandler(this.index); // TODO: Populate history this.history = {}; } getHistory() { // record operations that modify the graph return this.history; } toJSON() { return { vertices: Array.from(this.vertices.values()), edgeTypes: Array.from(this.edgeTypes.keys()), edges: Array.from(this.edgeTypes.values()).flatMap((edges) => Array.from(edges.values())), history: this.getHistory(), }; } redraw() { // Call .reset() on all vertices and edges for (const vertex of this.vertices.values()) { vertex.reset(); } for (const edges of this.edgeTypes.values()) { for (const edge of edges.values()) { edge.reset(); } } // Clear the target div this.flowchart?.reset(); this.flowchart?.init(); // Draw all vertices and edges for (const vertex of this.vertices.values()) { vertex.displayVertex(); } // Let's flatmap and dedupe by [from, to] since each edge // renders all comorphic edges as well. const edgesFrom = new Map(); // edgeMap[from][to] = edge for (const edges of this.edgeTypes.values()) { for (const edge of edges.values()) { const edgesTo = edgesFrom.get(edge.from) || new Map(); edgesTo.set(edge.to, edge); edgesFrom.set(edge.from, edgesTo); } } for (const edgesTo of edgesFrom.values()) { for (const edge of edgesTo.values()) { edge.displayEdge(); } } // Ensure rerender this.flowchart?.render(); } withFlowchart() { this.scene?.withSectionFlowchart(); this.flowchart = this.scene?.lastFlowchart; if (this.editable) { this.editorDoc = new Document('WDGControls', this.flowchart.box.el); this.resetEditorDocument(); } this.errorDoc = new Document('WDGErrors', this.flowchart.box.el); return this; } resetEditorDocument() { this.editorDoc.clear(); Vertex.prepareEditorDocument(this, this.editorDoc); } addVertex(type, id, data, label, options) { // Supports simple case of auto-incremented numeric ids if (typeof id === 'object') { data = id; id = this.nextVertexId++; } id = (typeof id === 'number') ? id.toString(10) : id; if (this.vertices.has(id)) { throw new Error(`Vertex already exists with id: ${id}`); } const vertex = new Vertex(this, type, id, data, { ...options, label }); this.vertices.set(id, vertex); vertex.displayVertex(); return vertex; } getVertex(id) { id = (typeof id === 'number') ? id.toString(10) : id; return this.vertices.get(id); } getVertexData(id) { return this.getVertex(id)?.data; } getVerticesData() { return Array.from(this.vertices.values()).map(({ data }) => data); } getEdge(type, from, to) { from = from instanceof Vertex ? from : this.getVertex(from); to = to instanceof Vertex ? to : this.getVertex(to); if (!from || !to) { return undefined; } const edges = this.edgeTypes.get(type); const edgeKey = Edge.getKey({ from, to, type }); return edges?.get(edgeKey); } getEdgeWeight(type, from, to) { return this.getEdge(type, from, to)?.weight; } setEdgeWeight(type, from, to, weight, data, options) { from = from instanceof Vertex ? from : this.getVertex(from); to = to instanceof Vertex ? to : this.getVertex(to); const edge = new Edge(this, type, from, to, weight, data, options); let edges = this.edgeTypes.get(type); if (!edges) { edges = new Map(); this.edgeTypes.set(type, edges); } const edgeKey = Edge.getKey(edge); edges.set(edgeKey, edge); edge.displayEdgeNode(); return edge; } addEdge(type, from, to, weight, data, options) { from = from instanceof Vertex ? from : this.getVertex(from); to = to instanceof Vertex ? to : this.getVertex(to); const existingEdges = this.getEdges(type, from, to); if (this.getEdge(type, from, to)) { throw new Error(`Edge ${type} from ${from.id} to ${to.id} already exists`); } const edge = this.setEdgeWeight(type, from, to, weight, data, options); from.edges.from.push(edge); to.edges.to.push(edge); if (existingEdges.length) { edge.displayEdgeNode(); } else { edge.displayEdge(); } return edge; } getEdges(type, from, to) { from = from instanceof Vertex ? from : this.getVertex(from); to = to instanceof Vertex ? to : this.getVertex(to); const edgeTypes = type ? [type] : Array.from(this.edgeTypes.keys()); return edgeTypes.flatMap((edgeType) => { const edges = this.edgeTypes.get(edgeType); return Array.from(edges?.values() || []).filter((edge) => { const matchFrom = from === null || from === undefined || from === edge.from; const matchTo = to === null || to === undefined || to === edge.to; return matchFrom && matchTo; }); }); } countVertices(type) { if (!type) { return this.vertices.size; } return Array.from(this.vertices.values()).filter((vertex) => vertex.type === type).length; } deleteEdge(type, from, to) { from = from instanceof Vertex ? from : this.getVertex(from); to = to instanceof Vertex ? to : this.getVertex(to); const edges = this.edgeTypes.get(type); const edgeKey = Edge.getKey({ type, from, to }); if (!edges) return; const edge = edges.get(edgeKey); if (!edge) return; to.edges.from.forEach((x, i) => (x === edge) && to.edges.from.splice(i, 1)); from.edges.to.forEach((x, i) => (x === edge) && from.edges.to.splice(i, 1)); edges.delete(edgeKey); if (edges.size === 0) { this.edgeTypes.delete(type); } } deleteVertex(id) { const vertex = this.getVertex(id); for (const { type, from, to } of [...vertex.edges.to, ...vertex.edges.from]) { this.deleteEdge(type, from, to); } this.vertices.delete(id); } }