Radix cross Linux Build System

Cross-platform build system is designed to build distributions of different operating systems for a set of target devices

51 Commits   3 Branches   2 Tags
Index: requires_tree_html.template
===================================================================
--- requires_tree_html.template	(nonexistent)
+++ requires_tree_html.template	(revision 5)
@@ -0,0 +1,717 @@
+<!DOCTYPE html>
+<html>
+ <head>
+  <meta charset="utf-8">
+  <meta name="viewport" content="width=device-width, initial-scale=1.0">
+  <meta name="owner" content="Andrey V.Kosteltsev">
+  <meta name="author" content="Andrey V.Kosteltsev">
+  <meta http-equiv="content-type" content="text/html; charset=UTF-8">
+  <meta http-equiv="Content-script-type" content="text/javascript">
+  <meta http-equiv="Content-Style-Type" content="text/css">
+
+  <link href="" rel="icon" type="image/x-icon" />
+
+  <title>@ROOT@ &#8211; Requires Tree</title>
+
+  <style>
+   @import url(https://fonts.googleapis.com/css?family=Roboto:400,700italic,700,500italic,500,400italic&subset=cyrillic-ext,latin);
+   @import url(https://fonts.googleapis.com/css?family=Cousine:400,400italic,700,700italic&subset=cyrillic-ext,latin);
+  </style>
+
+  <style>
+   body, html {
+     margin: 0 0 0 0;
+   }
+
+   #front_wrapper {
+     margin: 0 auto;
+     height: 100vh;
+     position: relative;
+     overflow: auto;
+     background-color: #ececec;
+   }
+
+   #spinner {
+     margin: 0 auto;
+     min-height: 256px;
+     text-align: center;
+     display: flex;
+     align-items: center;
+   }
+
+   #tree_view {
+     margin: 0 auto;
+     min-height: 256px;
+     width: 2720px;
+     border: 0px solid #e7e7e7;
+   }
+
+   .header-wrapper {
+     height: 160px;
+     width: 100%;
+     margin: 0 auto;
+     position: relative;
+     background: transparent;
+   }
+
+   .content-wrapper {
+     background-color: #ffffff;
+   }
+
+   .footer-wrapper {
+     background: #ececec;
+   }
+
+   .content {
+     width: 1018px;
+     min-height: 256px;
+     padding: 18px 3px 12px 3px;
+     margin: 0 auto;
+     background-color: #fdfdfd;
+     position: relative;
+     overflow: hidden;
+     align: center;
+     border: 1px solid #e7e7e7;
+   }
+
+   .footer {
+     width: 1022px;
+     height: 48px;
+     margin: 0 auto;
+
+     -moz-border-radius-topleft: 0px;
+     -moz-border-radius-topright: 0px;
+     -moz-border-radius-bottomright: 4px;
+     -moz-border-radius-bottomleft: 4px;
+
+     -webkit-border-top-left-radius: 0px;
+     -webkit-border-top-right-radius: 0px;
+     -webkit-border-bottom-left-radius: 4px;
+     -webkit-border-bottom-right-radius: 4px;
+
+     border-top-left-radius: 0px;
+     border-top-right-radius: 0px;
+     border-bottom-left-radius: 4px;
+     border-bottom-right-radius: 4px;
+
+     border: 1px solid #545454;
+     background-color: #4c4c4c;
+     background: linear-gradient(288deg, rgb(84, 84, 84), rgb(76, 76, 76));
+   }
+
+   .footer-top {
+     margin: 2px auto 1px auto;
+     color: #ffffff;
+     text-align: center;
+   }
+
+   .footer-bottom {
+     margin: 0 8px 0 8px;
+     min-height: 20px;
+     color: #ffffff;
+     font-size: 10px;
+   }
+
+   .logo {
+     width: 1024px;
+     height: 80px;
+     margin: 0 auto;
+     background-color: transparent;
+   }
+
+   .navigator {
+     width: 1024px;
+     height: 79px;
+     margin: 0 auto;
+     padding: 1px 0 0;
+
+     -moz-border-radius-topleft: 4px;
+     -moz-border-radius-topright: 4px;
+     -moz-border-radius-bottomright: 0px;
+     -moz-border-radius-bottomleft: 0px;
+
+     -webkit-border-top-left-radius: 4px;
+     -webkit-border-top-right-radius: 4px;
+     -webkit-border-bottom-left-radius: 0px;
+     -webkit-border-bottom-right-radius: 0px;
+
+     border-top-left-radius: 4px;
+     border-top-right-radius: 4px;
+     border-bottom-left-radius: 0px;
+     border-bottom-right-radius: 0px;
+
+     border: 1px solid #545454;
+     background-color: #4c4c4c;
+     background: linear-gradient(288deg, rgb(84, 84, 84), rgb(76, 76, 76));
+   }
+
+   .copyright {
+     color: #f0f0ea;
+     text-decoration: none;
+     font-family: 'Roboto', helvetica, arial, sans-serif;
+     font-weight: bold;
+     font-style: normal;
+     font-size: 12px;
+   }
+
+   .copyright:hover {
+     text-decoration: underline;
+   }
+
+
+   .date-title {
+     height: 16px;
+     font: 12px 'Roboto', sans-serif;
+     font-weight: bold;
+     padding-top: 6px;
+     margin-bottom: -10px;
+     padding-left: 16px;
+     color: #c0c0c0;
+   }
+   .time-title {
+     color: #82946f;
+   }
+   .hardware-title {
+     height: 20px;
+     float: right;
+     text-align: right;
+     padding-right: 16px;
+     width: 512px; font: 14px 'Roboto', sans-serif;
+     font-weight: bold;
+     color: #f0f0ea;
+   }
+   .hw-title {
+     font: 10px 'Roboto', sans-serif;
+     font-weight: bold;
+     color: #cadaba;
+   }
+   .tree-title {
+     height: 42px;
+     padding-left: 16px;
+     font: 28px 'Roboto', sans-serif;
+     font-weight: bold;
+     color: #f0f0ea;
+   }
+   .tree-hw-title {
+     color: #cadaba;
+   }
+
+   /* SVG spinner icon animation */
+   .spinner {
+     -webkit-animation: rotate 2s linear infinite;
+             animation: rotate 2s linear infinite;
+     z-index: 2;
+     position: relative;
+     top: 50%;
+     left: 50%;
+     margin: -25px 0 0 -25px;
+     width: 50px;
+     height: 50px;
+   }
+   .spinner-text {
+     z-index: 2;
+     position: absolute;
+     top: 0;
+     left: 0;
+     margin: 36px;
+     font: 28px 'Roboto', sans-serif;
+     color: #c0c0c0;
+   }
+   .spinner .path {
+     stroke: #cccccc;
+     stroke-linecap: round;
+     -webkit-animation: dash 1.5s ease-in-out infinite;
+             animation: dash 1.5s ease-in-out infinite;
+   }
+
+   @-webkit-keyframes rotate {
+     100% { -webkit-transform: rotate(360deg); transform: rotate(360deg); }
+   }
+   @keyframes rotate {
+     100% { -webkit-transform: rotate(360deg); transform: rotate(360deg); }
+   }
+   @-webkit-keyframes dash {
+       0% { stroke-dasharray:  1, 150; stroke-dashoffset:    0; }
+      50% { stroke-dasharray: 90, 150; stroke-dashoffset:  -35; }
+     100% { stroke-dasharray: 90, 150; stroke-dashoffset: -124; }
+   }
+   @keyframes dash {
+       0% { stroke-dasharray:  1, 150; stroke-dashoffset:    0; }
+      50% { stroke-dasharray: 90, 150; stroke-dashoffset:  -35; }
+     100% { stroke-dasharray: 90, 150; stroke-dashoffset: -124; }
+   }
+
+
+   .node {
+     cursor: pointer;
+   }
+   .node text {
+     font: 14px 'Cousine', monospace;
+   }
+
+   .tree-tooltip {
+     position: absolute;
+     text-align: left;
+     padding: 16px 16px 8px;
+     background-color: #fafafa;
+     border: 1px solid #71ad93;
+     border-radius: 8px;
+     pointer-events: none;
+     color: #343434;
+     -webkit-box-shadow: 0 0 5px #aaa;
+     box-shadow: 0 0 5px #aaa;
+   }
+   .tooltip-header {
+     font: 14px Roboto, sans-serif;
+     font-weight: bold;
+     color: DarkRed;
+
+     white-space: nowrap;
+     text-align: left;
+   }
+   .tooltip-header-not-packaged {
+     font: 11px Cousine,monospace;
+     font-weight: bold;
+     color: DarkRed;
+
+     white-space: nowrap;
+
+     padding-left: 8px;
+     padding-right: 8px;
+     padding-bottom: 8px;
+     text-align: left;
+   }
+   .tooltip-description {
+     font: 14px Roboto, sans-serif;
+     font-style: italic;
+     font-weight: bold;
+     color: #343434;
+
+     white-space: nowrap;
+     text-align: left;
+     padding-left: 1.5em;
+     padding-top: .5em;
+     font-style: italic;
+   }
+   .tooltip-content {
+     font: 11px 'Cousine', monospace;
+     font-weight: bold;
+
+     white-space: pre;
+     margin: 12px 0 8px;
+   }
+   .flavour {
+     color: DarkBlue;
+   }
+
+
+   @media (min-width: 1200px) {
+     .navigator { width: 1140px; }
+     .logo      { width: 1140px; }
+     .footer    { width: 1140px; }
+     .content   { width: 1134px; }
+   }
+   @media (min-width: 992px) and (max-width: 1199px) {
+     .navigator { width: 960px; }
+     .logo      { width: 960px; }
+     .footer    { width: 960px; }
+     .content   { width: 954px; }
+   }
+   @media (min-width: 768px) and (max-width: 991px) {
+     .navigator { width: 720px; }
+     .logo      { width: 720px; }
+     .footer    { width: 720px; }
+     .content   { width: 714px; }
+   }
+   @media (min-width: 576px) and (max-width: 767px) {
+     .navigator { width: 540px; }
+     .logo      { width: 540px; }
+     .footer    { width: 540px; }
+     .content   { width: 534px; }
+
+     .node text { font-size: 12px; }
+
+     .tooltip-header      { font-size: 12px; }
+     .tooltip-description { font-size: 12px; }
+     .tooltip-content     { font-size: 10px; }
+   }
+   @media (max-width: 575px) {
+     .navigator { width: 480px; }
+     .logo      { width: 480px; }
+     .footer    { width: 480px; }
+     .content   { width: 474px; }
+
+     .node text { font-size: 12px; }
+
+     .tooltip-header      { font-size: 12px; }
+     .tooltip-description { font-size: 12px; }
+     .tooltip-content     { font-size: 10px; }
+   }
+  </style>
+
+  <script src="https://code.jquery.com/jquery-3.4.1.min.js"></script>
+  <script src="https://code.jquery.com/ui/1.12.1/jquery-ui.min.js"></script>
+  <script src="https://d3js.org/d3.v5.min.js"></script>
+  <script>
+   !function(o){function t(o,t){if(!(o.originalEvent.touches.length>1)){o.preventDefault();var e=o.originalEvent.changedTouches[0],n=document.createEvent("MouseEvents");n.initMouseEvent(t,!0,!0,window,1,e.screenX,e.screenY,e.clientX,e.clientY,!1,!1,!1,!1,0,null),o.target.dispatchEvent(n)}}if(o.support.touch="ontouchend"in document,o.support.touch){var e,n,u=o.ui.mouse.prototype,c=u._mouseInit,i=u._mouseDestroy;u._touchStart=function(o){var u=this;!n&&u._mouseCapture(o.originalEvent.changedTouches[0])&&(n=!0,u._touchMoved=!1,e=o,t(o,"mouseover"),t(o,"mousemove"),t(o,"mousedown"))},u._touchMove=function(o){if(n){var u=e.originalEvent.touches[0].screenX,c=e.originalEvent.touches[0].screenY,i=o.originalEvent.touches[0].screenX,r=o.originalEvent.touches[0].screenY;if(u===i&&c===r)return void(this._touchMoved=!1);this._touchMoved=!0,t(o,"mousemove")}},u._touchEnd=function(o){n&&(t(o,"mouseup"),t(o,"mouseout"),this._touchMoved||t(o,"click"),n=!1)},u._mouseInit=function(){var t=this;t.element.bind({touchstart:o.proxy(t,"_touchStart"),touchmove:o.proxy(t,"_touchMove"),touchend:o.proxy(t,"_touchEnd")}),c.call(t)},u._mouseDestroy=function(){var t=this;t.element.unbind({touchstart:o.proxy(t,"_touchStart"),touchmove:o.proxy(t,"_touchMove"),touchend:o.proxy(t,"_touchEnd")}),i.call(t)}}}(jQuery);
+   $(function() {
+     $( "#tree_view" ).draggable();
+   });
+  </script>
+  <script>
+   function load_json( url, callback ) {
+     var xobj = new XMLHttpRequest();
+     xobj.overrideMimeType("application/json");
+     xobj.open('GET', url, true);
+     xobj.onreadystatechange = function () {
+       if (xobj.readyState == 4 && xobj.status == "200") {
+         callback(xobj.responseText);
+       }
+     };
+     xobj.send(null);
+   }
+
+   var pkgs;
+
+   $(document).ready(function() {
+     load_json( '@JSON_PKGS_FILE@', function(response) {
+       pkgs = JSON.parse(response);
+     });
+
+     $('#tree_view')
+       .mousedown(function() { $(this).css( 'cursor', 'grab' ); })
+       .mouseup(  function() { $(this).css( 'cursor', 'auto' ); });
+   });
+  </script>
+ </head>
+ <body>
+  <div id="front_wrapper">
+   <div class="header-wrapper">
+    <div class="logo"></div>
+    <div class="navigator">
+     <div style="height: 36px;">
+       <div class="date-title">@YEAR@-@MONTH@-@DAY@&nbsp;&nbsp;<span class="time-title">@HOUR@:@MINUTE@:@SECOND@</span></div>
+       <div class="hardware-title">
+        <span class="hw-title">HARDWARE:</span> @HARDWARE@
+       </div>
+     </div>
+     <div class="tree-title">
+      <span class="tree-hw-title">@ROOT@</span> &#8211; Requires Tree
+     </div>
+    </div> <!-- "navigator" -->
+   </div> <!-- "header_wrapper" -->
+
+   <div class="content-wrapper">
+    <div class="content">
+     <div id="spinner">
+      <svg class="spinner" viewBox="0 0 50 50"><circle class="path" cx="25" cy="25" r="20" fill="none" stroke-width="5"></circle></svg>
+      <div class="spinner-text">Loading ...</div>
+     </div>
+     <div id="tree_view" class="ui-widget-content">
+     </div>
+    </div> <!-- "content" -->
+   </div> <!-- "content_wrapper" -->
+
+   <div class="footer-wrapper">
+    <div class="footer">
+     <div class="footer-top">
+      <a class="copyright" target="_blank" href="@BUG_URL@">&#169; @COPYING@</a>
+     </div>
+     <div class="footer-bottom">
+     </div>
+    </div> <!-- "footer" -->
+   </div> <!-- "footer_wrapper" -->
+  </div> <!-- "front_wrapper" -->
+
+  <script>
+   var margin = {top: 20, right: 120, bottom: 20, left: 220},
+       width = @SVG_WIDTH@ - margin.right - margin.left,
+       height = @SVG_HEIGHT@ - margin.top - margin.bottom;
+
+   var i = 0,
+       duration = 750,
+       root = 0;
+
+   var treemap = d3.tree()
+       .size([height, width]);
+
+   var svg = d3.select(document.getElementById( 'tree_view' )).append("svg")
+       .attr("width", width + margin.right + margin.left)
+       .attr("height", height + margin.top + margin.bottom)
+       .append("g")
+       .attr("transform", "translate(" + margin.left + "," + margin.top + ")");
+
+   var div = d3.select(document.getElementById( 'front_wrapper' )).append("div")
+       .attr("class", "tree-tooltip")
+       .style("display", "none")
+       .style("opacity", 0);
+
+
+   load_json( '@JSON_TREE_FILE@', function(response) {
+     var treeData = JSON.parse(response);
+
+     /* Assigns parent, children, height, depth: */
+     root = d3.hierarchy(treeData, function(d) { return d.children; });
+
+     root.x0 = height / 2;
+     root.y0 = 0;
+
+     function collapse(d) {
+       if( d.children ) {
+         d._children = d.children;
+         d._children.forEach(collapse);
+         d.children = null;
+       }
+     }
+
+     document.getElementById('spinner').remove();
+     root.children.forEach(collapse);
+     update(root);
+   });
+
+
+   function update(source) {
+
+     var tree = treemap( root );
+
+     /* Compute the new tree layout. */
+     var nodes = tree.descendants(),
+         links = tree.descendants().slice(1);
+
+     /* Normalize for fixed-depth. */
+     nodes.forEach(function(d) { d.y = d.depth * 220; });
+
+     /* Update the nodes . . . */
+     var node = svg.selectAll("g.node")
+       .data(nodes, function(d) { return d.id || (d.id = ++i); });
+
+     /* Enter any new nodes at the parent's previous position. */
+     var nodeEnter = node.enter().append("g")
+       .attr("class", "node")
+       .attr("transform", function(d) { return "translate(" + source.y0 + "," + source.x0 + ")"; })
+       .on("click", click)
+       .on("mouseover", function(d) {
+         div.transition()
+           .duration(200)
+           .style("opacity", .92);
+         {
+           var content = '<div class="tooltip-header-not-packaged">' + 'void' + '</div>';
+
+           if( d.name === "void" ) {
+             /* draw div.tree-tooltip to get actual size */
+             div.html( content )
+               .style("left", (d3.event.pageX + document.getElementById( 'front_wrapper' ).scrollLeft + 12) + "px")
+               .style("top",  (d3.event.pageY + document.getElementById( 'front_wrapper' ).scrollTop  + 12) + "px");
+           }
+           else
+           {
+             /* find package in the pkgs array: */
+             var pkg = pkgs.find(obj => { return obj.id === d.data.name; });
+
+             if( pkg === undefined )
+             {
+               content = '<div class="tooltip-header-not-packaged">' + 'not packaged collection' + '</div>';
+               /* draw div.tree-tooltip to get actual size */
+               div.html( content )
+                 .style("left", (d3.event.pageX + document.getElementById( 'front_wrapper' ).scrollLeft + 12) + "px")
+                 .style("top",  (d3.event.pageY + document.getElementById( 'front_wrapper' ).scrollTop  + 12) + "px");
+             }
+             else
+             {
+               content  = '<div class="tooltip-header">' + pkg.name + '</div>' +
+                           '<div class="tooltip-description">' + pkg.description + '</div>' +
+                            '<div class="tooltip-content">' +
+                            '               group: ' + pkg.group + '\n' +
+                            '        architecture: ' + pkg.arch + '\n' +
+                            '            hardware: ' + pkg.hardware + '\n';
+               if( pkg.flavour !== undefined )
+               {
+                 content += '             <span class="flavour">edition</span>: ' + pkg.flavour + '\n';
+               }
+                 content += '             license: ' + pkg.license + '\n' +
+                            '      bug report url: ' + root.data.distro[2] + '\n' +
+                            '        distribution: ' + root.data.distro[0] + '-' + root.data.distro[1] + '\n' +
+                            '     package tarball: ' + pkg.name + '-' + pkg.version + '-' + pkg.arch + '-' + root.data.distro[0] + '-' + root.data.distro[1] + '.' + '@TARBALL_SUFFIX@' + '\n' +
+                            '   uncompressed size: ' + pkg.uncompressed_size + '\n' +
+                            '     number of files: ' + pkg.total_files + '\n' +
+                            '</div>' +
+                           '</div>' +
+                          '</div>';
+
+               /* draw div.tree-tooltip to get actual size */
+               div.html( content )
+                 .style("left", (d3.event.pageX + document.getElementById( 'front_wrapper' ).scrollLeft + 12) + "px")
+                 .style("top",  (d3.event.pageY + document.getElementById( 'front_wrapper' ).scrollTop  + 12) + "px");
+             }
+           }
+
+           /* draw div.tree-tooltip at actual position */
+
+           var cW = $( window ).width();
+           var cH = $( window ).height();
+           var cX = d3.event.pageX;
+           var cY = d3.event.pageY;
+           var tW = $('div.tree-tooltip').width();
+           var tH = $('div.tree-tooltip').height();
+           var oX;
+           var oY;
+           var dX = ( cW - cX ) - ( tW + 12 );
+           var dY = ( cH - cY ) - ( tH + 12 );
+
+           /* shift left to according to width=16 of browser vertical scroll bar */
+           if( dX <= 24 ) { dX = 24 - dX; }
+           else           { dX = 0;       }
+
+           /* shift top to according to width=16 of browser horizontal scroll bar */
+           if( dY <= 24 ) { dY = 24 - dY; }
+           else           { dY = 0;       }
+
+           if( ( cW - cX ) < ( tW + 12 ) ) { oX = - 12 - tW; } else { oX = 12 - dX; }
+           if( ( cH - cY ) < ( tH + 12 ) ) { oY = - 12 - tH; } else { oY = 12 - dY; }
+
+           if( (( cW - cX ) < ( tW + 12 )) && (cX < ( tW + 12 )) )
+           {
+             /* in this case we have to center tooltip */
+             oX = - (tW + 12) / 2 + (cW/2 - cX);
+           }
+
+           div.html( content )
+             .style("left", (d3.event.pageX + document.getElementById( 'front_wrapper' ).scrollLeft + oX) + "px")
+             .style("top",  (d3.event.pageY + document.getElementById( 'front_wrapper' ).scrollTop +  oY) + "px")
+             .style("display","block");
+         }
+       })
+       .on("mouseout", function(d) {
+         div.transition()
+           .duration(500)
+           .style("opacity", 0);
+       });
+
+     nodeEnter.append("circle")
+       .attr('class', 'node')
+     /* Additional attributes (see the 'style' section) */
+       .attr("stroke", "#5d5d5d")
+       .attr("stroke-width", "1.0")
+     /* End of additional attributes */
+       .attr("r", 1e-6)
+       .style("fill", function(d) { return d._children ? "#abd8d4" : "#fff"; });
+
+     nodeEnter.append("text")
+       .attr("x", function(d) { return d.children || d._children ? -10 : 10; })
+       .attr("dy", "-.35em")
+       .attr("text-anchor", function(d) { return d.children || d._children ? "end" : "start"; })
+       .text(function(d) { return (d.data.name.indexOf(":",0) > 0 ) ? d.data.name.substr(d.data.name.indexOf(":",0) + 1) : d.data.name; })
+       .style("fill-opacity", 1);
+
+     /* Update */
+     var nodeUpdate = nodeEnter.merge(node);
+
+     /* Transition nodes to their new position. */
+     nodeUpdate.transition()
+       .duration(duration)
+       .attr("transform", function(d) { return "translate(" + d.y + "," + d.x + ")"; });
+
+     nodeUpdate.select("circle.node")
+       .attr("r", 4.5)
+       .style("fill", function(d) {
+         if( d._children )
+         {
+           return "#abd8d4";
+         }
+         else
+         {
+           if( d.children == undefined )
+           {
+             if( d.name == "void" )
+             {
+               return "#c9c9c9";
+             }
+             else
+             {
+               return "#fff";
+             }
+           }
+           else
+           {
+             return "#d2ebd8";
+           }
+         }
+       })
+       .attr('cursor', 'pointer');
+
+     /* Transition exiting nodes to the parent's new position. */
+     var nodeExit = node.exit().transition()
+       .duration(duration)
+       .attr("transform", function(d) { return "translate(" + source.y + "," + source.x + ")"; })
+       .remove();
+
+     nodeExit.select("circle")
+       .attr("r", 1e-6);
+
+     nodeExit.select("text")
+       .style("fill-opacity", 1e-6);
+
+     /* Update the links . . . */
+     var link = svg.selectAll('path.link')
+       .data(links, function(d) { return d.id; });
+
+     /* Enter any new links at the parent's previous position. */
+     var linkEnter = link.enter().insert('path', 'g')
+       .attr("class", "link")
+       .attr("d", function(d) {
+         var o = {x: source.x0, y: source.y0};
+         return diagonal(o, o);
+       });
+
+     /* Update */
+     var linkUpdate = linkEnter.merge(link);
+
+     /* Transition links to their new position. */
+     linkUpdate.transition()
+       .duration(duration)
+     /* Additional attributes (see the 'style' section) */
+       .style("fill", "none")
+       .attr("stroke", "DarkGray")
+       .attr("stroke-width", "1.5")
+     /* End of additional attributes */
+       .attr("d", function(d){ return diagonal(d, d.parent) });
+
+     /* Transition exiting nodes to the parent's new position. */
+     var linkExit = link.exit().transition()
+       .duration(duration)
+       .attr("d", function(d) {
+         var o = {x: source.x, y: source.y};
+         return diagonal(o, o);
+       })
+       .remove();
+
+     /* Stash the old positions for transition. */
+     nodes.forEach(function(d) {
+       d.x0 = d.x;
+       d.y0 = d.y;
+     });
+
+     /* Creates a curved (diagonal) path from parent to the child nodes. */
+     function diagonal(s, d) {
+       path = `M ${s.y} ${s.x}
+               C ${(s.y + d.y) / 2} ${s.x},
+                 ${(s.y + d.y) / 2} ${d.x},
+                 ${d.y} ${d.x}`;
+       return path;
+     }
+
+     /* Toggle children on click. */
+     function click(d) {
+       if (d.children) {
+         d._children = d.children;
+         d.children = null;
+       } else {
+         d.children = d._children;
+         d._children = null;
+       }
+       update(d);
+     }
+   }
+  </script>
+ </body>
+</html>