function generateMandala(){function t(t,e,a){void 0===e&&(e="y1"),void 0===a&&(a="y2");var r=yScale(t[e]),i=yScale(t[a]),l=(sliceHeight-r)*Math.tan(-sliceAngle/2),n=(sliceHeight-i)*Math.tan(sliceAngle/2);return{x1:l,x2:n,y1:r,y2:i}}function e(t){var e=t.x1,a=t.x2,r=t.y1,i=t.y2;return"M"+e+","+r+" L"+a+","+i}var a=d3.select("#vis-container");a.selectAll("*").remove();var r=a.append("svg").attr("width",width).attr("height",height),i=Math.floor(360*Math.random()),l=.3*Math.random()+.7,n=.15*Math.random()+.05,s=d3.hsl(i,l,n);d3.select("body").style("background"),r.append("rect").attr("class","mandala-bg").attr("width",width).attr("height",height).style("fill",s);var d=r.append("g").attr("transform","translate("+padding.left+" "+padding.top+")");animate&&d.transition().duration(2500).attrTween("transform",function(){return d3.interpolateString("translate("+padding.left+" "+padding.top+") rotate(0 "+plotAreaWidth/2+" "+plotAreaHeight/2+")","translate("+padding.left+" "+padding.top+") rotate(360 "+plotAreaWidth/2+" "+plotAreaHeight/2+")")});var c=d.append("defs"),o=c.append("radialGradient").attr("id","bg-shading").attr("gradientUnits","userSpaceOnUse");o.append("stop").attr("offset","0%").attr("stop-color","#000").attr("stop-opacity",0),o.append("stop").attr("offset","100%").attr("stop-color","#000").attr("stop-opacity",.2),r.insert("rect","g").attr("class","mandala-bg-shading").attr("width",width).attr("height",height).style("fill","url(#bg-shading)");var p=c.append("clipPath").attr("id","marks-clip").append("circle").attr("cx",plotAreaWidth/2).attr("cy",plotAreaHeight/2).attr("r",0).style("fill","#fff");animate?p.transition().ease(d3.easeLinear).duration(2e3).attr("r",plotAreaHeight/2+5):p.attr("r",plotAreaHeight/2+5);var u=d.append("g").attr("class","slices-group").attr("clip-path","url(#marks-clip)"),h=u.append("g").attr("id","ref-slice").attr("class","slice").attr("transform","translate("+plotAreaWidth/2+" 0)").attr("clip-path","url(#slice-clip)"),y=d3.range(numSlices-1).map(function(t,e){return{id:e+1,href:"#ref-slice",transform:"rotate("+(e+1)*sliceAngle*(180/Math.PI)+" "+plotAreaWidth/2+" "+sliceHeight+")"}}),g=u.selectAll("copy-slice").data(y);g.enter().append("use").attr("xlink:href",function(t){return t.href}).attr("transform",function(t){return t.transform}),h.append("path").attr("class","slice-bg").attr("transform","translate(0 "+sliceHeight+")").attr("d",arc({innerRadius:0,outerRadius:sliceHeight,startAngle:-(sliceAngle/2),endAngle:sliceAngle/2})).style("fill","none").style("stroke","tomato").style("opacity",0);var f="#fff",m=h.selectAll(".point").data(dataByType.point||[]);m.enter().append("circle").attr("class","point").attr("r",function(t){return t.r}).attr("cx",0).attr("cy",function(t){return yScale(t.y)}).style("fill",function(t){return t.filled?f:"none"}).style("stroke",function(t){return t.filled?"none":f});var v=h.selectAll(".arc").data(dataByType.arc||[]),S=d3.arc().innerRadius(function(t){return rScale(t.y-t.thickness)}).outerRadius(function(t){return rScale(t.y)}).startAngle(-sliceAngle/2-.1).endAngle(sliceAngle/2+.1);v.enter().append("path").attr("transform","translate(0 "+sliceHeight+")").attr("class","arc").attr("d",S).style("fill",f);var z=h.selectAll(".diagonalUp").data(dataByType.diagonalUp||[]);z.enter().append("path").attr("class","diagonalUp").attr("d",function(a){return e(t(a))}).style("stroke",f).style("fill",f);var A=h.selectAll(".diagonalDown").data(dataByType.diagonalDown||[]);A.enter().append("path").attr("class","diagonalDown").attr("d",function(a){return e(t(a))}).style("stroke",f).style("fill",f);var M=h.selectAll(".x").data(dataByType.x||[]),k=M.enter().append("g").attr("class","x");k.append("path").attr("d",function(a){return e(t(a))}).style("stroke",f).style("fill",f),k.append("path").attr("d",function(a){return e(t(a,"y2","y1"))}).style("stroke",f).style("fill",f);var w=h.selectAll(".arrow").data(dataByType.arrow||[]),H=w.enter().append("g").attr("class","arrow");H.append("path").attr("d",function(a){return e(t(a,"y1","yMid"))}).style("stroke",f).style("fill",f),H.append("path").attr("d",function(a){return e(t(a,"y2","yMid"))}).style("stroke",f).style("fill",f)}var markTypes=["x","arrow","arc","point"],animate=!0,numMarks=30,cumulativeSize=0,prevType,data=d3.range(numMarks).map(function(t,e){var a,r;do r=!0,a=markTypes[Math.floor(Math.random()*markTypes.length)],e>5&&"arrow"===a&&(r=!1);while(!r);prevType=a;var i;if("point"===a){var l=Math.ceil(20*Math.random())+10;i={type:a,r:l/5,size:l,cumulativeSize:cumulativeSize,y:cumulativeSize+l/2,filled:Math.random()>.3}}else if("arc"===a){var n=Math.ceil(20*Math.random())+2;i={type:a,thickness:Math.round(n/4),size:n,cumulativeSize:cumulativeSize,y:cumulativeSize+n/2}}else if("diagonalUp"===a){var s=Math.ceil(10*Math.random())+3;i={type:a,size:s,cumulativeSize:cumulativeSize,y1:cumulativeSize,y2:cumulativeSize+s}}else if("diagonalDown"===a){var d=Math.ceil(10*Math.random())+3;i={type:a,size:d,cumulativeSize:cumulativeSize,y1:cumulativeSize+d,y2:cumulativeSize}}else if("x"===a){var c=Math.ceil(10*Math.random())+3;i={type:a,size:c,cumulativeSize:cumulativeSize,y1:cumulativeSize+c,y2:cumulativeSize}}else if("arrow"===a){var o=Math.ceil(10*Math.random())+3;i={type:a,size:o,cumulativeSize:cumulativeSize,y1:cumulativeSize+o,yMid:cumulativeSize+o/2,y2:cumulativeSize}}else i={size:0};return i.id=e,cumulativeSize+=i.size,i}),dataByType=d3.nest().key(function(t){return t.type}).object(data),width=600,height=600,padding={top:20,right:20,bottom:20,left:20},plotAreaWidth=width-padding.left-padding.right,plotAreaHeight=height-padding.top-padding.bottom,numSlices=32,sliceHeight=plotAreaHeight/2,sliceAngle=2*Math.PI/numSlices,yScale=d3.scaleLinear().domain([0,cumulativeSize]).range([sliceHeight,0]),rScale=d3.scaleLinear().domain([0,cumulativeSize]).range([0,sliceHeight]),arc=d3.arc();generateMandala(),d3.select("#make-mandala").on("click",generateMandala); //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["script.js"],"names":["generateMandala","dToLine","d","y1Key","y2Key","const","y1","yScale","y2","x1","sliceHeight","Math","tan","sliceAngle","x2","toPath","ref","container","d3","select","selectAll","remove","svg","append","attr","width","height","bgHue","floor","random","bgSaturation","bgLightness","bg","hsl","style","g","padding","animate","transition","duration","attrTween","interpolateString","plotAreaWidth","plotAreaHeight","defs","mandalaBgGrad","insert","marksClip","ease","easeLinear","gSlices","slice","copySlices","range","numSlices","map","i","id","href","transform","PI","sliceBinding","data","enter","arc","innerRadius","outerRadius","startAngle","endAngle","markColor","points","dataByType","point","r","y","filled","arcs","interiorArc","rScale","thickness","diagUp","diagonalUp","diagDown","diagonalDown","xMarks","x","xMarkGs","arrowMarks","arrow","arrowMarkGs","markTypes","numMarks","cumulativeSize","prevType","let","type","validType","length","item","size","ceil","round","yMid","nest","key","object","top","right","bottom","left","scaleLinear","domain","on"],"mappings":"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","file":"script.js","sourcesContent":["// mark types\n// const markTypes = ['diagonalUp', 'diagonalDown', 'x', 'arc', 'point']; // 'square'];\nconst markTypes = ['x', 'arrow', 'arc', 'point']; // 'square'];\nconst animate = true;\nconst numMarks = 30;\n\n// generate random data\nlet cumulativeSize = 0;\nlet prevType;\nconst data = d3.range(numMarks).map((d, i) => {\n  let type;\n  let validType;\n  do {\n    validType = true;\n    type = markTypes[Math.floor(Math.random() * markTypes.length)];\n\n    if (i > 5 && type === 'arrow') {\n      validType = false;\n    }\n  } while (!validType);\n\n  // type = 'arrow';\n  prevType = type;\n\n  let item;\n\n  if (type === 'point') {\n    const size = Math.ceil(Math.random() * 20) + 10;\n    item = {\n      type,\n      r: size / 5,\n      size,\n      cumulativeSize,\n      y: cumulativeSize + (size / 2),\n      filled: Math.random() > 0.3,\n    };\n  } else if (type === 'arc') {\n    const size = Math.ceil(Math.random() * 20) + 2;\n    item = {\n      type,\n      thickness: Math.round(size / 4),\n      size,\n      cumulativeSize,\n      y: cumulativeSize + (size / 2),\n    };\n  } else if (type === 'diagonalUp') {\n    const size = Math.ceil(Math.random() * 10) + 3;\n    item = {\n      type,\n      size,\n      cumulativeSize,\n      y1: cumulativeSize,\n      y2: cumulativeSize + size,\n    };\n  } else if (type === 'diagonalDown') {\n    const size = Math.ceil(Math.random() * 10) + 3;\n    item = {\n      type,\n      size,\n      cumulativeSize,\n      y1: cumulativeSize + size,\n      y2: cumulativeSize,\n    };\n  } else if (type === 'x') {\n    const size = Math.ceil(Math.random() * 10) + 3;\n    item = {\n      type,\n      size,\n      cumulativeSize,\n      y1: cumulativeSize + size,\n      y2: cumulativeSize,\n    };\n  } else if (type === 'arrow') {\n    const size = Math.ceil(Math.random() * 10) + 3;\n    item = {\n      type,\n      size,\n      cumulativeSize,\n      y1: cumulativeSize + size,\n      yMid: cumulativeSize + (size / 2),\n      y2: cumulativeSize,\n    };\n  } else {\n    item = { size: 0 };\n  }\n\n  item.id = i;\n  cumulativeSize += item.size;\n\n  return item;\n});\n\nconst dataByType = d3.nest().key(d => d.type).object(data);\n\n// outer svg dimensions\nconst width = 600;\nconst height = 600;\n\n// padding around the chart\nconst padding = {\n  top: 20,\n  right: 20,\n  bottom: 20,\n  left: 20,\n};\n\n// inner chart dimensions, where the dots are plotted\nconst plotAreaWidth = width - padding.left - padding.right;\nconst plotAreaHeight = height - padding.top - padding.bottom;\n\n// size of an individual slice\nconst numSlices = 32;\nconst sliceHeight = plotAreaHeight / 2;\nconst sliceAngle = (2 * Math.PI) / numSlices;\n\n// initialize scales\nconst yScale = d3.scaleLinear().domain([0, cumulativeSize]).range([sliceHeight, 0]);\nconst rScale = d3.scaleLinear().domain([0, cumulativeSize]).range([0, sliceHeight]);\n\nconst arc = d3.arc();\n\nfunction generateMandala() {\n\n\t// select the root container where the chart will be added\n\tconst container = d3.select('#vis-container');\n\n\t// clear any old contents\n\tcontainer.selectAll('*').remove();\n\n\t// initialize main SVG\n\tconst svg = container.append('svg')\n\t  .attr('width', width)\n\t  .attr('height', height);\n\n\n\tconst bgHue = Math.floor(Math.random() * 360);\n\tconst bgSaturation = (Math.random() * 0.3) + 0.7;\n\tconst bgLightness = (Math.random() * 0.15) + 0.05;\n\n\tconst bg = d3.hsl(bgHue, bgSaturation, bgLightness);\n\td3.select('body').style('background');\n\n\t// draw the background\n\tsvg.append('rect')\n\t  .attr('class', 'mandala-bg')\n\t  .attr('width', width)\n\t  .attr('height', height)\n\t  .style('fill', bg);\n\n\t// the main <g> where all the chart content goes inside\n\tconst g = svg.append('g')\n\t  .attr('transform', `translate(${padding.left} ${padding.top})`);\n\n\tif (animate) {\n\t  g.transition()\n\t    .duration(2500)\n\t    .attrTween('transform', () =>\n\t      d3.interpolateString(`translate(${padding.left} ${padding.top}) rotate(0 ${plotAreaWidth / 2} ${plotAreaHeight / 2})`,\n\t        `translate(${padding.left} ${padding.top}) rotate(360 ${plotAreaWidth / 2} ${plotAreaHeight / 2})`));\n\t}\n\n\n\tconst defs = g.append('defs');\n\n\t// clip path for slices disabled to allow some slight overlap for things like arcs\n\t// add the slice as a clip path\n\t// defs.append('clipPath')\n\t//   .attr('id', 'slice-clip')\n\t//   .append('path')\n\t//   .attr('transform', `translate(0 ${sliceHeight})`)\n\t//   .attr('d', arc({\n\t//     innerRadius: 0,\n\t//     outerRadius: sliceHeight,\n\t//     startAngle: -(sliceAngle / 2),\n\t//     endAngle: sliceAngle / 2,\n\t//   }))\n\t//   .style('fill', 'tomato')\n\t//   .style('stroke', 'tomato')\n\t//   .style('stroke-width', 5);\n\n\t// radial gradient for background\n\tconst mandalaBgGrad = defs.append('radialGradient')\n\t  .attr('id', 'bg-shading')\n\t  .attr('gradientUnits', 'userSpaceOnUse');\n\n\tmandalaBgGrad.append('stop')\n\t  .attr('offset', '0%')\n\t  .attr('stop-color', '#000')\n\t  .attr('stop-opacity', 0.0);\n\n\tmandalaBgGrad.append('stop')\n\t  .attr('offset', '100%')\n\t  .attr('stop-color', '#000')\n\t  .attr('stop-opacity', 0.2);\n\n\tsvg.insert('rect', 'g')\n\t  .attr('class', 'mandala-bg-shading')\n\t  .attr('width', width)\n\t  .attr('height', height)\n\t  .style('fill', 'url(#bg-shading)');\n\n\t// add in a big clip for all the marks\n\tconst marksClip = defs.append('clipPath')\n\t  .attr('id', 'marks-clip')\n\t  .append('circle')\n\t  .attr('cx', plotAreaWidth / 2)\n\t  .attr('cy', plotAreaHeight / 2)\n\t  .attr('r', 0)\n\t  .style('fill', '#fff');\n\n\tif (animate) {\n\t  marksClip.transition()\n\t    .ease(d3.easeLinear)\n\t    .duration(2000)\n\t    .attr('r', (plotAreaHeight / 2) + 5);\n\t} else {\n\t  marksClip\n\t    .attr('r', (plotAreaHeight / 2) + 5);\n\t}\n\n\tconst gSlices = g.append('g')\n\t  .attr('class', 'slices-group')\n\t  .attr('clip-path', 'url(#marks-clip)');\n\n\t// create the group to be repeated\n\tconst slice = gSlices.append('g')\n\t  .attr('id', 'ref-slice')\n\t  .attr('class', 'slice')\n\t  .attr('transform', `translate(${plotAreaWidth / 2} 0)`)\n\t  .attr('clip-path', 'url(#slice-clip)');\n\n\t// add in copies of this slice\n\tconst copySlices = d3.range(numSlices - 1).map((d, i) => ({\n\t  id: i + 1,\n\t  href: '#ref-slice',\n\t  transform: `rotate(${(i + 1) * sliceAngle * (180 / Math.PI)} ${plotAreaWidth / 2} ${sliceHeight})`,\n\t}));\n\n\tconst sliceBinding = gSlices.selectAll('copy-slice').data(copySlices);\n\tsliceBinding.enter().append('use')\n\t  .attr('xlink:href', d => d.href)\n\t  .attr('transform', d => d.transform);\n\n\t// build up the slice\n\tslice.append('path')\n\t  .attr('class', 'slice-bg')\n\t  .attr('transform', `translate(0 ${sliceHeight})`)\n\t  .attr('d', arc({\n\t    innerRadius: 0,\n\t    outerRadius: sliceHeight,\n\t    startAngle: -(sliceAngle / 2),\n\t    endAngle: sliceAngle / 2,\n\t  }))\n\t  .style('fill', 'none')\n\t  .style('stroke', 'tomato')\n\t  .style('opacity', 0.0);\n\n\tconst markColor = '#fff';\n\n\t// add points to the slice\n\tconst points = slice.selectAll('.point').data(dataByType.point || []);\n\n\tpoints.enter()\n\t  .append('circle')\n\t  .attr('class', 'point')\n\t  .attr('r', d => d.r)\n\t  .attr('cx', 0)\n\t  .attr('cy', d => yScale(d.y))\n\t  .style('fill', d => (d.filled ? markColor : 'none'))\n\t  .style('stroke', d => (d.filled ? 'none' : markColor));\n\n\t// add arcs\n\tconst arcs = slice.selectAll('.arc').data(dataByType.arc || []);\n\n\tconst interiorArc = d3.arc()\n\t  .innerRadius(d => rScale(d.y - d.thickness))\n\t  .outerRadius(d => rScale(d.y))\n\t  .startAngle((-sliceAngle / 2) - 0.1) // slight padding to ensure overlap\n\t  .endAngle((sliceAngle / 2) + 0.1);\n\n\tarcs.enter()\n\t  .append('path')\n\t  .attr('transform', `translate(0 ${sliceHeight})`)\n\t  .attr('class', 'arc')\n\t  .attr('d', interiorArc)\n\t  .style('fill', markColor);\n\n\t// add diagonal line up\n\tconst diagUp = slice.selectAll('.diagonalUp').data(dataByType.diagonalUp || []);\n\n\tfunction dToLine(d, y1Key = 'y1', y2Key = 'y2') {\n\t  const y1 = yScale(d[y1Key]);\n\t  const y2 = yScale(d[y2Key]);\n\t  const x1 = (sliceHeight - y1) * Math.tan(-sliceAngle / 2);\n\t  const x2 = (sliceHeight - y2) * Math.tan(sliceAngle / 2);\n\n\t  return {\n\t    x1,\n\t    x2,\n\t    y1,\n\t    y2,\n\t  };\n\t}\n\n\tfunction toPath({ x1, x2, y1, y2 }) {\n\t  return `M${x1},${y1} L${x2},${y2}`;\n\t}\n\n\tdiagUp.enter()\n\t  .append('path')\n\t  .attr('class', 'diagonalUp')\n\t  .attr('d', d => toPath(dToLine(d)))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n\n\t// add diagonal down\n\tconst diagDown = slice.selectAll('.diagonalDown').data(dataByType.diagonalDown || []);\n\n\tdiagDown.enter()\n\t  .append('path')\n\t  .attr('class', 'diagonalDown')\n\t  .attr('d', d => toPath(dToLine(d)))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n\n\n\t// add X marks\n\tconst xMarks = slice.selectAll('.x').data(dataByType.x || []);\n\n\tconst xMarkGs = xMarks.enter()\n\t  .append('g')\n\t  .attr('class', 'x');\n\n\txMarkGs.append('path')\n\t  .attr('d', d => toPath(dToLine(d)))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n\n\txMarkGs.append('path')\n\t  .attr('d', d => toPath(dToLine(d, 'y2', 'y1')))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n\n\t// add X marks\n\tconst arrowMarks = slice.selectAll('.arrow').data(dataByType.arrow || []);\n\n\tconst arrowMarkGs = arrowMarks.enter()\n\t  .append('g')\n\t  .attr('class', 'arrow');\n\n\tarrowMarkGs.append('path')\n\t  .attr('d', d => toPath(dToLine(d, 'y1', 'yMid')))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n\n\tarrowMarkGs.append('path')\n\t  .attr('d', d => toPath(dToLine(d, 'y2', 'yMid')))\n\t  .style('stroke', markColor)\n\t  .style('fill', markColor);\n}\n\ngenerateMandala();\n\nd3.select('#make-mandala').on('click', generateMandala);\n\n"]}