UNPKG

gojs

Version:

Interactive diagrams, charts, and graphs, such as trees, flowcharts, orgcharts, UML, BPMN, or business diagrams

434 lines (414 loc) 21.2 kB
<!DOCTYPE html> <html lang="en"> <head> <meta charset="utf-8"/> <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no, viewport-fit=cover"/> <meta name="description" content="A Sankey diagram with the thickness of links indicating the flow quantity."/> <link rel="stylesheet" href="../assets/css/style.css"/> <!-- Copyright 1998-2023 by Northwoods Software Corporation. --> <title>Sankey Diagram</title> </head> <body> <!-- This top nav is not part of the sample code --> <nav id="navTop" class="w-full z-30 top-0 text-white bg-nwoods-primary"> <div class="w-full container max-w-screen-lg mx-auto flex flex-wrap sm:flex-nowrap items-center justify-between mt-0 py-2"> <div class="md:pl-4"> <a class="text-white hover:text-white no-underline hover:no-underline font-bold text-2xl lg:text-4xl rounded-lg hover:bg-nwoods-secondary " href="../"> <h1 class="my-0 p-1 ">GoJS</h1> </a> </div> <button id="topnavButton" class="rounded-lg sm:hidden focus:outline-none focus:ring" aria-label="Navigation"> <svg fill="currentColor" viewBox="0 0 20 20" class="w-6 h-6"> <path id="topnavOpen" fill-rule="evenodd" d="M3 5a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM3 10a1 1 0 011-1h12a1 1 0 110 2H4a1 1 0 01-1-1zM9 15a1 1 0 011-1h6a1 1 0 110 2h-6a1 1 0 01-1-1z" clip-rule="evenodd"></path> <path id="topnavClosed" class="hidden" fill-rule="evenodd" d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" clip-rule="evenodd"></path> </svg> </button> <div id="topnavList" class="hidden sm:block items-center w-auto mt-0 text-white p-0 z-20"> <ul class="list-reset list-none font-semibold flex justify-end flex-wrap sm:flex-nowrap items-center px-0 pb-0"> <li class="p-1 sm:p-0"><a class="topnav-link" href="../learn/">Learn</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="../samples/">Samples</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="../intro/">Intro</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="../api/">API</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="https://www.nwoods.com/products/register.html">Register</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="../download.html">Download</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="https://forum.nwoods.com/c/gojs/11">Forum</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="https://www.nwoods.com/contact.html" target="_blank" rel="noopener" onclick="getOutboundLink('https://www.nwoods.com/contact.html', 'contact');">Contact</a></li> <li class="p-1 sm:p-0"><a class="topnav-link" href="https://www.nwoods.com/sales/index.html" target="_blank" rel="noopener" onclick="getOutboundLink('https://www.nwoods.com/sales/index.html', 'buy');">Buy</a></li> </ul> </div> </div> <hr class="border-b border-gray-600 opacity-50 my-0 py-0" /> </nav> <div class="md:flex flex-col md:flex-row md:min-h-screen w-full max-w-screen-xl mx-auto"> <div id="navSide" class="flex flex-col w-full md:w-48 text-gray-700 bg-white flex-shrink-0"></div> <!-- * * * * * * * * * * * * * --> <!-- Start of GoJS sample code --> <script src="../release/go.js"></script> <div id="allSampleContent" class="p-4 w-full"> <script id="code"> class SankeyLayout extends go.LayeredDigraphLayout { constructor() { super(); this.alignOption = go.LayeredDigraphLayout.AlignAll; } // determine the desired height of each node/vertex, // based on the thicknesses of the connected links; // actually modify the height of each node's SHAPE makeNetwork(coll) { var net = super.makeNetwork(coll); this.diagram.nodes.each(node => { // figure out how tall the node's bar should be var height = this.getAutoHeightForNode(node); var shape = node.findObject("SHAPE"); if (shape) shape.height = height; var text = node.findObject("TEXT"); var ltext = node.findObject("LTEXT"); var font = "bold " + Math.max(12, Math.round(height / 8)) + "pt Segoe UI, sans-serif" if (text) text.font = font; if (ltext) ltext.font = font; // and update the vertex's dimensions accordingly var v = net.findVertex(node); if (v !== null) { node.ensureBounds(); var r = node.actualBounds; v.width = r.width; v.height = r.height; v.focusY = v.height/2; } }); return net; } getAutoHeightForNode(node) { var heightIn = 0; var it = node.findLinksInto() while (it.next()) { var link = it.value; heightIn += link.computeThickness(); } var heightOut = 0; var it = node.findLinksOutOf() while (it.next()) { var link = it.value; heightOut += link.computeThickness(); } var h = Math.max(heightIn, heightOut); if (h < 10) h = 10; return h; } // treat dummy vertexes as having the thickness of the link that they are in nodeMinColumnSpace(v, topleft) { if (v.node === null) { if (v.edgesCount >= 1) { var max = 1; var it = v.edges; while (it.next()) { var edge = it.value; if (edge.link != null) { var t = edge.link.computeThickness(); if (t > max) max = t; break; } } return Math.max(2, Math.ceil(max / this.columnSpacing)); } return 2; } return super.nodeMinColumnSpace(v, topleft); } // treat dummy vertexes as being thicker, so that the Bezier curves are gentler nodeMinLayerSpace(v, topleft) { if (v.node === null) return 100; return super.nodeMinLayerSpace(v, topleft); } assignLayers() { super.assignLayers(); var maxlayer = this.maxLayer; // now make sure every vertex with no outputs is maxlayer for (var it = this.network.vertexes.iterator; it.next();) { var v = it.value; var node = v.node; if (v.destinationVertexes.count == 0) { v.layer = 0; } if (v.sourceVertexes.count == 0) { v.layer = maxlayer; } } // 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). } commitLayout() { super.commitLayout(); for (var it = this.network.edges.iterator; it.next();) { var link = it.value.link; if (link && link.curve === go.Link.Bezier) { // depend on Link.adjusting === go.Link.End to fix up the end points of the links // without losing the intermediate points of the route as determined by LayeredDigraphLayout link.invalidateRoute(); } } } } // end of SankeyLayout function init() { // Since 2.2 you can also author concise templates with method chaining instead of GraphObject.make // For details, see https://gojs.net/latest/intro/buildingObjects.html const $ = go.GraphObject.make; // for conciseness in defining templates myDiagram = new go.Diagram("myDiagramDiv", // the ID of the DIV HTML element { initialAutoScale: go.Diagram.UniformToFill, "animationManager.isEnabled": false, layout: $(SankeyLayout, { setsPortSpots: false, // to allow the "Side" spots on the nodes to take effect direction: 0, // rightwards layeringOption: go.LayeredDigraphLayout.LayerOptimalLinkLength, packOption: go.LayeredDigraphLayout.PackStraighten || go.LayeredDigraphLayout.PackMedian, layerSpacing: 100, // lots of space between layers, for nicer thick links columnSpacing: 1 }) }); var colors = ["#AC193D/#BF1E4B", "#2672EC/#2E8DEF", "#8C0095/#A700AE", "#5133AB/#643EBF", "#008299/#00A0B1", "#D24726/#DC572E", "#008A00/#00A600", "#094AB2/#0A5BC4"]; // this function provides a common style for the TextBlocks function textStyle() { return { font: "bold 12pt Segoe UI, sans-serif", stroke: "black", margin: 5 }; } // define the Node template myDiagram.nodeTemplate = $(go.Node, go.Panel.Horizontal, { locationObjectName: "SHAPE", locationSpot: go.Spot.Left, portSpreading: go.Node.SpreadingPacked // rather than the default go.Node.SpreadingEvenly }, $(go.TextBlock, textStyle(), { name: "LTEXT" }, new go.Binding("text", "ltext")), $(go.Shape, { name: "SHAPE", fill: "#2E8DEF", // default fill color strokeWidth: 0, portId: "", fromSpot: go.Spot.RightSide, toSpot: go.Spot.LeftSide, height: 10, width: 20 }, new go.Binding("fill", "color")), $(go.TextBlock, textStyle(), { name: "TEXT" }, new go.Binding("text")) ); function getAutoLinkColor(data) { var nodedata = myDiagram.model.findNodeDataForKey(data.from); var hex = nodedata.color; if (hex.charAt(0) == '#') { var rgb = parseInt(hex.slice(1, 7), 16); var r = rgb >> 16; var g = rgb >> 8 & 0xFF; var b = rgb & 0xFF; var alpha = 0.4; if (data.width <= 2) alpha = 1; var rgba = "rgba(" + r + "," + g + "," + b + ", " + alpha + ")"; return rgba; } return "rgba(173, 173, 173, 0.25)"; } // define the Link template var linkSelectionAdornmentTemplate = $(go.Adornment, "Link", $(go.Shape, { isPanelMain: true, fill: null, stroke: "rgba(0, 0, 255, 0.3)", strokeWidth: 0 }) // use selection object's strokeWidth ); myDiagram.linkTemplate = $(go.Link, go.Link.Bezier, { selectionAdornmentTemplate: linkSelectionAdornmentTemplate, layerName: "Background", fromEndSegmentLength: 150, toEndSegmentLength: 150, adjusting: go.Link.End }, $(go.Shape, { strokeWidth: 4, stroke: "rgba(173, 173, 173, 0.25)" }, new go.Binding("stroke", "", getAutoLinkColor), new go.Binding("strokeWidth", "width")) ); // read in the JSON-format data from the "mySavedModel" element load(); } function load() { myDiagram.model = go.Model.fromJson(document.getElementById("mySavedModel").value); } window.addEventListener('DOMContentLoaded', init); </script> <div id="sample"> <div id="myDiagramDiv" style="border: solid 1px black; width: 99%; height: 850px"></div> <p> A Sankey diagram is a type of flow diagram where the Link thickness is proportional to the flow quantity. </p> <div> <div> <button onclick="load()">Read</button> </div> <textarea id="mySavedModel" style="width:100%;height:250px"> { "class": "go.GraphLinksModel", "nodeDataArray": [ {"key":"Coal reserves", "text":"Coal reserves", "color":"#9d75c2"}, {"key":"Coal imports", "text":"Coal imports", "color":"#9d75c2"}, {"key":"Oil reserves", "text":"Oil\nreserves", "color":"#9d75c2"}, {"key":"Oil imports", "text":"Oil imports", "color":"#9d75c2"}, {"key":"Gas reserves", "text":"Gas reserves", "color":"#a1e194"}, {"key":"Gas imports", "text":"Gas imports", "color":"#a1e194"}, {"key":"UK land based bioenergy", "text":"UK land based bioenergy", "color":"#f6bcd5"}, {"key":"Marine algae", "text":"Marine algae", "color":"#681313"}, {"key":"Agricultural 'waste'", "text":"Agricultural 'waste'", "color":"#3483ba"}, {"key":"Other waste", "text":"Other waste", "color":"#c9b7d8"}, {"key":"Biomass imports", "text":"Biomass imports", "color":"#fea19f"}, {"key":"Biofuel imports", "text":"Biofuel imports", "color":"#d93c3c"}, {"key":"Coal", "text":"Coal", "color":"#9d75c2"}, {"key":"Oil", "text":"Oil", "color":"#9d75c2"}, {"key":"Natural gas", "text":"Natural\ngas", "color":"#a6dce6"}, {"key":"Solar", "text":"Solar", "color":"#c9a59d"}, {"key":"Solar PV", "text":"Solar PV", "color":"#c9a59d"}, {"key":"Bio-conversion", "text":"Bio-conversion", "color":"#b5cbe9"}, {"key":"Solid", "text":"Solid", "color":"#40a840"}, {"key":"Liquid", "text":"Liquid", "color":"#fe8b25"}, {"key":"Gas", "text":"Gas", "color":"#a1e194"}, {"key":"Nuclear", "text":"Nuclear", "color":"#fea19f"}, {"key":"Thermal generation", "text":"Thermal\ngeneration", "color":"#3483ba"}, {"key":"CHP", "text":"CHP", "color":"yellow"}, {"key":"Electricity imports", "text":"Electricity imports", "color":"yellow"}, {"key":"Wind", "text":"Wind", "color":"#cbcbcb"}, {"key":"Tidal", "text":"Tidal", "color":"#6f3a5f"}, {"key":"Wave", "text":"Wave", "color":"#8b8b8b"}, {"key":"Geothermal", "text":"Geothermal", "color":"#556171"}, {"key":"Hydro", "text":"Hydro", "color":"#7c3e06"}, {"key":"Electricity grid", "text":"Electricity grid", "color":"#e483c7"}, {"key":"H2 conversion", "text":"H2 conversion", "color":"#868686"}, {"key":"Solar Thermal", "text":"Solar Thermal", "color":"#c9a59d"}, {"key":"H2", "text":"H2", "color":"#868686"}, {"key":"Pumped heat", "text":"Pumped heat", "color":"#96665c"}, {"key":"District heating", "text":"District heating", "color":"#c9b7d8"}, {"key":"Losses", "ltext":"Losses", "color":"#fec184"}, {"key":"Over generation / exports", "ltext":"Over generation / exports", "color":"#f6bcd5"}, {"key":"Heating and cooling - homes", "ltext":"Heating and cooling - homes", "color":"#c7a39b"}, {"key":"Road transport", "ltext":"Road transport", "color":"#cbcbcb"}, {"key":"Heating and cooling - commercial", "ltext":"Heating and cooling - commercial", "color":"#c9a59d"}, {"key":"Industry", "ltext":"Industry", "color":"#96665c"}, {"key":"Lighting & appliances - homes", "ltext":"Lighting & appliances - homes", "color":"#2dc3d2"}, {"key":"Lighting & appliances - commercial", "ltext":"Lighting & appliances - commercial", "color":"#2dc3d2"}, {"key":"Agriculture", "ltext":"Agriculture", "color":"#5c5c10"}, {"key":"Rail transport", "ltext":"Rail transport", "color":"#6b6b45"}, {"key":"Domestic aviation", "ltext":"Domestic aviation", "color":"#40a840"}, {"key":"National navigation", "ltext":"National navigation", "color":"#a1e194"}, {"key":"International aviation", "ltext":"International aviation", "color":"#fec184"}, {"key":"International shipping", "ltext":"International shipping", "color":"#fec184"}, {"key":"Geosequestration", "ltext":"Geosequestration", "color":"#fec184"} ], "linkDataArray": [ {"from":"Coal reserves", "to":"Coal", "width":31}, {"from":"Coal imports", "to":"Coal", "width":86}, {"from":"Oil reserves", "to":"Oil", "width":244}, {"from":"Oil imports", "to":"Oil", "width":1}, {"from":"Gas reserves", "to":"Natural gas", "width":182}, {"from":"Gas imports", "to":"Natural gas", "width":61}, {"from":"UK land based bioenergy", "to":"Bio-conversion", "width":1}, {"from":"Marine algae", "to":"Bio-conversion", "width":1}, {"from":"Agricultural 'waste'", "to":"Bio-conversion", "width":1}, {"from":"Other waste", "to":"Bio-conversion", "width":8}, {"from":"Other waste", "to":"Solid", "width":1}, {"from":"Biomass imports", "to":"Solid", "width":1}, {"from":"Biofuel imports", "to":"Liquid", "width":1}, {"from":"Coal", "to":"Solid", "width":117}, {"from":"Oil", "to":"Liquid", "width":244}, {"from":"Natural gas", "to":"Gas", "width":244}, {"from":"Solar", "to":"Solar PV", "width":1}, {"from":"Solar PV", "to":"Electricity grid", "width":1}, {"from":"Solar", "to":"Solar Thermal", "width":1}, {"from":"Bio-conversion", "to":"Solid", "width":3}, {"from":"Bio-conversion", "to":"Liquid", "width":1}, {"from":"Bio-conversion", "to":"Gas", "width":5}, {"from":"Bio-conversion", "to":"Losses", "width":1}, {"from":"Solid", "to":"Over generation / exports", "width":1}, {"from":"Liquid", "to":"Over generation / exports", "width":18}, {"from":"Gas", "to":"Over generation / exports", "width":1}, {"from":"Solid", "to":"Thermal generation", "width":106}, {"from":"Liquid", "to":"Thermal generation", "width":2}, {"from":"Gas", "to":"Thermal generation", "width":87}, {"from":"Nuclear", "to":"Thermal generation", "width":41}, {"from":"Thermal generation", "to":"District heating", "width":2}, {"from":"Thermal generation", "to":"Electricity grid", "width":92}, {"from":"Thermal generation", "to":"Losses", "width":142}, {"from":"Solid", "to":"CHP", "width":1}, {"from":"Liquid", "to":"CHP", "width":1}, {"from":"Gas", "to":"CHP", "width":1}, {"from":"CHP", "to":"Electricity grid", "width":1}, {"from":"CHP", "to":"Losses", "width":1}, {"from":"Electricity imports", "to":"Electricity grid", "width":1}, {"from":"Wind", "to":"Electricity grid", "width":1}, {"from":"Tidal", "to":"Electricity grid", "width":1}, {"from":"Wave", "to":"Electricity grid", "width":1}, {"from":"Geothermal", "to":"Electricity grid", "width":1}, {"from":"Hydro", "to":"Electricity grid", "width":1}, {"from":"Electricity grid", "to":"H2 conversion", "width":1}, {"from":"Electricity grid", "to":"Over generation / exports", "width":1}, {"from":"Electricity grid", "to":"Losses", "width":6}, {"from":"Gas", "to":"H2 conversion", "width":1}, {"from":"H2 conversion", "to":"H2", "width":1}, {"from":"H2 conversion", "to":"Losses", "width":1}, {"from":"Solar Thermal", "to":"Heating and cooling - homes", "width":1}, {"from":"H2", "to":"Road transport", "width":1}, {"from":"Pumped heat", "to":"Heating and cooling - homes", "width":1}, {"from":"Pumped heat", "to":"Heating and cooling - commercial", "width":1}, {"from":"CHP", "to":"Heating and cooling - homes", "width":1}, {"from":"CHP", "to":"Heating and cooling - commercial", "width":1}, {"from":"District heating", "to":"Heating and cooling - homes", "width":1}, {"from":"District heating", "to":"Heating and cooling - commercial", "width":1}, {"from":"District heating", "to":"Industry", "width":2}, {"from":"Electricity grid", "to":"Heating and cooling - homes", "width":7}, {"from":"Solid", "to":"Heating and cooling - homes", "width":3}, {"from":"Liquid", "to":"Heating and cooling - homes", "width":3}, {"from":"Gas", "to":"Heating and cooling - homes", "width":81}, {"from":"Electricity grid", "to":"Heating and cooling - commercial", "width":7}, {"from":"Solid", "to":"Heating and cooling - commercial", "width":1}, {"from":"Liquid", "to":"Heating and cooling - commercial", "width":2}, {"from":"Gas", "to":"Heating and cooling - commercial", "width":19}, {"from":"Electricity grid", "to":"Lighting & appliances - homes", "width":21}, {"from":"Gas", "to":"Lighting & appliances - homes", "width":2}, {"from":"Electricity grid", "to":"Lighting & appliances - commercial", "width":18}, {"from":"Gas", "to":"Lighting & appliances - commercial", "width":2}, {"from":"Electricity grid", "to":"Industry", "width":30}, {"from":"Solid", "to":"Industry", "width":13}, {"from":"Liquid", "to":"Industry", "width":34}, {"from":"Gas", "to":"Industry", "width":54}, {"from":"Electricity grid", "to":"Agriculture", "width":1}, {"from":"Solid", "to":"Agriculture", "width":1}, {"from":"Liquid", "to":"Agriculture", "width":1}, {"from":"Gas", "to":"Agriculture", "width":1}, {"from":"Electricity grid", "to":"Road transport", "width":1}, {"from":"Liquid", "to":"Road transport", "width":122}, {"from":"Electricity grid", "to":"Rail transport", "width":2}, {"from":"Liquid", "to":"Rail transport", "width":1}, {"from":"Liquid", "to":"Domestic aviation", "width":2}, {"from":"Liquid", "to":"National navigation", "width":4}, {"from":"Liquid", "to":"International aviation", "width":38}, {"from":"Liquid", "to":"International shipping", "width":13}, {"from":"Electricity grid", "to":"Geosequestration", "width":1}, {"from":"Gas", "to":"Losses", "width":2} ]} </textarea> </div> This sample demonstrates one way of generating a Sankey or flow diagram. The data was derived from <a href="https://tamc.github.io/Sankey/">https://tamc.github.io/Sankey/</a>. </div> </div> <!-- * * * * * * * * * * * * * --> <!-- End of GoJS sample code --> </div> </body> <!-- This script is part of the gojs.net website, and is not needed to run the sample --> <script src="../assets/js/goSamples.js"></script> </html>